@stackmemoryai/stackmemory 0.3.6 → 0.3.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (213) hide show
  1. package/dist/agents/verifiers/base-verifier.js.map +2 -2
  2. package/dist/agents/verifiers/formatter-verifier.js.map +2 -2
  3. package/dist/agents/verifiers/llm-judge.js.map +2 -2
  4. package/dist/cli/claude-sm.js +24 -13
  5. package/dist/cli/claude-sm.js.map +2 -2
  6. package/dist/cli/codex-sm.js +24 -13
  7. package/dist/cli/codex-sm.js.map +2 -2
  8. package/dist/cli/commands/agent.js.map +2 -2
  9. package/dist/cli/commands/chromadb.js +217 -32
  10. package/dist/cli/commands/chromadb.js.map +2 -2
  11. package/dist/cli/commands/clear.js +12 -1
  12. package/dist/cli/commands/clear.js.map +2 -2
  13. package/dist/cli/commands/context.js +13 -2
  14. package/dist/cli/commands/context.js.map +2 -2
  15. package/dist/cli/commands/dashboard.js.map +2 -2
  16. package/dist/cli/commands/gc.js +202 -0
  17. package/dist/cli/commands/gc.js.map +7 -0
  18. package/dist/cli/commands/handoff.js +12 -1
  19. package/dist/cli/commands/handoff.js.map +2 -2
  20. package/dist/cli/commands/infinite-storage.js +32 -21
  21. package/dist/cli/commands/infinite-storage.js.map +2 -2
  22. package/dist/cli/commands/linear-create.js +13 -2
  23. package/dist/cli/commands/linear-create.js.map +2 -2
  24. package/dist/cli/commands/linear-list.js +12 -1
  25. package/dist/cli/commands/linear-list.js.map +2 -2
  26. package/dist/cli/commands/linear-migrate.js +12 -1
  27. package/dist/cli/commands/linear-migrate.js.map +2 -2
  28. package/dist/cli/commands/linear-test.js +12 -1
  29. package/dist/cli/commands/linear-test.js.map +2 -2
  30. package/dist/cli/commands/linear-unified.js +262 -0
  31. package/dist/cli/commands/linear-unified.js.map +7 -0
  32. package/dist/cli/commands/linear.js +17 -6
  33. package/dist/cli/commands/linear.js.map +2 -2
  34. package/dist/cli/commands/monitor.js.map +2 -2
  35. package/dist/cli/commands/onboard.js.map +2 -2
  36. package/dist/cli/commands/quality.js.map +2 -2
  37. package/dist/cli/commands/search.js.map +2 -2
  38. package/dist/cli/commands/session.js.map +2 -2
  39. package/dist/cli/commands/skills.js +12 -1
  40. package/dist/cli/commands/skills.js.map +2 -2
  41. package/dist/cli/commands/storage.js +18 -7
  42. package/dist/cli/commands/storage.js.map +2 -2
  43. package/dist/cli/commands/tasks.js.map +2 -2
  44. package/dist/cli/commands/tui.js +13 -2
  45. package/dist/cli/commands/tui.js.map +2 -2
  46. package/dist/cli/commands/webhook.js +14 -3
  47. package/dist/cli/commands/webhook.js.map +2 -2
  48. package/dist/cli/commands/workflow.js +14 -3
  49. package/dist/cli/commands/workflow.js.map +2 -2
  50. package/dist/cli/commands/worktree.js.map +2 -2
  51. package/dist/cli/index.js +18 -5
  52. package/dist/cli/index.js.map +2 -2
  53. package/dist/core/config/config-manager.js.map +2 -2
  54. package/dist/core/context/auto-context.js.map +2 -2
  55. package/dist/core/context/compaction-handler.js.map +2 -2
  56. package/dist/core/context/context-bridge.js.map +2 -2
  57. package/dist/core/context/dual-stack-manager.js.map +2 -2
  58. package/dist/core/context/frame-database.js.map +2 -2
  59. package/dist/core/context/frame-digest.js.map +2 -2
  60. package/dist/core/context/frame-handoff-manager.js.map +2 -2
  61. package/dist/core/context/frame-manager.js +12 -1
  62. package/dist/core/context/frame-manager.js.map +2 -2
  63. package/dist/core/context/frame-stack.js.map +2 -2
  64. package/dist/core/context/incremental-gc.js +279 -0
  65. package/dist/core/context/incremental-gc.js.map +7 -0
  66. package/dist/core/context/permission-manager.js +12 -1
  67. package/dist/core/context/permission-manager.js.map +2 -2
  68. package/dist/core/context/refactored-frame-manager.js.map +2 -2
  69. package/dist/core/context/shared-context-layer.js +12 -1
  70. package/dist/core/context/shared-context-layer.js.map +2 -2
  71. package/dist/core/context/stack-merge-resolver.js.map +2 -2
  72. package/dist/core/context/validation.js.map +2 -2
  73. package/dist/core/database/batch-operations.js.map +2 -2
  74. package/dist/core/database/connection-pool.js.map +2 -2
  75. package/dist/core/database/migration-manager.js.map +2 -2
  76. package/dist/core/database/paradedb-adapter.js.map +2 -2
  77. package/dist/core/database/query-cache.js.map +2 -2
  78. package/dist/core/database/query-router.js.map +2 -2
  79. package/dist/core/database/sqlite-adapter.js.map +2 -2
  80. package/dist/core/digest/enhanced-hybrid-digest.js.map +2 -2
  81. package/dist/core/errors/recovery.js.map +2 -2
  82. package/dist/core/merge/resolution-engine.js.map +2 -2
  83. package/dist/core/monitoring/error-handler.js.map +2 -2
  84. package/dist/core/monitoring/logger.js +14 -3
  85. package/dist/core/monitoring/logger.js.map +2 -2
  86. package/dist/core/monitoring/metrics.js +13 -2
  87. package/dist/core/monitoring/metrics.js.map +2 -2
  88. package/dist/core/monitoring/progress-tracker.js +12 -1
  89. package/dist/core/monitoring/progress-tracker.js.map +2 -2
  90. package/dist/core/monitoring/session-monitor.js.map +2 -2
  91. package/dist/core/performance/context-cache.js.map +2 -2
  92. package/dist/core/performance/lazy-context-loader.js.map +2 -2
  93. package/dist/core/performance/monitor.js.map +2 -2
  94. package/dist/core/performance/optimized-frame-context.js.map +2 -2
  95. package/dist/core/performance/performance-benchmark.js.map +2 -2
  96. package/dist/core/performance/performance-profiler.js +12 -1
  97. package/dist/core/performance/performance-profiler.js.map +2 -2
  98. package/dist/core/performance/streaming-jsonl-parser.js.map +2 -2
  99. package/dist/core/persistence/postgres-adapter.js.map +2 -2
  100. package/dist/core/projects/project-manager.js.map +2 -2
  101. package/dist/core/retrieval/context-retriever.js.map +2 -2
  102. package/dist/core/retrieval/graph-retrieval.js.map +2 -2
  103. package/dist/core/retrieval/llm-context-retrieval.js.map +2 -2
  104. package/dist/core/retrieval/retrieval-benchmarks.js.map +2 -2
  105. package/dist/core/retrieval/summary-generator.js.map +2 -2
  106. package/dist/core/session/clear-survival.js.map +2 -2
  107. package/dist/core/session/handoff-generator.js.map +2 -2
  108. package/dist/core/session/session-manager.js +16 -5
  109. package/dist/core/session/session-manager.js.map +2 -2
  110. package/dist/core/skills/skill-storage.js +13 -2
  111. package/dist/core/skills/skill-storage.js.map +2 -2
  112. package/dist/core/storage/chromadb-adapter.js.map +2 -2
  113. package/dist/core/storage/chromadb-simple.js.map +2 -2
  114. package/dist/core/storage/infinite-storage.js.map +2 -2
  115. package/dist/core/storage/railway-optimized-storage.js +19 -8
  116. package/dist/core/storage/railway-optimized-storage.js.map +2 -2
  117. package/dist/core/storage/remote-storage.js +12 -1
  118. package/dist/core/storage/remote-storage.js.map +2 -2
  119. package/dist/core/trace/cli-trace-wrapper.js +16 -5
  120. package/dist/core/trace/cli-trace-wrapper.js.map +2 -2
  121. package/dist/core/trace/db-trace-wrapper.js.map +2 -2
  122. package/dist/core/trace/debug-trace.js +21 -10
  123. package/dist/core/trace/debug-trace.js.map +2 -2
  124. package/dist/core/trace/index.js +46 -35
  125. package/dist/core/trace/index.js.map +2 -2
  126. package/dist/core/trace/trace-demo.js +12 -1
  127. package/dist/core/trace/trace-demo.js.map +2 -2
  128. package/dist/core/trace/trace-detector.js.map +2 -2
  129. package/dist/core/trace/trace-store.js.map +2 -2
  130. package/dist/core/utils/update-checker.js.map +2 -2
  131. package/dist/core/worktree/worktree-manager.js.map +2 -2
  132. package/dist/features/analytics/api/analytics-api.js.map +2 -2
  133. package/dist/features/analytics/core/analytics-service.js +12 -1
  134. package/dist/features/analytics/core/analytics-service.js.map +2 -2
  135. package/dist/features/analytics/queries/metrics-queries.js.map +2 -2
  136. package/dist/features/tasks/pebbles-task-store.js.map +2 -2
  137. package/dist/features/tui/components/analytics-panel.js.map +2 -2
  138. package/dist/features/tui/components/pr-tracker.js.map +2 -2
  139. package/dist/features/tui/components/session-monitor.js.map +2 -2
  140. package/dist/features/tui/components/subagent-fleet.js.map +2 -2
  141. package/dist/features/tui/components/task-board.js +650 -2
  142. package/dist/features/tui/components/task-board.js.map +2 -2
  143. package/dist/features/tui/index.js +16 -5
  144. package/dist/features/tui/index.js.map +2 -2
  145. package/dist/features/tui/services/data-service.js +25 -14
  146. package/dist/features/tui/services/data-service.js.map +2 -2
  147. package/dist/features/tui/services/linear-task-reader.js.map +2 -2
  148. package/dist/features/tui/services/websocket-client.js +13 -2
  149. package/dist/features/tui/services/websocket-client.js.map +2 -2
  150. package/dist/features/tui/terminal-compat.js +27 -16
  151. package/dist/features/tui/terminal-compat.js.map +2 -2
  152. package/dist/features/web/client/stores/task-store.js.map +2 -2
  153. package/dist/features/web/server/index.js +13 -2
  154. package/dist/features/web/server/index.js.map +2 -2
  155. package/dist/integrations/claude-code/enhanced-pre-clear-hooks.js.map +2 -2
  156. package/dist/integrations/claude-code/lifecycle-hooks.js.map +2 -2
  157. package/dist/integrations/claude-code/post-task-hooks.js.map +2 -2
  158. package/dist/integrations/linear/auth.js +17 -6
  159. package/dist/integrations/linear/auth.js.map +2 -2
  160. package/dist/integrations/linear/auto-sync.js.map +2 -2
  161. package/dist/integrations/linear/client.js.map +2 -2
  162. package/dist/integrations/linear/config.js.map +2 -2
  163. package/dist/integrations/linear/migration.js.map +2 -2
  164. package/dist/integrations/linear/oauth-server.js +13 -2
  165. package/dist/integrations/linear/oauth-server.js.map +2 -2
  166. package/dist/integrations/linear/rest-client.js.map +2 -2
  167. package/dist/integrations/linear/sync-enhanced.js +202 -0
  168. package/dist/integrations/linear/sync-enhanced.js.map +7 -0
  169. package/dist/integrations/linear/sync-manager.js.map +2 -2
  170. package/dist/integrations/linear/sync-service.js +12 -1
  171. package/dist/integrations/linear/sync-service.js.map +2 -2
  172. package/dist/integrations/linear/sync.js +34 -3
  173. package/dist/integrations/linear/sync.js.map +2 -2
  174. package/dist/integrations/linear/unified-sync.js +560 -0
  175. package/dist/integrations/linear/unified-sync.js.map +7 -0
  176. package/dist/integrations/linear/webhook-handler.js +12 -1
  177. package/dist/integrations/linear/webhook-handler.js.map +2 -2
  178. package/dist/integrations/linear/webhook-server.js +14 -3
  179. package/dist/integrations/linear/webhook-server.js.map +2 -2
  180. package/dist/integrations/linear/webhook.js +12 -1
  181. package/dist/integrations/linear/webhook.js.map +2 -2
  182. package/dist/integrations/mcp/handlers/context-handlers.js.map +2 -2
  183. package/dist/integrations/mcp/handlers/linear-handlers.js.map +2 -2
  184. package/dist/integrations/mcp/handlers/skill-handlers.js +13 -2
  185. package/dist/integrations/mcp/handlers/skill-handlers.js.map +2 -2
  186. package/dist/integrations/mcp/handlers/task-handlers.js.map +2 -2
  187. package/dist/integrations/mcp/handlers/trace-handlers.js.map +2 -2
  188. package/dist/integrations/mcp/middleware/tool-scoring.js.map +2 -2
  189. package/dist/integrations/mcp/refactored-server.js +15 -4
  190. package/dist/integrations/mcp/refactored-server.js.map +2 -2
  191. package/dist/integrations/mcp/server.js +12 -1
  192. package/dist/integrations/mcp/server.js.map +2 -2
  193. package/dist/integrations/mcp/tool-definitions.js.map +2 -2
  194. package/dist/integrations/pg-aiguide/embedding-provider.js +13 -2
  195. package/dist/integrations/pg-aiguide/embedding-provider.js.map +2 -2
  196. package/dist/integrations/pg-aiguide/semantic-search.js.map +2 -2
  197. package/dist/mcp/stackmemory-mcp-server.js +12 -1
  198. package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
  199. package/dist/middleware/exponential-rate-limiter.js.map +2 -2
  200. package/dist/servers/production/auth-middleware.js +13 -2
  201. package/dist/servers/production/auth-middleware.js.map +2 -2
  202. package/dist/servers/railway/index.js +22 -11
  203. package/dist/servers/railway/index.js.map +2 -2
  204. package/dist/services/config-service.js.map +2 -2
  205. package/dist/services/context-service.js.map +2 -2
  206. package/dist/skills/claude-skills.js +105 -2
  207. package/dist/skills/claude-skills.js.map +2 -2
  208. package/dist/skills/dashboard-launcher.js.map +2 -2
  209. package/dist/skills/repo-ingestion-skill.js +561 -0
  210. package/dist/skills/repo-ingestion-skill.js.map +7 -0
  211. package/dist/utils/logger.js +12 -1
  212. package/dist/utils/logger.js.map +2 -2
  213. package/package.json +5 -1
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/retrieval/summary-generator.ts"],
4
- "sourcesContent": ["/**\n * Compressed Summary Generator\n * Creates compact summaries of project memory for LLM analysis\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FrameManager,\n Frame,\n Anchor,\n Event,\n} from '../context/frame-manager.js';\nimport { TraceDetector } from '../trace/trace-detector.js';\nimport {\n CompressedSummary,\n RecentSessionSummary,\n HistoricalPatterns,\n QueryableIndices,\n SummaryStats,\n FrameSummary,\n OperationSummary,\n FileSummary,\n ErrorSummary,\n DecisionSummary,\n IssueSummary,\n ToolSequence,\n ActivityPattern,\n RetrievalConfig,\n DEFAULT_RETRIEVAL_CONFIG,\n} from './types.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport class CompressedSummaryGenerator {\n private db: Database.Database;\n private frameManager: FrameManager;\n private traceDetector?: TraceDetector;\n private projectId: string;\n private config: RetrievalConfig;\n private cache: Map<\n string,\n { summary: CompressedSummary; expiresAt: number }\n > = new Map();\n\n constructor(\n db: Database.Database,\n frameManager: FrameManager,\n projectId: string,\n config: Partial<RetrievalConfig> = {},\n traceDetector?: TraceDetector\n ) {\n this.db = db;\n this.frameManager = frameManager;\n this.projectId = projectId;\n this.config = { ...DEFAULT_RETRIEVAL_CONFIG, ...config };\n this.traceDetector = traceDetector;\n }\n\n /**\n * Generate a compressed summary for LLM analysis\n */\n public generateSummary(\n options: {\n maxFrames?: number;\n timeRangeHours?: number;\n forceRefresh?: boolean;\n } = {}\n ): CompressedSummary {\n const cacheKey = `summary_${this.projectId}_${options.maxFrames || this.config.maxSummaryFrames}`;\n\n // Check cache\n if (!options.forceRefresh) {\n const cached = this.cache.get(cacheKey);\n if (cached && cached.expiresAt > Date.now()) {\n logger.debug('Using cached summary', { projectId: this.projectId });\n return cached.summary;\n }\n }\n\n const startTime = Date.now();\n const maxFrames = options.maxFrames || this.config.maxSummaryFrames;\n const timeRangeHours = options.timeRangeHours || 24;\n\n // Generate all components\n const recentSession = this.generateRecentSessionSummary(\n maxFrames,\n timeRangeHours\n );\n const historicalPatterns = this.generateHistoricalPatterns();\n const queryableIndices = this.generateQueryableIndices();\n const stats = this.generateStats();\n\n const summary: CompressedSummary = {\n projectId: this.projectId,\n generatedAt: Date.now(),\n recentSession,\n historicalPatterns,\n queryableIndices,\n stats,\n };\n\n // Cache the result\n this.cache.set(cacheKey, {\n summary,\n expiresAt: Date.now() + this.config.cacheTtlSeconds * 1000,\n });\n\n logger.info('Generated compressed summary', {\n projectId: this.projectId,\n frames: recentSession.frames.length,\n generationTimeMs: Date.now() - startTime,\n });\n\n return summary;\n }\n\n /**\n * Generate recent session summary\n */\n private generateRecentSessionSummary(\n maxFrames: number,\n timeRangeHours: number\n ): RecentSessionSummary {\n const cutoffTime = Math.floor(Date.now() / 1000) - timeRangeHours * 3600;\n\n // Get recent frames\n const frames = this.getRecentFrames(maxFrames, cutoffTime);\n const frameSummaries = frames.map((f) => this.summarizeFrame(f));\n\n // Get dominant operations\n const dominantOperations = this.getDominantOperations(cutoffTime);\n\n // Get files touched\n const filesTouched = this.getFilesTouched(cutoffTime);\n\n // Get errors encountered\n const errorsEncountered = this.getErrorsEncountered(cutoffTime);\n\n // Calculate time range\n const timestamps = frames.map((f) => f.created_at).filter((t) => t);\n const start = timestamps.length > 0 ? Math.min(...timestamps) : cutoffTime;\n const end =\n timestamps.length > 0\n ? Math.max(...timestamps)\n : Math.floor(Date.now() / 1000);\n\n return {\n frames: frameSummaries,\n dominantOperations,\n filesTouched,\n errorsEncountered,\n timeRange: {\n start: start * 1000,\n end: end * 1000,\n durationMs: (end - start) * 1000,\n },\n };\n }\n\n /**\n * Generate historical patterns\n */\n private generateHistoricalPatterns(): HistoricalPatterns {\n return {\n topicFrameCounts: this.getTopicFrameCounts(),\n keyDecisions: this.getKeyDecisions(),\n recurringIssues: this.getRecurringIssues(),\n commonToolSequences: this.getCommonToolSequences(),\n activityPatterns: this.getActivityPatterns(),\n };\n }\n\n /**\n * Generate queryable indices\n */\n private generateQueryableIndices(): QueryableIndices {\n return {\n byErrorType: this.indexByErrorType(),\n byTimeframe: this.indexByTimeframe(),\n byContributor: this.indexByContributor(),\n byTopic: this.indexByTopic(),\n byFile: this.indexByFile(),\n };\n }\n\n /**\n * Generate summary statistics\n */\n private generateStats(): SummaryStats {\n try {\n const frameStats =\n (this.db\n .prepare(\n `\n SELECT \n COUNT(*) as totalFrames,\n MIN(created_at) as oldestFrame,\n MAX(created_at) as newestFrame,\n AVG(depth) as avgDepth\n FROM frames\n WHERE project_id = ?\n `\n )\n .get(this.projectId) as any) || {};\n\n const eventCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const anchorCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const decisionCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ? AND a.type = 'DECISION'\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const totalFrames = frameStats.totalFrames || 0;\n\n return {\n totalFrames,\n totalEvents: eventCount.count || 0,\n totalAnchors: anchorCount.count || 0,\n totalDecisions: decisionCount.count || 0,\n oldestFrame: (frameStats.oldestFrame || 0) * 1000,\n newestFrame: (frameStats.newestFrame || 0) * 1000,\n avgFrameDepth: frameStats.avgDepth || 0,\n avgEventsPerFrame:\n totalFrames > 0 ? (eventCount.count || 0) / totalFrames : 0,\n };\n } catch (error) {\n logger.warn('Error generating stats, using defaults', { error });\n return {\n totalFrames: 0,\n totalEvents: 0,\n totalAnchors: 0,\n totalDecisions: 0,\n oldestFrame: 0,\n newestFrame: 0,\n avgFrameDepth: 0,\n avgEventsPerFrame: 0,\n };\n }\n }\n\n // Helper methods for recent session\n\n private getRecentFrames(limit: number, cutoffTime: number): Frame[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT * FROM frames\n WHERE project_id = ? AND created_at >= ?\n ORDER BY created_at DESC\n LIMIT ?\n `\n )\n .all(this.projectId, cutoffTime, limit) 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 } catch {\n return [];\n }\n }\n\n private summarizeFrame(frame: Frame): FrameSummary {\n // Get event count\n const eventCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events WHERE frame_id = ?\n `\n )\n .get(frame.frame_id) as any) || { count: 0 };\n\n // Get anchor count\n const anchorCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors WHERE frame_id = ?\n `\n )\n .get(frame.frame_id) as any) || { count: 0 };\n\n // Calculate score based on activity\n const score = this.calculateFrameScore(\n frame,\n eventCount.count,\n anchorCount.count\n );\n\n return {\n frameId: frame.frame_id,\n name: frame.name,\n type: frame.type,\n depth: frame.depth,\n eventCount: eventCount.count,\n anchorCount: anchorCount.count,\n score,\n createdAt: frame.created_at * 1000,\n closedAt: frame.closed_at ? frame.closed_at * 1000 : undefined,\n digestPreview: frame.digest_text?.substring(0, 100),\n };\n }\n\n private calculateFrameScore(\n frame: Frame,\n eventCount: number,\n anchorCount: number\n ): number {\n let score = 0.3; // Base score\n\n // Activity bonus\n score += Math.min(eventCount / 50, 0.3);\n score += Math.min(anchorCount / 10, 0.2);\n\n // Recency bonus\n const ageHours = (Date.now() / 1000 - frame.created_at) / 3600;\n if (ageHours < 1) score += 0.2;\n else if (ageHours < 6) score += 0.1;\n\n // Open frame bonus\n if (frame.state === 'active') score += 0.1;\n\n return Math.min(score, 1.0);\n }\n\n private getDominantOperations(cutoffTime: number): OperationSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n e.event_type as operation,\n COUNT(*) as count,\n MAX(e.ts) as lastOccurrence,\n SUM(CASE WHEN json_extract(e.payload, '$.success') = 1 THEN 1 ELSE 0 END) as successCount\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? AND e.ts >= ?\n GROUP BY e.event_type\n ORDER BY count DESC\n LIMIT 10\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n return rows.map((row) => ({\n operation: row.operation,\n count: row.count,\n lastOccurrence: row.lastOccurrence * 1000,\n successRate: row.count > 0 ? row.successCount / row.count : 0,\n }));\n } catch {\n return [];\n }\n }\n\n private getFilesTouched(cutoffTime: number): FileSummary[] {\n try {\n // Extract file paths from event payloads\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.file_path') as path,\n e.event_type as operation,\n MAX(e.ts) as lastModified\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND e.ts >= ?\n AND json_extract(e.payload, '$.file_path') IS NOT NULL\n GROUP BY json_extract(e.payload, '$.file_path'), e.event_type\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n // Aggregate by file\n const fileMap = new Map<string, FileSummary>();\n for (const row of rows) {\n if (!row.path) continue;\n\n const existing = fileMap.get(row.path);\n if (existing) {\n existing.operationCount++;\n existing.operations.push(row.operation);\n existing.lastModified = Math.max(\n existing.lastModified,\n row.lastModified * 1000\n );\n } else {\n fileMap.set(row.path, {\n path: row.path,\n operationCount: 1,\n lastModified: row.lastModified * 1000,\n operations: [row.operation],\n });\n }\n }\n\n return Array.from(fileMap.values())\n .sort((a, b) => b.operationCount - a.operationCount)\n .slice(0, 20);\n } catch {\n return [];\n }\n }\n\n private getErrorsEncountered(cutoffTime: number): ErrorSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.error_type') as errorType,\n json_extract(e.payload, '$.error') as message,\n COUNT(*) as count,\n MAX(e.ts) as lastOccurrence\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND e.ts >= ?\n AND (json_extract(e.payload, '$.error') IS NOT NULL \n OR json_extract(e.payload, '$.error_type') IS NOT NULL)\n GROUP BY json_extract(e.payload, '$.error_type'), json_extract(e.payload, '$.error')\n ORDER BY count DESC\n LIMIT 15\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n return rows.map((row) => ({\n errorType: row.errorType || 'unknown',\n message: row.message || '',\n count: row.count,\n lastOccurrence: row.lastOccurrence * 1000,\n resolved: false, // Would need more context to determine\n }));\n } catch {\n return [];\n }\n }\n\n // Helper methods for historical patterns\n\n private getTopicFrameCounts(): Record<string, number> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT type, COUNT(*) as count\n FROM frames\n WHERE project_id = ?\n GROUP BY type\n `\n )\n .all(this.projectId) as any[];\n\n const counts: Record<string, number> = {};\n for (const row of rows) {\n counts[row.type] = row.count;\n }\n return counts;\n } catch {\n return {};\n }\n }\n\n private getKeyDecisions(): DecisionSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n a.anchor_id as id,\n a.text,\n a.frame_id as frameId,\n a.created_at as timestamp,\n a.priority\n FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ? AND a.type = 'DECISION'\n ORDER BY a.priority DESC, a.created_at DESC\n LIMIT 20\n `\n )\n .all(this.projectId) as any[];\n\n return rows.map((row) => ({\n id: row.id,\n text: row.text,\n frameId: row.frameId,\n timestamp: row.timestamp * 1000,\n impact:\n row.priority >= 7 ? 'high' : row.priority >= 4 ? 'medium' : 'low',\n }));\n } catch {\n return [];\n }\n }\n\n private getRecurringIssues(): IssueSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.error_type') as issueType,\n COUNT(*) as occurrenceCount,\n MAX(e.ts) as lastSeen\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.error_type') IS NOT NULL\n GROUP BY json_extract(e.payload, '$.error_type')\n HAVING COUNT(*) > 1\n ORDER BY occurrenceCount DESC\n LIMIT 10\n `\n )\n .all(this.projectId) as any[];\n\n return rows.map((row) => ({\n issueType: row.issueType,\n occurrenceCount: row.occurrenceCount,\n lastSeen: row.lastSeen * 1000,\n resolutionRate: 0.5, // Would need resolution tracking\n }));\n } catch {\n return [];\n }\n }\n\n private getCommonToolSequences(): ToolSequence[] {\n // Use trace detector if available\n if (this.traceDetector) {\n const stats = this.traceDetector.getStatistics();\n const sequences: ToolSequence[] = [];\n\n for (const [type, count] of Object.entries(stats.tracesByType)) {\n sequences.push({\n pattern: type,\n frequency: count,\n avgDuration: 0, // Would need more data\n successRate: 0.8, // Estimate\n });\n }\n\n return sequences;\n }\n\n return [];\n }\n\n private getActivityPatterns(): ActivityPattern[] {\n try {\n // Get hourly distribution\n const hourlyRows = this.db\n .prepare(\n `\n SELECT \n strftime('%H', datetime(e.ts, 'unixepoch')) as hour,\n COUNT(*) as count\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n GROUP BY hour\n ORDER BY count DESC\n `\n )\n .all(this.projectId) as any[];\n\n const peakHours = hourlyRows.slice(0, 3).map((r) => `${r.hour}:00`);\n const totalEvents = hourlyRows.reduce((sum, r) => sum + r.count, 0);\n\n return [\n {\n periodType: 'hourly',\n peakPeriods: peakHours,\n avgEventsPerPeriod: totalEvents / 24,\n },\n ];\n } catch {\n return [];\n }\n }\n\n // Helper methods for queryable indices\n\n private indexByErrorType(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT DISTINCT\n json_extract(e.payload, '$.error_type') as errorType,\n f.frame_id\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.error_type') IS NOT NULL\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n for (const row of rows) {\n if (!row.errorType) continue;\n if (!index[row.errorType]) index[row.errorType] = [];\n if (!index[row.errorType].includes(row.frame_id)) {\n index[row.errorType].push(row.frame_id);\n }\n }\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByTimeframe(): Record<string, string[]> {\n try {\n const now = Math.floor(Date.now() / 1000);\n const timeframes = {\n last_hour: now - 3600,\n last_day: now - 86400,\n last_week: now - 604800,\n last_month: now - 2592000,\n };\n\n const index: Record<string, string[]> = {};\n\n for (const [label, cutoff] of Object.entries(timeframes)) {\n const rows = this.db\n .prepare(\n `\n SELECT frame_id FROM frames\n WHERE project_id = ? AND created_at >= ?\n `\n )\n .all(this.projectId, cutoff) as any[];\n\n index[label] = rows.map((r) => r.frame_id);\n }\n\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByContributor(): Record<string, string[]> {\n // Would need user tracking - return empty for now\n return {};\n }\n\n private indexByTopic(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT frame_id, type, name FROM frames\n WHERE project_id = ?\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n\n for (const row of rows) {\n // Index by frame type\n if (!index[row.type]) index[row.type] = [];\n index[row.type].push(row.frame_id);\n\n // Index by keywords in name\n const keywords = this.extractKeywords(row.name);\n for (const keyword of keywords) {\n if (!index[keyword]) index[keyword] = [];\n if (!index[keyword].includes(row.frame_id)) {\n index[keyword].push(row.frame_id);\n }\n }\n }\n\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByFile(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT DISTINCT\n json_extract(e.payload, '$.file_path') as filePath,\n f.frame_id\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.file_path') IS NOT NULL\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n for (const row of rows) {\n if (!row.filePath) continue;\n if (!index[row.filePath]) index[row.filePath] = [];\n if (!index[row.filePath].includes(row.frame_id)) {\n index[row.filePath].push(row.frame_id);\n }\n }\n return index;\n } catch {\n return {};\n }\n }\n\n private extractKeywords(text: string): string[] {\n const stopWords = new Set([\n 'the',\n 'a',\n 'an',\n 'and',\n 'or',\n 'but',\n 'in',\n 'on',\n 'at',\n 'to',\n 'for',\n ]);\n return text\n .toLowerCase()\n .split(/\\W+/)\n .filter((word) => word.length > 2 && !stopWords.has(word));\n }\n\n /**\n * Clear the cache\n */\n public clearCache(): void {\n this.cache.clear();\n logger.debug('Summary cache cleared', { projectId: this.projectId });\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Compressed Summary Generator\n * Creates compact summaries of project memory for LLM analysis\n */\n\nimport Database from 'better-sqlite3';\nimport {\n FrameManager,\n Frame,\n Anchor,\n Event,\n} from '../context/frame-manager.js';\nimport { TraceDetector } from '../trace/trace-detector.js';\nimport {\n CompressedSummary,\n RecentSessionSummary,\n HistoricalPatterns,\n QueryableIndices,\n SummaryStats,\n FrameSummary,\n OperationSummary,\n FileSummary,\n ErrorSummary,\n DecisionSummary,\n IssueSummary,\n ToolSequence,\n ActivityPattern,\n RetrievalConfig,\n DEFAULT_RETRIEVAL_CONFIG,\n} from './types.js';\nimport { logger } from '../monitoring/logger.js';\n\nexport class CompressedSummaryGenerator {\n private db: Database.Database;\n private frameManager: FrameManager;\n private traceDetector?: TraceDetector;\n private projectId: string;\n private config: RetrievalConfig;\n private cache: Map<\n string,\n { summary: CompressedSummary; expiresAt: number }\n > = new Map();\n\n constructor(\n db: Database.Database,\n frameManager: FrameManager,\n projectId: string,\n config: Partial<RetrievalConfig> = {},\n traceDetector?: TraceDetector\n ) {\n this.db = db;\n this.frameManager = frameManager;\n this.projectId = projectId;\n this.config = { ...DEFAULT_RETRIEVAL_CONFIG, ...config };\n this.traceDetector = traceDetector;\n }\n\n /**\n * Generate a compressed summary for LLM analysis\n */\n public generateSummary(\n options: {\n maxFrames?: number;\n timeRangeHours?: number;\n forceRefresh?: boolean;\n } = {}\n ): CompressedSummary {\n const cacheKey = `summary_${this.projectId}_${options.maxFrames || this.config.maxSummaryFrames}`;\n\n // Check cache\n if (!options.forceRefresh) {\n const cached = this.cache.get(cacheKey);\n if (cached && cached.expiresAt > Date.now()) {\n logger.debug('Using cached summary', { projectId: this.projectId });\n return cached.summary;\n }\n }\n\n const startTime = Date.now();\n const maxFrames = options.maxFrames || this.config.maxSummaryFrames;\n const timeRangeHours = options.timeRangeHours || 24;\n\n // Generate all components\n const recentSession = this.generateRecentSessionSummary(\n maxFrames,\n timeRangeHours\n );\n const historicalPatterns = this.generateHistoricalPatterns();\n const queryableIndices = this.generateQueryableIndices();\n const stats = this.generateStats();\n\n const summary: CompressedSummary = {\n projectId: this.projectId,\n generatedAt: Date.now(),\n recentSession,\n historicalPatterns,\n queryableIndices,\n stats,\n };\n\n // Cache the result\n this.cache.set(cacheKey, {\n summary,\n expiresAt: Date.now() + this.config.cacheTtlSeconds * 1000,\n });\n\n logger.info('Generated compressed summary', {\n projectId: this.projectId,\n frames: recentSession.frames.length,\n generationTimeMs: Date.now() - startTime,\n });\n\n return summary;\n }\n\n /**\n * Generate recent session summary\n */\n private generateRecentSessionSummary(\n maxFrames: number,\n timeRangeHours: number\n ): RecentSessionSummary {\n const cutoffTime = Math.floor(Date.now() / 1000) - timeRangeHours * 3600;\n\n // Get recent frames\n const frames = this.getRecentFrames(maxFrames, cutoffTime);\n const frameSummaries = frames.map((f) => this.summarizeFrame(f));\n\n // Get dominant operations\n const dominantOperations = this.getDominantOperations(cutoffTime);\n\n // Get files touched\n const filesTouched = this.getFilesTouched(cutoffTime);\n\n // Get errors encountered\n const errorsEncountered = this.getErrorsEncountered(cutoffTime);\n\n // Calculate time range\n const timestamps = frames.map((f) => f.created_at).filter((t) => t);\n const start = timestamps.length > 0 ? Math.min(...timestamps) : cutoffTime;\n const end =\n timestamps.length > 0\n ? Math.max(...timestamps)\n : Math.floor(Date.now() / 1000);\n\n return {\n frames: frameSummaries,\n dominantOperations,\n filesTouched,\n errorsEncountered,\n timeRange: {\n start: start * 1000,\n end: end * 1000,\n durationMs: (end - start) * 1000,\n },\n };\n }\n\n /**\n * Generate historical patterns\n */\n private generateHistoricalPatterns(): HistoricalPatterns {\n return {\n topicFrameCounts: this.getTopicFrameCounts(),\n keyDecisions: this.getKeyDecisions(),\n recurringIssues: this.getRecurringIssues(),\n commonToolSequences: this.getCommonToolSequences(),\n activityPatterns: this.getActivityPatterns(),\n };\n }\n\n /**\n * Generate queryable indices\n */\n private generateQueryableIndices(): QueryableIndices {\n return {\n byErrorType: this.indexByErrorType(),\n byTimeframe: this.indexByTimeframe(),\n byContributor: this.indexByContributor(),\n byTopic: this.indexByTopic(),\n byFile: this.indexByFile(),\n };\n }\n\n /**\n * Generate summary statistics\n */\n private generateStats(): SummaryStats {\n try {\n const frameStats =\n (this.db\n .prepare(\n `\n SELECT \n COUNT(*) as totalFrames,\n MIN(created_at) as oldestFrame,\n MAX(created_at) as newestFrame,\n AVG(depth) as avgDepth\n FROM frames\n WHERE project_id = ?\n `\n )\n .get(this.projectId) as any) || {};\n\n const eventCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const anchorCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const decisionCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ? AND a.type = 'DECISION'\n `\n )\n .get(this.projectId) as any) || { count: 0 };\n\n const totalFrames = frameStats.totalFrames || 0;\n\n return {\n totalFrames,\n totalEvents: eventCount.count || 0,\n totalAnchors: anchorCount.count || 0,\n totalDecisions: decisionCount.count || 0,\n oldestFrame: (frameStats.oldestFrame || 0) * 1000,\n newestFrame: (frameStats.newestFrame || 0) * 1000,\n avgFrameDepth: frameStats.avgDepth || 0,\n avgEventsPerFrame:\n totalFrames > 0 ? (eventCount.count || 0) / totalFrames : 0,\n };\n } catch (error: unknown) {\n logger.warn('Error generating stats, using defaults', { error });\n return {\n totalFrames: 0,\n totalEvents: 0,\n totalAnchors: 0,\n totalDecisions: 0,\n oldestFrame: 0,\n newestFrame: 0,\n avgFrameDepth: 0,\n avgEventsPerFrame: 0,\n };\n }\n }\n\n // Helper methods for recent session\n\n private getRecentFrames(limit: number, cutoffTime: number): Frame[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT * FROM frames\n WHERE project_id = ? AND created_at >= ?\n ORDER BY created_at DESC\n LIMIT ?\n `\n )\n .all(this.projectId, cutoffTime, limit) 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 } catch {\n return [];\n }\n }\n\n private summarizeFrame(frame: Frame): FrameSummary {\n // Get event count\n const eventCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events WHERE frame_id = ?\n `\n )\n .get(frame.frame_id) as any) || { count: 0 };\n\n // Get anchor count\n const anchorCount = (this.db\n .prepare(\n `\n SELECT COUNT(*) as count FROM anchors WHERE frame_id = ?\n `\n )\n .get(frame.frame_id) as any) || { count: 0 };\n\n // Calculate score based on activity\n const score = this.calculateFrameScore(\n frame,\n eventCount.count,\n anchorCount.count\n );\n\n return {\n frameId: frame.frame_id,\n name: frame.name,\n type: frame.type,\n depth: frame.depth,\n eventCount: eventCount.count,\n anchorCount: anchorCount.count,\n score,\n createdAt: frame.created_at * 1000,\n closedAt: frame.closed_at ? frame.closed_at * 1000 : undefined,\n digestPreview: frame.digest_text?.substring(0, 100),\n };\n }\n\n private calculateFrameScore(\n frame: Frame,\n eventCount: number,\n anchorCount: number\n ): number {\n let score = 0.3; // Base score\n\n // Activity bonus\n score += Math.min(eventCount / 50, 0.3);\n score += Math.min(anchorCount / 10, 0.2);\n\n // Recency bonus\n const ageHours = (Date.now() / 1000 - frame.created_at) / 3600;\n if (ageHours < 1) score += 0.2;\n else if (ageHours < 6) score += 0.1;\n\n // Open frame bonus\n if (frame.state === 'active') score += 0.1;\n\n return Math.min(score, 1.0);\n }\n\n private getDominantOperations(cutoffTime: number): OperationSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n e.event_type as operation,\n COUNT(*) as count,\n MAX(e.ts) as lastOccurrence,\n SUM(CASE WHEN json_extract(e.payload, '$.success') = 1 THEN 1 ELSE 0 END) as successCount\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? AND e.ts >= ?\n GROUP BY e.event_type\n ORDER BY count DESC\n LIMIT 10\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n return rows.map((row) => ({\n operation: row.operation,\n count: row.count,\n lastOccurrence: row.lastOccurrence * 1000,\n successRate: row.count > 0 ? row.successCount / row.count : 0,\n }));\n } catch {\n return [];\n }\n }\n\n private getFilesTouched(cutoffTime: number): FileSummary[] {\n try {\n // Extract file paths from event payloads\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.file_path') as path,\n e.event_type as operation,\n MAX(e.ts) as lastModified\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND e.ts >= ?\n AND json_extract(e.payload, '$.file_path') IS NOT NULL\n GROUP BY json_extract(e.payload, '$.file_path'), e.event_type\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n // Aggregate by file\n const fileMap = new Map<string, FileSummary>();\n for (const row of rows) {\n if (!row.path) continue;\n\n const existing = fileMap.get(row.path);\n if (existing) {\n existing.operationCount++;\n existing.operations.push(row.operation);\n existing.lastModified = Math.max(\n existing.lastModified,\n row.lastModified * 1000\n );\n } else {\n fileMap.set(row.path, {\n path: row.path,\n operationCount: 1,\n lastModified: row.lastModified * 1000,\n operations: [row.operation],\n });\n }\n }\n\n return Array.from(fileMap.values())\n .sort((a, b) => b.operationCount - a.operationCount)\n .slice(0, 20);\n } catch {\n return [];\n }\n }\n\n private getErrorsEncountered(cutoffTime: number): ErrorSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.error_type') as errorType,\n json_extract(e.payload, '$.error') as message,\n COUNT(*) as count,\n MAX(e.ts) as lastOccurrence\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND e.ts >= ?\n AND (json_extract(e.payload, '$.error') IS NOT NULL \n OR json_extract(e.payload, '$.error_type') IS NOT NULL)\n GROUP BY json_extract(e.payload, '$.error_type'), json_extract(e.payload, '$.error')\n ORDER BY count DESC\n LIMIT 15\n `\n )\n .all(this.projectId, cutoffTime) as any[];\n\n return rows.map((row) => ({\n errorType: row.errorType || 'unknown',\n message: row.message || '',\n count: row.count,\n lastOccurrence: row.lastOccurrence * 1000,\n resolved: false, // Would need more context to determine\n }));\n } catch {\n return [];\n }\n }\n\n // Helper methods for historical patterns\n\n private getTopicFrameCounts(): Record<string, number> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT type, COUNT(*) as count\n FROM frames\n WHERE project_id = ?\n GROUP BY type\n `\n )\n .all(this.projectId) as any[];\n\n const counts: Record<string, number> = {};\n for (const row of rows) {\n counts[row.type] = row.count;\n }\n return counts;\n } catch {\n return {};\n }\n }\n\n private getKeyDecisions(): DecisionSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n a.anchor_id as id,\n a.text,\n a.frame_id as frameId,\n a.created_at as timestamp,\n a.priority\n FROM anchors a\n JOIN frames f ON a.frame_id = f.frame_id\n WHERE f.project_id = ? AND a.type = 'DECISION'\n ORDER BY a.priority DESC, a.created_at DESC\n LIMIT 20\n `\n )\n .all(this.projectId) as any[];\n\n return rows.map((row) => ({\n id: row.id,\n text: row.text,\n frameId: row.frameId,\n timestamp: row.timestamp * 1000,\n impact:\n row.priority >= 7 ? 'high' : row.priority >= 4 ? 'medium' : 'low',\n }));\n } catch {\n return [];\n }\n }\n\n private getRecurringIssues(): IssueSummary[] {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT \n json_extract(e.payload, '$.error_type') as issueType,\n COUNT(*) as occurrenceCount,\n MAX(e.ts) as lastSeen\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.error_type') IS NOT NULL\n GROUP BY json_extract(e.payload, '$.error_type')\n HAVING COUNT(*) > 1\n ORDER BY occurrenceCount DESC\n LIMIT 10\n `\n )\n .all(this.projectId) as any[];\n\n return rows.map((row) => ({\n issueType: row.issueType,\n occurrenceCount: row.occurrenceCount,\n lastSeen: row.lastSeen * 1000,\n resolutionRate: 0.5, // Would need resolution tracking\n }));\n } catch {\n return [];\n }\n }\n\n private getCommonToolSequences(): ToolSequence[] {\n // Use trace detector if available\n if (this.traceDetector) {\n const stats = this.traceDetector.getStatistics();\n const sequences: ToolSequence[] = [];\n\n for (const [type, count] of Object.entries(stats.tracesByType)) {\n sequences.push({\n pattern: type,\n frequency: count,\n avgDuration: 0, // Would need more data\n successRate: 0.8, // Estimate\n });\n }\n\n return sequences;\n }\n\n return [];\n }\n\n private getActivityPatterns(): ActivityPattern[] {\n try {\n // Get hourly distribution\n const hourlyRows = this.db\n .prepare(\n `\n SELECT \n strftime('%H', datetime(e.ts, 'unixepoch')) as hour,\n COUNT(*) as count\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n GROUP BY hour\n ORDER BY count DESC\n `\n )\n .all(this.projectId) as any[];\n\n const peakHours = hourlyRows.slice(0, 3).map((r) => `${r.hour}:00`);\n const totalEvents = hourlyRows.reduce((sum, r) => sum + r.count, 0);\n\n return [\n {\n periodType: 'hourly',\n peakPeriods: peakHours,\n avgEventsPerPeriod: totalEvents / 24,\n },\n ];\n } catch {\n return [];\n }\n }\n\n // Helper methods for queryable indices\n\n private indexByErrorType(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT DISTINCT\n json_extract(e.payload, '$.error_type') as errorType,\n f.frame_id\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.error_type') IS NOT NULL\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n for (const row of rows) {\n if (!row.errorType) continue;\n if (!index[row.errorType]) index[row.errorType] = [];\n if (!index[row.errorType].includes(row.frame_id)) {\n index[row.errorType].push(row.frame_id);\n }\n }\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByTimeframe(): Record<string, string[]> {\n try {\n const now = Math.floor(Date.now() / 1000);\n const timeframes = {\n last_hour: now - 3600,\n last_day: now - 86400,\n last_week: now - 604800,\n last_month: now - 2592000,\n };\n\n const index: Record<string, string[]> = {};\n\n for (const [label, cutoff] of Object.entries(timeframes)) {\n const rows = this.db\n .prepare(\n `\n SELECT frame_id FROM frames\n WHERE project_id = ? AND created_at >= ?\n `\n )\n .all(this.projectId, cutoff) as any[];\n\n index[label] = rows.map((r) => r.frame_id);\n }\n\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByContributor(): Record<string, string[]> {\n // Would need user tracking - return empty for now\n return {};\n }\n\n private indexByTopic(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT frame_id, type, name FROM frames\n WHERE project_id = ?\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n\n for (const row of rows) {\n // Index by frame type\n if (!index[row.type]) index[row.type] = [];\n index[row.type].push(row.frame_id);\n\n // Index by keywords in name\n const keywords = this.extractKeywords(row.name);\n for (const keyword of keywords) {\n if (!index[keyword]) index[keyword] = [];\n if (!index[keyword].includes(row.frame_id)) {\n index[keyword].push(row.frame_id);\n }\n }\n }\n\n return index;\n } catch {\n return {};\n }\n }\n\n private indexByFile(): Record<string, string[]> {\n try {\n const rows = this.db\n .prepare(\n `\n SELECT DISTINCT\n json_extract(e.payload, '$.file_path') as filePath,\n f.frame_id\n FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ? \n AND json_extract(e.payload, '$.file_path') IS NOT NULL\n `\n )\n .all(this.projectId) as any[];\n\n const index: Record<string, string[]> = {};\n for (const row of rows) {\n if (!row.filePath) continue;\n if (!index[row.filePath]) index[row.filePath] = [];\n if (!index[row.filePath].includes(row.frame_id)) {\n index[row.filePath].push(row.frame_id);\n }\n }\n return index;\n } catch {\n return {};\n }\n }\n\n private extractKeywords(text: string): string[] {\n const stopWords = new Set([\n 'the',\n 'a',\n 'an',\n 'and',\n 'or',\n 'but',\n 'in',\n 'on',\n 'at',\n 'to',\n 'for',\n ]);\n return text\n .toLowerCase()\n .split(/\\W+/)\n .filter((word) => word.length > 2 && !stopWords.has(word));\n }\n\n /**\n * Clear the cache\n */\n public clearCache(): void {\n this.cache.clear();\n logger.debug('Summary cache cleared', { projectId: this.projectId });\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/session/clear-survival.ts"],
4
- "sourcesContent": ["/**\n * Clear Survival System for StackMemory\n * Inspired by Continuous-Claude's \"Clear, don't compact\" philosophy\n *\n * Allows StackMemory to survive /clear operations by:\n * 1. Detecting when context is getting full (>70%)\n * 2. Saving critical state to external ledgers\n * 3. Restoring from ledgers after /clear\n * 4. Maintaining continuity across session resets\n */\n\nimport { Frame, Trace, Context, Digest } from '../types';\nimport { FrameManager } from '../frame/frame-manager';\nimport { DatabaseManager } from '../storage/database-manager';\nimport { HandoffGenerator } from './handoff-generator';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface ContinuityLedger {\n version: '1.0.0';\n timestamp: string;\n session_id: string;\n project: string;\n branch?: string;\n\n // Critical state to preserve\n active_frame_stack: FrameSummary[];\n key_decisions: Decision[];\n active_tasks: Task[];\n critical_context: string[];\n recent_achievements: Achievement[];\n\n // Navigation aids\n current_focus: string;\n next_actions: string[];\n warnings: string[];\n\n // Metrics for restoration\n original_token_count: number;\n compressed_token_count: number;\n compression_ratio: number;\n}\n\ninterface FrameSummary {\n id: string;\n type: string;\n description: string;\n depth: number;\n key_events: string[];\n digest?: string;\n}\n\ninterface Decision {\n id: string;\n decision: string;\n rationale: string;\n impact: 'low' | 'medium' | 'high' | 'critical';\n still_applies: boolean;\n}\n\ninterface Task {\n id: string;\n title: string;\n status: 'pending' | 'in_progress' | 'blocked' | 'completed';\n priority: 'low' | 'medium' | 'high' | 'critical';\n context: string;\n}\n\ninterface Achievement {\n description: string;\n impact: string;\n timestamp: string;\n}\n\nexport class ClearSurvival {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private handoffGenerator: HandoffGenerator;\n private ledgerPath: string;\n private continuityPath: string;\n\n // Thresholds\n private readonly CONTEXT_WARNING_THRESHOLD = 0.6; // 60%\n private readonly CONTEXT_CRITICAL_THRESHOLD = 0.7; // 70%\n private readonly CONTEXT_MAX_THRESHOLD = 0.85; // 85% - force save\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n handoffGenerator: HandoffGenerator,\n projectRoot: string\n ) {\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.handoffGenerator = handoffGenerator;\n this.ledgerPath = path.join(projectRoot, '.stackmemory', 'ledgers');\n this.continuityPath = path.join(projectRoot, '.stackmemory', 'continuity');\n }\n\n /**\n * Monitor context usage and trigger saves when needed\n */\n async monitorContextUsage(\n currentTokens: number,\n maxTokens: number\n ): Promise<'ok' | 'warning' | 'critical' | 'saved'> {\n const usage = currentTokens / maxTokens;\n\n if (usage < this.CONTEXT_WARNING_THRESHOLD) {\n return 'ok';\n }\n\n if (usage >= this.CONTEXT_MAX_THRESHOLD) {\n // Force save at 85%\n await this.saveContinuityLedger();\n return 'saved';\n }\n\n if (usage >= this.CONTEXT_CRITICAL_THRESHOLD) {\n // Suggest save at 70%\n console.warn(\n `\u26A0\uFE0F Context at ${Math.round(usage * 100)}% - Consider /clear after saving`\n );\n return 'critical';\n }\n\n // Warning at 60%\n console.warn(`Context at ${Math.round(usage * 100)}% - Approaching limit`);\n return 'warning';\n }\n\n /**\n * Save continuity ledger before /clear\n */\n async saveContinuityLedger(): Promise<ContinuityLedger> {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const session = await this.dbManager.getSession(sessionId);\n\n // Get current state\n const frameStack = await this.getCompressedFrameStack();\n const decisions = await this.getCriticalDecisions();\n const tasks = await this.getActiveTasks();\n const context = await this.getCriticalContext();\n const achievements = await this.getRecentAchievements();\n\n // Calculate token counts (simplified)\n const originalTokens = await this.estimateCurrentTokens();\n const compressedTokens = this.estimateLedgerTokens(\n frameStack,\n decisions,\n tasks\n );\n\n const ledger: ContinuityLedger = {\n version: '1.0.0',\n timestamp: new Date().toISOString(),\n session_id: sessionId,\n project: session?.project || 'unknown',\n branch: session?.metadata?.branch,\n\n active_frame_stack: frameStack,\n key_decisions: decisions,\n active_tasks: tasks,\n critical_context: context,\n recent_achievements: achievements,\n\n current_focus: await this.getCurrentFocus(),\n next_actions: await this.suggestNextActions(tasks),\n warnings: await this.getWarnings(),\n\n original_token_count: originalTokens,\n compressed_token_count: compressedTokens,\n compression_ratio: originalTokens / compressedTokens,\n };\n\n // Save to file (overwrites previous continuity ledger)\n await this.saveLedgerToFile(ledger);\n\n // Also create a timestamped backup\n await this.saveBackupLedger(ledger);\n\n console.log(\n `\u2705 Continuity ledger saved (${Math.round(ledger.compression_ratio)}x compression)`\n );\n\n return ledger;\n }\n\n /**\n * Restore from continuity ledger after /clear\n */\n async restoreFromLedger(): Promise<boolean> {\n try {\n const ledger = await this.loadLatestLedger();\n if (!ledger) {\n console.log('No continuity ledger found');\n return false;\n }\n\n console.log(`\uD83D\uDCDA Restoring from ledger (${ledger.timestamp})`);\n\n // Restore frame stack structure (not full content)\n await this.restoreFrameStructure(ledger.active_frame_stack);\n\n // Restore key decisions as anchors\n await this.restoreDecisions(ledger.key_decisions);\n\n // Restore active tasks\n await this.restoreTasks(ledger.active_tasks);\n\n // Log restoration summary\n console.log(`\u2705 Restored:`);\n console.log(` - ${ledger.active_frame_stack.length} frames`);\n console.log(` - ${ledger.key_decisions.length} decisions`);\n console.log(` - ${ledger.active_tasks.length} tasks`);\n console.log(` - Current focus: ${ledger.current_focus}`);\n\n if (ledger.warnings.length > 0) {\n console.warn(`\u26A0\uFE0F Warnings:`, ledger.warnings);\n }\n\n return true;\n } catch (error) {\n console.error('Failed to restore from ledger:', error);\n return false;\n }\n }\n\n /**\n * Generate markdown summary for human review\n */\n async generateLedgerMarkdown(ledger: ContinuityLedger): Promise<string> {\n const lines: string[] = [\n `# Continuity Ledger`,\n `**Saved**: ${new Date(ledger.timestamp).toLocaleString()}`,\n `**Project**: ${ledger.project}${ledger.branch ? ` (${ledger.branch})` : ''}`,\n `**Compression**: ${Math.round(ledger.compression_ratio)}x (${ledger.original_token_count} \u2192 ${ledger.compressed_token_count} tokens)`,\n '',\n\n `## \uD83C\uDFAF Current Focus`,\n ledger.current_focus,\n '',\n\n `## \uD83D\uDCDA Active Frame Stack (${ledger.active_frame_stack.length})`,\n ...ledger.active_frame_stack.map(\n (f) => `${' '.repeat(f.depth)}\u2514\u2500 ${f.type}: ${f.description}`\n ),\n '',\n\n `## \uD83C\uDFAF Active Tasks (${ledger.active_tasks.filter((t) => t.status !== 'completed').length})`,\n ...ledger.active_tasks\n .filter((t) => t.status !== 'completed')\n .sort((a, b) => {\n const priority = { critical: 0, high: 1, medium: 2, low: 3 };\n return priority[a.priority] - priority[b.priority];\n })\n .map((t) => `- [${t.priority}] ${t.title} (${t.status})`),\n '',\n\n `## \uD83D\uDD11 Key Decisions`,\n ...ledger.key_decisions\n .filter((d) => d.still_applies)\n .map((d) => `- **${d.decision}**\\n ${d.rationale}`),\n '',\n\n `## \u2705 Recent Achievements`,\n ...ledger.recent_achievements.map(\n (a) => `- ${a.description} \u2192 ${a.impact}`\n ),\n '',\n\n `## \u27A1\uFE0F Next Actions`,\n ...ledger.next_actions.map((a, i) => `${i + 1}. ${a}`),\n '',\n\n ledger.warnings.length > 0 ? `## \u26A0\uFE0F Warnings` : '',\n ...ledger.warnings.map((w) => `- ${w}`),\n ];\n\n return lines.filter((l) => l !== '').join('\\n');\n }\n\n /**\n * Check if /clear is recommended\n */\n async shouldClear(\n currentTokens: number,\n maxTokens: number\n ): Promise<{\n recommended: boolean;\n reason?: string;\n alternative?: string;\n }> {\n const usage = currentTokens / maxTokens;\n\n if (usage < this.CONTEXT_WARNING_THRESHOLD) {\n return { recommended: false };\n }\n\n // Check if we have redundant frames\n const frameStack = await this.frameManager.getStack();\n const redundantFrames = frameStack.frames.filter(\n (f) => f.status === 'closed' && !f.metadata?.critical\n ).length;\n\n if (usage >= this.CONTEXT_CRITICAL_THRESHOLD) {\n if (redundantFrames > 5) {\n return {\n recommended: true,\n reason: `Context at ${Math.round(usage * 100)}% with ${redundantFrames} closed frames`,\n alternative: 'Consider saving ledger and clearing',\n };\n }\n }\n\n return {\n recommended: false,\n alternative: `Context at ${Math.round(usage * 100)}% but manageable`,\n };\n }\n\n // Private helper methods\n\n private async getCompressedFrameStack(): Promise<FrameSummary[]> {\n const stack = await this.frameManager.getStack();\n\n return stack.frames.map((frame, index) => ({\n id: frame.id,\n type: frame.type,\n description: frame.description || 'Unnamed frame',\n depth: index,\n key_events: this.extractKeyEvents(frame),\n digest: frame.digest?.summary,\n }));\n }\n\n private extractKeyEvents(frame: Frame): string[] {\n const events: string[] = [];\n\n // Extract from metadata\n if (frame.metadata?.decision) {\n events.push(`Decision: ${frame.metadata.decision}`);\n }\n if (frame.metadata?.error) {\n events.push(`Error: ${frame.metadata.error}`);\n }\n if (frame.metadata?.achievement) {\n events.push(`Achievement: ${frame.metadata.achievement}`);\n }\n\n return events;\n }\n\n private async getCriticalDecisions(): Promise<Decision[]> {\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 100\n );\n\n return traces\n .filter((t) => t.type === 'decision')\n .map((t) => ({\n id: t.id,\n decision: t.content.decision || '',\n rationale: t.content.rationale || '',\n impact: this.assessImpact(t),\n still_applies: !t.metadata?.superseded,\n }))\n .filter((d) => d.impact !== 'low');\n }\n\n private assessImpact(trace: Trace): Decision['impact'] {\n const content = JSON.stringify(trace.content).toLowerCase();\n\n if (content.includes('architecture') || content.includes('critical')) {\n return 'critical';\n }\n if (content.includes('important') || content.includes('significant')) {\n return 'high';\n }\n if (content.includes('minor') || content.includes('small')) {\n return 'low';\n }\n\n return 'medium';\n }\n\n private async getActiveTasks(): Promise<Task[]> {\n const frames = await this.dbManager.getRecentFrames(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n return frames\n .filter((f) => f.type === 'task')\n .map((f) => ({\n id: f.id,\n title: f.description || 'Untitled task',\n status: this.getTaskStatus(f),\n priority: this.getTaskPriority(f),\n context: f.metadata?.context || '',\n }));\n }\n\n private getTaskStatus(frame: Frame): Task['status'] {\n if (frame.status === 'closed' && frame.metadata?.completed) {\n return 'completed';\n }\n if (frame.metadata?.blocked) return 'blocked';\n if (frame.status === 'open') return 'in_progress';\n return 'pending';\n }\n\n private getTaskPriority(frame: Frame): Task['priority'] {\n const priority = frame.metadata?.priority;\n if (['critical', 'high', 'medium', 'low'].includes(priority)) {\n return priority as Task['priority'];\n }\n return 'medium';\n }\n\n private async getCriticalContext(): Promise<string[]> {\n const context: string[] = [];\n\n // Add project-specific context\n const session = await this.dbManager.getSession(\n await this.dbManager.getCurrentSessionId()\n );\n if (session?.metadata?.key_facts) {\n context.push(...session.metadata.key_facts);\n }\n\n // Add recent important discoveries\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n const discoveries = traces\n .filter((t) => t.metadata?.important || t.type === 'discovery')\n .map((t) => t.content.summary || t.content.description)\n .filter(Boolean)\n .slice(0, 5);\n\n context.push(...discoveries);\n\n return context;\n }\n\n private async getRecentAchievements(): Promise<Achievement[]> {\n const frames = await this.dbManager.getRecentFrames(\n await this.dbManager.getCurrentSessionId(),\n 20\n );\n\n return frames\n .filter((f) => f.status === 'closed' && f.metadata?.achievement)\n .map((f) => ({\n description: f.metadata.achievement,\n impact: f.metadata.impact || 'completed task',\n timestamp: f.closedAt || f.createdAt,\n }))\n .slice(0, 5);\n }\n\n private async getCurrentFocus(): Promise<string> {\n const stack = await this.frameManager.getStack();\n const activeFrame = stack.frames.find((f) => f.status === 'open');\n\n if (!activeFrame) {\n return 'No active focus';\n }\n\n return `${activeFrame.type}: ${activeFrame.description || 'In progress'}`;\n }\n\n private async suggestNextActions(tasks: Task[]): Promise<string[]> {\n const suggestions: string[] = [];\n\n // Continue in-progress tasks\n const inProgress = tasks.filter((t) => t.status === 'in_progress');\n if (inProgress.length > 0) {\n suggestions.push(`Continue: ${inProgress[0].title}`);\n }\n\n // Start high-priority pending tasks\n const highPriority = tasks.filter(\n (t) => t.status === 'pending' && t.priority === 'high'\n );\n if (highPriority.length > 0) {\n suggestions.push(`Start: ${highPriority[0].title}`);\n }\n\n // Unblock blocked tasks\n const blocked = tasks.filter((t) => t.status === 'blocked');\n if (blocked.length > 0) {\n suggestions.push(`Unblock: ${blocked[0].title}`);\n }\n\n return suggestions.slice(0, 3);\n }\n\n private async getWarnings(): Promise<string[]> {\n const warnings: string[] = [];\n\n const tasks = await this.getActiveTasks();\n const blocked = tasks.filter((t) => t.status === 'blocked');\n\n if (blocked.length > 0) {\n warnings.push(`${blocked.length} tasks blocked`);\n }\n\n const critical = tasks.filter(\n (t) => t.priority === 'critical' && t.status !== 'completed'\n );\n if (critical.length > 0) {\n warnings.push(`${critical.length} critical tasks pending`);\n }\n\n return warnings;\n }\n\n private async estimateCurrentTokens(): Promise<number> {\n // Simplified estimation\n const frames = await this.frameManager.getStack();\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 100\n );\n\n const frameTokens = frames.frames.length * 200; // Rough estimate\n const traceTokens = traces.length * 100; // Rough estimate\n\n return frameTokens + traceTokens;\n }\n\n private estimateLedgerTokens(\n frames: FrameSummary[],\n decisions: Decision[],\n tasks: Task[]\n ): number {\n // Rough estimation\n return frames.length * 50 + decisions.length * 30 + tasks.length * 20;\n }\n\n private async saveLedgerToFile(ledger: ContinuityLedger): Promise<void> {\n await fs.mkdir(this.continuityPath, { recursive: true });\n\n // Save as CONTINUITY_CLAUDE-latest.json (overwrites)\n const latestPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.json'\n );\n await fs.writeFile(latestPath, JSON.stringify(ledger, null, 2), 'utf-8');\n\n // Also save markdown version\n const markdown = await this.generateLedgerMarkdown(ledger);\n const mdPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.md'\n );\n await fs.writeFile(mdPath, markdown, 'utf-8');\n }\n\n private async saveBackupLedger(ledger: ContinuityLedger): Promise<void> {\n await fs.mkdir(this.ledgerPath, { recursive: true });\n\n const timestamp = ledger.timestamp.replace(/[:.]/g, '-');\n const backupPath = path.join(this.ledgerPath, `ledger-${timestamp}.json`);\n await fs.writeFile(backupPath, JSON.stringify(ledger, null, 2), 'utf-8');\n }\n\n private async loadLatestLedger(): Promise<ContinuityLedger | null> {\n try {\n const latestPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.json'\n );\n const content = await fs.readFile(latestPath, 'utf-8');\n return JSON.parse(content) as ContinuityLedger;\n } catch (error) {\n return null;\n }\n }\n\n private async restoreFrameStructure(frames: FrameSummary[]): Promise<void> {\n // Create lightweight frame references (not full frames)\n for (const summary of frames) {\n await this.frameManager.push({\n type: summary.type,\n description: summary.description,\n metadata: {\n restored_from_ledger: true,\n original_id: summary.id,\n key_events: summary.key_events,\n digest: summary.digest,\n },\n });\n }\n }\n\n private async restoreDecisions(decisions: Decision[]): Promise<void> {\n for (const decision of decisions) {\n if (decision.still_applies) {\n await this.dbManager.addAnchor({\n type: 'decision',\n content: {\n decision: decision.decision,\n rationale: decision.rationale,\n impact: decision.impact,\n },\n metadata: {\n restored_from_ledger: true,\n original_id: decision.id,\n },\n });\n }\n }\n }\n\n private async restoreTasks(tasks: Task[]): Promise<void> {\n for (const task of tasks) {\n if (task.status !== 'completed') {\n await this.frameManager.push({\n type: 'task',\n description: task.title,\n metadata: {\n status: task.status,\n priority: task.priority,\n context: task.context,\n restored_from_ledger: true,\n original_id: task.id,\n },\n });\n }\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Clear Survival System for StackMemory\n * Inspired by Continuous-Claude's \"Clear, don't compact\" philosophy\n *\n * Allows StackMemory to survive /clear operations by:\n * 1. Detecting when context is getting full (>70%)\n * 2. Saving critical state to external ledgers\n * 3. Restoring from ledgers after /clear\n * 4. Maintaining continuity across session resets\n */\n\nimport { Frame, Trace, Context, Digest } from '../types';\nimport { FrameManager } from '../frame/frame-manager';\nimport { DatabaseManager } from '../storage/database-manager';\nimport { HandoffGenerator } from './handoff-generator';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface ContinuityLedger {\n version: '1.0.0';\n timestamp: string;\n session_id: string;\n project: string;\n branch?: string;\n\n // Critical state to preserve\n active_frame_stack: FrameSummary[];\n key_decisions: Decision[];\n active_tasks: Task[];\n critical_context: string[];\n recent_achievements: Achievement[];\n\n // Navigation aids\n current_focus: string;\n next_actions: string[];\n warnings: string[];\n\n // Metrics for restoration\n original_token_count: number;\n compressed_token_count: number;\n compression_ratio: number;\n}\n\ninterface FrameSummary {\n id: string;\n type: string;\n description: string;\n depth: number;\n key_events: string[];\n digest?: string;\n}\n\ninterface Decision {\n id: string;\n decision: string;\n rationale: string;\n impact: 'low' | 'medium' | 'high' | 'critical';\n still_applies: boolean;\n}\n\ninterface Task {\n id: string;\n title: string;\n status: 'pending' | 'in_progress' | 'blocked' | 'completed';\n priority: 'low' | 'medium' | 'high' | 'critical';\n context: string;\n}\n\ninterface Achievement {\n description: string;\n impact: string;\n timestamp: string;\n}\n\nexport class ClearSurvival {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private handoffGenerator: HandoffGenerator;\n private ledgerPath: string;\n private continuityPath: string;\n\n // Thresholds\n private readonly CONTEXT_WARNING_THRESHOLD = 0.6; // 60%\n private readonly CONTEXT_CRITICAL_THRESHOLD = 0.7; // 70%\n private readonly CONTEXT_MAX_THRESHOLD = 0.85; // 85% - force save\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n handoffGenerator: HandoffGenerator,\n projectRoot: string\n ) {\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.handoffGenerator = handoffGenerator;\n this.ledgerPath = path.join(projectRoot, '.stackmemory', 'ledgers');\n this.continuityPath = path.join(projectRoot, '.stackmemory', 'continuity');\n }\n\n /**\n * Monitor context usage and trigger saves when needed\n */\n async monitorContextUsage(\n currentTokens: number,\n maxTokens: number\n ): Promise<'ok' | 'warning' | 'critical' | 'saved'> {\n const usage = currentTokens / maxTokens;\n\n if (usage < this.CONTEXT_WARNING_THRESHOLD) {\n return 'ok';\n }\n\n if (usage >= this.CONTEXT_MAX_THRESHOLD) {\n // Force save at 85%\n await this.saveContinuityLedger();\n return 'saved';\n }\n\n if (usage >= this.CONTEXT_CRITICAL_THRESHOLD) {\n // Suggest save at 70%\n console.warn(\n `\u26A0\uFE0F Context at ${Math.round(usage * 100)}% - Consider /clear after saving`\n );\n return 'critical';\n }\n\n // Warning at 60%\n console.warn(`Context at ${Math.round(usage * 100)}% - Approaching limit`);\n return 'warning';\n }\n\n /**\n * Save continuity ledger before /clear\n */\n async saveContinuityLedger(): Promise<ContinuityLedger> {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const session = await this.dbManager.getSession(sessionId);\n\n // Get current state\n const frameStack = await this.getCompressedFrameStack();\n const decisions = await this.getCriticalDecisions();\n const tasks = await this.getActiveTasks();\n const context = await this.getCriticalContext();\n const achievements = await this.getRecentAchievements();\n\n // Calculate token counts (simplified)\n const originalTokens = await this.estimateCurrentTokens();\n const compressedTokens = this.estimateLedgerTokens(\n frameStack,\n decisions,\n tasks\n );\n\n const ledger: ContinuityLedger = {\n version: '1.0.0',\n timestamp: new Date().toISOString(),\n session_id: sessionId,\n project: session?.project || 'unknown',\n branch: session?.metadata?.branch,\n\n active_frame_stack: frameStack,\n key_decisions: decisions,\n active_tasks: tasks,\n critical_context: context,\n recent_achievements: achievements,\n\n current_focus: await this.getCurrentFocus(),\n next_actions: await this.suggestNextActions(tasks),\n warnings: await this.getWarnings(),\n\n original_token_count: originalTokens,\n compressed_token_count: compressedTokens,\n compression_ratio: originalTokens / compressedTokens,\n };\n\n // Save to file (overwrites previous continuity ledger)\n await this.saveLedgerToFile(ledger);\n\n // Also create a timestamped backup\n await this.saveBackupLedger(ledger);\n\n console.log(\n `\u2705 Continuity ledger saved (${Math.round(ledger.compression_ratio)}x compression)`\n );\n\n return ledger;\n }\n\n /**\n * Restore from continuity ledger after /clear\n */\n async restoreFromLedger(): Promise<boolean> {\n try {\n const ledger = await this.loadLatestLedger();\n if (!ledger) {\n console.log('No continuity ledger found');\n return false;\n }\n\n console.log(`\uD83D\uDCDA Restoring from ledger (${ledger.timestamp})`);\n\n // Restore frame stack structure (not full content)\n await this.restoreFrameStructure(ledger.active_frame_stack);\n\n // Restore key decisions as anchors\n await this.restoreDecisions(ledger.key_decisions);\n\n // Restore active tasks\n await this.restoreTasks(ledger.active_tasks);\n\n // Log restoration summary\n console.log(`\u2705 Restored:`);\n console.log(` - ${ledger.active_frame_stack.length} frames`);\n console.log(` - ${ledger.key_decisions.length} decisions`);\n console.log(` - ${ledger.active_tasks.length} tasks`);\n console.log(` - Current focus: ${ledger.current_focus}`);\n\n if (ledger.warnings.length > 0) {\n console.warn(`\u26A0\uFE0F Warnings:`, ledger.warnings);\n }\n\n return true;\n } catch (error: unknown) {\n console.error('Failed to restore from ledger:', error);\n return false;\n }\n }\n\n /**\n * Generate markdown summary for human review\n */\n async generateLedgerMarkdown(ledger: ContinuityLedger): Promise<string> {\n const lines: string[] = [\n `# Continuity Ledger`,\n `**Saved**: ${new Date(ledger.timestamp).toLocaleString()}`,\n `**Project**: ${ledger.project}${ledger.branch ? ` (${ledger.branch})` : ''}`,\n `**Compression**: ${Math.round(ledger.compression_ratio)}x (${ledger.original_token_count} \u2192 ${ledger.compressed_token_count} tokens)`,\n '',\n\n `## \uD83C\uDFAF Current Focus`,\n ledger.current_focus,\n '',\n\n `## \uD83D\uDCDA Active Frame Stack (${ledger.active_frame_stack.length})`,\n ...ledger.active_frame_stack.map(\n (f) => `${' '.repeat(f.depth)}\u2514\u2500 ${f.type}: ${f.description}`\n ),\n '',\n\n `## \uD83C\uDFAF Active Tasks (${ledger.active_tasks.filter((t) => t.status !== 'completed').length})`,\n ...ledger.active_tasks\n .filter((t) => t.status !== 'completed')\n .sort((a, b) => {\n const priority = { critical: 0, high: 1, medium: 2, low: 3 };\n return priority[a.priority] - priority[b.priority];\n })\n .map((t) => `- [${t.priority}] ${t.title} (${t.status})`),\n '',\n\n `## \uD83D\uDD11 Key Decisions`,\n ...ledger.key_decisions\n .filter((d) => d.still_applies)\n .map((d) => `- **${d.decision}**\\n ${d.rationale}`),\n '',\n\n `## \u2705 Recent Achievements`,\n ...ledger.recent_achievements.map(\n (a) => `- ${a.description} \u2192 ${a.impact}`\n ),\n '',\n\n `## \u27A1\uFE0F Next Actions`,\n ...ledger.next_actions.map((a, i) => `${i + 1}. ${a}`),\n '',\n\n ledger.warnings.length > 0 ? `## \u26A0\uFE0F Warnings` : '',\n ...ledger.warnings.map((w) => `- ${w}`),\n ];\n\n return lines.filter((l) => l !== '').join('\\n');\n }\n\n /**\n * Check if /clear is recommended\n */\n async shouldClear(\n currentTokens: number,\n maxTokens: number\n ): Promise<{\n recommended: boolean;\n reason?: string;\n alternative?: string;\n }> {\n const usage = currentTokens / maxTokens;\n\n if (usage < this.CONTEXT_WARNING_THRESHOLD) {\n return { recommended: false };\n }\n\n // Check if we have redundant frames\n const frameStack = await this.frameManager.getStack();\n const redundantFrames = frameStack.frames.filter(\n (f) => f.status === 'closed' && !f.metadata?.critical\n ).length;\n\n if (usage >= this.CONTEXT_CRITICAL_THRESHOLD) {\n if (redundantFrames > 5) {\n return {\n recommended: true,\n reason: `Context at ${Math.round(usage * 100)}% with ${redundantFrames} closed frames`,\n alternative: 'Consider saving ledger and clearing',\n };\n }\n }\n\n return {\n recommended: false,\n alternative: `Context at ${Math.round(usage * 100)}% but manageable`,\n };\n }\n\n // Private helper methods\n\n private async getCompressedFrameStack(): Promise<FrameSummary[]> {\n const stack = await this.frameManager.getStack();\n\n return stack.frames.map((frame, index) => ({\n id: frame.id,\n type: frame.type,\n description: frame.description || 'Unnamed frame',\n depth: index,\n key_events: this.extractKeyEvents(frame),\n digest: frame.digest?.summary,\n }));\n }\n\n private extractKeyEvents(frame: Frame): string[] {\n const events: string[] = [];\n\n // Extract from metadata\n if (frame.metadata?.decision) {\n events.push(`Decision: ${frame.metadata.decision}`);\n }\n if (frame.metadata?.error) {\n events.push(`Error: ${frame.metadata.error}`);\n }\n if (frame.metadata?.achievement) {\n events.push(`Achievement: ${frame.metadata.achievement}`);\n }\n\n return events;\n }\n\n private async getCriticalDecisions(): Promise<Decision[]> {\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 100\n );\n\n return traces\n .filter((t) => t.type === 'decision')\n .map((t) => ({\n id: t.id,\n decision: t.content.decision || '',\n rationale: t.content.rationale || '',\n impact: this.assessImpact(t),\n still_applies: !t.metadata?.superseded,\n }))\n .filter((d) => d.impact !== 'low');\n }\n\n private assessImpact(trace: Trace): Decision['impact'] {\n const content = JSON.stringify(trace.content).toLowerCase();\n\n if (content.includes('architecture') || content.includes('critical')) {\n return 'critical';\n }\n if (content.includes('important') || content.includes('significant')) {\n return 'high';\n }\n if (content.includes('minor') || content.includes('small')) {\n return 'low';\n }\n\n return 'medium';\n }\n\n private async getActiveTasks(): Promise<Task[]> {\n const frames = await this.dbManager.getRecentFrames(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n return frames\n .filter((f) => f.type === 'task')\n .map((f) => ({\n id: f.id,\n title: f.description || 'Untitled task',\n status: this.getTaskStatus(f),\n priority: this.getTaskPriority(f),\n context: f.metadata?.context || '',\n }));\n }\n\n private getTaskStatus(frame: Frame): Task['status'] {\n if (frame.status === 'closed' && frame.metadata?.completed) {\n return 'completed';\n }\n if (frame.metadata?.blocked) return 'blocked';\n if (frame.status === 'open') return 'in_progress';\n return 'pending';\n }\n\n private getTaskPriority(frame: Frame): Task['priority'] {\n const priority = frame.metadata?.priority;\n if (['critical', 'high', 'medium', 'low'].includes(priority)) {\n return priority as Task['priority'];\n }\n return 'medium';\n }\n\n private async getCriticalContext(): Promise<string[]> {\n const context: string[] = [];\n\n // Add project-specific context\n const session = await this.dbManager.getSession(\n await this.dbManager.getCurrentSessionId()\n );\n if (session?.metadata?.key_facts) {\n context.push(...session.metadata.key_facts);\n }\n\n // Add recent important discoveries\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n const discoveries = traces\n .filter((t) => t.metadata?.important || t.type === 'discovery')\n .map((t) => t.content.summary || t.content.description)\n .filter(Boolean)\n .slice(0, 5);\n\n context.push(...discoveries);\n\n return context;\n }\n\n private async getRecentAchievements(): Promise<Achievement[]> {\n const frames = await this.dbManager.getRecentFrames(\n await this.dbManager.getCurrentSessionId(),\n 20\n );\n\n return frames\n .filter((f) => f.status === 'closed' && f.metadata?.achievement)\n .map((f) => ({\n description: f.metadata.achievement,\n impact: f.metadata.impact || 'completed task',\n timestamp: f.closedAt || f.createdAt,\n }))\n .slice(0, 5);\n }\n\n private async getCurrentFocus(): Promise<string> {\n const stack = await this.frameManager.getStack();\n const activeFrame = stack.frames.find((f) => f.status === 'open');\n\n if (!activeFrame) {\n return 'No active focus';\n }\n\n return `${activeFrame.type}: ${activeFrame.description || 'In progress'}`;\n }\n\n private async suggestNextActions(tasks: Task[]): Promise<string[]> {\n const suggestions: string[] = [];\n\n // Continue in-progress tasks\n const inProgress = tasks.filter((t) => t.status === 'in_progress');\n if (inProgress.length > 0) {\n suggestions.push(`Continue: ${inProgress[0].title}`);\n }\n\n // Start high-priority pending tasks\n const highPriority = tasks.filter(\n (t) => t.status === 'pending' && t.priority === 'high'\n );\n if (highPriority.length > 0) {\n suggestions.push(`Start: ${highPriority[0].title}`);\n }\n\n // Unblock blocked tasks\n const blocked = tasks.filter((t) => t.status === 'blocked');\n if (blocked.length > 0) {\n suggestions.push(`Unblock: ${blocked[0].title}`);\n }\n\n return suggestions.slice(0, 3);\n }\n\n private async getWarnings(): Promise<string[]> {\n const warnings: string[] = [];\n\n const tasks = await this.getActiveTasks();\n const blocked = tasks.filter((t) => t.status === 'blocked');\n\n if (blocked.length > 0) {\n warnings.push(`${blocked.length} tasks blocked`);\n }\n\n const critical = tasks.filter(\n (t) => t.priority === 'critical' && t.status !== 'completed'\n );\n if (critical.length > 0) {\n warnings.push(`${critical.length} critical tasks pending`);\n }\n\n return warnings;\n }\n\n private async estimateCurrentTokens(): Promise<number> {\n // Simplified estimation\n const frames = await this.frameManager.getStack();\n const traces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 100\n );\n\n const frameTokens = frames.frames.length * 200; // Rough estimate\n const traceTokens = traces.length * 100; // Rough estimate\n\n return frameTokens + traceTokens;\n }\n\n private estimateLedgerTokens(\n frames: FrameSummary[],\n decisions: Decision[],\n tasks: Task[]\n ): number {\n // Rough estimation\n return frames.length * 50 + decisions.length * 30 + tasks.length * 20;\n }\n\n private async saveLedgerToFile(ledger: ContinuityLedger): Promise<void> {\n await fs.mkdir(this.continuityPath, { recursive: true });\n\n // Save as CONTINUITY_CLAUDE-latest.json (overwrites)\n const latestPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.json'\n );\n await fs.writeFile(latestPath, JSON.stringify(ledger, null, 2), 'utf-8');\n\n // Also save markdown version\n const markdown = await this.generateLedgerMarkdown(ledger);\n const mdPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.md'\n );\n await fs.writeFile(mdPath, markdown, 'utf-8');\n }\n\n private async saveBackupLedger(ledger: ContinuityLedger): Promise<void> {\n await fs.mkdir(this.ledgerPath, { recursive: true });\n\n const timestamp = ledger.timestamp.replace(/[:.]/g, '-');\n const backupPath = path.join(this.ledgerPath, `ledger-${timestamp}.json`);\n await fs.writeFile(backupPath, JSON.stringify(ledger, null, 2), 'utf-8');\n }\n\n private async loadLatestLedger(): Promise<ContinuityLedger | null> {\n try {\n const latestPath = path.join(\n this.continuityPath,\n 'CONTINUITY_CLAUDE-latest.json'\n );\n const content = await fs.readFile(latestPath, 'utf-8');\n return JSON.parse(content) as ContinuityLedger;\n } catch (error: unknown) {\n return null;\n }\n }\n\n private async restoreFrameStructure(frames: FrameSummary[]): Promise<void> {\n // Create lightweight frame references (not full frames)\n for (const summary of frames) {\n await this.frameManager.push({\n type: summary.type,\n description: summary.description,\n metadata: {\n restored_from_ledger: true,\n original_id: summary.id,\n key_events: summary.key_events,\n digest: summary.digest,\n },\n });\n }\n }\n\n private async restoreDecisions(decisions: Decision[]): Promise<void> {\n for (const decision of decisions) {\n if (decision.still_applies) {\n await this.dbManager.addAnchor({\n type: 'decision',\n content: {\n decision: decision.decision,\n rationale: decision.rationale,\n impact: decision.impact,\n },\n metadata: {\n restored_from_ledger: true,\n original_id: decision.id,\n },\n });\n }\n }\n }\n\n private async restoreTasks(tasks: Task[]): Promise<void> {\n for (const task of tasks) {\n if (task.status !== 'completed') {\n await this.frameManager.push({\n type: 'task',\n description: task.title,\n metadata: {\n status: task.status,\n priority: task.priority,\n context: task.context,\n restored_from_ledger: true,\n original_id: task.id,\n },\n });\n }\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/session/handoff-generator.ts"],
4
- "sourcesContent": ["/**\n * Session Handoff Generator for StackMemory\n * Inspired by Continuous-Claude's handoff documents\n *\n * Generates structured transfer documents when sessions end\n * and loads them when new sessions begin\n */\n\nimport { Frame, Trace, Context } from '../types';\nimport { FrameManager } from '../frame/frame-manager';\nimport { DatabaseManager } from '../storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface HandoffDocument {\n session_id: string;\n timestamp: string;\n project: string;\n branch?: string;\n\n // Current state\n active_frame_path: string[];\n active_tasks: TaskSummary[];\n pending_decisions: DecisionPoint[];\n blockers: Blocker[];\n\n // Recent context\n recent_files: FileEdit[];\n recent_commands: CommandExecution[];\n recent_errors: ErrorContext[];\n\n // Key insights\n patterns_detected: string[];\n approaches_tried: ApproachSummary[];\n successful_strategies: string[];\n\n // Next steps\n suggested_next_actions: string[];\n warnings: string[];\n\n // Metrics\n session_duration_minutes: number;\n frames_created: number;\n tool_calls_made: number;\n decisions_recorded: number;\n}\n\ninterface TaskSummary {\n id: string;\n title: string;\n status: 'pending' | 'in_progress' | 'blocked' | 'completed';\n progress_percentage: number;\n blocker?: string;\n}\n\ninterface DecisionPoint {\n decision: string;\n rationale: string;\n alternatives_considered?: string[];\n timestamp: string;\n}\n\ninterface Blocker {\n description: string;\n attempted_solutions: string[];\n suggested_approach?: string;\n severity: 'low' | 'medium' | 'high' | 'critical';\n}\n\ninterface FileEdit {\n path: string;\n operations: ('created' | 'modified' | 'deleted')[];\n line_changes: { added: number; removed: number };\n}\n\ninterface CommandExecution {\n command: string;\n success: boolean;\n output_summary?: string;\n}\n\ninterface ErrorContext {\n error: string;\n context: string;\n resolved: boolean;\n resolution?: string;\n}\n\ninterface ApproachSummary {\n approach: string;\n outcome: 'successful' | 'failed' | 'partial';\n learnings?: string;\n}\n\nexport class HandoffGenerator {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private handoffDir: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n projectRoot: string\n ) {\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.handoffDir = path.join(projectRoot, '.stackmemory', 'handoffs');\n }\n\n /**\n * Generate a handoff document for the current session\n */\n async generateHandoff(sessionId: string): Promise<HandoffDocument> {\n const session = await this.dbManager.getSession(sessionId);\n if (!session) throw new Error(`Session ${sessionId} not found`);\n\n // Get active frame stack\n const activeFramePath = await this.getActiveFramePath();\n\n // Get recent activity\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n const recentFrames = await this.dbManager.getRecentFrames(sessionId, 20);\n\n // Extract key information\n const tasks = await this.extractTasks(recentFrames);\n const decisions = await this.extractDecisions(recentTraces);\n const blockers = await this.extractBlockers(recentTraces, recentFrames);\n const fileEdits = await this.extractFileEdits(recentTraces);\n const commands = await this.extractCommands(recentTraces);\n const errors = await this.extractErrors(recentTraces);\n const patterns = await this.detectPatterns(recentTraces);\n const approaches = await this.extractApproaches(recentFrames);\n\n // Calculate metrics\n const sessionDuration = Math.floor(\n (Date.now() - new Date(session.startedAt).getTime()) / 60000\n );\n\n const handoff: HandoffDocument = {\n session_id: sessionId,\n timestamp: new Date().toISOString(),\n project: session.project,\n branch: session.metadata?.branch,\n\n active_frame_path: activeFramePath,\n active_tasks: tasks,\n pending_decisions: decisions.filter((d) => !d.resolved),\n blockers: blockers,\n\n recent_files: fileEdits.slice(0, 10),\n recent_commands: commands.slice(0, 10),\n recent_errors: errors.slice(0, 5),\n\n patterns_detected: patterns,\n approaches_tried: approaches,\n successful_strategies: this.extractSuccessfulStrategies(approaches),\n\n suggested_next_actions: await this.suggestNextActions(\n tasks,\n blockers,\n activeFramePath\n ),\n warnings: await this.generateWarnings(errors, blockers),\n\n session_duration_minutes: sessionDuration,\n frames_created: recentFrames.length,\n tool_calls_made: recentTraces.filter((t) => t.type === 'tool_call')\n .length,\n decisions_recorded: decisions.length,\n };\n\n // Save to file\n await this.saveHandoff(handoff);\n\n return handoff;\n }\n\n /**\n * Load the most recent handoff document\n */\n async loadHandoff(): Promise<HandoffDocument | null> {\n try {\n await fs.mkdir(this.handoffDir, { recursive: true });\n\n const files = await fs.readdir(this.handoffDir);\n const handoffFiles = files\n .filter((f) => f.endsWith('.json'))\n .sort()\n .reverse();\n\n if (handoffFiles.length === 0) return null;\n\n const mostRecent = handoffFiles[0];\n const content = await fs.readFile(\n path.join(this.handoffDir, mostRecent),\n 'utf-8'\n );\n\n return JSON.parse(content) as HandoffDocument;\n } catch (error) {\n console.error('Error loading handoff:', error);\n return null;\n }\n }\n\n /**\n * Generate a markdown summary of the handoff\n */\n async generateMarkdownSummary(handoff: HandoffDocument): Promise<string> {\n const lines: string[] = [\n `# Session Handoff`,\n `**Generated**: ${new Date(handoff.timestamp).toLocaleString()}`,\n `**Project**: ${handoff.project}`,\n handoff.branch ? `**Branch**: ${handoff.branch}` : '',\n `**Duration**: ${handoff.session_duration_minutes} minutes`,\n '',\n\n `## Current Context`,\n `**Active Frame Path**: ${handoff.active_frame_path.join(' \u2192 ')}`,\n '',\n\n `## Active Tasks (${handoff.active_tasks.length})`,\n ...handoff.active_tasks.map(\n (t) =>\n `- [${t.status}] ${t.title} (${t.progress_percentage}%)${\n t.blocker ? ` \u26A0\uFE0F Blocked: ${t.blocker}` : ''\n }`\n ),\n '',\n\n handoff.blockers.length > 0 ? '## Blockers' : '',\n ...handoff.blockers.map(\n (b) =>\n `- **${b.severity}**: ${b.description}\\n Tried: ${b.attempted_solutions.join(\n ', '\n )}`\n ),\n '',\n\n handoff.pending_decisions.length > 0 ? '## Pending Decisions' : '',\n ...handoff.pending_decisions.map(\n (d) => `- **${d.decision}**\\n Rationale: ${d.rationale}`\n ),\n '',\n\n '## Recent Activity',\n `- Files edited: ${handoff.recent_files.length}`,\n `- Commands run: ${handoff.recent_commands.length}`,\n `- Errors encountered: ${handoff.recent_errors.length}`,\n '',\n\n handoff.patterns_detected.length > 0 ? '## Patterns Detected' : '',\n ...handoff.patterns_detected.map((p) => `- ${p}`),\n '',\n\n handoff.successful_strategies.length > 0\n ? '## Successful Strategies'\n : '',\n ...handoff.successful_strategies.map((s) => `- ${s}`),\n '',\n\n '## Suggested Next Actions',\n ...handoff.suggested_next_actions.map((a) => `1. ${a}`),\n '',\n\n handoff.warnings.length > 0 ? '## \u26A0\uFE0F Warnings' : '',\n ...handoff.warnings.map((w) => `- ${w}`),\n ];\n\n return lines.filter((l) => l !== '').join('\\n');\n }\n\n /**\n * Auto-detect session end and trigger handoff\n */\n async detectSessionEnd(sessionId: string): Promise<boolean> {\n const idleThreshold = 5 * 60 * 1000; // 5 minutes\n const lastActivity = await this.dbManager.getLastActivityTime(sessionId);\n\n if (!lastActivity) return false;\n\n const idleTime = Date.now() - lastActivity.getTime();\n if (idleTime > idleThreshold) {\n await this.generateHandoff(sessionId);\n return true;\n }\n\n return false;\n }\n\n // Private helper methods\n\n private async getActiveFramePath(): Promise<string[]> {\n const stack = await this.frameManager.getStack();\n return stack.frames.map((f) => f.description || f.type);\n }\n\n private async extractTasks(frames: Frame[]): Promise<TaskSummary[]> {\n return frames\n .filter((f) => f.type === 'task')\n .map((f) => ({\n id: f.id,\n title: f.description || 'Untitled task',\n status: this.getTaskStatus(f),\n progress_percentage: f.metadata?.progress || 0,\n blocker: f.metadata?.blocker,\n }));\n }\n\n private getTaskStatus(frame: Frame): TaskSummary['status'] {\n if (frame.status === 'closed') return 'completed';\n if (frame.metadata?.blocker) return 'blocked';\n if (frame.status === 'open') return 'in_progress';\n return 'pending';\n }\n\n private async extractDecisions(traces: Trace[]): Promise<DecisionPoint[]> {\n return traces\n .filter((t) => t.type === 'decision')\n .map((t) => ({\n decision: t.content.decision || '',\n rationale: t.content.rationale || '',\n alternatives_considered: t.content.alternatives,\n timestamp: t.timestamp,\n resolved: t.metadata?.resolved || false,\n }));\n }\n\n private async extractBlockers(\n traces: Trace[],\n frames: Frame[]\n ): Promise<Blocker[]> {\n const blockers: Blocker[] = [];\n\n // Extract from error traces\n const errorTraces = traces.filter(\n (t) => t.type === 'error' && !t.metadata?.resolved\n );\n\n for (const trace of errorTraces) {\n blockers.push({\n description: trace.content.error || 'Unknown error',\n attempted_solutions: trace.metadata?.attempts || [],\n suggested_approach: trace.metadata?.suggestion,\n severity: this.getErrorSeverity(trace),\n });\n }\n\n // Extract from blocked frames\n const blockedFrames = frames.filter((f) => f.metadata?.blocker);\n for (const frame of blockedFrames) {\n blockers.push({\n description: frame.metadata.blocker,\n attempted_solutions: frame.metadata.attempts || [],\n severity: 'medium',\n });\n }\n\n return blockers;\n }\n\n private getErrorSeverity(trace: Trace): Blocker['severity'] {\n const error = trace.content.error?.toLowerCase() || '';\n if (error.includes('critical') || error.includes('fatal'))\n return 'critical';\n if (error.includes('error') || error.includes('fail')) return 'high';\n if (error.includes('warning')) return 'medium';\n return 'low';\n }\n\n private async extractFileEdits(traces: Trace[]): Promise<FileEdit[]> {\n const fileMap = new Map<string, FileEdit>();\n\n const editTraces = traces.filter((t) =>\n ['edit', 'write', 'create', 'delete'].includes(t.type)\n );\n\n for (const trace of editTraces) {\n const path = trace.content.file_path || trace.content.path;\n if (!path) continue;\n\n if (!fileMap.has(path)) {\n fileMap.set(path, {\n path,\n operations: [],\n line_changes: { added: 0, removed: 0 },\n });\n }\n\n const file = fileMap.get(path)!;\n const op = this.getFileOperation(trace.type);\n if (!file.operations.includes(op)) {\n file.operations.push(op);\n }\n\n file.line_changes.added += trace.metadata?.lines_added || 0;\n file.line_changes.removed += trace.metadata?.lines_removed || 0;\n }\n\n return Array.from(fileMap.values());\n }\n\n private getFileOperation(traceType: string): FileEdit['operations'][0] {\n switch (traceType) {\n case 'create':\n case 'write':\n return 'created';\n case 'edit':\n return 'modified';\n case 'delete':\n return 'deleted';\n default:\n return 'modified';\n }\n }\n\n private async extractCommands(traces: Trace[]): Promise<CommandExecution[]> {\n return traces\n .filter((t) => t.type === 'bash' || t.type === 'command')\n .map((t) => ({\n command: t.content.command || '',\n success: !t.metadata?.error,\n output_summary: t.content.output?.substring(0, 100),\n }));\n }\n\n private async extractErrors(traces: Trace[]): Promise<ErrorContext[]> {\n return traces\n .filter((t) => t.type === 'error')\n .map((t) => ({\n error: t.content.error || '',\n context: t.content.context || '',\n resolved: t.metadata?.resolved || false,\n resolution: t.metadata?.resolution,\n }));\n }\n\n private async detectPatterns(traces: Trace[]): Promise<string[]> {\n const patterns: string[] = [];\n\n // Detect TDD pattern\n const testFirst = traces.some(\n (t) =>\n t.type === 'test' &&\n traces.some(\n (t2) => t2.type === 'implement' && t2.timestamp > t.timestamp\n )\n );\n if (testFirst) patterns.push('Test-Driven Development');\n\n // Detect refactoring pattern\n const refactoring =\n traces.filter(\n (t) =>\n t.content.description?.includes('refactor') ||\n t.metadata?.operation === 'refactor'\n ).length > 3;\n if (refactoring) patterns.push('Active Refactoring');\n\n // Detect debugging pattern\n const debugging =\n traces.filter((t) => t.type === 'error' || t.type === 'debug').length > 5;\n if (debugging) patterns.push('Deep Debugging Session');\n\n return patterns;\n }\n\n private async extractApproaches(frames: Frame[]): Promise<ApproachSummary[]> {\n return frames\n .filter((f) => f.metadata?.approach)\n .map((f) => ({\n approach: f.metadata.approach,\n outcome: this.getApproachOutcome(f),\n learnings: f.metadata.learnings,\n }));\n }\n\n private getApproachOutcome(frame: Frame): ApproachSummary['outcome'] {\n if (frame.status === 'closed' && frame.metadata?.success)\n return 'successful';\n if (frame.status === 'closed' && !frame.metadata?.success) return 'failed';\n return 'partial';\n }\n\n private extractSuccessfulStrategies(approaches: ApproachSummary[]): string[] {\n return approaches\n .filter((a) => a.outcome === 'successful')\n .map((a) => a.approach);\n }\n\n private async suggestNextActions(\n tasks: TaskSummary[],\n blockers: Blocker[],\n framePath: string[]\n ): Promise<string[]> {\n const suggestions: string[] = [];\n\n // Resume in-progress tasks\n const inProgress = tasks.filter((t) => t.status === 'in_progress');\n if (inProgress.length > 0) {\n suggestions.push(`Resume task: ${inProgress[0].title}`);\n }\n\n // Address critical blockers\n const criticalBlockers = blockers.filter((b) => b.severity === 'critical');\n if (criticalBlockers.length > 0) {\n suggestions.push(\n `Resolve critical blocker: ${criticalBlockers[0].description}`\n );\n }\n\n // Complete nearly done tasks\n const nearlyDone = tasks.filter((t) => t.progress_percentage >= 80);\n if (nearlyDone.length > 0) {\n suggestions.push(\n `Complete task: ${nearlyDone[0].title} (${nearlyDone[0].progress_percentage}% done)`\n );\n }\n\n return suggestions;\n }\n\n private async generateWarnings(\n errors: ErrorContext[],\n blockers: Blocker[]\n ): Promise<string[]> {\n const warnings: string[] = [];\n\n // Unresolved errors\n const unresolved = errors.filter((e) => !e.resolved);\n if (unresolved.length > 0) {\n warnings.push(`${unresolved.length} unresolved errors`);\n }\n\n // Critical blockers\n const critical = blockers.filter((b) => b.severity === 'critical');\n if (critical.length > 0) {\n warnings.push(\n `${critical.length} critical blockers need immediate attention`\n );\n }\n\n return warnings;\n }\n\n private async saveHandoff(handoff: HandoffDocument): Promise<void> {\n await fs.mkdir(this.handoffDir, { recursive: true });\n\n const filename = `${handoff.timestamp.replace(/[:.]/g, '-')}.json`;\n const filepath = path.join(this.handoffDir, filename);\n\n await fs.writeFile(filepath, JSON.stringify(handoff, null, 2), 'utf-8');\n\n // Also save markdown summary\n const markdown = await this.generateMarkdownSummary(handoff);\n const mdPath = filepath.replace('.json', '.md');\n await fs.writeFile(mdPath, markdown, 'utf-8');\n }\n}\n"],
5
- "mappings": "AAWA,YAAY,QAAQ;AACpB,YAAY,UAAU;AAkFf,MAAM,iBAAiB;AAAA,EACpB;AAAA,EACA;AAAA,EACA;AAAA,EAER,YACE,cACA,WACA,aACA;AACA,SAAK,eAAe;AACpB,SAAK,YAAY;AACjB,SAAK,aAAa,KAAK,KAAK,aAAa,gBAAgB,UAAU;AAAA,EACrE;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,gBAAgB,WAA6C;AACjE,UAAM,UAAU,MAAM,KAAK,UAAU,WAAW,SAAS;AACzD,QAAI,CAAC,QAAS,OAAM,IAAI,MAAM,WAAW,SAAS,YAAY;AAG9D,UAAM,kBAAkB,MAAM,KAAK,mBAAmB;AAGtD,UAAM,eAAe,MAAM,KAAK,UAAU,gBAAgB,WAAW,GAAG;AACxE,UAAM,eAAe,MAAM,KAAK,UAAU,gBAAgB,WAAW,EAAE;AAGvE,UAAM,QAAQ,MAAM,KAAK,aAAa,YAAY;AAClD,UAAM,YAAY,MAAM,KAAK,iBAAiB,YAAY;AAC1D,UAAM,WAAW,MAAM,KAAK,gBAAgB,cAAc,YAAY;AACtE,UAAM,YAAY,MAAM,KAAK,iBAAiB,YAAY;AAC1D,UAAM,WAAW,MAAM,KAAK,gBAAgB,YAAY;AACxD,UAAM,SAAS,MAAM,KAAK,cAAc,YAAY;AACpD,UAAM,WAAW,MAAM,KAAK,eAAe,YAAY;AACvD,UAAM,aAAa,MAAM,KAAK,kBAAkB,YAAY;AAG5D,UAAM,kBAAkB,KAAK;AAAA,OAC1B,KAAK,IAAI,IAAI,IAAI,KAAK,QAAQ,SAAS,EAAE,QAAQ,KAAK;AAAA,IACzD;AAEA,UAAM,UAA2B;AAAA,MAC/B,YAAY;AAAA,MACZ,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,MAClC,SAAS,QAAQ;AAAA,MACjB,QAAQ,QAAQ,UAAU;AAAA,MAE1B,mBAAmB;AAAA,MACnB,cAAc;AAAA,MACd,mBAAmB,UAAU,OAAO,CAAC,MAAM,CAAC,EAAE,QAAQ;AAAA,MACtD;AAAA,MAEA,cAAc,UAAU,MAAM,GAAG,EAAE;AAAA,MACnC,iBAAiB,SAAS,MAAM,GAAG,EAAE;AAAA,MACrC,eAAe,OAAO,MAAM,GAAG,CAAC;AAAA,MAEhC,mBAAmB;AAAA,MACnB,kBAAkB;AAAA,MAClB,uBAAuB,KAAK,4BAA4B,UAAU;AAAA,MAElE,wBAAwB,MAAM,KAAK;AAAA,QACjC;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,MACA,UAAU,MAAM,KAAK,iBAAiB,QAAQ,QAAQ;AAAA,MAEtD,0BAA0B;AAAA,MAC1B,gBAAgB,aAAa;AAAA,MAC7B,iBAAiB,aAAa,OAAO,CAAC,MAAM,EAAE,SAAS,WAAW,EAC/D;AAAA,MACH,oBAAoB,UAAU;AAAA,IAChC;AAGA,UAAM,KAAK,YAAY,OAAO;AAE9B,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,cAA+C;AACnD,QAAI;AACF,YAAM,GAAG,MAAM,KAAK,YAAY,EAAE,WAAW,KAAK,CAAC;AAEnD,YAAM,QAAQ,MAAM,GAAG,QAAQ,KAAK,UAAU;AAC9C,YAAM,eAAe,MAClB,OAAO,CAAC,MAAM,EAAE,SAAS,OAAO,CAAC,EACjC,KAAK,EACL,QAAQ;AAEX,UAAI,aAAa,WAAW,EAAG,QAAO;AAEtC,YAAM,aAAa,aAAa,CAAC;AACjC,YAAM,UAAU,MAAM,GAAG;AAAA,QACvB,KAAK,KAAK,KAAK,YAAY,UAAU;AAAA,QACrC;AAAA,MACF;AAEA,aAAO,KAAK,MAAM,OAAO;AAAA,IAC3B,SAAS,OAAO;AACd,cAAQ,MAAM,0BAA0B,KAAK;AAC7C,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,wBAAwB,SAA2C;AACvE,UAAM,QAAkB;AAAA,MACtB;AAAA,MACA,kBAAkB,IAAI,KAAK,QAAQ,SAAS,EAAE,eAAe,CAAC;AAAA,MAC9D,gBAAgB,QAAQ,OAAO;AAAA,MAC/B,QAAQ,SAAS,eAAe,QAAQ,MAAM,KAAK;AAAA,MACnD,iBAAiB,QAAQ,wBAAwB;AAAA,MACjD;AAAA,MAEA;AAAA,MACA,0BAA0B,QAAQ,kBAAkB,KAAK,UAAK,CAAC;AAAA,MAC/D;AAAA,MAEA,oBAAoB,QAAQ,aAAa,MAAM;AAAA,MAC/C,GAAG,QAAQ,aAAa;AAAA,QACtB,CAAC,MACC,MAAM,EAAE,MAAM,KAAK,EAAE,KAAK,KAAK,EAAE,mBAAmB,KAClD,EAAE,UAAU,0BAAgB,EAAE,OAAO,KAAK,EAC5C;AAAA,MACJ;AAAA,MACA;AAAA,MAEA,QAAQ,SAAS,SAAS,IAAI,gBAAgB;AAAA,MAC9C,GAAG,QAAQ,SAAS;AAAA,QAClB,CAAC,MACC,OAAO,EAAE,QAAQ,OAAO,EAAE,WAAW;AAAA,WAAc,EAAE,oBAAoB;AAAA,UACvE;AAAA,QACF,CAAC;AAAA,MACL;AAAA,MACA;AAAA,MAEA,QAAQ,kBAAkB,SAAS,IAAI,yBAAyB;AAAA,MAChE,GAAG,QAAQ,kBAAkB;AAAA,QAC3B,CAAC,MAAM,OAAO,EAAE,QAAQ;AAAA,eAAoB,EAAE,SAAS;AAAA,MACzD;AAAA,MACA;AAAA,MAEA;AAAA,MACA,mBAAmB,QAAQ,aAAa,MAAM;AAAA,MAC9C,mBAAmB,QAAQ,gBAAgB,MAAM;AAAA,MACjD,yBAAyB,QAAQ,cAAc,MAAM;AAAA,MACrD;AAAA,MAEA,QAAQ,kBAAkB,SAAS,IAAI,yBAAyB;AAAA,MAChE,GAAG,QAAQ,kBAAkB,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;AAAA,MAChD;AAAA,MAEA,QAAQ,sBAAsB,SAAS,IACnC,6BACA;AAAA,MACJ,GAAG,QAAQ,sBAAsB,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;AAAA,MACpD;AAAA,MAEA;AAAA,MACA,GAAG,QAAQ,uBAAuB,IAAI,CAAC,MAAM,MAAM,CAAC,EAAE;AAAA,MACtD;AAAA,MAEA,QAAQ,SAAS,SAAS,IAAI,6BAAmB;AAAA,MACjD,GAAG,QAAQ,SAAS,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;AAAA,IACzC;AAEA,WAAO,MAAM,OAAO,CAAC,MAAM,MAAM,EAAE,EAAE,KAAK,IAAI;AAAA,EAChD;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBAAiB,WAAqC;AAC1D,UAAM,gBAAgB,IAAI,KAAK;AAC/B,UAAM,eAAe,MAAM,KAAK,UAAU,oBAAoB,SAAS;AAEvE,QAAI,CAAC,aAAc,QAAO;AAE1B,UAAM,WAAW,KAAK,IAAI,IAAI,aAAa,QAAQ;AACnD,QAAI,WAAW,eAAe;AAC5B,YAAM,KAAK,gBAAgB,SAAS;AACpC,aAAO;AAAA,IACT;AAEA,WAAO;AAAA,EACT;AAAA;AAAA,EAIA,MAAc,qBAAwC;AACpD,UAAM,QAAQ,MAAM,KAAK,aAAa,SAAS;AAC/C,WAAO,MAAM,OAAO,IAAI,CAAC,MAAM,EAAE,eAAe,EAAE,IAAI;AAAA,EACxD;AAAA,EAEA,MAAc,aAAa,QAAyC;AAClE,WAAO,OACJ,OAAO,CAAC,MAAM,EAAE,SAAS,MAAM,EAC/B,IAAI,CAAC,OAAO;AAAA,MACX,IAAI,EAAE;AAAA,MACN,OAAO,EAAE,eAAe;AAAA,MACxB,QAAQ,KAAK,cAAc,CAAC;AAAA,MAC5B,qBAAqB,EAAE,UAAU,YAAY;AAAA,MAC7C,SAAS,EAAE,UAAU;AAAA,IACvB,EAAE;AAAA,EACN;AAAA,EAEQ,cAAc,OAAqC;AACzD,QAAI,MAAM,WAAW,SAAU,QAAO;AACtC,QAAI,MAAM,UAAU,QAAS,QAAO;AACpC,QAAI,MAAM,WAAW,OAAQ,QAAO;AACpC,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBAAiB,QAA2C;AACxE,WAAO,OACJ,OAAO,CAAC,MAAM,EAAE,SAAS,UAAU,EACnC,IAAI,CAAC,OAAO;AAAA,MACX,UAAU,EAAE,QAAQ,YAAY;AAAA,MAChC,WAAW,EAAE,QAAQ,aAAa;AAAA,MAClC,yBAAyB,EAAE,QAAQ;AAAA,MACnC,WAAW,EAAE;AAAA,MACb,UAAU,EAAE,UAAU,YAAY;AAAA,IACpC,EAAE;AAAA,EACN;AAAA,EAEA,MAAc,gBACZ,QACA,QACoB;AACpB,UAAM,WAAsB,CAAC;AAG7B,UAAM,cAAc,OAAO;AAAA,MACzB,CAAC,MAAM,EAAE,SAAS,WAAW,CAAC,EAAE,UAAU;AAAA,IAC5C;AAEA,eAAW,SAAS,aAAa;AAC/B,eAAS,KAAK;AAAA,QACZ,aAAa,MAAM,QAAQ,SAAS;AAAA,QACpC,qBAAqB,MAAM,UAAU,YAAY,CAAC;AAAA,QAClD,oBAAoB,MAAM,UAAU;AAAA,QACpC,UAAU,KAAK,iBAAiB,KAAK;AAAA,MACvC,CAAC;AAAA,IACH;AAGA,UAAM,gBAAgB,OAAO,OAAO,CAAC,MAAM,EAAE,UAAU,OAAO;AAC9D,eAAW,SAAS,eAAe;AACjC,eAAS,KAAK;AAAA,QACZ,aAAa,MAAM,SAAS;AAAA,QAC5B,qBAAqB,MAAM,SAAS,YAAY,CAAC;AAAA,QACjD,UAAU;AAAA,MACZ,CAAC;AAAA,IACH;AAEA,WAAO;AAAA,EACT;AAAA,EAEQ,iBAAiB,OAAmC;AAC1D,UAAM,QAAQ,MAAM,QAAQ,OAAO,YAAY,KAAK;AACpD,QAAI,MAAM,SAAS,UAAU,KAAK,MAAM,SAAS,OAAO;AACtD,aAAO;AACT,QAAI,MAAM,SAAS,OAAO,KAAK,MAAM,SAAS,MAAM,EAAG,QAAO;AAC9D,QAAI,MAAM,SAAS,SAAS,EAAG,QAAO;AACtC,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBAAiB,QAAsC;AACnE,UAAM,UAAU,oBAAI,IAAsB;AAE1C,UAAM,aAAa,OAAO;AAAA,MAAO,CAAC,MAChC,CAAC,QAAQ,SAAS,UAAU,QAAQ,EAAE,SAAS,EAAE,IAAI;AAAA,IACvD;AAEA,eAAW,SAAS,YAAY;AAC9B,YAAMA,QAAO,MAAM,QAAQ,aAAa,MAAM,QAAQ;AACtD,UAAI,CAACA,MAAM;AAEX,UAAI,CAAC,QAAQ,IAAIA,KAAI,GAAG;AACtB,gBAAQ,IAAIA,OAAM;AAAA,UAChB,MAAAA;AAAA,UACA,YAAY,CAAC;AAAA,UACb,cAAc,EAAE,OAAO,GAAG,SAAS,EAAE;AAAA,QACvC,CAAC;AAAA,MACH;AAEA,YAAM,OAAO,QAAQ,IAAIA,KAAI;AAC7B,YAAM,KAAK,KAAK,iBAAiB,MAAM,IAAI;AAC3C,UAAI,CAAC,KAAK,WAAW,SAAS,EAAE,GAAG;AACjC,aAAK,WAAW,KAAK,EAAE;AAAA,MACzB;AAEA,WAAK,aAAa,SAAS,MAAM,UAAU,eAAe;AAC1D,WAAK,aAAa,WAAW,MAAM,UAAU,iBAAiB;AAAA,IAChE;AAEA,WAAO,MAAM,KAAK,QAAQ,OAAO,CAAC;AAAA,EACpC;AAAA,EAEQ,iBAAiB,WAA8C;AACrE,YAAQ,WAAW;AAAA,MACjB,KAAK;AAAA,MACL,KAAK;AACH,eAAO;AAAA,MACT,KAAK;AACH,eAAO;AAAA,MACT,KAAK;AACH,eAAO;AAAA,MACT;AACE,eAAO;AAAA,IACX;AAAA,EACF;AAAA,EAEA,MAAc,gBAAgB,QAA8C;AAC1E,WAAO,OACJ,OAAO,CAAC,MAAM,EAAE,SAAS,UAAU,EAAE,SAAS,SAAS,EACvD,IAAI,CAAC,OAAO;AAAA,MACX,SAAS,EAAE,QAAQ,WAAW;AAAA,MAC9B,SAAS,CAAC,EAAE,UAAU;AAAA,MACtB,gBAAgB,EAAE,QAAQ,QAAQ,UAAU,GAAG,GAAG;AAAA,IACpD,EAAE;AAAA,EACN;AAAA,EAEA,MAAc,cAAc,QAA0C;AACpE,WAAO,OACJ,OAAO,CAAC,MAAM,EAAE,SAAS,OAAO,EAChC,IAAI,CAAC,OAAO;AAAA,MACX,OAAO,EAAE,QAAQ,SAAS;AAAA,MAC1B,SAAS,EAAE,QAAQ,WAAW;AAAA,MAC9B,UAAU,EAAE,UAAU,YAAY;AAAA,MAClC,YAAY,EAAE,UAAU;AAAA,IAC1B,EAAE;AAAA,EACN;AAAA,EAEA,MAAc,eAAe,QAAoC;AAC/D,UAAM,WAAqB,CAAC;AAG5B,UAAM,YAAY,OAAO;AAAA,MACvB,CAAC,MACC,EAAE,SAAS,UACX,OAAO;AAAA,QACL,CAAC,OAAO,GAAG,SAAS,eAAe,GAAG,YAAY,EAAE;AAAA,MACtD;AAAA,IACJ;AACA,QAAI,UAAW,UAAS,KAAK,yBAAyB;AAGtD,UAAM,cACJ,OAAO;AAAA,MACL,CAAC,MACC,EAAE,QAAQ,aAAa,SAAS,UAAU,KAC1C,EAAE,UAAU,cAAc;AAAA,IAC9B,EAAE,SAAS;AACb,QAAI,YAAa,UAAS,KAAK,oBAAoB;AAGnD,UAAM,YACJ,OAAO,OAAO,CAAC,MAAM,EAAE,SAAS,WAAW,EAAE,SAAS,OAAO,EAAE,SAAS;AAC1E,QAAI,UAAW,UAAS,KAAK,wBAAwB;AAErD,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,kBAAkB,QAA6C;AAC3E,WAAO,OACJ,OAAO,CAAC,MAAM,EAAE,UAAU,QAAQ,EAClC,IAAI,CAAC,OAAO;AAAA,MACX,UAAU,EAAE,SAAS;AAAA,MACrB,SAAS,KAAK,mBAAmB,CAAC;AAAA,MAClC,WAAW,EAAE,SAAS;AAAA,IACxB,EAAE;AAAA,EACN;AAAA,EAEQ,mBAAmB,OAA0C;AACnE,QAAI,MAAM,WAAW,YAAY,MAAM,UAAU;AAC/C,aAAO;AACT,QAAI,MAAM,WAAW,YAAY,CAAC,MAAM,UAAU,QAAS,QAAO;AAClE,WAAO;AAAA,EACT;AAAA,EAEQ,4BAA4B,YAAyC;AAC3E,WAAO,WACJ,OAAO,CAAC,MAAM,EAAE,YAAY,YAAY,EACxC,IAAI,CAAC,MAAM,EAAE,QAAQ;AAAA,EAC1B;AAAA,EAEA,MAAc,mBACZ,OACA,UACA,WACmB;AACnB,UAAM,cAAwB,CAAC;AAG/B,UAAM,aAAa,MAAM,OAAO,CAAC,MAAM,EAAE,WAAW,aAAa;AACjE,QAAI,WAAW,SAAS,GAAG;AACzB,kBAAY,KAAK,gBAAgB,WAAW,CAAC,EAAE,KAAK,EAAE;AAAA,IACxD;AAGA,UAAM,mBAAmB,SAAS,OAAO,CAAC,MAAM,EAAE,aAAa,UAAU;AACzE,QAAI,iBAAiB,SAAS,GAAG;AAC/B,kBAAY;AAAA,QACV,6BAA6B,iBAAiB,CAAC,EAAE,WAAW;AAAA,MAC9D;AAAA,IACF;AAGA,UAAM,aAAa,MAAM,OAAO,CAAC,MAAM,EAAE,uBAAuB,EAAE;AAClE,QAAI,WAAW,SAAS,GAAG;AACzB,kBAAY;AAAA,QACV,kBAAkB,WAAW,CAAC,EAAE,KAAK,KAAK,WAAW,CAAC,EAAE,mBAAmB;AAAA,MAC7E;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBACZ,QACA,UACmB;AACnB,UAAM,WAAqB,CAAC;AAG5B,UAAM,aAAa,OAAO,OAAO,CAAC,MAAM,CAAC,EAAE,QAAQ;AACnD,QAAI,WAAW,SAAS,GAAG;AACzB,eAAS,KAAK,GAAG,WAAW,MAAM,oBAAoB;AAAA,IACxD;AAGA,UAAM,WAAW,SAAS,OAAO,CAAC,MAAM,EAAE,aAAa,UAAU;AACjE,QAAI,SAAS,SAAS,GAAG;AACvB,eAAS;AAAA,QACP,GAAG,SAAS,MAAM;AAAA,MACpB;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,YAAY,SAAyC;AACjE,UAAM,GAAG,MAAM,KAAK,YAAY,EAAE,WAAW,KAAK,CAAC;AAEnD,UAAM,WAAW,GAAG,QAAQ,UAAU,QAAQ,SAAS,GAAG,CAAC;AAC3D,UAAM,WAAW,KAAK,KAAK,KAAK,YAAY,QAAQ;AAEpD,UAAM,GAAG,UAAU,UAAU,KAAK,UAAU,SAAS,MAAM,CAAC,GAAG,OAAO;AAGtE,UAAM,WAAW,MAAM,KAAK,wBAAwB,OAAO;AAC3D,UAAM,SAAS,SAAS,QAAQ,SAAS,KAAK;AAC9C,UAAM,GAAG,UAAU,QAAQ,UAAU,OAAO;AAAA,EAC9C;AACF;",
4
+ "sourcesContent": ["/**\n * Session Handoff Generator for StackMemory\n * Inspired by Continuous-Claude's handoff documents\n *\n * Generates structured transfer documents when sessions end\n * and loads them when new sessions begin\n */\n\nimport { Frame, Trace, Context } from '../types';\nimport { FrameManager } from '../frame/frame-manager';\nimport { DatabaseManager } from '../storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport interface HandoffDocument {\n session_id: string;\n timestamp: string;\n project: string;\n branch?: string;\n\n // Current state\n active_frame_path: string[];\n active_tasks: TaskSummary[];\n pending_decisions: DecisionPoint[];\n blockers: Blocker[];\n\n // Recent context\n recent_files: FileEdit[];\n recent_commands: CommandExecution[];\n recent_errors: ErrorContext[];\n\n // Key insights\n patterns_detected: string[];\n approaches_tried: ApproachSummary[];\n successful_strategies: string[];\n\n // Next steps\n suggested_next_actions: string[];\n warnings: string[];\n\n // Metrics\n session_duration_minutes: number;\n frames_created: number;\n tool_calls_made: number;\n decisions_recorded: number;\n}\n\ninterface TaskSummary {\n id: string;\n title: string;\n status: 'pending' | 'in_progress' | 'blocked' | 'completed';\n progress_percentage: number;\n blocker?: string;\n}\n\ninterface DecisionPoint {\n decision: string;\n rationale: string;\n alternatives_considered?: string[];\n timestamp: string;\n}\n\ninterface Blocker {\n description: string;\n attempted_solutions: string[];\n suggested_approach?: string;\n severity: 'low' | 'medium' | 'high' | 'critical';\n}\n\ninterface FileEdit {\n path: string;\n operations: ('created' | 'modified' | 'deleted')[];\n line_changes: { added: number; removed: number };\n}\n\ninterface CommandExecution {\n command: string;\n success: boolean;\n output_summary?: string;\n}\n\ninterface ErrorContext {\n error: string;\n context: string;\n resolved: boolean;\n resolution?: string;\n}\n\ninterface ApproachSummary {\n approach: string;\n outcome: 'successful' | 'failed' | 'partial';\n learnings?: string;\n}\n\nexport class HandoffGenerator {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private handoffDir: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n projectRoot: string\n ) {\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.handoffDir = path.join(projectRoot, '.stackmemory', 'handoffs');\n }\n\n /**\n * Generate a handoff document for the current session\n */\n async generateHandoff(sessionId: string): Promise<HandoffDocument> {\n const session = await this.dbManager.getSession(sessionId);\n if (!session) throw new Error(`Session ${sessionId} not found`);\n\n // Get active frame stack\n const activeFramePath = await this.getActiveFramePath();\n\n // Get recent activity\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n const recentFrames = await this.dbManager.getRecentFrames(sessionId, 20);\n\n // Extract key information\n const tasks = await this.extractTasks(recentFrames);\n const decisions = await this.extractDecisions(recentTraces);\n const blockers = await this.extractBlockers(recentTraces, recentFrames);\n const fileEdits = await this.extractFileEdits(recentTraces);\n const commands = await this.extractCommands(recentTraces);\n const errors = await this.extractErrors(recentTraces);\n const patterns = await this.detectPatterns(recentTraces);\n const approaches = await this.extractApproaches(recentFrames);\n\n // Calculate metrics\n const sessionDuration = Math.floor(\n (Date.now() - new Date(session.startedAt).getTime()) / 60000\n );\n\n const handoff: HandoffDocument = {\n session_id: sessionId,\n timestamp: new Date().toISOString(),\n project: session.project,\n branch: session.metadata?.branch,\n\n active_frame_path: activeFramePath,\n active_tasks: tasks,\n pending_decisions: decisions.filter((d) => !d.resolved),\n blockers: blockers,\n\n recent_files: fileEdits.slice(0, 10),\n recent_commands: commands.slice(0, 10),\n recent_errors: errors.slice(0, 5),\n\n patterns_detected: patterns,\n approaches_tried: approaches,\n successful_strategies: this.extractSuccessfulStrategies(approaches),\n\n suggested_next_actions: await this.suggestNextActions(\n tasks,\n blockers,\n activeFramePath\n ),\n warnings: await this.generateWarnings(errors, blockers),\n\n session_duration_minutes: sessionDuration,\n frames_created: recentFrames.length,\n tool_calls_made: recentTraces.filter((t) => t.type === 'tool_call')\n .length,\n decisions_recorded: decisions.length,\n };\n\n // Save to file\n await this.saveHandoff(handoff);\n\n return handoff;\n }\n\n /**\n * Load the most recent handoff document\n */\n async loadHandoff(): Promise<HandoffDocument | null> {\n try {\n await fs.mkdir(this.handoffDir, { recursive: true });\n\n const files = await fs.readdir(this.handoffDir);\n const handoffFiles = files\n .filter((f) => f.endsWith('.json'))\n .sort()\n .reverse();\n\n if (handoffFiles.length === 0) return null;\n\n const mostRecent = handoffFiles[0];\n const content = await fs.readFile(\n path.join(this.handoffDir, mostRecent),\n 'utf-8'\n );\n\n return JSON.parse(content) as HandoffDocument;\n } catch (error: unknown) {\n console.error('Error loading handoff:', error);\n return null;\n }\n }\n\n /**\n * Generate a markdown summary of the handoff\n */\n async generateMarkdownSummary(handoff: HandoffDocument): Promise<string> {\n const lines: string[] = [\n `# Session Handoff`,\n `**Generated**: ${new Date(handoff.timestamp).toLocaleString()}`,\n `**Project**: ${handoff.project}`,\n handoff.branch ? `**Branch**: ${handoff.branch}` : '',\n `**Duration**: ${handoff.session_duration_minutes} minutes`,\n '',\n\n `## Current Context`,\n `**Active Frame Path**: ${handoff.active_frame_path.join(' \u2192 ')}`,\n '',\n\n `## Active Tasks (${handoff.active_tasks.length})`,\n ...handoff.active_tasks.map(\n (t) =>\n `- [${t.status}] ${t.title} (${t.progress_percentage}%)${\n t.blocker ? ` \u26A0\uFE0F Blocked: ${t.blocker}` : ''\n }`\n ),\n '',\n\n handoff.blockers.length > 0 ? '## Blockers' : '',\n ...handoff.blockers.map(\n (b) =>\n `- **${b.severity}**: ${b.description}\\n Tried: ${b.attempted_solutions.join(\n ', '\n )}`\n ),\n '',\n\n handoff.pending_decisions.length > 0 ? '## Pending Decisions' : '',\n ...handoff.pending_decisions.map(\n (d) => `- **${d.decision}**\\n Rationale: ${d.rationale}`\n ),\n '',\n\n '## Recent Activity',\n `- Files edited: ${handoff.recent_files.length}`,\n `- Commands run: ${handoff.recent_commands.length}`,\n `- Errors encountered: ${handoff.recent_errors.length}`,\n '',\n\n handoff.patterns_detected.length > 0 ? '## Patterns Detected' : '',\n ...handoff.patterns_detected.map((p) => `- ${p}`),\n '',\n\n handoff.successful_strategies.length > 0\n ? '## Successful Strategies'\n : '',\n ...handoff.successful_strategies.map((s) => `- ${s}`),\n '',\n\n '## Suggested Next Actions',\n ...handoff.suggested_next_actions.map((a) => `1. ${a}`),\n '',\n\n handoff.warnings.length > 0 ? '## \u26A0\uFE0F Warnings' : '',\n ...handoff.warnings.map((w) => `- ${w}`),\n ];\n\n return lines.filter((l) => l !== '').join('\\n');\n }\n\n /**\n * Auto-detect session end and trigger handoff\n */\n async detectSessionEnd(sessionId: string): Promise<boolean> {\n const idleThreshold = 5 * 60 * 1000; // 5 minutes\n const lastActivity = await this.dbManager.getLastActivityTime(sessionId);\n\n if (!lastActivity) return false;\n\n const idleTime = Date.now() - lastActivity.getTime();\n if (idleTime > idleThreshold) {\n await this.generateHandoff(sessionId);\n return true;\n }\n\n return false;\n }\n\n // Private helper methods\n\n private async getActiveFramePath(): Promise<string[]> {\n const stack = await this.frameManager.getStack();\n return stack.frames.map((f) => f.description || f.type);\n }\n\n private async extractTasks(frames: Frame[]): Promise<TaskSummary[]> {\n return frames\n .filter((f) => f.type === 'task')\n .map((f) => ({\n id: f.id,\n title: f.description || 'Untitled task',\n status: this.getTaskStatus(f),\n progress_percentage: f.metadata?.progress || 0,\n blocker: f.metadata?.blocker,\n }));\n }\n\n private getTaskStatus(frame: Frame): TaskSummary['status'] {\n if (frame.status === 'closed') return 'completed';\n if (frame.metadata?.blocker) return 'blocked';\n if (frame.status === 'open') return 'in_progress';\n return 'pending';\n }\n\n private async extractDecisions(traces: Trace[]): Promise<DecisionPoint[]> {\n return traces\n .filter((t) => t.type === 'decision')\n .map((t) => ({\n decision: t.content.decision || '',\n rationale: t.content.rationale || '',\n alternatives_considered: t.content.alternatives,\n timestamp: t.timestamp,\n resolved: t.metadata?.resolved || false,\n }));\n }\n\n private async extractBlockers(\n traces: Trace[],\n frames: Frame[]\n ): Promise<Blocker[]> {\n const blockers: Blocker[] = [];\n\n // Extract from error traces\n const errorTraces = traces.filter(\n (t) => t.type === 'error' && !t.metadata?.resolved\n );\n\n for (const trace of errorTraces) {\n blockers.push({\n description: trace.content.error || 'Unknown error',\n attempted_solutions: trace.metadata?.attempts || [],\n suggested_approach: trace.metadata?.suggestion,\n severity: this.getErrorSeverity(trace),\n });\n }\n\n // Extract from blocked frames\n const blockedFrames = frames.filter((f) => f.metadata?.blocker);\n for (const frame of blockedFrames) {\n blockers.push({\n description: frame.metadata.blocker,\n attempted_solutions: frame.metadata.attempts || [],\n severity: 'medium',\n });\n }\n\n return blockers;\n }\n\n private getErrorSeverity(trace: Trace): Blocker['severity'] {\n const error = trace.content.error?.toLowerCase() || '';\n if (error.includes('critical') || error.includes('fatal'))\n return 'critical';\n if (error.includes('error') || error.includes('fail')) return 'high';\n if (error.includes('warning')) return 'medium';\n return 'low';\n }\n\n private async extractFileEdits(traces: Trace[]): Promise<FileEdit[]> {\n const fileMap = new Map<string, FileEdit>();\n\n const editTraces = traces.filter((t) =>\n ['edit', 'write', 'create', 'delete'].includes(t.type)\n );\n\n for (const trace of editTraces) {\n const path = trace.content.file_path || trace.content.path;\n if (!path) continue;\n\n if (!fileMap.has(path)) {\n fileMap.set(path, {\n path,\n operations: [],\n line_changes: { added: 0, removed: 0 },\n });\n }\n\n const file = fileMap.get(path)!;\n const op = this.getFileOperation(trace.type);\n if (!file.operations.includes(op)) {\n file.operations.push(op);\n }\n\n file.line_changes.added += trace.metadata?.lines_added || 0;\n file.line_changes.removed += trace.metadata?.lines_removed || 0;\n }\n\n return Array.from(fileMap.values());\n }\n\n private getFileOperation(traceType: string): FileEdit['operations'][0] {\n switch (traceType) {\n case 'create':\n case 'write':\n return 'created';\n case 'edit':\n return 'modified';\n case 'delete':\n return 'deleted';\n default:\n return 'modified';\n }\n }\n\n private async extractCommands(traces: Trace[]): Promise<CommandExecution[]> {\n return traces\n .filter((t) => t.type === 'bash' || t.type === 'command')\n .map((t) => ({\n command: t.content.command || '',\n success: !t.metadata?.error,\n output_summary: t.content.output?.substring(0, 100),\n }));\n }\n\n private async extractErrors(traces: Trace[]): Promise<ErrorContext[]> {\n return traces\n .filter((t) => t.type === 'error')\n .map((t) => ({\n error: t.content.error || '',\n context: t.content.context || '',\n resolved: t.metadata?.resolved || false,\n resolution: t.metadata?.resolution,\n }));\n }\n\n private async detectPatterns(traces: Trace[]): Promise<string[]> {\n const patterns: string[] = [];\n\n // Detect TDD pattern\n const testFirst = traces.some(\n (t) =>\n t.type === 'test' &&\n traces.some(\n (t2) => t2.type === 'implement' && t2.timestamp > t.timestamp\n )\n );\n if (testFirst) patterns.push('Test-Driven Development');\n\n // Detect refactoring pattern\n const refactoring =\n traces.filter(\n (t) =>\n t.content.description?.includes('refactor') ||\n t.metadata?.operation === 'refactor'\n ).length > 3;\n if (refactoring) patterns.push('Active Refactoring');\n\n // Detect debugging pattern\n const debugging =\n traces.filter((t) => t.type === 'error' || t.type === 'debug').length > 5;\n if (debugging) patterns.push('Deep Debugging Session');\n\n return patterns;\n }\n\n private async extractApproaches(frames: Frame[]): Promise<ApproachSummary[]> {\n return frames\n .filter((f) => f.metadata?.approach)\n .map((f) => ({\n approach: f.metadata.approach,\n outcome: this.getApproachOutcome(f),\n learnings: f.metadata.learnings,\n }));\n }\n\n private getApproachOutcome(frame: Frame): ApproachSummary['outcome'] {\n if (frame.status === 'closed' && frame.metadata?.success)\n return 'successful';\n if (frame.status === 'closed' && !frame.metadata?.success) return 'failed';\n return 'partial';\n }\n\n private extractSuccessfulStrategies(approaches: ApproachSummary[]): string[] {\n return approaches\n .filter((a) => a.outcome === 'successful')\n .map((a) => a.approach);\n }\n\n private async suggestNextActions(\n tasks: TaskSummary[],\n blockers: Blocker[],\n framePath: string[]\n ): Promise<string[]> {\n const suggestions: string[] = [];\n\n // Resume in-progress tasks\n const inProgress = tasks.filter((t) => t.status === 'in_progress');\n if (inProgress.length > 0) {\n suggestions.push(`Resume task: ${inProgress[0].title}`);\n }\n\n // Address critical blockers\n const criticalBlockers = blockers.filter((b) => b.severity === 'critical');\n if (criticalBlockers.length > 0) {\n suggestions.push(\n `Resolve critical blocker: ${criticalBlockers[0].description}`\n );\n }\n\n // Complete nearly done tasks\n const nearlyDone = tasks.filter((t) => t.progress_percentage >= 80);\n if (nearlyDone.length > 0) {\n suggestions.push(\n `Complete task: ${nearlyDone[0].title} (${nearlyDone[0].progress_percentage}% done)`\n );\n }\n\n return suggestions;\n }\n\n private async generateWarnings(\n errors: ErrorContext[],\n blockers: Blocker[]\n ): Promise<string[]> {\n const warnings: string[] = [];\n\n // Unresolved errors\n const unresolved = errors.filter((e) => !e.resolved);\n if (unresolved.length > 0) {\n warnings.push(`${unresolved.length} unresolved errors`);\n }\n\n // Critical blockers\n const critical = blockers.filter((b) => b.severity === 'critical');\n if (critical.length > 0) {\n warnings.push(\n `${critical.length} critical blockers need immediate attention`\n );\n }\n\n return warnings;\n }\n\n private async saveHandoff(handoff: HandoffDocument): Promise<void> {\n await fs.mkdir(this.handoffDir, { recursive: true });\n\n const filename = `${handoff.timestamp.replace(/[:.]/g, '-')}.json`;\n const filepath = path.join(this.handoffDir, filename);\n\n await fs.writeFile(filepath, JSON.stringify(handoff, null, 2), 'utf-8');\n\n // Also save markdown summary\n const markdown = await this.generateMarkdownSummary(handoff);\n const mdPath = filepath.replace('.json', '.md');\n await fs.writeFile(mdPath, markdown, 'utf-8');\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["path"]
7
7
  }
