139 lines
3.7 KiB
TypeScript
139 lines
3.7 KiB
TypeScript
import { Dictionary, groupBy, sumBy, minBy } from 'lodash'
|
|
import { Bet } from './bet'
|
|
import { getContractBetMetrics } from './calculate'
|
|
import {
|
|
Contract,
|
|
CPMMContract,
|
|
FreeResponseContract,
|
|
MultipleChoiceContract,
|
|
} from './contract'
|
|
import { PortfolioMetrics, User } from './user'
|
|
import { filterDefined } from './util/array'
|
|
|
|
const LOAN_DAILY_RATE = 0.01
|
|
|
|
const calculateNewLoan = (investedValue: number, loanTotal: number) => {
|
|
const netValue = investedValue - loanTotal
|
|
return netValue * LOAN_DAILY_RATE
|
|
}
|
|
|
|
export const getLoanUpdates = (
|
|
users: User[],
|
|
contractsById: { [contractId: string]: Contract },
|
|
portfolioByUser: { [userId: string]: PortfolioMetrics | undefined },
|
|
betsByUser: { [userId: string]: Bet[] }
|
|
) => {
|
|
const eligibleUsers = filterDefined(
|
|
users.map((user) =>
|
|
isUserEligibleForLoan(portfolioByUser[user.id]) ? user : undefined
|
|
)
|
|
)
|
|
|
|
const betUpdates = eligibleUsers
|
|
.map((user) => {
|
|
const updates = calculateLoanBetUpdates(
|
|
betsByUser[user.id] ?? [],
|
|
contractsById
|
|
).betUpdates
|
|
return updates.map((update) => ({ ...update, user }))
|
|
})
|
|
.flat()
|
|
|
|
const updatesByUser = groupBy(betUpdates, (update) => update.userId)
|
|
const userPayouts = Object.values(updatesByUser).map((updates) => {
|
|
return {
|
|
user: updates[0].user,
|
|
payout: sumBy(updates, (update) => update.newLoan),
|
|
}
|
|
})
|
|
|
|
return {
|
|
betUpdates,
|
|
userPayouts,
|
|
}
|
|
}
|
|
|
|
const isUserEligibleForLoan = (portfolio: PortfolioMetrics | undefined) => {
|
|
if (!portfolio) return true
|
|
|
|
const { balance, investmentValue } = portfolio
|
|
return balance + investmentValue > 0
|
|
}
|
|
|
|
const calculateLoanBetUpdates = (
|
|
bets: Bet[],
|
|
contractsById: Dictionary<Contract>
|
|
) => {
|
|
const betsByContract = groupBy(bets, (bet) => bet.contractId)
|
|
const contracts = filterDefined(
|
|
Object.keys(betsByContract).map((contractId) => contractsById[contractId])
|
|
).filter((c) => !c.isResolved)
|
|
|
|
const betUpdates = filterDefined(
|
|
contracts
|
|
.map((c) => {
|
|
if (c.mechanism === 'cpmm-1') {
|
|
return getBinaryContractLoanUpdate(c, betsByContract[c.id])
|
|
} else if (
|
|
c.outcomeType === 'FREE_RESPONSE' ||
|
|
c.outcomeType === 'MULTIPLE_CHOICE'
|
|
)
|
|
return getFreeResponseContractLoanUpdate(c, betsByContract[c.id])
|
|
else {
|
|
// Unsupported contract / mechanism for loans.
|
|
return []
|
|
}
|
|
})
|
|
.flat()
|
|
)
|
|
|
|
const totalNewLoan = sumBy(betUpdates, (loanUpdate) => loanUpdate.loanTotal)
|
|
|
|
return {
|
|
totalNewLoan,
|
|
betUpdates,
|
|
}
|
|
}
|
|
|
|
const getBinaryContractLoanUpdate = (contract: CPMMContract, bets: Bet[]) => {
|
|
const { invested } = getContractBetMetrics(contract, bets)
|
|
const loanAmount = sumBy(bets, (bet) => bet.loanAmount ?? 0)
|
|
const oldestBet = minBy(bets, (bet) => bet.createdTime)
|
|
|
|
const newLoan = calculateNewLoan(invested, loanAmount)
|
|
if (!isFinite(newLoan) || newLoan <= 0 || !oldestBet) return undefined
|
|
|
|
const loanTotal = (oldestBet.loanAmount ?? 0) + newLoan
|
|
|
|
return {
|
|
userId: oldestBet.userId,
|
|
contractId: contract.id,
|
|
betId: oldestBet.id,
|
|
newLoan,
|
|
loanTotal,
|
|
}
|
|
}
|
|
|
|
const getFreeResponseContractLoanUpdate = (
|
|
contract: FreeResponseContract | MultipleChoiceContract,
|
|
bets: Bet[]
|
|
) => {
|
|
const openBets = bets.filter((bet) => bet.isSold || bet.sale)
|
|
|
|
return openBets.map((bet) => {
|
|
const loanAmount = bet.loanAmount ?? 0
|
|
const newLoan = calculateNewLoan(bet.amount, loanAmount)
|
|
const loanTotal = loanAmount + newLoan
|
|
|
|
if (!isFinite(newLoan) || newLoan <= 0) return undefined
|
|
|
|
return {
|
|
userId: bet.userId,
|
|
contractId: contract.id,
|
|
betId: bet.id,
|
|
newLoan,
|
|
loanTotal,
|
|
}
|
|
})
|
|
}
|