@stackmemoryai/stackmemory 0.3.7 → 0.3.8

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 (190) hide show
  1. package/dist/agents/core/agent-task-manager.js +5 -5
  2. package/dist/agents/core/agent-task-manager.js.map +2 -2
  3. package/dist/agents/verifiers/base-verifier.js +2 -2
  4. package/dist/agents/verifiers/base-verifier.js.map +2 -2
  5. package/dist/cli/claude-sm.js +0 -11
  6. package/dist/cli/claude-sm.js.map +2 -2
  7. package/dist/cli/codex-sm.js +0 -11
  8. package/dist/cli/codex-sm.js.map +2 -2
  9. package/dist/cli/commands/chromadb.js +64 -34
  10. package/dist/cli/commands/chromadb.js.map +2 -2
  11. package/dist/cli/commands/clear.js +9 -13
  12. package/dist/cli/commands/clear.js.map +2 -2
  13. package/dist/cli/commands/config.js +43 -33
  14. package/dist/cli/commands/config.js.map +2 -2
  15. package/dist/cli/commands/context.js.map +2 -2
  16. package/dist/cli/commands/dashboard.js +41 -13
  17. package/dist/cli/commands/dashboard.js.map +2 -2
  18. package/dist/cli/commands/gc.js +69 -20
  19. package/dist/cli/commands/gc.js.map +2 -2
  20. package/dist/cli/commands/handoff.js.map +2 -2
  21. package/dist/cli/commands/infinite-storage.js +60 -19
  22. package/dist/cli/commands/infinite-storage.js.map +2 -2
  23. package/dist/cli/commands/linear-create.js +36 -8
  24. package/dist/cli/commands/linear-create.js.map +2 -2
  25. package/dist/cli/commands/linear-list.js +33 -10
  26. package/dist/cli/commands/linear-list.js.map +2 -2
  27. package/dist/cli/commands/linear-migrate.js +17 -4
  28. package/dist/cli/commands/linear-migrate.js.map +2 -2
  29. package/dist/cli/commands/linear-test.js +14 -6
  30. package/dist/cli/commands/linear-test.js.map +2 -2
  31. package/dist/cli/commands/linear-unified.js +123 -35
  32. package/dist/cli/commands/linear-unified.js.map +2 -2
  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 +35 -8
  36. package/dist/cli/commands/onboard.js.map +2 -2
  37. package/dist/cli/commands/quality.js +2 -7
  38. package/dist/cli/commands/quality.js.map +2 -2
  39. package/dist/cli/commands/session.js +23 -6
  40. package/dist/cli/commands/session.js.map +2 -2
  41. package/dist/cli/commands/skills.js +72 -27
  42. package/dist/cli/commands/skills.js.map +2 -2
  43. package/dist/cli/commands/storage.js +108 -38
  44. package/dist/cli/commands/storage.js.map +2 -2
  45. package/dist/cli/commands/tui.js.map +2 -2
  46. package/dist/cli/commands/webhook.js +57 -18
  47. package/dist/cli/commands/webhook.js.map +2 -2
  48. package/dist/cli/commands/workflow.js +8 -15
  49. package/dist/cli/commands/workflow.js.map +2 -2
  50. package/dist/cli/commands/worktree.js +34 -13
  51. package/dist/cli/commands/worktree.js.map +2 -2
  52. package/dist/cli/index.js +0 -11
  53. package/dist/cli/index.js.map +2 -2
  54. package/dist/core/config/types.js.map +1 -1
  55. package/dist/core/context/auto-context.js +10 -6
  56. package/dist/core/context/auto-context.js.map +2 -2
  57. package/dist/core/context/context-bridge.js.map +2 -2
  58. package/dist/core/context/frame-database.js +13 -3
  59. package/dist/core/context/frame-database.js.map +2 -2
  60. package/dist/core/context/frame-digest.js +7 -5
  61. package/dist/core/context/frame-digest.js.map +2 -2
  62. package/dist/core/context/frame-manager.js.map +2 -2
  63. package/dist/core/context/frame-stack.js +16 -5
  64. package/dist/core/context/frame-stack.js.map +2 -2
  65. package/dist/core/context/incremental-gc.js +10 -3
  66. package/dist/core/context/incremental-gc.js.map +2 -2
  67. package/dist/core/context/index.js.map +1 -1
  68. package/dist/core/context/permission-manager.js.map +2 -2
  69. package/dist/core/context/refactored-frame-manager.js +12 -3
  70. package/dist/core/context/refactored-frame-manager.js.map +2 -2
  71. package/dist/core/context/shared-context-layer.js +4 -2
  72. package/dist/core/context/shared-context-layer.js.map +2 -2
  73. package/dist/core/database/batch-operations.js +112 -86
  74. package/dist/core/database/batch-operations.js.map +2 -2
  75. package/dist/core/database/query-cache.js +19 -9
  76. package/dist/core/database/query-cache.js.map +2 -2
  77. package/dist/core/database/sqlite-adapter.js +1 -1
  78. package/dist/core/database/sqlite-adapter.js.map +2 -2
  79. package/dist/core/digest/enhanced-hybrid-digest.js +8 -2
  80. package/dist/core/digest/enhanced-hybrid-digest.js.map +2 -2
  81. package/dist/core/errors/recovery.js +9 -2
  82. package/dist/core/errors/recovery.js.map +2 -2
  83. package/dist/core/frame/workflow-templates-stub.js.map +1 -1
  84. package/dist/core/frame/workflow-templates.js +40 -1
  85. package/dist/core/frame/workflow-templates.js.map +2 -2
  86. package/dist/core/monitoring/logger.js +6 -1
  87. package/dist/core/monitoring/logger.js.map +2 -2
  88. package/dist/core/monitoring/metrics.js.map +2 -2
  89. package/dist/core/monitoring/progress-tracker.js.map +2 -2
  90. package/dist/core/performance/context-cache.js.map +2 -2
  91. package/dist/core/performance/lazy-context-loader.js +24 -20
  92. package/dist/core/performance/lazy-context-loader.js.map +2 -2
  93. package/dist/core/performance/optimized-frame-context.js +27 -12
  94. package/dist/core/performance/optimized-frame-context.js.map +2 -2
  95. package/dist/core/performance/performance-benchmark.js +10 -6
  96. package/dist/core/performance/performance-benchmark.js.map +2 -2
  97. package/dist/core/performance/performance-profiler.js +51 -14
  98. package/dist/core/performance/performance-profiler.js.map +2 -2
  99. package/dist/core/performance/streaming-jsonl-parser.js +5 -1
  100. package/dist/core/performance/streaming-jsonl-parser.js.map +2 -2
  101. package/dist/core/projects/project-manager.js +14 -20
  102. package/dist/core/projects/project-manager.js.map +2 -2
  103. package/dist/core/retrieval/context-retriever.js.map +1 -1
  104. package/dist/core/retrieval/llm-context-retrieval.js.map +2 -2
  105. package/dist/core/session/clear-survival-stub.js +5 -1
  106. package/dist/core/session/clear-survival-stub.js.map +2 -2
  107. package/dist/core/session/clear-survival.js +35 -0
  108. package/dist/core/session/clear-survival.js.map +2 -2
  109. package/dist/core/session/index.js.map +1 -1
  110. package/dist/core/session/session-manager.js.map +2 -2
  111. package/dist/core/storage/chromadb-adapter.js +6 -2
  112. package/dist/core/storage/chromadb-adapter.js.map +2 -2
  113. package/dist/core/storage/chromadb-simple.js +17 -5
  114. package/dist/core/storage/chromadb-simple.js.map +2 -2
  115. package/dist/core/storage/infinite-storage.js +109 -46
  116. package/dist/core/storage/infinite-storage.js.map +2 -2
  117. package/dist/core/storage/railway-optimized-storage.js +48 -22
  118. package/dist/core/storage/railway-optimized-storage.js.map +2 -2
  119. package/dist/core/storage/remote-storage.js +41 -23
  120. package/dist/core/storage/remote-storage.js.map +2 -2
  121. package/dist/core/trace/cli-trace-wrapper.js +9 -2
  122. package/dist/core/trace/cli-trace-wrapper.js.map +2 -2
  123. package/dist/core/trace/db-trace-wrapper.js +96 -68
  124. package/dist/core/trace/db-trace-wrapper.js.map +2 -2
  125. package/dist/core/trace/debug-trace.js +25 -8
  126. package/dist/core/trace/debug-trace.js.map +2 -2
  127. package/dist/core/trace/index.js +6 -2
  128. package/dist/core/trace/index.js.map +2 -2
  129. package/dist/core/trace/linear-api-wrapper.js +10 -5
  130. package/dist/core/trace/linear-api-wrapper.js.map +2 -2
  131. package/dist/core/trace/trace-demo.js +14 -10
  132. package/dist/core/trace/trace-demo.js.map +2 -2
  133. package/dist/core/trace/trace-detector.js +9 -2
  134. package/dist/core/trace/trace-detector.js.map +2 -2
  135. package/dist/core/trace/types.js.map +1 -1
  136. package/dist/core/utils/compression.js.map +1 -1
  137. package/dist/core/utils/update-checker.js.map +1 -1
  138. package/dist/core/worktree/worktree-manager.js +18 -7
  139. package/dist/core/worktree/worktree-manager.js.map +2 -2
  140. package/dist/features/analytics/core/analytics-service.js.map +2 -2
  141. package/dist/features/analytics/queries/metrics-queries.js +1 -1
  142. package/dist/features/analytics/queries/metrics-queries.js.map +2 -2
  143. package/dist/features/tasks/pebbles-task-store.js.map +1 -1
  144. package/dist/features/tui/components/analytics-panel.js +36 -15
  145. package/dist/features/tui/components/analytics-panel.js.map +2 -2
  146. package/dist/features/tui/components/pr-tracker.js +19 -7
  147. package/dist/features/tui/components/pr-tracker.js.map +2 -2
  148. package/dist/features/tui/components/session-monitor.js +22 -9
  149. package/dist/features/tui/components/session-monitor.js.map +2 -2
  150. package/dist/features/tui/components/subagent-fleet.js +20 -13
  151. package/dist/features/tui/components/subagent-fleet.js.map +2 -2
  152. package/dist/features/tui/components/task-board.js +26 -10
  153. package/dist/features/tui/components/task-board.js.map +2 -2
  154. package/dist/features/tui/index.js.map +2 -2
  155. package/dist/features/tui/services/data-service.js +6 -2
  156. package/dist/features/tui/services/data-service.js.map +2 -2
  157. package/dist/features/tui/services/linear-task-reader.js +3 -1
  158. package/dist/features/tui/services/linear-task-reader.js.map +2 -2
  159. package/dist/features/tui/services/websocket-client.js +3 -1
  160. package/dist/features/tui/services/websocket-client.js.map +2 -2
  161. package/dist/features/tui/terminal-compat.js +6 -2
  162. package/dist/features/tui/terminal-compat.js.map +2 -2
  163. package/dist/features/web/client/stores/task-store.js.map +2 -2
  164. package/dist/features/web/server/index.js +18 -10
  165. package/dist/features/web/server/index.js.map +2 -2
  166. package/dist/integrations/linear/sync-service.js +12 -13
  167. package/dist/integrations/linear/sync-service.js.map +2 -2
  168. package/dist/integrations/linear/sync.js +174 -12
  169. package/dist/integrations/linear/sync.js.map +2 -2
  170. package/dist/integrations/linear/unified-sync.js +1 -1
  171. package/dist/integrations/linear/unified-sync.js.map +1 -1
  172. package/dist/integrations/linear/webhook-server.js +15 -16
  173. package/dist/integrations/linear/webhook-server.js.map +2 -2
  174. package/dist/mcp/stackmemory-mcp-server.js +0 -11
  175. package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
  176. package/dist/servers/production/auth-middleware.js.map +2 -2
  177. package/dist/servers/railway/index.js.map +2 -2
  178. package/dist/services/config-service.js +6 -7
  179. package/dist/services/config-service.js.map +2 -2
  180. package/dist/services/context-service.js +11 -12
  181. package/dist/services/context-service.js.map +2 -2
  182. package/dist/skills/claude-skills.js +4 -2
  183. package/dist/skills/claude-skills.js.map +2 -2
  184. package/dist/skills/dashboard-launcher.js.map +2 -2
  185. package/dist/skills/repo-ingestion-skill.js.map +2 -2
  186. package/dist/utils/env.js +46 -0
  187. package/dist/utils/env.js.map +7 -0
  188. package/dist/utils/logger.js +0 -11
  189. package/dist/utils/logger.js.map +2 -2
  190. package/package.json +1 -1
