* 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>
112 lines
3.4 KiB
TypeScript
112 lines
3.4 KiB
TypeScript
import * as functions from 'firebase-functions'
|
|
import * as admin from 'firebase-admin'
|
|
|
|
import { Contract } from '../../common/contract'
|
|
import { User } from '../../common/user'
|
|
import { getNewMultiBetInfo } from '../../common/new-bet'
|
|
import { Answer } from '../../common/answer'
|
|
import { getValues } from './utils'
|
|
|
|
export const createAnswer = functions.runWith({ minInstances: 1 }).https.onCall(
|
|
async (
|
|
data: {
|
|
contractId: string
|
|
amount: number
|
|
text: string
|
|
},
|
|
context
|
|
) => {
|
|
const userId = context?.auth?.uid
|
|
if (!userId) return { status: 'error', message: 'Not authorized' }
|
|
|
|
const { contractId, amount, text } = data
|
|
|
|
if (amount <= 0 || isNaN(amount) || !isFinite(amount))
|
|
return { status: 'error', message: 'Invalid amount' }
|
|
|
|
if (!text || typeof text !== 'string' || text.length > 10000)
|
|
return { status: 'error', message: 'Invalid text' }
|
|
|
|
// Run as transaction to prevent race conditions.
|
|
return await firestore.runTransaction(async (transaction) => {
|
|
const userDoc = firestore.doc(`users/${userId}`)
|
|
const userSnap = await transaction.get(userDoc)
|
|
if (!userSnap.exists)
|
|
return { status: 'error', message: 'User not found' }
|
|
const user = userSnap.data() as User
|
|
|
|
if (user.balance < amount)
|
|
return { status: 'error', message: 'Insufficient balance' }
|
|
|
|
const contractDoc = firestore.doc(`contracts/${contractId}`)
|
|
const contractSnap = await transaction.get(contractDoc)
|
|
if (!contractSnap.exists)
|
|
return { status: 'error', message: 'Invalid contract' }
|
|
const contract = contractSnap.data() as Contract
|
|
|
|
if (contract.outcomeType !== 'FREE_RESPONSE')
|
|
return {
|
|
status: 'error',
|
|
message: 'Requires a free response contract',
|
|
}
|
|
|
|
const { closeTime } = contract
|
|
if (closeTime && Date.now() > closeTime)
|
|
return { status: 'error', message: 'Trading is closed' }
|
|
|
|
const [lastAnswer] = await getValues<Answer>(
|
|
firestore
|
|
.collection(`contracts/${contractId}/answers`)
|
|
.orderBy('number', 'desc')
|
|
.limit(1)
|
|
)
|
|
|
|
if (!lastAnswer)
|
|
return { status: 'error', message: 'Could not fetch last answer' }
|
|
|
|
const number = lastAnswer.number + 1
|
|
const id = `${number}`
|
|
|
|
const newAnswerDoc = firestore
|
|
.collection(`contracts/${contractId}/answers`)
|
|
.doc(id)
|
|
|
|
const answerId = newAnswerDoc.id
|
|
const { username, name, avatarUrl } = user
|
|
|
|
const answer: Answer = {
|
|
id,
|
|
number,
|
|
contractId,
|
|
createdTime: Date.now(),
|
|
userId: user.id,
|
|
username,
|
|
name,
|
|
avatarUrl,
|
|
text,
|
|
}
|
|
transaction.create(newAnswerDoc, answer)
|
|
|
|
const newBetDoc = firestore
|
|
.collection(`contracts/${contractId}/bets`)
|
|
.doc()
|
|
|
|
const { newBet, newPool, newTotalShares, newTotalBets, newBalance } =
|
|
getNewMultiBetInfo(user, answerId, amount, contract, newBetDoc.id)
|
|
|
|
transaction.create(newBetDoc, { ...newBet, isAnte: true })
|
|
transaction.update(contractDoc, {
|
|
pool: newPool,
|
|
totalShares: newTotalShares,
|
|
totalBets: newTotalBets,
|
|
answers: [...(contract.answers ?? []), answer],
|
|
})
|
|
transaction.update(userDoc, { balance: newBalance })
|
|
|
|
return { status: 'success', answerId, betId: newBetDoc.id }
|
|
})
|
|
}
|
|
)
|
|
|
|
const firestore = admin.firestore()
|