c9229ca2b8
* Add Firestore package and config * Upload basic Firebase Auth code * Basic ability to sign in and view profile * Move html head content to Next's _document * Apply dark theme to all DaisyUI components * Add contract page * Smaller width bet input * Allow users to create new contracts * Add back listenForContract * Add some buttons * Add Row, Col, and Spacer components * Implement skeleton ContractPage * Apply dark theme to all DaisyUI components * Fix hooks lints (#3) * Add background to bet panel * Sort contracts by creation time * Link to market creation from header * List your markets on account page * Set fullscreen black background * Correctly set seeds on new contracts * Code cleanups * Gratuitously cool font * Add creator name, fix ordering * Use Readex Pro as body font * Fixes according to code review Co-authored-by: jahooma <jahooma@gmail.com>
89 lines
2.3 KiB
TypeScript
89 lines
2.3 KiB
TypeScript
import { app } from './init'
|
|
import {
|
|
getFirestore,
|
|
doc,
|
|
setDoc,
|
|
deleteDoc,
|
|
where,
|
|
collection,
|
|
query,
|
|
getDocs,
|
|
onSnapshot,
|
|
orderBy,
|
|
} from 'firebase/firestore'
|
|
|
|
export type Contract = {
|
|
id: string // Chosen by creator; must be unique
|
|
creatorId: string
|
|
creatorName: string
|
|
|
|
question: string
|
|
description: string // More info about what the contract is about
|
|
outcomeType: 'BINARY' // | 'MULTI' | 'interval' | 'date'
|
|
// outcomes: ['YES', 'NO']
|
|
seedAmounts: { YES: number; NO: number } // seedBets: [number, number]
|
|
|
|
createdTime: number // Milliseconds since epoch
|
|
lastUpdatedTime: number // If the question or description was changed
|
|
closeTime?: number // When no more trading is allowed
|
|
|
|
// isResolved: boolean
|
|
resolutionTime?: 10293849 // When the contract creator resolved the market; 0 if unresolved
|
|
resolution?: 'YES' | 'NO' | 'CANCEL' // Chosen by creator; must be one of outcomes
|
|
}
|
|
|
|
export type Bet = {
|
|
id: string
|
|
userId: string
|
|
contractId: string
|
|
|
|
size: number // Amount of USD bid
|
|
outcome: 'YES' | 'NO' // Chosen outcome
|
|
createdTime: number
|
|
|
|
dpmWeight: number // Dynamic Parimutuel weight
|
|
}
|
|
|
|
const db = getFirestore(app)
|
|
const contractCollection = collection(db, 'contracts')
|
|
|
|
// Push contract to Firestore
|
|
export async function setContract(contract: Contract) {
|
|
const docRef = doc(db, 'contracts', contract.id)
|
|
await setDoc(docRef, contract)
|
|
}
|
|
|
|
export async function deleteContract(contractId: string) {
|
|
const docRef = doc(db, 'contracts', contractId)
|
|
await deleteDoc(docRef)
|
|
}
|
|
|
|
export async function listContracts(creatorId: string): Promise<Contract[]> {
|
|
const q = query(
|
|
contractCollection,
|
|
where('creatorId', '==', creatorId),
|
|
orderBy('createdTime', 'desc')
|
|
)
|
|
const snapshot = await getDocs(q)
|
|
const contracts: Contract[] = []
|
|
snapshot.forEach((doc) => contracts.push(doc.data() as Contract))
|
|
return contracts
|
|
}
|
|
|
|
export function listenForContract(
|
|
contractId: string,
|
|
setContract: (contract: Contract) => void
|
|
) {
|
|
const contractRef = doc(contractCollection, contractId)
|
|
return onSnapshot(contractRef, (contractSnap) => {
|
|
setContract(contractSnap.data() as Contract)
|
|
})
|
|
}
|
|
|
|
// Push bet to Firestore
|
|
// TODO: Should bets be subcollections under its contract?
|
|
export async function setBet(bet: Bet) {
|
|
const docRef = doc(db, 'bets', bet.id)
|
|
await setDoc(docRef, bet)
|
|
}
|