798253f887
* Challenge bets * Store avatar url * Fix before and after probs * Check balance before creation * Calculate winning shares * pretty * Change winning value * Set shares to equal each other * Fix share challenge link * pretty * remove lib refs * Probability of bet is set to market * Remove peer pill * Cleanup * Button on contract page * don't show challenge if not binary or if resolved * challenge button (WIP) * fix accept challenge: don't change pool/probability * Opengraph preview [WIP] * elim lib * Edit og card props * Change challenge text * New card gen attempt * Get challenge on server * challenge button styling * Use env domain * Remove other window ref * Use challenge creator as avatar * Remove user name * Remove s from property, replace prob with outcome * challenge form * share text * Add in challenge parts to template and url * Challenge url params optional * Add challenge params to parse request * Parse please * Don't remove prob * Challenge card styling * Challenge card styling * Challenge card styling * Challenge card styling * Challenge card styling * Challenge card styling * Challenge card styling * Challenge card styling * Add to readme about how to dev og-image * Add emojis * button: gradient background, 2xl size * beautify accept bet screen * update question button * Add separate challenge template * Accepted challenge sharing card, fix accept bet call * accept challenge button * challenge winner page * create challenge screen * Your outcome/cost=> acceptorOutcome/cost * New create challenge panel * Fix main merge * Add challenge slug to bet and filter by it * Center title * Add helper text * Add FAQ section * Lint * Columnize the user areas in preview link too * Absolutely position * Spacing * Orientation * Restyle challenges list, cache contract name * Make copying easy on mobile * Link spacing * Fix spacing * qr codes! * put your challenges first * eslint * Changes to contract buttons and create challenge modal * Change titles around for current bet * Add back in contract title after winning * Cleanup * Add challenge enabled flag * Spacing of switch button * Put sharing qr code in modal Co-authored-by: mantikoros <sgrugett@gmail.com>
165 lines
4.9 KiB
TypeScript
165 lines
4.9 KiB
TypeScript
import { z } from 'zod'
|
|
import { APIError, newEndpoint, validate } from './api'
|
|
import { log } from './utils'
|
|
import { Contract, CPMMBinaryContract } from '../../common/contract'
|
|
import { User } from '../../common/user'
|
|
import * as admin from 'firebase-admin'
|
|
import { FieldValue } from 'firebase-admin/firestore'
|
|
import { removeUndefinedProps } from '../../common/util/object'
|
|
import { Acceptance, Challenge } from '../../common/challenge'
|
|
import { CandidateBet } from '../../common/new-bet'
|
|
import { createChallengeAcceptedNotification } from './create-notification'
|
|
import { noFees } from '../../common/fees'
|
|
import { formatMoney, formatPercent } from '../../common/util/format'
|
|
|
|
const bodySchema = z.object({
|
|
contractId: z.string(),
|
|
challengeSlug: z.string(),
|
|
outcomeType: z.literal('BINARY'),
|
|
closeTime: z.number().gte(Date.now()),
|
|
})
|
|
const firestore = admin.firestore()
|
|
|
|
export const acceptchallenge = newEndpoint({}, async (req, auth) => {
|
|
const { challengeSlug, contractId } = validate(bodySchema, req.body)
|
|
|
|
const result = await firestore.runTransaction(async (trans) => {
|
|
const contractDoc = firestore.doc(`contracts/${contractId}`)
|
|
const userDoc = firestore.doc(`users/${auth.uid}`)
|
|
const challengeDoc = firestore.doc(
|
|
`contracts/${contractId}/challenges/${challengeSlug}`
|
|
)
|
|
const [contractSnap, userSnap, challengeSnap] = await trans.getAll(
|
|
contractDoc,
|
|
userDoc,
|
|
challengeDoc
|
|
)
|
|
if (!contractSnap.exists) throw new APIError(400, 'Contract not found.')
|
|
if (!userSnap.exists) throw new APIError(400, 'User not found.')
|
|
if (!challengeSnap.exists) throw new APIError(400, 'Challenge not found.')
|
|
|
|
const anyContract = contractSnap.data() as Contract
|
|
const user = userSnap.data() as User
|
|
const challenge = challengeSnap.data() as Challenge
|
|
|
|
if (challenge.acceptances.length > 0)
|
|
throw new APIError(400, 'Challenge already accepted.')
|
|
|
|
const creatorDoc = firestore.doc(`users/${challenge.creatorId}`)
|
|
const creatorSnap = await trans.get(creatorDoc)
|
|
if (!creatorSnap.exists) throw new APIError(400, 'User not found.')
|
|
const creator = creatorSnap.data() as User
|
|
|
|
const {
|
|
creatorAmount,
|
|
acceptorOutcome,
|
|
creatorOutcome,
|
|
creatorOutcomeProb,
|
|
acceptorAmount,
|
|
} = challenge
|
|
|
|
if (user.balance < acceptorAmount)
|
|
throw new APIError(400, 'Insufficient balance.')
|
|
|
|
const contract = anyContract as CPMMBinaryContract
|
|
const shares = (1 / creatorOutcomeProb) * creatorAmount
|
|
const createdTime = Date.now()
|
|
const probOfYes =
|
|
creatorOutcome === 'YES' ? creatorOutcomeProb : 1 - creatorOutcomeProb
|
|
|
|
log(
|
|
'Creating challenge bet for',
|
|
user.username,
|
|
shares,
|
|
acceptorOutcome,
|
|
'shares',
|
|
'at',
|
|
formatPercent(creatorOutcomeProb),
|
|
'for',
|
|
formatMoney(acceptorAmount)
|
|
)
|
|
|
|
const yourNewBet: CandidateBet = removeUndefinedProps({
|
|
orderAmount: acceptorAmount,
|
|
amount: acceptorAmount,
|
|
shares,
|
|
isCancelled: false,
|
|
contractId: contract.id,
|
|
outcome: acceptorOutcome,
|
|
probBefore: probOfYes,
|
|
probAfter: probOfYes,
|
|
loanAmount: 0,
|
|
createdTime,
|
|
fees: noFees,
|
|
challengeSlug: challenge.slug,
|
|
})
|
|
|
|
const yourNewBetDoc = contractDoc.collection('bets').doc()
|
|
trans.create(yourNewBetDoc, {
|
|
id: yourNewBetDoc.id,
|
|
userId: user.id,
|
|
...yourNewBet,
|
|
})
|
|
|
|
trans.update(userDoc, { balance: FieldValue.increment(-yourNewBet.amount) })
|
|
|
|
const creatorNewBet: CandidateBet = removeUndefinedProps({
|
|
orderAmount: creatorAmount,
|
|
amount: creatorAmount,
|
|
shares,
|
|
isCancelled: false,
|
|
contractId: contract.id,
|
|
outcome: creatorOutcome,
|
|
probBefore: probOfYes,
|
|
probAfter: probOfYes,
|
|
loanAmount: 0,
|
|
createdTime,
|
|
fees: noFees,
|
|
challengeSlug: challenge.slug,
|
|
})
|
|
const creatorBetDoc = contractDoc.collection('bets').doc()
|
|
trans.create(creatorBetDoc, {
|
|
id: creatorBetDoc.id,
|
|
userId: creator.id,
|
|
...creatorNewBet,
|
|
})
|
|
|
|
trans.update(creatorDoc, {
|
|
balance: FieldValue.increment(-creatorNewBet.amount),
|
|
})
|
|
|
|
const volume = contract.volume + yourNewBet.amount + creatorNewBet.amount
|
|
trans.update(contractDoc, { volume })
|
|
|
|
trans.update(
|
|
challengeDoc,
|
|
removeUndefinedProps({
|
|
acceptedByUserIds: [user.id],
|
|
acceptances: [
|
|
{
|
|
userId: user.id,
|
|
betId: yourNewBetDoc.id,
|
|
createdTime,
|
|
amount: acceptorAmount,
|
|
userUsername: user.username,
|
|
userName: user.name,
|
|
userAvatarUrl: user.avatarUrl,
|
|
} as Acceptance,
|
|
],
|
|
})
|
|
)
|
|
|
|
await createChallengeAcceptedNotification(
|
|
user,
|
|
creator,
|
|
challenge,
|
|
acceptorAmount,
|
|
contract
|
|
)
|
|
log('Done, sent notification.')
|
|
return yourNewBetDoc
|
|
})
|
|
|
|
return { betId: result.id }
|
|
})
|