@@ -3,6 +3,17 @@ import * as fs from "fs/promises";
3
3
  import * as path from "path";
4
4
  import { logger } from "../monitoring/logger.js";
5
5
  import { SystemError, ErrorCode } from "../errors/index.js";
6
+ function getEnv(key, defaultValue) {
7
+ const value = process.env[key];
8
+ if (value === void 0) {
9
+ if (defaultValue !== void 0) return defaultValue;
10
+ throw new Error(`Environment variable ${key} is required`);
11
+ }
12
+ return value;
13
+ }
14
+ function getOptionalEnv(key) {
15
+ return process.env[key];
16
+ }
6
17
  var FrameQueryMode = /* @__PURE__ */ ((FrameQueryMode2) => {
7
18
  FrameQueryMode2["CURRENT_SESSION"] = "current";
8
19
  FrameQueryMode2["PROJECT_ACTIVE"] = "project";
@@ -17,7 +28,7 @@ class SessionManager {
17
28
  STALE_THRESHOLD = 24 * 60 * 60 * 1e3;
18
29
  // 24 hours
19
30
  constructor() {
20
- const homeDir = process.env.HOME || process.env.USERPROFILE || "";
31
+ const homeDir = process.env["HOME"] || process.env["USERPROFILE"] || "";
21
32
  this.sessionsDir = path.join(homeDir, ".stackmemory", "sessions");
22
33
  }
23
34
  static getInstance() {
@@ -51,7 +62,7 @@ class SessionManager {
51
62
  return session;
52
63
  }
53
64
  }
54
- const envSessionId = process.env.STACKMEMORY_SESSION;
65
+ const envSessionId = process.env["STACKMEMORY_SESSION"];
55
66
  if (envSessionId) {
56
67
  const session = await this.loadSession(envSessionId);
57
68
  if (session) {
@@ -96,9 +107,9 @@ class SessionManager {
96
107
  lastActiveAt: Date.now(),
97
108
  metadata: {
98
109
  ...params.metadata,
99
- user: process.env.USER,
100
- environment: process.env.NODE_ENV || "development",
101
- cliVersion: process.env.npm_package_version
110
+ user: process.env["USER"],
111
+ environment: process.env["NODE_ENV"] || "development",
112
+ cliVersion: process.env["npm_package_version"]
102
113
  },
103
114
  state: "active"
104
115
  };
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/session/session-manager.ts"],
4
- "sourcesContent": ["/**\n * Session Management for StackMemory\n * Provides session persistence and recovery across CLI invocations\n */\n\nimport { v4 as uuidv4 } from 'uuid';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as crypto from 'crypto';\nimport { logger } from '../monitoring/logger.js';\nimport { SystemError, ErrorCode } from '../errors/index.js';\n\nexport interface Session {\n sessionId: string;\n runId: string;\n projectId: string;\n branch?: string;\n startedAt: number;\n lastActiveAt: number;\n metadata: {\n user?: string;\n environment?: string;\n tags?: string[];\n cliVersion?: string;\n };\n state: 'active' | 'suspended' | 'closed';\n}\n\nexport interface SessionOptions {\n sessionId?: string;\n projectPath?: string;\n branch?: string;\n metadata?: Session['metadata'];\n}\n\nexport enum FrameQueryMode {\n CURRENT_SESSION = 'current',\n PROJECT_ACTIVE = 'project',\n ALL_ACTIVE = 'all',\n HISTORICAL = 'historical',\n}\n\nexport class SessionManager {\n private static instance: SessionManager;\n private sessionsDir: string;\n private currentSession: Session | null = null;\n private readonly STALE_THRESHOLD = 24 * 60 * 60 * 1000; // 24 hours\n\n private constructor() {\n const homeDir = process.env.HOME || process.env.USERPROFILE || '';\n this.sessionsDir = path.join(homeDir, '.stackmemory', 'sessions');\n }\n\n static getInstance(): SessionManager {\n if (!SessionManager.instance) {\n SessionManager.instance = new SessionManager();\n }\n return SessionManager.instance;\n }\n\n async initialize(): Promise<void> {\n try {\n await fs.mkdir(this.sessionsDir, { recursive: true });\n await fs.mkdir(path.join(this.sessionsDir, 'projects'), {\n recursive: true,\n });\n await fs.mkdir(path.join(this.sessionsDir, 'history'), {\n recursive: true,\n });\n } catch (error) {\n throw new SystemError(\n 'Failed to initialize session directories',\n ErrorCode.INITIALIZATION_ERROR,\n { error, sessionsDir: this.sessionsDir }\n );\n }\n }\n\n async getOrCreateSession(options?: SessionOptions): Promise<Session> {\n // 1. Check explicit session ID\n if (options?.sessionId) {\n const session = await this.loadSession(options.sessionId);\n if (session) {\n this.currentSession = session;\n return session;\n }\n }\n\n // 2. Check environment variable\n const envSessionId = process.env.STACKMEMORY_SESSION;\n if (envSessionId) {\n const session = await this.loadSession(envSessionId);\n if (session) {\n this.currentSession = session;\n return session;\n }\n }\n\n // 3. Check project + branch context\n const projectHash = await this.getProjectHash(options?.projectPath);\n const branch =\n options?.branch || (await this.getGitBranch(options?.projectPath));\n\n if (projectHash) {\n // Try project+branch session\n const branchSession = await this.findProjectBranchSession(\n projectHash,\n branch\n );\n if (branchSession && this.isSessionRecent(branchSession)) {\n await this.touchSession(branchSession);\n this.currentSession = branchSession;\n return branchSession;\n }\n\n // Try last active for project\n const lastActive = await this.findLastActiveSession(projectHash);\n if (lastActive && this.isSessionRecent(lastActive)) {\n await this.touchSession(lastActive);\n this.currentSession = lastActive;\n return lastActive;\n }\n }\n\n // 4. Create new session\n const newSession = await this.createSession({\n projectId: projectHash || 'global',\n branch,\n metadata: options?.metadata,\n });\n\n this.currentSession = newSession;\n return newSession;\n }\n\n async createSession(params: {\n projectId: string;\n branch?: string;\n metadata?: Session['metadata'];\n }): Promise<Session> {\n const session: Session = {\n sessionId: uuidv4(),\n runId: uuidv4(),\n projectId: params.projectId,\n branch: params.branch,\n startedAt: Date.now(),\n lastActiveAt: Date.now(),\n metadata: {\n ...params.metadata,\n user: process.env.USER,\n environment: process.env.NODE_ENV || 'development',\n cliVersion: process.env.npm_package_version,\n },\n state: 'active',\n };\n\n await this.saveSession(session);\n await this.setProjectActiveSession(params.projectId, session.sessionId);\n\n // Set as current session\n this.currentSession = session;\n\n logger.info('Created new session', {\n sessionId: session.sessionId,\n projectId: session.projectId,\n branch: session.branch,\n });\n\n return session;\n }\n\n async loadSession(sessionId: string): Promise<Session | null> {\n try {\n const sessionPath = path.join(this.sessionsDir, `${sessionId}.json`);\n const data = await fs.readFile(sessionPath, 'utf-8');\n return JSON.parse(data) as Session;\n } catch (error) {\n // Check history\n try {\n const historyPath = path.join(\n this.sessionsDir,\n 'history',\n `${sessionId}.json`\n );\n const data = await fs.readFile(historyPath, 'utf-8');\n return JSON.parse(data) as Session;\n } catch {\n return null;\n }\n }\n }\n\n async saveSession(session: Session): Promise<void> {\n const sessionPath = path.join(\n this.sessionsDir,\n `${session.sessionId}.json`\n );\n await fs.writeFile(sessionPath, JSON.stringify(session, null, 2));\n }\n\n async suspendSession(sessionId?: string): Promise<void> {\n const id = sessionId || this.currentSession?.sessionId;\n if (!id) return;\n\n const session = await this.loadSession(id);\n if (session) {\n session.state = 'suspended';\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n }\n }\n\n async resumeSession(sessionId: string): Promise<Session> {\n const session = await this.loadSession(sessionId);\n if (!session) {\n throw new SystemError('Session not found', ErrorCode.NOT_FOUND, {\n sessionId,\n });\n }\n\n session.state = 'active';\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n\n this.currentSession = session;\n return session;\n }\n\n async closeSession(sessionId?: string): Promise<void> {\n const id = sessionId || this.currentSession?.sessionId;\n if (!id) return;\n\n const session = await this.loadSession(id);\n if (session) {\n session.state = 'closed';\n session.lastActiveAt = Date.now();\n\n // Move to history\n const sessionPath = path.join(\n this.sessionsDir,\n `${session.sessionId}.json`\n );\n const historyPath = path.join(\n this.sessionsDir,\n 'history',\n `${session.sessionId}.json`\n );\n\n await fs.rename(sessionPath, historyPath);\n }\n }\n\n async listSessions(filter?: {\n projectId?: string;\n state?: Session['state'];\n branch?: string;\n }): Promise<Session[]> {\n const sessions: Session[] = [];\n\n // Load active sessions\n const files = await fs.readdir(this.sessionsDir);\n for (const file of files) {\n if (file.endsWith('.json')) {\n const session = await this.loadSession(file.replace('.json', ''));\n if (session) {\n sessions.push(session);\n }\n }\n }\n\n // Apply filters\n return sessions.filter((s) => {\n if (filter?.projectId && s.projectId !== filter.projectId) return false;\n if (filter?.state && s.state !== filter.state) return false;\n if (filter?.branch && s.branch !== filter.branch) return false;\n return true;\n });\n }\n\n async mergeSessions(sourceId: string, targetId: string): Promise<Session> {\n const source = await this.loadSession(sourceId);\n const target = await this.loadSession(targetId);\n\n if (!source || !target) {\n throw new SystemError(\n 'Session not found for merge',\n ErrorCode.NOT_FOUND,\n { sourceId, targetId }\n );\n }\n\n // Merge metadata\n target.metadata = {\n ...target.metadata,\n ...source.metadata,\n tags: [...(target.metadata.tags || []), ...(source.metadata.tags || [])],\n };\n\n // Update timestamps\n target.lastActiveAt = Date.now();\n\n // Close source session\n await this.closeSession(sourceId);\n await this.saveSession(target);\n\n logger.info('Merged sessions', {\n source: sourceId,\n target: targetId,\n });\n\n return target;\n }\n\n async cleanupStaleSessions(\n maxAge: number = 30 * 24 * 60 * 60 * 1000\n ): Promise<number> {\n const historyDir = path.join(this.sessionsDir, 'history');\n const files = await fs.readdir(historyDir);\n const cutoff = Date.now() - maxAge;\n let cleaned = 0;\n\n for (const file of files) {\n if (file.endsWith('.json')) {\n const filePath = path.join(historyDir, file);\n const stats = await fs.stat(filePath);\n\n if (stats.mtimeMs < cutoff) {\n await fs.unlink(filePath);\n cleaned++;\n }\n }\n }\n\n logger.info(`Cleaned up ${cleaned} stale sessions`);\n return cleaned;\n }\n\n getCurrentSession(): Session | null {\n return this.currentSession;\n }\n\n getSessionRunId(): string {\n return this.currentSession?.runId || uuidv4();\n }\n\n private async getProjectHash(projectPath?: string): Promise<string | null> {\n try {\n const cwd = projectPath || process.cwd();\n const pathModule = await import('path');\n\n // Try to get git remote first (consistent with project-manager)\n let identifier: string;\n try {\n const { execSync } = await import('child_process');\n identifier = execSync('git config --get remote.origin.url', {\n cwd,\n encoding: 'utf-8',\n timeout: 5000,\n }).trim();\n } catch {\n // Fall back to directory path\n identifier = cwd;\n }\n\n // Use same algorithm as project-manager.generateProjectId\n const cleaned = identifier\n .replace(/\\.git$/, '')\n .replace(/[^a-zA-Z0-9-]/g, '-')\n .toLowerCase();\n\n return cleaned.substring(cleaned.length - 50);\n } catch {\n return null;\n }\n }\n\n private async getGitBranch(\n projectPath?: string\n ): Promise<string | undefined> {\n try {\n const { execSync } = await import('child_process');\n const cwd = projectPath || process.cwd();\n const branch = execSync('git rev-parse --abbrev-ref HEAD', {\n cwd,\n encoding: 'utf-8',\n }).trim();\n return branch;\n } catch {\n return undefined;\n }\n }\n\n private async findProjectBranchSession(\n projectHash: string,\n branch?: string\n ): Promise<Session | null> {\n if (!branch) return null;\n\n const sessions = await this.listSessions({\n projectId: projectHash,\n state: 'active',\n branch,\n });\n\n return sessions.sort((a, b) => b.lastActiveAt - a.lastActiveAt)[0] || null;\n }\n\n private async findLastActiveSession(\n projectHash: string\n ): Promise<Session | null> {\n const sessions = await this.listSessions({\n projectId: projectHash,\n state: 'active',\n });\n\n return sessions.sort((a, b) => b.lastActiveAt - a.lastActiveAt)[0] || null;\n }\n\n private async setProjectActiveSession(\n projectId: string,\n sessionId: string\n ): Promise<void> {\n const projectFile = path.join(\n this.sessionsDir,\n 'projects',\n `${projectId}.json`\n );\n await fs.writeFile(\n projectFile,\n JSON.stringify(\n {\n projectId,\n activeSessionId: sessionId,\n updatedAt: Date.now(),\n },\n null,\n 2\n )\n );\n }\n\n private isSessionRecent(session: Session): boolean {\n return Date.now() - session.lastActiveAt < this.STALE_THRESHOLD;\n }\n\n private async touchSession(session: Session): Promise<void> {\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n }\n}\n\nexport const sessionManager = SessionManager.getInstance();\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Session Management for StackMemory\n * Provides session persistence and recovery across CLI invocations\n */\n\nimport { v4 as uuidv4 } from 'uuid';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as crypto from 'crypto';\nimport { logger } from '../monitoring/logger.js';\nimport { SystemError, ErrorCode } from '../errors/index.js';\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new Error(`Environment variable ${key} is required`);\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\n\nexport interface Session {\n sessionId: string;\n runId: string;\n projectId: string;\n branch?: string;\n startedAt: number;\n lastActiveAt: number;\n metadata: {\n user?: string;\n environment?: string;\n tags?: string[];\n cliVersion?: string;\n };\n state: 'active' | 'suspended' | 'closed';\n}\n\nexport interface SessionOptions {\n sessionId?: string;\n projectPath?: string;\n branch?: string;\n metadata?: Session['metadata'];\n}\n\nexport enum FrameQueryMode {\n CURRENT_SESSION = 'current',\n PROJECT_ACTIVE = 'project',\n ALL_ACTIVE = 'all',\n HISTORICAL = 'historical',\n}\n\nexport class SessionManager {\n private static instance: SessionManager;\n private sessionsDir: string;\n private currentSession: Session | null = null;\n private readonly STALE_THRESHOLD = 24 * 60 * 60 * 1000; // 24 hours\n\n private constructor() {\n const homeDir = process.env['HOME'] || process.env['USERPROFILE'] || '';\n this.sessionsDir = path.join(homeDir, '.stackmemory', 'sessions');\n }\n\n static getInstance(): SessionManager {\n if (!SessionManager.instance) {\n SessionManager.instance = new SessionManager();\n }\n return SessionManager.instance;\n }\n\n async initialize(): Promise<void> {\n try {\n await fs.mkdir(this.sessionsDir, { recursive: true });\n await fs.mkdir(path.join(this.sessionsDir, 'projects'), {\n recursive: true,\n });\n await fs.mkdir(path.join(this.sessionsDir, 'history'), {\n recursive: true,\n });\n } catch (error: unknown) {\n throw new SystemError(\n 'Failed to initialize session directories',\n ErrorCode.INITIALIZATION_ERROR,\n { error, sessionsDir: this.sessionsDir }\n );\n }\n }\n\n async getOrCreateSession(options?: SessionOptions): Promise<Session> {\n // 1. Check explicit session ID\n if (options?.sessionId) {\n const session = await this.loadSession(options.sessionId);\n if (session) {\n this.currentSession = session;\n return session;\n }\n }\n\n // 2. Check environment variable\n const envSessionId = process.env['STACKMEMORY_SESSION'];\n if (envSessionId) {\n const session = await this.loadSession(envSessionId);\n if (session) {\n this.currentSession = session;\n return session;\n }\n }\n\n // 3. Check project + branch context\n const projectHash = await this.getProjectHash(options?.projectPath);\n const branch =\n options?.branch || (await this.getGitBranch(options?.projectPath));\n\n if (projectHash) {\n // Try project+branch session\n const branchSession = await this.findProjectBranchSession(\n projectHash,\n branch\n );\n if (branchSession && this.isSessionRecent(branchSession)) {\n await this.touchSession(branchSession);\n this.currentSession = branchSession;\n return branchSession;\n }\n\n // Try last active for project\n const lastActive = await this.findLastActiveSession(projectHash);\n if (lastActive && this.isSessionRecent(lastActive)) {\n await this.touchSession(lastActive);\n this.currentSession = lastActive;\n return lastActive;\n }\n }\n\n // 4. Create new session\n const newSession = await this.createSession({\n projectId: projectHash || 'global',\n branch,\n metadata: options?.metadata,\n });\n\n this.currentSession = newSession;\n return newSession;\n }\n\n async createSession(params: {\n projectId: string;\n branch?: string;\n metadata?: Session['metadata'];\n }): Promise<Session> {\n const session: Session = {\n sessionId: uuidv4(),\n runId: uuidv4(),\n projectId: params.projectId,\n branch: params.branch,\n startedAt: Date.now(),\n lastActiveAt: Date.now(),\n metadata: {\n ...params.metadata,\n user: process.env['USER'],\n environment: process.env['NODE_ENV'] || 'development',\n cliVersion: process.env['npm_package_version'],\n },\n state: 'active',\n };\n\n await this.saveSession(session);\n await this.setProjectActiveSession(params.projectId, session.sessionId);\n\n // Set as current session\n this.currentSession = session;\n\n logger.info('Created new session', {\n sessionId: session.sessionId,\n projectId: session.projectId,\n branch: session.branch,\n });\n\n return session;\n }\n\n async loadSession(sessionId: string): Promise<Session | null> {\n try {\n const sessionPath = path.join(this.sessionsDir, `${sessionId}.json`);\n const data = await fs.readFile(sessionPath, 'utf-8');\n return JSON.parse(data) as Session;\n } catch (error: unknown) {\n // Check history\n try {\n const historyPath = path.join(\n this.sessionsDir,\n 'history',\n `${sessionId}.json`\n );\n const data = await fs.readFile(historyPath, 'utf-8');\n return JSON.parse(data) as Session;\n } catch {\n return null;\n }\n }\n }\n\n async saveSession(session: Session): Promise<void> {\n const sessionPath = path.join(\n this.sessionsDir,\n `${session.sessionId}.json`\n );\n await fs.writeFile(sessionPath, JSON.stringify(session, null, 2));\n }\n\n async suspendSession(sessionId?: string): Promise<void> {\n const id = sessionId || this.currentSession?.sessionId;\n if (!id) return;\n\n const session = await this.loadSession(id);\n if (session) {\n session.state = 'suspended';\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n }\n }\n\n async resumeSession(sessionId: string): Promise<Session> {\n const session = await this.loadSession(sessionId);\n if (!session) {\n throw new SystemError('Session not found', ErrorCode.NOT_FOUND, {\n sessionId,\n });\n }\n\n session.state = 'active';\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n\n this.currentSession = session;\n return session;\n }\n\n async closeSession(sessionId?: string): Promise<void> {\n const id = sessionId || this.currentSession?.sessionId;\n if (!id) return;\n\n const session = await this.loadSession(id);\n if (session) {\n session.state = 'closed';\n session.lastActiveAt = Date.now();\n\n // Move to history\n const sessionPath = path.join(\n this.sessionsDir,\n `${session.sessionId}.json`\n );\n const historyPath = path.join(\n this.sessionsDir,\n 'history',\n `${session.sessionId}.json`\n );\n\n await fs.rename(sessionPath, historyPath);\n }\n }\n\n async listSessions(filter?: {\n projectId?: string;\n state?: Session['state'];\n branch?: string;\n }): Promise<Session[]> {\n const sessions: Session[] = [];\n\n // Load active sessions\n const files = await fs.readdir(this.sessionsDir);\n for (const file of files) {\n if (file.endsWith('.json')) {\n const session = await this.loadSession(file.replace('.json', ''));\n if (session) {\n sessions.push(session);\n }\n }\n }\n\n // Apply filters\n return sessions.filter((s) => {\n if (filter?.projectId && s.projectId !== filter.projectId) return false;\n if (filter?.state && s.state !== filter.state) return false;\n if (filter?.branch && s.branch !== filter.branch) return false;\n return true;\n });\n }\n\n async mergeSessions(sourceId: string, targetId: string): Promise<Session> {\n const source = await this.loadSession(sourceId);\n const target = await this.loadSession(targetId);\n\n if (!source || !target) {\n throw new SystemError(\n 'Session not found for merge',\n ErrorCode.NOT_FOUND,\n { sourceId, targetId }\n );\n }\n\n // Merge metadata\n target.metadata = {\n ...target.metadata,\n ...source.metadata,\n tags: [...(target.metadata.tags || []), ...(source.metadata.tags || [])],\n };\n\n // Update timestamps\n target.lastActiveAt = Date.now();\n\n // Close source session\n await this.closeSession(sourceId);\n await this.saveSession(target);\n\n logger.info('Merged sessions', {\n source: sourceId,\n target: targetId,\n });\n\n return target;\n }\n\n async cleanupStaleSessions(\n maxAge: number = 30 * 24 * 60 * 60 * 1000\n ): Promise<number> {\n const historyDir = path.join(this.sessionsDir, 'history');\n const files = await fs.readdir(historyDir);\n const cutoff = Date.now() - maxAge;\n let cleaned = 0;\n\n for (const file of files) {\n if (file.endsWith('.json')) {\n const filePath = path.join(historyDir, file);\n const stats = await fs.stat(filePath);\n\n if (stats.mtimeMs < cutoff) {\n await fs.unlink(filePath);\n cleaned++;\n }\n }\n }\n\n logger.info(`Cleaned up ${cleaned} stale sessions`);\n return cleaned;\n }\n\n getCurrentSession(): Session | null {\n return this.currentSession;\n }\n\n getSessionRunId(): string {\n return this.currentSession?.runId || uuidv4();\n }\n\n private async getProjectHash(projectPath?: string): Promise<string | null> {\n try {\n const cwd = projectPath || process.cwd();\n const pathModule = await import('path');\n\n // Try to get git remote first (consistent with project-manager)\n let identifier: string;\n try {\n const { execSync } = await import('child_process');\n identifier = execSync('git config --get remote.origin.url', {\n cwd,\n encoding: 'utf-8',\n timeout: 5000,\n }).trim();\n } catch {\n // Fall back to directory path\n identifier = cwd;\n }\n\n // Use same algorithm as project-manager.generateProjectId\n const cleaned = identifier\n .replace(/\\.git$/, '')\n .replace(/[^a-zA-Z0-9-]/g, '-')\n .toLowerCase();\n\n return cleaned.substring(cleaned.length - 50);\n } catch {\n return null;\n }\n }\n\n private async getGitBranch(\n projectPath?: string\n ): Promise<string | undefined> {\n try {\n const { execSync } = await import('child_process');\n const cwd = projectPath || process.cwd();\n const branch = execSync('git rev-parse --abbrev-ref HEAD', {\n cwd,\n encoding: 'utf-8',\n }).trim();\n return branch;\n } catch {\n return undefined;\n }\n }\n\n private async findProjectBranchSession(\n projectHash: string,\n branch?: string\n ): Promise<Session | null> {\n if (!branch) return null;\n\n const sessions = await this.listSessions({\n projectId: projectHash,\n state: 'active',\n branch,\n });\n\n return sessions.sort((a, b) => b.lastActiveAt - a.lastActiveAt)[0] || null;\n }\n\n private async findLastActiveSession(\n projectHash: string\n ): Promise<Session | null> {\n const sessions = await this.listSessions({\n projectId: projectHash,\n state: 'active',\n });\n\n return sessions.sort((a, b) => b.lastActiveAt - a.lastActiveAt)[0] || null;\n }\n\n private async setProjectActiveSession(\n projectId: string,\n sessionId: string\n ): Promise<void> {\n const projectFile = path.join(\n this.sessionsDir,\n 'projects',\n `${projectId}.json`\n );\n await fs.writeFile(\n projectFile,\n JSON.stringify(\n {\n projectId,\n activeSessionId: sessionId,\n updatedAt: Date.now(),\n },\n null,\n 2\n )\n );\n }\n\n private isSessionRecent(session: Session): boolean {\n return Date.now() - session.lastActiveAt < this.STALE_THRESHOLD;\n }\n\n private async touchSession(session: Session): Promise<void> {\n session.lastActiveAt = Date.now();\n await this.saveSession(session);\n }\n}\n\nexport const sessionManager = SessionManager.getInstance();\n"],
5
+ "mappings": "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",
6
6
  "names": ["FrameQueryMode"]
7
7
  }