import * as functions from 'firebase-functions' import * as admin from 'firebase-admin' import { compact } from 'lodash' import { getContract, getUser, getValues } from './utils' import { ContractComment } from '../../common/comment' import { Bet } from '../../common/bet' import { Answer } from '../../common/answer' import { getLargestPosition } from '../../common/calculate' import { maxBy } from 'lodash' import { createCommentOrAnswerOrUpdatedContractNotification, replied_users_info, } from './create-notification' import { parseMentions, richTextToString } from '../../common/util/parse' import { addUserToContractFollowers } from './follow-market' const firestore = admin.firestore() export const onCreateCommentOnContract = functions .runWith({ secrets: ['MAILGUN_KEY'] }) .firestore.document('contracts/{contractId}/comments/{commentId}') .onCreate(async (change, context) => { const { contractId } = context.params as { contractId: string } const { eventId } = context const contract = await getContract(contractId) if (!contract) throw new Error('Could not find contract corresponding with comment') await change.ref.update({ contractSlug: contract.slug, contractQuestion: contract.question, }) const comment = change.data() as ContractComment const lastCommentTime = comment.createdTime const commentCreator = await getUser(comment.userId) if (!commentCreator) throw new Error('Could not find comment creator') await addUserToContractFollowers(contract.id, commentCreator.id) await firestore .collection('contracts') .doc(contract.id) .update({ lastCommentTime, lastUpdatedTime: Date.now() }) const previousBetsQuery = await firestore .collection('contracts') .doc(contractId) .collection('bets') .where('createdTime', '<', comment.createdTime) .get() const previousBets = previousBetsQuery.docs.map((d) => d.data() as Bet) const position = getLargestPosition( contract, previousBets.filter((b) => b.userId === comment.userId && !b.isAnte) ) if (position) { const fields: { [k: string]: unknown } = { commenterPositionShares: position.shares, commenterPositionOutcome: position.outcome, } const previousProb = contract.outcomeType === 'BINARY' ? maxBy(previousBets, (bet) => bet.createdTime)?.probAfter : undefined if (previousProb != null) { fields.commenterPositionProb = previousProb } await change.ref.update(fields) } let bet: Bet | undefined let answer: Answer | undefined if (comment.answerOutcome) { answer = contract.outcomeType === 'FREE_RESPONSE' && contract.answers ? contract.answers?.find( (answer) => answer.id === comment.answerOutcome ) : undefined } else if (comment.betId) { const betSnapshot = await firestore .collection('contracts') .doc(contractId) .collection('bets') .doc(comment.betId) .get() bet = betSnapshot.data() as Bet answer = contract.outcomeType === 'FREE_RESPONSE' && contract.answers ? contract.answers.find((answer) => answer.id === bet?.outcome) : undefined await change.ref.update({ betOutcome: bet.outcome, betAmount: bet.amount, }) } const comments = await getValues( firestore.collection('contracts').doc(contractId).collection('comments') ) const repliedToType = answer ? 'answer' : comment.replyToCommentId ? 'comment' : undefined const repliedUserId = comment.replyToCommentId ? comments.find((c) => c.id === comment.replyToCommentId)?.userId : answer?.userId const mentionedUsers = compact(parseMentions(comment.content)) const repliedUsers: replied_users_info = {} // The parent of the reply chain could be a comment or an answer if (repliedUserId && repliedToType) repliedUsers[repliedUserId] = { repliedToType, repliedToAnswerText: answer ? answer.text : undefined, repliedToId: comment.replyToCommentId || answer?.id, bet: bet, } const commentsInSameReplyChain = comments.filter((c) => repliedToType === 'answer' ? c.answerOutcome === answer?.id : repliedToType === 'comment' ? c.replyToCommentId === comment.replyToCommentId : false ) // The rest of the children in the chain are always comments commentsInSameReplyChain.forEach((c) => { if (c.userId !== comment.userId && c.userId !== repliedUserId) { repliedUsers[c.userId] = { repliedToType: 'comment', repliedToAnswerText: undefined, repliedToId: c.id, bet: undefined, } } }) await createCommentOrAnswerOrUpdatedContractNotification( comment.id, 'comment', 'created', commentCreator, eventId, richTextToString(comment.content), contract, { repliedUsersInfo: repliedUsers, taggedUserIds: mentionedUsers, } ) })