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_WEEKLY_RATE = 0.05
 | |
| 
 | |
| const calculateNewLoan = (investedValue: number, loanTotal: number) => {
 | |
|   const netValue = investedValue - loanTotal
 | |
|   return netValue * LOAN_WEEKLY_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,
 | |
|     }
 | |
|   })
 | |
| }
 |