b2501d8145
* Answer datatype and MULTI outcome type for Contract
* Create free answer contract
* Automatically sort Tailwind classes with Prettier (#45)
* Add Prettier Tailwind plugin
* Autoformat Tailwind classes with Prettier
* Allow for non-binary contracts in contract page and related components
* logo with white inside, transparent bg
* Create answer
* Some UI for showing answers
* Answer bet panel
* Convert rest of calcuate file to generic multi contracts
* Working betting with ante'd NONE answer
* Numbered answers. Layout & calculation tweaks
* Can bet. More layout tweaks!
* Resolve answer UI
* Resolve multi market
* Resolved market UI
* Fix feed and cards for multi contracts
* Sell bets. Various fixes
* Tweaks for trades page
* Always dev mode
* Create answer bet has isAnte: true
* Fix card showing 0% for multi contracts
* Fix grouped bets feed for multi outcomes
* None option converted to none of the above label at bottom of list. Button to resolve none.
* Tweaks to no answers yet, resolve button layout
* Show ante bets on new answers in the feed
* Update placeholder text for description
* Consolidate firestore rules for subcollections
* Remove Contract and Bet type params. Use string type for outcomes.
* Increase char limit to 10k for answers. Preserve line breaks.
* Don't show resolve options after answer chosen
* Fix type error in script
* Remove NONE resolution option
* Change outcomeType to include 'MULTI' and 'FREE_RESPONSE'
* Show bet probability change and payout when creating answer
* User info change: also change answers
* Append answers to contract field 'answers'
* sort trades by resolved
* Don't include trailing !:,.; in links
* Stop flooring inputs into formatMoney
* Revert "Stop flooring inputs into formatMoney"
This reverts commit 2f7ab18429
.
* Consistently floor user.balance
* Expand create panel on focus
From Richard Hanania's feedback
* welcome email: include link to manifold
* Fix home page in dev on branches that are not free-response
* Close emails (#50)
* script init for stephen dev
* market close emails
* order of operations
* template email
* sendMarketCloseEmail: handle unsubscribe
* remove debugging
* marketCloseEmails: every hour
* sendMarketCloseEmails: check undefined
* marketCloseEmails: "every hour" => "every 1 hours"
* Set up a read API using Vercel serverless functions (#49)
* Set up read API using Vercel serverless functions
Featuring:
/api/v0/markets
/api/v0/market/[contractId]
/api/v0/slug/[contractSlug]
* Include tags in API
* Tweaks. Remove filter for only binary contract
* Fix bet probability change for NO bets
* Put back isProd calculation
Co-authored-by: Austin Chen <akrolsmir@gmail.com>
Co-authored-by: mantikoros <sgrugett@gmail.com>
Co-authored-by: mantikoros <95266179+mantikoros@users.noreply.github.com>
241 lines
6.5 KiB
TypeScript
241 lines
6.5 KiB
TypeScript
import * as _ from 'lodash'
|
|
import { Bet } from './bet'
|
|
import { Contract } from './contract'
|
|
import { FEES } from './fees'
|
|
|
|
export function getProbability(totalShares: { [outcome: string]: number }) {
|
|
// For binary contracts only.
|
|
return getOutcomeProbability(totalShares, 'YES')
|
|
}
|
|
|
|
export function getOutcomeProbability(
|
|
totalShares: {
|
|
[outcome: string]: number
|
|
},
|
|
outcome: string
|
|
) {
|
|
const squareSum = _.sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
|
const shares = totalShares[outcome] ?? 0
|
|
return shares ** 2 / squareSum
|
|
}
|
|
|
|
export function getProbabilityAfterBet(
|
|
totalShares: {
|
|
[outcome: string]: number
|
|
},
|
|
outcome: string,
|
|
bet: number
|
|
) {
|
|
const shares = calculateShares(totalShares, bet, outcome)
|
|
|
|
const prevShares = totalShares[outcome] ?? 0
|
|
const newTotalShares = { ...totalShares, [outcome]: prevShares + shares }
|
|
|
|
return getOutcomeProbability(newTotalShares, outcome)
|
|
}
|
|
|
|
export function calculateShares(
|
|
totalShares: {
|
|
[outcome: string]: number
|
|
},
|
|
bet: number,
|
|
betChoice: string
|
|
) {
|
|
const squareSum = _.sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
|
const shares = totalShares[betChoice] ?? 0
|
|
|
|
const c = 2 * bet * Math.sqrt(squareSum)
|
|
|
|
return Math.sqrt(bet ** 2 + shares ** 2 + c) - shares
|
|
}
|
|
|
|
export function calculateRawShareValue(
|
|
totalShares: {
|
|
[outcome: string]: number
|
|
},
|
|
shares: number,
|
|
betChoice: string
|
|
) {
|
|
const currentValue = Math.sqrt(
|
|
_.sumBy(Object.values(totalShares), (shares) => shares ** 2)
|
|
)
|
|
|
|
const postSaleValue = Math.sqrt(
|
|
_.sumBy(Object.keys(totalShares), (outcome) =>
|
|
outcome === betChoice
|
|
? Math.max(0, totalShares[outcome] - shares) ** 2
|
|
: totalShares[outcome] ** 2
|
|
)
|
|
)
|
|
|
|
return currentValue - postSaleValue
|
|
}
|
|
|
|
export function calculateMoneyRatio(
|
|
contract: Contract,
|
|
bet: Bet,
|
|
shareValue: number
|
|
) {
|
|
const { totalShares, totalBets, pool } = contract
|
|
const { outcome, amount } = bet
|
|
|
|
const p = getOutcomeProbability(totalShares, outcome)
|
|
|
|
const actual = _.sum(Object.values(pool)) - shareValue
|
|
|
|
const betAmount = p * amount
|
|
|
|
const expected =
|
|
_.sumBy(
|
|
Object.keys(totalBets),
|
|
(outcome) =>
|
|
getOutcomeProbability(totalShares, outcome) *
|
|
(totalBets as { [outcome: string]: number })[outcome]
|
|
) - betAmount
|
|
|
|
if (actual <= 0 || expected <= 0) return 0
|
|
|
|
return actual / expected
|
|
}
|
|
|
|
export function calculateShareValue(contract: Contract, bet: Bet) {
|
|
const { pool, totalShares } = contract
|
|
const { shares, outcome } = bet
|
|
|
|
const shareValue = calculateRawShareValue(totalShares, shares, outcome)
|
|
const f = calculateMoneyRatio(contract, bet, shareValue)
|
|
|
|
const myPool = pool[outcome]
|
|
const adjShareValue = Math.min(Math.min(1, f) * shareValue, myPool)
|
|
return adjShareValue
|
|
}
|
|
|
|
export function calculateSaleAmount(contract: Contract, bet: Bet) {
|
|
const { amount } = bet
|
|
const winnings = calculateShareValue(contract, bet)
|
|
return deductFees(amount, winnings)
|
|
}
|
|
|
|
export function calculatePayout(contract: Contract, bet: Bet, outcome: string) {
|
|
if (outcome === 'CANCEL') return calculateCancelPayout(contract, bet)
|
|
if (outcome === 'MKT') return calculateMktPayout(contract, bet)
|
|
|
|
return calculateStandardPayout(contract, bet, outcome)
|
|
}
|
|
|
|
export function calculateCancelPayout(contract: Contract, bet: Bet) {
|
|
const { totalBets, pool } = contract
|
|
const betTotal = _.sum(Object.values(totalBets))
|
|
const poolTotal = _.sum(Object.values(pool))
|
|
|
|
return (bet.amount / betTotal) * poolTotal
|
|
}
|
|
|
|
export function calculateStandardPayout(
|
|
contract: Contract,
|
|
bet: Bet,
|
|
outcome: string
|
|
) {
|
|
const { amount, outcome: betOutcome, shares } = bet
|
|
if (betOutcome !== outcome) return 0
|
|
|
|
const { totalShares, phantomShares, pool } = contract
|
|
if (!totalShares[outcome]) return 0
|
|
|
|
const poolTotal = _.sum(Object.values(pool))
|
|
|
|
const total =
|
|
totalShares[outcome] - (phantomShares ? phantomShares[outcome] : 0)
|
|
|
|
const winnings = (shares / total) * poolTotal
|
|
// profit can be negative if using phantom shares
|
|
return amount + (1 - FEES) * Math.max(0, winnings - amount)
|
|
}
|
|
|
|
export function calculatePayoutAfterCorrectBet(contract: Contract, bet: Bet) {
|
|
const { totalShares, pool, totalBets } = contract
|
|
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,
|
|
},
|
|
}
|
|
|
|
return calculateStandardPayout(newContract, bet, outcome)
|
|
}
|
|
|
|
function calculateMktPayout(contract: Contract, bet: Bet) {
|
|
if (contract.outcomeType === 'BINARY')
|
|
return calculateBinaryMktPayout(contract, bet)
|
|
|
|
const { totalShares, pool } = contract
|
|
|
|
const totalPool = _.sum(Object.values(pool))
|
|
const sharesSquareSum = _.sumBy(
|
|
Object.values(totalShares),
|
|
(shares) => shares ** 2
|
|
)
|
|
|
|
const weightedShareTotal = _.sumBy(Object.keys(totalShares), (outcome) => {
|
|
// Avoid O(n^2) by reusing sharesSquareSum for prob.
|
|
const shares = totalShares[outcome]
|
|
const prob = shares ** 2 / sharesSquareSum
|
|
return prob * shares
|
|
})
|
|
|
|
const { outcome, amount, shares } = bet
|
|
|
|
const betP = getOutcomeProbability(totalShares, outcome)
|
|
const winnings = ((betP * shares) / weightedShareTotal) * totalPool
|
|
|
|
return deductFees(amount, winnings)
|
|
}
|
|
|
|
function calculateBinaryMktPayout(contract: Contract, bet: Bet) {
|
|
const { resolutionProbability, totalShares, phantomShares } = contract
|
|
const p =
|
|
resolutionProbability !== undefined
|
|
? resolutionProbability
|
|
: getProbability(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 deductFees(amount, winnings)
|
|
}
|
|
|
|
export function resolvedPayout(contract: Contract, bet: Bet) {
|
|
if (contract.resolution)
|
|
return calculatePayout(contract, bet, contract.resolution)
|
|
throw new Error('Contract was not resolved')
|
|
}
|
|
|
|
export const deductFees = (betAmount: number, winnings: number) => {
|
|
return winnings > betAmount
|
|
? betAmount + (1 - FEES) * (winnings - betAmount)
|
|
: winnings
|
|
}
|