2022-07-19 22:29:41 +00:00
|
|
|
import * as functions from 'firebase-functions'
|
|
|
|
import * as admin from 'firebase-admin'
|
|
|
|
import { uniq } from 'lodash'
|
2022-09-22 21:57:48 +00:00
|
|
|
import { Bet } from '../../common/bet'
|
|
|
|
import { Contract } from '../../common/contract'
|
2022-07-19 23:08:51 +00:00
|
|
|
import { log } from './utils'
|
2022-09-22 21:57:48 +00:00
|
|
|
import { removeUndefinedProps } from '../../common/util/object'
|
2022-09-26 21:43:27 +00:00
|
|
|
import { DAY_MS, HOUR_MS } from '../../common/util/time'
|
2022-07-19 22:29:41 +00:00
|
|
|
|
2022-09-22 21:57:48 +00:00
|
|
|
export const scoreContracts = functions
|
|
|
|
.runWith({ memory: '4GB', timeoutSeconds: 540 })
|
|
|
|
.pubsub.schedule('every 1 hours')
|
2022-07-19 22:29:41 +00:00
|
|
|
.onRun(async () => {
|
|
|
|
await scoreContractsInternal()
|
|
|
|
})
|
|
|
|
const firestore = admin.firestore()
|
|
|
|
|
|
|
|
async function scoreContractsInternal() {
|
|
|
|
const now = Date.now()
|
2022-09-26 21:43:27 +00:00
|
|
|
const hourAgo = now - HOUR_MS
|
|
|
|
const dayAgo = now - DAY_MS
|
|
|
|
const threeDaysAgo = now - DAY_MS * 3
|
2022-07-19 23:08:51 +00:00
|
|
|
const activeContractsSnap = await firestore
|
2022-07-19 22:29:41 +00:00
|
|
|
.collection('contracts')
|
2022-09-26 21:43:27 +00:00
|
|
|
.where('lastUpdatedTime', '>', hourAgo)
|
2022-07-19 22:29:41 +00:00
|
|
|
.get()
|
2022-07-19 23:08:51 +00:00
|
|
|
const activeContracts = activeContractsSnap.docs.map(
|
|
|
|
(doc) => doc.data() as Contract
|
|
|
|
)
|
|
|
|
// We have to downgrade previously active contracts to allow the new ones to bubble up
|
|
|
|
const previouslyActiveContractsSnap = await firestore
|
|
|
|
.collection('contracts')
|
|
|
|
.where('popularityScore', '>', 0)
|
|
|
|
.get()
|
|
|
|
const activeContractIds = activeContracts.map((c) => c.id)
|
|
|
|
const previouslyActiveContracts = previouslyActiveContractsSnap.docs
|
|
|
|
.map((doc) => doc.data() as Contract)
|
|
|
|
.filter((c) => !activeContractIds.includes(c.id))
|
|
|
|
|
|
|
|
const contracts = activeContracts.concat(previouslyActiveContracts)
|
|
|
|
log(`Found ${contracts.length} contracts to score`)
|
2022-07-19 22:29:41 +00:00
|
|
|
|
2022-07-19 23:08:51 +00:00
|
|
|
for (const contract of contracts) {
|
2022-07-19 22:29:41 +00:00
|
|
|
const bets = await firestore
|
2022-07-19 23:08:51 +00:00
|
|
|
.collection(`contracts/${contract.id}/bets`)
|
2022-09-26 21:43:27 +00:00
|
|
|
.where('createdTime', '>', threeDaysAgo)
|
2022-07-19 22:29:41 +00:00
|
|
|
.get()
|
|
|
|
const bettors = bets.docs
|
|
|
|
.map((doc) => doc.data() as Bet)
|
|
|
|
.map((bet) => bet.userId)
|
2022-09-22 21:57:48 +00:00
|
|
|
const popularityScore = uniq(bettors).length
|
|
|
|
|
2022-09-26 21:43:27 +00:00
|
|
|
const wasCreatedToday = contract.createdTime > dayAgo
|
|
|
|
|
2022-09-22 21:57:48 +00:00
|
|
|
let dailyScore: number | undefined
|
2022-09-26 21:43:27 +00:00
|
|
|
if (
|
|
|
|
contract.outcomeType === 'BINARY' &&
|
|
|
|
contract.mechanism === 'cpmm-1' &&
|
|
|
|
!wasCreatedToday
|
|
|
|
) {
|
2022-09-22 21:57:48 +00:00
|
|
|
const percentChange = Math.abs(contract.probChanges.day)
|
|
|
|
dailyScore = popularityScore * percentChange
|
|
|
|
}
|
|
|
|
|
|
|
|
if (
|
|
|
|
contract.popularityScore !== popularityScore ||
|
|
|
|
contract.dailyScore !== dailyScore
|
|
|
|
) {
|
2022-07-19 22:29:41 +00:00
|
|
|
await firestore
|
|
|
|
.collection('contracts')
|
2022-07-19 23:08:51 +00:00
|
|
|
.doc(contract.id)
|
2022-09-22 21:57:48 +00:00
|
|
|
.update(removeUndefinedProps({ popularityScore, dailyScore }))
|
|
|
|
}
|
2022-07-19 22:29:41 +00:00
|
|
|
}
|
|
|
|
}
|