From 94d3b2c2e7230c14aff48d64f04dc454bf762b9a Mon Sep 17 00:00:00 2001 From: Xiao Yijun Date: Mon, 4 Mar 2024 14:05:23 +0800 Subject: [PATCH] refactor(schemas): regenerate dau table data from logs (#5455) --- .../next-1709528944-regenerate-dau-data.ts | 49 +++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 packages/schemas/alterations/next-1709528944-regenerate-dau-data.ts diff --git a/packages/schemas/alterations/next-1709528944-regenerate-dau-data.ts b/packages/schemas/alterations/next-1709528944-regenerate-dau-data.ts new file mode 100644 index 000000000..1dc1ac5a6 --- /dev/null +++ b/packages/schemas/alterations/next-1709528944-regenerate-dau-data.ts @@ -0,0 +1,49 @@ +import { generateStandardId } from '@logto/shared/universal'; +import { sql } from 'slonik'; + +import type { AlterationScript } from '../lib/types/alteration.js'; + +type ActiveUserInteractionLog = { + tenantId: string; + userId: string; + createdAt: number; +}; + +const alteration: AlterationScript = { + up: async (pool) => { + // Delete all record from `daily_active_users` table + await pool.query(sql`delete from daily_active_users;`); + + // Retrieve all active user logs from `logs` table + const { rows: interactionLogs } = await pool.query(sql` + select tenant_id, payload->>'userId' as user_id, created_at + from logs + where payload->>'userId' is not null and key like 'ExchangeTokenBy.%' and payload->>'result' = 'Success' + `); + + if (interactionLogs.length === 0) { + console.log('No active user interaction logs found, skip alteration'); + return; + } + + // Generate DAU data from active user logs + for (const { tenantId, userId, createdAt } of interactionLogs) { + /** + * Note: we ignore the conflict here because conflict data may be inserted when staging. + */ + // eslint-disable-next-line no-await-in-loop + await pool.query(sql` + insert into daily_active_users (id, tenant_id, user_id, date) + values (${generateStandardId()},${tenantId}, ${userId}, ${new Date( + createdAt + ).toISOString()}) + on conflict do nothing; + `); + } + }, + down: async (pool) => { + // Cannot be reverted + }, +}; + +export default alteration;