import { app } from './init' import { getFirestore, doc, setDoc, deleteDoc, where, collection, query, getDocs, onSnapshot, orderBy, getDoc, } from 'firebase/firestore' import dayjs from 'dayjs' import { Bet, getRecentBets } from './bets' import _ from 'lodash' export type Contract = { id: string slug: string // auto-generated; must be unique creatorId: string creatorName: string creatorUsername: string question: string description: string // More info about what the contract is about outcomeType: 'BINARY' // | 'MULTI' | 'interval' | 'date' // outcomes: ['YES', 'NO'] startPool: { YES: number; NO: number } pool: { YES: number; NO: number } totalShares: { YES: number; NO: number } totalBets: { YES: number; NO: 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?: number // When the contract creator resolved the market resolution?: 'YES' | 'NO' | 'CANCEL' // Chosen by creator; must be one of outcomes } export function path(contract: Contract) { // For now, derive username from creatorName return `/${contract.creatorUsername}/${contract.slug}` } export function compute(contract: Contract) { const { pool, startPool, createdTime, resolutionTime, isResolved } = contract const truePool = pool.YES + pool.NO - startPool.YES - startPool.NO const prob = pool.YES ** 2 / (pool.YES ** 2 + pool.NO ** 2) const probPercent = Math.round(prob * 100) + '%' const startProb = startPool.YES ** 2 / (startPool.YES ** 2 + startPool.NO ** 2) const createdDate = dayjs(createdTime).format('MMM D') const resolvedDate = isResolved ? dayjs(resolutionTime).format('MMM D') : undefined return { truePool, probPercent, startProb, createdDate, resolvedDate } } 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 pushNewContract(contract: Omit) { const newContractRef = doc(contractCollection) const fullContract: Contract = { ...contract, id: newContractRef.id } await setDoc(newContractRef, fullContract) return fullContract } export async function getContractFromId(contractId: string) { const docRef = doc(db, 'contracts', contractId) const result = await getDoc(docRef) return result.exists() ? (result.data() as Contract) : undefined } export async function getContractFromSlug(slug: string) { const q = query(contractCollection, where('slug', '==', slug)) const snapshot = await getDocs(q) return snapshot.empty ? undefined : (snapshot.docs[0].data() as Contract) } export async function deleteContract(contractId: string) { const docRef = doc(db, 'contracts', contractId) await deleteDoc(docRef) } export async function listContracts(creatorId: string): Promise { const q = query( contractCollection, where('creatorId', '==', creatorId), orderBy('createdTime', 'desc') ) const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data() as Contract) } export async function listAllContracts(): Promise { const q = query(contractCollection, orderBy('createdTime', 'desc')) const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data() as Contract) } export function listenForContracts( setContracts: (contracts: Contract[]) => void ) { const q = query(contractCollection, orderBy('createdTime', 'desc')) return onSnapshot(q, (snap) => { setContracts(snap.docs.map((doc) => doc.data() as Contract)) }) } export function listenForContract( contractId: string, setContract: (contract: Contract | null) => void ) { const contractRef = doc(contractCollection, contractId) return onSnapshot(contractRef, (contractSnap) => { setContract((contractSnap.data() ?? null) as Contract | null) }) } export function computeHotContracts(recentBets: Bet[]) { const contractBets = _.groupBy(recentBets, (bet) => bet.contractId) const hotContractIds = _.sortBy(Object.keys(contractBets), (contractId) => _.sumBy(contractBets[contractId], (bet) => -1 * bet.amount) ).slice(0, 4) return hotContractIds } export async function getHotContracts() { const oneDay = 1000 * 60 * 60 * 24 const recentBets = await getRecentBets(oneDay) return computeHotContracts(recentBets) }