import dayjs from 'dayjs' import { collection, deleteDoc, doc, getDoc, getDocs, limit, orderBy, query, setDoc, startAfter, updateDoc, where, } from 'firebase/firestore' import { sortBy, sum } from 'lodash' import { coll, getValues, listenForValue, listenForValues } from './utils' import { BinaryContract, Contract } from 'common/contract' import { getDpmProbability } from 'common/calculate-dpm' import { createRNG, shuffle } from 'common/util/random' import { getCpmmProbability } from 'common/calculate-cpmm' import { formatMoney, formatPercent } from 'common/util/format' import { DAY_MS } from 'common/util/time' import { MAX_FEED_CONTRACTS } from 'common/recommended-contracts' import { Bet } from 'common/bet' import { Comment } from 'common/comment' import { ENV_CONFIG } from 'common/envs/constants' export const contracts = coll('contracts') export type { Contract } export function contractPath(contract: Contract) { return `/${contract.creatorUsername}/${contract.slug}` } export function homeContractPath(contract: Contract) { return `/home?c=${contract.slug}` } export function contractUrl(contract: Contract) { return `https://${ENV_CONFIG.domain}${contractPath(contract)}` } export function contractMetrics(contract: Contract) { const { createdTime, resolutionTime, isResolved } = contract const createdDate = dayjs(createdTime).format('MMM D') const resolvedDate = isResolved ? dayjs(resolutionTime).format('MMM D') : undefined const volumeLabel = `${formatMoney(contract.volume)} bet` return { volumeLabel, createdDate, resolvedDate } } export function contractPool(contract: Contract) { return contract.mechanism === 'cpmm-1' ? formatMoney(contract.totalLiquidity) : contract.mechanism === 'dpm-2' ? formatMoney(sum(Object.values(contract.pool))) : 'Empty pool' } export function getBinaryProb(contract: BinaryContract) { const { pool, resolutionProbability, mechanism } = contract return ( resolutionProbability ?? (mechanism === 'cpmm-1' ? getCpmmProbability(pool, contract.p) : getDpmProbability(contract.totalShares)) ) } export function getBinaryProbPercent(contract: BinaryContract) { return formatPercent(getBinaryProb(contract)) } export function tradingAllowed(contract: Contract) { return ( !contract.isResolved && (!contract.closeTime || contract.closeTime > Date.now()) ) } // Push contract to Firestore export async function setContract(contract: Contract) { await setDoc(doc(contracts, contract.id), contract) } export async function updateContract( contractId: string, update: Partial ) { await updateDoc(doc(contracts, contractId), update) } export async function getContractFromId(contractId: string) { const result = await getDoc(doc(contracts, contractId)) return result.exists() ? result.data() : undefined } export async function getContractFromSlug(slug: string) { const q = query(contracts, where('slug', '==', slug)) const snapshot = await getDocs(q) return snapshot.empty ? undefined : snapshot.docs[0].data() } export async function deleteContract(contractId: string) { await deleteDoc(doc(contracts, contractId)) } export async function listContracts(creatorId: string): Promise { const q = query( contracts, where('creatorId', '==', creatorId), orderBy('createdTime', 'desc') ) const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data()) } export async function listContractsByGroupSlug( slug: string ): Promise { const q = query(contracts, where('groupSlugs', 'array-contains', slug)) const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data()) } export async function listTaggedContractsCaseInsensitive( tag: string ): Promise { const q = query( contracts, where('lowercaseTags', 'array-contains', tag.toLowerCase()), orderBy('createdTime', 'desc') ) const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data()) } export async function listAllContracts( n: number, before?: string ): Promise { let q = query(contracts, orderBy('createdTime', 'desc'), limit(n)) if (before != null) { const snap = await getDoc(doc(contracts, before)) q = query(q, startAfter(snap)) } const snapshot = await getDocs(q) return snapshot.docs.map((doc) => doc.data()) } export function listenForContracts( setContracts: (contracts: Contract[]) => void ) { const q = query(contracts, orderBy('createdTime', 'desc')) return listenForValues(q, setContracts) } export function listenForUserContracts( creatorId: string, setContracts: (contracts: Contract[]) => void ) { const q = query( contracts, where('creatorId', '==', creatorId), orderBy('createdTime', 'desc') ) return listenForValues(q, setContracts) } const activeContractsQuery = query( contracts, where('isResolved', '==', false), where('visibility', '==', 'public'), where('volume7Days', '>', 0) ) export function getActiveContracts() { return getValues(activeContractsQuery) } export function listenForActiveContracts( setContracts: (contracts: Contract[]) => void ) { return listenForValues(activeContractsQuery, setContracts) } const inactiveContractsQuery = query( contracts, where('isResolved', '==', false), where('closeTime', '>', Date.now()), where('visibility', '==', 'public'), where('volume24Hours', '==', 0) ) export function getInactiveContracts() { return getValues(inactiveContractsQuery) } export function listenForInactiveContracts( setContracts: (contracts: Contract[]) => void ) { return listenForValues(inactiveContractsQuery, setContracts) } const newContractsQuery = query( contracts, where('isResolved', '==', false), where('volume7Days', '==', 0), where('createdTime', '>', Date.now() - 7 * DAY_MS) ) export function listenForNewContracts( setContracts: (contracts: Contract[]) => void ) { return listenForValues(newContractsQuery, setContracts) } export function listenForContract( contractId: string, setContract: (contract: Contract | null) => void ) { const contractRef = doc(contracts, contractId) return listenForValue(contractRef, setContract) } function chooseRandomSubset(contracts: Contract[], count: number) { const fiveMinutes = 5 * 60 * 1000 const seed = Math.round(Date.now() / fiveMinutes).toString() shuffle(contracts, createRNG(seed)) return contracts.slice(0, count) } const hotContractsQuery = query( contracts, where('isResolved', '==', false), where('visibility', '==', 'public'), orderBy('volume24Hours', 'desc'), limit(16) ) export function listenForHotContracts( setHotContracts: (contracts: Contract[]) => void ) { return listenForValues(hotContractsQuery, (contracts) => { const hotContracts = sortBy( chooseRandomSubset(contracts, 4), (contract) => contract.volume24Hours ) setHotContracts(hotContracts) }) } export async function getHotContracts() { const data = await getValues(hotContractsQuery) return sortBy( chooseRandomSubset(data, 10), (contract) => -1 * contract.volume24Hours ) } export async function getContractsBySlugs(slugs: string[]) { const q = query(contracts, where('slug', 'in', slugs)) const snapshot = await getDocs(q) const data = snapshot.docs.map((doc) => doc.data()) return sortBy(data, (contract) => -1 * contract.volume24Hours) } const topWeeklyQuery = query( contracts, where('isResolved', '==', false), orderBy('volume7Days', 'desc'), limit(MAX_FEED_CONTRACTS) ) export async function getTopWeeklyContracts() { return await getValues(topWeeklyQuery) } const closingSoonQuery = query( contracts, where('isResolved', '==', false), where('visibility', '==', 'public'), where('closeTime', '>', Date.now()), orderBy('closeTime', 'asc'), limit(6) ) export async function getClosingSoonContracts() { const data = await getValues(closingSoonQuery) return sortBy(chooseRandomSubset(data, 2), (contract) => contract.closeTime) } export async function getRecentBetsAndComments(contract: Contract) { const contractDoc = doc(contracts, contract.id) const [recentBets, recentComments] = await Promise.all([ getValues( query( collection(contractDoc, 'bets'), where('createdTime', '>', Date.now() - DAY_MS), orderBy('createdTime', 'desc'), limit(1) ) ), getValues( query( collection(contractDoc, 'comments'), where('createdTime', '>', Date.now() - 3 * DAY_MS), orderBy('createdTime', 'desc'), limit(3) ) ), ]) return { contract, recentBets, recentComments, } }