2022-05-01 16:36:54 +00:00
|
|
|
import * as _ from 'lodash'
|
|
|
|
import * as functions from 'firebase-functions'
|
|
|
|
import * as admin from 'firebase-admin'
|
|
|
|
|
|
|
|
import { getValue, getValues } from './utils'
|
2022-05-10 20:59:38 +00:00
|
|
|
import { Contract } from 'common/contract'
|
|
|
|
import { logInterpolation } from 'common/util/math'
|
|
|
|
import { DAY_MS } from 'common/util/time'
|
2022-05-01 16:36:54 +00:00
|
|
|
import {
|
|
|
|
getProbability,
|
|
|
|
getOutcomeProbability,
|
|
|
|
getTopAnswer,
|
2022-05-10 20:59:38 +00:00
|
|
|
} from 'common/calculate'
|
|
|
|
import { Bet } from 'common/bet'
|
|
|
|
import { Comment } from 'common/comment'
|
|
|
|
import { User } from 'common/user'
|
2022-05-01 16:36:54 +00:00
|
|
|
import {
|
|
|
|
getContractScore,
|
|
|
|
MAX_FEED_CONTRACTS,
|
2022-05-10 20:59:38 +00:00
|
|
|
} from 'common/recommended-contracts'
|
2022-05-01 16:36:54 +00:00
|
|
|
import { callCloudFunction } from './call-cloud-function'
|
|
|
|
|
|
|
|
const firestore = admin.firestore()
|
|
|
|
|
|
|
|
export const updateFeed = functions.pubsub
|
|
|
|
.schedule('every 60 minutes')
|
|
|
|
.onRun(async () => {
|
|
|
|
const users = await getValues<User>(firestore.collection('users'))
|
|
|
|
|
|
|
|
const batchSize = 100
|
|
|
|
const userBatches: User[][] = []
|
|
|
|
for (let i = 0; i < users.length; i += batchSize) {
|
|
|
|
userBatches.push(users.slice(i, i + batchSize))
|
|
|
|
}
|
|
|
|
|
|
|
|
await Promise.all(
|
|
|
|
userBatches.map(async (users) =>
|
|
|
|
callCloudFunction('updateFeedBatch', { users })
|
|
|
|
)
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
export const updateFeedBatch = functions.https.onCall(
|
|
|
|
async (data: { users: User[] }) => {
|
|
|
|
const { users } = data
|
|
|
|
const contracts = await getFeedContracts()
|
|
|
|
|
|
|
|
await Promise.all(users.map((user) => doUserFeedUpdate(user, contracts)))
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
export async function getFeedContracts() {
|
|
|
|
// Get contracts bet on or created in last week.
|
|
|
|
const contracts = await Promise.all([
|
|
|
|
getValues<Contract>(
|
|
|
|
firestore
|
|
|
|
.collection('contracts')
|
|
|
|
.where('isResolved', '==', false)
|
|
|
|
.where('volume7Days', '>', 0)
|
|
|
|
),
|
|
|
|
|
|
|
|
getValues<Contract>(
|
|
|
|
firestore
|
|
|
|
.collection('contracts')
|
|
|
|
.where('isResolved', '==', false)
|
|
|
|
.where('createdTime', '>', Date.now() - DAY_MS * 7)
|
|
|
|
.where('volume7Days', '==', 0)
|
|
|
|
),
|
|
|
|
]).then(([activeContracts, inactiveContracts]) => {
|
|
|
|
const combined = [...activeContracts, ...inactiveContracts]
|
|
|
|
// Remove closed contracts.
|
|
|
|
return combined.filter((c) => (c.closeTime ?? Infinity) > Date.now())
|
|
|
|
})
|
|
|
|
|
|
|
|
return contracts
|
|
|
|
}
|
|
|
|
|
|
|
|
export const doUserFeedUpdate = async (user: User, contracts: Contract[]) => {
|
|
|
|
const userCacheCollection = firestore.collection(
|
|
|
|
`private-users/${user.id}/cache`
|
|
|
|
)
|
|
|
|
const [wordScores, lastViewedTime] = await Promise.all([
|
|
|
|
getValue<{ [word: string]: number }>(userCacheCollection.doc('wordScores')),
|
|
|
|
getValue<{ [contractId: string]: number }>(
|
|
|
|
userCacheCollection.doc('lastViewTime')
|
|
|
|
),
|
|
|
|
]).then((dicts) => dicts.map((dict) => dict ?? {}))
|
|
|
|
|
|
|
|
const scoredContracts = contracts.map((contract) => {
|
|
|
|
const score = scoreContract(
|
|
|
|
contract,
|
|
|
|
wordScores,
|
|
|
|
lastViewedTime[contract.id]
|
|
|
|
)
|
|
|
|
return [contract, score] as [Contract, number]
|
|
|
|
})
|
|
|
|
|
|
|
|
const sortedContracts = _.sortBy(
|
|
|
|
scoredContracts,
|
|
|
|
([_, score]) => score
|
|
|
|
).reverse()
|
|
|
|
|
|
|
|
// console.log(sortedContracts.map(([c, score]) => c.question + ': ' + score))
|
|
|
|
|
|
|
|
const feedContracts = sortedContracts
|
|
|
|
.slice(0, MAX_FEED_CONTRACTS)
|
|
|
|
.map(([c]) => c)
|
|
|
|
|
|
|
|
const feed = await Promise.all(
|
|
|
|
feedContracts.map((contract) => getRecentBetsAndComments(contract))
|
|
|
|
)
|
|
|
|
|
|
|
|
await userCacheCollection.doc('feed').set({ feed })
|
|
|
|
}
|
|
|
|
|
|
|
|
function scoreContract(
|
|
|
|
contract: Contract,
|
|
|
|
wordScores: { [word: string]: number },
|
|
|
|
viewTime: number | undefined
|
|
|
|
) {
|
|
|
|
const recommendationScore = getContractScore(contract, wordScores)
|
|
|
|
const activityScore = getActivityScore(contract, viewTime)
|
|
|
|
// const lastViewedScore = getLastViewedScore(viewTime)
|
|
|
|
return recommendationScore * activityScore
|
|
|
|
}
|
|
|
|
|
|
|
|
function getActivityScore(contract: Contract, viewTime: number | undefined) {
|
|
|
|
const { createdTime, lastBetTime, lastCommentTime, outcomeType } = contract
|
|
|
|
const hasNewComments =
|
|
|
|
lastCommentTime && (!viewTime || lastCommentTime > viewTime)
|
|
|
|
const newCommentScore = hasNewComments ? 1 : 0.5
|
|
|
|
|
|
|
|
const timeSinceLastComment = Date.now() - (lastCommentTime ?? createdTime)
|
|
|
|
const commentDaysAgo = timeSinceLastComment / DAY_MS
|
|
|
|
const commentTimeScore =
|
|
|
|
0.25 + 0.75 * (1 - logInterpolation(0, 3, commentDaysAgo))
|
|
|
|
|
|
|
|
const timeSinceLastBet = Date.now() - (lastBetTime ?? createdTime)
|
|
|
|
const betDaysAgo = timeSinceLastBet / DAY_MS
|
|
|
|
const betTimeScore = 0.5 + 0.5 * (1 - logInterpolation(0, 3, betDaysAgo))
|
|
|
|
|
|
|
|
let prob = 0.5
|
|
|
|
if (outcomeType === 'BINARY') {
|
|
|
|
prob = getProbability(contract)
|
|
|
|
} else if (outcomeType === 'FREE_RESPONSE') {
|
|
|
|
const topAnswer = getTopAnswer(contract)
|
|
|
|
if (topAnswer)
|
|
|
|
prob = Math.max(0.5, getOutcomeProbability(contract, topAnswer.id))
|
|
|
|
}
|
|
|
|
const frac = 1 - Math.abs(prob - 0.5) ** 2 / 0.25
|
|
|
|
const probScore = 0.5 + frac * 0.5
|
|
|
|
|
|
|
|
const { volume24Hours, volume7Days } = contract
|
|
|
|
const combinedVolume = Math.log(volume24Hours + 1) + Math.log(volume7Days + 1)
|
|
|
|
const volumeScore = 0.5 + 0.5 * logInterpolation(4, 20, combinedVolume)
|
|
|
|
|
|
|
|
const score =
|
|
|
|
newCommentScore * commentTimeScore * betTimeScore * probScore * volumeScore
|
|
|
|
|
|
|
|
// Map score to [0.5, 1] since no recent activty is not a deal breaker.
|
|
|
|
const mappedScore = 0.5 + 0.5 * score
|
|
|
|
const newMappedScore = 0.7 + 0.3 * score
|
|
|
|
|
|
|
|
const isNew = Date.now() < contract.createdTime + DAY_MS
|
|
|
|
return isNew ? newMappedScore : mappedScore
|
|
|
|
}
|
|
|
|
|
|
|
|
function getLastViewedScore(viewTime: number | undefined) {
|
|
|
|
if (viewTime === undefined) {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
const daysAgo = (Date.now() - viewTime) / DAY_MS
|
|
|
|
|
|
|
|
if (daysAgo < 0.5) {
|
|
|
|
const frac = logInterpolation(0, 0.5, daysAgo)
|
|
|
|
return 0.5 + 0.25 * frac
|
|
|
|
}
|
|
|
|
|
|
|
|
const frac = logInterpolation(0.5, 14, daysAgo)
|
|
|
|
return 0.75 + 0.25 * frac
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getRecentBetsAndComments(contract: Contract) {
|
|
|
|
const contractDoc = firestore.collection('contracts').doc(contract.id)
|
|
|
|
|
|
|
|
const [recentBets, recentComments] = await Promise.all([
|
|
|
|
getValues<Bet>(
|
|
|
|
contractDoc
|
|
|
|
.collection('bets')
|
|
|
|
.where('createdTime', '>', Date.now() - DAY_MS)
|
|
|
|
.orderBy('createdTime', 'desc')
|
|
|
|
.limit(1)
|
|
|
|
),
|
|
|
|
|
|
|
|
getValues<Comment>(
|
|
|
|
contractDoc
|
|
|
|
.collection('comments')
|
|
|
|
.where('createdTime', '>', Date.now() - 3 * DAY_MS)
|
|
|
|
.orderBy('createdTime', 'desc')
|
|
|
|
.limit(3)
|
|
|
|
),
|
|
|
|
])
|
|
|
|
|
|
|
|
return {
|
|
|
|
contract,
|
|
|
|
recentBets,
|
|
|
|
recentComments,
|
|
|
|
}
|
|
|
|
}
|