2022-05-22 08:36:05 +00:00
|
|
|
import { cloneDeep, range, sum, sumBy, sortBy, mapValues } from 'lodash'
|
2022-05-19 17:42:03 +00:00
|
|
|
import { Bet, NumericBet } from './bet'
|
|
|
|
import {
|
|
|
|
Binary,
|
|
|
|
DPM,
|
|
|
|
FreeResponse,
|
|
|
|
FullContract,
|
|
|
|
Numeric,
|
|
|
|
NumericContract,
|
|
|
|
} from './contract'
|
2022-03-15 22:27:51 +00:00
|
|
|
import { DPM_FEES } from './fees'
|
2022-05-19 17:42:03 +00:00
|
|
|
import { normpdf } from '../common/util/math'
|
|
|
|
import { addObjects } from './util/object'
|
2022-03-15 22:27:51 +00:00
|
|
|
|
|
|
|
export function getDpmProbability(totalShares: { [outcome: string]: number }) {
|
|
|
|
// For binary contracts only.
|
|
|
|
return getDpmOutcomeProbability(totalShares, 'YES')
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getDpmOutcomeProbability(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
outcome: string
|
|
|
|
) {
|
2022-05-22 08:36:05 +00:00
|
|
|
const squareSum = sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
2022-03-15 22:27:51 +00:00
|
|
|
const shares = totalShares[outcome] ?? 0
|
|
|
|
return shares ** 2 / squareSum
|
|
|
|
}
|
|
|
|
|
2022-05-19 17:42:03 +00:00
|
|
|
export function getDpmOutcomeProbabilities(totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
}) {
|
2022-05-22 08:36:05 +00:00
|
|
|
const squareSum = sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
|
|
|
return mapValues(totalShares, (shares) => shares ** 2 / squareSum)
|
2022-05-19 17:42:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getNumericBets(
|
|
|
|
contract: NumericContract,
|
|
|
|
bucket: string,
|
|
|
|
betAmount: number,
|
|
|
|
variance: number
|
|
|
|
) {
|
|
|
|
const { bucketCount } = contract
|
|
|
|
const bucketNumber = parseInt(bucket)
|
2022-05-22 08:36:05 +00:00
|
|
|
const buckets = range(0, bucketCount)
|
2022-05-19 17:42:03 +00:00
|
|
|
|
|
|
|
const mean = bucketNumber / bucketCount
|
|
|
|
|
|
|
|
const allDensities = buckets.map((i) =>
|
|
|
|
normpdf(i / bucketCount, mean, variance)
|
|
|
|
)
|
2022-05-22 08:36:05 +00:00
|
|
|
const densitySum = sum(allDensities)
|
2022-05-19 17:42:03 +00:00
|
|
|
|
|
|
|
const rawBetAmounts = allDensities
|
|
|
|
.map((d) => (d / densitySum) * betAmount)
|
|
|
|
.map((x) => (x >= 1 / bucketCount ? x : 0))
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const rawSum = sum(rawBetAmounts)
|
2022-05-19 17:42:03 +00:00
|
|
|
const scaledBetAmounts = rawBetAmounts.map((x) => (x / rawSum) * betAmount)
|
|
|
|
|
|
|
|
const bets = scaledBetAmounts
|
|
|
|
.map((x, i) => (x > 0 ? [i.toString(), x] : undefined))
|
|
|
|
.filter((x) => x != undefined) as [string, number][]
|
|
|
|
|
|
|
|
return bets
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getMappedBucket = (value: number, contract: NumericContract) => {
|
|
|
|
const { bucketCount, min, max } = contract
|
|
|
|
|
|
|
|
const index = Math.floor(((value - min) / (max - min)) * bucketCount)
|
|
|
|
const bucket = Math.max(Math.min(index, bucketCount - 1), 0)
|
|
|
|
|
|
|
|
return `${bucket}`
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getValueFromBucket = (
|
|
|
|
bucket: string,
|
|
|
|
contract: NumericContract
|
|
|
|
) => {
|
|
|
|
const { bucketCount, min, max } = contract
|
|
|
|
const index = parseInt(bucket)
|
|
|
|
const value = min + (index / bucketCount) * (max - min)
|
|
|
|
const rounded = Math.round(value * 1e4) / 1e4
|
|
|
|
return rounded
|
|
|
|
}
|
|
|
|
|
|
|
|
export const getExpectedValue = (contract: NumericContract) => {
|
|
|
|
const { bucketCount, min, max, totalShares } = contract
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const totalShareSum = sumBy(
|
2022-05-19 17:42:03 +00:00
|
|
|
Object.values(totalShares),
|
|
|
|
(shares) => shares ** 2
|
|
|
|
)
|
2022-05-22 08:36:05 +00:00
|
|
|
const probs = range(0, bucketCount).map(
|
2022-05-19 17:42:03 +00:00
|
|
|
(i) => totalShares[i] ** 2 / totalShareSum
|
|
|
|
)
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const values = range(0, bucketCount).map(
|
2022-05-19 17:42:03 +00:00
|
|
|
(i) =>
|
|
|
|
// use mid point within bucket
|
|
|
|
0.5 * (min + (i / bucketCount) * (max - min)) +
|
|
|
|
0.5 * (min + ((i + 1) / bucketCount) * (max - min))
|
|
|
|
)
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const weightedValues = range(0, bucketCount).map((i) => probs[i] * values[i])
|
2022-05-19 17:42:03 +00:00
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const expectation = sum(weightedValues)
|
2022-05-19 17:42:03 +00:00
|
|
|
const rounded = Math.round(expectation * 1e2) / 1e2
|
|
|
|
return rounded
|
|
|
|
}
|
|
|
|
|
2022-03-15 22:27:51 +00:00
|
|
|
export function getDpmOutcomeProbabilityAfterBet(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
outcome: string,
|
|
|
|
bet: number
|
|
|
|
) {
|
|
|
|
const shares = calculateDpmShares(totalShares, bet, outcome)
|
|
|
|
|
|
|
|
const prevShares = totalShares[outcome] ?? 0
|
|
|
|
const newTotalShares = { ...totalShares, [outcome]: prevShares + shares }
|
|
|
|
|
|
|
|
return getDpmOutcomeProbability(newTotalShares, outcome)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getDpmProbabilityAfterSale(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
outcome: string,
|
|
|
|
shares: number
|
|
|
|
) {
|
|
|
|
const prevShares = totalShares[outcome] ?? 0
|
|
|
|
const newTotalShares = { ...totalShares, [outcome]: prevShares - shares }
|
|
|
|
|
|
|
|
const predictionOutcome = outcome === 'NO' ? 'YES' : outcome
|
|
|
|
return getDpmOutcomeProbability(newTotalShares, predictionOutcome)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmShares(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
bet: number,
|
|
|
|
betChoice: string
|
|
|
|
) {
|
2022-05-22 08:36:05 +00:00
|
|
|
const squareSum = sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
2022-03-15 22:27:51 +00:00
|
|
|
const shares = totalShares[betChoice] ?? 0
|
|
|
|
|
|
|
|
const c = 2 * bet * Math.sqrt(squareSum)
|
|
|
|
|
|
|
|
return Math.sqrt(bet ** 2 + shares ** 2 + c) - shares
|
|
|
|
}
|
|
|
|
|
2022-05-19 17:42:03 +00:00
|
|
|
export function calculateNumericDpmShares(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
bets: [string, number][]
|
|
|
|
) {
|
|
|
|
const shares: number[] = []
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
totalShares = cloneDeep(totalShares)
|
2022-05-19 17:42:03 +00:00
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const order = sortBy(
|
2022-05-19 17:42:03 +00:00
|
|
|
bets.map(([, amount], i) => [amount, i]),
|
|
|
|
([amount]) => amount
|
|
|
|
).map(([, i]) => i)
|
|
|
|
|
2022-05-24 20:17:39 +00:00
|
|
|
for (const i of order) {
|
2022-05-19 17:42:03 +00:00
|
|
|
const [bucket, bet] = bets[i]
|
|
|
|
shares[i] = calculateDpmShares(totalShares, bet, bucket)
|
|
|
|
totalShares = addObjects(totalShares, { [bucket]: shares[i] })
|
|
|
|
}
|
|
|
|
|
|
|
|
return { shares, totalShares }
|
|
|
|
}
|
|
|
|
|
2022-03-15 22:27:51 +00:00
|
|
|
export function calculateDpmRawShareValue(
|
|
|
|
totalShares: {
|
|
|
|
[outcome: string]: number
|
|
|
|
},
|
|
|
|
shares: number,
|
|
|
|
betChoice: string
|
|
|
|
) {
|
|
|
|
const currentValue = Math.sqrt(
|
2022-05-22 08:36:05 +00:00
|
|
|
sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
2022-03-15 22:27:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const postSaleValue = Math.sqrt(
|
2022-05-22 08:36:05 +00:00
|
|
|
sumBy(Object.keys(totalShares), (outcome) =>
|
2022-03-15 22:27:51 +00:00
|
|
|
outcome === betChoice
|
|
|
|
? Math.max(0, totalShares[outcome] - shares) ** 2
|
|
|
|
: totalShares[outcome] ** 2
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
return currentValue - postSaleValue
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmMoneyRatio(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet,
|
|
|
|
shareValue: number
|
|
|
|
) {
|
|
|
|
const { totalShares, totalBets, pool } = contract
|
|
|
|
const { outcome, amount } = bet
|
|
|
|
|
|
|
|
const p = getDpmOutcomeProbability(totalShares, outcome)
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const actual = sum(Object.values(pool)) - shareValue
|
2022-03-15 22:27:51 +00:00
|
|
|
|
|
|
|
const betAmount = p * amount
|
|
|
|
|
|
|
|
const expected =
|
2022-05-22 08:36:05 +00:00
|
|
|
sumBy(
|
2022-03-15 22:27:51 +00:00
|
|
|
Object.keys(totalBets),
|
|
|
|
(outcome) =>
|
|
|
|
getDpmOutcomeProbability(totalShares, outcome) *
|
|
|
|
(totalBets as { [outcome: string]: number })[outcome]
|
|
|
|
) - betAmount
|
|
|
|
|
|
|
|
if (actual <= 0 || expected <= 0) return 0
|
|
|
|
|
|
|
|
return actual / expected
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmShareValue(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
|
|
|
const { pool, totalShares } = contract
|
|
|
|
const { shares, outcome } = bet
|
|
|
|
|
|
|
|
const shareValue = calculateDpmRawShareValue(totalShares, shares, outcome)
|
|
|
|
const f = calculateDpmMoneyRatio(contract, bet, shareValue)
|
|
|
|
|
|
|
|
const myPool = pool[outcome]
|
|
|
|
const adjShareValue = Math.min(Math.min(1, f) * shareValue, myPool)
|
|
|
|
return adjShareValue
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmSaleAmount(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
|
|
|
const { amount } = bet
|
|
|
|
const winnings = calculateDpmShareValue(contract, bet)
|
|
|
|
return deductDpmFees(amount, winnings)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmPayout(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet,
|
|
|
|
outcome: string
|
|
|
|
) {
|
|
|
|
if (outcome === 'CANCEL') return calculateDpmCancelPayout(contract, bet)
|
|
|
|
if (outcome === 'MKT') return calculateMktDpmPayout(contract, bet)
|
|
|
|
|
|
|
|
return calculateStandardDpmPayout(contract, bet, outcome)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmCancelPayout(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
|
|
|
const { totalBets, pool } = contract
|
2022-05-22 08:36:05 +00:00
|
|
|
const betTotal = sum(Object.values(totalBets))
|
|
|
|
const poolTotal = sum(Object.values(pool))
|
2022-03-15 22:27:51 +00:00
|
|
|
|
|
|
|
return (bet.amount / betTotal) * poolTotal
|
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateStandardDpmPayout(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet,
|
|
|
|
outcome: string
|
|
|
|
) {
|
2022-05-19 17:42:03 +00:00
|
|
|
const { outcome: betOutcome } = bet
|
|
|
|
const isNumeric = contract.outcomeType === 'NUMERIC'
|
|
|
|
if (!isNumeric && betOutcome !== outcome) return 0
|
|
|
|
|
|
|
|
const shares = isNumeric
|
|
|
|
? ((bet as NumericBet).allOutcomeShares ?? {})[outcome]
|
|
|
|
: bet.shares
|
|
|
|
|
|
|
|
if (!shares) return 0
|
2022-03-15 22:27:51 +00:00
|
|
|
|
|
|
|
const { totalShares, phantomShares, pool } = contract
|
|
|
|
if (!totalShares[outcome]) return 0
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const poolTotal = sum(Object.values(pool))
|
2022-03-15 22:27:51 +00:00
|
|
|
|
|
|
|
const total =
|
|
|
|
totalShares[outcome] - (phantomShares ? phantomShares[outcome] : 0)
|
|
|
|
|
|
|
|
const winnings = (shares / total) * poolTotal
|
2022-05-19 17:42:03 +00:00
|
|
|
|
|
|
|
const amount = isNumeric
|
|
|
|
? (bet as NumericBet).allBetAmounts[outcome]
|
|
|
|
: bet.amount
|
|
|
|
|
|
|
|
const payout = amount + (1 - DPM_FEES) * Math.max(0, winnings - amount)
|
|
|
|
return payout
|
2022-03-15 22:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function calculateDpmPayoutAfterCorrectBet(
|
|
|
|
contract: FullContract<DPM, any>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
2022-05-19 17:42:03 +00:00
|
|
|
const { totalShares, pool, totalBets, outcomeType } = contract
|
2022-03-15 22:27:51 +00:00
|
|
|
const { shares, amount, outcome } = bet
|
|
|
|
|
|
|
|
const prevShares = totalShares[outcome] ?? 0
|
|
|
|
const prevPool = pool[outcome] ?? 0
|
|
|
|
const prevTotalBet = totalBets[outcome] ?? 0
|
|
|
|
|
|
|
|
const newContract = {
|
|
|
|
...contract,
|
|
|
|
totalShares: {
|
|
|
|
...totalShares,
|
|
|
|
[outcome]: prevShares + shares,
|
|
|
|
},
|
|
|
|
pool: {
|
|
|
|
...pool,
|
|
|
|
[outcome]: prevPool + amount,
|
|
|
|
},
|
|
|
|
totalBets: {
|
|
|
|
...totalBets,
|
|
|
|
[outcome]: prevTotalBet + amount,
|
|
|
|
},
|
2022-05-19 17:42:03 +00:00
|
|
|
outcomeType:
|
|
|
|
outcomeType === 'NUMERIC'
|
|
|
|
? 'FREE_RESPONSE' // hack to show payout at particular bet point estimate
|
|
|
|
: outcomeType,
|
2022-03-15 22:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return calculateStandardDpmPayout(newContract, bet, outcome)
|
|
|
|
}
|
|
|
|
|
2022-05-19 17:42:03 +00:00
|
|
|
function calculateMktDpmPayout(
|
|
|
|
contract: FullContract<DPM, Binary | FreeResponse | Numeric>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
2022-03-15 22:27:51 +00:00
|
|
|
if (contract.outcomeType === 'BINARY')
|
|
|
|
return calculateBinaryMktDpmPayout(contract, bet)
|
|
|
|
|
2022-05-19 17:42:03 +00:00
|
|
|
const { totalShares, pool, resolutions, outcomeType } = contract
|
2022-03-18 17:22:42 +00:00
|
|
|
|
2022-03-18 19:59:06 +00:00
|
|
|
let probs: { [outcome: string]: number }
|
2022-03-15 22:27:51 +00:00
|
|
|
|
2022-03-18 19:59:06 +00:00
|
|
|
if (resolutions) {
|
2022-05-22 08:36:05 +00:00
|
|
|
const probTotal = sum(Object.values(resolutions))
|
|
|
|
probs = mapValues(
|
2022-03-18 19:59:06 +00:00
|
|
|
totalShares,
|
|
|
|
(_, outcome) => (resolutions[outcome] ?? 0) / probTotal
|
|
|
|
)
|
|
|
|
} else {
|
2022-05-22 08:36:05 +00:00
|
|
|
const squareSum = sum(
|
2022-03-18 19:59:06 +00:00
|
|
|
Object.values(totalShares).map((shares) => shares ** 2)
|
|
|
|
)
|
2022-05-22 08:36:05 +00:00
|
|
|
probs = mapValues(totalShares, (shares) => shares ** 2 / squareSum)
|
2022-03-18 19:59:06 +00:00
|
|
|
}
|
2022-03-15 22:27:51 +00:00
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const weightedShareTotal = sumBy(Object.keys(totalShares), (outcome) => {
|
2022-03-18 19:59:06 +00:00
|
|
|
return probs[outcome] * totalShares[outcome]
|
2022-03-15 22:27:51 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
const { outcome, amount, shares } = bet
|
|
|
|
|
2022-05-19 17:42:03 +00:00
|
|
|
const poolFrac =
|
|
|
|
outcomeType === 'NUMERIC'
|
2022-05-22 08:36:05 +00:00
|
|
|
? sumBy(
|
2022-05-19 17:42:03 +00:00
|
|
|
Object.keys((bet as NumericBet).allOutcomeShares ?? {}),
|
|
|
|
(outcome) => {
|
|
|
|
return (
|
|
|
|
(probs[outcome] * (bet as NumericBet).allOutcomeShares[outcome]) /
|
|
|
|
weightedShareTotal
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
: (probs[outcome] * shares) / weightedShareTotal
|
|
|
|
|
2022-05-22 08:36:05 +00:00
|
|
|
const totalPool = sum(Object.values(pool))
|
2022-03-18 19:59:06 +00:00
|
|
|
const winnings = poolFrac * totalPool
|
2022-03-15 22:27:51 +00:00
|
|
|
return deductDpmFees(amount, winnings)
|
|
|
|
}
|
|
|
|
|
|
|
|
function calculateBinaryMktDpmPayout(
|
|
|
|
contract: FullContract<DPM, Binary>,
|
|
|
|
bet: Bet
|
|
|
|
) {
|
|
|
|
const { resolutionProbability, totalShares, phantomShares } = contract
|
|
|
|
const p =
|
|
|
|
resolutionProbability !== undefined
|
|
|
|
? resolutionProbability
|
|
|
|
: getDpmProbability(totalShares)
|
|
|
|
|
|
|
|
const pool = contract.pool.YES + contract.pool.NO
|
|
|
|
|
|
|
|
const weightedShareTotal =
|
|
|
|
p * (totalShares.YES - (phantomShares?.YES ?? 0)) +
|
|
|
|
(1 - p) * (totalShares.NO - (phantomShares?.NO ?? 0))
|
|
|
|
|
|
|
|
const { outcome, amount, shares } = bet
|
|
|
|
|
|
|
|
const betP = outcome === 'YES' ? p : 1 - p
|
|
|
|
const winnings = ((betP * shares) / weightedShareTotal) * pool
|
|
|
|
|
|
|
|
return deductDpmFees(amount, winnings)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function resolvedDpmPayout(contract: FullContract<DPM, any>, bet: Bet) {
|
|
|
|
if (contract.resolution)
|
|
|
|
return calculateDpmPayout(contract, bet, contract.resolution)
|
|
|
|
throw new Error('Contract was not resolved')
|
|
|
|
}
|
|
|
|
|
|
|
|
export const deductDpmFees = (betAmount: number, winnings: number) => {
|
|
|
|
return winnings > betAmount
|
|
|
|
? betAmount + (1 - DPM_FEES) * (winnings - betAmount)
|
|
|
|
: winnings
|
|
|
|
}
|
|
|
|
|
|
|
|
export const calcDpmInitialPool = (
|
|
|
|
initialProbInt: number,
|
|
|
|
ante: number,
|
|
|
|
phantomAnte: number
|
|
|
|
) => {
|
|
|
|
const p = initialProbInt / 100.0
|
|
|
|
const totalAnte = phantomAnte + ante
|
|
|
|
|
|
|
|
const sharesYes = Math.sqrt(p * totalAnte ** 2)
|
|
|
|
const sharesNo = Math.sqrt(totalAnte ** 2 - sharesYes ** 2)
|
|
|
|
|
|
|
|
const poolYes = p * ante
|
|
|
|
const poolNo = (1 - p) * ante
|
|
|
|
|
|
|
|
const phantomYes = Math.sqrt(p) * phantomAnte
|
|
|
|
const phantomNo = Math.sqrt(1 - p) * phantomAnte
|
|
|
|
|
|
|
|
return { sharesYes, sharesNo, poolYes, poolNo, phantomYes, phantomNo }
|
|
|
|
}
|