@@ -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: any) => \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', {\n pattern: pattern.source,\n invalidated: count,\n });\n return count;\n }\n\n /**\n * Get cache metrics\n */\n getMetrics(): CacheMetrics {\n return {\n ...this.metrics,\n hitRate:\n 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]) => ({\n key,\n entry,\n }));\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', {\n removed,\n remaining: this.cache.size,\n });\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', {\n projectId,\n totalInvalidated: total,\n });\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\n .map((p: any) => (typeof p === 'object' ? JSON.stringify(p) : String(p)))\n .join(':');\n\n return `${queryName}:${paramsStr}`;\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -334,7 +334,7 @@ class SQLiteAdapter extends FeatureAwareDatabaseAdapter {
334
334
  // Limited search (basic LIKE queries)
335
335
  async search(options) {
336
336
  if (!this.db) throw new Error("Database not connected");
337
- let sql = `
337
+ const sql = `
338
338
  SELECT *,
339
339
  CASE
340
340
  WHEN name LIKE ? THEN 1.0
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/database/sqlite-adapter.ts"],
4
- "sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n EventRow,\n AnchorRow,\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';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: false, // Could enable with FTS5\n supportsVectorSearch: false,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const currentVersion = await this.getSchemaVersion();\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\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db) throw new Error('Database not connected');\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch {\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\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 JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db) throw new Error('Database not connected');\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\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 if (!this.db) throw new Error('Database not connected');\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\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 return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Limited search (basic LIKE queries)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db) throw new Error('Database not connected');\n\n // SQLite doesn't support HAVING on non-aggregate queries, so we filter in application\n let sql = `\n SELECT *, \n CASE \n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db.prepare(sql).all(...params) as any[];\n\n // Apply score threshold in application layer\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n // Apply limit and offset in application layer if threshold is used\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n // Not supported in SQLite\n logger.warn('Vector search not supported in SQLite adapter');\n return [];\n }\n\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n // Fall back to text search only\n return this.search({ query: textQuery, ...weights });\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\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 if (!this.db) throw new Error('Database not connected');\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db) throw new Error('Database not connected');\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch {}\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\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 await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite export`);\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite import`);\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * SQLite Database Adapter\n * Maintains backward compatibility with existing SQLite implementation\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FeatureAwareDatabaseAdapter,\n DatabaseFeatures,\n SearchOptions,\n QueryOptions,\n AggregationOptions,\n BulkOperation,\n DatabaseStats,\n CountResult,\n VersionResult,\n FrameRow,\n EventRow,\n AnchorRow,\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';\nimport * as path from 'path';\n\nexport interface SQLiteConfig {\n dbPath: string;\n walMode?: boolean;\n busyTimeout?: number;\n cacheSize?: number;\n synchronous?: 'OFF' | 'NORMAL' | 'FULL' | 'EXTRA';\n}\n\nexport class SQLiteAdapter extends FeatureAwareDatabaseAdapter {\n private db: Database.Database | null = null;\n private readonly dbPath: string;\n private inTransactionFlag = false;\n\n constructor(projectId: string, config: SQLiteConfig) {\n super(projectId, config);\n this.dbPath = config.dbPath;\n }\n\n getFeatures(): DatabaseFeatures {\n return {\n supportsFullTextSearch: false, // Could enable with FTS5\n supportsVectorSearch: false,\n supportsPartitioning: false,\n supportsAnalytics: false,\n supportsCompression: false,\n supportsMaterializedViews: false,\n supportsParallelQueries: false,\n };\n }\n\n async connect(): Promise<void> {\n if (this.db) return;\n\n const config = this.config as SQLiteConfig;\n\n // Ensure directory exists\n const dir = path.dirname(this.dbPath);\n await fs.mkdir(dir, { recursive: true });\n\n this.db = new Database(this.dbPath);\n\n // Configure SQLite for better performance\n if (config.walMode !== false) {\n this.db.pragma('journal_mode = WAL');\n }\n\n if (config.busyTimeout) {\n this.db.pragma(`busy_timeout = ${config.busyTimeout}`);\n }\n\n if (config.cacheSize) {\n this.db.pragma(`cache_size = ${config.cacheSize}`);\n }\n\n if (config.synchronous) {\n this.db.pragma(`synchronous = ${config.synchronous}`);\n }\n\n logger.info('SQLite database connected', { dbPath: this.dbPath });\n }\n\n async disconnect(): Promise<void> {\n if (!this.db) return;\n\n this.db.close();\n this.db = null;\n logger.info('SQLite database disconnected');\n }\n\n /**\n * Get raw database handle for testing purposes\n * @internal\n */\n getRawDatabase(): Database.Database | null {\n return this.db;\n }\n\n isConnected(): boolean {\n return this.db !== null && this.db.open;\n }\n\n async ping(): Promise<boolean> {\n if (!this.db) return false;\n\n try {\n this.db.prepare('SELECT 1').get();\n return true;\n } catch {\n return false;\n }\n }\n\n async initializeSchema(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT REFERENCES frames(frame_id),\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER DEFAULT (unixepoch()),\n closed_at INTEGER\n );\n\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n frame_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL,\n ts INTEGER DEFAULT (unixepoch()),\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER DEFAULT 0,\n created_at INTEGER DEFAULT (unixepoch()),\n metadata TEXT DEFAULT '{}',\n FOREIGN KEY(frame_id) REFERENCES frames(frame_id)\n );\n\n CREATE TABLE IF NOT EXISTS schema_version (\n version INTEGER PRIMARY KEY,\n applied_at INTEGER DEFAULT (unixepoch())\n );\n\n -- Indexes for performance\n CREATE INDEX IF NOT EXISTS idx_frames_run ON frames(run_id);\n CREATE INDEX IF NOT EXISTS idx_frames_project ON frames(project_id);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_frames_state ON frames(state);\n CREATE INDEX IF NOT EXISTS idx_frames_created ON frames(created_at DESC);\n CREATE INDEX IF NOT EXISTS idx_events_frame ON events(frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame ON anchors(frame_id);\n\n -- Set initial schema version if not exists\n INSERT OR IGNORE INTO schema_version (version) VALUES (1);\n `);\n }\n\n async migrateSchema(targetVersion: number): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const currentVersion = await this.getSchemaVersion();\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\n this.db.prepare('UPDATE schema_version SET version = ?').run(v);\n }\n }\n\n async getSchemaVersion(): Promise<number> {\n if (!this.db) throw new Error('Database not connected');\n\n try {\n const result = this.db\n .prepare('SELECT MAX(version) as version FROM schema_version')\n .get() as VersionResult;\n return result?.version || 0;\n } catch {\n return 0;\n }\n }\n\n // Frame operations\n async createFrame(frame: Partial<Frame>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const frameId = frame.frame_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO frames (\n frame_id, run_id, project_id, parent_frame_id, depth,\n type, name, state, inputs, outputs, digest_text, digest_json\n ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n frameId,\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 JSON.stringify(frame.inputs || {}),\n JSON.stringify(frame.outputs || {}),\n frame.digest_text || null,\n JSON.stringify(frame.digest_json || {})\n );\n\n return frameId;\n }\n\n async getFrame(frameId: string): Promise<Frame | null> {\n if (!this.db) throw new Error('Database not connected');\n\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as FrameRow | undefined;\n\n if (!row) return null;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n }\n\n async updateFrame(frameId: string, updates: Partial<Frame>): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n const fields = [];\n const values = [];\n\n if (updates.state !== undefined) {\n fields.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n fields.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n fields.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n fields.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n fields.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (fields.length === 0) return;\n\n values.push(frameId);\n\n this.db\n .prepare(\n `\n UPDATE frames SET ${fields.join(', ')} WHERE frame_id = ?\n `\n )\n .run(...values);\n }\n\n async deleteFrame(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n // Delete in order due to foreign keys\n await this.deleteFrameAnchors(frameId);\n await this.deleteFrameEvents(frameId);\n\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n }\n\n async getActiveFrames(runId?: string): Promise<Frame[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = \"SELECT * FROM frames WHERE state = 'active'\";\n const params = [];\n\n if (runId) {\n query += ' AND run_id = ?';\n params.push(runId);\n }\n\n query += ' ORDER BY depth ASC, created_at ASC';\n\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\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 if (!this.db) throw new Error('Database not connected');\n\n const eventId = event.event_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO events (event_id, run_id, frame_id, seq, event_type, payload, ts)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n eventId,\n event.run_id,\n event.frame_id,\n event.seq || 0,\n event.event_type,\n JSON.stringify(event.payload || {}),\n event.ts || Date.now()\n );\n\n return eventId;\n }\n\n async getFrameEvents(\n frameId: string,\n options?: QueryOptions\n ): Promise<Event[]> {\n if (!this.db) throw new Error('Database not connected');\n\n let query = 'SELECT * FROM events WHERE frame_id = ?';\n query += this.buildOrderByClause(\n options?.orderBy || 'seq',\n options?.orderDirection\n );\n query += this.buildLimitClause(options?.limit, options?.offset);\n\n const rows = this.db.prepare(query).all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n payload: JSON.parse(row.payload || '{}'),\n }));\n }\n\n async deleteFrameEvents(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n }\n\n // Anchor operations\n async createAnchor(anchor: Partial<Anchor>): Promise<string> {\n if (!this.db) throw new Error('Database not connected');\n\n const anchorId = anchor.anchor_id || this.generateId();\n\n this.db\n .prepare(\n `\n INSERT INTO anchors (anchor_id, frame_id, project_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?, ?)\n `\n )\n .run(\n anchorId,\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 return anchorId;\n }\n\n async getFrameAnchors(frameId: string): Promise<Anchor[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const rows = this.db\n .prepare(\n `\n SELECT * FROM anchors WHERE frame_id = ? \n ORDER BY priority DESC, created_at ASC\n `\n )\n .all(frameId) as any[];\n\n return rows.map((row) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n }\n\n async deleteFrameAnchors(frameId: string): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n }\n\n // Limited search (basic LIKE queries)\n async search(\n options: SearchOptions\n ): Promise<Array<Frame & { score: number }>> {\n if (!this.db) throw new Error('Database not connected');\n\n // SQLite doesn't support HAVING on non-aggregate queries, so we filter in application\n const sql = `\n SELECT *, \n CASE \n WHEN name LIKE ? THEN 1.0\n WHEN digest_text LIKE ? THEN 0.8\n WHEN inputs LIKE ? THEN 0.6\n ELSE 0.5\n END as score\n FROM frames\n WHERE name LIKE ? OR digest_text LIKE ? OR inputs LIKE ?\n ORDER BY score DESC\n `;\n\n const params = Array(6).fill(`%${options.query}%`);\n\n let rows = this.db.prepare(sql).all(...params) as any[];\n\n // Apply score threshold in application layer\n if (options.scoreThreshold) {\n rows = rows.filter((row) => row.score >= options.scoreThreshold);\n }\n\n // Apply limit and offset in application layer if threshold is used\n if (options.limit || options.offset) {\n const start = options.offset || 0;\n const end = options.limit ? start + options.limit : rows.length;\n rows = rows.slice(start, end);\n }\n\n return rows.map((row) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n }\n\n async searchByVector(\n embedding: number[],\n options?: QueryOptions\n ): Promise<Array<Frame & { similarity: number }>> {\n // Not supported in SQLite\n logger.warn('Vector search not supported in SQLite adapter');\n return [];\n }\n\n async searchHybrid(\n textQuery: string,\n embedding: number[],\n weights?: { text: number; vector: number }\n ): Promise<Array<Frame & { score: number }>> {\n // Fall back to text search only\n return this.search({ query: textQuery, ...weights });\n }\n\n // Basic aggregation\n async aggregate(\n table: string,\n options: AggregationOptions\n ): Promise<Record<string, any>[]> {\n if (!this.db) throw new Error('Database not connected');\n\n const metrics = options.metrics\n .map(\n (m) =>\n `${m.operation}(${m.field}) AS ${m.alias || `${m.operation}_${m.field}`}`\n )\n .join(', ');\n\n let sql = `SELECT ${options.groupBy.join(', ')}, ${metrics} FROM ${table}`;\n sql += ` GROUP BY ${options.groupBy.join(', ')}`;\n\n if (options.having) {\n const havingClauses = Object.entries(options.having).map(\n ([key, value]) =>\n `${key} ${typeof value === 'object' ? value.op : '='} ?`\n );\n sql += ` HAVING ${havingClauses.join(' AND ')}`;\n }\n\n return this.db\n .prepare(sql)\n .all(...Object.values(options.having || {})) as any[];\n }\n\n // Pattern detection (basic)\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 if (!this.db) throw new Error('Database not connected');\n\n let sql = `\n SELECT type as pattern, type, COUNT(*) as frequency, MAX(created_at) as last_seen\n FROM frames\n `;\n\n const params = [];\n if (timeRange) {\n sql += ' WHERE created_at >= ? AND created_at <= ?';\n params.push(\n Math.floor(timeRange.start.getTime() / 1000),\n Math.floor(timeRange.end.getTime() / 1000)\n );\n }\n\n sql += ' GROUP BY type HAVING COUNT(*) > 1 ORDER BY frequency DESC';\n\n const rows = this.db.prepare(sql).all(...params) as any[];\n\n return rows.map((row) => ({\n pattern: row.pattern,\n type: row.type,\n frequency: row.frequency,\n lastSeen: new Date(row.last_seen * 1000),\n }));\n }\n\n // Bulk operations\n async executeBulk(operations: BulkOperation[]): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n await this.inTransaction(async () => {\n for (const op of operations) {\n switch (op.type) {\n case 'insert':\n // Build insert dynamically based on data\n const insertCols = Object.keys(op.data);\n const insertPlaceholders = insertCols.map(() => '?').join(',');\n this.db!.prepare(\n `INSERT INTO ${op.table} (${insertCols.join(',')}) VALUES (${insertPlaceholders})`\n ).run(...Object.values(op.data));\n break;\n\n case 'update':\n const updateSets = Object.keys(op.data)\n .map((k) => `${k} = ?`)\n .join(',');\n const whereClause = this.buildWhereClause(op.where || {});\n this.db!.prepare(\n `UPDATE ${op.table} SET ${updateSets} ${whereClause}`\n ).run(...Object.values(op.data), ...Object.values(op.where || {}));\n break;\n\n case 'delete':\n const deleteWhere = this.buildWhereClause(op.where || {});\n this.db!.prepare(`DELETE FROM ${op.table} ${deleteWhere}`).run(\n ...Object.values(op.where || {})\n );\n break;\n }\n }\n });\n }\n\n async vacuum(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('vacuum');\n logger.info('SQLite database vacuumed');\n }\n\n async analyze(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.pragma('analyze');\n logger.info('SQLite database analyzed');\n }\n\n // Statistics\n async getStats(): Promise<DatabaseStats> {\n if (!this.db) throw new Error('Database not connected');\n\n const stats = {\n totalFrames: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as CountResult\n ).count,\n activeFrames: (\n this.db\n .prepare(\n \"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\"\n )\n .get() as CountResult\n ).count,\n totalEvents: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as CountResult\n ).count,\n totalAnchors: (\n this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as CountResult\n ).count,\n diskUsage: 0,\n };\n\n // Get file size\n try {\n const fileStats = await fs.stat(this.dbPath);\n stats.diskUsage = fileStats.size;\n } catch {}\n\n return stats;\n }\n\n async getQueryStats(): Promise<\n Array<{\n query: string;\n calls: number;\n meanTime: number;\n totalTime: number;\n }>\n > {\n // SQLite doesn't have built-in query stats\n logger.warn('Query stats not available for SQLite');\n return [];\n }\n\n // Transaction support\n async beginTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('BEGIN').run();\n this.inTransactionFlag = true;\n }\n\n async commitTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('COMMIT').run();\n this.inTransactionFlag = false;\n }\n\n async rollbackTransaction(): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n this.db.prepare('ROLLBACK').run();\n this.inTransactionFlag = false;\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 await this.rollbackTransaction();\n throw error;\n }\n }\n\n // Export/Import\n async exportData(\n tables: string[],\n format: 'json' | 'parquet' | 'csv'\n ): Promise<Buffer> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite export`);\n }\n\n const data: Record<string, any[]> = {};\n\n for (const table of tables) {\n data[table] = this.db.prepare(`SELECT * FROM ${table}`).all();\n }\n\n return Buffer.from(JSON.stringify(data, null, 2));\n }\n\n async importData(\n data: Buffer,\n format: 'json' | 'parquet' | 'csv',\n options?: { truncate?: boolean; upsert?: boolean }\n ): Promise<void> {\n if (!this.db) throw new Error('Database not connected');\n\n if (format !== 'json') {\n throw new Error(`Format ${format} not supported for SQLite import`);\n }\n\n const parsed = JSON.parse(data.toString());\n\n await this.inTransaction(async () => {\n for (const [table, rows] of Object.entries(parsed)) {\n if (options?.truncate) {\n this.db!.prepare(`DELETE FROM ${table}`).run();\n }\n\n for (const row of rows as any[]) {\n const cols = Object.keys(row);\n const placeholders = cols.map(() => '?').join(',');\n\n if (options?.upsert) {\n const updates = cols.map((c) => `${c} = excluded.${c}`).join(',');\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})\n ON CONFLICT DO UPDATE SET ${updates}`\n ).run(...Object.values(row));\n } else {\n this.db!.prepare(\n `INSERT INTO ${table} (${cols.join(',')}) VALUES (${placeholders})`\n ).run(...Object.values(row));\n }\n }\n }\n });\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -49,7 +49,10 @@ class EnhancedHybridDigestGenerator extends HybridDigestGenerator {
49
49
  timeSinceLastInput: now - this.lastInputTime
50
50
  });
