chore: avoid using lastSignedIn

This commit is contained in:
Ephraim Atta-Duncan
2025-01-29 07:25:07 +00:00
parent 413c3cbc6e
commit cdb980bd4c
3 changed files with 49 additions and 15 deletions

15
package-lock.json generated
View File

@ -35722,6 +35722,21 @@
"engines": { "engines": {
"node": ">=6" "node": ">=6"
} }
},
"packages/trpc/node_modules/@next/swc-win32-ia32-msvc": {
"version": "14.2.6",
"resolved": "https://registry.npmjs.org/@next/swc-win32-ia32-msvc/-/swc-win32-ia32-msvc-14.2.6.tgz",
"integrity": "sha512-hNukAxq7hu4o5/UjPp5jqoBEtrpCbOmnUqZSKNJG8GrUVzfq0ucdhQFVrHcLRMvQcwqqDh1a5AJN9ORnNDpgBQ==",
"cpu": [
"ia32"
],
"optional": true,
"os": [
"win32"
],
"engines": {
"node": ">= 10"
}
} }
} }
} }

View File

@ -1,7 +1,11 @@
import { DateTime } from 'luxon'; import { DateTime } from 'luxon';
import { prisma } from '@documenso/prisma'; import { kyselyPrisma, prisma, sql } from '@documenso/prisma';
import { DocumentStatus, SubscriptionStatus } from '@documenso/prisma/client'; import {
DocumentStatus,
SubscriptionStatus,
UserSecurityAuditLogType,
} from '@documenso/prisma/client';
export const getUsersCount = async () => { export const getUsersCount = async () => {
return await prisma.user.count(); return await prisma.user.count();
@ -94,17 +98,26 @@ type GetMonthlyActiveUsersQueryResult = Array<{
}>; }>;
export const getMonthlyActiveUsers = async () => { export const getMonthlyActiveUsers = async () => {
const result = await prisma.$queryRaw<GetMonthlyActiveUsersQueryResult>` const qb = kyselyPrisma.$kysely
SELECT .selectFrom('UserSecurityAuditLog')
DATE_TRUNC('month', "lastSignedIn") AS "month", .select(({ fn }) => [
COUNT(DISTINCT "id") as "count", fn<Date>('DATE_TRUNC', [sql.lit('MONTH'), 'UserSecurityAuditLog.createdAt']).as('month'),
SUM(COUNT(DISTINCT "id")) OVER (ORDER BY DATE_TRUNC('month', "lastSignedIn")) as "cume_count" fn.count('userId').distinct().as('count'),
FROM "User" fn
WHERE "lastSignedIn" >= NOW() - INTERVAL '1 year' AND "disabled" = false .sum(fn.count('userId').distinct())
GROUP BY DATE_TRUNC('month', "lastSignedIn") .over((ob) =>
ORDER BY "month" DESC // eslint-disable-next-line @typescript-eslint/no-explicit-any
LIMIT 12 ob.orderBy(fn('DATE_TRUNC', [sql.lit('MONTH'), 'UserSecurityAuditLog.createdAt']) as any),
`; )
.as('cume_count'),
])
// @ts-expect-error - Raw SQL enum casting not properly typed by Kysely
.where(sql`type = ${UserSecurityAuditLogType.SIGN_IN}::"UserSecurityAuditLogType"`)
.groupBy(({ fn }) => fn('DATE_TRUNC', [sql.lit('MONTH'), 'UserSecurityAuditLog.createdAt']))
.orderBy('month', 'desc')
.limit(12);
const result = await qb.execute();
return result.map((row) => ({ return result.map((row) => ({
month: DateTime.fromJSDate(row.month).toFormat('yyyy-MM'), month: DateTime.fromJSDate(row.month).toFormat('yyyy-MM'),

View File

@ -13,8 +13,14 @@ import { SUPPORTED_LANGUAGES } from '@documenso/lib/constants/i18n';
import { DEFAULT_DOCUMENT_TIME_ZONE, TIME_ZONES } from '@documenso/lib/constants/time-zones'; import { DEFAULT_DOCUMENT_TIME_ZONE, TIME_ZONES } from '@documenso/lib/constants/time-zones';
import type { TDocument } from '@documenso/lib/types/document'; import type { TDocument } from '@documenso/lib/types/document';
import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth'; import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth';
import { DocumentVisibility, TeamMemberRole } from '@documenso/prisma/client'; import {
import { DocumentStatus, type Field, type Recipient, SendStatus } from '@documenso/prisma/client'; DocumentStatus,
DocumentVisibility,
type Field,
type Recipient,
SendStatus,
TeamMemberRole,
} from '@documenso/prisma/client';
import { import {
DocumentGlobalAuthAccessSelect, DocumentGlobalAuthAccessSelect,
DocumentGlobalAuthAccessTooltip, DocumentGlobalAuthAccessTooltip,