@stackmemoryai/stackmemory 0.3.5 → 0.3.7

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.
Files changed (220) hide show
  1. package/dist/agents/verifiers/base-verifier.js.map +2 -2
  2. package/dist/agents/verifiers/formatter-verifier.js.map +2 -2
  3. package/dist/agents/verifiers/llm-judge.js.map +2 -2
  4. package/dist/cli/claude-sm.js +24 -13
  5. package/dist/cli/claude-sm.js.map +2 -2
  6. package/dist/cli/codex-sm.js +24 -13
  7. package/dist/cli/codex-sm.js.map +2 -2
  8. package/dist/cli/commands/agent.js.map +2 -2
  9. package/dist/cli/commands/chromadb.js +217 -32
  10. package/dist/cli/commands/chromadb.js.map +2 -2
  11. package/dist/cli/commands/clear.js +12 -1
  12. package/dist/cli/commands/clear.js.map +2 -2
  13. package/dist/cli/commands/context.js +13 -2
  14. package/dist/cli/commands/context.js.map +2 -2
  15. package/dist/cli/commands/dashboard.js.map +2 -2
  16. package/dist/cli/commands/gc.js +202 -0
  17. package/dist/cli/commands/gc.js.map +7 -0
  18. package/dist/cli/commands/handoff.js +12 -1
  19. package/dist/cli/commands/handoff.js.map +2 -2
  20. package/dist/cli/commands/infinite-storage.js +251 -0
  21. package/dist/cli/commands/infinite-storage.js.map +7 -0
  22. package/dist/cli/commands/linear-create.js +13 -2
  23. package/dist/cli/commands/linear-create.js.map +2 -2
  24. package/dist/cli/commands/linear-list.js +12 -1
  25. package/dist/cli/commands/linear-list.js.map +2 -2
  26. package/dist/cli/commands/linear-migrate.js +12 -1
  27. package/dist/cli/commands/linear-migrate.js.map +2 -2
  28. package/dist/cli/commands/linear-test.js +12 -1
  29. package/dist/cli/commands/linear-test.js.map +2 -2
  30. package/dist/cli/commands/linear-unified.js +262 -0
  31. package/dist/cli/commands/linear-unified.js.map +7 -0
  32. package/dist/cli/commands/linear.js +17 -6
  33. package/dist/cli/commands/linear.js.map +2 -2
  34. package/dist/cli/commands/monitor.js.map +2 -2
  35. package/dist/cli/commands/onboard.js.map +2 -2
  36. package/dist/cli/commands/quality.js.map +2 -2
  37. package/dist/cli/commands/search.js.map +2 -2
  38. package/dist/cli/commands/session.js.map +2 -2
  39. package/dist/cli/commands/skills.js +12 -1
  40. package/dist/cli/commands/skills.js.map +2 -2
  41. package/dist/cli/commands/storage.js +18 -7
  42. package/dist/cli/commands/storage.js.map +2 -2
  43. package/dist/cli/commands/tasks.js.map +2 -2
  44. package/dist/cli/commands/tui.js +13 -2
  45. package/dist/cli/commands/tui.js.map +2 -2
  46. package/dist/cli/commands/webhook.js +14 -3
  47. package/dist/cli/commands/webhook.js.map +2 -2
  48. package/dist/cli/commands/workflow.js +14 -3
  49. package/dist/cli/commands/workflow.js.map +2 -2
  50. package/dist/cli/commands/worktree.js.map +2 -2
  51. package/dist/cli/index.js +20 -5
  52. package/dist/cli/index.js.map +2 -2
  53. package/dist/core/config/config-manager.js.map +2 -2
  54. package/dist/core/context/auto-context.js.map +2 -2
  55. package/dist/core/context/compaction-handler.js.map +2 -2
  56. package/dist/core/context/context-bridge.js.map +2 -2
  57. package/dist/core/context/dual-stack-manager.js.map +2 -2
  58. package/dist/core/context/frame-database.js.map +2 -2
  59. package/dist/core/context/frame-digest.js.map +2 -2
  60. package/dist/core/context/frame-handoff-manager.js.map +2 -2
  61. package/dist/core/context/frame-manager.js +12 -1
  62. package/dist/core/context/frame-manager.js.map +2 -2
  63. package/dist/core/context/frame-stack.js.map +2 -2
  64. package/dist/core/context/incremental-gc.js +279 -0
  65. package/dist/core/context/incremental-gc.js.map +7 -0
  66. package/dist/core/context/permission-manager.js +12 -1
  67. package/dist/core/context/permission-manager.js.map +2 -2
  68. package/dist/core/context/refactored-frame-manager.js.map +2 -2
  69. package/dist/core/context/shared-context-layer.js +12 -1
  70. package/dist/core/context/shared-context-layer.js.map +2 -2
  71. package/dist/core/context/stack-merge-resolver.js.map +2 -2
  72. package/dist/core/context/validation.js.map +2 -2
  73. package/dist/core/database/batch-operations.js.map +2 -2
  74. package/dist/core/database/connection-pool.js.map +2 -2
  75. package/dist/core/database/migration-manager.js.map +2 -2
  76. package/dist/core/database/paradedb-adapter.js.map +2 -2
  77. package/dist/core/database/query-cache.js.map +2 -2
  78. package/dist/core/database/query-router.js.map +2 -2
  79. package/dist/core/database/sqlite-adapter.js.map +2 -2
  80. package/dist/core/digest/enhanced-hybrid-digest.js.map +2 -2
  81. package/dist/core/errors/recovery.js.map +2 -2
  82. package/dist/core/merge/resolution-engine.js.map +2 -2
  83. package/dist/core/monitoring/error-handler.js.map +2 -2
  84. package/dist/core/monitoring/logger.js +14 -3
  85. package/dist/core/monitoring/logger.js.map +2 -2
  86. package/dist/core/monitoring/metrics.js +13 -2
  87. package/dist/core/monitoring/metrics.js.map +2 -2
  88. package/dist/core/monitoring/progress-tracker.js +12 -1
  89. package/dist/core/monitoring/progress-tracker.js.map +2 -2
  90. package/dist/core/monitoring/session-monitor.js.map +2 -2
  91. package/dist/core/performance/context-cache.js.map +2 -2
  92. package/dist/core/performance/lazy-context-loader.js.map +2 -2
  93. package/dist/core/performance/monitor.js.map +2 -2
  94. package/dist/core/performance/optimized-frame-context.js.map +2 -2
  95. package/dist/core/performance/performance-benchmark.js.map +2 -2
  96. package/dist/core/performance/performance-profiler.js +12 -1
  97. package/dist/core/performance/performance-profiler.js.map +2 -2
  98. package/dist/core/performance/streaming-jsonl-parser.js.map +2 -2
  99. package/dist/core/persistence/postgres-adapter.js.map +2 -2
  100. package/dist/core/projects/project-manager.js.map +2 -2
  101. package/dist/core/retrieval/context-retriever.js.map +2 -2
  102. package/dist/core/retrieval/graph-retrieval.js.map +2 -2
  103. package/dist/core/retrieval/llm-context-retrieval.js.map +2 -2
  104. package/dist/core/retrieval/retrieval-benchmarks.js.map +2 -2
  105. package/dist/core/retrieval/summary-generator.js.map +2 -2
  106. package/dist/core/session/clear-survival.js.map +2 -2
  107. package/dist/core/session/handoff-generator.js.map +2 -2
  108. package/dist/core/session/session-manager.js +16 -5
  109. package/dist/core/session/session-manager.js.map +2 -2
  110. package/dist/core/skills/skill-storage.js +13 -2
  111. package/dist/core/skills/skill-storage.js.map +2 -2
  112. package/dist/core/storage/chromadb-adapter.js.map +2 -2
  113. package/dist/core/storage/chromadb-simple.js +160 -0
  114. package/dist/core/storage/chromadb-simple.js.map +7 -0
  115. package/dist/core/storage/infinite-storage.js +443 -0
  116. package/dist/core/storage/infinite-storage.js.map +7 -0
  117. package/dist/core/storage/railway-optimized-storage.js +19 -8
  118. package/dist/core/storage/railway-optimized-storage.js.map +2 -2
  119. package/dist/core/storage/remote-storage.js +12 -1
  120. package/dist/core/storage/remote-storage.js.map +2 -2
  121. package/dist/core/trace/cli-trace-wrapper.js +16 -5
  122. package/dist/core/trace/cli-trace-wrapper.js.map +2 -2
  123. package/dist/core/trace/db-trace-wrapper.js.map +2 -2
  124. package/dist/core/trace/debug-trace.js +21 -10
  125. package/dist/core/trace/debug-trace.js.map +2 -2
  126. package/dist/core/trace/index.js +46 -35
  127. package/dist/core/trace/index.js.map +2 -2
  128. package/dist/core/trace/trace-demo.js +12 -1
  129. package/dist/core/trace/trace-demo.js.map +2 -2
  130. package/dist/core/trace/trace-detector.js.map +2 -2
  131. package/dist/core/trace/trace-store.js.map +2 -2
  132. package/dist/core/utils/compression.js +79 -0
  133. package/dist/core/utils/compression.js.map +7 -0
  134. package/dist/core/utils/update-checker.js.map +2 -2
  135. package/dist/core/worktree/worktree-manager.js.map +2 -2
  136. package/dist/features/analytics/api/analytics-api.js.map +2 -2
  137. package/dist/features/analytics/core/analytics-service.js +12 -1
  138. package/dist/features/analytics/core/analytics-service.js.map +2 -2
  139. package/dist/features/analytics/queries/metrics-queries.js.map +2 -2
  140. package/dist/features/tasks/pebbles-task-store.js.map +2 -2
  141. package/dist/features/tui/components/analytics-panel.js.map +2 -2
  142. package/dist/features/tui/components/pr-tracker.js.map +2 -2
  143. package/dist/features/tui/components/session-monitor.js.map +2 -2
  144. package/dist/features/tui/components/subagent-fleet.js.map +2 -2
  145. package/dist/features/tui/components/task-board.js +650 -2
  146. package/dist/features/tui/components/task-board.js.map +2 -2
  147. package/dist/features/tui/index.js +16 -5
  148. package/dist/features/tui/index.js.map +2 -2
  149. package/dist/features/tui/services/data-service.js +35 -52
  150. package/dist/features/tui/services/data-service.js.map +2 -2
  151. package/dist/features/tui/services/linear-task-reader.js +100 -0
  152. package/dist/features/tui/services/linear-task-reader.js.map +7 -0
  153. package/dist/features/tui/services/websocket-client.js +13 -2
  154. package/dist/features/tui/services/websocket-client.js.map +2 -2
  155. package/dist/features/tui/terminal-compat.js +27 -16
  156. package/dist/features/tui/terminal-compat.js.map +2 -2
  157. package/dist/features/web/client/stores/task-store.js +22 -0
  158. package/dist/features/web/client/stores/task-store.js.map +7 -0
  159. package/dist/features/web/server/index.js +182 -0
  160. package/dist/features/web/server/index.js.map +7 -0
  161. package/dist/integrations/claude-code/enhanced-pre-clear-hooks.js.map +2 -2
  162. package/dist/integrations/claude-code/lifecycle-hooks.js.map +2 -2
  163. package/dist/integrations/claude-code/post-task-hooks.js.map +2 -2
  164. package/dist/integrations/linear/auth.js +17 -6
  165. package/dist/integrations/linear/auth.js.map +2 -2
  166. package/dist/integrations/linear/auto-sync.js.map +2 -2
  167. package/dist/integrations/linear/client.js.map +2 -2
  168. package/dist/integrations/linear/config.js.map +2 -2
  169. package/dist/integrations/linear/migration.js.map +2 -2
  170. package/dist/integrations/linear/oauth-server.js +13 -2
  171. package/dist/integrations/linear/oauth-server.js.map +2 -2
  172. package/dist/integrations/linear/rest-client.js.map +2 -2
  173. package/dist/integrations/linear/sync-enhanced.js +202 -0
  174. package/dist/integrations/linear/sync-enhanced.js.map +7 -0
  175. package/dist/integrations/linear/sync-manager.js.map +2 -2
  176. package/dist/integrations/linear/sync-service.js +12 -1
  177. package/dist/integrations/linear/sync-service.js.map +2 -2
  178. package/dist/integrations/linear/sync.js +34 -3
  179. package/dist/integrations/linear/sync.js.map +2 -2
  180. package/dist/integrations/linear/unified-sync.js +560 -0
  181. package/dist/integrations/linear/unified-sync.js.map +7 -0
  182. package/dist/integrations/linear/webhook-handler.js +12 -1
  183. package/dist/integrations/linear/webhook-handler.js.map +2 -2
  184. package/dist/integrations/linear/webhook-server.js +14 -3
  185. package/dist/integrations/linear/webhook-server.js.map +2 -2
  186. package/dist/integrations/linear/webhook.js +12 -1
  187. package/dist/integrations/linear/webhook.js.map +2 -2
  188. package/dist/integrations/mcp/handlers/context-handlers.js.map +2 -2
  189. package/dist/integrations/mcp/handlers/linear-handlers.js.map +2 -2
  190. package/dist/integrations/mcp/handlers/skill-handlers.js +13 -2
  191. package/dist/integrations/mcp/handlers/skill-handlers.js.map +2 -2
  192. package/dist/integrations/mcp/handlers/task-handlers.js.map +2 -2
  193. package/dist/integrations/mcp/handlers/trace-handlers.js.map +2 -2
  194. package/dist/integrations/mcp/middleware/tool-scoring.js.map +2 -2
  195. package/dist/integrations/mcp/refactored-server.js +15 -4
  196. package/dist/integrations/mcp/refactored-server.js.map +2 -2
  197. package/dist/integrations/mcp/server.js +12 -1
  198. package/dist/integrations/mcp/server.js.map +2 -2
  199. package/dist/integrations/mcp/tool-definitions.js.map +2 -2
  200. package/dist/integrations/pg-aiguide/embedding-provider.js +13 -2
  201. package/dist/integrations/pg-aiguide/embedding-provider.js.map +2 -2
  202. package/dist/integrations/pg-aiguide/semantic-search.js.map +2 -2
  203. package/dist/mcp/stackmemory-mcp-server.js +12 -1
  204. package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
  205. package/dist/middleware/exponential-rate-limiter.js.map +2 -2
  206. package/dist/servers/production/auth-middleware.js +13 -2
  207. package/dist/servers/production/auth-middleware.js.map +2 -2
  208. package/dist/servers/railway/index.js +22 -11
  209. package/dist/servers/railway/index.js.map +2 -2
  210. package/dist/services/config-service.js.map +2 -2
  211. package/dist/services/context-service.js.map +2 -2
  212. package/dist/skills/claude-skills.js +150 -1
  213. package/dist/skills/claude-skills.js.map +2 -2
  214. package/dist/skills/dashboard-launcher.js +212 -0
  215. package/dist/skills/dashboard-launcher.js.map +7 -0
  216. package/dist/skills/repo-ingestion-skill.js +561 -0
  217. package/dist/skills/repo-ingestion-skill.js.map +7 -0
  218. package/dist/utils/logger.js +12 -1
  219. package/dist/utils/logger.js.map +2 -2
  220. package/package.json +7 -1
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/database/migration-manager.ts"],
4
- "sourcesContent": ["/**\n * Migration Manager for Dual-Write Strategy\n * Enables seamless migration between SQLite and ParadeDB with zero downtime\n */\n\nimport { EventEmitter } from 'events';\nimport { DatabaseAdapter } from './database-adapter.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport interface MigrationConfig {\n sourceAdapter: DatabaseAdapter;\n targetAdapter: DatabaseAdapter;\n batchSize?: number;\n retryAttempts?: number;\n retryDelayMs?: number;\n verifyData?: boolean;\n enableDualWrite?: boolean;\n progressCallback?: (progress: MigrationProgress) => void;\n}\n\nexport interface MigrationProgress {\n phase:\n | 'initializing'\n | 'migrating'\n | 'verifying'\n | 'completing'\n | 'completed'\n | 'failed';\n totalRecords: number;\n processedRecords: number;\n percentage: number;\n startTime: Date;\n estimatedEndTime?: Date;\n currentTable?: string;\n errors: Array<{ table: string; error: string; timestamp: Date }>;\n warnings: Array<{ table: string; warning: string; timestamp: Date }>;\n}\n\nexport interface MigrationStrategy {\n type: 'online' | 'offline' | 'dual-write';\n allowWrites: boolean;\n verifyIntegrity: boolean;\n fallbackOnError: boolean;\n}\n\nexport interface TableMigrationPlan {\n table: string;\n priority: number;\n estimatedRows: number;\n dependencies: string[];\n strategy: 'full' | 'incremental' | 'skip';\n}\n\nexport class MigrationManager extends EventEmitter {\n private config: Required<MigrationConfig>;\n private progress: MigrationProgress;\n private isRunning = false;\n private isPaused = false;\n private abortController?: AbortController;\n\n constructor(config: MigrationConfig) {\n super();\n\n this.validateConfig(config);\n this.config = this.normalizeConfig(config);\n this.progress = this.initializeProgress();\n }\n\n private validateConfig(config: MigrationConfig): void {\n if (!config.sourceAdapter || !config.targetAdapter) {\n throw new Error('Source and target adapters are required');\n }\n\n if (\n config.batchSize &&\n (config.batchSize < 1 || config.batchSize > 10000)\n ) {\n throw new Error('Batch size must be between 1 and 10000');\n }\n\n if (\n config.retryAttempts &&\n (config.retryAttempts < 0 || config.retryAttempts > 10)\n ) {\n throw new Error('Retry attempts must be between 0 and 10');\n }\n\n if (\n config.retryDelayMs &&\n (config.retryDelayMs < 0 || config.retryDelayMs > 30000)\n ) {\n throw new Error('Retry delay must be between 0 and 30000ms');\n }\n }\n\n private normalizeConfig(config: MigrationConfig): Required<MigrationConfig> {\n return {\n ...config,\n batchSize: config.batchSize ?? 1000,\n retryAttempts: config.retryAttempts ?? 3,\n retryDelayMs: config.retryDelayMs ?? 1000,\n verifyData: config.verifyData ?? true,\n enableDualWrite: config.enableDualWrite ?? true,\n progressCallback: config.progressCallback ?? (() => {}),\n };\n }\n\n private initializeProgress(): MigrationProgress {\n return {\n phase: 'initializing',\n totalRecords: 0,\n processedRecords: 0,\n percentage: 0,\n startTime: new Date(),\n errors: [],\n warnings: [],\n };\n }\n\n async planMigration(): Promise<TableMigrationPlan[]> {\n logger.info('Planning migration strategy');\n\n const plan: TableMigrationPlan[] = [];\n const tables = ['frames', 'events', 'anchors'];\n\n for (const table of tables) {\n try {\n const stats = await this.config.sourceAdapter.getStats();\n const estimatedRows = this.estimateTableRows(table, stats);\n\n plan.push({\n table,\n priority: this.getTablePriority(table),\n estimatedRows,\n dependencies: this.getTableDependencies(table),\n strategy: 'full',\n });\n } catch (error) {\n logger.warn(`Failed to estimate rows for table ${table}:`, error);\n plan.push({\n table,\n priority: this.getTablePriority(table),\n estimatedRows: 0,\n dependencies: this.getTableDependencies(table),\n strategy: 'skip',\n });\n }\n }\n\n // Sort by priority (dependencies first)\n plan.sort((a, b) => a.priority - b.priority);\n\n const totalRecords = plan.reduce((sum, p) => sum + p.estimatedRows, 0);\n this.progress.totalRecords = totalRecords;\n\n logger.info(\n `Migration plan: ${plan.length} tables, ~${totalRecords} records`\n );\n return plan;\n }\n\n private estimateTableRows(table: string, stats: any): number {\n switch (table) {\n case 'frames':\n return stats.totalFrames || 0;\n case 'events':\n return stats.totalEvents || 0;\n case 'anchors':\n return stats.totalAnchors || 0;\n default:\n return 0;\n }\n }\n\n private getTablePriority(table: string): number {\n const priorities = { frames: 1, events: 2, anchors: 3 };\n return priorities[table as keyof typeof priorities] || 99;\n }\n\n private getTableDependencies(table: string): string[] {\n const dependencies = {\n frames: [],\n events: ['frames'],\n anchors: ['frames'],\n };\n return dependencies[table as keyof typeof dependencies] || [];\n }\n\n async migrate(\n strategy: MigrationStrategy = {\n type: 'online',\n allowWrites: true,\n verifyIntegrity: true,\n fallbackOnError: true,\n }\n ): Promise<void> {\n if (this.isRunning) {\n throw new Error('Migration already in progress');\n }\n\n this.isRunning = true;\n this.abortController = new AbortController();\n\n try {\n logger.info('Starting database migration', strategy);\n this.updateProgress({ phase: 'initializing' });\n\n // Validate adapters\n await this.validateAdapters();\n\n // Create migration plan\n const plan = await this.planMigration();\n\n // Initialize target schema\n await this.initializeTargetSchema();\n\n // Enable dual-write if requested\n if (strategy.type === 'dual-write' && this.config.enableDualWrite) {\n await this.enableDualWrite();\n }\n\n // Execute migration\n this.updateProgress({ phase: 'migrating' });\n await this.executeMigrationPlan(plan, strategy);\n\n // Verify data integrity\n if (strategy.verifyIntegrity) {\n this.updateProgress({ phase: 'verifying' });\n await this.verifyDataIntegrity(plan);\n }\n\n // Complete migration\n this.updateProgress({ phase: 'completing' });\n await this.completeMigration(strategy);\n\n this.updateProgress({ phase: 'completed', percentage: 100 });\n logger.info('Migration completed successfully');\n this.emit('completed', this.progress);\n } catch (error) {\n this.updateProgress({ phase: 'failed' });\n\n // Sanitize error for logging\n const sanitizedError = this.sanitizeError(error);\n logger.error('Migration failed:', sanitizedError);\n\n if (strategy.fallbackOnError) {\n try {\n await this.rollbackMigration();\n } catch (rollbackError) {\n logger.error('Rollback failed:', this.sanitizeError(rollbackError));\n }\n }\n\n // Create user-safe error message\n const userError = new Error('Migration failed. Check logs for details.');\n this.emit('failed', userError);\n throw userError;\n } finally {\n this.isRunning = false;\n this.abortController = undefined;\n }\n }\n\n private async validateAdapters(): Promise<void> {\n logger.debug('Validating database adapters');\n\n // Check source adapter\n if (!this.config.sourceAdapter.isConnected()) {\n await this.config.sourceAdapter.connect();\n }\n\n if (!(await this.config.sourceAdapter.ping())) {\n throw new Error('Source adapter is not responding');\n }\n\n // Check target adapter\n if (!this.config.targetAdapter.isConnected()) {\n await this.config.targetAdapter.connect();\n }\n\n if (!(await this.config.targetAdapter.ping())) {\n throw new Error('Target adapter is not responding');\n }\n\n // Verify schema compatibility\n const sourceVersion = await this.config.sourceAdapter.getSchemaVersion();\n const targetVersion = await this.config.targetAdapter.getSchemaVersion();\n\n if (sourceVersion !== targetVersion) {\n logger.warn(\n `Schema version mismatch: source=${sourceVersion}, target=${targetVersion}`\n );\n this.addWarning('Schema version mismatch detected');\n }\n }\n\n private async initializeTargetSchema(): Promise<void> {\n logger.debug('Initializing target schema');\n\n try {\n await this.config.targetAdapter.initializeSchema();\n } catch (error) {\n logger.error('Failed to initialize target schema:', error);\n throw new Error(`Target schema initialization failed: ${error}`);\n }\n }\n\n private async enableDualWrite(): Promise<void> {\n logger.info('Enabling dual-write mode');\n // This would typically involve configuring the application to write to both databases\n // For now, we'll just log the intention\n this.addWarning(\n 'Dual-write mode enabled - ensure application routes writes to both adapters'\n );\n }\n\n private async executeMigrationPlan(\n plan: TableMigrationPlan[],\n strategy: MigrationStrategy\n ): Promise<void> {\n for (const tablePlan of plan) {\n if (this.abortController?.signal.aborted) {\n throw new Error('Migration aborted by user');\n }\n\n if (tablePlan.strategy === 'skip') {\n logger.info(`Skipping table: ${tablePlan.table}`);\n continue;\n }\n\n this.updateProgress({ currentTable: tablePlan.table });\n await this.migrateTable(tablePlan, strategy);\n }\n }\n\n private async migrateTable(\n plan: TableMigrationPlan,\n strategy: MigrationStrategy\n ): Promise<void> {\n logger.info(`Migrating table: ${plan.table} (~${plan.estimatedRows} rows)`);\n\n let offset = 0;\n let migratedRows = 0;\n\n while (true) {\n if (this.abortController?.signal.aborted || this.isPaused) {\n break;\n }\n\n try {\n // Get batch of data from source\n const batch = await this.getBatch(\n plan.table,\n offset,\n this.config.batchSize\n );\n\n if (batch.length === 0) {\n break; // No more data\n }\n\n // Migrate batch to target\n await this.migrateBatch(plan.table, batch);\n\n migratedRows += batch.length;\n offset += this.config.batchSize;\n\n this.progress.processedRecords += batch.length;\n this.updateProgressPercentage();\n\n // Adaptive delay based on system resources\n await this.sleep(this.calculateAdaptiveDelay());\n } catch (error) {\n this.addError(plan.table, `Batch migration failed: ${error}`);\n\n if (this.config.retryAttempts > 0) {\n await this.retryBatch(plan.table, offset, this.config.batchSize);\n } else {\n throw error;\n }\n }\n }\n\n logger.info(\n `Completed migrating table ${plan.table}: ${migratedRows} rows`\n );\n }\n\n private async getBatch(\n table: string,\n offset: number,\n limit: number\n ): Promise<any[]> {\n // Validate table name against whitelist\n const allowedTables = ['frames', 'events', 'anchors'] as const;\n if (!allowedTables.includes(table as any)) {\n throw new Error(`Invalid table name: ${table}`);\n }\n\n // Validate and bound parameters\n const safeLimit = Math.max(1, Math.min(limit, 10000));\n const safeOffset = Math.max(0, offset);\n\n const options = {\n limit: safeLimit,\n offset: safeOffset,\n orderBy: 'created_at',\n orderDirection: 'ASC' as const,\n };\n\n switch (table) {\n case 'frames':\n // This would need to be implemented in the adapter\n return []; // Placeholder\n case 'events':\n return []; // Placeholder\n case 'anchors':\n return []; // Placeholder\n default:\n throw new Error(`Unsupported table: ${table}`);\n }\n }\n\n private async migrateBatch(table: string, batch: any[]): Promise<void> {\n // Validate table name\n const allowedTables = ['frames', 'events', 'anchors'] as const;\n if (!allowedTables.includes(table as any)) {\n throw new Error(`Invalid table name: ${table}`);\n }\n\n // Use transaction for batch safety\n await this.config.targetAdapter.inTransaction(async (adapter) => {\n const operations = batch.map((row) => ({\n type: 'insert' as const,\n table,\n data: this.validateRowData(table, row),\n }));\n\n await adapter.executeBulk(operations);\n });\n }\n\n private validateRowData(table: string, row: any): any {\n if (!row || typeof row !== 'object') {\n throw new Error(`Invalid row data for table ${table}`);\n }\n\n switch (table) {\n case 'frames':\n return this.validateFrameRow(row);\n case 'events':\n return this.validateEventRow(row);\n case 'anchors':\n return this.validateAnchorRow(row);\n default:\n throw new Error(`Unknown table: ${table}`);\n }\n }\n\n private validateFrameRow(row: any): any {\n const required = [\n 'frame_id',\n 'project_id',\n 'run_id',\n 'type',\n 'name',\n 'state',\n 'depth',\n ];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in frame row`);\n }\n }\n return row;\n }\n\n private validateEventRow(row: any): any {\n const required = ['event_id', 'frame_id', 'seq', 'type', 'text'];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in event row`);\n }\n }\n return row;\n }\n\n private validateAnchorRow(row: any): any {\n const required = ['anchor_id', 'frame_id', 'type', 'text', 'priority'];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in anchor row`);\n }\n }\n return row;\n }\n\n private async retryBatch(\n table: string,\n offset: number,\n batchSize: number\n ): Promise<void> {\n for (let attempt = 1; attempt <= this.config.retryAttempts; attempt++) {\n try {\n await this.sleep(this.config.retryDelayMs * attempt);\n\n const batch = await this.getBatch(table, offset, batchSize);\n await this.migrateBatch(table, batch);\n\n logger.info(`Retry successful for table ${table} at offset ${offset}`);\n return;\n } catch (error) {\n logger.warn(\n `Retry ${attempt}/${this.config.retryAttempts} failed:`,\n error\n );\n\n if (attempt === this.config.retryAttempts) {\n throw new Error(\n `Failed after ${this.config.retryAttempts} retries: ${error}`\n );\n }\n }\n }\n }\n\n private async verifyDataIntegrity(plan: TableMigrationPlan[]): Promise<void> {\n logger.info('Verifying data integrity');\n\n for (const tablePlan of plan) {\n if (tablePlan.strategy === 'skip') continue;\n\n try {\n const sourceStats = await this.config.sourceAdapter.getStats();\n const targetStats = await this.config.targetAdapter.getStats();\n\n const sourceCount = this.estimateTableRows(\n tablePlan.table,\n sourceStats\n );\n const targetCount = this.estimateTableRows(\n tablePlan.table,\n targetStats\n );\n\n if (sourceCount !== targetCount) {\n this.addError(\n tablePlan.table,\n `Row count mismatch: source=${sourceCount}, target=${targetCount}`\n );\n } else {\n logger.debug(\n `Table ${tablePlan.table} verified: ${sourceCount} rows`\n );\n }\n } catch (error) {\n this.addError(tablePlan.table, `Verification failed: ${error}`);\n }\n }\n\n if (this.progress.errors.length > 0) {\n throw new Error(\n `Data integrity verification failed with ${this.progress.errors.length} errors`\n );\n }\n }\n\n private async completeMigration(strategy: MigrationStrategy): Promise<void> {\n logger.info('Completing migration');\n\n // Update target schema version if needed\n const sourceVersion = await this.config.sourceAdapter.getSchemaVersion();\n await this.config.targetAdapter.migrateSchema(sourceVersion);\n\n // Analyze target database for optimal performance\n await this.config.targetAdapter.analyze();\n\n logger.info('Migration completion tasks finished');\n }\n\n private async rollbackMigration(): Promise<void> {\n logger.warn('Rolling back migration');\n\n try {\n // This would typically involve cleaning up the target database\n // For now, we'll just log the intention\n logger.warn(\n 'Rollback would clean target database - implement based on strategy'\n );\n } catch (error) {\n logger.error('Rollback failed:', error);\n }\n }\n\n private updateProgress(updates: Partial<MigrationProgress>): void {\n Object.assign(this.progress, updates);\n this.updateProgressPercentage();\n\n if (this.progress.totalRecords > 0) {\n const elapsed = Date.now() - this.progress.startTime.getTime();\n const rate = this.progress.processedRecords / (elapsed / 1000);\n const remaining =\n this.progress.totalRecords - this.progress.processedRecords;\n\n if (rate > 0) {\n this.progress.estimatedEndTime = new Date(\n Date.now() + (remaining / rate) * 1000\n );\n }\n }\n\n this.config.progressCallback(this.progress);\n this.emit('progress', this.progress);\n }\n\n private updateProgressPercentage(): void {\n if (this.progress.totalRecords > 0) {\n this.progress.percentage = Math.min(\n 100,\n (this.progress.processedRecords / this.progress.totalRecords) * 100\n );\n }\n }\n\n private addError(table: string, error: string): void {\n this.progress.errors.push({\n table,\n error,\n timestamp: new Date(),\n });\n\n logger.error(`Migration error for table ${table}: ${error}`);\n }\n\n private addWarning(warning: string, table?: string): void {\n this.progress.warnings.push({\n table: table || 'general',\n warning,\n timestamp: new Date(),\n });\n\n logger.warn(`Migration warning: ${warning}`);\n }\n\n private sanitizeError(error: any): any {\n if (error instanceof Error) {\n return {\n name: error.name,\n message: error.message,\n // Exclude stack traces and sensitive data for security\n };\n }\n return { message: 'Unknown error occurred' };\n }\n\n private calculateAdaptiveDelay(): number {\n const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024;\n\n // Adaptive delay based on system resources\n if (memoryUsage > 400) return 100;\n if (memoryUsage > 300) return 50;\n return 10;\n }\n\n private sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n }\n\n pause(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.isPaused = true;\n logger.info('Migration paused');\n this.emit('paused');\n }\n\n resume(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.isPaused = false;\n logger.info('Migration resumed');\n this.emit('resumed');\n }\n\n abort(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.abortController?.abort();\n logger.info('Migration aborted');\n this.emit('aborted');\n }\n\n getProgress(): MigrationProgress {\n return { ...this.progress };\n }\n\n isActive(): boolean {\n return this.isRunning;\n }\n\n async estimateDuration(): Promise<{\n estimatedMinutes: number;\n confidence: 'low' | 'medium' | 'high';\n }> {\n const plan = await this.planMigration();\n const totalRecords = plan.reduce((sum, p) => sum + p.estimatedRows, 0);\n\n // Rough estimate: 1000 records per second\n const estimatedSeconds = totalRecords / 1000;\n const estimatedMinutes = Math.ceil(estimatedSeconds / 60);\n\n let confidence: 'low' | 'medium' | 'high' = 'medium';\n if (totalRecords < 10000) confidence = 'high';\n if (totalRecords > 100000) confidence = 'low';\n\n return { estimatedMinutes, confidence };\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Migration Manager for Dual-Write Strategy\n * Enables seamless migration between SQLite and ParadeDB with zero downtime\n */\n\nimport { EventEmitter } from 'events';\nimport { DatabaseAdapter } from './database-adapter.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport interface MigrationConfig {\n sourceAdapter: DatabaseAdapter;\n targetAdapter: DatabaseAdapter;\n batchSize?: number;\n retryAttempts?: number;\n retryDelayMs?: number;\n verifyData?: boolean;\n enableDualWrite?: boolean;\n progressCallback?: (progress: MigrationProgress) => void;\n}\n\nexport interface MigrationProgress {\n phase:\n | 'initializing'\n | 'migrating'\n | 'verifying'\n | 'completing'\n | 'completed'\n | 'failed';\n totalRecords: number;\n processedRecords: number;\n percentage: number;\n startTime: Date;\n estimatedEndTime?: Date;\n currentTable?: string;\n errors: Array<{ table: string; error: string; timestamp: Date }>;\n warnings: Array<{ table: string; warning: string; timestamp: Date }>;\n}\n\nexport interface MigrationStrategy {\n type: 'online' | 'offline' | 'dual-write';\n allowWrites: boolean;\n verifyIntegrity: boolean;\n fallbackOnError: boolean;\n}\n\nexport interface TableMigrationPlan {\n table: string;\n priority: number;\n estimatedRows: number;\n dependencies: string[];\n strategy: 'full' | 'incremental' | 'skip';\n}\n\nexport class MigrationManager extends EventEmitter {\n private config: Required<MigrationConfig>;\n private progress: MigrationProgress;\n private isRunning = false;\n private isPaused = false;\n private abortController?: AbortController;\n\n constructor(config: MigrationConfig) {\n super();\n\n this.validateConfig(config);\n this.config = this.normalizeConfig(config);\n this.progress = this.initializeProgress();\n }\n\n private validateConfig(config: MigrationConfig): void {\n if (!config.sourceAdapter || !config.targetAdapter) {\n throw new Error('Source and target adapters are required');\n }\n\n if (\n config.batchSize &&\n (config.batchSize < 1 || config.batchSize > 10000)\n ) {\n throw new Error('Batch size must be between 1 and 10000');\n }\n\n if (\n config.retryAttempts &&\n (config.retryAttempts < 0 || config.retryAttempts > 10)\n ) {\n throw new Error('Retry attempts must be between 0 and 10');\n }\n\n if (\n config.retryDelayMs &&\n (config.retryDelayMs < 0 || config.retryDelayMs > 30000)\n ) {\n throw new Error('Retry delay must be between 0 and 30000ms');\n }\n }\n\n private normalizeConfig(config: MigrationConfig): Required<MigrationConfig> {\n return {\n ...config,\n batchSize: config.batchSize ?? 1000,\n retryAttempts: config.retryAttempts ?? 3,\n retryDelayMs: config.retryDelayMs ?? 1000,\n verifyData: config.verifyData ?? true,\n enableDualWrite: config.enableDualWrite ?? true,\n progressCallback: config.progressCallback ?? (() => {}),\n };\n }\n\n private initializeProgress(): MigrationProgress {\n return {\n phase: 'initializing',\n totalRecords: 0,\n processedRecords: 0,\n percentage: 0,\n startTime: new Date(),\n errors: [],\n warnings: [],\n };\n }\n\n async planMigration(): Promise<TableMigrationPlan[]> {\n logger.info('Planning migration strategy');\n\n const plan: TableMigrationPlan[] = [];\n const tables = ['frames', 'events', 'anchors'];\n\n for (const table of tables) {\n try {\n const stats = await this.config.sourceAdapter.getStats();\n const estimatedRows = this.estimateTableRows(table, stats);\n\n plan.push({\n table,\n priority: this.getTablePriority(table),\n estimatedRows,\n dependencies: this.getTableDependencies(table),\n strategy: 'full',\n });\n } catch (error: unknown) {\n logger.warn(`Failed to estimate rows for table ${table}:`, error);\n plan.push({\n table,\n priority: this.getTablePriority(table),\n estimatedRows: 0,\n dependencies: this.getTableDependencies(table),\n strategy: 'skip',\n });\n }\n }\n\n // Sort by priority (dependencies first)\n plan.sort((a, b) => a.priority - b.priority);\n\n const totalRecords = plan.reduce((sum, p) => sum + p.estimatedRows, 0);\n this.progress.totalRecords = totalRecords;\n\n logger.info(\n `Migration plan: ${plan.length} tables, ~${totalRecords} records`\n );\n return plan;\n }\n\n private estimateTableRows(table: string, stats: any): number {\n switch (table) {\n case 'frames':\n return stats.totalFrames || 0;\n case 'events':\n return stats.totalEvents || 0;\n case 'anchors':\n return stats.totalAnchors || 0;\n default:\n return 0;\n }\n }\n\n private getTablePriority(table: string): number {\n const priorities = { frames: 1, events: 2, anchors: 3 };\n return priorities[table as keyof typeof priorities] || 99;\n }\n\n private getTableDependencies(table: string): string[] {\n const dependencies = {\n frames: [],\n events: ['frames'],\n anchors: ['frames'],\n };\n return dependencies[table as keyof typeof dependencies] || [];\n }\n\n async migrate(\n strategy: MigrationStrategy = {\n type: 'online',\n allowWrites: true,\n verifyIntegrity: true,\n fallbackOnError: true,\n }\n ): Promise<void> {\n if (this.isRunning) {\n throw new Error('Migration already in progress');\n }\n\n this.isRunning = true;\n this.abortController = new AbortController();\n\n try {\n logger.info('Starting database migration', strategy);\n this.updateProgress({ phase: 'initializing' });\n\n // Validate adapters\n await this.validateAdapters();\n\n // Create migration plan\n const plan = await this.planMigration();\n\n // Initialize target schema\n await this.initializeTargetSchema();\n\n // Enable dual-write if requested\n if (strategy.type === 'dual-write' && this.config.enableDualWrite) {\n await this.enableDualWrite();\n }\n\n // Execute migration\n this.updateProgress({ phase: 'migrating' });\n await this.executeMigrationPlan(plan, strategy);\n\n // Verify data integrity\n if (strategy.verifyIntegrity) {\n this.updateProgress({ phase: 'verifying' });\n await this.verifyDataIntegrity(plan);\n }\n\n // Complete migration\n this.updateProgress({ phase: 'completing' });\n await this.completeMigration(strategy);\n\n this.updateProgress({ phase: 'completed', percentage: 100 });\n logger.info('Migration completed successfully');\n this.emit('completed', this.progress);\n } catch (error: unknown) {\n this.updateProgress({ phase: 'failed' });\n\n // Sanitize error for logging\n const sanitizedError = this.sanitizeError(error);\n logger.error('Migration failed:', sanitizedError);\n\n if (strategy.fallbackOnError) {\n try {\n await this.rollbackMigration();\n } catch (rollbackError: unknown) {\n logger.error('Rollback failed:', this.sanitizeError(rollbackError));\n }\n }\n\n // Create user-safe error message\n const userError = new Error('Migration failed. Check logs for details.');\n this.emit('failed', userError);\n throw userError;\n } finally {\n this.isRunning = false;\n this.abortController = undefined;\n }\n }\n\n private async validateAdapters(): Promise<void> {\n logger.debug('Validating database adapters');\n\n // Check source adapter\n if (!this.config.sourceAdapter.isConnected()) {\n await this.config.sourceAdapter.connect();\n }\n\n if (!(await this.config.sourceAdapter.ping())) {\n throw new Error('Source adapter is not responding');\n }\n\n // Check target adapter\n if (!this.config.targetAdapter.isConnected()) {\n await this.config.targetAdapter.connect();\n }\n\n if (!(await this.config.targetAdapter.ping())) {\n throw new Error('Target adapter is not responding');\n }\n\n // Verify schema compatibility\n const sourceVersion = await this.config.sourceAdapter.getSchemaVersion();\n const targetVersion = await this.config.targetAdapter.getSchemaVersion();\n\n if (sourceVersion !== targetVersion) {\n logger.warn(\n `Schema version mismatch: source=${sourceVersion}, target=${targetVersion}`\n );\n this.addWarning('Schema version mismatch detected');\n }\n }\n\n private async initializeTargetSchema(): Promise<void> {\n logger.debug('Initializing target schema');\n\n try {\n await this.config.targetAdapter.initializeSchema();\n } catch (error: unknown) {\n logger.error('Failed to initialize target schema:', error);\n throw new Error(`Target schema initialization failed: ${error}`);\n }\n }\n\n private async enableDualWrite(): Promise<void> {\n logger.info('Enabling dual-write mode');\n // This would typically involve configuring the application to write to both databases\n // For now, we'll just log the intention\n this.addWarning(\n 'Dual-write mode enabled - ensure application routes writes to both adapters'\n );\n }\n\n private async executeMigrationPlan(\n plan: TableMigrationPlan[],\n strategy: MigrationStrategy\n ): Promise<void> {\n for (const tablePlan of plan) {\n if (this.abortController?.signal.aborted) {\n throw new Error('Migration aborted by user');\n }\n\n if (tablePlan.strategy === 'skip') {\n logger.info(`Skipping table: ${tablePlan.table}`);\n continue;\n }\n\n this.updateProgress({ currentTable: tablePlan.table });\n await this.migrateTable(tablePlan, strategy);\n }\n }\n\n private async migrateTable(\n plan: TableMigrationPlan,\n strategy: MigrationStrategy\n ): Promise<void> {\n logger.info(`Migrating table: ${plan.table} (~${plan.estimatedRows} rows)`);\n\n let offset = 0;\n let migratedRows = 0;\n\n while (true) {\n if (this.abortController?.signal.aborted || this.isPaused) {\n break;\n }\n\n try {\n // Get batch of data from source\n const batch = await this.getBatch(\n plan.table,\n offset,\n this.config.batchSize\n );\n\n if (batch.length === 0) {\n break; // No more data\n }\n\n // Migrate batch to target\n await this.migrateBatch(plan.table, batch);\n\n migratedRows += batch.length;\n offset += this.config.batchSize;\n\n this.progress.processedRecords += batch.length;\n this.updateProgressPercentage();\n\n // Adaptive delay based on system resources\n await this.sleep(this.calculateAdaptiveDelay());\n } catch (error: unknown) {\n this.addError(plan.table, `Batch migration failed: ${error}`);\n\n if (this.config.retryAttempts > 0) {\n await this.retryBatch(plan.table, offset, this.config.batchSize);\n } else {\n throw error;\n }\n }\n }\n\n logger.info(\n `Completed migrating table ${plan.table}: ${migratedRows} rows`\n );\n }\n\n private async getBatch(\n table: string,\n offset: number,\n limit: number\n ): Promise<any[]> {\n // Validate table name against whitelist\n const allowedTables = ['frames', 'events', 'anchors'] as const;\n if (!allowedTables.includes(table as any)) {\n throw new Error(`Invalid table name: ${table}`);\n }\n\n // Validate and bound parameters\n const safeLimit = Math.max(1, Math.min(limit, 10000));\n const safeOffset = Math.max(0, offset);\n\n const options = {\n limit: safeLimit,\n offset: safeOffset,\n orderBy: 'created_at',\n orderDirection: 'ASC' as const,\n };\n\n switch (table) {\n case 'frames':\n // This would need to be implemented in the adapter\n return []; // Placeholder\n case 'events':\n return []; // Placeholder\n case 'anchors':\n return []; // Placeholder\n default:\n throw new Error(`Unsupported table: ${table}`);\n }\n }\n\n private async migrateBatch(table: string, batch: any[]): Promise<void> {\n // Validate table name\n const allowedTables = ['frames', 'events', 'anchors'] as const;\n if (!allowedTables.includes(table as any)) {\n throw new Error(`Invalid table name: ${table}`);\n }\n\n // Use transaction for batch safety\n await this.config.targetAdapter.inTransaction(async (adapter) => {\n const operations = batch.map((row) => ({\n type: 'insert' as const,\n table,\n data: this.validateRowData(table, row),\n }));\n\n await adapter.executeBulk(operations);\n });\n }\n\n private validateRowData(table: string, row: any): any {\n if (!row || typeof row !== 'object') {\n throw new Error(`Invalid row data for table ${table}`);\n }\n\n switch (table) {\n case 'frames':\n return this.validateFrameRow(row);\n case 'events':\n return this.validateEventRow(row);\n case 'anchors':\n return this.validateAnchorRow(row);\n default:\n throw new Error(`Unknown table: ${table}`);\n }\n }\n\n private validateFrameRow(row: any): any {\n const required = [\n 'frame_id',\n 'project_id',\n 'run_id',\n 'type',\n 'name',\n 'state',\n 'depth',\n ];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in frame row`);\n }\n }\n return row;\n }\n\n private validateEventRow(row: any): any {\n const required = ['event_id', 'frame_id', 'seq', 'type', 'text'];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in event row`);\n }\n }\n return row;\n }\n\n private validateAnchorRow(row: any): any {\n const required = ['anchor_id', 'frame_id', 'type', 'text', 'priority'];\n for (const field of required) {\n if (!(field in row)) {\n throw new Error(`Missing required field ${field} in anchor row`);\n }\n }\n return row;\n }\n\n private async retryBatch(\n table: string,\n offset: number,\n batchSize: number\n ): Promise<void> {\n for (let attempt = 1; attempt <= this.config.retryAttempts; attempt++) {\n try {\n await this.sleep(this.config.retryDelayMs * attempt);\n\n const batch = await this.getBatch(table, offset, batchSize);\n await this.migrateBatch(table, batch);\n\n logger.info(`Retry successful for table ${table} at offset ${offset}`);\n return;\n } catch (error: unknown) {\n logger.warn(\n `Retry ${attempt}/${this.config.retryAttempts} failed:`,\n error\n );\n\n if (attempt === this.config.retryAttempts) {\n throw new Error(\n `Failed after ${this.config.retryAttempts} retries: ${error}`\n );\n }\n }\n }\n }\n\n private async verifyDataIntegrity(plan: TableMigrationPlan[]): Promise<void> {\n logger.info('Verifying data integrity');\n\n for (const tablePlan of plan) {\n if (tablePlan.strategy === 'skip') continue;\n\n try {\n const sourceStats = await this.config.sourceAdapter.getStats();\n const targetStats = await this.config.targetAdapter.getStats();\n\n const sourceCount = this.estimateTableRows(\n tablePlan.table,\n sourceStats\n );\n const targetCount = this.estimateTableRows(\n tablePlan.table,\n targetStats\n );\n\n if (sourceCount !== targetCount) {\n this.addError(\n tablePlan.table,\n `Row count mismatch: source=${sourceCount}, target=${targetCount}`\n );\n } else {\n logger.debug(\n `Table ${tablePlan.table} verified: ${sourceCount} rows`\n );\n }\n } catch (error: unknown) {\n this.addError(tablePlan.table, `Verification failed: ${error}`);\n }\n }\n\n if (this.progress.errors.length > 0) {\n throw new Error(\n `Data integrity verification failed with ${this.progress.errors.length} errors`\n );\n }\n }\n\n private async completeMigration(strategy: MigrationStrategy): Promise<void> {\n logger.info('Completing migration');\n\n // Update target schema version if needed\n const sourceVersion = await this.config.sourceAdapter.getSchemaVersion();\n await this.config.targetAdapter.migrateSchema(sourceVersion);\n\n // Analyze target database for optimal performance\n await this.config.targetAdapter.analyze();\n\n logger.info('Migration completion tasks finished');\n }\n\n private async rollbackMigration(): Promise<void> {\n logger.warn('Rolling back migration');\n\n try {\n // This would typically involve cleaning up the target database\n // For now, we'll just log the intention\n logger.warn(\n 'Rollback would clean target database - implement based on strategy'\n );\n } catch (error: unknown) {\n logger.error('Rollback failed:', error);\n }\n }\n\n private updateProgress(updates: Partial<MigrationProgress>): void {\n Object.assign(this.progress, updates);\n this.updateProgressPercentage();\n\n if (this.progress.totalRecords > 0) {\n const elapsed = Date.now() - this.progress.startTime.getTime();\n const rate = this.progress.processedRecords / (elapsed / 1000);\n const remaining =\n this.progress.totalRecords - this.progress.processedRecords;\n\n if (rate > 0) {\n this.progress.estimatedEndTime = new Date(\n Date.now() + (remaining / rate) * 1000\n );\n }\n }\n\n this.config.progressCallback(this.progress);\n this.emit('progress', this.progress);\n }\n\n private updateProgressPercentage(): void {\n if (this.progress.totalRecords > 0) {\n this.progress.percentage = Math.min(\n 100,\n (this.progress.processedRecords / this.progress.totalRecords) * 100\n );\n }\n }\n\n private addError(table: string, error: string): void {\n this.progress.errors.push({\n table,\n error,\n timestamp: new Date(),\n });\n\n logger.error(`Migration error for table ${table}: ${error}`);\n }\n\n private addWarning(warning: string, table?: string): void {\n this.progress.warnings.push({\n table: table || 'general',\n warning,\n timestamp: new Date(),\n });\n\n logger.warn(`Migration warning: ${warning}`);\n }\n\n private sanitizeError(error: any): any {\n if (error instanceof Error) {\n return {\n name: error.name,\n message: error.message,\n // Exclude stack traces and sensitive data for security\n };\n }\n return { message: 'Unknown error occurred' };\n }\n\n private calculateAdaptiveDelay(): number {\n const memoryUsage = process.memoryUsage().heapUsed / 1024 / 1024;\n\n // Adaptive delay based on system resources\n if (memoryUsage > 400) return 100;\n if (memoryUsage > 300) return 50;\n return 10;\n }\n\n private sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n }\n\n pause(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.isPaused = true;\n logger.info('Migration paused');\n this.emit('paused');\n }\n\n resume(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.isPaused = false;\n logger.info('Migration resumed');\n this.emit('resumed');\n }\n\n abort(): void {\n if (!this.isRunning) {\n throw new Error('No migration in progress');\n }\n\n this.abortController?.abort();\n logger.info('Migration aborted');\n this.emit('aborted');\n }\n\n getProgress(): MigrationProgress {\n return { ...this.progress };\n }\n\n isActive(): boolean {\n return this.isRunning;\n }\n\n async estimateDuration(): Promise<{\n estimatedMinutes: number;\n confidence: 'low' | 'medium' | 'high';\n }> {\n const plan = await this.planMigration();\n const totalRecords = plan.reduce((sum, p) => sum + p.estimatedRows, 0);\n\n // Rough estimate: 1000 records per second\n const estimatedSeconds = totalRecords / 1000;\n const estimatedMinutes = Math.ceil(estimatedSeconds / 60);\n\n let confidence: 'low' | 'medium' | 'high' = 'medium';\n if (totalRecords < 10000) confidence = 'high';\n if (totalRecords > 100000) confidence = 'low';\n\n return { estimatedMinutes, confidence };\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/database/paradedb-adapter.ts"],
4
- "sourcesContent": ["/**\n * ParadeDB Database Adapter\n * Advanced PostgreSQL with built-in search (BM25) and analytics capabilities\n */\n\nimport { Pool, PoolClient, QueryResult } from 'pg';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/frame-manager.js';\nimport { logger } from '../monitoring/logger.js';\nimport * as fs from 'fs/promises';\n\nexport interface ParadeDBConfig {\n connectionString?: string;\n host?: string;\n port?: number;\n database?: string;\n user?: string;\n password?: string;\n ssl?: boolean | { rejectUnauthorized?: boolean };\n max?: number; // Max pool size\n idleTimeoutMillis?: number;\n connectionTimeoutMillis?: number;\n statementTimeout?: number;\n enableBM25?: boolean;\n enableVector?: boolean;\n enableAnalytics?: boolean;\n}\n\nexport class ParadeDBAdapter extends FeatureAwareDatabaseAdapter {\n private pool: Pool | null = null;\n private activeClient: PoolClient | null = null;\n\n constructor(projectId: string, config: ParadeDBConfig) {\n super(projectId, config);\n }\n\n getFeatures(): DatabaseFeatures {\n const config = this.config as ParadeDBConfig;\n return {\n supportsFullTextSearch: config.enableBM25 !== false,\n supportsVectorSearch: config.enableVector !== false,\n supportsPartitioning: true,\n supportsAnalytics: config.enableAnalytics !== false,\n supportsCompression: true,\n supportsMaterializedViews: true,\n supportsParallelQueries: true,\n };\n }\n\n async connect(): Promise<void> {\n if (this.pool) return;\n\n const config = this.config as ParadeDBConfig;\n\n this.pool = new Pool({\n connectionString: config.connectionString,\n host: config.host || 'localhost',\n port: config.port || 5432,\n database: config.database || 'stackmemory',\n user: config.user,\n password: config.password,\n ssl: config.ssl,\n max: config.max || 20,\n idleTimeoutMillis: config.idleTimeoutMillis || 30000,\n connectionTimeoutMillis: config.connectionTimeoutMillis || 2000,\n statement_timeout: config.statementTimeout || 30000,\n });\n\n // Test connection\n const client = await this.pool.connect();\n try {\n await client.query('SELECT 1');\n logger.info('ParadeDB connected successfully');\n } finally {\n client.release();\n }\n }\n\n async disconnect(): Promise<void> {\n if (!this.pool) return;\n\n await this.pool.end();\n this.pool = null;\n logger.info('ParadeDB disconnected');\n }\n\n isConnected(): boolean {\n return this.pool !== null && !this.pool.ended;\n }\n\n async ping(): Promise<boolean> {\n if (!this.pool) return false;\n\n try {\n const client = await this.pool.connect();\n try {\n await client.query('SELECT 1');\n return true;\n } finally {\n client.release();\n }\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('BEGIN');\n\n // Enable required extensions\n await client.query(`\n CREATE EXTENSION IF NOT EXISTS \"uuid-ossp\";\n CREATE EXTENSION IF NOT EXISTS \"pg_trgm\";\n CREATE EXTENSION IF NOT EXISTS \"btree_gin\";\n `);\n\n // Enable ParadeDB extensions if configured\n const config = this.config as ParadeDBConfig;\n\n if (config.enableBM25 !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS pg_search;');\n }\n\n if (config.enableVector !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS vector;');\n }\n\n if (config.enableAnalytics !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS pg_analytics;');\n }\n\n // Create main tables with partitioning support\n await client.query(`\n -- Main frames table\n CREATE TABLE IF NOT EXISTS frames (\n frame_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n run_id UUID NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id UUID REFERENCES frames(frame_id) ON DELETE CASCADE,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n score FLOAT DEFAULT 0.5,\n inputs JSONB DEFAULT '{}',\n outputs JSONB DEFAULT '{}',\n metadata JSONB DEFAULT '{}',\n digest_text TEXT,\n digest_json JSONB DEFAULT '{}',\n content TEXT, -- For full-text search\n embedding vector(768), -- For vector search\n created_at TIMESTAMPTZ DEFAULT NOW(),\n closed_at TIMESTAMPTZ,\n CONSTRAINT check_state CHECK (state IN ('active', 'closed', 'suspended'))\n ) PARTITION BY RANGE (created_at);\n\n -- Create partitions for time-based data\n CREATE TABLE IF NOT EXISTS frames_recent PARTITION OF frames\n FOR VALUES FROM (NOW() - INTERVAL '30 days') TO (NOW() + INTERVAL '1 day');\n \n CREATE TABLE IF NOT EXISTS frames_archive PARTITION OF frames\n FOR VALUES FROM ('2020-01-01') TO (NOW() - INTERVAL '30 days');\n\n -- Events table\n CREATE TABLE IF NOT EXISTS events (\n event_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n run_id UUID NOT NULL,\n frame_id UUID NOT NULL REFERENCES frames(frame_id) ON DELETE CASCADE,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload JSONB NOT NULL DEFAULT '{}',\n ts TIMESTAMPTZ DEFAULT NOW()\n );\n\n -- Anchors table\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n frame_id UUID NOT NULL REFERENCES frames(frame_id) ON DELETE CASCADE,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n metadata JSONB DEFAULT '{}',\n created_at TIMESTAMPTZ DEFAULT NOW()\n );\n\n -- Schema version tracking\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at TIMESTAMPTZ DEFAULT NOW(),\n description TEXT\n );\n `);\n\n // Create indexes for performance\n await client.query(`\n -- Standard B-tree indexes\n CREATE INDEX IF NOT EXISTS idx_frames_run_id ON frames USING btree(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project_id ON frames USING btree(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames USING btree(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames USING btree(state);\n CREATE INDEX IF NOT EXISTS idx_frames_type ON frames USING btree(type);\n CREATE INDEX IF NOT EXISTS idx_frames_created_at ON frames USING btree(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_frames_score ON frames USING btree(score DESC);\n\n -- GIN indexes for JSONB\n CREATE INDEX IF NOT EXISTS idx_frames_inputs ON frames USING gin(inputs);\n CREATE INDEX IF NOT EXISTS idx_frames_outputs ON frames USING gin(outputs);\n CREATE INDEX IF NOT EXISTS idx_frames_metadata ON frames USING gin(metadata);\n CREATE INDEX IF NOT EXISTS idx_frames_digest ON frames USING gin(digest_json);\n\n -- Trigram index for fuzzy text search\n CREATE INDEX IF NOT EXISTS idx_frames_name_trgm ON frames USING gin(name gin_trgm_ops);\n CREATE INDEX IF NOT EXISTS idx_frames_content_trgm ON frames USING gin(content gin_trgm_ops);\n\n -- Event indexes\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events USING btree(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events USING btree(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_events_type ON events USING btree(event_type);\n CREATE INDEX IF NOT EXISTS idx_events_ts ON events USING btree(ts DESC);\n\n -- Anchor indexes\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors USING btree(frame_id);\n CREATE INDEX IF NOT EXISTS idx_anchors_type ON anchors USING btree(type);\n CREATE INDEX IF NOT EXISTS idx_anchors_priority ON anchors USING btree(priority DESC);\n `);\n\n // Create BM25 search index if enabled\n if (config.enableBM25 !== false) {\n await client.query(`\n -- Create BM25 index for full-text search\n CALL paradedb.create_bm25_test_table(\n index_name => 'frames_search_idx',\n table_name => 'frames',\n schema_name => 'public',\n key_field => 'frame_id',\n text_fields => paradedb.field('name') || \n paradedb.field('content') || \n paradedb.field('digest_text'),\n numeric_fields => paradedb.field('score') || \n paradedb.field('depth'),\n json_fields => paradedb.field('metadata', flatten => true),\n datetime_fields => paradedb.field('created_at')\n );\n `);\n }\n\n // Create vector index if enabled\n if (config.enableVector !== false) {\n await client.query(`\n -- HNSW index for vector similarity search\n CREATE INDEX IF NOT EXISTS idx_frames_embedding \n ON frames USING hnsw (embedding vector_cosine_ops)\n WITH (m = 16, ef_construction = 64);\n `);\n }\n\n // Create materialized views for patterns\n await client.query(`\n CREATE MATERIALIZED VIEW IF NOT EXISTS pattern_summary AS\n WITH pattern_extraction AS (\n SELECT \n project_id,\n type as pattern_type,\n metadata->>'error' as error_pattern,\n COUNT(*) as frequency,\n MAX(score) as max_score,\n MAX(created_at) as last_seen,\n MIN(created_at) as first_seen\n FROM frames\n WHERE created_at > NOW() - INTERVAL '30 days'\n GROUP BY project_id, pattern_type, error_pattern\n )\n SELECT * FROM pattern_extraction\n WHERE frequency > 3;\n\n CREATE UNIQUE INDEX IF NOT EXISTS idx_pattern_summary_unique \n ON pattern_summary(project_id, pattern_type, error_pattern);\n `);\n\n // Set initial schema version\n await client.query(`\n INSERT INTO schema_version (version, description) \n VALUES (1, 'Initial ParadeDB schema with search and analytics')\n ON CONFLICT (version) DO NOTHING;\n `);\n\n await client.query('COMMIT');\n logger.info('ParadeDB schema initialized successfully');\n } catch (error) {\n await client.query('ROLLBACK');\n throw error;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT MAX(version) as version FROM schema_version'\n );\n const currentVersion = result.rows[0]?.version || 0;\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here based on version\n await client.query(\n 'INSERT INTO schema_version (version, description) VALUES ($1, $2)',\n [v, `Migration to version ${v}`]\n );\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT MAX(version) as version FROM schema_version'\n );\n return result.rows[0]?.version || 0;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, score, inputs, outputs, metadata,\n digest_text, digest_json, content\n ) VALUES (\n COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5,\n $6, $7, $8, $9, $10, $11, $12, $13, $14, $15\n ) RETURNING frame_id\n `,\n [\n frame.frame_id || null,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n frame.score || 0.5,\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n JSON.stringify(frame.metadata || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {}),\n frame.content || `${frame.name} ${frame.digest_text || ''}`,\n ]\n );\n\n return result.rows[0].frame_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT * FROM frames WHERE frame_id = $1',\n [frameId]\n );\n\n if (result.rows.length === 0) return null;\n\n const row = result.rows[0];\n return {\n ...row,\n frame_id: row.frame_id,\n run_id: row.run_id,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n };\n } finally {\n this.releaseClient(client);\n }\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n const client = await this.getClient();\n\n try {\n const fields = [];\n const values = [];\n let paramCount = 1;\n\n if (updates.state !== undefined) {\n fields.push(`state = $${paramCount++}`);\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push(`outputs = $${paramCount++}`);\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.score !== undefined) {\n fields.push(`score = $${paramCount++}`);\n values.push(updates.score);\n }\n\n if (updates.digest_text !== undefined) {\n fields.push(`digest_text = $${paramCount++}`);\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push(`digest_json = $${paramCount++}`);\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push(`closed_at = $${paramCount++}`);\n values.push(new Date(updates.closed_at));\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n await client.query(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = $${paramCount}\n `,\n values\n );\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n // CASCADE delete handles events and anchors\n await client.query('DELETE FROM frames WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n const client = await this.getClient();\n\n try {\n let query = 'SELECT * FROM frames WHERE state = $1';\n const params: any[] = ['active'];\n\n if (runId) {\n query += ' AND run_id = $2';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const result = await client.query(query, params);\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5, $6, $7)\n RETURNING event_id\n `,\n [\n event.event_id || null,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts ? new Date(event.ts) : new Date(),\n ]\n );\n\n return result.rows[0].event_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n const client = await this.getClient();\n\n try {\n let query = 'SELECT * FROM events WHERE frame_id = $1';\n const params: any[] = [frameId];\n\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const result = await client.query(query, params);\n\n return result.rows.map((row) => ({\n ...row,\n ts: row.ts.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('DELETE FROM events WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5, $6, $7)\n RETURNING anchor_id\n `,\n [\n anchor.anchor_id || null,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {}),\n ]\n );\n\n return result.rows[0].anchor_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n SELECT * FROM anchors WHERE frame_id = $1 \n ORDER BY priority DESC, created_at ASC\n `,\n [frameId]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('DELETE FROM anchors WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Advanced search with BM25\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n const client = await this.getClient();\n\n try {\n const config = this.config as ParadeDBConfig;\n\n if (config.enableBM25 !== false) {\n // Use ParadeDB BM25 search\n const result = await client.query(\n `\n SELECT f.*, s.score_bm25 as score\n FROM frames_search_idx.search(\n query => $1,\n limit_rows => $2,\n offset_rows => $3\n ) s\n JOIN frames f ON f.frame_id = s.frame_id\n WHERE ($4::float IS NULL OR s.score_bm25 >= $4)\n ORDER BY s.score_bm25 DESC\n `,\n [\n options.query,\n options.limit || 100,\n options.offset || 0,\n options.scoreThreshold || null,\n ]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } else {\n // Fallback to PostgreSQL full-text search\n const result = await client.query(\n `\n SELECT *,\n ts_rank(\n to_tsvector('english', COALESCE(name, '') || ' ' || COALESCE(content, '')),\n plainto_tsquery('english', $1)\n ) as score\n FROM frames\n WHERE to_tsvector('english', COALESCE(name, '') || ' ' || COALESCE(content, ''))\n @@ plainto_tsquery('english', $1)\n ORDER BY score DESC\n LIMIT $2 OFFSET $3\n `,\n [options.query, options.limit || 100, options.offset || 0]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Vector similarity search\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n const client = await this.getClient();\n\n try {\n const config = this.config as ParadeDBConfig;\n\n if (config.enableVector === false) {\n logger.warn('Vector search not enabled in ParadeDB configuration');\n return [];\n }\n\n const result = await client.query(\n `\n SELECT *,\n 1 - (embedding <=> $1::vector) as similarity\n FROM frames\n WHERE embedding IS NOT NULL\n ORDER BY embedding <=> $1::vector\n LIMIT $2 OFFSET $3\n `,\n [\n `[${embedding.join(',')}]`,\n options?.limit || 100,\n options?.offset || 0,\n ]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Hybrid search combining BM25 and vector\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n const client = await this.getClient();\n\n try {\n const textWeight = weights?.text || 0.6;\n const vectorWeight = weights?.vector || 0.4;\n\n const result = await client.query(\n `\n WITH bm25_results AS (\n SELECT frame_id, score_bm25\n FROM frames_search_idx.search(\n query => $1,\n limit_rows => 200\n )\n ),\n vector_results AS (\n SELECT frame_id,\n 1 - (embedding <=> $2::vector) as score_vector\n FROM frames\n WHERE embedding IS NOT NULL\n ORDER BY embedding <=> $2::vector\n LIMIT 200\n )\n SELECT f.*,\n (COALESCE(b.score_bm25, 0) * $3 + \n COALESCE(v.score_vector, 0) * $4) as score\n FROM frames f\n LEFT JOIN bm25_results b ON f.frame_id = b.frame_id\n LEFT JOIN vector_results v ON f.frame_id = v.frame_id\n WHERE b.frame_id IS NOT NULL OR v.frame_id IS NOT NULL\n ORDER BY score DESC\n LIMIT $5\n `,\n [textQuery, `[${embedding.join(',')}]`, textWeight, vectorWeight, 100]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Advanced aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n const client = await this.getClient();\n\n try {\n const metrics = options.metrics\n .map((m) => {\n const alias = m.alias || `${m.operation}_${m.field}`;\n return `${m.operation}(${m.field}) AS \"${alias}\"`;\n })\n .join(', ');\n\n let query = `\n SELECT ${options.groupBy.map((g) => `\"${g}\"`).join(', ')}, ${metrics}\n FROM ${table}\n GROUP BY ${options.groupBy.map((g) => `\"${g}\"`).join(', ')}\n `;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value], i) => {\n return `${key} ${typeof value === 'object' ? value.op : '='} $${i + 1}`;\n }\n );\n query += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n const result = await client.query(\n query,\n Object.values(options.having || {})\n );\n return result.rows;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Pattern detection with analytics\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n const client = await this.getClient();\n\n try {\n // Use materialized view for better performance\n const result = await client.query(\n `\n SELECT \n COALESCE(error_pattern, pattern_type) as pattern,\n pattern_type as type,\n frequency,\n last_seen\n FROM pattern_summary\n WHERE project_id = $1\n AND ($2::timestamptz IS NULL OR last_seen >= $2)\n AND ($3::timestamptz IS NULL OR first_seen <= $3)\n ORDER BY frequency DESC, last_seen DESC\n LIMIT 100\n `,\n [this.projectId, timeRange?.start || null, timeRange?.end || null]\n );\n\n return result.rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: row.last_seen,\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n await this.inTransaction(async () => {\n const client = this.activeClient!;\n\n for (const op of operations) {\n switch (op.type) {\n case 'insert': {\n const cols = Object.keys(op.data);\n const values = Object.values(op.data);\n const placeholders = values.map((_, i) => `$${i + 1}`).join(',');\n\n await client.query(\n `INSERT INTO ${op.table} (${cols.join(',')}) VALUES (${placeholders})`,\n values\n );\n break;\n }\n\n case 'update': {\n const sets = Object.keys(op.data)\n .map((k, i) => `${k} = $${i + 1}`)\n .join(',');\n const whereClause = this.buildWhereClausePostgres(\n op.where || {},\n Object.keys(op.data).length\n );\n const values = [\n ...Object.values(op.data),\n ...Object.values(op.where || {}),\n ];\n\n await client.query(\n `UPDATE ${op.table} SET ${sets} ${whereClause}`,\n values\n );\n break;\n }\n\n case 'delete': {\n const whereClause = this.buildWhereClausePostgres(\n op.where || {},\n 0\n );\n await client.query(\n `DELETE FROM ${op.table} ${whereClause}`,\n Object.values(op.where || {})\n );\n break;\n }\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('VACUUM ANALYZE frames');\n await client.query('VACUUM ANALYZE events');\n await client.query('VACUUM ANALYZE anchors');\n\n // Refresh materialized views\n await client.query(\n 'REFRESH MATERIALIZED VIEW CONCURRENTLY pattern_summary'\n );\n\n logger.info('ParadeDB vacuum and analyze completed');\n } finally {\n this.releaseClient(client);\n }\n }\n\n async analyze(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('ANALYZE frames');\n await client.query('ANALYZE events');\n await client.query('ANALYZE anchors');\n logger.info('ParadeDB analyze completed');\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(`\n SELECT\n (SELECT COUNT(*) FROM frames) as total_frames,\n (SELECT COUNT(*) FROM frames WHERE state = 'active') as active_frames,\n (SELECT COUNT(*) FROM events) as total_events,\n (SELECT COUNT(*) FROM anchors) as total_anchors,\n pg_database_size(current_database()) as disk_usage\n `);\n\n return {\n totalFrames: parseInt(result.rows[0].total_frames),\n activeFrames: parseInt(result.rows[0].active_frames),\n totalEvents: parseInt(result.rows[0].total_events),\n totalAnchors: parseInt(result.rows[0].total_anchors),\n diskUsage: parseInt(result.rows[0].disk_usage),\n };\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n const client = await this.getClient();\n\n try {\n const result = await client.query(`\n SELECT \n query,\n calls,\n mean_exec_time as mean_time,\n total_exec_time as total_time\n FROM pg_stat_statements\n WHERE query NOT LIKE '%pg_stat_statements%'\n ORDER BY total_exec_time DESC\n LIMIT 100\n `);\n\n return result.rows.map((row) => ({\n query: row.query,\n calls: parseInt(row.calls),\n meanTime: parseFloat(row.mean_time),\n totalTime: parseFloat(row.total_time),\n }));\n } catch (error) {\n logger.warn('pg_stat_statements not available', error);\n return [];\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n this.activeClient = await this.pool!.connect();\n await this.activeClient.query('BEGIN');\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.activeClient) throw new Error('No active transaction');\n\n await this.activeClient.query('COMMIT');\n this.activeClient.release();\n this.activeClient = null;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.activeClient) throw new Error('No active transaction');\n\n await this.activeClient.query('ROLLBACK');\n this.activeClient.release();\n this.activeClient = null;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error) {\n try {\n await this.rollbackTransaction();\n } catch (rollbackError) {\n // Log rollback failure but don't mask original error\n console.error('Transaction rollback failed:', rollbackError);\n // Connection might be in bad state - mark as unusable if connection pool exists\n if (this.connectionPool) {\n this.connectionPool.markConnectionAsBad(this.client);\n }\n }\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n const client = await this.getClient();\n\n try {\n if (format === 'json') {\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n const result = await client.query(`SELECT * FROM ${table}`);\n data[table] = result.rows;\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n } else if (format === 'csv') {\n // Export as CSV using COPY\n const chunks: string[] = [];\n\n for (const table of tables) {\n const result = await client.query(`\n COPY (SELECT * FROM ${table}) TO STDOUT WITH CSV HEADER\n `);\n chunks.push(result.toString());\n }\n\n return Buffer.from(chunks.join('\\n\\n'));\n } else {\n throw new Error(\n `Format ${format} not yet implemented for ParadeDB export`\n );\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n const client = await this.getClient();\n\n try {\n if (format === 'json') {\n const parsed = JSON.parse(data.toString());\n\n await client.query('BEGIN');\n\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n await client.query(`TRUNCATE TABLE ${table} CASCADE`);\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const values = Object.values(row);\n const placeholders = values.map((_, i) => `$${i + 1}`).join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = EXCLUDED.${c}`).join(',');\n await client.query(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`,\n values\n );\n } else {\n await client.query(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`,\n values\n );\n }\n }\n }\n\n await client.query('COMMIT');\n } else {\n throw new Error(\n `Format ${format} not yet implemented for ParadeDB import`\n );\n }\n } catch (error) {\n await client.query('ROLLBACK');\n throw error;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Helper methods\n private async getClient(): Promise<PoolClient> {\n if (this.activeClient) {\n return this.activeClient;\n }\n\n if (!this.pool) {\n throw new Error('Database not connected');\n }\n\n return await this.pool.connect();\n }\n\n private releaseClient(client: PoolClient): void {\n if (client !== this.activeClient) {\n client.release();\n }\n }\n\n private buildWhereClausePostgres(\n conditions: Record<string, any>,\n startParam: number\n ): string {\n const clauses = Object.entries(conditions).map(([key, value], i) => {\n const paramNum = startParam + i + 1;\n\n if (value === null) {\n return `${key} IS NULL`;\n } else if (Array.isArray(value)) {\n const placeholders = value.map((_, j) => `$${paramNum + j}`).join(',');\n return `${key} IN (${placeholders})`;\n } else {\n return `${key} = $${paramNum}`;\n }\n });\n\n return clauses.length > 0 ? `WHERE ${clauses.join(' AND ')}` : '';\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * ParadeDB Database Adapter\n * Advanced PostgreSQL with built-in search (BM25) and analytics capabilities\n */\n\nimport { Pool, PoolClient, QueryResult } from 'pg';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n} from './database-adapter.js';\nimport type { Frame, Event, Anchor } from '../context/frame-manager.js';\nimport { logger } from '../monitoring/logger.js';\nimport * as fs from 'fs/promises';\n\nexport interface ParadeDBConfig {\n connectionString?: string;\n host?: string;\n port?: number;\n database?: string;\n user?: string;\n password?: string;\n ssl?: boolean | { rejectUnauthorized?: boolean };\n max?: number; // Max pool size\n idleTimeoutMillis?: number;\n connectionTimeoutMillis?: number;\n statementTimeout?: number;\n enableBM25?: boolean;\n enableVector?: boolean;\n enableAnalytics?: boolean;\n}\n\nexport class ParadeDBAdapter extends FeatureAwareDatabaseAdapter {\n private pool: Pool | null = null;\n private activeClient: PoolClient | null = null;\n\n constructor(projectId: string, config: ParadeDBConfig) {\n super(projectId, config);\n }\n\n getFeatures(): DatabaseFeatures {\n const config = this.config as ParadeDBConfig;\n return {\n supportsFullTextSearch: config.enableBM25 !== false,\n supportsVectorSearch: config.enableVector !== false,\n supportsPartitioning: true,\n supportsAnalytics: config.enableAnalytics !== false,\n supportsCompression: true,\n supportsMaterializedViews: true,\n supportsParallelQueries: true,\n };\n }\n\n async connect(): Promise<void> {\n if (this.pool) return;\n\n const config = this.config as ParadeDBConfig;\n\n this.pool = new Pool({\n connectionString: config.connectionString,\n host: config.host || 'localhost',\n port: config.port || 5432,\n database: config.database || 'stackmemory',\n user: config.user,\n password: config.password,\n ssl: config.ssl,\n max: config.max || 20,\n idleTimeoutMillis: config.idleTimeoutMillis || 30000,\n connectionTimeoutMillis: config.connectionTimeoutMillis || 2000,\n statement_timeout: config.statementTimeout || 30000,\n });\n\n // Test connection\n const client = await this.pool.connect();\n try {\n await client.query('SELECT 1');\n logger.info('ParadeDB connected successfully');\n } finally {\n client.release();\n }\n }\n\n async disconnect(): Promise<void> {\n if (!this.pool) return;\n\n await this.pool.end();\n this.pool = null;\n logger.info('ParadeDB disconnected');\n }\n\n isConnected(): boolean {\n return this.pool !== null && !this.pool.ended;\n }\n\n async ping(): Promise<boolean> {\n if (!this.pool) return false;\n\n try {\n const client = await this.pool.connect();\n try {\n await client.query('SELECT 1');\n return true;\n } finally {\n client.release();\n }\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('BEGIN');\n\n // Enable required extensions\n await client.query(`\n CREATE EXTENSION IF NOT EXISTS \"uuid-ossp\";\n CREATE EXTENSION IF NOT EXISTS \"pg_trgm\";\n CREATE EXTENSION IF NOT EXISTS \"btree_gin\";\n `);\n\n // Enable ParadeDB extensions if configured\n const config = this.config as ParadeDBConfig;\n\n if (config.enableBM25 !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS pg_search;');\n }\n\n if (config.enableVector !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS vector;');\n }\n\n if (config.enableAnalytics !== false) {\n await client.query('CREATE EXTENSION IF NOT EXISTS pg_analytics;');\n }\n\n // Create main tables with partitioning support\n await client.query(`\n -- Main frames table\n CREATE TABLE IF NOT EXISTS frames (\n frame_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n run_id UUID NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id UUID REFERENCES frames(frame_id) ON DELETE CASCADE,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n score FLOAT DEFAULT 0.5,\n inputs JSONB DEFAULT '{}',\n outputs JSONB DEFAULT '{}',\n metadata JSONB DEFAULT '{}',\n digest_text TEXT,\n digest_json JSONB DEFAULT '{}',\n content TEXT, -- For full-text search\n embedding vector(768), -- For vector search\n created_at TIMESTAMPTZ DEFAULT NOW(),\n closed_at TIMESTAMPTZ,\n CONSTRAINT check_state CHECK (state IN ('active', 'closed', 'suspended'))\n ) PARTITION BY RANGE (created_at);\n\n -- Create partitions for time-based data\n CREATE TABLE IF NOT EXISTS frames_recent PARTITION OF frames\n FOR VALUES FROM (NOW() - INTERVAL '30 days') TO (NOW() + INTERVAL '1 day');\n \n CREATE TABLE IF NOT EXISTS frames_archive PARTITION OF frames\n FOR VALUES FROM ('2020-01-01') TO (NOW() - INTERVAL '30 days');\n\n -- Events table\n CREATE TABLE IF NOT EXISTS events (\n event_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n run_id UUID NOT NULL,\n frame_id UUID NOT NULL REFERENCES frames(frame_id) ON DELETE CASCADE,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload JSONB NOT NULL DEFAULT '{}',\n ts TIMESTAMPTZ DEFAULT NOW()\n );\n\n -- Anchors table\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),\n frame_id UUID NOT NULL REFERENCES frames(frame_id) ON DELETE CASCADE,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n metadata JSONB DEFAULT '{}',\n created_at TIMESTAMPTZ DEFAULT NOW()\n );\n\n -- Schema version tracking\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at TIMESTAMPTZ DEFAULT NOW(),\n description TEXT\n );\n `);\n\n // Create indexes for performance\n await client.query(`\n -- Standard B-tree indexes\n CREATE INDEX IF NOT EXISTS idx_frames_run_id ON frames USING btree(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project_id ON frames USING btree(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames USING btree(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames USING btree(state);\n CREATE INDEX IF NOT EXISTS idx_frames_type ON frames USING btree(type);\n CREATE INDEX IF NOT EXISTS idx_frames_created_at ON frames USING btree(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_frames_score ON frames USING btree(score DESC);\n\n -- GIN indexes for JSONB\n CREATE INDEX IF NOT EXISTS idx_frames_inputs ON frames USING gin(inputs);\n CREATE INDEX IF NOT EXISTS idx_frames_outputs ON frames USING gin(outputs);\n CREATE INDEX IF NOT EXISTS idx_frames_metadata ON frames USING gin(metadata);\n CREATE INDEX IF NOT EXISTS idx_frames_digest ON frames USING gin(digest_json);\n\n -- Trigram index for fuzzy text search\n CREATE INDEX IF NOT EXISTS idx_frames_name_trgm ON frames USING gin(name gin_trgm_ops);\n CREATE INDEX IF NOT EXISTS idx_frames_content_trgm ON frames USING gin(content gin_trgm_ops);\n\n -- Event indexes\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events USING btree(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events USING btree(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_events_type ON events USING btree(event_type);\n CREATE INDEX IF NOT EXISTS idx_events_ts ON events USING btree(ts DESC);\n\n -- Anchor indexes\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors USING btree(frame_id);\n CREATE INDEX IF NOT EXISTS idx_anchors_type ON anchors USING btree(type);\n CREATE INDEX IF NOT EXISTS idx_anchors_priority ON anchors USING btree(priority DESC);\n `);\n\n // Create BM25 search index if enabled\n if (config.enableBM25 !== false) {\n await client.query(`\n -- Create BM25 index for full-text search\n CALL paradedb.create_bm25_test_table(\n index_name => 'frames_search_idx',\n table_name => 'frames',\n schema_name => 'public',\n key_field => 'frame_id',\n text_fields => paradedb.field('name') || \n paradedb.field('content') || \n paradedb.field('digest_text'),\n numeric_fields => paradedb.field('score') || \n paradedb.field('depth'),\n json_fields => paradedb.field('metadata', flatten => true),\n datetime_fields => paradedb.field('created_at')\n );\n `);\n }\n\n // Create vector index if enabled\n if (config.enableVector !== false) {\n await client.query(`\n -- HNSW index for vector similarity search\n CREATE INDEX IF NOT EXISTS idx_frames_embedding \n ON frames USING hnsw (embedding vector_cosine_ops)\n WITH (m = 16, ef_construction = 64);\n `);\n }\n\n // Create materialized views for patterns\n await client.query(`\n CREATE MATERIALIZED VIEW IF NOT EXISTS pattern_summary AS\n WITH pattern_extraction AS (\n SELECT \n project_id,\n type as pattern_type,\n metadata->>'error' as error_pattern,\n COUNT(*) as frequency,\n MAX(score) as max_score,\n MAX(created_at) as last_seen,\n MIN(created_at) as first_seen\n FROM frames\n WHERE created_at > NOW() - INTERVAL '30 days'\n GROUP BY project_id, pattern_type, error_pattern\n )\n SELECT * FROM pattern_extraction\n WHERE frequency > 3;\n\n CREATE UNIQUE INDEX IF NOT EXISTS idx_pattern_summary_unique \n ON pattern_summary(project_id, pattern_type, error_pattern);\n `);\n\n // Set initial schema version\n await client.query(`\n INSERT INTO schema_version (version, description) \n VALUES (1, 'Initial ParadeDB schema with search and analytics')\n ON CONFLICT (version) DO NOTHING;\n `);\n\n await client.query('COMMIT');\n logger.info('ParadeDB schema initialized successfully');\n } catch (error: unknown) {\n await client.query('ROLLBACK');\n throw error;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT MAX(version) as version FROM schema_version'\n );\n const currentVersion = result.rows[0]?.version || 0;\n\n if (currentVersion >= targetVersion) {\n logger.info('Schema already at target version', {\n currentVersion,\n targetVersion,\n });\n return;\n }\n\n // Apply migrations sequentially\n for (let v = currentVersion + 1; v <= targetVersion; v++) {\n logger.info(`Applying migration to version ${v}`);\n // Migration logic would go here based on version\n await client.query(\n 'INSERT INTO schema_version (version, description) VALUES ($1, $2)',\n [v, `Migration to version ${v}`]\n );\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT MAX(version) as version FROM schema_version'\n );\n return result.rows[0]?.version || 0;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, score, inputs, outputs, metadata,\n digest_text, digest_json, content\n ) VALUES (\n COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5,\n $6, $7, $8, $9, $10, $11, $12, $13, $14, $15\n ) RETURNING frame_id\n `,\n [\n frame.frame_id || null,\n frame.run_id,\n frame.project_id || this.projectId,\n frame.parent_frame_id || null,\n frame.depth || 0,\n frame.type,\n frame.name,\n frame.state || 'active',\n frame.score || 0.5,\n JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n JSON.stringify(frame.metadata || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {}),\n frame.content || `${frame.name} ${frame.digest_text || ''}`,\n ]\n );\n\n return result.rows[0].frame_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n 'SELECT * FROM frames WHERE frame_id = $1',\n [frameId]\n );\n\n if (result.rows.length === 0) return null;\n\n const row = result.rows[0];\n return {\n ...row,\n frame_id: row.frame_id,\n run_id: row.run_id,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n };\n } finally {\n this.releaseClient(client);\n }\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n const client = await this.getClient();\n\n try {\n const fields = [];\n const values = [];\n let paramCount = 1;\n\n if (updates.state !== undefined) {\n fields.push(`state = $${paramCount++}`);\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push(`outputs = $${paramCount++}`);\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.score !== undefined) {\n fields.push(`score = $${paramCount++}`);\n values.push(updates.score);\n }\n\n if (updates.digest_text !== undefined) {\n fields.push(`digest_text = $${paramCount++}`);\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push(`digest_json = $${paramCount++}`);\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push(`closed_at = $${paramCount++}`);\n values.push(new Date(updates.closed_at));\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n await client.query(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = $${paramCount}\n `,\n values\n );\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n // CASCADE delete handles events and anchors\n await client.query('DELETE FROM frames WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n const client = await this.getClient();\n\n try {\n let query = 'SELECT * FROM frames WHERE state = $1';\n const params: any[] = ['active'];\n\n if (runId) {\n query += ' AND run_id = $2';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const result = await client.query(query, params);\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async closeFrame(frameId: string, outputs?: any): Promise<void> {\n await this.updateFrame(frameId, {\n state: 'closed',\n outputs,\n closed_at: Date.now(),\n });\n }\n\n // Event operations\n async createEvent(event: Partial<Event>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5, $6, $7)\n RETURNING event_id\n `,\n [\n event.event_id || null,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts ? new Date(event.ts) : new Date(),\n ]\n );\n\n return result.rows[0].event_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n const client = await this.getClient();\n\n try {\n let query = 'SELECT * FROM events WHERE frame_id = $1';\n const params: any[] = [frameId];\n\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const result = await client.query(query, params);\n\n return result.rows.map((row) => ({\n ...row,\n ts: row.ts.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('DELETE FROM events WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (COALESCE($1::uuid, uuid_generate_v4()), $2, $3, $4, $5, $6, $7)\n RETURNING anchor_id\n `,\n [\n anchor.anchor_id || null,\n anchor.frame_id,\n anchor.project_id || this.projectId,\n anchor.type,\n anchor.text,\n anchor.priority || 0,\n JSON.stringify(anchor.metadata || {}),\n ]\n );\n\n return result.rows[0].anchor_id;\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(\n `\n SELECT * FROM anchors WHERE frame_id = $1 \n ORDER BY priority DESC, created_at ASC\n `,\n [frameId]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('DELETE FROM anchors WHERE frame_id = $1', [frameId]);\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Advanced search with BM25\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n const client = await this.getClient();\n\n try {\n const config = this.config as ParadeDBConfig;\n\n if (config.enableBM25 !== false) {\n // Use ParadeDB BM25 search\n const result = await client.query(\n `\n SELECT f.*, s.score_bm25 as score\n FROM frames_search_idx.search(\n query => $1,\n limit_rows => $2,\n offset_rows => $3\n ) s\n JOIN frames f ON f.frame_id = s.frame_id\n WHERE ($4::float IS NULL OR s.score_bm25 >= $4)\n ORDER BY s.score_bm25 DESC\n `,\n [\n options.query,\n options.limit || 100,\n options.offset || 0,\n options.scoreThreshold || null,\n ]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } else {\n // Fallback to PostgreSQL full-text search\n const result = await client.query(\n `\n SELECT *,\n ts_rank(\n to_tsvector('english', COALESCE(name, '') || ' ' || COALESCE(content, '')),\n plainto_tsquery('english', $1)\n ) as score\n FROM frames\n WHERE to_tsvector('english', COALESCE(name, '') || ' ' || COALESCE(content, ''))\n @@ plainto_tsquery('english', $1)\n ORDER BY score DESC\n LIMIT $2 OFFSET $3\n `,\n [options.query, options.limit || 100, options.offset || 0]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Vector similarity search\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n const client = await this.getClient();\n\n try {\n const config = this.config as ParadeDBConfig;\n\n if (config.enableVector === false) {\n logger.warn('Vector search not enabled in ParadeDB configuration');\n return [];\n }\n\n const result = await client.query(\n `\n SELECT *,\n 1 - (embedding <=> $1::vector) as similarity\n FROM frames\n WHERE embedding IS NOT NULL\n ORDER BY embedding <=> $1::vector\n LIMIT $2 OFFSET $3\n `,\n [\n `[${embedding.join(',')}]`,\n options?.limit || 100,\n options?.offset || 0,\n ]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Hybrid search combining BM25 and vector\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n const client = await this.getClient();\n\n try {\n const textWeight = weights?.text || 0.6;\n const vectorWeight = weights?.vector || 0.4;\n\n const result = await client.query(\n `\n WITH bm25_results AS (\n SELECT frame_id, score_bm25\n FROM frames_search_idx.search(\n query => $1,\n limit_rows => 200\n )\n ),\n vector_results AS (\n SELECT frame_id,\n 1 - (embedding <=> $2::vector) as score_vector\n FROM frames\n WHERE embedding IS NOT NULL\n ORDER BY embedding <=> $2::vector\n LIMIT 200\n )\n SELECT f.*,\n (COALESCE(b.score_bm25, 0) * $3 + \n COALESCE(v.score_vector, 0) * $4) as score\n FROM frames f\n LEFT JOIN bm25_results b ON f.frame_id = b.frame_id\n LEFT JOIN vector_results v ON f.frame_id = v.frame_id\n WHERE b.frame_id IS NOT NULL OR v.frame_id IS NOT NULL\n ORDER BY score DESC\n LIMIT $5\n `,\n [textQuery, `[${embedding.join(',')}]`, textWeight, vectorWeight, 100]\n );\n\n return result.rows.map((row) => ({\n ...row,\n created_at: row.created_at.getTime(),\n closed_at: row.closed_at?.getTime(),\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Advanced aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n const client = await this.getClient();\n\n try {\n const metrics = options.metrics\n .map((m) => {\n const alias = m.alias || `${m.operation}_${m.field}`;\n return `${m.operation}(${m.field}) AS \"${alias}\"`;\n })\n .join(', ');\n\n let query = `\n SELECT ${options.groupBy.map((g) => `\"${g}\"`).join(', ')}, ${metrics}\n FROM ${table}\n GROUP BY ${options.groupBy.map((g) => `\"${g}\"`).join(', ')}\n `;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value], i) => {\n return `${key} ${typeof value === 'object' ? value.op : '='} $${i + 1}`;\n }\n );\n query += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n const result = await client.query(\n query,\n Object.values(options.having || {})\n );\n return result.rows;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Pattern detection with analytics\n async detectPatterns(timeRange?: { start: Date; end: Date }): Promise<\n Array<{\n pattern: string;\n type: string;\n frequency: number;\n lastSeen: Date;\n }>\n > {\n const client = await this.getClient();\n\n try {\n // Use materialized view for better performance\n const result = await client.query(\n `\n SELECT \n COALESCE(error_pattern, pattern_type) as pattern,\n pattern_type as type,\n frequency,\n last_seen\n FROM pattern_summary\n WHERE project_id = $1\n AND ($2::timestamptz IS NULL OR last_seen >= $2)\n AND ($3::timestamptz IS NULL OR first_seen <= $3)\n ORDER BY frequency DESC, last_seen DESC\n LIMIT 100\n `,\n [this.projectId, timeRange?.start || null, timeRange?.end || null]\n );\n\n return result.rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: row.last_seen,\n }));\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n await this.inTransaction(async () => {\n const client = this.activeClient!;\n\n for (const op of operations) {\n switch (op.type) {\n case 'insert': {\n const cols = Object.keys(op.data);\n const values = Object.values(op.data);\n const placeholders = values.map((_, i) => `$${i + 1}`).join(',');\n\n await client.query(\n `INSERT INTO ${op.table} (${cols.join(',')}) VALUES (${placeholders})`,\n values\n );\n break;\n }\n\n case 'update': {\n const sets = Object.keys(op.data)\n .map((k, i) => `${k} = $${i + 1}`)\n .join(',');\n const whereClause = this.buildWhereClausePostgres(\n op.where || {},\n Object.keys(op.data).length\n );\n const values = [\n ...Object.values(op.data),\n ...Object.values(op.where || {}),\n ];\n\n await client.query(\n `UPDATE ${op.table} SET ${sets} ${whereClause}`,\n values\n );\n break;\n }\n\n case 'delete': {\n const whereClause = this.buildWhereClausePostgres(\n op.where || {},\n 0\n );\n await client.query(\n `DELETE FROM ${op.table} ${whereClause}`,\n Object.values(op.where || {})\n );\n break;\n }\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('VACUUM ANALYZE frames');\n await client.query('VACUUM ANALYZE events');\n await client.query('VACUUM ANALYZE anchors');\n\n // Refresh materialized views\n await client.query(\n 'REFRESH MATERIALIZED VIEW CONCURRENTLY pattern_summary'\n );\n\n logger.info('ParadeDB vacuum and analyze completed');\n } finally {\n this.releaseClient(client);\n }\n }\n\n async analyze(): Promise<void> {\n const client = await this.getClient();\n\n try {\n await client.query('ANALYZE frames');\n await client.query('ANALYZE events');\n await client.query('ANALYZE anchors');\n logger.info('ParadeDB analyze completed');\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n const client = await this.getClient();\n\n try {\n const result = await client.query(`\n SELECT\n (SELECT COUNT(*) FROM frames) as total_frames,\n (SELECT COUNT(*) FROM frames WHERE state = 'active') as active_frames,\n (SELECT COUNT(*) FROM events) as total_events,\n (SELECT COUNT(*) FROM anchors) as total_anchors,\n pg_database_size(current_database()) as disk_usage\n `);\n\n return {\n totalFrames: parseInt(result.rows[0].total_frames),\n activeFrames: parseInt(result.rows[0].active_frames),\n totalEvents: parseInt(result.rows[0].total_events),\n totalAnchors: parseInt(result.rows[0].total_anchors),\n diskUsage: parseInt(result.rows[0].disk_usage),\n };\n } finally {\n this.releaseClient(client);\n }\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n const client = await this.getClient();\n\n try {\n const result = await client.query(`\n SELECT \n query,\n calls,\n mean_exec_time as mean_time,\n total_exec_time as total_time\n FROM pg_stat_statements\n WHERE query NOT LIKE '%pg_stat_statements%'\n ORDER BY total_exec_time DESC\n LIMIT 100\n `);\n\n return result.rows.map((row) => ({\n query: row.query,\n calls: parseInt(row.calls),\n meanTime: parseFloat(row.mean_time),\n totalTime: parseFloat(row.total_time),\n }));\n } catch (error: unknown) {\n logger.warn('pg_stat_statements not available', error);\n return [];\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n this.activeClient = await this.pool!.connect();\n await this.activeClient.query('BEGIN');\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.activeClient) throw new Error('No active transaction');\n\n await this.activeClient.query('COMMIT');\n this.activeClient.release();\n this.activeClient = null;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.activeClient) throw new Error('No active transaction');\n\n await this.activeClient.query('ROLLBACK');\n this.activeClient.release();\n this.activeClient = null;\n }\n\n async inTransaction(\n callback: (adapter: DatabaseAdapter) => Promise<void>\n ): Promise<void> {\n await this.beginTransaction();\n\n try {\n await callback(this);\n await this.commitTransaction();\n } catch (error: unknown) {\n try {\n await this.rollbackTransaction();\n } catch (rollbackError: unknown) {\n // Log rollback failure but don't mask original error\n console.error('Transaction rollback failed:', rollbackError);\n // Connection might be in bad state - mark as unusable if connection pool exists\n if (this.connectionPool) {\n this.connectionPool.markConnectionAsBad(this.client);\n }\n }\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n const client = await this.getClient();\n\n try {\n if (format === 'json') {\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n const result = await client.query(`SELECT * FROM ${table}`);\n data[table] = result.rows;\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n } else if (format === 'csv') {\n // Export as CSV using COPY\n const chunks: string[] = [];\n\n for (const table of tables) {\n const result = await client.query(`\n COPY (SELECT * FROM ${table}) TO STDOUT WITH CSV HEADER\n `);\n chunks.push(result.toString());\n }\n\n return Buffer.from(chunks.join('\\n\\n'));\n } else {\n throw new Error(\n `Format ${format} not yet implemented for ParadeDB export`\n );\n }\n } finally {\n this.releaseClient(client);\n }\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n const client = await this.getClient();\n\n try {\n if (format === 'json') {\n const parsed = JSON.parse(data.toString());\n\n await client.query('BEGIN');\n\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n await client.query(`TRUNCATE TABLE ${table} CASCADE`);\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const values = Object.values(row);\n const placeholders = values.map((_, i) => `$${i + 1}`).join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = EXCLUDED.${c}`).join(',');\n await client.query(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`,\n values\n );\n } else {\n await client.query(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`,\n values\n );\n }\n }\n }\n\n await client.query('COMMIT');\n } else {\n throw new Error(\n `Format ${format} not yet implemented for ParadeDB import`\n );\n }\n } catch (error: unknown) {\n await client.query('ROLLBACK');\n throw error;\n } finally {\n this.releaseClient(client);\n }\n }\n\n // Helper methods\n private async getClient(): Promise<PoolClient> {\n if (this.activeClient) {\n return this.activeClient;\n }\n\n if (!this.pool) {\n throw new Error('Database not connected');\n }\n\n return await this.pool.connect();\n }\n\n private releaseClient(client: PoolClient): void {\n if (client !== this.activeClient) {\n client.release();\n }\n }\n\n private buildWhereClausePostgres(\n conditions: Record<string, any>,\n startParam: number\n ): string {\n const clauses = Object.entries(conditions).map(([key, value], i) => {\n const paramNum = startParam + i + 1;\n\n if (value === null) {\n return `${key} IS NULL`;\n } else if (Array.isArray(value)) {\n const placeholders = value.map((_, j) => `$${paramNum + j}`).join(',');\n return `${key} IN (${placeholders})`;\n } else {\n return `${key} = $${paramNum}`;\n }\n });\n\n return clauses.length > 0 ? `WHERE ${clauses.join(' AND ')}` : '';\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/database/query-cache.ts"],
4
- "sourcesContent": ["/**\n * Query Result Cache\n * LRU cache for frequently accessed database query results\n */\n\nimport { logger } from '../monitoring/logger.js';\n\nexport interface CacheOptions {\n maxSize?: number;\n ttlMs?: number;\n enableMetrics?: boolean;\n}\n\nexport interface CacheEntry<T> {\n value: T;\n createdAt: number;\n accessCount: number;\n lastAccessed: number;\n}\n\nexport interface CacheMetrics {\n hits: number;\n misses: number;\n evictions: number;\n totalQueries: number;\n hitRate: number;\n size: number;\n maxSize: number;\n}\n\nexport class LRUQueryCache<T = any> {\n private cache = new Map<string, CacheEntry<T>>();\n private maxSize: number;\n private ttlMs: number;\n private enableMetrics: boolean;\n \n // Metrics\n private metrics: Omit<CacheMetrics, 'hitRate' | 'size' | 'maxSize'> = {\n hits: 0,\n misses: 0,\n evictions: 0,\n totalQueries: 0,\n };\n\n constructor(options: CacheOptions = {}) {\n this.maxSize = options.maxSize ?? 1000;\n this.ttlMs = options.ttlMs ?? 300000; // 5 minutes default\n this.enableMetrics = options.enableMetrics ?? true;\n\n logger.info('Query cache initialized', {\n maxSize: this.maxSize,\n ttlMs: this.ttlMs,\n enableMetrics: this.enableMetrics,\n });\n }\n\n /**\n * Get a value from cache\n */\n get(key: string): T | undefined {\n if (this.enableMetrics) {\n this.metrics.totalQueries++;\n }\n\n const entry = this.cache.get(key);\n \n if (!entry) {\n if (this.enableMetrics) {\n this.metrics.misses++;\n }\n return undefined;\n }\n\n // Check TTL\n const now = Date.now();\n if (now - entry.createdAt > this.ttlMs) {\n this.cache.delete(key);\n if (this.enableMetrics) {\n this.metrics.misses++;\n this.metrics.evictions++;\n }\n logger.debug('Cache entry expired', { key, age: now - entry.createdAt });\n return undefined;\n }\n\n // Update access stats and move to end (most recently used)\n entry.accessCount++;\n entry.lastAccessed = now;\n this.cache.delete(key);\n this.cache.set(key, entry);\n\n if (this.enableMetrics) {\n this.metrics.hits++;\n }\n\n logger.debug('Cache hit', { key, accessCount: entry.accessCount });\n return entry.value;\n }\n\n /**\n * Set a value in cache\n */\n set(key: string, value: T): void {\n const now = Date.now();\n \n // Remove existing entry if present\n if (this.cache.has(key)) {\n this.cache.delete(key);\n }\n \n // Evict least recently used entries if at capacity\n while (this.cache.size >= this.maxSize) {\n const firstKey = this.cache.keys().next().value;\n if (firstKey !== undefined) {\n this.cache.delete(firstKey);\n if (this.enableMetrics) {\n this.metrics.evictions++;\n }\n logger.debug('Evicted LRU entry', { key: firstKey });\n } else {\n break;\n }\n }\n\n // Add new entry\n const entry: CacheEntry<T> = {\n value,\n createdAt: now,\n accessCount: 0,\n lastAccessed: now,\n };\n\n this.cache.set(key, entry);\n logger.debug('Cache set', { key, size: this.cache.size });\n }\n\n /**\n * Delete a specific key\n */\n delete(key: string): boolean {\n const deleted = this.cache.delete(key);\n if (deleted) {\n logger.debug('Cache delete', { key });\n }\n return deleted;\n }\n\n /**\n * Clear all cached entries\n */\n clear(): void {\n const size = this.cache.size;\n this.cache.clear();\n logger.info('Cache cleared', { previousSize: size });\n }\n\n /**\n * Invalidate entries matching a pattern\n */\n invalidatePattern(pattern: RegExp): number {\n let count = 0;\n for (const key of this.cache.keys()) {\n if (pattern.test(key)) {\n this.cache.delete(key);\n count++;\n }\n }\n logger.info('Pattern invalidation', { pattern: pattern.source, invalidated: count });\n return count;\n }\n\n /**\n * Get cache metrics\n */\n getMetrics(): CacheMetrics {\n return {\n ...this.metrics,\n hitRate: this.metrics.totalQueries > 0 \n ? this.metrics.hits / this.metrics.totalQueries \n : 0,\n size: this.cache.size,\n maxSize: this.maxSize,\n };\n }\n\n /**\n * Get cache contents for debugging\n */\n debug(): Array<{ key: string; entry: CacheEntry<T> }> {\n return Array.from(this.cache.entries()).map(([key, entry]) => ({ key, entry }));\n }\n\n /**\n * Cleanup expired entries\n */\n cleanup(): number {\n const now = Date.now();\n let removed = 0;\n\n for (const [key, entry] of this.cache.entries()) {\n if (now - entry.createdAt > this.ttlMs) {\n this.cache.delete(key);\n removed++;\n if (this.enableMetrics) {\n this.metrics.evictions++;\n }\n }\n }\n\n if (removed > 0) {\n logger.info('Cache cleanup completed', { removed, remaining: this.cache.size });\n }\n\n return removed;\n }\n}\n\n/**\n * Query cache specifically designed for StackMemory operations\n */\nexport class StackMemoryQueryCache {\n private frameCache = new LRUQueryCache<any>({ maxSize: 500, ttlMs: 300000 }); // 5 min\n private eventCache = new LRUQueryCache<any>({ maxSize: 1000, ttlMs: 180000 }); // 3 min \n private anchorCache = new LRUQueryCache<any>({ maxSize: 200, ttlMs: 600000 }); // 10 min\n private digestCache = new LRUQueryCache<any>({ maxSize: 100, ttlMs: 900000 }); // 15 min\n\n /**\n * Cache frame data\n */\n cacheFrame(frameId: string, data: any): void {\n this.frameCache.set(`frame:${frameId}`, data);\n }\n\n getFrame(frameId: string): any {\n return this.frameCache.get(`frame:${frameId}`);\n }\n\n /**\n * Cache frame context assemblies (expensive operations)\n */\n cacheFrameContext(frameId: string, context: any): void {\n this.frameCache.set(`context:${frameId}`, context);\n }\n\n getFrameContext(frameId: string): any {\n return this.frameCache.get(`context:${frameId}`);\n }\n\n /**\n * Cache events for a frame\n */\n cacheFrameEvents(frameId: string, events: any[]): void {\n this.eventCache.set(`events:${frameId}`, events);\n }\n\n getFrameEvents(frameId: string): any[] {\n return this.eventCache.get(`events:${frameId}`);\n }\n\n /**\n * Cache anchors\n */\n cacheAnchors(frameId: string, anchors: any[]): void {\n this.anchorCache.set(`anchors:${frameId}`, anchors);\n }\n\n getAnchors(frameId: string): any[] {\n return this.anchorCache.get(`anchors:${frameId}`);\n }\n\n /**\n * Cache digest data\n */\n cacheDigest(frameId: string, digest: any): void {\n this.digestCache.set(`digest:${frameId}`, digest);\n }\n\n getDigest(frameId: string): any {\n return this.digestCache.get(`digest:${frameId}`);\n }\n\n /**\n * Invalidate caches for a specific frame\n */\n invalidateFrame(frameId: string): void {\n this.frameCache.delete(`frame:${frameId}`);\n this.frameCache.delete(`context:${frameId}`);\n this.eventCache.delete(`events:${frameId}`);\n this.anchorCache.delete(`anchors:${frameId}`);\n this.digestCache.delete(`digest:${frameId}`);\n \n logger.info('Invalidated frame caches', { frameId });\n }\n\n /**\n * Invalidate all caches for a project \n */\n invalidateProject(projectId: string): void {\n const pattern = new RegExp(`^(frame|context|events|anchors|digest):.+`);\n \n let total = 0;\n total += this.frameCache.invalidatePattern(pattern);\n total += this.eventCache.invalidatePattern(pattern);\n total += this.anchorCache.invalidatePattern(pattern);\n total += this.digestCache.invalidatePattern(pattern);\n \n logger.info('Invalidated project caches', { projectId, totalInvalidated: total });\n }\n\n /**\n * Get comprehensive cache metrics\n */\n getMetrics() {\n return {\n frame: this.frameCache.getMetrics(),\n event: this.eventCache.getMetrics(),\n anchor: this.anchorCache.getMetrics(),\n digest: this.digestCache.getMetrics(),\n };\n }\n\n /**\n * Cleanup all caches\n */\n cleanup(): void {\n this.frameCache.cleanup();\n this.eventCache.cleanup();\n this.anchorCache.cleanup();\n this.digestCache.cleanup();\n }\n\n /**\n * Clear all caches\n */\n clear(): void {\n this.frameCache.clear();\n this.eventCache.clear();\n this.anchorCache.clear();\n this.digestCache.clear();\n \n logger.info('All StackMemory caches cleared');\n }\n}\n\n// Global cache instance\nlet globalCache: StackMemoryQueryCache | null = null;\n\n/**\n * Get or create global query cache\n */\nexport function getQueryCache(): StackMemoryQueryCache {\n if (!globalCache) {\n globalCache = new StackMemoryQueryCache();\n }\n return globalCache;\n}\n\n/**\n * Create a cache key from query parameters\n */\nexport function createCacheKey(queryName: string, params: any[]): string {\n const paramsStr = params.map(p => \n typeof p === 'object' ? JSON.stringify(p) : String(p)\n ).join(':');\n \n return `${queryName}:${paramsStr}`;\n}"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Query Result Cache\n * LRU cache for frequently accessed database query results\n */\n\nimport { logger } from '../monitoring/logger.js';\n\nexport interface CacheOptions {\n maxSize?: number;\n ttlMs?: number;\n enableMetrics?: boolean;\n}\n\nexport interface CacheEntry<T> {\n value: T;\n createdAt: number;\n accessCount: number;\n lastAccessed: number;\n}\n\nexport interface CacheMetrics {\n hits: number;\n misses: number;\n evictions: number;\n totalQueries: number;\n hitRate: number;\n size: number;\n maxSize: number;\n}\n\nexport class LRUQueryCache<T = any> {\n private cache = new Map<string, CacheEntry<T>>();\n private maxSize: number;\n private ttlMs: number;\n private enableMetrics: boolean;\n \n // Metrics\n private metrics: Omit<CacheMetrics, 'hitRate' | 'size' | 'maxSize'> = {\n hits: 0,\n misses: 0,\n evictions: 0,\n totalQueries: 0,\n };\n\n constructor(options: CacheOptions = {}) {\n this.maxSize = options.maxSize ?? 1000;\n this.ttlMs = options.ttlMs ?? 300000; // 5 minutes default\n this.enableMetrics = options.enableMetrics ?? true;\n\n logger.info('Query cache initialized', {\n maxSize: this.maxSize,\n ttlMs: this.ttlMs,\n enableMetrics: this.enableMetrics,\n });\n }\n\n /**\n * Get a value from cache\n */\n get(key: string): T | undefined {\n if (this.enableMetrics) {\n this.metrics.totalQueries++;\n }\n\n const entry = this.cache.get(key);\n \n if (!entry) {\n if (this.enableMetrics) {\n this.metrics.misses++;\n }\n return undefined;\n }\n\n // Check TTL\n const now = Date.now();\n if (now - entry.createdAt > this.ttlMs) {\n this.cache.delete(key);\n if (this.enableMetrics) {\n this.metrics.misses++;\n this.metrics.evictions++;\n }\n logger.debug('Cache entry expired', { key, age: now - entry.createdAt });\n return undefined;\n }\n\n // Update access stats and move to end (most recently used)\n entry.accessCount++;\n entry.lastAccessed = now;\n this.cache.delete(key);\n this.cache.set(key, entry);\n\n if (this.enableMetrics) {\n this.metrics.hits++;\n }\n\n logger.debug('Cache hit', { key, accessCount: entry.accessCount });\n return entry.value;\n }\n\n /**\n * Set a value in cache\n */\n set(key: string, value: T): void {\n const now = Date.now();\n \n // Remove existing entry if present\n if (this.cache.has(key)) {\n this.cache.delete(key);\n }\n \n // Evict least recently used entries if at capacity\n while (this.cache.size >= this.maxSize) {\n const firstKey = this.cache.keys().next().value;\n if (firstKey !== undefined) {\n this.cache.delete(firstKey);\n if (this.enableMetrics) {\n this.metrics.evictions++;\n }\n logger.debug('Evicted LRU entry', { key: firstKey });\n } else {\n break;\n }\n }\n\n // Add new entry\n const entry: CacheEntry<T> = {\n value,\n createdAt: now,\n accessCount: 0,\n lastAccessed: now,\n };\n\n this.cache.set(key, entry);\n logger.debug('Cache set', { key, size: this.cache.size });\n }\n\n /**\n * Delete a specific key\n */\n delete(key: string): boolean {\n const deleted = this.cache.delete(key);\n if (deleted) {\n logger.debug('Cache delete', { key });\n }\n return deleted;\n }\n\n /**\n * Clear all cached entries\n */\n clear(): void {\n const size = this.cache.size;\n this.cache.clear();\n logger.info('Cache cleared', { previousSize: size });\n }\n\n /**\n * Invalidate entries matching a pattern\n */\n invalidatePattern(pattern: RegExp): number {\n let count = 0;\n for (const key of this.cache.keys()) {\n if (pattern.test(key)) {\n this.cache.delete(key);\n count++;\n }\n }\n logger.info('Pattern invalidation', { pattern: pattern.source, invalidated: count });\n return count;\n }\n\n /**\n * Get cache metrics\n */\n getMetrics(): CacheMetrics {\n return {\n ...this.metrics,\n hitRate: this.metrics.totalQueries > 0 \n ? this.metrics.hits / this.metrics.totalQueries \n : 0,\n size: this.cache.size,\n maxSize: this.maxSize,\n };\n }\n\n /**\n * Get cache contents for debugging\n */\n debug(): Array<{ key: string; entry: CacheEntry<T> }> {\n return Array.from(this.cache.entries()).map(([key, entry]) => ({ key, entry }));\n }\n\n /**\n * Cleanup expired entries\n */\n cleanup(): number {\n const now = Date.now();\n let removed = 0;\n\n for (const [key, entry] of this.cache.entries()) {\n if (now - entry.createdAt > this.ttlMs) {\n this.cache.delete(key);\n removed++;\n if (this.enableMetrics) {\n this.metrics.evictions++;\n }\n }\n }\n\n if (removed > 0) {\n logger.info('Cache cleanup completed', { removed, remaining: this.cache.size });\n }\n\n return removed;\n }\n}\n\n/**\n * Query cache specifically designed for StackMemory operations\n */\nexport class StackMemoryQueryCache {\n private frameCache = new LRUQueryCache<any>({ maxSize: 500, ttlMs: 300000 }); // 5 min\n private eventCache = new LRUQueryCache<any>({ maxSize: 1000, ttlMs: 180000 }); // 3 min \n private anchorCache = new LRUQueryCache<any>({ maxSize: 200, ttlMs: 600000 }); // 10 min\n private digestCache = new LRUQueryCache<any>({ maxSize: 100, ttlMs: 900000 }); // 15 min\n\n /**\n * Cache frame data\n */\n cacheFrame(frameId: string, data: any): void {\n this.frameCache.set(`frame:${frameId}`, data);\n }\n\n getFrame(frameId: string): any {\n return this.frameCache.get(`frame:${frameId}`);\n }\n\n /**\n * Cache frame context assemblies (expensive operations)\n */\n cacheFrameContext(frameId: string, context: any): void {\n this.frameCache.set(`context:${frameId}`, context);\n }\n\n getFrameContext(frameId: string): any {\n return this.frameCache.get(`context:${frameId}`);\n }\n\n /**\n * Cache events for a frame\n */\n cacheFrameEvents(frameId: string, events: any[]): void {\n this.eventCache.set(`events:${frameId}`, events);\n }\n\n getFrameEvents(frameId: string): any[] {\n return this.eventCache.get(`events:${frameId}`);\n }\n\n /**\n * Cache anchors\n */\n cacheAnchors(frameId: string, anchors: any[]): void {\n this.anchorCache.set(`anchors:${frameId}`, anchors);\n }\n\n getAnchors(frameId: string): any[] {\n return this.anchorCache.get(`anchors:${frameId}`);\n }\n\n /**\n * Cache digest data\n */\n cacheDigest(frameId: string, digest: any): void {\n this.digestCache.set(`digest:${frameId}`, digest);\n }\n\n getDigest(frameId: string): any {\n return this.digestCache.get(`digest:${frameId}`);\n }\n\n /**\n * Invalidate caches for a specific frame\n */\n invalidateFrame(frameId: string): void {\n this.frameCache.delete(`frame:${frameId}`);\n this.frameCache.delete(`context:${frameId}`);\n this.eventCache.delete(`events:${frameId}`);\n this.anchorCache.delete(`anchors:${frameId}`);\n this.digestCache.delete(`digest:${frameId}`);\n \n logger.info('Invalidated frame caches', { frameId });\n }\n\n /**\n * Invalidate all caches for a project \n */\n invalidateProject(projectId: string): void {\n const pattern = new RegExp(`^(frame|context|events|anchors|digest):.+`);\n \n let total = 0;\n total += this.frameCache.invalidatePattern(pattern);\n total += this.eventCache.invalidatePattern(pattern);\n total += this.anchorCache.invalidatePattern(pattern);\n total += this.digestCache.invalidatePattern(pattern);\n \n logger.info('Invalidated project caches', { projectId, totalInvalidated: total });\n }\n\n /**\n * Get comprehensive cache metrics\n */\n getMetrics() {\n return {\n frame: this.frameCache.getMetrics(),\n event: this.eventCache.getMetrics(),\n anchor: this.anchorCache.getMetrics(),\n digest: this.digestCache.getMetrics(),\n };\n }\n\n /**\n * Cleanup all caches\n */\n cleanup(): void {\n this.frameCache.cleanup();\n this.eventCache.cleanup();\n this.anchorCache.cleanup();\n this.digestCache.cleanup();\n }\n\n /**\n * Clear all caches\n */\n clear(): void {\n this.frameCache.clear();\n this.eventCache.clear();\n this.anchorCache.clear();\n this.digestCache.clear();\n \n logger.info('All StackMemory caches cleared');\n }\n}\n\n// Global cache instance\nlet globalCache: StackMemoryQueryCache | null = null;\n\n/**\n * Get or create global query cache\n */\nexport function getQueryCache(): StackMemoryQueryCache {\n if (!globalCache) {\n globalCache = new StackMemoryQueryCache();\n }\n return globalCache;\n}\n\n/**\n * Create a cache key from query parameters\n */\nexport function createCacheKey(queryName: string, params: any[]): string {\n const paramsStr = params.map((p: any) => \n typeof p === 'object' ? JSON.stringify(p) : String(p)\n ).join(':');\n \n return `${queryName}:${paramsStr}`;\n}"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }