add subsidy
This commit is contained in:
parent
4b8d381da5
commit
760ebcf38f
|
@ -1,4 +1,4 @@
|
||||||
import { addCpmmLiquidity, getCpmmLiquidity } from './calculate-cpmm'
|
import { getCpmmLiquidity } from './calculate-cpmm'
|
||||||
import { CPMMContract } from './contract'
|
import { CPMMContract } from './contract'
|
||||||
import { LiquidityProvision } from './liquidity-provision'
|
import { LiquidityProvision } from './liquidity-provision'
|
||||||
|
|
||||||
|
@ -8,25 +8,23 @@ export const getNewLiquidityProvision = (
|
||||||
contract: CPMMContract,
|
contract: CPMMContract,
|
||||||
newLiquidityProvisionId: string
|
newLiquidityProvisionId: string
|
||||||
) => {
|
) => {
|
||||||
const { pool, p, totalLiquidity } = contract
|
const { pool, p, totalLiquidity, subsidyPool } = contract
|
||||||
|
|
||||||
const { newPool, newP } = addCpmmLiquidity(pool, p, amount)
|
const liquidity = getCpmmLiquidity(pool, p)
|
||||||
|
|
||||||
const liquidity =
|
|
||||||
getCpmmLiquidity(newPool, newP) - getCpmmLiquidity(pool, newP)
|
|
||||||
|
|
||||||
const newLiquidityProvision: LiquidityProvision = {
|
const newLiquidityProvision: LiquidityProvision = {
|
||||||
id: newLiquidityProvisionId,
|
id: newLiquidityProvisionId,
|
||||||
userId: userId,
|
userId: userId,
|
||||||
contractId: contract.id,
|
contractId: contract.id,
|
||||||
amount,
|
amount,
|
||||||
pool: newPool,
|
pool,
|
||||||
p: newP,
|
p,
|
||||||
liquidity,
|
liquidity,
|
||||||
createdTime: Date.now(),
|
createdTime: Date.now(),
|
||||||
}
|
}
|
||||||
|
|
||||||
const newTotalLiquidity = (totalLiquidity ?? 0) + amount
|
const newTotalLiquidity = (totalLiquidity ?? 0) + amount
|
||||||
|
const newSubsidyPool = (subsidyPool ?? 0) + amount
|
||||||
|
|
||||||
return { newLiquidityProvision, newPool, newP, newTotalLiquidity }
|
return { newLiquidityProvision, newTotalLiquidity, newSubsidyPool }
|
||||||
}
|
}
|
||||||
|
|
|
@ -91,7 +91,8 @@ export type CPMM = {
|
||||||
mechanism: 'cpmm-1'
|
mechanism: 'cpmm-1'
|
||||||
pool: { [outcome: string]: number }
|
pool: { [outcome: string]: number }
|
||||||
p: number // probability constant in y^p * n^(1-p) = k
|
p: number // probability constant in y^p * n^(1-p) = k
|
||||||
totalLiquidity: number // in M$
|
totalLiquidity: number // for historical reasons, this the total subsidy amount added in M$
|
||||||
|
subsidyPool: number // current value of subsidy pool in M$
|
||||||
prob: number
|
prob: number
|
||||||
probChanges: {
|
probChanges: {
|
||||||
day: number
|
day: number
|
||||||
|
|
|
@ -111,6 +111,7 @@ const getBinaryCpmmProps = (initialProb: number, ante: number) => {
|
||||||
mechanism: 'cpmm-1',
|
mechanism: 'cpmm-1',
|
||||||
outcomeType: 'BINARY',
|
outcomeType: 'BINARY',
|
||||||
totalLiquidity: ante,
|
totalLiquidity: ante,
|
||||||
|
subsidyPool: 0,
|
||||||
initialProbability: p,
|
initialProbability: p,
|
||||||
p,
|
p,
|
||||||
pool: pool,
|
pool: pool,
|
||||||
|
|
78
functions/src/add-subsidy.ts
Normal file
78
functions/src/add-subsidy.ts
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
import * as admin from 'firebase-admin'
|
||||||
|
import { z } from 'zod'
|
||||||
|
|
||||||
|
import { Contract, CPMMContract } from '../../common/contract'
|
||||||
|
import { User } from '../../common/user'
|
||||||
|
import { getNewLiquidityProvision } from '../../common/add-liquidity'
|
||||||
|
import { APIError, newEndpoint, validate } from './api'
|
||||||
|
|
||||||
|
const bodySchema = z.object({
|
||||||
|
contractId: z.string(),
|
||||||
|
amount: z.number().gt(0),
|
||||||
|
})
|
||||||
|
|
||||||
|
export const addsubsidy = newEndpoint({}, async (req, auth) => {
|
||||||
|
const { amount, contractId } = validate(bodySchema, req.body)
|
||||||
|
|
||||||
|
if (!isFinite(amount) || amount < 1) throw new APIError(400, 'Invalid amount')
|
||||||
|
|
||||||
|
// run as transaction to prevent race conditions
|
||||||
|
return await firestore.runTransaction(async (transaction) => {
|
||||||
|
const userDoc = firestore.doc(`users/${auth.uid}`)
|
||||||
|
const userSnap = await transaction.get(userDoc)
|
||||||
|
if (!userSnap.exists) throw new APIError(400, 'User not found')
|
||||||
|
const user = userSnap.data() as User
|
||||||
|
|
||||||
|
const contractDoc = firestore.doc(`contracts/${contractId}`)
|
||||||
|
const contractSnap = await transaction.get(contractDoc)
|
||||||
|
if (!contractSnap.exists) throw new APIError(400, 'Invalid contract')
|
||||||
|
const contract = contractSnap.data() as Contract
|
||||||
|
if (
|
||||||
|
contract.mechanism !== 'cpmm-1' ||
|
||||||
|
(contract.outcomeType !== 'BINARY' &&
|
||||||
|
contract.outcomeType !== 'PSEUDO_NUMERIC')
|
||||||
|
)
|
||||||
|
throw new APIError(400, 'Invalid contract')
|
||||||
|
|
||||||
|
const { closeTime } = contract
|
||||||
|
if (closeTime && Date.now() > closeTime)
|
||||||
|
throw new APIError(400, 'Trading is closed')
|
||||||
|
|
||||||
|
if (user.balance < amount) throw new APIError(400, 'Insufficient balance')
|
||||||
|
|
||||||
|
const newLiquidityProvisionDoc = firestore
|
||||||
|
.collection(`contracts/${contractId}/liquidity`)
|
||||||
|
.doc()
|
||||||
|
|
||||||
|
const { newLiquidityProvision, newTotalLiquidity, newSubsidyPool } =
|
||||||
|
getNewLiquidityProvision(
|
||||||
|
user.id,
|
||||||
|
amount,
|
||||||
|
contract,
|
||||||
|
newLiquidityProvisionDoc.id
|
||||||
|
)
|
||||||
|
|
||||||
|
transaction.update(contractDoc, {
|
||||||
|
subsidyPool: newSubsidyPool,
|
||||||
|
totalLiquidity: newTotalLiquidity,
|
||||||
|
} as Partial<CPMMContract>)
|
||||||
|
|
||||||
|
const newBalance = user.balance - amount
|
||||||
|
const newTotalDeposits = user.totalDeposits - amount
|
||||||
|
|
||||||
|
if (!isFinite(newBalance)) {
|
||||||
|
throw new APIError(500, 'Invalid user balance for ' + user.username)
|
||||||
|
}
|
||||||
|
|
||||||
|
transaction.update(userDoc, {
|
||||||
|
balance: newBalance,
|
||||||
|
totalDeposits: newTotalDeposits,
|
||||||
|
})
|
||||||
|
|
||||||
|
transaction.create(newLiquidityProvisionDoc, newLiquidityProvision)
|
||||||
|
|
||||||
|
return newLiquidityProvision
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const firestore = admin.firestore()
|
|
@ -44,8 +44,6 @@ export * from './sell-bet'
|
||||||
export * from './sell-shares'
|
export * from './sell-shares'
|
||||||
export * from './claim-manalink'
|
export * from './claim-manalink'
|
||||||
export * from './create-market'
|
export * from './create-market'
|
||||||
export * from './add-liquidity'
|
|
||||||
export * from './withdraw-liquidity'
|
|
||||||
export * from './create-group'
|
export * from './create-group'
|
||||||
export * from './resolve-market'
|
export * from './resolve-market'
|
||||||
export * from './unsubscribe'
|
export * from './unsubscribe'
|
||||||
|
@ -53,6 +51,7 @@ export * from './stripe'
|
||||||
export * from './mana-bonus-email'
|
export * from './mana-bonus-email'
|
||||||
export * from './close-market'
|
export * from './close-market'
|
||||||
export * from './update-comment-bounty'
|
export * from './update-comment-bounty'
|
||||||
|
export * from './add-subsidy'
|
||||||
|
|
||||||
import { health } from './health'
|
import { health } from './health'
|
||||||
import { transact } from './transact'
|
import { transact } from './transact'
|
||||||
|
|
Loading…
Reference in New Issue
Block a user