51
51
  this.processQueue().catch((error) => {
52
- logger.error("Error processing digest queue during idle", error instanceof Error ? error : new Error(String(error)));
52
+ logger.error(
53
+ "Error processing digest queue during idle",
54
+ error instanceof Error ? error : new Error(String(error))
55
+ );
53
56
  });
54
57
  }
55
58
  }
@@ -76,7 +79,10 @@ class EnhancedHybridDigestGenerator extends HybridDigestGenerator {
76
79
  });
77
80
  this.prioritizeFrame(frameId);
78
81
  this.processQueue().catch((error) => {
79
- logger.error("Error processing digest on frame close", error instanceof Error ? error : new Error(String(error)));
82
+ logger.error(
83
+ "Error processing digest on frame close",
84
+ error instanceof Error ? error : new Error(String(error))
85
+ );
80
86
  });
81
87
  }
82
88
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/digest/enhanced-hybrid-digest.ts"],
4
- "sourcesContent": ["/**\n * Enhanced Hybrid Digest Generator for STA-96\n * Implements 60/40 split: deterministic extraction + AI-generated insights\n */\n\nimport Database from 'better-sqlite3';\nimport {\n HybridDigest,\n DeterministicDigest,\n AIGeneratedDigest,\n DigestConfig,\n DigestInput,\n DigestGenerationRequest,\n DigestQueueStats,\n DigestLLMProvider,\n DigestStatus,\n DEFAULT_DIGEST_CONFIG,\n} from './types.js';\nimport { HybridDigestGenerator } from './hybrid-digest-generator.js';\nimport { logger } from '../monitoring/logger.js';\n\n/**\n * Enhanced AI digest with 40% content\n */\nexport interface EnhancedAIDigest extends AIGeneratedDigest {\n /** Key decisions made and their reasoning */\n keyDecisions?: string[];\n /** Important insights discovered */\n insights?: string[];\n /** Suggested next steps */\n nextSteps?: string[];\n /** Patterns detected */\n patterns?: string[];\n /** Technical debt or improvements identified */\n technicalDebt?: string[];\n}\n\n/**\n * Idle detection configuration\n */\nexport interface IdleDetectionConfig {\n /** No tool calls threshold (ms) */\n noToolCallThreshold: number;\n /** No user input threshold (ms) */\n noInputThreshold: number;\n /** Frame closed immediately triggers processing */\n processOnFrameClose: boolean;\n /** Check interval (ms) */\n checkInterval: number;\n}\n\nconst DEFAULT_IDLE_CONFIG: IdleDetectionConfig = {\n noToolCallThreshold: 30000, // 30 seconds\n noInputThreshold: 60000, // 60 seconds\n processOnFrameClose: true,\n checkInterval: 10000, // Check every 10 seconds\n};\n\n/**\n * Enhanced Hybrid Digest Generator\n * Implements 60% deterministic + 40% AI insights\n */\nexport class EnhancedHybridDigestGenerator extends HybridDigestGenerator {\n private idleConfig: IdleDetectionConfig;\n private lastToolCallTime: number = Date.now();\n private lastInputTime: number = Date.now();\n private idleCheckInterval?: NodeJS.Timeout;\n private activeFrames = new Set<string>();\n\n constructor(\n db: Database.Database,\n config: Partial<DigestConfig> = {},\n llmProvider?: DigestLLMProvider,\n idleConfig: Partial<IdleDetectionConfig> = {}\n ) {\n // Update config to reflect 60/40 split\n const enhancedConfig = {\n ...config,\n maxTokens: config.maxTokens || 200, // Keep under 200 tokens as per requirement\n enableAIGeneration: config.enableAIGeneration ?? true,\n };\n\n super(db, enhancedConfig, llmProvider);\n this.idleConfig = { ...DEFAULT_IDLE_CONFIG, ...idleConfig };\n this.startIdleDetection();\n }\n\n /**\n * Start idle detection monitoring\n */\n private startIdleDetection(): void {\n this.idleCheckInterval = setInterval(() => {\n this.checkIdleState();\n }, this.idleConfig.checkInterval);\n }\n\n /**\n * Check if system is idle and trigger processing\n */\n private checkIdleState(): void {\n const now = Date.now();\n\n // Check for idle based on tool calls\n const toolCallIdle =\n now - this.lastToolCallTime > this.idleConfig.noToolCallThreshold;\n\n // Check for idle based on user input\n const inputIdle =\n now - this.lastInputTime > this.idleConfig.noInputThreshold;\n\n if (toolCallIdle || inputIdle) {\n logger.debug('Idle state detected, triggering digest processing', {\n toolCallIdle,\n inputIdle,\n timeSinceLastToolCall: now - this.lastToolCallTime,\n timeSinceLastInput: now - this.lastInputTime,\n });\n\n this.processQueue().catch((error) => {\n logger.error('Error processing digest queue during idle', error instanceof Error ? error : new Error(String(error)));\n });\n }\n }\n\n /**\n * Record tool call activity\n */\n public recordToolCall(): void {\n this.lastToolCallTime = Date.now();\n }\n\n /**\n * Record user input activity\n */\n public recordUserInput(): void {\n this.lastInputTime = Date.now();\n }\n\n /**\n * Handle frame closure - immediately trigger digest if configured\n */\n public onFrameClosed(frameId: string): void {\n this.activeFrames.delete(frameId);\n\n if (this.idleConfig.processOnFrameClose) {\n logger.info('Frame closed, triggering immediate digest processing', {\n frameId,\n });\n\n // Process this specific frame with high priority\n this.prioritizeFrame(frameId);\n this.processQueue().catch((error) => {\n logger.error('Error processing digest on frame close', error instanceof Error ? error : new Error(String(error)));\n });\n }\n }\n\n /**\n * Handle frame opened\n */\n public onFrameOpened(frameId: string): void {\n this.activeFrames.add(frameId);\n }\n\n /**\n * Prioritize a specific frame for processing\n */\n private prioritizeFrame(frameId: string): void {\n try {\n this.db\n .prepare(\n `\n UPDATE digest_queue \n SET priority = 'high', updated_at = unixepoch()\n WHERE frame_id = ? AND status = 'pending'\n `\n )\n .run(frameId);\n } catch (error: unknown) {\n logger.error('Failed to prioritize frame', error as Error);\n }\n }\n\n /**\n * Enhanced AI generation with 40% content\n */\n protected async generateEnhancedAI(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): Promise<EnhancedAIDigest> {\n if (!this.llmProvider) {\n throw new Error('No LLM provider configured');\n }\n\n // Build enhanced prompt for 40% AI content\n const prompt = this.buildEnhancedPrompt(input, deterministic);\n\n // Generate with LLM\n const response = await this.llmProvider.generateSummary(\n input,\n deterministic,\n this.config.maxTokens\n );\n\n // Parse and structure the enhanced response\n const enhanced: EnhancedAIDigest = {\n ...response,\n keyDecisions: this.extractKeyDecisions(response),\n insights: this.extractInsights(response),\n nextSteps: this.extractNextSteps(response),\n patterns: this.detectPatterns(input, deterministic),\n technicalDebt: this.identifyTechnicalDebt(input, deterministic),\n };\n\n return enhanced;\n }\n\n /**\n * Build enhanced prompt for AI generation\n */\n private buildEnhancedPrompt(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string {\n const parts: string[] = [\n `Analyze this development frame and provide insights (max ${this.config.maxTokens} tokens):`,\n '',\n `Frame: ${input.frame.name} (${input.frame.type})`,\n `Duration: ${deterministic.durationSeconds}s`,\n `Files Modified: ${deterministic.filesModified.length}`,\n `Tool Calls: ${deterministic.toolCallCount}`,\n `Errors: ${deterministic.errorsEncountered.length}`,\n '',\n 'Provide:',\n '1. Key decisions made and why (2-3 items)',\n '2. Important insights or learnings (1-2 items)',\n '3. Suggested next steps (2-3 items)',\n '4. Any patterns or anti-patterns observed',\n '5. Technical debt or improvements needed',\n '',\n 'Be concise and actionable. Focus on value, not description.',\n ];\n\n return parts.join('\\n');\n }\n\n /**\n * Extract key decisions from AI response\n */\n private extractKeyDecisions(response: AIGeneratedDigest): string[] {\n // This would parse the AI response for decision patterns\n // For now, return placeholder\n return [];\n }\n\n /**\n * Extract insights from AI response\n */\n private extractInsights(response: AIGeneratedDigest): string[] {\n const insights: string[] = [];\n if (response.insight) {\n insights.push(response.insight);\n }\n return insights;\n }\n\n /**\n * Extract next steps from AI response\n */\n private extractNextSteps(response: AIGeneratedDigest): string[] {\n // Parse AI response for next steps\n return [];\n }\n\n /**\n * Detect patterns in the frame activity\n */\n private detectPatterns(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string[] {\n const patterns: string[] = [];\n\n // Detect retry patterns\n if (deterministic.errorsEncountered.some((e) => e.count > 2)) {\n patterns.push('Multiple retry attempts detected');\n }\n\n // Detect test-driven development\n const hasTests = deterministic.testsRun.length > 0;\n const hasCodeChanges = deterministic.filesModified.some(\n (f) => f.operation === 'modify' && !f.path.includes('test')\n );\n if (hasTests && hasCodeChanges) {\n patterns.push('Test-driven development pattern observed');\n }\n\n // Detect refactoring patterns\n const manyFileChanges = deterministic.filesModified.length > 5;\n const noNewFiles = !deterministic.filesModified.some(\n (f) => f.operation === 'create'\n );\n if (manyFileChanges && noNewFiles) {\n patterns.push('Refactoring pattern detected');\n }\n\n return patterns;\n }\n\n /**\n * Identify technical debt\n */\n private identifyTechnicalDebt(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string[] {\n const debt: string[] = [];\n\n // Check for missing tests\n if (\n deterministic.filesModified.length > 3 &&\n deterministic.testsRun.length === 0\n ) {\n debt.push('Code changes without corresponding tests');\n }\n\n // Check for unresolved errors\n const unresolvedErrors = deterministic.errorsEncountered.filter(\n (e) => !e.resolved\n );\n if (unresolvedErrors.length > 0) {\n debt.push(`${unresolvedErrors.length} unresolved errors remain`);\n }\n\n // Check for TODOs in decisions\n if (deterministic.decisions.some((d) => d.toLowerCase().includes('todo'))) {\n debt.push('TODOs added to codebase');\n }\n\n return debt;\n }\n\n /**\n * Generate digest with 60/40 split\n */\n public generateDigest(input: DigestInput): HybridDigest {\n // Record activity\n this.recordToolCall();\n\n // Generate base digest (60% deterministic)\n const digest = super.generateDigest(input);\n\n // Ensure AI generation is queued for 40% content\n if (this.config.enableAIGeneration && this.llmProvider) {\n digest.status = 'ai_pending';\n }\n\n return digest;\n }\n\n /**\n * Handle interruption gracefully\n */\n public handleInterruption(): void {\n logger.info('User activity detected, pausing digest processing');\n\n // Update timestamps\n this.recordUserInput();\n this.recordToolCall();\n\n // Don't stop processing entirely, just deprioritize\n this.db\n .prepare(\n `\n UPDATE digest_queue \n SET priority = 'low'\n WHERE status = 'processing'\n `\n )\n .run();\n }\n\n /**\n * Get idle status\n */\n public getIdleStatus(): {\n isIdle: boolean;\n timeSinceLastToolCall: number;\n timeSinceLastInput: number;\n activeFrames: number;\n } {\n const now = Date.now();\n return {\n isIdle:\n now - this.lastToolCallTime > this.idleConfig.noToolCallThreshold ||\n now - this.lastInputTime > this.idleConfig.noInputThreshold,\n timeSinceLastToolCall: now - this.lastToolCallTime,\n timeSinceLastInput: now - this.lastInputTime,\n activeFrames: this.activeFrames.size,\n };\n }\n\n /**\n * Cleanup on shutdown\n */\n public shutdown(): void {\n if (this.idleCheckInterval) {\n clearInterval(this.idleCheckInterval);\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Enhanced Hybrid Digest Generator for STA-96\n * Implements 60/40 split: deterministic extraction + AI-generated insights\n */\n\nimport Database from 'better-sqlite3';\nimport {\n HybridDigest,\n DeterministicDigest,\n AIGeneratedDigest,\n DigestConfig,\n DigestInput,\n DigestGenerationRequest,\n DigestQueueStats,\n DigestLLMProvider,\n DigestStatus,\n DEFAULT_DIGEST_CONFIG,\n} from './types.js';\nimport { HybridDigestGenerator } from './hybrid-digest-generator.js';\nimport { logger } from '../monitoring/logger.js';\n\n/**\n * Enhanced AI digest with 40% content\n */\nexport interface EnhancedAIDigest extends AIGeneratedDigest {\n /** Key decisions made and their reasoning */\n keyDecisions?: string[];\n /** Important insights discovered */\n insights?: string[];\n /** Suggested next steps */\n nextSteps?: string[];\n /** Patterns detected */\n patterns?: string[];\n /** Technical debt or improvements identified */\n technicalDebt?: string[];\n}\n\n/**\n * Idle detection configuration\n */\nexport interface IdleDetectionConfig {\n /** No tool calls threshold (ms) */\n noToolCallThreshold: number;\n /** No user input threshold (ms) */\n noInputThreshold: number;\n /** Frame closed immediately triggers processing */\n processOnFrameClose: boolean;\n /** Check interval (ms) */\n checkInterval: number;\n}\n\nconst DEFAULT_IDLE_CONFIG: IdleDetectionConfig = {\n noToolCallThreshold: 30000, // 30 seconds\n noInputThreshold: 60000, // 60 seconds\n processOnFrameClose: true,\n checkInterval: 10000, // Check every 10 seconds\n};\n\n/**\n * Enhanced Hybrid Digest Generator\n * Implements 60% deterministic + 40% AI insights\n */\nexport class EnhancedHybridDigestGenerator extends HybridDigestGenerator {\n private idleConfig: IdleDetectionConfig;\n private lastToolCallTime: number = Date.now();\n private lastInputTime: number = Date.now();\n private idleCheckInterval?: NodeJS.Timeout;\n private activeFrames = new Set<string>();\n\n constructor(\n db: Database.Database,\n config: Partial<DigestConfig> = {},\n llmProvider?: DigestLLMProvider,\n idleConfig: Partial<IdleDetectionConfig> = {}\n ) {\n // Update config to reflect 60/40 split\n const enhancedConfig = {\n ...config,\n maxTokens: config.maxTokens || 200, // Keep under 200 tokens as per requirement\n enableAIGeneration: config.enableAIGeneration ?? true,\n };\n\n super(db, enhancedConfig, llmProvider);\n this.idleConfig = { ...DEFAULT_IDLE_CONFIG, ...idleConfig };\n this.startIdleDetection();\n }\n\n /**\n * Start idle detection monitoring\n */\n private startIdleDetection(): void {\n this.idleCheckInterval = setInterval(() => {\n this.checkIdleState();\n }, this.idleConfig.checkInterval);\n }\n\n /**\n * Check if system is idle and trigger processing\n */\n private checkIdleState(): void {\n const now = Date.now();\n\n // Check for idle based on tool calls\n const toolCallIdle =\n now - this.lastToolCallTime > this.idleConfig.noToolCallThreshold;\n\n // Check for idle based on user input\n const inputIdle =\n now - this.lastInputTime > this.idleConfig.noInputThreshold;\n\n if (toolCallIdle || inputIdle) {\n logger.debug('Idle state detected, triggering digest processing', {\n toolCallIdle,\n inputIdle,\n timeSinceLastToolCall: now - this.lastToolCallTime,\n timeSinceLastInput: now - this.lastInputTime,\n });\n\n this.processQueue().catch((error) => {\n logger.error(\n 'Error processing digest queue during idle',\n error instanceof Error ? error : new Error(String(error))\n );\n });\n }\n }\n\n /**\n * Record tool call activity\n */\n public recordToolCall(): void {\n this.lastToolCallTime = Date.now();\n }\n\n /**\n * Record user input activity\n */\n public recordUserInput(): void {\n this.lastInputTime = Date.now();\n }\n\n /**\n * Handle frame closure - immediately trigger digest if configured\n */\n public onFrameClosed(frameId: string): void {\n this.activeFrames.delete(frameId);\n\n if (this.idleConfig.processOnFrameClose) {\n logger.info('Frame closed, triggering immediate digest processing', {\n frameId,\n });\n\n // Process this specific frame with high priority\n this.prioritizeFrame(frameId);\n this.processQueue().catch((error) => {\n logger.error(\n 'Error processing digest on frame close',\n error instanceof Error ? error : new Error(String(error))\n );\n });\n }\n }\n\n /**\n * Handle frame opened\n */\n public onFrameOpened(frameId: string): void {\n this.activeFrames.add(frameId);\n }\n\n /**\n * Prioritize a specific frame for processing\n */\n private prioritizeFrame(frameId: string): void {\n try {\n this.db\n .prepare(\n `\n UPDATE digest_queue \n SET priority = 'high', updated_at = unixepoch()\n WHERE frame_id = ? AND status = 'pending'\n `\n )\n .run(frameId);\n } catch (error: unknown) {\n logger.error('Failed to prioritize frame', error as Error);\n }\n }\n\n /**\n * Enhanced AI generation with 40% content\n */\n protected async generateEnhancedAI(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): Promise<EnhancedAIDigest> {\n if (!this.llmProvider) {\n throw new Error('No LLM provider configured');\n }\n\n // Build enhanced prompt for 40% AI content\n const prompt = this.buildEnhancedPrompt(input, deterministic);\n\n // Generate with LLM\n const response = await this.llmProvider.generateSummary(\n input,\n deterministic,\n this.config.maxTokens\n );\n\n // Parse and structure the enhanced response\n const enhanced: EnhancedAIDigest = {\n ...response,\n keyDecisions: this.extractKeyDecisions(response),\n insights: this.extractInsights(response),\n nextSteps: this.extractNextSteps(response),\n patterns: this.detectPatterns(input, deterministic),\n technicalDebt: this.identifyTechnicalDebt(input, deterministic),\n };\n\n return enhanced;\n }\n\n /**\n * Build enhanced prompt for AI generation\n */\n private buildEnhancedPrompt(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string {\n const parts: string[] = [\n `Analyze this development frame and provide insights (max ${this.config.maxTokens} tokens):`,\n '',\n `Frame: ${input.frame.name} (${input.frame.type})`,\n `Duration: ${deterministic.durationSeconds}s`,\n `Files Modified: ${deterministic.filesModified.length}`,\n `Tool Calls: ${deterministic.toolCallCount}`,\n `Errors: ${deterministic.errorsEncountered.length}`,\n '',\n 'Provide:',\n '1. Key decisions made and why (2-3 items)',\n '2. Important insights or learnings (1-2 items)',\n '3. Suggested next steps (2-3 items)',\n '4. Any patterns or anti-patterns observed',\n '5. Technical debt or improvements needed',\n '',\n 'Be concise and actionable. Focus on value, not description.',\n ];\n\n return parts.join('\\n');\n }\n\n /**\n * Extract key decisions from AI response\n */\n private extractKeyDecisions(response: AIGeneratedDigest): string[] {\n // This would parse the AI response for decision patterns\n // For now, return placeholder\n return [];\n }\n\n /**\n * Extract insights from AI response\n */\n private extractInsights(response: AIGeneratedDigest): string[] {\n const insights: string[] = [];\n if (response.insight) {\n insights.push(response.insight);\n }\n return insights;\n }\n\n /**\n * Extract next steps from AI response\n */\n private extractNextSteps(response: AIGeneratedDigest): string[] {\n // Parse AI response for next steps\n return [];\n }\n\n /**\n * Detect patterns in the frame activity\n */\n private detectPatterns(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string[] {\n const patterns: string[] = [];\n\n // Detect retry patterns\n if (deterministic.errorsEncountered.some((e) => e.count > 2)) {\n patterns.push('Multiple retry attempts detected');\n }\n\n // Detect test-driven development\n const hasTests = deterministic.testsRun.length > 0;\n const hasCodeChanges = deterministic.filesModified.some(\n (f) => f.operation === 'modify' && !f.path.includes('test')\n );\n if (hasTests && hasCodeChanges) {\n patterns.push('Test-driven development pattern observed');\n }\n\n // Detect refactoring patterns\n const manyFileChanges = deterministic.filesModified.length > 5;\n const noNewFiles = !deterministic.filesModified.some(\n (f) => f.operation === 'create'\n );\n if (manyFileChanges && noNewFiles) {\n patterns.push('Refactoring pattern detected');\n }\n\n return patterns;\n }\n\n /**\n * Identify technical debt\n */\n private identifyTechnicalDebt(\n input: DigestInput,\n deterministic: DeterministicDigest\n ): string[] {\n const debt: string[] = [];\n\n // Check for missing tests\n if (\n deterministic.filesModified.length > 3 &&\n deterministic.testsRun.length === 0\n ) {\n debt.push('Code changes without corresponding tests');\n }\n\n // Check for unresolved errors\n const unresolvedErrors = deterministic.errorsEncountered.filter(\n (e) => !e.resolved\n );\n if (unresolvedErrors.length > 0) {\n debt.push(`${unresolvedErrors.length} unresolved errors remain`);\n }\n\n // Check for TODOs in decisions\n if (deterministic.decisions.some((d) => d.toLowerCase().includes('todo'))) {\n debt.push('TODOs added to codebase');\n }\n\n return debt;\n }\n\n /**\n * Generate digest with 60/40 split\n */\n public generateDigest(input: DigestInput): HybridDigest {\n // Record activity\n this.recordToolCall();\n\n // Generate base digest (60% deterministic)\n const digest = super.generateDigest(input);\n\n // Ensure AI generation is queued for 40% content\n if (this.config.enableAIGeneration && this.llmProvider) {\n digest.status = 'ai_pending';\n }\n\n return digest;\n }\n\n /**\n * Handle interruption gracefully\n */\n public handleInterruption(): void {\n logger.info('User activity detected, pausing digest processing');\n\n // Update timestamps\n this.recordUserInput();\n this.recordToolCall();\n\n // Don't stop processing entirely, just deprioritize\n this.db\n .prepare(\n `\n UPDATE digest_queue \n SET priority = 'low'\n WHERE status = 'processing'\n `\n )\n .run();\n }\n\n /**\n * Get idle status\n */\n public getIdleStatus(): {\n isIdle: boolean;\n timeSinceLastToolCall: number;\n timeSinceLastInput: number;\n activeFrames: number;\n } {\n const now = Date.now();\n return {\n isIdle:\n now - this.lastToolCallTime > this.idleConfig.noToolCallThreshold ||\n now - this.lastInputTime > this.idleConfig.noInputThreshold,\n timeSinceLastToolCall: now - this.lastToolCallTime,\n timeSinceLastInput: now - this.lastInputTime,\n activeFrames: this.activeFrames.size,\n };\n }\n\n /**\n * Cleanup on shutdown\n */\n public shutdown(): void {\n if (this.idleCheckInterval) {\n clearInterval(this.idleCheckInterval);\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -46,7 +46,12 @@ async function retry(fn, options = {}) {
46
46
  if (!isRetryableError(error) || attempt === maxAttempts) {
47
47
  throw error;
48
48
  }
49
- const delay = calculateBackoff(attempt, initialDelay, maxDelay, backoffFactor);
49
+ const delay = calculateBackoff(
50
+ attempt,
51
+ initialDelay,
52
+ maxDelay,
53
+ backoffFactor
54
+ );
50
55
  logger.warn(`Retry attempt ${attempt}/${maxAttempts} after ${delay}ms`, {
51
56
  error: getErrorMessage(error),
52
57
  attempt,
@@ -199,7 +204,9 @@ async function withTimeout(fn, timeoutMs, timeoutMessage) {
199
204
  new Promise(
200
205
  (_, reject) => setTimeout(
201
206
  () => reject(
202
- new Error(timeoutMessage ?? `Operation timed out after ${timeoutMs}ms`)
207
+ new Error(
208
+ timeoutMessage ?? `Operation timed out after ${timeoutMs}ms`
209
+ )
203
210
  ),
204
211
  timeoutMs
205
212
  )
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/errors/recovery.ts"],
4
- "sourcesContent": ["/**\n * Error recovery utilities for StackMemory\n * Provides retry logic, circuit breakers, and fallback mechanisms\n */\n\nimport { logger } from '../monitoring/logger.js';\nimport {\n StackMemoryError,\n isRetryableError,\n getErrorMessage,\n ErrorContext,\n} from './index.js';\n\nexport interface RetryOptions {\n maxAttempts?: number;\n initialDelay?: number;\n maxDelay?: number;\n backoffFactor?: number;\n timeout?: number;\n onRetry?: (attempt: number, error: unknown) => void;\n}\n\nexport interface CircuitBreakerOptions {\n failureThreshold?: number;\n resetTimeout?: number;\n halfOpenRequests?: number;\n}\n\nexport enum CircuitState {\n CLOSED = 'closed',\n OPEN = 'open',\n HALF_OPEN = 'half_open',\n}\n\n/**\n * Exponential backoff with jitter\n */\nexport function calculateBackoff(\n attempt: number,\n initialDelay: number,\n maxDelay: number,\n factor: number\n): number {\n const exponentialDelay = Math.min(\n initialDelay * Math.pow(factor, attempt - 1),\n maxDelay\n );\n // Add jitter (0-25% of delay)\n const jitter = exponentialDelay * Math.random() * 0.25;\n return Math.floor(exponentialDelay + jitter);\n}\n\n/**\n * Retry with exponential backoff\n */\nexport async function retry<T>(\n fn: () => Promise<T>,\n options: RetryOptions = {}\n): Promise<T> {\n const {\n maxAttempts = 3,\n initialDelay = 1000,\n maxDelay = 30000,\n backoffFactor = 2,\n timeout,\n onRetry,\n } = options;\n\n let lastError: unknown;\n\n for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n try {\n // Add timeout if specified\n if (timeout) {\n return await Promise.race([\n fn(),\n new Promise<T>((_, reject) =>\n setTimeout(\n () => reject(new Error(`Operation timed out after ${timeout}ms`)),\n timeout\n )\n ),\n ]);\n }\n return await fn();\n } catch (error: unknown) {\n lastError = error;\n\n // Don't retry if not retryable or last attempt\n if (!isRetryableError(error) || attempt === maxAttempts) {\n throw error;\n }\n\n const delay = calculateBackoff(attempt, initialDelay, maxDelay, backoffFactor);\n\n logger.warn(`Retry attempt ${attempt}/${maxAttempts} after ${delay}ms`, {\n error: getErrorMessage(error),\n attempt,\n delay,\n });\n\n if (onRetry) {\n onRetry(attempt, error);\n }\n\n await new Promise((resolve) => setTimeout(resolve, delay));\n }\n }\n\n throw lastError;\n}\n\n/**\n * Circuit breaker implementation\n */\nexport class CircuitBreaker<T> {\n private state: CircuitState = CircuitState.CLOSED;\n private failures = 0;\n private successCount = 0;\n private lastFailTime?: Date;\n private readonly options: Required<CircuitBreakerOptions>;\n\n constructor(\n private readonly name: string,\n options: CircuitBreakerOptions = {}\n ) {\n this.options = {\n failureThreshold: options.failureThreshold ?? 5,\n resetTimeout: options.resetTimeout ?? 60000,\n halfOpenRequests: options.halfOpenRequests ?? 3,\n };\n }\n\n async execute<R = T>(fn: () => Promise<R>): Promise<R> {\n // Check if circuit should transition from OPEN to HALF_OPEN\n if (this.state === CircuitState.OPEN) {\n const timeSinceLastFailure = this.lastFailTime\n ? Date.now() - this.lastFailTime.getTime()\n : 0;\n\n if (timeSinceLastFailure >= this.options.resetTimeout) {\n this.state = CircuitState.HALF_OPEN;\n this.successCount = 0;\n logger.info(`Circuit breaker ${this.name} entering half-open state`);\n } else {\n throw new Error(\n `Circuit breaker ${this.name} is OPEN. Retry after ${\n this.options.resetTimeout - timeSinceLastFailure\n }ms`\n );\n }\n }\n\n try {\n const result = await fn();\n\n // Handle success\n if (this.state === CircuitState.HALF_OPEN) {\n this.successCount++;\n if (this.successCount >= this.options.halfOpenRequests) {\n this.state = CircuitState.CLOSED;\n this.failures = 0;\n logger.info(`Circuit breaker ${this.name} is now CLOSED`);\n }\n } else {\n this.failures = 0;\n }\n\n return result;\n } catch (error: unknown) {\n this.handleFailure(error);\n throw error;\n }\n }\n\n private handleFailure(error: unknown): void {\n this.failures++;\n this.lastFailTime = new Date();\n\n if (this.state === CircuitState.HALF_OPEN) {\n this.state = CircuitState.OPEN;\n logger.error(\n `Circuit breaker ${this.name} reopened due to failure in half-open state`\n );\n } else if (\n this.state === CircuitState.CLOSED &&\n this.failures >= this.options.failureThreshold\n ) {\n this.state = CircuitState.OPEN;\n logger.error(\n `Circuit breaker ${this.name} opened after ${this.failures} failures`\n );\n }\n }\n\n getState(): CircuitState {\n return this.state;\n }\n\n reset(): void {\n this.state = CircuitState.CLOSED;\n this.failures = 0;\n this.successCount = 0;\n this.lastFailTime = undefined;\n logger.info(`Circuit breaker ${this.name} manually reset`);\n }\n}\n\n/**\n * Fallback with multiple strategies\n */\nexport async function withFallback<T>(\n primary: () => Promise<T>,\n fallbacks: Array<() => Promise<T>>,\n context?: ErrorContext\n): Promise<T> {\n const errors: unknown[] = [];\n\n // Try primary\n try {\n return await primary();\n } catch (error: unknown) {\n errors.push(error);\n logger.warn('Primary operation failed, trying fallbacks', {\n error: getErrorMessage(error),\n context,\n });\n }\n\n // Try fallbacks in order\n for (let i = 0; i < fallbacks.length; i++) {\n try {\n const result = await fallbacks[i]();\n logger.info(`Fallback ${i + 1} succeeded`, { context });\n return result;\n } catch (error: unknown) {\n errors.push(error);\n if (i < fallbacks.length - 1) {\n logger.warn(`Fallback ${i + 1} failed, trying next`, {\n error: getErrorMessage(error),\n context,\n });\n }\n }\n }\n\n // All attempts failed\n throw new Error(\n `All attempts failed. Errors: ${errors.map(getErrorMessage).join(', ')}`\n );\n}\n\n/**\n * Bulkhead pattern - limit concurrent operations\n */\nexport class Bulkhead {\n private running = 0;\n private queue: Array<() => void> = [];\n\n constructor(\n private readonly name: string,\n private readonly maxConcurrent: number\n ) {}\n\n async execute<T>(fn: () => Promise<T>): Promise<T> {\n if (this.running >= this.maxConcurrent) {\n await new Promise<void>((resolve) => {\n this.queue.push(resolve);\n });\n }\n\n this.running++;\n\n try {\n return await fn();\n } finally {\n this.running--;\n const next = this.queue.shift();\n if (next) {\n next();\n }\n }\n }\n\n getStats() {\n return {\n running: this.running,\n queued: this.queue.length,\n maxConcurrent: this.maxConcurrent,\n };\n }\n}\n\n/**\n * Timeout wrapper with proper cleanup\n */\nexport async function withTimeout<T>(\n fn: () => Promise<T>,\n timeoutMs: number,\n timeoutMessage?: string\n): Promise<T> {\n return Promise.race([\n fn(),\n new Promise<T>((_, reject) =>\n setTimeout(\n () =>\n reject(\n new Error(timeoutMessage ?? `Operation timed out after ${timeoutMs}ms`)\n ),\n timeoutMs\n )\n ),\n ]);\n}\n\n/**\n * Graceful degradation helper\n */\nexport async function gracefulDegrade<T, F>(\n fn: () => Promise<T>,\n defaultValue: F,\n logContext?: ErrorContext\n): Promise<T | F> {\n try {\n return await fn();\n } catch (error: unknown) {\n logger.warn('Operation failed, using default value', {\n error: getErrorMessage(error),\n ...logContext,\n });\n return defaultValue;\n }\n}\n\n/**\n * Create a resilient operation with multiple recovery strategies\n */\nexport function createResilientOperation<T>(\n name: string,\n options: {\n retry?: RetryOptions;\n circuitBreaker?: CircuitBreakerOptions;\n bulkhead?: number;\n timeout?: number;\n fallback?: () => Promise<T>;\n } = {}\n) {\n const circuitBreaker = options.circuitBreaker\n ? new CircuitBreaker<T>(name, options.circuitBreaker)\n : null;\n const bulkhead = options.bulkhead\n ? new Bulkhead(name, options.bulkhead)\n : null;\n\n return async (fn: () => Promise<T>): Promise<T> => {\n let currentFn = fn;\n\n // Wrap with bulkhead if configured\n if (bulkhead) {\n const wrapped = currentFn;\n currentFn = () => bulkhead.execute(wrapped);\n }\n\n // Wrap with timeout if configured\n if (options.timeout) {\n const wrapped = currentFn;\n currentFn = () => withTimeout(wrapped, options.timeout!);\n }\n\n // Wrap with retry if configured\n if (options.retry) {\n const wrapped = currentFn;\n currentFn = () => retry(wrapped, options.retry!);\n }\n\n // Wrap with circuit breaker if configured\n if (circuitBreaker) {\n const wrapped = currentFn;\n currentFn = () => circuitBreaker.execute(wrapped);\n }\n\n // Execute with fallback if configured\n if (options.fallback) {\n return withFallback(currentFn, [options.fallback]);\n }\n\n return currentFn();\n };\n}"],
5
- "mappings": "AAKA,SAAS,cAAc;AACvB;AAAA,EAEE;AAAA,EACA;AAAA,OAEK;AAiBA,IAAK,eAAL,kBAAKA,kBAAL;AACL,EAAAA,cAAA,YAAS;AACT,EAAAA,cAAA,UAAO;AACP,EAAAA,cAAA,eAAY;AAHF,SAAAA;AAAA,GAAA;AASL,SAAS,iBACd,SACA,cACA,UACA,QACQ;AACR,QAAM,mBAAmB,KAAK;AAAA,IAC5B,eAAe,KAAK,IAAI,QAAQ,UAAU,CAAC;AAAA,IAC3C;AAAA,EACF;AAEA,QAAM,SAAS,mBAAmB,KAAK,OAAO,IAAI;AAClD,SAAO,KAAK,MAAM,mBAAmB,MAAM;AAC7C;AAKA,eAAsB,MACpB,IACA,UAAwB,CAAC,GACb;AACZ,QAAM;AAAA,IACJ,cAAc;AAAA,IACd,eAAe;AAAA,IACf,WAAW;AAAA,IACX,gBAAgB;AAAA,IAChB;AAAA,IACA;AAAA,EACF,IAAI;AAEJ,MAAI;AAEJ,WAAS,UAAU,GAAG,WAAW,aAAa,WAAW;AACvD,QAAI;AAEF,UAAI,SAAS;AACX,eAAO,MAAM,QAAQ,KAAK;AAAA,UACxB,GAAG;AAAA,UACH,IAAI;AAAA,YAAW,CAAC,GAAG,WACjB;AAAA,cACE,MAAM,OAAO,IAAI,MAAM,6BAA6B,OAAO,IAAI,CAAC;AAAA,cAChE;AAAA,YACF;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACH;AACA,aAAO,MAAM,GAAG;AAAA,IAClB,SAAS,OAAgB;AACvB,kBAAY;AAGZ,UAAI,CAAC,iBAAiB,KAAK,KAAK,YAAY,aAAa;AACvD,cAAM;AAAA,MACR;AAEA,YAAM,QAAQ,iBAAiB,SAAS,cAAc,UAAU,aAAa;AAE7E,aAAO,KAAK,iBAAiB,OAAO,IAAI,WAAW,UAAU,KAAK,MAAM;AAAA,QACtE,OAAO,gBAAgB,KAAK;AAAA,QAC5B;AAAA,QACA;AAAA,MACF,CAAC;AAED,UAAI,SAAS;AACX,gBAAQ,SAAS,KAAK;AAAA,MACxB;AAEA,YAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,KAAK,CAAC;AAAA,IAC3D;AAAA,EACF;AAEA,QAAM;AACR;AAKO,MAAM,eAAkB;AAAA,EAO7B,YACmB,MACjB,UAAiC,CAAC,GAClC;AAFiB;AAGjB,SAAK,UAAU;AAAA,MACb,kBAAkB,QAAQ,oBAAoB;AAAA,MAC9C,cAAc,QAAQ,gBAAgB;AAAA,MACtC,kBAAkB,QAAQ,oBAAoB;AAAA,IAChD;AAAA,EACF;AAAA,EAfQ,QAAsB;AAAA,EACtB,WAAW;AAAA,EACX,eAAe;AAAA,EACf;AAAA,EACS;AAAA,EAajB,MAAM,QAAe,IAAkC;AAErD,QAAI,KAAK,UAAU,mBAAmB;AACpC,YAAM,uBAAuB,KAAK,eAC9B,KAAK,IAAI,IAAI,KAAK,aAAa,QAAQ,IACvC;AAEJ,UAAI,wBAAwB,KAAK,QAAQ,cAAc;AACrD,aAAK,QAAQ;AACb,aAAK,eAAe;AACpB,eAAO,KAAK,mBAAmB,KAAK,IAAI,2BAA2B;AAAA,MACrE,OAAO;AACL,cAAM,IAAI;AAAA,UACR,mBAAmB,KAAK,IAAI,yBAC1B,KAAK,QAAQ,eAAe,oBAC9B;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAEA,QAAI;AACF,YAAM,SAAS,MAAM,GAAG;AAGxB,UAAI,KAAK,UAAU,6BAAwB;AACzC,aAAK;AACL,YAAI,KAAK,gBAAgB,KAAK,QAAQ,kBAAkB;AACtD,eAAK,QAAQ;AACb,eAAK,WAAW;AAChB,iBAAO,KAAK,mBAAmB,KAAK,IAAI,gBAAgB;AAAA,QAC1D;AAAA,MACF,OAAO;AACL,aAAK,WAAW;AAAA,MAClB;AAEA,aAAO;AAAA,IACT,SAAS,OAAgB;AACvB,WAAK,cAAc,KAAK;AACxB,YAAM;AAAA,IACR;AAAA,EACF;AAAA,EAEQ,cAAc,OAAsB;AAC1C,SAAK;AACL,SAAK,eAAe,oBAAI,KAAK;AAE7B,QAAI,KAAK,UAAU,6BAAwB;AACzC,WAAK,QAAQ;AACb,aAAO;AAAA,QACL,mBAAmB,KAAK,IAAI;AAAA,MAC9B;AAAA,IACF,WACE,KAAK,UAAU,yBACf,KAAK,YAAY,KAAK,QAAQ,kBAC9B;AACA,WAAK,QAAQ;AACb,aAAO;AAAA,QACL,mBAAmB,KAAK,IAAI,iBAAiB,KAAK,QAAQ;AAAA,MAC5D;AAAA,IACF;AAAA,EACF;AAAA,EAEA,WAAyB;AACvB,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,QAAc;AACZ,SAAK,QAAQ;AACb,SAAK,WAAW;AAChB,SAAK,eAAe;AACpB,SAAK,eAAe;AACpB,WAAO,KAAK,mBAAmB,KAAK,IAAI,iBAAiB;AAAA,EAC3D;AACF;AAKA,eAAsB,aACpB,SACA,WACA,SACY;AACZ,QAAM,SAAoB,CAAC;AAG3B,MAAI;AACF,WAAO,MAAM,QAAQ;AAAA,EACvB,SAAS,OAAgB;AACvB,WAAO,KAAK,KAAK;AACjB,WAAO,KAAK,8CAA8C;AAAA,MACxD,OAAO,gBAAgB,KAAK;AAAA,MAC5B;AAAA,IACF,CAAC;AAAA,EACH;AAGA,WAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,QAAI;AACF,YAAM,SAAS,MAAM,UAAU,CAAC,EAAE;AAClC,aAAO,KAAK,YAAY,IAAI,CAAC,cAAc,EAAE,QAAQ,CAAC;AACtD,aAAO;AAAA,IACT,SAAS,OAAgB;AACvB,aAAO,KAAK,KAAK;AACjB,UAAI,IAAI,UAAU,SAAS,GAAG;AAC5B,eAAO,KAAK,YAAY,IAAI,CAAC,wBAAwB;AAAA,UACnD,OAAO,gBAAgB,KAAK;AAAA,UAC5B;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAGA,QAAM,IAAI;AAAA,IACR,gCAAgC,OAAO,IAAI,eAAe,EAAE,KAAK,IAAI,CAAC;AAAA,EACxE;AACF;AAKO,MAAM,SAAS;AAAA,EAIpB,YACmB,MACA,eACjB;AAFiB;AACA;AAAA,EAChB;AAAA,EANK,UAAU;AAAA,EACV,QAA2B,CAAC;AAAA,EAOpC,MAAM,QAAW,IAAkC;AACjD,QAAI,KAAK,WAAW,KAAK,eAAe;AACtC,YAAM,IAAI,QAAc,CAAC,YAAY;AACnC,aAAK,MAAM,KAAK,OAAO;AAAA,MACzB,CAAC;AAAA,IACH;AAEA,SAAK;AAEL,QAAI;AACF,aAAO,MAAM,GAAG;AAAA,IAClB,UAAE;AACA,WAAK;AACL,YAAM,OAAO,KAAK,MAAM,MAAM;AAC9B,UAAI,MAAM;AACR,aAAK;AAAA,MACP;AAAA,IACF;AAAA,EACF;AAAA,EAEA,WAAW;AACT,WAAO;AAAA,MACL,SAAS,KAAK;AAAA,MACd,QAAQ,KAAK,MAAM;AAAA,MACnB,eAAe,KAAK;AAAA,IACtB;AAAA,EACF;AACF;AAKA,eAAsB,YACpB,IACA,WACA,gBACY;AACZ,SAAO,QAAQ,KAAK;AAAA,IAClB,GAAG;AAAA,IACH,IAAI;AAAA,MAAW,CAAC,GAAG,WACjB;AAAA,QACE,MACE;AAAA,UACE,IAAI,MAAM,kBAAkB,6BAA6B,SAAS,IAAI;AAAA,QACxE;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF,CAAC;AACH;AAKA,eAAsB,gBACpB,IACA,cACA,YACgB;AAChB,MAAI;AACF,WAAO,MAAM,GAAG;AAAA,EAClB,SAAS,OAAgB;AACvB,WAAO,KAAK,yCAAyC;AAAA,MACnD,OAAO,gBAAgB,KAAK;AAAA,MAC5B,GAAG;AAAA,IACL,CAAC;AACD,WAAO;AAAA,EACT;AACF;AAKO,SAAS,yBACd,MACA,UAMI,CAAC,GACL;AACA,QAAM,iBAAiB,QAAQ,iBAC3B,IAAI,eAAkB,MAAM,QAAQ,cAAc,IAClD;AACJ,QAAM,WAAW,QAAQ,WACrB,IAAI,SAAS,MAAM,QAAQ,QAAQ,IACnC;AAEJ,SAAO,OAAO,OAAqC;AACjD,QAAI,YAAY;AAGhB,QAAI,UAAU;AACZ,YAAM,UAAU;AAChB,kBAAY,MAAM,SAAS,QAAQ,OAAO;AAAA,IAC5C;AAGA,QAAI,QAAQ,SAAS;AACnB,YAAM,UAAU;AAChB,kBAAY,MAAM,YAAY,SAAS,QAAQ,OAAQ;AAAA,IACzD;AAGA,QAAI,QAAQ,OAAO;AACjB,YAAM,UAAU;AAChB,kBAAY,MAAM,MAAM,SAAS,QAAQ,KAAM;AAAA,IACjD;AAGA,QAAI,gBAAgB;AAClB,YAAM,UAAU;AAChB,kBAAY,MAAM,eAAe,QAAQ,OAAO;AAAA,IAClD;AAGA,QAAI,QAAQ,UAAU;AACpB,aAAO,aAAa,WAAW,CAAC,QAAQ,QAAQ,CAAC;AAAA,IACnD;AAEA,WAAO,UAAU;AAAA,EACnB;AACF;",
4
+ "sourcesContent": ["/**\n * Error recovery utilities for StackMemory\n * Provides retry logic, circuit breakers, and fallback mechanisms\n */\n\nimport { logger } from '../monitoring/logger.js';\nimport {\n StackMemoryError,\n isRetryableError,\n getErrorMessage,\n ErrorContext,\n} from './index.js';\n\nexport interface RetryOptions {\n maxAttempts?: number;\n initialDelay?: number;\n maxDelay?: number;\n backoffFactor?: number;\n timeout?: number;\n onRetry?: (attempt: number, error: unknown) => void;\n}\n\nexport interface CircuitBreakerOptions {\n failureThreshold?: number;\n resetTimeout?: number;\n halfOpenRequests?: number;\n}\n\nexport enum CircuitState {\n CLOSED = 'closed',\n OPEN = 'open',\n HALF_OPEN = 'half_open',\n}\n\n/**\n * Exponential backoff with jitter\n */\nexport function calculateBackoff(\n attempt: number,\n initialDelay: number,\n maxDelay: number,\n factor: number\n): number {\n const exponentialDelay = Math.min(\n initialDelay * Math.pow(factor, attempt - 1),\n maxDelay\n );\n // Add jitter (0-25% of delay)\n const jitter = exponentialDelay * Math.random() * 0.25;\n return Math.floor(exponentialDelay + jitter);\n}\n\n/**\n * Retry with exponential backoff\n */\nexport async function retry<T>(\n fn: () => Promise<T>,\n options: RetryOptions = {}\n): Promise<T> {\n const {\n maxAttempts = 3,\n initialDelay = 1000,\n maxDelay = 30000,\n backoffFactor = 2,\n timeout,\n onRetry,\n } = options;\n\n let lastError: unknown;\n\n for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n try {\n // Add timeout if specified\n if (timeout) {\n return await Promise.race([\n fn(),\n new Promise<T>((_, reject) =>\n setTimeout(\n () => reject(new Error(`Operation timed out after ${timeout}ms`)),\n timeout\n )\n ),\n ]);\n }\n return await fn();\n } catch (error: unknown) {\n lastError = error;\n\n // Don't retry if not retryable or last attempt\n if (!isRetryableError(error) || attempt === maxAttempts) {\n throw error;\n }\n\n const delay = calculateBackoff(\n attempt,\n initialDelay,\n maxDelay,\n backoffFactor\n );\n\n logger.warn(`Retry attempt ${attempt}/${maxAttempts} after ${delay}ms`, {\n error: getErrorMessage(error),\n attempt,\n delay,\n });\n\n if (onRetry) {\n onRetry(attempt, error);\n }\n\n await new Promise((resolve) => setTimeout(resolve, delay));\n }\n }\n\n throw lastError;\n}\n\n/**\n * Circuit breaker implementation\n */\nexport class CircuitBreaker<T> {\n private state: CircuitState = CircuitState.CLOSED;\n private failures = 0;\n private successCount = 0;\n private lastFailTime?: Date;\n private readonly options: Required<CircuitBreakerOptions>;\n\n constructor(\n private readonly name: string,\n options: CircuitBreakerOptions = {}\n ) {\n this.options = {\n failureThreshold: options.failureThreshold ?? 5,\n resetTimeout: options.resetTimeout ?? 60000,\n halfOpenRequests: options.halfOpenRequests ?? 3,\n };\n }\n\n async execute<R = T>(fn: () => Promise<R>): Promise<R> {\n // Check if circuit should transition from OPEN to HALF_OPEN\n if (this.state === CircuitState.OPEN) {\n const timeSinceLastFailure = this.lastFailTime\n ? Date.now() - this.lastFailTime.getTime()\n : 0;\n\n if (timeSinceLastFailure >= this.options.resetTimeout) {\n this.state = CircuitState.HALF_OPEN;\n this.successCount = 0;\n logger.info(`Circuit breaker ${this.name} entering half-open state`);\n } else {\n throw new Error(\n `Circuit breaker ${this.name} is OPEN. Retry after ${\n this.options.resetTimeout - timeSinceLastFailure\n }ms`\n );\n }\n }\n\n try {\n const result = await fn();\n\n // Handle success\n if (this.state === CircuitState.HALF_OPEN) {\n this.successCount++;\n if (this.successCount >= this.options.halfOpenRequests) {\n this.state = CircuitState.CLOSED;\n this.failures = 0;\n logger.info(`Circuit breaker ${this.name} is now CLOSED`);\n }\n } else {\n this.failures = 0;\n }\n\n return result;\n } catch (error: unknown) {\n this.handleFailure(error);\n throw error;\n }\n }\n\n private handleFailure(error: unknown): void {\n this.failures++;\n this.lastFailTime = new Date();\n\n if (this.state === CircuitState.HALF_OPEN) {\n this.state = CircuitState.OPEN;\n logger.error(\n `Circuit breaker ${this.name} reopened due to failure in half-open state`\n );\n } else if (\n this.state === CircuitState.CLOSED &&\n this.failures >= this.options.failureThreshold\n ) {\n this.state = CircuitState.OPEN;\n logger.error(\n `Circuit breaker ${this.name} opened after ${this.failures} failures`\n );\n }\n }\n\n getState(): CircuitState {\n return this.state;\n }\n\n reset(): void {\n this.state = CircuitState.CLOSED;\n this.failures = 0;\n this.successCount = 0;\n this.lastFailTime = undefined;\n logger.info(`Circuit breaker ${this.name} manually reset`);\n }\n}\n\n/**\n * Fallback with multiple strategies\n */\nexport async function withFallback<T>(\n primary: () => Promise<T>,\n fallbacks: Array<() => Promise<T>>,\n context?: ErrorContext\n): Promise<T> {\n const errors: unknown[] = [];\n\n // Try primary\n try {\n return await primary();\n } catch (error: unknown) {\n errors.push(error);\n logger.warn('Primary operation failed, trying fallbacks', {\n error: getErrorMessage(error),\n context,\n });\n }\n\n // Try fallbacks in order\n for (let i = 0; i < fallbacks.length; i++) {\n try {\n const result = await fallbacks[i]();\n logger.info(`Fallback ${i + 1} succeeded`, { context });\n return result;\n } catch (error: unknown) {\n errors.push(error);\n if (i < fallbacks.length - 1) {\n logger.warn(`Fallback ${i + 1} failed, trying next`, {\n error: getErrorMessage(error),\n context,\n });\n }\n }\n }\n\n // All attempts failed\n throw new Error(\n `All attempts failed. Errors: ${errors.map(getErrorMessage).join(', ')}`\n );\n}\n\n/**\n * Bulkhead pattern - limit concurrent operations\n */\nexport class Bulkhead {\n private running = 0;\n private queue: Array<() => void> = [];\n\n constructor(\n private readonly name: string,\n private readonly maxConcurrent: number\n ) {}\n\n async execute<T>(fn: () => Promise<T>): Promise<T> {\n if (this.running >= this.maxConcurrent) {\n await new Promise<void>((resolve) => {\n this.queue.push(resolve);\n });\n }\n\n this.running++;\n\n try {\n return await fn();\n } finally {\n this.running--;\n const next = this.queue.shift();\n if (next) {\n next();\n }\n }\n }\n\n getStats() {\n return {\n running: this.running,\n queued: this.queue.length,\n maxConcurrent: this.maxConcurrent,\n };\n }\n}\n\n/**\n * Timeout wrapper with proper cleanup\n */\nexport async function withTimeout<T>(\n fn: () => Promise<T>,\n timeoutMs: number,\n timeoutMessage?: string\n): Promise<T> {\n return Promise.race([\n fn(),\n new Promise<T>((_, reject) =>\n setTimeout(\n () =>\n reject(\n new Error(\n timeoutMessage ?? `Operation timed out after ${timeoutMs}ms`\n )\n ),\n timeoutMs\n )\n ),\n ]);\n}\n\n/**\n * Graceful degradation helper\n */\nexport async function gracefulDegrade<T, F>(\n fn: () => Promise<T>,\n defaultValue: F,\n logContext?: ErrorContext\n): Promise<T | F> {\n try {\n return await fn();\n } catch (error: unknown) {\n logger.warn('Operation failed, using default value', {\n error: getErrorMessage(error),\n ...logContext,\n });\n return defaultValue;\n }\n}\n\n/**\n * Create a resilient operation with multiple recovery strategies\n */\nexport function createResilientOperation<T>(\n name: string,\n options: {\n retry?: RetryOptions;\n circuitBreaker?: CircuitBreakerOptions;\n bulkhead?: number;\n timeout?: number;\n fallback?: () => Promise<T>;\n } = {}\n) {\n const circuitBreaker = options.circuitBreaker\n ? new CircuitBreaker<T>(name, options.circuitBreaker)\n : null;\n const bulkhead = options.bulkhead\n ? new Bulkhead(name, options.bulkhead)\n : null;\n\n return async (fn: () => Promise<T>): Promise<T> => {\n let currentFn = fn;\n\n // Wrap with bulkhead if configured\n if (bulkhead) {\n const wrapped = currentFn;\n currentFn = () => bulkhead.execute(wrapped);\n }\n\n // Wrap with timeout if configured\n if (options.timeout) {\n const wrapped = currentFn;\n currentFn = () => withTimeout(wrapped, options.timeout!);\n }\n\n // Wrap with retry if configured\n if (options.retry) {\n const wrapped = currentFn;\n currentFn = () => retry(wrapped, options.retry!);\n }\n\n // Wrap with circuit breaker if configured\n if (circuitBreaker) {\n const wrapped = currentFn;\n currentFn = () => circuitBreaker.execute(wrapped);\n }\n\n // Execute with fallback if configured\n if (options.fallback) {\n return withFallback(currentFn, [options.fallback]);\n }\n\n return currentFn();\n };\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["CircuitState"]
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/frame/workflow-templates-stub.ts"],
4
- "sourcesContent": ["/**\n * Stub workflow templates for testing\n */\n\nexport const workflowTemplates = {\n tdd: {\n name: 'Test-Driven Development',\n description: 'Write tests first, then implement',\n phases: [\n { name: 'write-failing-tests', description: 'Write tests that fail' },\n { name: 'implement-code', description: 'Make tests pass' },\n { name: 'refactor', description: 'Clean up code' }\n ]\n },\n feature: {\n name: 'Feature Development',\n description: 'Develop a new feature',\n phases: [\n { name: 'design', description: 'Design the feature' },\n { name: 'implement', description: 'Build the feature' },\n { name: 'test', description: 'Test the feature' }\n ]\n },\n bugfix: {\n name: 'Bug Fix',\n description: 'Fix a reported bug',\n phases: [\n { name: 'reproduce', description: 'Reproduce the bug' },\n { name: 'fix', description: 'Fix the bug' },\n { name: 'verify', description: 'Verify the fix' }\n ]\n },\n refactor: {\n name: 'Refactoring',\n description: 'Improve code structure',\n phases: [\n { name: 'analyze', description: 'Analyze current code' },\n { name: 'refactor', description: 'Refactor code' },\n { name: 'test', description: 'Ensure no regressions' }\n ]\n }\n};"],
4
+ "sourcesContent": ["/**\n * Stub workflow templates for testing\n */\n\nexport const workflowTemplates = {\n tdd: {\n name: 'Test-Driven Development',\n description: 'Write tests first, then implement',\n phases: [\n { name: 'write-failing-tests', description: 'Write tests that fail' },\n { name: 'implement-code', description: 'Make tests pass' },\n { name: 'refactor', description: 'Clean up code' },\n ],\n },\n feature: {\n name: 'Feature Development',\n description: 'Develop a new feature',\n phases: [\n { name: 'design', description: 'Design the feature' },\n { name: 'implement', description: 'Build the feature' },\n { name: 'test', description: 'Test the feature' },\n ],\n },\n bugfix: {\n name: 'Bug Fix',\n description: 'Fix a reported bug',\n phases: [\n { name: 'reproduce', description: 'Reproduce the bug' },\n { name: 'fix', description: 'Fix the bug' },\n { name: 'verify', description: 'Verify the fix' },\n ],\n },\n refactor: {\n name: 'Refactoring',\n description: 'Improve code structure',\n phases: [\n { name: 'analyze', description: 'Analyze current code' },\n { name: 'refactor', description: 'Refactor code' },\n { name: 'test', description: 'Ensure no regressions' },\n ],\n },\n};\n"],
5
5
  "mappings": "AAIO,MAAM,oBAAoB;AAAA,EAC/B,KAAK;AAAA,IACH,MAAM;AAAA,IACN,aAAa;AAAA,IACb,QAAQ;AAAA,MACN,EAAE,MAAM,uBAAuB,aAAa,wBAAwB;AAAA,MACpE,EAAE,MAAM,kBAAkB,aAAa,kBAAkB;AAAA,MACzD,EAAE,MAAM,YAAY,aAAa,gBAAgB;AAAA,IACnD;AAAA,EACF;AAAA,EACA,SAAS;AAAA,IACP,MAAM;AAAA,IACN,aAAa;AAAA,IACb,QAAQ;AAAA,MACN,EAAE,MAAM,UAAU,aAAa,qBAAqB;AAAA,MACpD,EAAE,MAAM,aAAa,aAAa,oBAAoB;AAAA,MACtD,EAAE,MAAM,QAAQ,aAAa,mBAAmB;AAAA,IAClD;AAAA,EACF;AAAA,EACA,QAAQ;AAAA,IACN,MAAM;AAAA,IACN,aAAa;AAAA,IACb,QAAQ;AAAA,MACN,EAAE,MAAM,aAAa,aAAa,oBAAoB;AAAA,MACtD,EAAE,MAAM,OAAO,aAAa,cAAc;AAAA,MAC1C,EAAE,MAAM,UAAU,aAAa,iBAAiB;AAAA,IAClD;AAAA,EACF;AAAA,EACA,UAAU;AAAA,IACR,MAAM;AAAA,IACN,aAAa;AAAA,IACb,QAAQ;AAAA,MACN,EAAE,MAAM,WAAW,aAAa,uBAAuB;AAAA,MACvD,EAAE,MAAM,YAAY,aAAa,gBAAgB;AAAA,MACjD,EAAE,MAAM,QAAQ,aAAa,wBAAwB;AAAA,IACvD;AAAA,EACF;AACF;",
6
6
  "names": []
7
7
  }
@@ -270,7 +270,46 @@ class WorkflowTemplates {
270
270
  ];
271
271
  }
272
272
  }
273
+ const workflowTemplates = {
274
+ tdd: {
275
+ name: "Test-Driven Development",
276
+ description: "Write tests first, then implement",
277
+ phases: [
278
+ { name: "write-failing-tests", description: "Write tests that fail" },
279
+ { name: "implement-code", description: "Make tests pass" },
280
+ { name: "refactor", description: "Clean up code" }
281
+ ]
282
+ },
283
+ feature: {
284
+ name: "Feature Development",
285
+ description: "Develop a new feature",
286
+ phases: [
287
+ { name: "design", description: "Design the feature" },
288
+ { name: "implement", description: "Build the feature" },
289
+ { name: "test", description: "Test the feature" }
290
+ ]
291
+ },
292
+ bugfix: {
293
+ name: "Bug Fix",
294
+ description: "Fix a reported bug",
295
+ phases: [
296
+ { name: "reproduce", description: "Reproduce the bug" },
297
+ { name: "fix", description: "Fix the bug" },
298
+ { name: "verify", description: "Verify the fix" }
299
+ ]
300
+ },
301
+ refactor: {
302
+ name: "Refactoring",
303
+ description: "Improve code structure",
304
+ phases: [
305
+ { name: "analyze", description: "Analyze current code" },
306
+ { name: "refactor", description: "Refactor code" },
307
+ { name: "test", description: "Ensure no regressions" }
308
+ ]
309
+ }
310
+ };
273
311
  export {
274
- WorkflowTemplates
312
+ WorkflowTemplates,
313
+ workflowTemplates
275
314
  };
276
315
  //# sourceMappingURL=workflow-templates.js.map