@happyvertical/smrt-reports 0.36.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +26 -0
- package/CLAUDE.md +1 -0
- package/LICENSE +7 -0
- package/dist/aggregate.d.ts +25 -0
- package/dist/aggregate.js +6 -0
- package/dist/aggregate.js.map +1 -0
- package/dist/compiler.d.ts +78 -0
- package/dist/compiler.js +218 -0
- package/dist/compiler.js.map +1 -0
- package/dist/decorators.d.ts +90 -0
- package/dist/decorators.js +84 -0
- package/dist/decorators.js.map +1 -0
- package/dist/index.d.ts +407 -0
- package/dist/index.js +162 -0
- package/dist/index.js.map +1 -0
- package/dist/manifest.json +1639 -0
- package/dist/refresh.d.ts +106 -0
- package/dist/refresh.js +778 -0
- package/dist/refresh.js.map +1 -0
- package/dist/scheduler.d.ts +112 -0
- package/dist/scheduler.js +449 -0
- package/dist/scheduler.js.map +1 -0
- package/dist/smrt-knowledge.json +584 -0
- package/dist/state.d.ts +108 -0
- package/dist/state.js +320 -0
- package/dist/state.js.map +1 -0
- package/dist/types.d.ts +114 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +87 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"refresh.js","sources":["../src/refresh.ts"],"sourcesContent":["import { createHash, randomUUID } from 'node:crypto';\nimport { ObjectRegistry, type SmrtObject } from '@happyvertical/smrt-core';\nimport { toSnakeCase } from '@happyvertical/smrt-core/utils';\nimport { getTenantId, withTenant } from '@happyvertical/smrt-tenancy';\nimport {\n buildAggregate,\n buildWhere,\n type DatabaseInterface,\n type SqlAdapterType,\n tableExists,\n validateColumnName,\n} from '@happyvertical/sql';\nimport {\n buildReportDefinition,\n compileReportDefinition,\n getReportGroupingColumns,\n} from './compiler.js';\nimport {\n assertReportTablesReady,\n REPORT_LOCKS_TABLE,\n REPORT_RUNS_TABLE,\n REPORT_WATERMARKS_TABLE,\n scopeKeyForTenant,\n} from './state.js';\nimport type {\n AggregateSelectExpr,\n AggregateSpec,\n ReportDefinition,\n ReportRefreshMode,\n ReportRefreshOptions,\n ReportRefreshResult,\n ReportSource,\n} from './types.js';\n\ntype ReportCtor = new (...args: any[]) => SmrtObject;\n\ninterface RefreshScope {\n tenantId: string | null;\n scopeKey: string;\n sourceTenantColumn: string | null;\n reportTenantColumn: string | null;\n}\n\ninterface WatermarkConfig {\n watermarkColumn: string;\n softDeleteColumn: string;\n}\n\ninterface RefreshWorkResult {\n rowCount: number;\n changedGroupCount: number;\n watermarkBefore?: string | null;\n watermarkAfter?: string | null;\n mode: ReportRefreshMode;\n}\n\ninterface ReportLockHandle {\n ownerId: string;\n release(): Promise<void>;\n}\n\nfunction isSqlAdapterType(value: unknown): value is SqlAdapterType {\n return (\n value === 'sqlite' ||\n value === 'postgres' ||\n value === 'duckdb' ||\n value === 'json'\n );\n}\n\nfunction adapterTypeFromDb(\n db: DatabaseInterface,\n fallback?: SqlAdapterType,\n): SqlAdapterType {\n if (isSqlAdapterType(fallback)) return fallback;\n\n const url = String(db.url ?? '');\n if (url.startsWith('postgres://') || url.startsWith('postgresql://')) {\n return 'postgres';\n }\n if (url.endsWith('.duckdb')) {\n return 'duckdb';\n }\n if (url === ':memory:' || url.endsWith('.sqlite') || url.endsWith('.db')) {\n return 'sqlite';\n }\n throw new Error(\n 'Report refresh requires a database adapter type. Pass { adapterType }.',\n );\n}\n\nfunction stableReportId(values: unknown[]): string {\n const hash = createHash('sha256')\n .update(JSON.stringify(values))\n .digest('hex');\n const variant = ((Number.parseInt(hash[16], 16) & 0x3) | 0x8).toString(16);\n return [\n hash.slice(0, 8),\n hash.slice(8, 12),\n `4${hash.slice(13, 16)}`,\n `${variant}${hash.slice(17, 20)}`,\n hash.slice(20, 32),\n ].join('-');\n}\n\nfunction getReportTableName(reportCtor: ReportCtor): string {\n const registered =\n ObjectRegistry.getClassByConstructor(reportCtor) ??\n ObjectRegistry.getClass(reportCtor.name);\n const tableName = registered\n ? ObjectRegistry.getTableName(registered.qualifiedName ?? registered.name)\n : ObjectRegistry.getTableName(reportCtor.name);\n if (!tableName) {\n throw new Error(`No report table registered for ${reportCtor.name}`);\n }\n return tableName;\n}\n\nfunction getActualGroupingColumns(\n definition: Pick<ReportDefinition, 'fields'>,\n): string[] {\n return definition.fields\n .filter(\n (field) =>\n field.report?.kind === 'group' || field.report?.kind === 'bucket',\n )\n .map((field) => field.columnName ?? toSnakeCase(field.fieldName));\n}\n\nfunction materializeRows(\n rows: Record<string, any>[],\n definition: ReportDefinition,\n refreshedAt: Date,\n scope: RefreshScope,\n): Record<string, any>[] {\n const groupingColumns = getActualGroupingColumns(definition);\n const now = refreshedAt.toISOString();\n\n return rows.map((row, index) => {\n const groupingValues =\n groupingColumns.length === 0\n ? [definition.reportClassName, scope.scopeKey, index]\n : [\n definition.reportClassName,\n scope.scopeKey,\n ...groupingColumns.map((column) => row[column]),\n ];\n const id = stableReportId(groupingValues);\n const tenantFields = scope.reportTenantColumn\n ? { [scope.reportTenantColumn]: scope.tenantId }\n : {};\n\n return {\n id,\n slug: id,\n context: scope.scopeKey,\n ...tenantFields,\n ...row,\n refreshed_at: now,\n created_at: now,\n updated_at: now,\n };\n });\n}\n\nfunction sourceClassName(source: ReportSource): string {\n if (typeof source === 'string') return source;\n return source.name;\n}\n\nasync function fieldColumn(\n className: string,\n requested: string,\n): Promise<string | null> {\n const fields = await ObjectRegistry.getAllFields(className);\n const requestedColumn = toSnakeCase(requested);\n\n for (const [fieldName] of fields.entries()) {\n if (fieldName === requested || toSnakeCase(fieldName) === requestedColumn) {\n return requestedColumn;\n }\n }\n\n return null;\n}\n\nasync function tenantColumn(className: string): Promise<string | null> {\n const registered = ObjectRegistry.getClass(className);\n const configuredField = registered?.tenantScopedConfig?.field;\n if (configuredField) {\n return toSnakeCase(configuredField);\n }\n\n const fields = await ObjectRegistry.getAllFields(className);\n for (const [fieldName, field] of fields.entries()) {\n if (field?._meta?.__tenancy?.isTenantIdField) {\n return toSnakeCase(fieldName);\n }\n }\n\n const tenantIdField = await fieldColumn(className, 'tenantId');\n return tenantIdField;\n}\n\nasync function resolveScope(\n definition: ReportDefinition,\n options: ReportRefreshOptions,\n): Promise<RefreshScope> {\n const tenantId =\n options.tenantId !== undefined ? options.tenantId : (getTenantId() ?? null);\n const sourceTenantColumn = await tenantColumn(definition.sourceClassName);\n const reportTenantColumn = await tenantColumn(definition.reportClassName);\n\n if (tenantId && !sourceTenantColumn) {\n throw new Error(\n `Tenant-scoped refresh for ${definition.reportClassName} requires source ` +\n `${definition.sourceClassName} to expose a tenantId field.`,\n );\n }\n if (tenantId && !reportTenantColumn) {\n throw new Error(\n `Tenant-scoped refresh for ${definition.reportClassName} requires the report table to expose a tenantId field.`,\n );\n }\n\n return {\n tenantId,\n scopeKey: scopeKeyForTenant(tenantId),\n sourceTenantColumn,\n reportTenantColumn,\n };\n}\n\nfunction tenantWhere(scope: RefreshScope): Record<string, unknown> {\n if (!scope.sourceTenantColumn) return {};\n return { [scope.sourceTenantColumn]: scope.tenantId };\n}\n\nfunction andWhere(\n base: AggregateSpec['where'],\n extra: Record<string, unknown>,\n): AggregateSpec['where'] {\n if (Object.keys(extra).length === 0) return base;\n if (!base) return extra;\n if (Array.isArray(base)) {\n return base.map((andGroup) => [...andGroup, extra]);\n }\n return { ...base, ...extra };\n}\n\nfunction andHaving(\n base: AggregateSpec['having'],\n extra: Record<string, unknown>,\n): AggregateSpec['having'] {\n if (Object.keys(extra).length === 0) return base;\n if (!base) return extra;\n if (Array.isArray(base)) {\n return base.map((andGroup) => [...andGroup, extra]);\n }\n return { ...base, ...extra };\n}\n\nfunction withRefreshFilters(\n spec: AggregateSpec,\n scope: RefreshScope,\n extraWhere: Record<string, unknown> = {},\n): AggregateSpec {\n return {\n ...spec,\n where: andWhere(spec.where, {\n ...tenantWhere(scope),\n ...extraWhere,\n }),\n };\n}\n\nasync function resolveWatermarkConfig(\n definition: ReportDefinition,\n): Promise<WatermarkConfig> {\n const configured = definition.refresh ?? {};\n const watermarkField = configured.watermarkColumn ?? 'updatedAt';\n const softDeleteField = configured.softDeleteColumn ?? 'deletedAt';\n const watermarkColumn = await fieldColumn(\n definition.sourceClassName,\n watermarkField,\n );\n const softDeleteColumn = await fieldColumn(\n definition.sourceClassName,\n softDeleteField,\n );\n\n if (!watermarkColumn) {\n throw new Error(\n `Incremental report ${definition.reportClassName} requires source ` +\n `${definition.sourceClassName} to define watermark column '${watermarkField}'.`,\n );\n }\n if (!softDeleteColumn) {\n throw new Error(\n `Incremental report ${definition.reportClassName} requires source ` +\n `${definition.sourceClassName} to define soft-delete column '${softDeleteField}'.`,\n );\n }\n\n return { watermarkColumn, softDeleteColumn };\n}\n\nfunction compileChangedGroupSpec(\n definition: ReportDefinition,\n scope: RefreshScope,\n watermark: WatermarkConfig,\n watermarkBefore: string,\n): AggregateSpec {\n const spec = compileReportDefinition(definition);\n const select = spec.select.filter(\n (expr): expr is AggregateSelectExpr => !('fn' in expr),\n );\n\n return withRefreshFilters(\n {\n ...spec,\n // Ignore the report filter while finding changed groups so rows that\n // just stopped matching `report.where` still trigger a recompute/delete.\n where: undefined,\n select:\n select.length > 0\n ? select\n : [{ fn: 'count', as: '__smrt_changed_count' }],\n groupBy: select.length > 0 ? spec.groupBy : [],\n having: undefined,\n },\n scope,\n { [`${watermark.watermarkColumn} >`]: watermarkBefore },\n );\n}\n\nfunction compileAffectedGroupSpec(\n definition: ReportDefinition,\n scope: RefreshScope,\n watermark: WatermarkConfig,\n groupRow: Record<string, unknown>,\n): AggregateSpec {\n const groupingColumns = getActualGroupingColumns(definition);\n const groupHaving = Object.fromEntries(\n groupingColumns.map((column) => [column, groupRow[column]]),\n );\n const spec = withRefreshFilters(compileReportDefinition(definition), scope, {\n [watermark.softDeleteColumn]: null,\n });\n return {\n ...spec,\n having: andHaving(spec.having, groupHaving),\n };\n}\n\nfunction conflictColumns(\n definition: ReportDefinition,\n scope: RefreshScope,\n): string[] {\n const groupingColumns = getActualGroupingColumns(definition);\n if (groupingColumns.length === 0) return ['id'];\n return [\n ...(scope.reportTenantColumn ? [scope.reportTenantColumn] : []),\n ...groupingColumns,\n ];\n}\n\nfunction predicateSql(\n tableName: string,\n predicate: Record<string, unknown>,\n): { sql: string; values: unknown[] } {\n validateColumnName(tableName);\n const clauses: string[] = [];\n const values: unknown[] = [];\n\n for (const [column, value] of Object.entries(predicate)) {\n validateColumnName(column);\n if (value === null || value === undefined) {\n clauses.push(`${column} IS NULL`);\n continue;\n }\n clauses.push(`${column} = ?`);\n values.push(value);\n }\n\n return {\n sql: clauses.length > 0 ? `WHERE ${clauses.join(' AND ')}` : '',\n values,\n };\n}\n\nfunction dedupeGroupRows(\n definition: ReportDefinition,\n rows: Array<Record<string, unknown> | null>,\n): Record<string, unknown>[] {\n const groupingColumns = getActualGroupingColumns(definition);\n const groups = new Map<string, Record<string, unknown>>();\n\n for (const row of rows) {\n if (!row) continue;\n const groupRow =\n groupingColumns.length === 0\n ? {}\n : Object.fromEntries(\n groupingColumns.map((column) => [column, row[column] ?? null]),\n );\n const key =\n groupingColumns.length === 0\n ? '__smrt_global_group__'\n : JSON.stringify(groupingColumns.map((column) => groupRow[column]));\n if (!groups.has(key)) {\n groups.set(key, groupRow);\n }\n }\n\n return [...groups.values()];\n}\n\nasync function readMaterializedGroups(\n db: DatabaseInterface,\n tableName: string,\n definition: ReportDefinition,\n scope: RefreshScope,\n): Promise<Record<string, unknown>[]> {\n const groupingColumns = getActualGroupingColumns(definition);\n if (groupingColumns.length === 0) {\n return [{}];\n }\n\n const where = scope.reportTenantColumn\n ? predicateSql(tableName, { [scope.reportTenantColumn]: scope.tenantId })\n : { sql: '', values: [] as unknown[] };\n const columns = groupingColumns.map((column) => validateColumnName(column));\n const result = await db.query(\n `SELECT ${columns.join(', ')} FROM ${validateColumnName(tableName)} ${where.sql}`,\n ...where.values,\n );\n\n return result.rows\n .map((row) => parseMaybeJson<Record<string, unknown>>(row))\n .filter((row): row is Record<string, unknown> => Boolean(row));\n}\n\nasync function deleteMaterializedGroup(\n db: DatabaseInterface,\n tableName: string,\n definition: ReportDefinition,\n scope: RefreshScope,\n groupRow: Record<string, unknown>,\n): Promise<void> {\n const predicate: Record<string, unknown> = {};\n if (scope.reportTenantColumn) {\n predicate[scope.reportTenantColumn] = scope.tenantId;\n }\n for (const column of getActualGroupingColumns(definition)) {\n predicate[column] = groupRow[column];\n }\n\n const where = predicateSql(tableName, predicate);\n await db.query(\n `DELETE FROM ${validateColumnName(tableName)} ${where.sql}`,\n ...where.values,\n );\n}\n\nasync function replaceRows(\n db: DatabaseInterface,\n tableName: string,\n rows: Record<string, any>[],\n scope: RefreshScope,\n): Promise<void> {\n validateColumnName(tableName);\n const run = async (tx: DatabaseInterface) => {\n if (scope.reportTenantColumn) {\n const where = predicateSql(tableName, {\n [scope.reportTenantColumn]: scope.tenantId,\n });\n await tx.query(`DELETE FROM ${tableName} ${where.sql}`, ...where.values);\n } else {\n await tx.query(`DELETE FROM ${tableName}`);\n }\n if (rows.length > 0) {\n await tx.insert(tableName, rows);\n }\n };\n\n if (db.transaction) {\n await db.transaction(run);\n return;\n }\n\n await run(db);\n}\n\nfunction parseMaybeJson<T>(value: T | string | null | undefined): T | null {\n if (value == null) return null;\n if (typeof value !== 'string') return value;\n try {\n return JSON.parse(value) as T;\n } catch {\n return value as T;\n }\n}\n\nfunction watermarkValue(value: unknown): string | null {\n if (value === null || value === undefined) return null;\n if (value instanceof Date) return value.toISOString();\n return String(value);\n}\n\nasync function readWatermark(\n db: DatabaseInterface,\n definition: ReportDefinition,\n scope: RefreshScope,\n watermarkColumn: string,\n): Promise<string | null> {\n const result = await db.query(\n `SELECT watermark_value FROM ${REPORT_WATERMARKS_TABLE}\n WHERE report_class = ?\n AND scope_key = ?\n AND source_class = ?\n AND watermark_column = ?\n LIMIT 1`,\n definition.reportClassName,\n scope.scopeKey,\n definition.sourceClassName,\n watermarkColumn,\n );\n const row = result.rows[0] as { watermark_value?: unknown } | undefined;\n return watermarkValue(row?.watermark_value);\n}\n\nasync function writeWatermark(\n db: DatabaseInterface,\n definition: ReportDefinition,\n scope: RefreshScope,\n watermarkColumn: string,\n value: string | null,\n runId: string | undefined,\n): Promise<void> {\n if (value === null) return;\n const id = stableReportId([\n 'watermark',\n definition.reportClassName,\n scope.scopeKey,\n definition.sourceClassName,\n watermarkColumn,\n ]);\n const now = new Date().toISOString();\n await db.upsert(\n REPORT_WATERMARKS_TABLE,\n ['report_class', 'scope_key', 'source_class', 'watermark_column'],\n {\n id,\n slug: id,\n context: scope.scopeKey,\n tenant_id: scope.tenantId,\n scope_key: scope.scopeKey,\n report_class: definition.reportClassName,\n source_class: definition.sourceClassName,\n watermark_column: watermarkColumn,\n watermark_value: value,\n last_run_id: runId ?? null,\n created_at: now,\n updated_at: now,\n },\n );\n}\n\nasync function maxWatermark(\n db: DatabaseInterface,\n definition: ReportDefinition,\n scope: RefreshScope,\n watermarkColumn: string,\n adapterType: SqlAdapterType,\n): Promise<string | null> {\n const where = andWhere(undefined, tenantWhere(scope));\n const builtWhere = where\n ? buildWhere(where, 1, adapterType)\n : { sql: '', values: [] };\n const result = await db.query(\n `SELECT MAX(${validateColumnName(watermarkColumn)}) AS watermark FROM ${validateColumnName(definition.sourceTable)} ${builtWhere.sql}`,\n ...builtWhere.values,\n );\n const row = result.rows[0] as { watermark?: unknown } | undefined;\n return watermarkValue(row?.watermark);\n}\n\nasync function startRun(\n db: DatabaseInterface,\n definition: ReportDefinition,\n scope: RefreshScope,\n options: ReportRefreshOptions,\n mode: ReportRefreshMode,\n watermarkBefore?: string | null,\n): Promise<string | undefined> {\n if (options.trackRuns === false) return undefined;\n const id = randomUUID();\n const now = new Date().toISOString();\n await db.insert(REPORT_RUNS_TABLE, {\n id,\n slug: id,\n context: scope.scopeKey,\n tenant_id: scope.tenantId,\n scope_key: scope.scopeKey,\n report_class: definition.reportClassName,\n source_class: definition.sourceClassName,\n mode,\n trigger: options.trigger ?? 'manual',\n status: 'running',\n started_at: now,\n row_count: 0,\n changed_group_count: 0,\n watermark_before: watermarkBefore ?? null,\n watermark_after: null,\n error: null,\n metadata: {\n scheduleId: options.scheduleId,\n },\n created_at: now,\n updated_at: now,\n });\n return id;\n}\n\nasync function completeRun(\n db: DatabaseInterface,\n runId: string | undefined,\n status: 'success' | 'failed' | 'skipped',\n data: {\n rowCount?: number;\n changedGroupCount?: number;\n watermarkAfter?: string | null;\n error?: unknown;\n } = {},\n): Promise<void> {\n if (!runId) return;\n const error =\n data.error instanceof Error\n ? data.error.message\n : data.error\n ? String(data.error)\n : null;\n await db.update(\n REPORT_RUNS_TABLE,\n { id: runId },\n {\n status,\n completed_at: new Date().toISOString(),\n row_count: data.rowCount ?? 0,\n changed_group_count: data.changedGroupCount ?? 0,\n watermark_after: data.watermarkAfter ?? null,\n error,\n updated_at: new Date().toISOString(),\n },\n );\n}\n\nasync function acquireLock(\n db: DatabaseInterface,\n definition: ReportDefinition,\n scope: RefreshScope,\n ttlMs: number,\n): Promise<ReportLockHandle | null> {\n const now = new Date();\n const nowIso = now.toISOString();\n const ownerId = randomUUID();\n const expiresAt = new Date(now.getTime() + ttlMs);\n const expiresAtIso = expiresAt.toISOString();\n const id = stableReportId([\n 'lock',\n definition.reportClassName,\n scope.scopeKey,\n ]);\n await db.query(\n `INSERT INTO ${REPORT_LOCKS_TABLE} (\n id,\n slug,\n context,\n tenant_id,\n scope_key,\n report_class,\n owner_id,\n acquired_at,\n heartbeat_at,\n expires_at,\n created_at,\n updated_at\n )\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n ON CONFLICT(report_class, scope_key) DO UPDATE SET\n tenant_id = excluded.tenant_id,\n owner_id = excluded.owner_id,\n acquired_at = excluded.acquired_at,\n heartbeat_at = excluded.heartbeat_at,\n expires_at = excluded.expires_at,\n updated_at = excluded.updated_at\n WHERE ${REPORT_LOCKS_TABLE}.owner_id IS NULL\n OR ${REPORT_LOCKS_TABLE}.expires_at IS NULL\n OR ${REPORT_LOCKS_TABLE}.expires_at <= ?`,\n id,\n id,\n scope.scopeKey,\n scope.tenantId,\n scope.scopeKey,\n definition.reportClassName,\n ownerId,\n nowIso,\n nowIso,\n expiresAtIso,\n nowIso,\n nowIso,\n nowIso,\n );\n\n const claimed = await db.query(\n `SELECT owner_id FROM ${REPORT_LOCKS_TABLE}\n WHERE report_class = ? AND scope_key = ?\n LIMIT 1`,\n definition.reportClassName,\n scope.scopeKey,\n );\n const row = claimed.rows[0] as { owner_id?: unknown } | undefined;\n if (row?.owner_id !== ownerId) {\n return null;\n }\n\n return {\n ownerId,\n async release() {\n await db.query(\n `UPDATE ${REPORT_LOCKS_TABLE}\n SET owner_id = NULL,\n heartbeat_at = NULL,\n expires_at = NULL,\n updated_at = ?\n WHERE report_class = ?\n AND scope_key = ?\n AND owner_id = ?`,\n new Date().toISOString(),\n definition.reportClassName,\n scope.scopeKey,\n ownerId,\n );\n },\n };\n}\n\nasync function rebuildReport(\n db: DatabaseInterface,\n definition: ReportDefinition,\n reportTable: string,\n scope: RefreshScope,\n adapterType: SqlAdapterType,\n runId: string | undefined,\n watermark?: WatermarkConfig | null,\n): Promise<RefreshWorkResult> {\n const explicitWatermarkColumn =\n definition.refresh?.watermarkColumn &&\n (await fieldColumn(\n definition.sourceClassName,\n definition.refresh.watermarkColumn,\n ));\n const explicitSoftDeleteColumn =\n definition.refresh?.softDeleteColumn &&\n (await fieldColumn(\n definition.sourceClassName,\n definition.refresh.softDeleteColumn,\n ));\n const watermarkColumn =\n watermark?.watermarkColumn ?? explicitWatermarkColumn ?? null;\n const softDeleteColumn =\n watermark?.softDeleteColumn ?? explicitSoftDeleteColumn ?? null;\n const deletedFilter = softDeleteColumn ? { [softDeleteColumn]: null } : {};\n const spec = withRefreshFilters(\n compileReportDefinition(definition),\n scope,\n deletedFilter,\n );\n const aggregate = buildAggregate(spec, 1, adapterType);\n const result = await db.query(aggregate.sql, ...aggregate.values);\n const refreshedAt = new Date();\n const rows = materializeRows(result.rows, definition, refreshedAt, scope);\n\n await replaceRows(db, reportTable, rows, scope);\n\n const watermarkAfter = watermarkColumn\n ? await maxWatermark(db, definition, scope, watermarkColumn, adapterType)\n : null;\n if (watermarkColumn) {\n await writeWatermark(\n db,\n definition,\n scope,\n watermarkColumn,\n watermarkAfter,\n runId,\n );\n }\n\n return {\n rowCount: rows.length,\n changedGroupCount: rows.length,\n watermarkAfter,\n mode: 'rebuild',\n };\n}\n\nasync function incrementalReport(\n db: DatabaseInterface,\n definition: ReportDefinition,\n reportTable: string,\n scope: RefreshScope,\n adapterType: SqlAdapterType,\n runId: string | undefined,\n changedRows: Record<string, unknown>[] = [],\n): Promise<RefreshWorkResult> {\n const watermark = await resolveWatermarkConfig(definition);\n const watermarkBefore = await readWatermark(\n db,\n definition,\n scope,\n watermark.watermarkColumn,\n );\n\n if (!watermarkBefore) {\n const seeded = await rebuildReport(\n db,\n definition,\n reportTable,\n scope,\n adapterType,\n runId,\n watermark,\n );\n return {\n ...seeded,\n watermarkBefore,\n };\n }\n\n const changedSpec = compileChangedGroupSpec(\n definition,\n scope,\n watermark,\n watermarkBefore,\n );\n const changedAggregate = buildAggregate(changedSpec, 1, adapterType);\n const changed = await db.query(\n changedAggregate.sql,\n ...changedAggregate.values,\n );\n const changedGroups = changed.rows.map((row) =>\n parseMaybeJson<Record<string, unknown>>(row),\n );\n const materializedGroups =\n changedGroups.length > 0 || changedRows.length > 0\n ? await readMaterializedGroups(db, reportTable, definition, scope)\n : [];\n const affectedGroups = dedupeGroupRows(definition, [\n ...changedGroups,\n ...materializedGroups,\n ]);\n\n if (affectedGroups.length === 0) {\n return {\n rowCount: 0,\n changedGroupCount: 0,\n watermarkBefore,\n watermarkAfter: watermarkBefore,\n mode: 'incremental',\n };\n }\n\n let rowCount = 0;\n const conflicts = conflictColumns(definition, scope);\n\n for (const groupRow of affectedGroups) {\n const affectedSpec = compileAffectedGroupSpec(\n definition,\n scope,\n watermark,\n groupRow,\n );\n const aggregate = buildAggregate(affectedSpec, 1, adapterType);\n const result = await db.query(aggregate.sql, ...aggregate.values);\n const refreshedAt = new Date();\n const rows = materializeRows(result.rows, definition, refreshedAt, scope);\n\n if (rows.length === 0) {\n await deleteMaterializedGroup(\n db,\n reportTable,\n definition,\n scope,\n groupRow,\n );\n continue;\n }\n\n for (const row of rows) {\n await db.upsert(reportTable, conflicts, row);\n rowCount += 1;\n }\n }\n\n const watermarkAfter = await maxWatermark(\n db,\n definition,\n scope,\n watermark.watermarkColumn,\n adapterType,\n );\n await writeWatermark(\n db,\n definition,\n scope,\n watermark.watermarkColumn,\n watermarkAfter,\n runId,\n );\n\n return {\n rowCount,\n changedGroupCount: affectedGroups.length,\n watermarkBefore,\n watermarkAfter,\n mode: 'incremental',\n };\n}\n\nfunction requiredReportRuntimeTables(\n mode: ReportRefreshMode,\n options: ReportRefreshOptions,\n): string[] {\n const tables = new Set<string>();\n if (options.trackRuns !== false) tables.add(REPORT_RUNS_TABLE);\n if (mode === 'incremental') tables.add(REPORT_WATERMARKS_TABLE);\n if (options.lock !== false) tables.add(REPORT_LOCKS_TABLE);\n return [...tables];\n}\n\nasync function refreshReportOnce(\n reportCtor: ReportCtor,\n options: ReportRefreshOptions,\n): Promise<ReportRefreshResult> {\n if (!options.db) {\n throw new Error('refreshReport requires a database handle');\n }\n\n const definition = await buildReportDefinition(reportCtor);\n const reportTable = getReportTableName(reportCtor);\n const requestedMode = options.mode ?? definition.refresh?.mode ?? 'rebuild';\n const exists = await tableExists(options.db, reportTable);\n if (!exists) {\n throw new Error(\n `Report table '${reportTable}' does not exist for ${reportCtor.name}. ` +\n `Run smrt db:migrate before refreshing reports.`,\n );\n }\n\n const requiredTables = requiredReportRuntimeTables(requestedMode, options);\n if (requiredTables.length > 0) {\n await assertReportTablesReady(options.db, requiredTables);\n }\n\n const adapterType = adapterTypeFromDb(options.db, options.adapterType);\n const scope = await resolveScope(definition, options);\n const lock =\n options.lock === false\n ? null\n : await acquireLock(\n options.db,\n definition,\n scope,\n options.lockTtlMs ?? 15 * 60 * 1000,\n );\n\n if (options.lock !== false && !lock) {\n const runId = await startRun(\n options.db,\n definition,\n scope,\n options,\n requestedMode,\n );\n await completeRun(options.db, runId, 'skipped');\n return {\n rowCount: 0,\n refreshedAt: new Date(),\n mode: requestedMode,\n tenantId: scope.tenantId,\n runId,\n changedGroupCount: 0,\n skipped: true,\n };\n }\n\n let runId: string | undefined;\n try {\n runId = await startRun(\n options.db,\n definition,\n scope,\n options,\n requestedMode,\n );\n const work =\n requestedMode === 'incremental'\n ? await incrementalReport(\n options.db,\n definition,\n reportTable,\n scope,\n adapterType,\n runId,\n options.changedRows,\n )\n : await rebuildReport(\n options.db,\n definition,\n reportTable,\n scope,\n adapterType,\n runId,\n );\n\n await completeRun(options.db, runId, 'success', {\n rowCount: work.rowCount,\n changedGroupCount: work.changedGroupCount,\n watermarkAfter: work.watermarkAfter,\n });\n\n return {\n rowCount: work.rowCount,\n refreshedAt: new Date(),\n mode: work.mode,\n tenantId: scope.tenantId,\n runId,\n changedGroupCount: work.changedGroupCount,\n };\n } catch (error) {\n await completeRun(options.db, runId, 'failed', { error });\n throw error;\n } finally {\n await lock?.release();\n }\n}\n\nexport async function refreshReport(\n reportCtor: ReportCtor,\n options: ReportRefreshOptions = {},\n): Promise<ReportRefreshResult> {\n if (options.tenantIds && options.tenantIds.length > 0) {\n if (!options.db) {\n throw new Error('refreshReport requires a database handle');\n }\n const results: ReportRefreshResult[] = [];\n for (const tenantId of options.tenantIds) {\n const result = await withTenant({ tenantId }, () =>\n refreshReportOnce(reportCtor, {\n ...options,\n tenantId,\n tenantIds: undefined,\n }),\n );\n results.push(result);\n }\n return {\n rowCount: results.reduce((sum, result) => sum + result.rowCount, 0),\n refreshedAt: new Date(),\n mode: results[0]?.mode ?? options.mode ?? 'rebuild',\n tenantId: null,\n changedGroupCount: results.reduce(\n (sum, result) => sum + (result.changedGroupCount ?? 0),\n 0,\n ),\n tenantResults: results,\n };\n }\n\n return refreshReportOnce(reportCtor, options);\n}\n\nexport function reportRowIdentity(\n row: Record<string, any>,\n definition: ReportDefinition,\n): string {\n const groupingColumns = getReportGroupingColumns(definition);\n return stableReportId(\n groupingColumns.map((column) => row[toSnakeCase(column)]),\n );\n}\n\nexport function reportSourceClassName(source: ReportSource): string {\n return sourceClassName(source);\n}\n"],"names":["runId"],"mappings":"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"}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { DatabaseInterface } from '@happyvertical/sql';
|
|
2
|
+
import { EventEmitter } from 'node:events';
|
|
3
|
+
import { SmrtJob } from '@happyvertical/smrt-jobs';
|
|
4
|
+
import { SmrtObject } from '@happyvertical/smrt-core';
|
|
5
|
+
import { SqlAdapterType } from '@happyvertical/sql';
|
|
6
|
+
|
|
7
|
+
export declare function enqueueReportRefresh(options: EnqueueReportRefreshOptions): Promise<SmrtJob>;
|
|
8
|
+
|
|
9
|
+
export declare interface EnqueueReportRefreshOptions extends ReportRefreshJobArgs {
|
|
10
|
+
report?: ReportCtor;
|
|
11
|
+
reportClass: string;
|
|
12
|
+
db: DatabaseInterface;
|
|
13
|
+
queue?: string;
|
|
14
|
+
priority?: number;
|
|
15
|
+
timeout?: number;
|
|
16
|
+
maxAttempts?: number;
|
|
17
|
+
tenantJobCap?: number;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export declare function ensureReportRefreshSchedules(options: EnsureReportSchedulesOptions): Promise<void>;
|
|
21
|
+
|
|
22
|
+
export declare interface EnsureReportSchedulesOptions {
|
|
23
|
+
db: DatabaseInterface;
|
|
24
|
+
reports: ReportCtor[];
|
|
25
|
+
tenantIds?: string[];
|
|
26
|
+
queue?: string;
|
|
27
|
+
priority?: number;
|
|
28
|
+
timeout?: number;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export declare function registerReportRefreshInterceptor(options: ReportRefreshInterceptorOptions): () => boolean;
|
|
32
|
+
|
|
33
|
+
declare type ReportCtor = new (...args: any[]) => SmrtObject;
|
|
34
|
+
|
|
35
|
+
export declare interface ReportRefreshInterceptorOptions {
|
|
36
|
+
db: DatabaseInterface;
|
|
37
|
+
reports: ReportCtor[];
|
|
38
|
+
enqueue?: boolean;
|
|
39
|
+
queue?: string;
|
|
40
|
+
priority?: number;
|
|
41
|
+
timeout?: number;
|
|
42
|
+
tenantJobCap?: number;
|
|
43
|
+
name?: string;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
export declare interface ReportRefreshJobArgs {
|
|
47
|
+
reportClass?: string;
|
|
48
|
+
mode?: ReportRefreshMode;
|
|
49
|
+
trigger?: ReportRefreshTrigger;
|
|
50
|
+
tenantId?: string | null;
|
|
51
|
+
tenantIds?: string[];
|
|
52
|
+
scheduleId?: string;
|
|
53
|
+
adapterType?: SqlAdapterType;
|
|
54
|
+
changedRows?: Record<string, unknown>[];
|
|
55
|
+
_scheduleId?: string;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
declare type ReportRefreshMode = 'rebuild' | 'incremental';
|
|
59
|
+
|
|
60
|
+
declare type ReportRefreshTrigger = 'manual' | 'schedule' | 'change' | 'ttl' | 'job';
|
|
61
|
+
|
|
62
|
+
export declare interface ReportScheduleInfo {
|
|
63
|
+
id: string;
|
|
64
|
+
reportClass: string;
|
|
65
|
+
tenantId: string | null;
|
|
66
|
+
cron: string;
|
|
67
|
+
mode: ReportRefreshMode;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export declare class ReportScheduleRunner extends EventEmitter {
|
|
71
|
+
readonly id: string;
|
|
72
|
+
private readonly config;
|
|
73
|
+
private db;
|
|
74
|
+
private running;
|
|
75
|
+
private pollTimer;
|
|
76
|
+
constructor(config?: ReportScheduleRunnerConfig);
|
|
77
|
+
initialize(db: DatabaseInterface): Promise<void>;
|
|
78
|
+
start(): Promise<void>;
|
|
79
|
+
stop(): Promise<void>;
|
|
80
|
+
isRunning(): boolean;
|
|
81
|
+
handleJobCompletion(scheduleId: string, success: boolean, errorMessage?: string): Promise<void>;
|
|
82
|
+
private startPolling;
|
|
83
|
+
poll(): Promise<void>;
|
|
84
|
+
private triggerSchedule;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export declare interface ReportScheduleRunnerConfig {
|
|
88
|
+
id?: string;
|
|
89
|
+
pollInterval?: number;
|
|
90
|
+
batchSize?: number;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
export declare interface ReportScheduleRunnerEvents {
|
|
94
|
+
'schedule:triggered': (schedule: ReportScheduleInfo) => void;
|
|
95
|
+
'schedule:error': (schedule: ReportScheduleInfo, error: Error) => void;
|
|
96
|
+
'schedule:completed': (scheduleId: string) => void;
|
|
97
|
+
'schedule:failed': (scheduleId: string, error: string) => void;
|
|
98
|
+
'runner:started': () => void;
|
|
99
|
+
'runner:stopped': () => void;
|
|
100
|
+
'runner:error': (error: Error) => void;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
export declare class SmrtReportRefreshTask extends SmrtObject {
|
|
104
|
+
tenantId: string | null;
|
|
105
|
+
reportClass: string;
|
|
106
|
+
mode: ReportRefreshMode;
|
|
107
|
+
trigger: ReportRefreshTrigger;
|
|
108
|
+
args: ReportRefreshJobArgs;
|
|
109
|
+
run(args?: ReportRefreshJobArgs): Promise<unknown>;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
export { }
|
|
@@ -0,0 +1,449 @@
|
|
|
1
|
+
import { createHash } from "node:crypto";
|
|
2
|
+
import { EventEmitter } from "node:events";
|
|
3
|
+
import { field, smrt, SmrtObject, ObjectRegistry, GlobalInterceptors } from "@happyvertical/smrt-core";
|
|
4
|
+
import { backgroundEligible, getNextCronDate, SmrtJobCollection, validateCronExpression } from "@happyvertical/smrt-jobs";
|
|
5
|
+
import { tenantId, TenantScoped, getTenantId } from "@happyvertical/smrt-tenancy";
|
|
6
|
+
import { buildReportDefinition } from "./compiler.js";
|
|
7
|
+
import { refreshReport } from "./refresh.js";
|
|
8
|
+
import { assertReportTablesReady, REPORT_SCHEDULES_TABLE, REPORT_SCHEDULER_TABLES, scopeKeyForTenant } from "./state.js";
|
|
9
|
+
var __defProp = Object.defineProperty;
|
|
10
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
11
|
+
var __decorateClass = (decorators, target, key, kind) => {
|
|
12
|
+
var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
|
|
13
|
+
for (var i = decorators.length - 1, decorator; i >= 0; i--)
|
|
14
|
+
if (decorator = decorators[i])
|
|
15
|
+
result = (kind ? decorator(target, key, result) : decorator(result)) || result;
|
|
16
|
+
if (kind && result) __defProp(target, key, result);
|
|
17
|
+
return result;
|
|
18
|
+
};
|
|
19
|
+
const INTERNAL_SURFACE = {
|
|
20
|
+
api: false,
|
|
21
|
+
cli: {
|
|
22
|
+
include: ["list", "get"],
|
|
23
|
+
skipApiCheck: true,
|
|
24
|
+
http: false
|
|
25
|
+
},
|
|
26
|
+
mcp: false
|
|
27
|
+
};
|
|
28
|
+
function stableUuid(values) {
|
|
29
|
+
const hash = createHash("sha256").update(JSON.stringify(values)).digest("hex");
|
|
30
|
+
const variant = (Number.parseInt(hash[16], 16) & 3 | 8).toString(16);
|
|
31
|
+
return [
|
|
32
|
+
hash.slice(0, 8),
|
|
33
|
+
hash.slice(8, 12),
|
|
34
|
+
`4${hash.slice(13, 16)}`,
|
|
35
|
+
`${variant}${hash.slice(17, 20)}`,
|
|
36
|
+
hash.slice(20, 32)
|
|
37
|
+
].join("-");
|
|
38
|
+
}
|
|
39
|
+
function canonicalClassName(reportCtor) {
|
|
40
|
+
const registered = ObjectRegistry.getClassByConstructor(reportCtor) ?? ObjectRegistry.getClass(reportCtor.name);
|
|
41
|
+
return registered?.qualifiedName ?? registered?.name ?? reportCtor.name;
|
|
42
|
+
}
|
|
43
|
+
function resolveReportClass(name) {
|
|
44
|
+
const registered = ObjectRegistry.getClassByQualifiedName(name) ?? ObjectRegistry.getClass(name);
|
|
45
|
+
if (!registered) {
|
|
46
|
+
throw new Error(`Unknown report class: ${name}`);
|
|
47
|
+
}
|
|
48
|
+
return registered.constructor;
|
|
49
|
+
}
|
|
50
|
+
function reportSourceName(source) {
|
|
51
|
+
if (typeof source === "string") return source;
|
|
52
|
+
return source.name;
|
|
53
|
+
}
|
|
54
|
+
function sourceMatches(definition, instance, context) {
|
|
55
|
+
const configured = definition.refresh?.onChange;
|
|
56
|
+
if (!configured || configured.length === 0) return false;
|
|
57
|
+
const eventNames = /* @__PURE__ */ new Set([
|
|
58
|
+
context.className,
|
|
59
|
+
instance.constructor.name
|
|
60
|
+
]);
|
|
61
|
+
const registered = ObjectRegistry.getClassByConstructor(
|
|
62
|
+
instance.constructor
|
|
63
|
+
);
|
|
64
|
+
if (registered?.qualifiedName) eventNames.add(registered.qualifiedName);
|
|
65
|
+
if (registered?.name) eventNames.add(registered.name);
|
|
66
|
+
for (const source of configured) {
|
|
67
|
+
const name = reportSourceName(source);
|
|
68
|
+
const registeredSource = ObjectRegistry.getClassByQualifiedName(name) ?? ObjectRegistry.getClass(name);
|
|
69
|
+
if (eventNames.has(name) || registeredSource?.name && eventNames.has(registeredSource.name) || registeredSource?.qualifiedName && eventNames.has(registeredSource.qualifiedName)) {
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
return false;
|
|
74
|
+
}
|
|
75
|
+
function tenantIdFromInstance(instance) {
|
|
76
|
+
const value = instance.tenantId;
|
|
77
|
+
return typeof value === "string" && value.length > 0 ? value : getTenantId() ?? null;
|
|
78
|
+
}
|
|
79
|
+
function changedRowSnapshot(instance) {
|
|
80
|
+
const serializable = instance.toJSON();
|
|
81
|
+
return serializable && typeof serializable === "object" ? serializable : {};
|
|
82
|
+
}
|
|
83
|
+
let SmrtReportRefreshTask = class extends SmrtObject {
|
|
84
|
+
tenantId = null;
|
|
85
|
+
reportClass = "";
|
|
86
|
+
mode = "incremental";
|
|
87
|
+
trigger = "job";
|
|
88
|
+
args = {};
|
|
89
|
+
async run(args = {}) {
|
|
90
|
+
const reportClass = args.reportClass || this.reportClass;
|
|
91
|
+
if (!reportClass) {
|
|
92
|
+
throw new Error("Report refresh job requires reportClass");
|
|
93
|
+
}
|
|
94
|
+
const reportCtor = resolveReportClass(reportClass);
|
|
95
|
+
return refreshReport(reportCtor, {
|
|
96
|
+
db: this.db,
|
|
97
|
+
mode: args.mode ?? this.mode,
|
|
98
|
+
trigger: args.trigger ?? this.trigger,
|
|
99
|
+
tenantId: args.tenantId,
|
|
100
|
+
tenantIds: args.tenantIds,
|
|
101
|
+
adapterType: args.adapterType,
|
|
102
|
+
scheduleId: args.scheduleId ?? args._scheduleId,
|
|
103
|
+
changedRows: args.changedRows
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
};
|
|
107
|
+
__decorateClass([
|
|
108
|
+
tenantId({ nullable: true })
|
|
109
|
+
], SmrtReportRefreshTask.prototype, "tenantId", 2);
|
|
110
|
+
__decorateClass([
|
|
111
|
+
field({ type: "text", required: true })
|
|
112
|
+
], SmrtReportRefreshTask.prototype, "reportClass", 2);
|
|
113
|
+
__decorateClass([
|
|
114
|
+
field({ type: "text", required: true })
|
|
115
|
+
], SmrtReportRefreshTask.prototype, "mode", 2);
|
|
116
|
+
__decorateClass([
|
|
117
|
+
field({ type: "text", required: true })
|
|
118
|
+
], SmrtReportRefreshTask.prototype, "trigger", 2);
|
|
119
|
+
__decorateClass([
|
|
120
|
+
field({ type: "json" })
|
|
121
|
+
], SmrtReportRefreshTask.prototype, "args", 2);
|
|
122
|
+
__decorateClass([
|
|
123
|
+
backgroundEligible()
|
|
124
|
+
], SmrtReportRefreshTask.prototype, "run", 1);
|
|
125
|
+
SmrtReportRefreshTask = __decorateClass([
|
|
126
|
+
TenantScoped({ mode: "optional" }),
|
|
127
|
+
smrt({
|
|
128
|
+
tableName: "_smrt_report_refresh_tasks",
|
|
129
|
+
...INTERNAL_SURFACE
|
|
130
|
+
})
|
|
131
|
+
], SmrtReportRefreshTask);
|
|
132
|
+
async function enqueueReportRefresh(options) {
|
|
133
|
+
await ObjectRegistry.ensureManifestLoaded("SmrtJob");
|
|
134
|
+
const collection = await SmrtJobCollection.create({ db: options.db });
|
|
135
|
+
const taskType = canonicalClassName(SmrtReportRefreshTask);
|
|
136
|
+
const scheduleId = options.scheduleId ?? options._scheduleId;
|
|
137
|
+
return collection.enqueueJob(
|
|
138
|
+
{
|
|
139
|
+
tenantId: options.tenantId ?? null,
|
|
140
|
+
queue: options.queue ?? "reports",
|
|
141
|
+
objectType: taskType,
|
|
142
|
+
objectId: null,
|
|
143
|
+
method: "run",
|
|
144
|
+
args: {
|
|
145
|
+
reportClass: options.reportClass,
|
|
146
|
+
mode: options.mode,
|
|
147
|
+
trigger: options.trigger ?? "job",
|
|
148
|
+
tenantId: options.tenantId,
|
|
149
|
+
tenantIds: options.tenantIds,
|
|
150
|
+
adapterType: options.adapterType,
|
|
151
|
+
changedRows: options.changedRows,
|
|
152
|
+
scheduleId,
|
|
153
|
+
_scheduleId: scheduleId
|
|
154
|
+
},
|
|
155
|
+
priority: options.priority ?? 70,
|
|
156
|
+
timeout: options.timeout ?? 36e5,
|
|
157
|
+
maxAttempts: options.maxAttempts ?? 3
|
|
158
|
+
},
|
|
159
|
+
{ tenantJobCap: options.tenantJobCap }
|
|
160
|
+
);
|
|
161
|
+
}
|
|
162
|
+
async function ensureReportRefreshSchedules(options) {
|
|
163
|
+
await assertReportTablesReady(options.db, REPORT_SCHEDULER_TABLES);
|
|
164
|
+
for (const reportCtor of options.reports) {
|
|
165
|
+
const definition = await buildReportDefinition(reportCtor);
|
|
166
|
+
const refresh = definition.refresh;
|
|
167
|
+
if (!refresh || refresh.manual) continue;
|
|
168
|
+
const reportClass = canonicalClassName(reportCtor);
|
|
169
|
+
const targetTenants = refresh.tenantFanout ? options.tenantIds : [null];
|
|
170
|
+
if (refresh.tenantFanout && (!targetTenants || targetTenants.length === 0)) {
|
|
171
|
+
throw new Error(
|
|
172
|
+
`${definition.reportClassName} refresh.tenantFanout requires tenantIds when creating schedules.`
|
|
173
|
+
);
|
|
174
|
+
}
|
|
175
|
+
const schedules = [
|
|
176
|
+
refresh.schedule ? {
|
|
177
|
+
cron: refresh.schedule,
|
|
178
|
+
mode: refresh.mode ?? "incremental",
|
|
179
|
+
trigger: "schedule"
|
|
180
|
+
} : null,
|
|
181
|
+
refresh.fullRebuildSchedule ? {
|
|
182
|
+
cron: refresh.fullRebuildSchedule,
|
|
183
|
+
mode: "rebuild",
|
|
184
|
+
trigger: "schedule"
|
|
185
|
+
} : null
|
|
186
|
+
].filter(Boolean);
|
|
187
|
+
for (const schedule of schedules) {
|
|
188
|
+
validateCronExpression(schedule.cron);
|
|
189
|
+
for (const tenantId2 of targetTenants ?? []) {
|
|
190
|
+
const scopeKey = scopeKeyForTenant(tenantId2);
|
|
191
|
+
const id = stableUuid([
|
|
192
|
+
"schedule",
|
|
193
|
+
reportClass,
|
|
194
|
+
scopeKey,
|
|
195
|
+
schedule.cron,
|
|
196
|
+
schedule.mode
|
|
197
|
+
]);
|
|
198
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
199
|
+
await options.db.upsert(
|
|
200
|
+
REPORT_SCHEDULES_TABLE,
|
|
201
|
+
["report_class", "scope_key", "cron", "mode"],
|
|
202
|
+
{
|
|
203
|
+
id,
|
|
204
|
+
slug: id,
|
|
205
|
+
context: scopeKey,
|
|
206
|
+
tenant_id: tenantId2,
|
|
207
|
+
scope_key: scopeKey,
|
|
208
|
+
report_class: reportClass,
|
|
209
|
+
cron: schedule.cron,
|
|
210
|
+
trigger: schedule.trigger,
|
|
211
|
+
mode: schedule.mode,
|
|
212
|
+
enabled: true,
|
|
213
|
+
status: "active",
|
|
214
|
+
next_run: getNextCronDate(schedule.cron).toISOString(),
|
|
215
|
+
last_run: null,
|
|
216
|
+
last_status: null,
|
|
217
|
+
last_error: null,
|
|
218
|
+
run_count: 0,
|
|
219
|
+
success_count: 0,
|
|
220
|
+
failure_count: 0,
|
|
221
|
+
running_count: 0,
|
|
222
|
+
max_concurrent: 1,
|
|
223
|
+
queue: options.queue ?? "reports",
|
|
224
|
+
priority: options.priority ?? 70,
|
|
225
|
+
timeout: options.timeout ?? 36e5,
|
|
226
|
+
created_at: now,
|
|
227
|
+
updated_at: now
|
|
228
|
+
}
|
|
229
|
+
);
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
class ReportScheduleRunner extends EventEmitter {
|
|
235
|
+
id;
|
|
236
|
+
config;
|
|
237
|
+
db = null;
|
|
238
|
+
running = false;
|
|
239
|
+
pollTimer = null;
|
|
240
|
+
constructor(config = {}) {
|
|
241
|
+
super();
|
|
242
|
+
this.config = {
|
|
243
|
+
id: config.id || `reports_${stableUuid([Date.now()]).slice(0, 8)}`,
|
|
244
|
+
pollInterval: config.pollInterval ?? 6e4,
|
|
245
|
+
batchSize: config.batchSize ?? 50
|
|
246
|
+
};
|
|
247
|
+
this.id = this.config.id;
|
|
248
|
+
}
|
|
249
|
+
async initialize(db) {
|
|
250
|
+
this.db = db;
|
|
251
|
+
await assertReportTablesReady(db, REPORT_SCHEDULER_TABLES);
|
|
252
|
+
}
|
|
253
|
+
async start() {
|
|
254
|
+
if (this.running) return;
|
|
255
|
+
if (!this.db) {
|
|
256
|
+
throw new Error(
|
|
257
|
+
"ReportScheduleRunner not initialized. Call initialize() first."
|
|
258
|
+
);
|
|
259
|
+
}
|
|
260
|
+
this.running = true;
|
|
261
|
+
this.startPolling();
|
|
262
|
+
this.emit("runner:started");
|
|
263
|
+
}
|
|
264
|
+
async stop() {
|
|
265
|
+
if (!this.running) return;
|
|
266
|
+
this.running = false;
|
|
267
|
+
if (this.pollTimer) {
|
|
268
|
+
clearTimeout(this.pollTimer);
|
|
269
|
+
this.pollTimer = null;
|
|
270
|
+
}
|
|
271
|
+
this.emit("runner:stopped");
|
|
272
|
+
}
|
|
273
|
+
isRunning() {
|
|
274
|
+
return this.running;
|
|
275
|
+
}
|
|
276
|
+
async handleJobCompletion(scheduleId, success, errorMessage) {
|
|
277
|
+
if (!this.db) return;
|
|
278
|
+
const now = (/* @__PURE__ */ new Date()).toISOString();
|
|
279
|
+
if (success) {
|
|
280
|
+
await this.db.query(
|
|
281
|
+
`UPDATE ${REPORT_SCHEDULES_TABLE}
|
|
282
|
+
SET running_count = CASE WHEN COALESCE(running_count, 0) > 0 THEN running_count - 1 ELSE 0 END,
|
|
283
|
+
last_run = ?,
|
|
284
|
+
last_status = 'success',
|
|
285
|
+
last_error = NULL,
|
|
286
|
+
run_count = COALESCE(run_count, 0) + 1,
|
|
287
|
+
success_count = COALESCE(success_count, 0) + 1,
|
|
288
|
+
updated_at = ?
|
|
289
|
+
WHERE id = ?`,
|
|
290
|
+
now,
|
|
291
|
+
now,
|
|
292
|
+
scheduleId
|
|
293
|
+
);
|
|
294
|
+
this.emit("schedule:completed", scheduleId);
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
const safeError = errorMessage ?? "Unknown error";
|
|
298
|
+
await this.db.query(
|
|
299
|
+
`UPDATE ${REPORT_SCHEDULES_TABLE}
|
|
300
|
+
SET running_count = CASE WHEN COALESCE(running_count, 0) > 0 THEN running_count - 1 ELSE 0 END,
|
|
301
|
+
last_run = ?,
|
|
302
|
+
last_status = 'failed',
|
|
303
|
+
last_error = ?,
|
|
304
|
+
run_count = COALESCE(run_count, 0) + 1,
|
|
305
|
+
failure_count = COALESCE(failure_count, 0) + 1,
|
|
306
|
+
updated_at = ?
|
|
307
|
+
WHERE id = ?`,
|
|
308
|
+
now,
|
|
309
|
+
safeError,
|
|
310
|
+
now,
|
|
311
|
+
scheduleId
|
|
312
|
+
);
|
|
313
|
+
this.emit("schedule:failed", scheduleId, safeError);
|
|
314
|
+
}
|
|
315
|
+
startPolling() {
|
|
316
|
+
const poll = async () => {
|
|
317
|
+
if (!this.running) return;
|
|
318
|
+
try {
|
|
319
|
+
await this.poll();
|
|
320
|
+
} catch (error) {
|
|
321
|
+
this.emit("runner:error", error);
|
|
322
|
+
}
|
|
323
|
+
if (this.running) {
|
|
324
|
+
this.pollTimer = setTimeout(poll, this.config.pollInterval);
|
|
325
|
+
if (typeof this.pollTimer.unref === "function") {
|
|
326
|
+
this.pollTimer.unref();
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
poll();
|
|
331
|
+
}
|
|
332
|
+
async poll() {
|
|
333
|
+
if (!this.db) return;
|
|
334
|
+
const result = await this.db.query(
|
|
335
|
+
`SELECT * FROM ${REPORT_SCHEDULES_TABLE}
|
|
336
|
+
WHERE enabled = true
|
|
337
|
+
AND status = 'active'
|
|
338
|
+
AND next_run <= ?
|
|
339
|
+
AND COALESCE(running_count, 0) < COALESCE(max_concurrent, 1)
|
|
340
|
+
ORDER BY next_run ASC
|
|
341
|
+
LIMIT ?`,
|
|
342
|
+
(/* @__PURE__ */ new Date()).toISOString(),
|
|
343
|
+
this.config.batchSize
|
|
344
|
+
);
|
|
345
|
+
for (const row of result.rows) {
|
|
346
|
+
await this.triggerSchedule(row);
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
async triggerSchedule(row) {
|
|
350
|
+
if (!this.db) return;
|
|
351
|
+
const schedule = {
|
|
352
|
+
id: String(row.id),
|
|
353
|
+
reportClass: String(row.report_class),
|
|
354
|
+
tenantId: typeof row.tenant_id === "string" && row.tenant_id.length > 0 ? row.tenant_id : null,
|
|
355
|
+
cron: String(row.cron),
|
|
356
|
+
mode: row.mode || "incremental"
|
|
357
|
+
};
|
|
358
|
+
try {
|
|
359
|
+
const nextRun = getNextCronDate(schedule.cron);
|
|
360
|
+
await enqueueReportRefresh({
|
|
361
|
+
db: this.db,
|
|
362
|
+
reportClass: schedule.reportClass,
|
|
363
|
+
mode: schedule.mode,
|
|
364
|
+
trigger: row.trigger || "schedule",
|
|
365
|
+
tenantId: schedule.tenantId,
|
|
366
|
+
scheduleId: schedule.id,
|
|
367
|
+
queue: String(row.queue || "reports"),
|
|
368
|
+
priority: Number(row.priority ?? 70),
|
|
369
|
+
timeout: Number(row.timeout ?? 36e5)
|
|
370
|
+
});
|
|
371
|
+
await this.db.query(
|
|
372
|
+
`UPDATE ${REPORT_SCHEDULES_TABLE}
|
|
373
|
+
SET running_count = COALESCE(running_count, 0) + 1,
|
|
374
|
+
next_run = ?,
|
|
375
|
+
updated_at = ?
|
|
376
|
+
WHERE id = ?`,
|
|
377
|
+
nextRun.toISOString(),
|
|
378
|
+
(/* @__PURE__ */ new Date()).toISOString(),
|
|
379
|
+
schedule.id
|
|
380
|
+
);
|
|
381
|
+
this.emit("schedule:triggered", schedule);
|
|
382
|
+
} catch (error) {
|
|
383
|
+
await this.db.query(
|
|
384
|
+
`UPDATE ${REPORT_SCHEDULES_TABLE}
|
|
385
|
+
SET last_error = ?,
|
|
386
|
+
updated_at = ?
|
|
387
|
+
WHERE id = ?`,
|
|
388
|
+
error instanceof Error ? error.message : String(error),
|
|
389
|
+
(/* @__PURE__ */ new Date()).toISOString(),
|
|
390
|
+
schedule.id
|
|
391
|
+
);
|
|
392
|
+
this.emit("schedule:error", schedule, error);
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
}
|
|
396
|
+
function registerReportRefreshInterceptor(options) {
|
|
397
|
+
const name = options.name ?? "smrt-reports-refresh";
|
|
398
|
+
GlobalInterceptors.register({
|
|
399
|
+
name,
|
|
400
|
+
priority: -10,
|
|
401
|
+
async afterSave(instance, context) {
|
|
402
|
+
await triggerReportsForInstance(options, instance, context);
|
|
403
|
+
},
|
|
404
|
+
async afterDelete(instance, context) {
|
|
405
|
+
await triggerReportsForInstance(options, instance, context);
|
|
406
|
+
}
|
|
407
|
+
});
|
|
408
|
+
return () => GlobalInterceptors.unregister(name);
|
|
409
|
+
}
|
|
410
|
+
async function triggerReportsForInstance(options, instance, context) {
|
|
411
|
+
for (const reportCtor of options.reports) {
|
|
412
|
+
const definition = await buildReportDefinition(reportCtor);
|
|
413
|
+
if (definition.refresh?.manual) continue;
|
|
414
|
+
if (!sourceMatches(definition, instance, context)) continue;
|
|
415
|
+
const mode = definition.refresh?.mode ?? "incremental";
|
|
416
|
+
const tenantId2 = tenantIdFromInstance(instance);
|
|
417
|
+
const changedRows = [changedRowSnapshot(instance)];
|
|
418
|
+
if (options.enqueue === false) {
|
|
419
|
+
await refreshReport(reportCtor, {
|
|
420
|
+
db: options.db,
|
|
421
|
+
mode,
|
|
422
|
+
trigger: "change",
|
|
423
|
+
tenantId: tenantId2,
|
|
424
|
+
changedRows
|
|
425
|
+
});
|
|
426
|
+
continue;
|
|
427
|
+
}
|
|
428
|
+
await enqueueReportRefresh({
|
|
429
|
+
db: options.db,
|
|
430
|
+
reportClass: canonicalClassName(reportCtor),
|
|
431
|
+
mode,
|
|
432
|
+
trigger: "change",
|
|
433
|
+
tenantId: tenantId2,
|
|
434
|
+
queue: options.queue,
|
|
435
|
+
priority: options.priority,
|
|
436
|
+
timeout: options.timeout,
|
|
437
|
+
tenantJobCap: options.tenantJobCap,
|
|
438
|
+
changedRows
|
|
439
|
+
});
|
|
440
|
+
}
|
|
441
|
+
}
|
|
442
|
+
export {
|
|
443
|
+
ReportScheduleRunner,
|
|
444
|
+
SmrtReportRefreshTask,
|
|
445
|
+
enqueueReportRefresh,
|
|
446
|
+
ensureReportRefreshSchedules,
|
|
447
|
+
registerReportRefreshInterceptor
|
|
448
|
+
};
|
|
449
|
+
//# sourceMappingURL=scheduler.js.map
|