2022-06-01 13:11:25 +00:00
|
|
|
import * as admin from 'firebase-admin'
|
|
|
|
import {
|
|
|
|
Notification,
|
|
|
|
notification_reason_types,
|
2022-06-06 17:36:59 +00:00
|
|
|
notification_source_update_types,
|
2022-06-01 13:11:25 +00:00
|
|
|
notification_source_types,
|
|
|
|
} from '../../common/notification'
|
|
|
|
import { User } from '../../common/user'
|
|
|
|
import { Contract } from '../../common/contract'
|
2022-06-06 17:36:59 +00:00
|
|
|
import { getUserByUsername, getValues } from './utils'
|
2022-06-01 13:11:25 +00:00
|
|
|
import { Comment } from '../../common/comment'
|
|
|
|
import { uniq } from 'lodash'
|
|
|
|
import { Bet } from '../../common/bet'
|
|
|
|
import { Answer } from '../../common/answer'
|
2022-06-06 17:36:59 +00:00
|
|
|
import { getContractBetMetrics } from '../../common/calculate'
|
|
|
|
import { removeUndefinedProps } from '../../common/util/object'
|
2022-06-01 13:11:25 +00:00
|
|
|
const firestore = admin.firestore()
|
|
|
|
|
|
|
|
type user_to_reason_texts = {
|
2022-06-06 17:36:59 +00:00
|
|
|
[userId: string]: { reason: notification_reason_types }
|
2022-06-01 13:11:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export const createNotification = async (
|
|
|
|
sourceId: string,
|
|
|
|
sourceType: notification_source_types,
|
2022-06-06 17:36:59 +00:00
|
|
|
sourceUpdateType: notification_source_update_types,
|
2022-06-01 13:11:25 +00:00
|
|
|
sourceUser: User,
|
2022-06-06 17:36:59 +00:00
|
|
|
idempotencyKey: string,
|
2022-06-08 14:43:24 +00:00
|
|
|
sourceText: string,
|
2022-06-06 17:36:59 +00:00
|
|
|
sourceContract?: Contract,
|
|
|
|
relatedSourceType?: notification_source_types,
|
2022-06-08 14:43:24 +00:00
|
|
|
relatedUserId?: string
|
2022-06-01 13:11:25 +00:00
|
|
|
) => {
|
|
|
|
const shouldGetNotification = (
|
|
|
|
userId: string,
|
|
|
|
userToReasonTexts: user_to_reason_texts
|
|
|
|
) => {
|
|
|
|
return (
|
|
|
|
sourceUser.id != userId &&
|
|
|
|
!Object.keys(userToReasonTexts).includes(userId)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const createUsersNotifications = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts
|
|
|
|
) => {
|
|
|
|
await Promise.all(
|
|
|
|
Object.keys(userToReasonTexts).map(async (userId) => {
|
|
|
|
const notificationRef = firestore
|
|
|
|
.collection(`/users/${userId}/notifications`)
|
|
|
|
.doc(idempotencyKey)
|
|
|
|
const notification: Notification = {
|
|
|
|
id: idempotencyKey,
|
|
|
|
userId,
|
|
|
|
reason: userToReasonTexts[userId].reason,
|
|
|
|
createdTime: Date.now(),
|
|
|
|
isSeen: false,
|
|
|
|
sourceId,
|
|
|
|
sourceType,
|
2022-06-06 17:36:59 +00:00
|
|
|
sourceUpdateType,
|
|
|
|
sourceContractId: sourceContract?.id,
|
2022-06-01 13:11:25 +00:00
|
|
|
sourceUserName: sourceUser.name,
|
|
|
|
sourceUserUsername: sourceUser.username,
|
|
|
|
sourceUserAvatarUrl: sourceUser.avatarUrl,
|
2022-06-08 14:43:24 +00:00
|
|
|
sourceText,
|
|
|
|
sourceContractTitle: sourceContract?.question,
|
|
|
|
sourceContractCreatorUsername: sourceContract?.creatorUsername,
|
|
|
|
sourceContractSlug: sourceContract?.slug,
|
2022-06-01 13:11:25 +00:00
|
|
|
}
|
2022-06-06 17:36:59 +00:00
|
|
|
await notificationRef.set(removeUndefinedProps(notification))
|
2022-06-01 13:11:25 +00:00
|
|
|
})
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-06-08 14:43:24 +00:00
|
|
|
const notifyUsersFollowers = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts
|
|
|
|
) => {
|
|
|
|
const followers = await firestore
|
|
|
|
.collectionGroup('follows')
|
|
|
|
.where('userId', '==', sourceUser.id)
|
|
|
|
.get()
|
|
|
|
|
|
|
|
followers.docs.forEach((doc) => {
|
|
|
|
const followerUserId = doc.ref.parent.parent?.id
|
|
|
|
if (
|
|
|
|
followerUserId &&
|
|
|
|
shouldGetNotification(followerUserId, userToReasonTexts)
|
|
|
|
) {
|
|
|
|
userToReasonTexts[followerUserId] = {
|
|
|
|
reason: 'you_follow_user',
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyRepliedUsers = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts
|
|
|
|
) => {
|
|
|
|
if (
|
|
|
|
!relatedSourceType ||
|
|
|
|
!relatedUserId ||
|
|
|
|
!shouldGetNotification(relatedUserId, userToReasonTexts)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
if (relatedSourceType === 'comment') {
|
|
|
|
userToReasonTexts[relatedUserId] = {
|
|
|
|
reason: 'reply_to_users_comment',
|
|
|
|
}
|
|
|
|
} else if (relatedSourceType === 'answer') {
|
|
|
|
userToReasonTexts[relatedUserId] = {
|
|
|
|
reason: 'reply_to_users_answer',
|
|
|
|
}
|
2022-06-06 16:54:25 +00:00
|
|
|
}
|
2022-06-06 17:36:59 +00:00
|
|
|
}
|
2022-06-01 13:11:25 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyFollowedUser = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts,
|
|
|
|
followedUserId: string
|
|
|
|
) => {
|
|
|
|
if (shouldGetNotification(followedUserId, userToReasonTexts))
|
|
|
|
userToReasonTexts[followedUserId] = {
|
|
|
|
reason: 'on_new_follow',
|
|
|
|
}
|
|
|
|
}
|
2022-06-01 13:11:25 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyTaggedUsers = async (userToReasonTexts: user_to_reason_texts) => {
|
|
|
|
if (!sourceText) return
|
|
|
|
const taggedUsers = sourceText.match(/@\w+/g)
|
|
|
|
if (!taggedUsers) return
|
|
|
|
// await all get tagged users:
|
|
|
|
const users = await Promise.all(
|
|
|
|
taggedUsers.map(async (username) => {
|
|
|
|
return await getUserByUsername(username.slice(1))
|
2022-06-06 16:54:25 +00:00
|
|
|
})
|
2022-06-06 17:36:59 +00:00
|
|
|
)
|
|
|
|
users.forEach((taggedUser) => {
|
|
|
|
if (taggedUser && shouldGetNotification(taggedUser.id, userToReasonTexts))
|
|
|
|
userToReasonTexts[taggedUser.id] = {
|
|
|
|
reason: 'tagged_user',
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-06-01 13:11:25 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyContractCreator = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts,
|
|
|
|
sourceContract: Contract
|
|
|
|
) => {
|
|
|
|
if (shouldGetNotification(sourceContract.creatorId, userToReasonTexts))
|
|
|
|
userToReasonTexts[sourceContract.creatorId] = {
|
|
|
|
reason: 'on_users_contract',
|
|
|
|
}
|
|
|
|
}
|
2022-06-01 13:11:25 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyOtherAnswerersOnContract = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts,
|
|
|
|
sourceContract: Contract
|
|
|
|
) => {
|
|
|
|
const answers = await getValues<Answer>(
|
|
|
|
firestore
|
|
|
|
.collection('contracts')
|
|
|
|
.doc(sourceContract.id)
|
|
|
|
.collection('answers')
|
|
|
|
)
|
|
|
|
const recipientUserIds = uniq(answers.map((answer) => answer.userId))
|
|
|
|
recipientUserIds.forEach((userId) => {
|
|
|
|
if (shouldGetNotification(userId, userToReasonTexts))
|
|
|
|
userToReasonTexts[userId] = {
|
|
|
|
reason: 'on_contract_with_users_answer',
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-06-01 13:11:25 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyOtherCommentersOnContract = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts,
|
|
|
|
sourceContract: Contract
|
|
|
|
) => {
|
|
|
|
const comments = await getValues<Comment>(
|
|
|
|
firestore
|
|
|
|
.collection('contracts')
|
|
|
|
.doc(sourceContract.id)
|
|
|
|
.collection('comments')
|
|
|
|
)
|
|
|
|
const recipientUserIds = uniq(comments.map((comment) => comment.userId))
|
|
|
|
recipientUserIds.forEach((userId) => {
|
|
|
|
if (shouldGetNotification(userId, userToReasonTexts))
|
|
|
|
userToReasonTexts[userId] = {
|
|
|
|
reason: 'on_contract_with_users_comment',
|
|
|
|
}
|
|
|
|
})
|
2022-06-01 13:11:25 +00:00
|
|
|
}
|
2022-06-06 16:52:11 +00:00
|
|
|
|
2022-06-06 17:36:59 +00:00
|
|
|
const notifyOtherBettorsOnContract = async (
|
|
|
|
userToReasonTexts: user_to_reason_texts,
|
|
|
|
sourceContract: Contract
|
|
|
|
) => {
|
|
|
|
const betsSnap = await firestore
|
|
|
|
.collection(`contracts/${sourceContract.id}/bets`)
|
|
|
|
.get()
|
|
|
|
const bets = betsSnap.docs.map((doc) => doc.data() as Bet)
|
|
|
|
// filter bets for only users that have an amount invested still
|
|
|
|
const recipientUserIds = uniq(bets.map((bet) => bet.userId)).filter(
|
|
|
|
(userId) => {
|
|
|
|
return (
|
|
|
|
getContractBetMetrics(
|
|
|
|
sourceContract,
|
|
|
|
bets.filter((bet) => bet.userId === userId)
|
|
|
|
).invested > 0
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
recipientUserIds.forEach((userId) => {
|
|
|
|
if (shouldGetNotification(userId, userToReasonTexts))
|
|
|
|
userToReasonTexts[userId] = {
|
|
|
|
reason: 'on_contract_with_users_shares_in',
|
|
|
|
}
|
|
|
|
})
|
2022-06-06 16:52:11 +00:00
|
|
|
}
|
2022-06-06 17:36:59 +00:00
|
|
|
|
|
|
|
// TODO: Update for liquidity.
|
|
|
|
// TODO: Notify users of their own closed but not resolved contracts.
|
|
|
|
const getUsersToNotify = async () => {
|
|
|
|
const userToReasonTexts: user_to_reason_texts = {}
|
|
|
|
// The following functions modify the userToReasonTexts object in place.
|
|
|
|
if (
|
|
|
|
sourceContract &&
|
|
|
|
(sourceType === 'comment' ||
|
|
|
|
sourceType === 'answer' ||
|
2022-06-08 14:43:24 +00:00
|
|
|
(sourceType === 'contract' &&
|
|
|
|
(sourceUpdateType === 'updated' || sourceUpdateType === 'resolved')))
|
2022-06-06 17:36:59 +00:00
|
|
|
) {
|
|
|
|
if (sourceType === 'comment') {
|
|
|
|
await notifyRepliedUsers(userToReasonTexts)
|
|
|
|
await notifyTaggedUsers(userToReasonTexts)
|
|
|
|
}
|
|
|
|
await notifyContractCreator(userToReasonTexts, sourceContract)
|
|
|
|
await notifyOtherAnswerersOnContract(userToReasonTexts, sourceContract)
|
|
|
|
await notifyOtherBettorsOnContract(userToReasonTexts, sourceContract)
|
|
|
|
await notifyOtherCommentersOnContract(userToReasonTexts, sourceContract)
|
|
|
|
} else if (sourceType === 'follow' && relatedUserId) {
|
|
|
|
await notifyFollowedUser(userToReasonTexts, relatedUserId)
|
2022-06-08 14:43:24 +00:00
|
|
|
} else if (sourceType === 'contract' && sourceUpdateType === 'created') {
|
|
|
|
await notifyUsersFollowers(userToReasonTexts)
|
2022-06-06 17:36:59 +00:00
|
|
|
}
|
|
|
|
return userToReasonTexts
|
|
|
|
}
|
|
|
|
|
|
|
|
const userToReasonTexts = await getUsersToNotify()
|
|
|
|
await createUsersNotifications(userToReasonTexts)
|
2022-06-01 13:11:25 +00:00
|
|
|
}
|