223 lines
6.0 KiB
TypeScript
223 lines
6.0 KiB
TypeScript
import {
|
|
MemberInWorkspace,
|
|
PrismaClient,
|
|
WorkspaceRole,
|
|
} from '@typebot.io/prisma'
|
|
import { isDefined } from '@typebot.io/lib'
|
|
import { getChatsLimit, getStorageLimit } from '@typebot.io/lib/pricing'
|
|
import { promptAndSetEnvironment } from './utils'
|
|
import { TelemetryEvent } from '@typebot.io/schemas/features/telemetry'
|
|
import { sendTelemetryEvents } from '@typebot.io/lib/telemetry/sendTelemetryEvent'
|
|
import { Workspace } from '@typebot.io/schemas'
|
|
|
|
const prisma = new PrismaClient()
|
|
|
|
type WorkspaceForDigest = Pick<
|
|
Workspace,
|
|
| 'id'
|
|
| 'plan'
|
|
| 'customChatsLimit'
|
|
| 'customStorageLimit'
|
|
| 'additionalChatsIndex'
|
|
| 'additionalStorageIndex'
|
|
| 'isQuarantined'
|
|
> & {
|
|
members: (Pick<MemberInWorkspace, 'role'> & {
|
|
user: { id: string; email: string | null }
|
|
})[]
|
|
}
|
|
|
|
export const sendTotalResultsDigest = async () => {
|
|
await promptAndSetEnvironment('production')
|
|
|
|
console.log("Generating total results yesterday's digest...")
|
|
const todayMidnight = new Date()
|
|
todayMidnight.setUTCHours(0, 0, 0, 0)
|
|
const yesterday = new Date(todayMidnight)
|
|
yesterday.setDate(yesterday.getDate() - 1)
|
|
|
|
const results = await prisma.result.groupBy({
|
|
by: ['typebotId'],
|
|
_count: {
|
|
_all: true,
|
|
},
|
|
where: {
|
|
hasStarted: true,
|
|
createdAt: {
|
|
gte: yesterday,
|
|
lt: todayMidnight,
|
|
},
|
|
},
|
|
})
|
|
|
|
console.log(
|
|
`Found ${results.reduce(
|
|
(total, result) => total + result._count._all,
|
|
0
|
|
)} results collected yesterday.`
|
|
)
|
|
|
|
const workspaces = await prisma.workspace.findMany({
|
|
where: {
|
|
typebots: {
|
|
some: {
|
|
id: { in: results.map((result) => result.typebotId) },
|
|
},
|
|
},
|
|
},
|
|
select: {
|
|
id: true,
|
|
typebots: { select: { id: true } },
|
|
members: {
|
|
select: { user: { select: { id: true, email: true } }, role: true },
|
|
},
|
|
additionalChatsIndex: true,
|
|
additionalStorageIndex: true,
|
|
customChatsLimit: true,
|
|
customStorageLimit: true,
|
|
plan: true,
|
|
isQuarantined: true,
|
|
},
|
|
})
|
|
|
|
const resultsWithWorkspaces = results
|
|
.flatMap((result) => {
|
|
const workspace = workspaces.find((workspace) =>
|
|
workspace.typebots.some((typebot) => typebot.id === result.typebotId)
|
|
)
|
|
if (!workspace) return
|
|
return workspace.members
|
|
.filter((member) => member.role !== WorkspaceRole.GUEST)
|
|
.map((member, memberIndex) => ({
|
|
userId: member.user.id,
|
|
workspace: workspace,
|
|
typebotId: result.typebotId,
|
|
totalResultsYesterday: result._count._all,
|
|
isFirstOfKind: memberIndex === 0 ? (true as const) : undefined,
|
|
}))
|
|
})
|
|
.filter(isDefined)
|
|
|
|
console.log('Computing workspaces limits...')
|
|
|
|
const workspaceLimitReachedEvents = await sendAlertIfLimitReached(
|
|
resultsWithWorkspaces
|
|
.filter((result) => result.isFirstOfKind)
|
|
.map((result) => result.workspace)
|
|
)
|
|
|
|
const newResultsCollectedEvents = resultsWithWorkspaces.map(
|
|
(result) =>
|
|
({
|
|
name: 'New results collected',
|
|
userId: result.userId,
|
|
workspaceId: result.workspace.id,
|
|
typebotId: result.typebotId,
|
|
data: {
|
|
total: result.totalResultsYesterday,
|
|
isFirstOfKind: result.isFirstOfKind,
|
|
},
|
|
} satisfies TelemetryEvent)
|
|
)
|
|
|
|
await sendTelemetryEvents(
|
|
workspaceLimitReachedEvents.concat(newResultsCollectedEvents)
|
|
)
|
|
|
|
console.log(
|
|
`Sent ${workspaceLimitReachedEvents.length} workspace limit reached events.`
|
|
)
|
|
console.log(
|
|
`Sent ${newResultsCollectedEvents.length} new results collected events.`
|
|
)
|
|
}
|
|
|
|
const sendAlertIfLimitReached = async (
|
|
workspaces: WorkspaceForDigest[]
|
|
): Promise<TelemetryEvent[]> => {
|
|
const events: TelemetryEvent[] = []
|
|
const taggedWorkspaces: string[] = []
|
|
for (const workspace of workspaces) {
|
|
if (taggedWorkspaces.includes(workspace.id) || workspace.isQuarantined)
|
|
continue
|
|
taggedWorkspaces.push(workspace.id)
|
|
const { totalChatsUsed, totalStorageUsed } = await getUsage(workspace.id)
|
|
const totalStorageUsedInGb = totalStorageUsed / 1024 / 1024 / 1024
|
|
const chatsLimit = getChatsLimit(workspace)
|
|
const storageLimit = getStorageLimit(workspace)
|
|
if (
|
|
(chatsLimit > 0 && totalChatsUsed >= chatsLimit) ||
|
|
(storageLimit > 0 && totalStorageUsedInGb >= storageLimit)
|
|
) {
|
|
events.push(
|
|
...workspace.members
|
|
.filter((member) => member.role === WorkspaceRole.ADMIN)
|
|
.map(
|
|
(member) =>
|
|
({
|
|
name: 'Workspace limit reached',
|
|
userId: member.user.id,
|
|
workspaceId: workspace.id,
|
|
data: {
|
|
totalChatsUsed,
|
|
totalStorageUsed: totalStorageUsedInGb,
|
|
chatsLimit,
|
|
storageLimit,
|
|
},
|
|
} satisfies TelemetryEvent)
|
|
)
|
|
)
|
|
continue
|
|
}
|
|
}
|
|
return events
|
|
}
|
|
|
|
const getUsage = async (workspaceId: string) => {
|
|
const now = new Date()
|
|
const firstDayOfMonth = new Date(now.getFullYear(), now.getMonth(), 1)
|
|
const firstDayOfNextMonth = new Date(now.getFullYear(), now.getMonth() + 1, 1)
|
|
const typebots = await prisma.typebot.findMany({
|
|
where: {
|
|
workspace: {
|
|
id: workspaceId,
|
|
},
|
|
},
|
|
select: { id: true },
|
|
})
|
|
|
|
const [
|
|
totalChatsUsed,
|
|
{
|
|
_sum: { storageUsed: totalStorageUsed },
|
|
},
|
|
] = await Promise.all([
|
|
prisma.result.count({
|
|
where: {
|
|
typebotId: { in: typebots.map((typebot) => typebot.id) },
|
|
hasStarted: true,
|
|
createdAt: {
|
|
gte: firstDayOfMonth,
|
|
lt: firstDayOfNextMonth,
|
|
},
|
|
},
|
|
}),
|
|
prisma.answer.aggregate({
|
|
where: {
|
|
storageUsed: { gt: 0 },
|
|
result: {
|
|
typebotId: { in: typebots.map((typebot) => typebot.id) },
|
|
},
|
|
},
|
|
_sum: { storageUsed: true },
|
|
}),
|
|
])
|
|
|
|
return {
|
|
totalChatsUsed,
|
|
totalStorageUsed: totalStorageUsed ?? 0,
|
|
}
|
|
}
|
|
|
|
sendTotalResultsDigest().then()
|