@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/integrations/claude-code/enhanced-pre-clear-hooks.ts"],
4
- "sourcesContent": ["/**\n * Enhanced Pre-Clear Context Preservation System\n * Comprehensive session state capture before /clear or /compact operations\n */\n\nimport { EventEmitter } from 'events';\nimport { ClearSurvival } from '../../core/session/clear-survival';\nimport { HandoffGenerator } from '../../core/session/handoff-generator';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { execSync } from 'child_process';\n\nexport interface PreClearContext {\n // Session metadata\n sessionId: string;\n timestamp: string;\n trigger:\n | 'manual_clear'\n | 'auto_compact'\n | 'context_overflow'\n | 'user_request';\n\n // Context analysis\n contextUsage: {\n estimatedTokens: number;\n maxTokens: number;\n percentage: number;\n components: {\n frames: number;\n traces: number;\n conversations: number;\n codeBlocks: number;\n };\n };\n\n // Active work state\n workingState: {\n currentTask: string;\n activeFiles: string[];\n recentCommands: string[];\n pendingActions: string[];\n blockers: string[];\n };\n\n // Conversation state\n conversationState: {\n lastUserMessage: string;\n lastAssistantMessage: string;\n conversationTopic: string;\n messageCount: number;\n recentContext: string[];\n };\n\n // Code context\n codeContext: {\n modifiedFiles: FileContext[];\n gitStatus: GitStatus;\n testResults?: TestResults;\n buildStatus?: BuildStatus;\n dependencies: DependencyInfo[];\n };\n\n // Cognitive state\n cognitiveState: {\n currentFocus: string;\n mentalModel: string[];\n assumptions: string[];\n hypotheses: string[];\n explorationPaths: string[];\n };\n\n // Environment snapshot\n environment: {\n workingDirectory: string;\n gitBranch: string;\n nodeVersion?: string;\n packageJson?: any;\n environmentVars: Record<string, string>;\n };\n}\n\ninterface FileContext {\n path: string;\n lastModified: string;\n changeType: 'created' | 'modified' | 'deleted';\n lineChanges: { added: number; removed: number };\n purpose: string;\n relatedFiles: string[];\n}\n\ninterface GitStatus {\n branch: string;\n ahead: number;\n behind: number;\n staged: string[];\n unstaged: string[];\n untracked: string[];\n lastCommit: {\n hash: string;\n message: string;\n timestamp: string;\n };\n}\n\ninterface TestResults {\n framework: string;\n passed: number;\n failed: number;\n skipped: number;\n coverage?: number;\n failures: string[];\n}\n\ninterface BuildStatus {\n success: boolean;\n errors: string[];\n warnings: string[];\n timestamp: string;\n}\n\ninterface DependencyInfo {\n name: string;\n version: string;\n type: 'dependency' | 'devDependency' | 'global';\n critical: boolean;\n}\n\nexport class EnhancedPreClearHooks extends EventEmitter {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private clearSurvival: ClearSurvival;\n private handoffGenerator: HandoffGenerator;\n private projectRoot: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n clearSurvival: ClearSurvival,\n handoffGenerator: HandoffGenerator,\n projectRoot: string\n ) {\n super();\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.clearSurvival = clearSurvival;\n this.handoffGenerator = handoffGenerator;\n this.projectRoot = projectRoot;\n }\n\n /**\n * Comprehensive pre-clear context capture\n */\n async capturePreClearContext(\n trigger: PreClearContext['trigger']\n ): Promise<PreClearContext> {\n console.log('\uD83D\uDD0D Capturing comprehensive session context...');\n\n const context: PreClearContext = {\n sessionId: await this.dbManager.getCurrentSessionId(),\n timestamp: new Date().toISOString(),\n trigger,\n\n contextUsage: await this.analyzeContextUsage(),\n workingState: await this.captureWorkingState(),\n conversationState: await this.captureConversationState(),\n codeContext: await this.captureCodeContext(),\n cognitiveState: await this.captureCognitiveState(),\n environment: await this.captureEnvironment(),\n };\n\n // Save comprehensive context\n await this.saveEnhancedContext(context);\n\n // Emit for other systems to react\n this.emit('context:captured', context);\n\n console.log('\u2705 Comprehensive context captured');\n return context;\n }\n\n /**\n * Analyze current context usage with detailed breakdown\n */\n private async analyzeContextUsage(): Promise<\n PreClearContext['contextUsage']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const frames = await this.dbManager.getRecentFrames(sessionId, 1000);\n const traces = await this.dbManager.getRecentTraces(sessionId, 1000);\n\n // Estimate token usage by component\n const frameTokens = frames.length * 200; // Average tokens per frame\n const traceTokens = traces.length * 100; // Average tokens per trace\n const conversationTokens = await this.estimateConversationTokens();\n const codeBlockTokens = await this.estimateCodeBlockTokens();\n\n const estimatedTokens =\n frameTokens + traceTokens + conversationTokens + codeBlockTokens;\n const maxTokens = 100000; // Claude's limit\n\n return {\n estimatedTokens,\n maxTokens,\n percentage: estimatedTokens / maxTokens,\n components: {\n frames: frameTokens,\n traces: traceTokens,\n conversations: conversationTokens,\n codeBlocks: codeBlockTokens,\n },\n };\n }\n\n /**\n * Capture current working state\n */\n private async captureWorkingState(): Promise<\n PreClearContext['workingState']\n > {\n const activeFrame = await this.getCurrentActiveFrame();\n const recentTraces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n // Extract active files from recent operations\n const activeFiles = this.extractActiveFiles(recentTraces);\n\n // Get recent commands\n const recentCommands = recentTraces\n .filter((t) => t.type === 'bash' || t.type === 'command')\n .map((t) => t.content.command)\n .slice(0, 10);\n\n // Identify pending actions from frame metadata\n const pendingActions = this.extractPendingActions(activeFrame);\n\n // Identify blockers\n const blockers = this.extractBlockers(recentTraces);\n\n return {\n currentTask: activeFrame?.description || 'No active task',\n activeFiles,\n recentCommands,\n pendingActions,\n blockers,\n };\n }\n\n /**\n * Capture conversation state and recent context\n */\n private async captureConversationState(): Promise<\n PreClearContext['conversationState']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Find last user and assistant messages\n const userMessages = recentTraces\n .filter((t) => t.type === 'user_message' || t.type === 'input')\n .slice(0, 5);\n\n const assistantMessages = recentTraces\n .filter((t) => t.type === 'assistant_message' || t.type === 'response')\n .slice(0, 5);\n\n // Extract conversation topic from recent interactions\n const conversationTopic = this.inferConversationTopic(recentTraces);\n\n // Build recent context summary\n const recentContext = this.buildRecentContextSummary(recentTraces);\n\n return {\n lastUserMessage:\n userMessages[0]?.content.message || 'No recent user message',\n lastAssistantMessage:\n assistantMessages[0]?.content.message || 'No recent assistant message',\n conversationTopic,\n messageCount: userMessages.length + assistantMessages.length,\n recentContext,\n };\n }\n\n /**\n * Capture comprehensive code context\n */\n private async captureCodeContext(): Promise<PreClearContext['codeContext']> {\n const gitStatus = await this.captureGitStatus();\n const modifiedFiles = await this.captureModifiedFiles();\n const testResults = await this.captureTestResults();\n const buildStatus = await this.captureBuildStatus();\n const dependencies = await this.captureDependencies();\n\n return {\n modifiedFiles,\n gitStatus,\n testResults,\n buildStatus,\n dependencies,\n };\n }\n\n /**\n * Capture cognitive state and mental model\n */\n private async captureCognitiveState(): Promise<\n PreClearContext['cognitiveState']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Extract cognitive elements from traces and frames\n const currentFocus = await this.extractCurrentFocus();\n const mentalModel = this.extractMentalModel(recentTraces);\n const assumptions = this.extractAssumptions(recentTraces);\n const hypotheses = this.extractHypotheses(recentTraces);\n const explorationPaths = this.extractExplorationPaths(recentTraces);\n\n return {\n currentFocus,\n mentalModel,\n assumptions,\n hypotheses,\n explorationPaths,\n };\n }\n\n /**\n * Capture environment snapshot\n */\n private async captureEnvironment(): Promise<PreClearContext['environment']> {\n const gitBranch = await this.getCurrentGitBranch();\n const packageJson = await this.getPackageJson();\n const environmentVars = this.getRelevantEnvVars();\n\n return {\n workingDirectory: this.projectRoot,\n gitBranch,\n nodeVersion: process.version,\n packageJson,\n environmentVars,\n };\n }\n\n /**\n * Save enhanced context to multiple locations for reliability\n */\n private async saveEnhancedContext(context: PreClearContext): Promise<void> {\n const timestamp = context.timestamp.replace(/[:.]/g, '-');\n\n // Save to primary location\n const primaryPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n `context-${timestamp}.json`\n );\n\n // Save to backup location\n const backupPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n 'latest-context.json'\n );\n\n // Create markdown summary\n const markdownPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n `context-${timestamp}.md`\n );\n\n await fs.mkdir(path.dirname(primaryPath), { recursive: true });\n\n // Save JSON\n await fs.writeFile(primaryPath, JSON.stringify(context, null, 2), 'utf-8');\n await fs.writeFile(backupPath, JSON.stringify(context, null, 2), 'utf-8');\n\n // Save markdown summary\n const markdown = this.generateMarkdownSummary(context);\n await fs.writeFile(markdownPath, markdown, 'utf-8');\n\n console.log(\n `\uD83D\uDCC1 Context saved to ${path.relative(this.projectRoot, primaryPath)}`\n );\n }\n\n /**\n * Generate human-readable markdown summary\n */\n private generateMarkdownSummary(context: PreClearContext): string {\n const lines = [\n `# Pre-Clear Context Snapshot`,\n `**Timestamp**: ${new Date(context.timestamp).toLocaleString()}`,\n `**Trigger**: ${context.trigger}`,\n `**Session ID**: ${context.sessionId}`,\n '',\n\n `## \uD83D\uDCCA Context Usage`,\n `- **Total Tokens**: ${context.contextUsage.estimatedTokens.toLocaleString()} / ${context.contextUsage.maxTokens.toLocaleString()} (${Math.round(context.contextUsage.percentage * 100)}%)`,\n `- **Frames**: ${context.contextUsage.components.frames} tokens`,\n `- **Traces**: ${context.contextUsage.components.traces} tokens`,\n `- **Conversations**: ${context.contextUsage.components.conversations} tokens`,\n `- **Code Blocks**: ${context.contextUsage.components.codeBlocks} tokens`,\n '',\n\n `## \uD83C\uDFAF Current Work State`,\n `**Task**: ${context.workingState.currentTask}`,\n `**Active Files** (${context.workingState.activeFiles.length}):`,\n ...context.workingState.activeFiles.slice(0, 10).map((f) => `- ${f}`),\n '',\n `**Recent Commands**:`,\n ...context.workingState.recentCommands\n .slice(0, 5)\n .map((c) => `- \\`${c}\\``),\n '',\n\n `## \uD83D\uDCAC Conversation State`,\n `**Topic**: ${context.conversationState.conversationTopic}`,\n `**Messages**: ${context.conversationState.messageCount}`,\n `**Last User**: ${context.conversationState.lastUserMessage.substring(0, 100)}...`,\n '',\n\n `## \uD83D\uDCDD Code Context`,\n `**Git Branch**: ${context.codeContext.gitStatus.branch}`,\n `**Modified Files**: ${context.codeContext.modifiedFiles.length}`,\n `**Staged**: ${context.codeContext.gitStatus.staged.length}`,\n `**Unstaged**: ${context.codeContext.gitStatus.unstaged.length}`,\n '',\n\n `## \uD83E\uDDE0 Cognitive State`,\n `**Current Focus**: ${context.cognitiveState.currentFocus}`,\n `**Mental Model**:`,\n ...context.cognitiveState.mentalModel.slice(0, 5).map((m) => `- ${m}`),\n '',\n\n `## \uD83C\uDF0D Environment`,\n `**Directory**: ${context.environment.workingDirectory}`,\n `**Node Version**: ${context.environment.nodeVersion}`,\n `**Git Branch**: ${context.environment.gitBranch}`,\n '',\n ];\n\n return lines.filter((l) => l !== undefined).join('\\n');\n }\n\n // Helper methods (simplified implementations)\n\n private async estimateConversationTokens(): Promise<number> {\n // Simplified - would analyze recent conversation history\n return 15000;\n }\n\n private async estimateCodeBlockTokens(): Promise<number> {\n // Simplified - would analyze code blocks in context\n return 8000;\n }\n\n private async getCurrentActiveFrame(): Promise<any> {\n const stack = await this.frameManager.getStack();\n return stack.frames.find((f) => f.status === 'open');\n }\n\n private extractActiveFiles(traces: any[]): string[] {\n const files = new Set<string>();\n traces.forEach((trace) => {\n if (trace.content?.file_path) files.add(trace.content.file_path);\n if (trace.content?.path) files.add(trace.content.path);\n });\n return Array.from(files).slice(0, 20);\n }\n\n private extractPendingActions(frame: any): string[] {\n if (!frame?.metadata?.pendingActions) return [];\n return frame.metadata.pendingActions;\n }\n\n private extractBlockers(traces: any[]): string[] {\n return traces\n .filter((t) => t.type === 'error' && !t.metadata?.resolved)\n .map((t) => t.content.error || 'Unknown error')\n .slice(0, 5);\n }\n\n private inferConversationTopic(traces: any[]): string {\n // Simplified - would use NLP to infer topic\n return 'Code implementation and debugging';\n }\n\n private buildRecentContextSummary(traces: any[]): string[] {\n return traces\n .slice(0, 10)\n .map(\n (t) =>\n `${t.type}: ${t.content.summary || t.content.description || 'No description'}`\n )\n .filter((s) => s.length > 10);\n }\n\n private async captureGitStatus(): Promise<GitStatus> {\n try {\n const branch = execSync('git branch --show-current', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n }).trim();\n const staged = execSync('git diff --cached --name-only', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n const unstaged = execSync('git diff --name-only', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n const untracked = execSync('git ls-files --others --exclude-standard', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n\n return {\n branch,\n ahead: 0, // Would implement git status parsing\n behind: 0,\n staged,\n unstaged,\n untracked,\n lastCommit: {\n hash: 'abc123', // Would get from git log\n message: 'Recent commit',\n timestamp: new Date().toISOString(),\n },\n };\n } catch (error) {\n return {\n branch: 'unknown',\n ahead: 0,\n behind: 0,\n staged: [],\n unstaged: [],\n untracked: [],\n lastCommit: { hash: '', message: '', timestamp: '' },\n };\n }\n }\n\n private async captureModifiedFiles(): Promise<FileContext[]> {\n try {\n const output = execSync('git diff --name-status', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n });\n return output\n .trim()\n .split('\\n')\n .filter(Boolean)\n .map((line) => {\n const [status, path] = line.split('\\t');\n return {\n path,\n lastModified: new Date().toISOString(),\n changeType:\n status === 'A'\n ? 'created'\n : status === 'D'\n ? 'deleted'\n : 'modified',\n lineChanges: { added: 0, removed: 0 }, // Would get from git diff --stat\n purpose: 'Code changes',\n relatedFiles: [],\n };\n });\n } catch (error) {\n return [];\n }\n }\n\n private async captureTestResults(): Promise<TestResults | undefined> {\n // Would implement test result parsing\n return undefined;\n }\n\n private async captureBuildStatus(): Promise<BuildStatus | undefined> {\n // Would implement build status checking\n return undefined;\n }\n\n private async captureDependencies(): Promise<DependencyInfo[]> {\n try {\n const packageJsonPath = path.join(this.projectRoot, 'package.json');\n const content = await fs.readFile(packageJsonPath, 'utf-8');\n const packageJson = JSON.parse(content);\n\n const deps: DependencyInfo[] = [];\n\n Object.entries(packageJson.dependencies || {}).forEach(\n ([name, version]) => {\n deps.push({\n name,\n version: version as string,\n type: 'dependency',\n critical: ['react', 'express', 'next'].includes(name),\n });\n }\n );\n\n return deps;\n } catch (error) {\n return [];\n }\n }\n\n private async extractCurrentFocus(): Promise<string> {\n const activeFrame = await this.getCurrentActiveFrame();\n return activeFrame?.description || 'No current focus';\n }\n\n private extractMentalModel(traces: any[]): string[] {\n // Would extract mental model concepts from traces\n return [\n 'Component architecture',\n 'Data flow patterns',\n 'Error handling strategy',\n ];\n }\n\n private extractAssumptions(traces: any[]): string[] {\n // Would extract assumptions from traces\n return [\n 'User input is validated',\n 'Database is available',\n 'Network is stable',\n ];\n }\n\n private extractHypotheses(traces: any[]): string[] {\n // Would extract hypotheses being tested\n return [\n 'Bug is in validation logic',\n 'Performance issue is database-related',\n ];\n }\n\n private extractExplorationPaths(traces: any[]): string[] {\n // Would extract different approaches being explored\n return [\n 'Try different algorithm',\n 'Refactor data structure',\n 'Add caching layer',\n ];\n }\n\n private async getCurrentGitBranch(): Promise<string> {\n try {\n return execSync('git branch --show-current', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n }).trim();\n } catch (error) {\n return 'unknown';\n }\n }\n\n private async getPackageJson(): Promise<any> {\n try {\n const content = await fs.readFile(\n path.join(this.projectRoot, 'package.json'),\n 'utf-8'\n );\n return JSON.parse(content);\n } catch (error) {\n return null;\n }\n }\n\n private getRelevantEnvVars(): Record<string, string> {\n const relevantVars = ['NODE_ENV', 'DEBUG', 'PORT', 'DATABASE_URL'];\n const result: Record<string, string> = {};\n\n relevantVars.forEach((varName) => {\n if (process.env[varName]) {\n result[varName] = process.env[varName]!;\n }\n });\n\n return result;\n }\n\n /**\n * Restore context after /clear\n */\n async restoreFromEnhancedContext(): Promise<boolean> {\n const latestPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n 'latest-context.json'\n );\n\n try {\n const content = await fs.readFile(latestPath, 'utf-8');\n const context: PreClearContext = JSON.parse(content);\n\n console.log('\uD83D\uDCDA Restoring enhanced context...');\n console.log(` Session: ${context.sessionId}`);\n console.log(` Task: ${context.workingState.currentTask}`);\n console.log(` Files: ${context.workingState.activeFiles.length}`);\n console.log(` Focus: ${context.cognitiveState.currentFocus}`);\n\n // Restore using existing systems\n await this.clearSurvival.restoreFromLedger();\n\n // Additional restoration logic would go here\n\n return true;\n } catch (error) {\n console.error('Failed to restore enhanced context:', error);\n return false;\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Enhanced Pre-Clear Context Preservation System\n * Comprehensive session state capture before /clear or /compact operations\n */\n\nimport { EventEmitter } from 'events';\nimport { ClearSurvival } from '../../core/session/clear-survival';\nimport { HandoffGenerator } from '../../core/session/handoff-generator';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { execSync } from 'child_process';\n\nexport interface PreClearContext {\n // Session metadata\n sessionId: string;\n timestamp: string;\n trigger:\n | 'manual_clear'\n | 'auto_compact'\n | 'context_overflow'\n | 'user_request';\n\n // Context analysis\n contextUsage: {\n estimatedTokens: number;\n maxTokens: number;\n percentage: number;\n components: {\n frames: number;\n traces: number;\n conversations: number;\n codeBlocks: number;\n };\n };\n\n // Active work state\n workingState: {\n currentTask: string;\n activeFiles: string[];\n recentCommands: string[];\n pendingActions: string[];\n blockers: string[];\n };\n\n // Conversation state\n conversationState: {\n lastUserMessage: string;\n lastAssistantMessage: string;\n conversationTopic: string;\n messageCount: number;\n recentContext: string[];\n };\n\n // Code context\n codeContext: {\n modifiedFiles: FileContext[];\n gitStatus: GitStatus;\n testResults?: TestResults;\n buildStatus?: BuildStatus;\n dependencies: DependencyInfo[];\n };\n\n // Cognitive state\n cognitiveState: {\n currentFocus: string;\n mentalModel: string[];\n assumptions: string[];\n hypotheses: string[];\n explorationPaths: string[];\n };\n\n // Environment snapshot\n environment: {\n workingDirectory: string;\n gitBranch: string;\n nodeVersion?: string;\n packageJson?: any;\n environmentVars: Record<string, string>;\n };\n}\n\ninterface FileContext {\n path: string;\n lastModified: string;\n changeType: 'created' | 'modified' | 'deleted';\n lineChanges: { added: number; removed: number };\n purpose: string;\n relatedFiles: string[];\n}\n\ninterface GitStatus {\n branch: string;\n ahead: number;\n behind: number;\n staged: string[];\n unstaged: string[];\n untracked: string[];\n lastCommit: {\n hash: string;\n message: string;\n timestamp: string;\n };\n}\n\ninterface TestResults {\n framework: string;\n passed: number;\n failed: number;\n skipped: number;\n coverage?: number;\n failures: string[];\n}\n\ninterface BuildStatus {\n success: boolean;\n errors: string[];\n warnings: string[];\n timestamp: string;\n}\n\ninterface DependencyInfo {\n name: string;\n version: string;\n type: 'dependency' | 'devDependency' | 'global';\n critical: boolean;\n}\n\nexport class EnhancedPreClearHooks extends EventEmitter {\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private clearSurvival: ClearSurvival;\n private handoffGenerator: HandoffGenerator;\n private projectRoot: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n clearSurvival: ClearSurvival,\n handoffGenerator: HandoffGenerator,\n projectRoot: string\n ) {\n super();\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n this.clearSurvival = clearSurvival;\n this.handoffGenerator = handoffGenerator;\n this.projectRoot = projectRoot;\n }\n\n /**\n * Comprehensive pre-clear context capture\n */\n async capturePreClearContext(\n trigger: PreClearContext['trigger']\n ): Promise<PreClearContext> {\n console.log('\uD83D\uDD0D Capturing comprehensive session context...');\n\n const context: PreClearContext = {\n sessionId: await this.dbManager.getCurrentSessionId(),\n timestamp: new Date().toISOString(),\n trigger,\n\n contextUsage: await this.analyzeContextUsage(),\n workingState: await this.captureWorkingState(),\n conversationState: await this.captureConversationState(),\n codeContext: await this.captureCodeContext(),\n cognitiveState: await this.captureCognitiveState(),\n environment: await this.captureEnvironment(),\n };\n\n // Save comprehensive context\n await this.saveEnhancedContext(context);\n\n // Emit for other systems to react\n this.emit('context:captured', context);\n\n console.log('\u2705 Comprehensive context captured');\n return context;\n }\n\n /**\n * Analyze current context usage with detailed breakdown\n */\n private async analyzeContextUsage(): Promise<\n PreClearContext['contextUsage']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const frames = await this.dbManager.getRecentFrames(sessionId, 1000);\n const traces = await this.dbManager.getRecentTraces(sessionId, 1000);\n\n // Estimate token usage by component\n const frameTokens = frames.length * 200; // Average tokens per frame\n const traceTokens = traces.length * 100; // Average tokens per trace\n const conversationTokens = await this.estimateConversationTokens();\n const codeBlockTokens = await this.estimateCodeBlockTokens();\n\n const estimatedTokens =\n frameTokens + traceTokens + conversationTokens + codeBlockTokens;\n const maxTokens = 100000; // Claude's limit\n\n return {\n estimatedTokens,\n maxTokens,\n percentage: estimatedTokens / maxTokens,\n components: {\n frames: frameTokens,\n traces: traceTokens,\n conversations: conversationTokens,\n codeBlocks: codeBlockTokens,\n },\n };\n }\n\n /**\n * Capture current working state\n */\n private async captureWorkingState(): Promise<\n PreClearContext['workingState']\n > {\n const activeFrame = await this.getCurrentActiveFrame();\n const recentTraces = await this.dbManager.getRecentTraces(\n await this.dbManager.getCurrentSessionId(),\n 50\n );\n\n // Extract active files from recent operations\n const activeFiles = this.extractActiveFiles(recentTraces);\n\n // Get recent commands\n const recentCommands = recentTraces\n .filter((t) => t.type === 'bash' || t.type === 'command')\n .map((t) => t.content.command)\n .slice(0, 10);\n\n // Identify pending actions from frame metadata\n const pendingActions = this.extractPendingActions(activeFrame);\n\n // Identify blockers\n const blockers = this.extractBlockers(recentTraces);\n\n return {\n currentTask: activeFrame?.description || 'No active task',\n activeFiles,\n recentCommands,\n pendingActions,\n blockers,\n };\n }\n\n /**\n * Capture conversation state and recent context\n */\n private async captureConversationState(): Promise<\n PreClearContext['conversationState']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Find last user and assistant messages\n const userMessages = recentTraces\n .filter((t) => t.type === 'user_message' || t.type === 'input')\n .slice(0, 5);\n\n const assistantMessages = recentTraces\n .filter((t) => t.type === 'assistant_message' || t.type === 'response')\n .slice(0, 5);\n\n // Extract conversation topic from recent interactions\n const conversationTopic = this.inferConversationTopic(recentTraces);\n\n // Build recent context summary\n const recentContext = this.buildRecentContextSummary(recentTraces);\n\n return {\n lastUserMessage:\n userMessages[0]?.content.message || 'No recent user message',\n lastAssistantMessage:\n assistantMessages[0]?.content.message || 'No recent assistant message',\n conversationTopic,\n messageCount: userMessages.length + assistantMessages.length,\n recentContext,\n };\n }\n\n /**\n * Capture comprehensive code context\n */\n private async captureCodeContext(): Promise<PreClearContext['codeContext']> {\n const gitStatus = await this.captureGitStatus();\n const modifiedFiles = await this.captureModifiedFiles();\n const testResults = await this.captureTestResults();\n const buildStatus = await this.captureBuildStatus();\n const dependencies = await this.captureDependencies();\n\n return {\n modifiedFiles,\n gitStatus,\n testResults,\n buildStatus,\n dependencies,\n };\n }\n\n /**\n * Capture cognitive state and mental model\n */\n private async captureCognitiveState(): Promise<\n PreClearContext['cognitiveState']\n > {\n const sessionId = await this.dbManager.getCurrentSessionId();\n const recentTraces = await this.dbManager.getRecentTraces(sessionId, 100);\n\n // Extract cognitive elements from traces and frames\n const currentFocus = await this.extractCurrentFocus();\n const mentalModel = this.extractMentalModel(recentTraces);\n const assumptions = this.extractAssumptions(recentTraces);\n const hypotheses = this.extractHypotheses(recentTraces);\n const explorationPaths = this.extractExplorationPaths(recentTraces);\n\n return {\n currentFocus,\n mentalModel,\n assumptions,\n hypotheses,\n explorationPaths,\n };\n }\n\n /**\n * Capture environment snapshot\n */\n private async captureEnvironment(): Promise<PreClearContext['environment']> {\n const gitBranch = await this.getCurrentGitBranch();\n const packageJson = await this.getPackageJson();\n const environmentVars = this.getRelevantEnvVars();\n\n return {\n workingDirectory: this.projectRoot,\n gitBranch,\n nodeVersion: process.version,\n packageJson,\n environmentVars,\n };\n }\n\n /**\n * Save enhanced context to multiple locations for reliability\n */\n private async saveEnhancedContext(context: PreClearContext): Promise<void> {\n const timestamp = context.timestamp.replace(/[:.]/g, '-');\n\n // Save to primary location\n const primaryPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n `context-${timestamp}.json`\n );\n\n // Save to backup location\n const backupPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n 'latest-context.json'\n );\n\n // Create markdown summary\n const markdownPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n `context-${timestamp}.md`\n );\n\n await fs.mkdir(path.dirname(primaryPath), { recursive: true });\n\n // Save JSON\n await fs.writeFile(primaryPath, JSON.stringify(context, null, 2), 'utf-8');\n await fs.writeFile(backupPath, JSON.stringify(context, null, 2), 'utf-8');\n\n // Save markdown summary\n const markdown = this.generateMarkdownSummary(context);\n await fs.writeFile(markdownPath, markdown, 'utf-8');\n\n console.log(\n `\uD83D\uDCC1 Context saved to ${path.relative(this.projectRoot, primaryPath)}`\n );\n }\n\n /**\n * Generate human-readable markdown summary\n */\n private generateMarkdownSummary(context: PreClearContext): string {\n const lines = [\n `# Pre-Clear Context Snapshot`,\n `**Timestamp**: ${new Date(context.timestamp).toLocaleString()}`,\n `**Trigger**: ${context.trigger}`,\n `**Session ID**: ${context.sessionId}`,\n '',\n\n `## \uD83D\uDCCA Context Usage`,\n `- **Total Tokens**: ${context.contextUsage.estimatedTokens.toLocaleString()} / ${context.contextUsage.maxTokens.toLocaleString()} (${Math.round(context.contextUsage.percentage * 100)}%)`,\n `- **Frames**: ${context.contextUsage.components.frames} tokens`,\n `- **Traces**: ${context.contextUsage.components.traces} tokens`,\n `- **Conversations**: ${context.contextUsage.components.conversations} tokens`,\n `- **Code Blocks**: ${context.contextUsage.components.codeBlocks} tokens`,\n '',\n\n `## \uD83C\uDFAF Current Work State`,\n `**Task**: ${context.workingState.currentTask}`,\n `**Active Files** (${context.workingState.activeFiles.length}):`,\n ...context.workingState.activeFiles.slice(0, 10).map((f) => `- ${f}`),\n '',\n `**Recent Commands**:`,\n ...context.workingState.recentCommands\n .slice(0, 5)\n .map((c) => `- \\`${c}\\``),\n '',\n\n `## \uD83D\uDCAC Conversation State`,\n `**Topic**: ${context.conversationState.conversationTopic}`,\n `**Messages**: ${context.conversationState.messageCount}`,\n `**Last User**: ${context.conversationState.lastUserMessage.substring(0, 100)}...`,\n '',\n\n `## \uD83D\uDCDD Code Context`,\n `**Git Branch**: ${context.codeContext.gitStatus.branch}`,\n `**Modified Files**: ${context.codeContext.modifiedFiles.length}`,\n `**Staged**: ${context.codeContext.gitStatus.staged.length}`,\n `**Unstaged**: ${context.codeContext.gitStatus.unstaged.length}`,\n '',\n\n `## \uD83E\uDDE0 Cognitive State`,\n `**Current Focus**: ${context.cognitiveState.currentFocus}`,\n `**Mental Model**:`,\n ...context.cognitiveState.mentalModel.slice(0, 5).map((m) => `- ${m}`),\n '',\n\n `## \uD83C\uDF0D Environment`,\n `**Directory**: ${context.environment.workingDirectory}`,\n `**Node Version**: ${context.environment.nodeVersion}`,\n `**Git Branch**: ${context.environment.gitBranch}`,\n '',\n ];\n\n return lines.filter((l) => l !== undefined).join('\\n');\n }\n\n // Helper methods (simplified implementations)\n\n private async estimateConversationTokens(): Promise<number> {\n // Simplified - would analyze recent conversation history\n return 15000;\n }\n\n private async estimateCodeBlockTokens(): Promise<number> {\n // Simplified - would analyze code blocks in context\n return 8000;\n }\n\n private async getCurrentActiveFrame(): Promise<any> {\n const stack = await this.frameManager.getStack();\n return stack.frames.find((f) => f.status === 'open');\n }\n\n private extractActiveFiles(traces: any[]): string[] {\n const files = new Set<string>();\n traces.forEach((trace) => {\n if (trace.content?.file_path) files.add(trace.content.file_path);\n if (trace.content?.path) files.add(trace.content.path);\n });\n return Array.from(files).slice(0, 20);\n }\n\n private extractPendingActions(frame: any): string[] {\n if (!frame?.metadata?.pendingActions) return [];\n return frame.metadata.pendingActions;\n }\n\n private extractBlockers(traces: any[]): string[] {\n return traces\n .filter((t) => t.type === 'error' && !t.metadata?.resolved)\n .map((t) => t.content.error || 'Unknown error')\n .slice(0, 5);\n }\n\n private inferConversationTopic(traces: any[]): string {\n // Simplified - would use NLP to infer topic\n return 'Code implementation and debugging';\n }\n\n private buildRecentContextSummary(traces: any[]): string[] {\n return traces\n .slice(0, 10)\n .map(\n (t) =>\n `${t.type}: ${t.content.summary || t.content.description || 'No description'}`\n )\n .filter((s) => s.length > 10);\n }\n\n private async captureGitStatus(): Promise<GitStatus> {\n try {\n const branch = execSync('git branch --show-current', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n }).trim();\n const staged = execSync('git diff --cached --name-only', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n const unstaged = execSync('git diff --name-only', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n const untracked = execSync('git ls-files --others --exclude-standard', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n })\n .trim()\n .split('\\n')\n .filter(Boolean);\n\n return {\n branch,\n ahead: 0, // Would implement git status parsing\n behind: 0,\n staged,\n unstaged,\n untracked,\n lastCommit: {\n hash: 'abc123', // Would get from git log\n message: 'Recent commit',\n timestamp: new Date().toISOString(),\n },\n };\n } catch (error: unknown) {\n return {\n branch: 'unknown',\n ahead: 0,\n behind: 0,\n staged: [],\n unstaged: [],\n untracked: [],\n lastCommit: { hash: '', message: '', timestamp: '' },\n };\n }\n }\n\n private async captureModifiedFiles(): Promise<FileContext[]> {\n try {\n const output = execSync('git diff --name-status', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n });\n return output\n .trim()\n .split('\\n')\n .filter(Boolean)\n .map((line) => {\n const [status, path] = line.split('\\t');\n return {\n path,\n lastModified: new Date().toISOString(),\n changeType:\n status === 'A'\n ? 'created'\n : status === 'D'\n ? 'deleted'\n : 'modified',\n lineChanges: { added: 0, removed: 0 }, // Would get from git diff --stat\n purpose: 'Code changes',\n relatedFiles: [],\n };\n });\n } catch (error: unknown) {\n return [];\n }\n }\n\n private async captureTestResults(): Promise<TestResults | undefined> {\n // Would implement test result parsing\n return undefined;\n }\n\n private async captureBuildStatus(): Promise<BuildStatus | undefined> {\n // Would implement build status checking\n return undefined;\n }\n\n private async captureDependencies(): Promise<DependencyInfo[]> {\n try {\n const packageJsonPath = path.join(this.projectRoot, 'package.json');\n const content = await fs.readFile(packageJsonPath, 'utf-8');\n const packageJson = JSON.parse(content);\n\n const deps: DependencyInfo[] = [];\n\n Object.entries(packageJson.dependencies || {}).forEach(\n ([name, version]) => {\n deps.push({\n name,\n version: version as string,\n type: 'dependency',\n critical: ['react', 'express', 'next'].includes(name),\n });\n }\n );\n\n return deps;\n } catch (error: unknown) {\n return [];\n }\n }\n\n private async extractCurrentFocus(): Promise<string> {\n const activeFrame = await this.getCurrentActiveFrame();\n return activeFrame?.description || 'No current focus';\n }\n\n private extractMentalModel(traces: any[]): string[] {\n // Would extract mental model concepts from traces\n return [\n 'Component architecture',\n 'Data flow patterns',\n 'Error handling strategy',\n ];\n }\n\n private extractAssumptions(traces: any[]): string[] {\n // Would extract assumptions from traces\n return [\n 'User input is validated',\n 'Database is available',\n 'Network is stable',\n ];\n }\n\n private extractHypotheses(traces: any[]): string[] {\n // Would extract hypotheses being tested\n return [\n 'Bug is in validation logic',\n 'Performance issue is database-related',\n ];\n }\n\n private extractExplorationPaths(traces: any[]): string[] {\n // Would extract different approaches being explored\n return [\n 'Try different algorithm',\n 'Refactor data structure',\n 'Add caching layer',\n ];\n }\n\n private async getCurrentGitBranch(): Promise<string> {\n try {\n return execSync('git branch --show-current', {\n encoding: 'utf-8',\n cwd: this.projectRoot,\n }).trim();\n } catch (error: unknown) {\n return 'unknown';\n }\n }\n\n private async getPackageJson(): Promise<any> {\n try {\n const content = await fs.readFile(\n path.join(this.projectRoot, 'package.json'),\n 'utf-8'\n );\n return JSON.parse(content);\n } catch (error: unknown) {\n return null;\n }\n }\n\n private getRelevantEnvVars(): Record<string, string> {\n const relevantVars = ['NODE_ENV', 'DEBUG', 'PORT', 'DATABASE_URL'];\n const result: Record<string, string> = {};\n\n relevantVars.forEach((varName) => {\n if (process.env[varName]) {\n result[varName] = process.env[varName]!;\n }\n });\n\n return result;\n }\n\n /**\n * Restore context after /clear\n */\n async restoreFromEnhancedContext(): Promise<boolean> {\n const latestPath = path.join(\n this.projectRoot,\n '.stackmemory',\n 'pre-clear',\n 'latest-context.json'\n );\n\n try {\n const content = await fs.readFile(latestPath, 'utf-8');\n const context: PreClearContext = JSON.parse(content);\n\n console.log('\uD83D\uDCDA Restoring enhanced context...');\n console.log(` Session: ${context.sessionId}`);\n console.log(` Task: ${context.workingState.currentTask}`);\n console.log(` Files: ${context.workingState.activeFiles.length}`);\n console.log(` Focus: ${context.cognitiveState.currentFocus}`);\n\n // Restore using existing systems\n await this.clearSurvival.restoreFromLedger();\n\n // Additional restoration logic would go here\n\n return true;\n } catch (error: unknown) {\n console.error('Failed to restore enhanced context:', error);\n return false;\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": ["path"]
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/integrations/claude-code/lifecycle-hooks.ts"],
4
- "sourcesContent": ["/**\n * Claude Code Lifecycle Hooks for StackMemory\n * Integrates with Claude Code's session lifecycle\n */\n\nimport { SessionMonitor } from '../../core/monitoring/session-monitor';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as os from 'os';\n\nexport interface ClaudeCodeHookConfig {\n projectRoot: string;\n autoTriggers: {\n onContextHigh: boolean; // Auto-save at 70%\n onContextCritical: boolean; // Force-save at 85%\n onSessionIdle: boolean; // Handoff on 5min idle\n onSessionEnd: boolean; // Handoff on close\n onClearCommand: boolean; // Save before /clear\n };\n claudeHooksPath?: string; // Path to .claude/hooks\n}\n\nexport class ClaudeCodeLifecycleHooks {\n private monitor?: SessionMonitor;\n private config: ClaudeCodeHookConfig;\n private isActive: boolean = false;\n private hookScripts: Map<string, string> = new Map();\n\n constructor(config: ClaudeCodeHookConfig) {\n this.config = {\n ...config,\n claudeHooksPath:\n config.claudeHooksPath || path.join(os.homedir(), '.claude', 'hooks'),\n };\n }\n\n /**\n * Initialize hooks and start monitoring\n */\n async initialize(): Promise<void> {\n if (this.isActive) return;\n\n // Initialize database and managers\n const dbPath = path.join(\n this.config.projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n const dbManager = new DatabaseManager(dbPath);\n await dbManager.initialize();\n\n const frameManager = new FrameManager(dbManager);\n\n // Create session monitor\n this.monitor = new SessionMonitor(\n frameManager,\n dbManager,\n this.config.projectRoot,\n {\n contextWarningThreshold: 0.6,\n contextCriticalThreshold: 0.7,\n contextAutoSaveThreshold: 0.85,\n idleTimeoutMinutes: 5,\n autoSaveLedger: this.config.autoTriggers.onContextCritical,\n autoGenerateHandoff: this.config.autoTriggers.onSessionIdle,\n sessionEndHandoff: this.config.autoTriggers.onSessionEnd,\n }\n );\n\n // Register event handlers\n this.registerEventHandlers();\n\n // Install Claude Code hooks\n await this.installClaudeHooks();\n\n // Start monitoring\n await this.monitor.start();\n\n this.isActive = true;\n console.log('\u2705 Claude Code lifecycle hooks initialized');\n }\n\n /**\n * Register event handlers for monitor events\n */\n private registerEventHandlers(): void {\n if (!this.monitor) return;\n\n // Context events\n this.monitor.on('context:warning', (data) => {\n console.log(`\u26A0\uFE0F Context at ${Math.round(data.percentage * 100)}%`);\n });\n\n this.monitor.on('context:high', async (data) => {\n if (this.config.autoTriggers.onContextHigh) {\n console.log('\uD83D\uDFE1 High context - preparing auto-save');\n await this.executeHook('on-context-high', data);\n }\n });\n\n this.monitor.on('context:ledger_saved', (data) => {\n console.log(`\u2705 Ledger saved (${data.compression}x compression)`);\n console.log('\uD83D\uDCA1 You can now safely use /clear');\n });\n\n // Handoff events\n this.monitor.on('handoff:generated', (data) => {\n console.log(`\uD83D\uDCCB Handoff saved (trigger: ${data.trigger})`);\n });\n }\n\n /**\n * Install hooks into Claude Code configuration\n */\n private async installClaudeHooks(): Promise<void> {\n // Create hooks directory if it doesn't exist\n await fs.mkdir(this.config.claudeHooksPath!, { recursive: true });\n\n // Install context monitor hook\n await this.installHook(\n 'on-message-submit',\n `\n#!/bin/bash\n# StackMemory Context Monitor Hook\n# Monitors token usage and triggers auto-save when needed\n\n# Get estimated token count from Claude (if available)\nTOKEN_COUNT=\\${CLAUDE_TOKEN_COUNT:-0}\nMAX_TOKENS=\\${CLAUDE_MAX_TOKENS:-100000}\n\nif [ \"\\$TOKEN_COUNT\" -gt 0 ]; then\n USAGE=\\$((TOKEN_COUNT * 100 / MAX_TOKENS))\n \n if [ \"\\$USAGE\" -gt 85 ]; then\n echo \"\uD83D\uDD34 Critical: Context at \\${USAGE}% - Auto-saving...\"\n stackmemory clear --save > /dev/null 2>&1\n echo \"\u2705 Ledger saved. Consider using /clear\"\n elif [ \"\\$USAGE\" -gt 70 ]; then\n echo \"\u26A0\uFE0F Warning: Context at \\${USAGE}%\"\n echo \"\uD83D\uDCA1 Run: stackmemory clear --save\"\n fi\nfi\n\n# Update activity timestamp\nstackmemory monitor --activity 2>/dev/null || true\n`\n );\n\n // Install session end hook\n await this.installHook(\n 'on-session-end',\n `\n#!/bin/bash\n# StackMemory Session End Hook\n# Generates handoff document when session ends\n\necho \"\uD83D\uDCE6 Saving session state...\"\n\n# Generate handoff\nstackmemory handoff --generate > /dev/null 2>&1 && echo \"\u2705 Handoff saved\"\n\n# Save ledger if context is significant\nCONTEXT_STATUS=$(stackmemory clear --check 2>/dev/null | grep -o '[0-9]\\\\+%' | head -1 | tr -d '%')\nif [ \"\\${CONTEXT_STATUS:-0}\" -gt 30 ]; then\n stackmemory clear --save > /dev/null 2>&1 && echo \"\u2705 Continuity ledger saved\"\nfi\n\necho \"\uD83D\uDC4B Session state preserved for next time\"\n`\n );\n\n // Install clear command interceptor\n await this.installHook(\n 'on-command-clear',\n `\n#!/bin/bash\n# StackMemory Clear Interceptor\n# Saves state before /clear command\n\necho \"\uD83D\uDD04 Preparing for /clear...\"\n\n# Save continuity ledger\nstackmemory clear --save > /dev/null 2>&1\necho \"\u2705 Continuity ledger saved\"\n\n# Generate quick handoff\nstackmemory handoff --generate > /dev/null 2>&1\necho \"\u2705 Handoff document saved\"\n\necho \"\u2705 Ready for /clear - context will be restored automatically\"\necho \"\uD83D\uDCA1 After /clear, run: stackmemory clear --restore\"\n`\n );\n\n // Install idle detector\n await this.installHook(\n 'on-idle-5min',\n `\n#!/bin/bash\n# StackMemory Idle Detector\n# Generates handoff after 5 minutes of inactivity\n\necho \"\u23F8\uFE0F Session idle - generating handoff...\"\nstackmemory handoff --generate > /dev/null 2>&1\necho \"\u2705 Handoff saved. Ready to resume anytime.\"\n`\n );\n }\n\n /**\n * Install a specific hook script\n */\n private async installHook(name: string, script: string): Promise<void> {\n const hookPath = path.join(this.config.claudeHooksPath!, name);\n\n // Store script content\n this.hookScripts.set(name, script);\n\n // Write hook file\n await fs.writeFile(hookPath, script.trim(), { mode: 0o755 });\n\n // Make executable\n await fs.chmod(hookPath, 0o755);\n }\n\n /**\n * Execute a hook with context\n */\n private async executeHook(hookName: string, context: any): Promise<void> {\n const hookPath = path.join(this.config.claudeHooksPath!, hookName);\n\n try {\n // Check if hook exists\n await fs.access(hookPath);\n\n // Execute hook with environment variables\n const { exec } = await import('child_process');\n const { promisify } = await import('util');\n const execAsync = promisify(exec);\n\n const env = {\n ...process.env,\n STACKMEMORY_CONTEXT: JSON.stringify(context),\n STACKMEMORY_PROJECT: this.config.projectRoot,\n };\n\n const { stdout, stderr } = await execAsync(hookPath, { env });\n\n if (stdout) console.log(stdout);\n if (stderr) console.error(stderr);\n } catch (error) {\n // Hook doesn't exist or failed\n console.debug(`Hook ${hookName} not found or failed:`, error);\n }\n }\n\n /**\n * Stop monitoring and cleanup\n */\n async stop(): Promise<void> {\n if (this.monitor) {\n await this.monitor.stop();\n }\n this.isActive = false;\n console.log('\uD83D\uDED1 Claude Code lifecycle hooks stopped');\n }\n\n /**\n * Get current status\n */\n getStatus(): any {\n return {\n isActive: this.isActive,\n monitorStatus: this.monitor?.getStatus(),\n config: this.config,\n installedHooks: Array.from(this.hookScripts.keys()),\n };\n }\n}\n\n/**\n * Global singleton instance\n */\nlet globalInstance: ClaudeCodeLifecycleHooks | undefined;\n\n/**\n * Initialize global hooks\n */\nexport async function initializeClaudeHooks(\n projectRoot?: string\n): Promise<ClaudeCodeLifecycleHooks> {\n if (!projectRoot) {\n projectRoot = process.cwd();\n }\n\n if (!globalInstance) {\n globalInstance = new ClaudeCodeLifecycleHooks({\n projectRoot,\n autoTriggers: {\n onContextHigh: true,\n onContextCritical: true,\n onSessionIdle: true,\n onSessionEnd: true,\n onClearCommand: true,\n },\n });\n\n await globalInstance.initialize();\n }\n\n return globalInstance;\n}\n\n/**\n * Get global instance\n */\nexport function getClaudeHooks(): ClaudeCodeLifecycleHooks | undefined {\n return globalInstance;\n}\n\n/**\n * Stop global hooks\n */\nexport async function stopClaudeHooks(): Promise<void> {\n if (globalInstance) {\n await globalInstance.stop();\n globalInstance = undefined;\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Claude Code Lifecycle Hooks for StackMemory\n * Integrates with Claude Code's session lifecycle\n */\n\nimport { SessionMonitor } from '../../core/monitoring/session-monitor';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as os from 'os';\n\nexport interface ClaudeCodeHookConfig {\n projectRoot: string;\n autoTriggers: {\n onContextHigh: boolean; // Auto-save at 70%\n onContextCritical: boolean; // Force-save at 85%\n onSessionIdle: boolean; // Handoff on 5min idle\n onSessionEnd: boolean; // Handoff on close\n onClearCommand: boolean; // Save before /clear\n };\n claudeHooksPath?: string; // Path to .claude/hooks\n}\n\nexport class ClaudeCodeLifecycleHooks {\n private monitor?: SessionMonitor;\n private config: ClaudeCodeHookConfig;\n private isActive: boolean = false;\n private hookScripts: Map<string, string> = new Map();\n\n constructor(config: ClaudeCodeHookConfig) {\n this.config = {\n ...config,\n claudeHooksPath:\n config.claudeHooksPath || path.join(os.homedir(), '.claude', 'hooks'),\n };\n }\n\n /**\n * Initialize hooks and start monitoring\n */\n async initialize(): Promise<void> {\n if (this.isActive) return;\n\n // Initialize database and managers\n const dbPath = path.join(\n this.config.projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n const dbManager = new DatabaseManager(dbPath);\n await dbManager.initialize();\n\n const frameManager = new FrameManager(dbManager);\n\n // Create session monitor\n this.monitor = new SessionMonitor(\n frameManager,\n dbManager,\n this.config.projectRoot,\n {\n contextWarningThreshold: 0.6,\n contextCriticalThreshold: 0.7,\n contextAutoSaveThreshold: 0.85,\n idleTimeoutMinutes: 5,\n autoSaveLedger: this.config.autoTriggers.onContextCritical,\n autoGenerateHandoff: this.config.autoTriggers.onSessionIdle,\n sessionEndHandoff: this.config.autoTriggers.onSessionEnd,\n }\n );\n\n // Register event handlers\n this.registerEventHandlers();\n\n // Install Claude Code hooks\n await this.installClaudeHooks();\n\n // Start monitoring\n await this.monitor.start();\n\n this.isActive = true;\n console.log('\u2705 Claude Code lifecycle hooks initialized');\n }\n\n /**\n * Register event handlers for monitor events\n */\n private registerEventHandlers(): void {\n if (!this.monitor) return;\n\n // Context events\n this.monitor.on('context:warning', (data) => {\n console.log(`\u26A0\uFE0F Context at ${Math.round(data.percentage * 100)}%`);\n });\n\n this.monitor.on('context:high', async (data) => {\n if (this.config.autoTriggers.onContextHigh) {\n console.log('\uD83D\uDFE1 High context - preparing auto-save');\n await this.executeHook('on-context-high', data);\n }\n });\n\n this.monitor.on('context:ledger_saved', (data) => {\n console.log(`\u2705 Ledger saved (${data.compression}x compression)`);\n console.log('\uD83D\uDCA1 You can now safely use /clear');\n });\n\n // Handoff events\n this.monitor.on('handoff:generated', (data) => {\n console.log(`\uD83D\uDCCB Handoff saved (trigger: ${data.trigger})`);\n });\n }\n\n /**\n * Install hooks into Claude Code configuration\n */\n private async installClaudeHooks(): Promise<void> {\n // Create hooks directory if it doesn't exist\n await fs.mkdir(this.config.claudeHooksPath!, { recursive: true });\n\n // Install context monitor hook\n await this.installHook(\n 'on-message-submit',\n `\n#!/bin/bash\n# StackMemory Context Monitor Hook\n# Monitors token usage and triggers auto-save when needed\n\n# Get estimated token count from Claude (if available)\nTOKEN_COUNT=\\${CLAUDE_TOKEN_COUNT:-0}\nMAX_TOKENS=\\${CLAUDE_MAX_TOKENS:-100000}\n\nif [ \"\\$TOKEN_COUNT\" -gt 0 ]; then\n USAGE=\\$((TOKEN_COUNT * 100 / MAX_TOKENS))\n \n if [ \"\\$USAGE\" -gt 85 ]; then\n echo \"\uD83D\uDD34 Critical: Context at \\${USAGE}% - Auto-saving...\"\n stackmemory clear --save > /dev/null 2>&1\n echo \"\u2705 Ledger saved. Consider using /clear\"\n elif [ \"\\$USAGE\" -gt 70 ]; then\n echo \"\u26A0\uFE0F Warning: Context at \\${USAGE}%\"\n echo \"\uD83D\uDCA1 Run: stackmemory clear --save\"\n fi\nfi\n\n# Update activity timestamp\nstackmemory monitor --activity 2>/dev/null || true\n`\n );\n\n // Install session end hook\n await this.installHook(\n 'on-session-end',\n `\n#!/bin/bash\n# StackMemory Session End Hook\n# Generates handoff document when session ends\n\necho \"\uD83D\uDCE6 Saving session state...\"\n\n# Generate handoff\nstackmemory handoff --generate > /dev/null 2>&1 && echo \"\u2705 Handoff saved\"\n\n# Save ledger if context is significant\nCONTEXT_STATUS=$(stackmemory clear --check 2>/dev/null | grep -o '[0-9]\\\\+%' | head -1 | tr -d '%')\nif [ \"\\${CONTEXT_STATUS:-0}\" -gt 30 ]; then\n stackmemory clear --save > /dev/null 2>&1 && echo \"\u2705 Continuity ledger saved\"\nfi\n\necho \"\uD83D\uDC4B Session state preserved for next time\"\n`\n );\n\n // Install clear command interceptor\n await this.installHook(\n 'on-command-clear',\n `\n#!/bin/bash\n# StackMemory Clear Interceptor\n# Saves state before /clear command\n\necho \"\uD83D\uDD04 Preparing for /clear...\"\n\n# Save continuity ledger\nstackmemory clear --save > /dev/null 2>&1\necho \"\u2705 Continuity ledger saved\"\n\n# Generate quick handoff\nstackmemory handoff --generate > /dev/null 2>&1\necho \"\u2705 Handoff document saved\"\n\necho \"\u2705 Ready for /clear - context will be restored automatically\"\necho \"\uD83D\uDCA1 After /clear, run: stackmemory clear --restore\"\n`\n );\n\n // Install idle detector\n await this.installHook(\n 'on-idle-5min',\n `\n#!/bin/bash\n# StackMemory Idle Detector\n# Generates handoff after 5 minutes of inactivity\n\necho \"\u23F8\uFE0F Session idle - generating handoff...\"\nstackmemory handoff --generate > /dev/null 2>&1\necho \"\u2705 Handoff saved. Ready to resume anytime.\"\n`\n );\n }\n\n /**\n * Install a specific hook script\n */\n private async installHook(name: string, script: string): Promise<void> {\n const hookPath = path.join(this.config.claudeHooksPath!, name);\n\n // Store script content\n this.hookScripts.set(name, script);\n\n // Write hook file\n await fs.writeFile(hookPath, script.trim(), { mode: 0o755 });\n\n // Make executable\n await fs.chmod(hookPath, 0o755);\n }\n\n /**\n * Execute a hook with context\n */\n private async executeHook(hookName: string, context: any): Promise<void> {\n const hookPath = path.join(this.config.claudeHooksPath!, hookName);\n\n try {\n // Check if hook exists\n await fs.access(hookPath);\n\n // Execute hook with environment variables\n const { exec } = await import('child_process');\n const { promisify } = await import('util');\n const execAsync = promisify(exec);\n\n const env = {\n ...process.env,\n STACKMEMORY_CONTEXT: JSON.stringify(context),\n STACKMEMORY_PROJECT: this.config.projectRoot,\n };\n\n const { stdout, stderr } = await execAsync(hookPath, { env });\n\n if (stdout) console.log(stdout);\n if (stderr) console.error(stderr);\n } catch (error: unknown) {\n // Hook doesn't exist or failed\n console.debug(`Hook ${hookName} not found or failed:`, error);\n }\n }\n\n /**\n * Stop monitoring and cleanup\n */\n async stop(): Promise<void> {\n if (this.monitor) {\n await this.monitor.stop();\n }\n this.isActive = false;\n console.log('\uD83D\uDED1 Claude Code lifecycle hooks stopped');\n }\n\n /**\n * Get current status\n */\n getStatus(): any {\n return {\n isActive: this.isActive,\n monitorStatus: this.monitor?.getStatus(),\n config: this.config,\n installedHooks: Array.from(this.hookScripts.keys()),\n };\n }\n}\n\n/**\n * Global singleton instance\n */\nlet globalInstance: ClaudeCodeLifecycleHooks | undefined;\n\n/**\n * Initialize global hooks\n */\nexport async function initializeClaudeHooks(\n projectRoot?: string\n): Promise<ClaudeCodeLifecycleHooks> {\n if (!projectRoot) {\n projectRoot = process.cwd();\n }\n\n if (!globalInstance) {\n globalInstance = new ClaudeCodeLifecycleHooks({\n projectRoot,\n autoTriggers: {\n onContextHigh: true,\n onContextCritical: true,\n onSessionIdle: true,\n onSessionEnd: true,\n onClearCommand: true,\n },\n });\n\n await globalInstance.initialize();\n }\n\n return globalInstance;\n}\n\n/**\n * Get global instance\n */\nexport function getClaudeHooks(): ClaudeCodeLifecycleHooks | undefined {\n return globalInstance;\n}\n\n/**\n * Stop global hooks\n */\nexport async function stopClaudeHooks(): Promise<void> {\n if (globalInstance) {\n await globalInstance.stop();\n globalInstance = undefined;\n }\n}\n"],
5
+ "mappings": "AAKA,SAAS,sBAAsB;AAC/B,SAAS,oBAAoB;AAC7B,SAAS,uBAAuB;AAChC,YAAY,QAAQ;AACpB,YAAY,UAAU;AACtB,YAAY,QAAQ;AAcb,MAAM,yBAAyB;AAAA,EAC5B;AAAA,EACA;AAAA,EACA,WAAoB;AAAA,EACpB,cAAmC,oBAAI,IAAI;AAAA,EAEnD,YAAY,QAA8B;AACxC,SAAK,SAAS;AAAA,MACZ,GAAG;AAAA,MACH,iBACE,OAAO,mBAAmB,KAAK,KAAK,GAAG,QAAQ,GAAG,WAAW,OAAO;AAAA,IACxE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAA4B;AAChC,QAAI,KAAK,SAAU;AAGnB,UAAM,SAAS,KAAK;AAAA,MAClB,KAAK,OAAO;AAAA,MACZ;AAAA,MACA;AAAA,MACA;AAAA,IACF;AACA,UAAM,YAAY,IAAI,gBAAgB,MAAM;AAC5C,UAAM,UAAU,WAAW;AAE3B,UAAM,eAAe,IAAI,aAAa,SAAS;AAG/C,SAAK,UAAU,IAAI;AAAA,MACjB;AAAA,MACA;AAAA,MACA,KAAK,OAAO;AAAA,MACZ;AAAA,QACE,yBAAyB;AAAA,QACzB,0BAA0B;AAAA,QAC1B,0BAA0B;AAAA,QAC1B,oBAAoB;AAAA,QACpB,gBAAgB,KAAK,OAAO,aAAa;AAAA,QACzC,qBAAqB,KAAK,OAAO,aAAa;AAAA,QAC9C,mBAAmB,KAAK,OAAO,aAAa;AAAA,MAC9C;AAAA,IACF;AAGA,SAAK,sBAAsB;AAG3B,UAAM,KAAK,mBAAmB;AAG9B,UAAM,KAAK,QAAQ,MAAM;AAEzB,SAAK,WAAW;AAChB,YAAQ,IAAI,gDAA2C;AAAA,EACzD;AAAA;AAAA;AAAA;AAAA,EAKQ,wBAA8B;AACpC,QAAI,CAAC,KAAK,QAAS;AAGnB,SAAK,QAAQ,GAAG,mBAAmB,CAAC,SAAS;AAC3C,cAAQ,IAAI,2BAAiB,KAAK,MAAM,KAAK,aAAa,GAAG,CAAC,GAAG;AAAA,IACnE,CAAC;AAED,SAAK,QAAQ,GAAG,gBAAgB,OAAO,SAAS;AAC9C,UAAI,KAAK,OAAO,aAAa,eAAe;AAC1C,gBAAQ,IAAI,8CAAuC;AACnD,cAAM,KAAK,YAAY,mBAAmB,IAAI;AAAA,MAChD;AAAA,IACF,CAAC;AAED,SAAK,QAAQ,GAAG,wBAAwB,CAAC,SAAS;AAChD,cAAQ,IAAI,wBAAmB,KAAK,WAAW,gBAAgB;AAC/D,cAAQ,IAAI,yCAAkC;AAAA,IAChD,CAAC;AAGD,SAAK,QAAQ,GAAG,qBAAqB,CAAC,SAAS;AAC7C,cAAQ,IAAI,qCAA8B,KAAK,OAAO,GAAG;AAAA,IAC3D,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,qBAAoC;AAEhD,UAAM,GAAG,MAAM,KAAK,OAAO,iBAAkB,EAAE,WAAW,KAAK,CAAC;AAGhE,UAAM,KAAK;AAAA,MACT;AAAA,MACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAyBF;AAGA,UAAM,KAAK;AAAA,MACT;AAAA,MACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAkBF;AAGA,UAAM,KAAK;AAAA,MACT;AAAA,MACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAkBF;AAGA,UAAM,KAAK;AAAA,MACT;AAAA,MACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IASF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,YAAY,MAAc,QAA+B;AACrE,UAAM,WAAW,KAAK,KAAK,KAAK,OAAO,iBAAkB,IAAI;AAG7D,SAAK,YAAY,IAAI,MAAM,MAAM;AAGjC,UAAM,GAAG,UAAU,UAAU,OAAO,KAAK,GAAG,EAAE,MAAM,IAAM,CAAC;AAG3D,UAAM,GAAG,MAAM,UAAU,GAAK;AAAA,EAChC;AAAA;AAAA;AAAA;AAAA,EAKA,MAAc,YAAY,UAAkB,SAA6B;AACvE,UAAM,WAAW,KAAK,KAAK,KAAK,OAAO,iBAAkB,QAAQ;AAEjE,QAAI;AAEF,YAAM,GAAG,OAAO,QAAQ;AAGxB,YAAM,EAAE,KAAK,IAAI,MAAM,OAAO,eAAe;AAC7C,YAAM,EAAE,UAAU,IAAI,MAAM,OAAO,MAAM;AACzC,YAAM,YAAY,UAAU,IAAI;AAEhC,YAAM,MAAM;AAAA,QACV,GAAG,QAAQ;AAAA,QACX,qBAAqB,KAAK,UAAU,OAAO;AAAA,QAC3C,qBAAqB,KAAK,OAAO;AAAA,MACnC;AAEA,YAAM,EAAE,QAAQ,OAAO,IAAI,MAAM,UAAU,UAAU,EAAE,IAAI,CAAC;AAE5D,UAAI,OAAQ,SAAQ,IAAI,MAAM;AAC9B,UAAI,OAAQ,SAAQ,MAAM,MAAM;AAAA,IAClC,SAAS,OAAgB;AAEvB,cAAQ,MAAM,QAAQ,QAAQ,yBAAyB,KAAK;AAAA,IAC9D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,OAAsB;AAC1B,QAAI,KAAK,SAAS;AAChB,YAAM,KAAK,QAAQ,KAAK;AAAA,IAC1B;AACA,SAAK,WAAW;AAChB,YAAQ,IAAI,+CAAwC;AAAA,EACtD;AAAA;AAAA;AAAA;AAAA,EAKA,YAAiB;AACf,WAAO;AAAA,MACL,UAAU,KAAK;AAAA,MACf,eAAe,KAAK,SAAS,UAAU;AAAA,MACvC,QAAQ,KAAK;AAAA,MACb,gBAAgB,MAAM,KAAK,KAAK,YAAY,KAAK,CAAC;AAAA,IACpD;AAAA,EACF;AACF;AAKA,IAAI;AAKJ,eAAsB,sBACpB,aACmC;AACnC,MAAI,CAAC,aAAa;AAChB,kBAAc,QAAQ,IAAI;AAAA,EAC5B;AAEA,MAAI,CAAC,gBAAgB;AACnB,qBAAiB,IAAI,yBAAyB;AAAA,MAC5C;AAAA,MACA,cAAc;AAAA,QACZ,eAAe;AAAA,QACf,mBAAmB;AAAA,QACnB,eAAe;AAAA,QACf,cAAc;AAAA,QACd,gBAAgB;AAAA,MAClB;AAAA,IACF,CAAC;AAED,UAAM,eAAe,WAAW;AAAA,EAClC;AAEA,SAAO;AACT;AAKO,SAAS,iBAAuD;AACrE,SAAO;AACT;AAKA,eAAsB,kBAAiC;AACrD,MAAI,gBAAgB;AAClB,UAAM,eAAe,KAAK;AAC1B,qBAAiB;AAAA,EACnB;AACF;",
6
6
  "names": []
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/integrations/claude-code/post-task-hooks.ts"],
4
- "sourcesContent": ["/**\n * Post-Task Completion Hooks for StackMemory\n * Automatically runs tests and code review after Claude completes tasks\n */\n\nimport { EventEmitter } from 'events';\nimport { execSync } from 'child_process';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\n\nexport interface PostTaskConfig {\n projectRoot: string;\n qualityGates: {\n runTests: boolean; // Auto-run tests after code changes\n requireTestCoverage: boolean; // Require test coverage for new code\n runCodeReview: boolean; // Auto-trigger code review agent\n runLinter: boolean; // Auto-run linter/formatter\n blockOnFailure: boolean; // Block further work if quality gates fail\n };\n testFrameworks: {\n detected: string[]; // Auto-detected test frameworks\n testCommand?: string; // Custom test command\n coverageCommand?: string; // Custom coverage command\n lintCommand?: string; // Custom lint command\n };\n reviewConfig: {\n reviewOnEveryChange: boolean; // Review every file change\n reviewOnTaskComplete: boolean; // Review when task frame closes\n focusAreas: string[]; // What to focus on in reviews\n skipPatterns: string[]; // Files/patterns to skip review\n };\n}\n\nexport interface TaskCompletionEvent {\n taskType: 'code_change' | 'task_complete' | 'file_modified' | 'frame_closed';\n files: string[];\n frameId: string;\n frameName: string;\n changes: {\n added: number;\n removed: number;\n modified: number;\n };\n metadata: Record<string, any>;\n}\n\nexport interface QualityGateResult {\n gate: string;\n passed: boolean;\n output: string;\n duration: number;\n issues?: QualityIssue[];\n}\n\nexport interface QualityIssue {\n type: 'test_failure' | 'lint_error' | 'coverage_low' | 'review_concern';\n file: string;\n line?: number;\n message: string;\n severity: 'error' | 'warning' | 'info';\n}\n\nexport class PostTaskHooks extends EventEmitter {\n private config: PostTaskConfig;\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private isActive: boolean = false;\n private lastProcessedFrame?: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n config: Partial<PostTaskConfig>\n ) {\n super();\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n\n // Default configuration\n this.config = {\n projectRoot: process.cwd(),\n qualityGates: {\n runTests: true,\n requireTestCoverage: false,\n runCodeReview: true,\n runLinter: true,\n blockOnFailure: false,\n },\n testFrameworks: {\n detected: [],\n },\n reviewConfig: {\n reviewOnEveryChange: false,\n reviewOnTaskComplete: true,\n focusAreas: [\n 'security',\n 'performance',\n 'maintainability',\n 'correctness',\n ],\n skipPatterns: ['*.test.ts', '*.spec.js', 'dist/', 'node_modules/'],\n },\n ...config,\n };\n }\n\n /**\n * Initialize post-task hooks\n */\n async initialize(): Promise<void> {\n if (this.isActive) return;\n\n // Detect test frameworks and commands\n await this.detectTestFrameworks();\n\n // Set up frame event listeners\n this.setupFrameListeners();\n\n // Set up file watchers for code changes\n await this.setupFileWatchers();\n\n this.isActive = true;\n console.log('\u2705 Post-task hooks initialized');\n this.emit('initialized', this.config);\n }\n\n /**\n * Detect available test frameworks and commands\n */\n private async detectTestFrameworks(): Promise<void> {\n const packageJsonPath = path.join(this.config.projectRoot, 'package.json');\n\n try {\n const packageJson = JSON.parse(\n await fs.readFile(packageJsonPath, 'utf-8')\n );\n const scripts = packageJson.scripts || {};\n const dependencies = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n };\n\n // Detect test frameworks\n const frameworks: string[] = [];\n if (dependencies.jest) frameworks.push('jest');\n if (dependencies.vitest) frameworks.push('vitest');\n if (dependencies.mocha) frameworks.push('mocha');\n if (dependencies.playwright) frameworks.push('playwright');\n if (dependencies.cypress) frameworks.push('cypress');\n\n this.config.testFrameworks.detected = frameworks;\n\n // Detect commands\n if (scripts.test) {\n this.config.testFrameworks.testCommand = 'npm test';\n } else if (scripts['test:run']) {\n this.config.testFrameworks.testCommand = 'npm run test:run';\n }\n\n if (scripts.coverage) {\n this.config.testFrameworks.coverageCommand = 'npm run coverage';\n }\n\n if (scripts.lint) {\n this.config.testFrameworks.lintCommand = 'npm run lint';\n }\n } catch (error) {\n console.warn('Could not detect test frameworks:', error);\n }\n }\n\n /**\n * Set up frame event listeners\n */\n private setupFrameListeners(): void {\n // Listen for frame closures (task completion)\n this.frameManager.on(\n 'frame:closed',\n async (frameId: string, frameData: any) => {\n if (frameData.type === 'task' || frameData.type === 'subtask') {\n await this.handleTaskCompletion({\n taskType: 'task_complete',\n frameId,\n frameName: frameData.name || 'Unnamed task',\n files: this.extractFilesFromFrame(frameData),\n changes: this.calculateChanges(frameData),\n metadata: frameData.metadata || {},\n });\n }\n }\n );\n\n // Listen for significant events\n this.frameManager.on(\n 'frame:event',\n async (frameId: string, eventType: string, data: any) => {\n if (eventType === 'code_change' || eventType === 'file_modified') {\n await this.handleTaskCompletion({\n taskType: 'code_change',\n frameId,\n frameName: data.description || 'Code change',\n files: data.files || [],\n changes: data.changes || { added: 0, removed: 0, modified: 1 },\n metadata: data,\n });\n }\n }\n );\n }\n\n /**\n * Set up file watchers for real-time code change detection\n */\n private async setupFileWatchers(): Promise<void> {\n try {\n const chokidar = await import('chokidar');\n\n const watcher = chokidar.watch(\n [\n '**/*.{ts,js,tsx,jsx,py,go,rs,java,cpp,c}',\n '!node_modules/**',\n '!dist/**',\n '!build/**',\n ],\n {\n cwd: this.config.projectRoot,\n ignored: /node_modules/,\n persistent: true,\n }\n );\n\n let changeQueue: string[] = [];\n let debounceTimer: NodeJS.Timeout | null = null;\n\n watcher.on('change', (filePath: string) => {\n changeQueue.push(filePath);\n\n // Debounce to avoid too many triggers\n if (debounceTimer) clearTimeout(debounceTimer);\n debounceTimer = setTimeout(async () => {\n if (changeQueue.length > 0) {\n await this.handleFileChanges([...changeQueue]);\n changeQueue = [];\n }\n }, 2000); // 2 second debounce\n });\n } catch (error) {\n console.warn('File watching not available:', error);\n }\n }\n\n /**\n * Handle task completion events\n */\n private async handleTaskCompletion(\n event: TaskCompletionEvent\n ): Promise<void> {\n // Avoid duplicate processing\n if (\n this.lastProcessedFrame === event.frameId &&\n event.taskType !== 'code_change'\n ) {\n return;\n }\n\n this.lastProcessedFrame = event.frameId;\n\n console.log(\n `\uD83D\uDD0D Task completed: ${event.frameName} (${event.files.length} files changed)`\n );\n this.emit('task:completed', event);\n\n // Run quality gates\n const results = await this.runQualityGates(event);\n\n // Check if gates passed\n const allPassed = results.every((r) => r.passed);\n\n if (allPassed) {\n console.log('\u2705 All quality gates passed');\n this.emit('quality:passed', { event, results });\n } else {\n console.log('\u26A0\uFE0F Quality gate failures detected');\n this.emit('quality:failed', { event, results });\n\n if (this.config.qualityGates.blockOnFailure) {\n await this.blockFurtherWork(results);\n }\n }\n\n // Record results in frame metadata\n await this.recordQualityResults(event.frameId, results);\n }\n\n /**\n * Handle file changes\n */\n private async handleFileChanges(files: string[]): Promise<void> {\n if (!this.config.reviewConfig.reviewOnEveryChange) return;\n\n // Filter out files that should be skipped\n const filteredFiles = files.filter((file) => {\n return !this.config.reviewConfig.skipPatterns.some((pattern) => {\n return file.includes(pattern.replace('*', ''));\n });\n });\n\n if (filteredFiles.length === 0) return;\n\n await this.handleTaskCompletion({\n taskType: 'file_modified',\n frameId: 'file-watcher',\n frameName: 'File changes detected',\n files: filteredFiles,\n changes: { added: 0, removed: 0, modified: filteredFiles.length },\n metadata: { trigger: 'file_watcher' },\n });\n }\n\n /**\n * Run all configured quality gates\n */\n private async runQualityGates(\n event: TaskCompletionEvent\n ): Promise<QualityGateResult[]> {\n const results: QualityGateResult[] = [];\n\n // Run linter first (fastest)\n if (this.config.qualityGates.runLinter) {\n results.push(await this.runLinter(event.files));\n }\n\n // Run tests (slower)\n if (this.config.qualityGates.runTests) {\n results.push(await this.runTests(event.files));\n }\n\n // Check test coverage\n if (this.config.qualityGates.requireTestCoverage) {\n results.push(await this.checkTestCoverage(event.files));\n }\n\n // Run code review (slowest, most thorough)\n if (this.config.qualityGates.runCodeReview) {\n results.push(await this.runCodeReview(event));\n }\n\n return results;\n }\n\n /**\n * Run linter on changed files\n */\n private async runLinter(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.lintCommand) {\n return {\n gate: 'linter',\n passed: true,\n output: 'No lint command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.lintCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 30000, // 30 second timeout\n });\n\n return {\n gate: 'linter',\n passed: true,\n output,\n duration: Date.now() - start,\n };\n } catch (error: any) {\n return {\n gate: 'linter',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: this.parseLintErrors(error.stdout || error.message),\n };\n }\n }\n\n /**\n * Run tests\n */\n private async runTests(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.testCommand) {\n return {\n gate: 'tests',\n passed: true,\n output: 'No test command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.testCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 120000, // 2 minute timeout\n });\n\n return {\n gate: 'tests',\n passed: true,\n output,\n duration: Date.now() - start,\n };\n } catch (error: any) {\n return {\n gate: 'tests',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: this.parseTestFailures(error.stdout || error.message),\n };\n }\n }\n\n /**\n * Check test coverage\n */\n private async checkTestCoverage(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.coverageCommand) {\n return {\n gate: 'coverage',\n passed: true,\n output: 'No coverage command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.coverageCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 120000,\n });\n\n // Parse coverage percentage (simplified)\n const coverageMatch = output.match(/(\\d+\\.?\\d*)%/);\n const coverage = coverageMatch ? parseFloat(coverageMatch[1]) : 0;\n const threshold = 80; // 80% coverage threshold\n\n return {\n gate: 'coverage',\n passed: coverage >= threshold,\n output,\n duration: Date.now() - start,\n issues:\n coverage < threshold\n ? [\n {\n type: 'coverage_low',\n file: 'overall',\n message: `Coverage ${coverage}% is below threshold ${threshold}%`,\n severity: 'warning' as const,\n },\n ]\n : undefined,\n };\n } catch (error: any) {\n return {\n gate: 'coverage',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: [\n {\n type: 'coverage_low',\n file: 'overall',\n message: 'Coverage check failed',\n severity: 'error' as const,\n },\n ],\n };\n }\n }\n\n /**\n * Run code review using AI agent\n */\n private async runCodeReview(\n event: TaskCompletionEvent\n ): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n // This would integrate with the code review agent\n const reviewPrompt = this.generateCodeReviewPrompt(event);\n\n // For now, simulate a review (in real implementation, call agent)\n const review = await this.callCodeReviewAgent(reviewPrompt, event.files);\n\n return {\n gate: 'code_review',\n passed: !review.issues || review.issues.length === 0,\n output: review.summary,\n duration: Date.now() - start,\n issues: review.issues,\n };\n } catch (error: any) {\n return {\n gate: 'code_review',\n passed: false,\n output: `Code review failed: ${error.message}`,\n duration: Date.now() - start,\n };\n }\n }\n\n /**\n * Generate code review prompt\n */\n private generateCodeReviewPrompt(event: TaskCompletionEvent): string {\n return `\nPlease review the following code changes:\n\nTask: ${event.frameName}\nFiles changed: ${event.files.join(', ')}\nChanges: +${event.changes.added}, -${event.changes.removed}, ~${event.changes.modified}\n\nFocus areas: ${this.config.reviewConfig.focusAreas.join(', ')}\n\nPlease check for:\n1. Security vulnerabilities\n2. Performance issues \n3. Code maintainability\n4. Correctness and logic errors\n5. Best practices adherence\n\nProvide specific, actionable feedback.\n`;\n }\n\n /**\n * Call code review agent (placeholder for actual implementation)\n */\n private async callCodeReviewAgent(\n prompt: string,\n files: string[]\n ): Promise<{\n summary: string;\n issues?: QualityIssue[];\n }> {\n // This would integrate with Claude Code's agent system\n // For now, return a mock review\n return {\n summary: `Reviewed ${files.length} files. Code quality looks good.`,\n issues: [],\n };\n }\n\n /**\n * Parse lint errors into structured issues\n */\n private parseLintErrors(output: string): QualityIssue[] {\n const issues: QualityIssue[] = [];\n\n // Simple parser for common lint formats\n const lines = output.split('\\n');\n for (const line of lines) {\n const match = line.match(\n /^(.+?):(\\d+):(\\d+):\\s*(error|warning):\\s*(.+)$/\n );\n if (match) {\n issues.push({\n type: 'lint_error',\n file: match[1],\n line: parseInt(match[2]),\n message: match[5],\n severity: match[4] as 'error' | 'warning',\n });\n }\n }\n\n return issues;\n }\n\n /**\n * Parse test failures into structured issues\n */\n private parseTestFailures(output: string): QualityIssue[] {\n const issues: QualityIssue[] = [];\n\n // Simple parser for test failures\n const lines = output.split('\\n');\n for (const line of lines) {\n if (line.includes('FAIL') || line.includes('\u2717')) {\n issues.push({\n type: 'test_failure',\n file: 'unknown',\n message: line.trim(),\n severity: 'error',\n });\n }\n }\n\n return issues;\n }\n\n /**\n * Block further work when quality gates fail\n */\n private async blockFurtherWork(results: QualityGateResult[]): Promise<void> {\n const failedGates = results.filter((r) => !r.passed);\n\n console.log('\uD83D\uDEAB Quality gates failed - blocking further work:');\n failedGates.forEach((gate) => {\n console.log(` ${gate.gate}: ${gate.output}`);\n if (gate.issues) {\n gate.issues.forEach((issue) => {\n console.log(\n ` - ${issue.severity}: ${issue.message} (${issue.file}:${issue.line || 0})`\n );\n });\n }\n });\n\n console.log('\\n\uD83D\uDD27 Fix these issues before continuing:');\n const allIssues = failedGates.flatMap((g) => g.issues || []);\n allIssues.forEach((issue, i) => {\n console.log(`${i + 1}. ${issue.message}`);\n });\n }\n\n /**\n * Record quality results in frame metadata\n */\n private async recordQualityResults(\n frameId: string,\n results: QualityGateResult[]\n ): Promise<void> {\n try {\n const frame = await this.frameManager.getFrame(frameId);\n if (frame) {\n frame.metadata = {\n ...frame.metadata,\n qualityGates: {\n timestamp: new Date().toISOString(),\n results,\n passed: results.every((r) => r.passed),\n totalDuration: results.reduce((sum, r) => sum + r.duration, 0),\n },\n };\n\n // Update frame with quality results\n await this.frameManager.updateFrame(frameId, frame);\n }\n } catch (error) {\n console.error('Failed to record quality results:', error);\n }\n }\n\n /**\n * Extract files from frame data\n */\n private extractFilesFromFrame(frameData: any): string[] {\n // Extract file paths from various possible locations\n const files: string[] = [];\n\n if (frameData.metadata?.files) {\n files.push(...frameData.metadata.files);\n }\n\n if (frameData.events) {\n frameData.events.forEach((event: any) => {\n if (event.type === 'file_change' && event.data?.file) {\n files.push(event.data.file);\n }\n });\n }\n\n return [...new Set(files)]; // Remove duplicates\n }\n\n /**\n * Calculate changes from frame data\n */\n private calculateChanges(frameData: any): {\n added: number;\n removed: number;\n modified: number;\n } {\n return {\n added: frameData.metadata?.linesAdded || 0,\n removed: frameData.metadata?.linesRemoved || 0,\n modified: frameData.metadata?.filesModified || 1,\n };\n }\n\n /**\n * Stop post-task hooks\n */\n async stop(): Promise<void> {\n this.isActive = false;\n this.removeAllListeners();\n console.log('\uD83D\uDED1 Post-task hooks stopped');\n }\n\n /**\n * Get current configuration\n */\n getConfig(): PostTaskConfig {\n return { ...this.config };\n }\n\n /**\n * Update configuration\n */\n updateConfig(updates: Partial<PostTaskConfig>): void {\n this.config = { ...this.config, ...updates };\n this.emit('config:updated', this.config);\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Post-Task Completion Hooks for StackMemory\n * Automatically runs tests and code review after Claude completes tasks\n */\n\nimport { EventEmitter } from 'events';\nimport { execSync } from 'child_process';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { FrameManager } from '../../core/frame/frame-manager';\nimport { DatabaseManager } from '../../core/storage/database-manager';\n\nexport interface PostTaskConfig {\n projectRoot: string;\n qualityGates: {\n runTests: boolean; // Auto-run tests after code changes\n requireTestCoverage: boolean; // Require test coverage for new code\n runCodeReview: boolean; // Auto-trigger code review agent\n runLinter: boolean; // Auto-run linter/formatter\n blockOnFailure: boolean; // Block further work if quality gates fail\n };\n testFrameworks: {\n detected: string[]; // Auto-detected test frameworks\n testCommand?: string; // Custom test command\n coverageCommand?: string; // Custom coverage command\n lintCommand?: string; // Custom lint command\n };\n reviewConfig: {\n reviewOnEveryChange: boolean; // Review every file change\n reviewOnTaskComplete: boolean; // Review when task frame closes\n focusAreas: string[]; // What to focus on in reviews\n skipPatterns: string[]; // Files/patterns to skip review\n };\n}\n\nexport interface TaskCompletionEvent {\n taskType: 'code_change' | 'task_complete' | 'file_modified' | 'frame_closed';\n files: string[];\n frameId: string;\n frameName: string;\n changes: {\n added: number;\n removed: number;\n modified: number;\n };\n metadata: Record<string, any>;\n}\n\nexport interface QualityGateResult {\n gate: string;\n passed: boolean;\n output: string;\n duration: number;\n issues?: QualityIssue[];\n}\n\nexport interface QualityIssue {\n type: 'test_failure' | 'lint_error' | 'coverage_low' | 'review_concern';\n file: string;\n line?: number;\n message: string;\n severity: 'error' | 'warning' | 'info';\n}\n\nexport class PostTaskHooks extends EventEmitter {\n private config: PostTaskConfig;\n private frameManager: FrameManager;\n private dbManager: DatabaseManager;\n private isActive: boolean = false;\n private lastProcessedFrame?: string;\n\n constructor(\n frameManager: FrameManager,\n dbManager: DatabaseManager,\n config: Partial<PostTaskConfig>\n ) {\n super();\n this.frameManager = frameManager;\n this.dbManager = dbManager;\n\n // Default configuration\n this.config = {\n projectRoot: process.cwd(),\n qualityGates: {\n runTests: true,\n requireTestCoverage: false,\n runCodeReview: true,\n runLinter: true,\n blockOnFailure: false,\n },\n testFrameworks: {\n detected: [],\n },\n reviewConfig: {\n reviewOnEveryChange: false,\n reviewOnTaskComplete: true,\n focusAreas: [\n 'security',\n 'performance',\n 'maintainability',\n 'correctness',\n ],\n skipPatterns: ['*.test.ts', '*.spec.js', 'dist/', 'node_modules/'],\n },\n ...config,\n };\n }\n\n /**\n * Initialize post-task hooks\n */\n async initialize(): Promise<void> {\n if (this.isActive) return;\n\n // Detect test frameworks and commands\n await this.detectTestFrameworks();\n\n // Set up frame event listeners\n this.setupFrameListeners();\n\n // Set up file watchers for code changes\n await this.setupFileWatchers();\n\n this.isActive = true;\n console.log('\u2705 Post-task hooks initialized');\n this.emit('initialized', this.config);\n }\n\n /**\n * Detect available test frameworks and commands\n */\n private async detectTestFrameworks(): Promise<void> {\n const packageJsonPath = path.join(this.config.projectRoot, 'package.json');\n\n try {\n const packageJson = JSON.parse(\n await fs.readFile(packageJsonPath, 'utf-8')\n );\n const scripts = packageJson.scripts || {};\n const dependencies = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n };\n\n // Detect test frameworks\n const frameworks: string[] = [];\n if (dependencies.jest) frameworks.push('jest');\n if (dependencies.vitest) frameworks.push('vitest');\n if (dependencies.mocha) frameworks.push('mocha');\n if (dependencies.playwright) frameworks.push('playwright');\n if (dependencies.cypress) frameworks.push('cypress');\n\n this.config.testFrameworks.detected = frameworks;\n\n // Detect commands\n if (scripts.test) {\n this.config.testFrameworks.testCommand = 'npm test';\n } else if (scripts['test:run']) {\n this.config.testFrameworks.testCommand = 'npm run test:run';\n }\n\n if (scripts.coverage) {\n this.config.testFrameworks.coverageCommand = 'npm run coverage';\n }\n\n if (scripts.lint) {\n this.config.testFrameworks.lintCommand = 'npm run lint';\n }\n } catch (error: unknown) {\n console.warn('Could not detect test frameworks:', error);\n }\n }\n\n /**\n * Set up frame event listeners\n */\n private setupFrameListeners(): void {\n // Listen for frame closures (task completion)\n this.frameManager.on(\n 'frame:closed',\n async (frameId: string, frameData: any) => {\n if (frameData.type === 'task' || frameData.type === 'subtask') {\n await this.handleTaskCompletion({\n taskType: 'task_complete',\n frameId,\n frameName: frameData.name || 'Unnamed task',\n files: this.extractFilesFromFrame(frameData),\n changes: this.calculateChanges(frameData),\n metadata: frameData.metadata || {},\n });\n }\n }\n );\n\n // Listen for significant events\n this.frameManager.on(\n 'frame:event',\n async (frameId: string, eventType: string, data: any) => {\n if (eventType === 'code_change' || eventType === 'file_modified') {\n await this.handleTaskCompletion({\n taskType: 'code_change',\n frameId,\n frameName: data.description || 'Code change',\n files: data.files || [],\n changes: data.changes || { added: 0, removed: 0, modified: 1 },\n metadata: data,\n });\n }\n }\n );\n }\n\n /**\n * Set up file watchers for real-time code change detection\n */\n private async setupFileWatchers(): Promise<void> {\n try {\n const chokidar = await import('chokidar');\n\n const watcher = chokidar.watch(\n [\n '**/*.{ts,js,tsx,jsx,py,go,rs,java,cpp,c}',\n '!node_modules/**',\n '!dist/**',\n '!build/**',\n ],\n {\n cwd: this.config.projectRoot,\n ignored: /node_modules/,\n persistent: true,\n }\n );\n\n let changeQueue: string[] = [];\n let debounceTimer: NodeJS.Timeout | null = null;\n\n watcher.on('change', (filePath: string) => {\n changeQueue.push(filePath);\n\n // Debounce to avoid too many triggers\n if (debounceTimer) clearTimeout(debounceTimer);\n debounceTimer = setTimeout(async () => {\n if (changeQueue.length > 0) {\n await this.handleFileChanges([...changeQueue]);\n changeQueue = [];\n }\n }, 2000); // 2 second debounce\n });\n } catch (error: unknown) {\n console.warn('File watching not available:', error);\n }\n }\n\n /**\n * Handle task completion events\n */\n private async handleTaskCompletion(\n event: TaskCompletionEvent\n ): Promise<void> {\n // Avoid duplicate processing\n if (\n this.lastProcessedFrame === event.frameId &&\n event.taskType !== 'code_change'\n ) {\n return;\n }\n\n this.lastProcessedFrame = event.frameId;\n\n console.log(\n `\uD83D\uDD0D Task completed: ${event.frameName} (${event.files.length} files changed)`\n );\n this.emit('task:completed', event);\n\n // Run quality gates\n const results = await this.runQualityGates(event);\n\n // Check if gates passed\n const allPassed = results.every((r) => r.passed);\n\n if (allPassed) {\n console.log('\u2705 All quality gates passed');\n this.emit('quality:passed', { event, results });\n } else {\n console.log('\u26A0\uFE0F Quality gate failures detected');\n this.emit('quality:failed', { event, results });\n\n if (this.config.qualityGates.blockOnFailure) {\n await this.blockFurtherWork(results);\n }\n }\n\n // Record results in frame metadata\n await this.recordQualityResults(event.frameId, results);\n }\n\n /**\n * Handle file changes\n */\n private async handleFileChanges(files: string[]): Promise<void> {\n if (!this.config.reviewConfig.reviewOnEveryChange) return;\n\n // Filter out files that should be skipped\n const filteredFiles = files.filter((file) => {\n return !this.config.reviewConfig.skipPatterns.some((pattern) => {\n return file.includes(pattern.replace('*', ''));\n });\n });\n\n if (filteredFiles.length === 0) return;\n\n await this.handleTaskCompletion({\n taskType: 'file_modified',\n frameId: 'file-watcher',\n frameName: 'File changes detected',\n files: filteredFiles,\n changes: { added: 0, removed: 0, modified: filteredFiles.length },\n metadata: { trigger: 'file_watcher' },\n });\n }\n\n /**\n * Run all configured quality gates\n */\n private async runQualityGates(\n event: TaskCompletionEvent\n ): Promise<QualityGateResult[]> {\n const results: QualityGateResult[] = [];\n\n // Run linter first (fastest)\n if (this.config.qualityGates.runLinter) {\n results.push(await this.runLinter(event.files));\n }\n\n // Run tests (slower)\n if (this.config.qualityGates.runTests) {\n results.push(await this.runTests(event.files));\n }\n\n // Check test coverage\n if (this.config.qualityGates.requireTestCoverage) {\n results.push(await this.checkTestCoverage(event.files));\n }\n\n // Run code review (slowest, most thorough)\n if (this.config.qualityGates.runCodeReview) {\n results.push(await this.runCodeReview(event));\n }\n\n return results;\n }\n\n /**\n * Run linter on changed files\n */\n private async runLinter(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.lintCommand) {\n return {\n gate: 'linter',\n passed: true,\n output: 'No lint command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.lintCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 30000, // 30 second timeout\n });\n\n return {\n gate: 'linter',\n passed: true,\n output,\n duration: Date.now() - start,\n };\n } catch (error: any) {\n return {\n gate: 'linter',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: this.parseLintErrors(error.stdout || error.message),\n };\n }\n }\n\n /**\n * Run tests\n */\n private async runTests(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.testCommand) {\n return {\n gate: 'tests',\n passed: true,\n output: 'No test command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.testCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 120000, // 2 minute timeout\n });\n\n return {\n gate: 'tests',\n passed: true,\n output,\n duration: Date.now() - start,\n };\n } catch (error: any) {\n return {\n gate: 'tests',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: this.parseTestFailures(error.stdout || error.message),\n };\n }\n }\n\n /**\n * Check test coverage\n */\n private async checkTestCoverage(files: string[]): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n if (!this.config.testFrameworks.coverageCommand) {\n return {\n gate: 'coverage',\n passed: true,\n output: 'No coverage command configured',\n duration: Date.now() - start,\n };\n }\n\n const output = execSync(this.config.testFrameworks.coverageCommand, {\n cwd: this.config.projectRoot,\n encoding: 'utf-8',\n timeout: 120000,\n });\n\n // Parse coverage percentage (simplified)\n const coverageMatch = output.match(/(\\d+\\.?\\d*)%/);\n const coverage = coverageMatch ? parseFloat(coverageMatch[1]) : 0;\n const threshold = 80; // 80% coverage threshold\n\n return {\n gate: 'coverage',\n passed: coverage >= threshold,\n output,\n duration: Date.now() - start,\n issues:\n coverage < threshold\n ? [\n {\n type: 'coverage_low',\n file: 'overall',\n message: `Coverage ${coverage}% is below threshold ${threshold}%`,\n severity: 'warning' as const,\n },\n ]\n : undefined,\n };\n } catch (error: any) {\n return {\n gate: 'coverage',\n passed: false,\n output: error.stdout || error.message,\n duration: Date.now() - start,\n issues: [\n {\n type: 'coverage_low',\n file: 'overall',\n message: 'Coverage check failed',\n severity: 'error' as const,\n },\n ],\n };\n }\n }\n\n /**\n * Run code review using AI agent\n */\n private async runCodeReview(\n event: TaskCompletionEvent\n ): Promise<QualityGateResult> {\n const start = Date.now();\n\n try {\n // This would integrate with the code review agent\n const reviewPrompt = this.generateCodeReviewPrompt(event);\n\n // For now, simulate a review (in real implementation, call agent)\n const review = await this.callCodeReviewAgent(reviewPrompt, event.files);\n\n return {\n gate: 'code_review',\n passed: !review.issues || review.issues.length === 0,\n output: review.summary,\n duration: Date.now() - start,\n issues: review.issues,\n };\n } catch (error: any) {\n return {\n gate: 'code_review',\n passed: false,\n output: `Code review failed: ${error.message}`,\n duration: Date.now() - start,\n };\n }\n }\n\n /**\n * Generate code review prompt\n */\n private generateCodeReviewPrompt(event: TaskCompletionEvent): string {\n return `\nPlease review the following code changes:\n\nTask: ${event.frameName}\nFiles changed: ${event.files.join(', ')}\nChanges: +${event.changes.added}, -${event.changes.removed}, ~${event.changes.modified}\n\nFocus areas: ${this.config.reviewConfig.focusAreas.join(', ')}\n\nPlease check for:\n1. Security vulnerabilities\n2. Performance issues \n3. Code maintainability\n4. Correctness and logic errors\n5. Best practices adherence\n\nProvide specific, actionable feedback.\n`;\n }\n\n /**\n * Call code review agent (placeholder for actual implementation)\n */\n private async callCodeReviewAgent(\n prompt: string,\n files: string[]\n ): Promise<{\n summary: string;\n issues?: QualityIssue[];\n }> {\n // This would integrate with Claude Code's agent system\n // For now, return a mock review\n return {\n summary: `Reviewed ${files.length} files. Code quality looks good.`,\n issues: [],\n };\n }\n\n /**\n * Parse lint errors into structured issues\n */\n private parseLintErrors(output: string): QualityIssue[] {\n const issues: QualityIssue[] = [];\n\n // Simple parser for common lint formats\n const lines = output.split('\\n');\n for (const line of lines) {\n const match = line.match(\n /^(.+?):(\\d+):(\\d+):\\s*(error|warning):\\s*(.+)$/\n );\n if (match) {\n issues.push({\n type: 'lint_error',\n file: match[1],\n line: parseInt(match[2]),\n message: match[5],\n severity: match[4] as 'error' | 'warning',\n });\n }\n }\n\n return issues;\n }\n\n /**\n * Parse test failures into structured issues\n */\n private parseTestFailures(output: string): QualityIssue[] {\n const issues: QualityIssue[] = [];\n\n // Simple parser for test failures\n const lines = output.split('\\n');\n for (const line of lines) {\n if (line.includes('FAIL') || line.includes('\u2717')) {\n issues.push({\n type: 'test_failure',\n file: 'unknown',\n message: line.trim(),\n severity: 'error',\n });\n }\n }\n\n return issues;\n }\n\n /**\n * Block further work when quality gates fail\n */\n private async blockFurtherWork(results: QualityGateResult[]): Promise<void> {\n const failedGates = results.filter((r) => !r.passed);\n\n console.log('\uD83D\uDEAB Quality gates failed - blocking further work:');\n failedGates.forEach((gate) => {\n console.log(` ${gate.gate}: ${gate.output}`);\n if (gate.issues) {\n gate.issues.forEach((issue) => {\n console.log(\n ` - ${issue.severity}: ${issue.message} (${issue.file}:${issue.line || 0})`\n );\n });\n }\n });\n\n console.log('\\n\uD83D\uDD27 Fix these issues before continuing:');\n const allIssues = failedGates.flatMap((g) => g.issues || []);\n allIssues.forEach((issue, i) => {\n console.log(`${i + 1}. ${issue.message}`);\n });\n }\n\n /**\n * Record quality results in frame metadata\n */\n private async recordQualityResults(\n frameId: string,\n results: QualityGateResult[]\n ): Promise<void> {\n try {\n const frame = await this.frameManager.getFrame(frameId);\n if (frame) {\n frame.metadata = {\n ...frame.metadata,\n qualityGates: {\n timestamp: new Date().toISOString(),\n results,\n passed: results.every((r) => r.passed),\n totalDuration: results.reduce((sum, r) => sum + r.duration, 0),\n },\n };\n\n // Update frame with quality results\n await this.frameManager.updateFrame(frameId, frame);\n }\n } catch (error: unknown) {\n console.error('Failed to record quality results:', error);\n }\n }\n\n /**\n * Extract files from frame data\n */\n private extractFilesFromFrame(frameData: any): string[] {\n // Extract file paths from various possible locations\n const files: string[] = [];\n\n if (frameData.metadata?.files) {\n files.push(...frameData.metadata.files);\n }\n\n if (frameData.events) {\n frameData.events.forEach((event: any) => {\n if (event.type === 'file_change' && event.data?.file) {\n files.push(event.data.file);\n }\n });\n }\n\n return [...new Set(files)]; // Remove duplicates\n }\n\n /**\n * Calculate changes from frame data\n */\n private calculateChanges(frameData: any): {\n added: number;\n removed: number;\n modified: number;\n } {\n return {\n added: frameData.metadata?.linesAdded || 0,\n removed: frameData.metadata?.linesRemoved || 0,\n modified: frameData.metadata?.filesModified || 1,\n };\n }\n\n /**\n * Stop post-task hooks\n */\n async stop(): Promise<void> {\n this.isActive = false;\n this.removeAllListeners();\n console.log('\uD83D\uDED1 Post-task hooks stopped');\n }\n\n /**\n * Get current configuration\n */\n getConfig(): PostTaskConfig {\n return { ...this.config };\n }\n\n /**\n * Update configuration\n */\n updateConfig(updates: Partial<PostTaskConfig>): void {\n this.config = { ...this.config, ...updates };\n this.emit('config:updated', this.config);\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -2,6 +2,17 @@ import { createHash, randomBytes } from "crypto";
2
2
  import { readFileSync, writeFileSync, existsSync } from "fs";
3
3
  import { join } from "path";
4
4
  import { logger } from "../../core/monitoring/logger.js";
5
+ function getEnv(key, defaultValue) {
6
+ const value = process.env[key];
7
+ if (value === void 0) {
8
+ if (defaultValue !== void 0) return defaultValue;
9
+ throw new Error(`Environment variable ${key} is required`);
10
+ }
11
+ return value;
12
+ }
13
+ function getOptionalEnv(key) {
14
+ return process.env[key];
15
+ }
5
16
  class LinearAuthManager {
6
17
  configPath;
7
18
  tokensPath;
@@ -214,9 +225,9 @@ class LinearOAuthSetup {
214
225
  */
215
226
  async setupInteractive() {
216
227
  const config = {
217
- clientId: process.env.LINEAR_CLIENT_ID || "",
218
- clientSecret: process.env.LINEAR_CLIENT_SECRET || "",
219
- redirectUri: process.env.LINEAR_REDIRECT_URI || "http://localhost:3456/auth/linear/callback",
228
+ clientId: process.env["LINEAR_CLIENT_ID"] || "",
229
+ clientSecret: process.env["LINEAR_CLIENT_SECRET"] || "",
230
+ redirectUri: process.env["LINEAR_REDIRECT_URI"] || "http://localhost:3456/auth/linear/callback",
220
231
  scopes: DEFAULT_LINEAR_SCOPES
221
232
  };
222
233
  if (!config.clientId || !config.clientSecret) {
@@ -235,7 +246,7 @@ class LinearOAuthSetup {
235
246
  }
236
247
  this.authManager.saveConfig(config);
237
248
  const { url, codeVerifier } = this.authManager.generateAuthUrl();
238
- process.env._LINEAR_CODE_VERIFIER = codeVerifier;
249
+ process.env["_LINEAR_CODE_VERIFIER"] = codeVerifier;
239
250
  return {
240
251
  authUrl: url,
241
252
  instructions: [
@@ -253,14 +264,14 @@ class LinearOAuthSetup {
253
264
  */
254
265
  async completeAuth(authCode) {
255
266
  try {
256
- const codeVerifier = process.env._LINEAR_CODE_VERIFIER;
267
+ const codeVerifier = process.env["_LINEAR_CODE_VERIFIER"];
257
268
  if (!codeVerifier) {
258
269
  throw new Error(
259
270
  "Code verifier not found. Please restart the setup process."
260
271
  );
261
272
  }
262
273
  await this.authManager.exchangeCodeForToken(authCode, codeVerifier);
263
- delete process.env._LINEAR_CODE_VERIFIER;
274
+ delete process.env["_LINEAR_CODE_VERIFIER"];
264
275
  logger.info("Linear OAuth setup completed successfully!");
265
276
  return true;
266
277
  } catch (error) {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/integrations/linear/auth.ts"],
4
- "sourcesContent": ["/**\n * Linear OAuth Authentication Setup\n * Handles initial OAuth flow and token management for Linear integration\n */\n\nimport { createHash, randomBytes } from 'crypto';\nimport { readFileSync, writeFileSync, existsSync } from 'fs';\nimport { join } from 'path';\nimport { logger } from '../../core/monitoring/logger.js';\n\nexport interface LinearAuthConfig {\n clientId: string;\n clientSecret: string;\n redirectUri: string;\n scopes: string[];\n}\n\nexport interface LinearTokens {\n accessToken: string;\n refreshToken?: string;\n expiresAt: number; // Unix timestamp\n scope: string[];\n}\n\nexport interface LinearAuthResult {\n accessToken: string;\n refreshToken?: string;\n expiresIn: number;\n scope: string;\n tokenType: string;\n}\n\nexport class LinearAuthManager {\n private configPath: string;\n private tokensPath: string;\n private config?: LinearAuthConfig;\n\n constructor(projectRoot: string) {\n const configDir = join(projectRoot, '.stackmemory');\n this.configPath = join(configDir, 'linear-config.json');\n this.tokensPath = join(configDir, 'linear-tokens.json');\n }\n\n /**\n * Check if Linear integration is configured\n */\n isConfigured(): boolean {\n return existsSync(this.configPath) && existsSync(this.tokensPath);\n }\n\n /**\n * Save OAuth application configuration\n */\n saveConfig(config: LinearAuthConfig): void {\n writeFileSync(this.configPath, JSON.stringify(config, null, 2));\n this.config = config;\n logger.info('Linear OAuth configuration saved');\n }\n\n /**\n * Load OAuth configuration\n */\n loadConfig(): LinearAuthConfig | null {\n if (!existsSync(this.configPath)) {\n return null;\n }\n\n try {\n const configData = readFileSync(this.configPath, 'utf8');\n this.config = JSON.parse(configData);\n return this.config!;\n } catch (error) {\n logger.error('Failed to load Linear configuration:', error as Error);\n return null;\n }\n }\n\n /**\n * Generate OAuth authorization URL with PKCE\n */\n generateAuthUrl(state?: string): { url: string; codeVerifier: string } {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n // Generate PKCE parameters\n const codeVerifier = randomBytes(32).toString('base64url');\n const codeChallenge = createHash('sha256')\n .update(codeVerifier)\n .digest('base64url');\n\n const params = new URLSearchParams({\n client_id: this.config.clientId,\n redirect_uri: this.config.redirectUri,\n response_type: 'code',\n scope: this.config.scopes.join(' '),\n code_challenge: codeChallenge,\n code_challenge_method: 'S256',\n actor: 'app', // Enable actor authorization for service accounts\n });\n\n if (state) {\n params.set('state', state);\n }\n\n const authUrl = `https://linear.app/oauth/authorize?${params.toString()}`;\n\n return { url: authUrl, codeVerifier };\n }\n\n /**\n * Exchange authorization code for access token\n */\n async exchangeCodeForToken(\n authCode: string,\n codeVerifier: string\n ): Promise<LinearTokens> {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n const tokenUrl = 'https://api.linear.app/oauth/token';\n\n const body = new URLSearchParams({\n grant_type: 'authorization_code',\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n redirect_uri: this.config.redirectUri,\n code: authCode,\n code_verifier: codeVerifier,\n });\n\n const response = await fetch(tokenUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded',\n Accept: 'application/json',\n },\n body: body.toString(),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Token exchange failed: ${response.status} ${errorText}`);\n }\n\n const result = (await response.json()) as LinearAuthResult;\n\n // Calculate expiration time (tokens expire in 24 hours)\n const expiresAt = Date.now() + result.expiresIn * 1000;\n\n const tokens: LinearTokens = {\n accessToken: result.accessToken,\n refreshToken: result.refreshToken,\n expiresAt,\n scope: result.scope.split(' '),\n };\n\n this.saveTokens(tokens);\n return tokens;\n }\n\n /**\n * Refresh access token using refresh token\n */\n async refreshAccessToken(): Promise<LinearTokens> {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n const currentTokens = this.loadTokens();\n if (!currentTokens?.refreshToken) {\n throw new Error('No refresh token available');\n }\n\n const tokenUrl = 'https://api.linear.app/oauth/token';\n\n const body = new URLSearchParams({\n grant_type: 'refresh_token',\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: currentTokens.refreshToken,\n });\n\n const response = await fetch(tokenUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded',\n Accept: 'application/json',\n },\n body: body.toString(),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Token refresh failed: ${response.status} ${errorText}`);\n }\n\n const result = (await response.json()) as LinearAuthResult;\n\n const tokens: LinearTokens = {\n accessToken: result.accessToken,\n refreshToken: result.refreshToken || currentTokens.refreshToken,\n expiresAt: Date.now() + result.expiresIn * 1000,\n scope: result.scope.split(' '),\n };\n\n this.saveTokens(tokens);\n return tokens;\n }\n\n /**\n * Get valid access token (refresh if needed)\n */\n async getValidToken(): Promise<string> {\n const tokens = this.loadTokens();\n if (!tokens) {\n throw new Error('No Linear tokens found. Please complete OAuth setup.');\n }\n\n // Check if token expires in next 5 minutes\n const fiveMinutes = 5 * 60 * 1000;\n if (tokens.expiresAt - Date.now() < fiveMinutes) {\n logger.info('Linear token expiring soon, refreshing...');\n const newTokens = await this.refreshAccessToken();\n return newTokens.accessToken;\n }\n\n return tokens.accessToken;\n }\n\n /**\n * Save tokens to file\n */\n private saveTokens(tokens: LinearTokens): void {\n writeFileSync(this.tokensPath, JSON.stringify(tokens, null, 2));\n logger.info('Linear tokens saved');\n }\n\n /**\n * Load tokens from file\n */\n loadTokens(): LinearTokens | null {\n if (!existsSync(this.tokensPath)) {\n return null;\n }\n\n try {\n const tokensData = readFileSync(this.tokensPath, 'utf8');\n return JSON.parse(tokensData);\n } catch (error) {\n logger.error('Failed to load Linear tokens:', error as Error);\n return null;\n }\n }\n\n /**\n * Clear stored tokens and config\n */\n clearAuth(): void {\n if (existsSync(this.tokensPath)) {\n writeFileSync(this.tokensPath, '');\n }\n if (existsSync(this.configPath)) {\n writeFileSync(this.configPath, '');\n }\n logger.info('Linear authentication cleared');\n }\n}\n\n/**\n * Default Linear OAuth scopes for task management\n */\nexport const DEFAULT_LINEAR_SCOPES = [\n 'read', // Read issues, projects, teams\n 'write', // Create and update issues\n 'admin', // Manage team settings and workflows\n];\n\n/**\n * Linear OAuth setup helper\n */\nexport class LinearOAuthSetup {\n private authManager: LinearAuthManager;\n\n constructor(projectRoot: string) {\n this.authManager = new LinearAuthManager(projectRoot);\n }\n\n /**\n * Interactive setup for Linear OAuth\n */\n async setupInteractive(): Promise<{\n authUrl: string;\n instructions: string[];\n }> {\n // For now, we'll provide manual setup instructions\n // In a full implementation, this could open a browser or use a local server\n\n const config: LinearAuthConfig = {\n clientId: process.env.LINEAR_CLIENT_ID || '',\n clientSecret: process.env.LINEAR_CLIENT_SECRET || '',\n redirectUri:\n process.env.LINEAR_REDIRECT_URI ||\n 'http://localhost:3456/auth/linear/callback',\n scopes: DEFAULT_LINEAR_SCOPES,\n };\n\n if (!config.clientId || !config.clientSecret) {\n return {\n authUrl: '',\n instructions: [\n '1. Create a Linear OAuth application at https://linear.app/settings/api',\n '2. Set redirect URI to: http://localhost:3456/auth/linear/callback',\n '3. Copy your Client ID and Client Secret',\n '4. Set environment variables:',\n ' export LINEAR_CLIENT_ID=\"your_client_id\"',\n ' export LINEAR_CLIENT_SECRET=\"your_client_secret\"',\n '5. Re-run this setup command',\n ],\n };\n }\n\n this.authManager.saveConfig(config);\n\n const { url, codeVerifier } = this.authManager.generateAuthUrl();\n\n // Store code verifier temporarily (in a real app, this would be in a secure session store)\n process.env._LINEAR_CODE_VERIFIER = codeVerifier;\n\n return {\n authUrl: url,\n instructions: [\n '1. Open this URL in your browser:',\n url,\n '',\n '2. Approve the StackMemory integration',\n '3. Copy the authorization code from the redirect URL',\n '4. Run: stackmemory linear authorize <code>',\n ],\n };\n }\n\n /**\n * Complete OAuth flow with authorization code\n */\n async completeAuth(authCode: string): Promise<boolean> {\n try {\n const codeVerifier = process.env._LINEAR_CODE_VERIFIER;\n if (!codeVerifier) {\n throw new Error(\n 'Code verifier not found. Please restart the setup process.'\n );\n }\n\n await this.authManager.exchangeCodeForToken(authCode, codeVerifier);\n delete process.env._LINEAR_CODE_VERIFIER;\n\n logger.info('Linear OAuth setup completed successfully!');\n return true;\n } catch (error) {\n logger.error('Failed to complete Linear OAuth setup:', error as Error);\n return false;\n }\n }\n\n /**\n * Test the Linear connection\n */\n async testConnection(): Promise<boolean> {\n try {\n const token = await this.authManager.getValidToken();\n\n // Test with a simple GraphQL query\n const response = await fetch('https://api.linear.app/graphql', {\n method: 'POST',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n query: 'query { viewer { id name email } }',\n }),\n });\n\n if (response.ok) {\n const result = (await response.json()) as {\n data?: { viewer?: { id: string; name: string; email: string } };\n };\n if (result.data?.viewer) {\n logger.info(\n `Connected to Linear as: ${result.data.viewer.name} (${result.data.viewer.email})`\n );\n return true;\n }\n }\n\n return false;\n } catch (error) {\n logger.error('Linear connection test failed:', error as Error);\n return false;\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Linear OAuth Authentication Setup\n * Handles initial OAuth flow and token management for Linear integration\n */\n\nimport { createHash, randomBytes } from 'crypto';\nimport { readFileSync, writeFileSync, existsSync } from 'fs';\nimport { join } from 'path';\nimport { logger } from '../../core/monitoring/logger.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 LinearAuthConfig {\n clientId: string;\n clientSecret: string;\n redirectUri: string;\n scopes: string[];\n}\n\nexport interface LinearTokens {\n accessToken: string;\n refreshToken?: string;\n expiresAt: number; // Unix timestamp\n scope: string[];\n}\n\nexport interface LinearAuthResult {\n accessToken: string;\n refreshToken?: string;\n expiresIn: number;\n scope: string;\n tokenType: string;\n}\n\nexport class LinearAuthManager {\n private configPath: string;\n private tokensPath: string;\n private config?: LinearAuthConfig;\n\n constructor(projectRoot: string) {\n const configDir = join(projectRoot, '.stackmemory');\n this.configPath = join(configDir, 'linear-config.json');\n this.tokensPath = join(configDir, 'linear-tokens.json');\n }\n\n /**\n * Check if Linear integration is configured\n */\n isConfigured(): boolean {\n return existsSync(this.configPath) && existsSync(this.tokensPath);\n }\n\n /**\n * Save OAuth application configuration\n */\n saveConfig(config: LinearAuthConfig): void {\n writeFileSync(this.configPath, JSON.stringify(config, null, 2));\n this.config = config;\n logger.info('Linear OAuth configuration saved');\n }\n\n /**\n * Load OAuth configuration\n */\n loadConfig(): LinearAuthConfig | null {\n if (!existsSync(this.configPath)) {\n return null;\n }\n\n try {\n const configData = readFileSync(this.configPath, 'utf8');\n this.config = JSON.parse(configData);\n return this.config!;\n } catch (error: unknown) {\n logger.error('Failed to load Linear configuration:', error as Error);\n return null;\n }\n }\n\n /**\n * Generate OAuth authorization URL with PKCE\n */\n generateAuthUrl(state?: string): { url: string; codeVerifier: string } {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n // Generate PKCE parameters\n const codeVerifier = randomBytes(32).toString('base64url');\n const codeChallenge = createHash('sha256')\n .update(codeVerifier)\n .digest('base64url');\n\n const params = new URLSearchParams({\n client_id: this.config.clientId,\n redirect_uri: this.config.redirectUri,\n response_type: 'code',\n scope: this.config.scopes.join(' '),\n code_challenge: codeChallenge,\n code_challenge_method: 'S256',\n actor: 'app', // Enable actor authorization for service accounts\n });\n\n if (state) {\n params.set('state', state);\n }\n\n const authUrl = `https://linear.app/oauth/authorize?${params.toString()}`;\n\n return { url: authUrl, codeVerifier };\n }\n\n /**\n * Exchange authorization code for access token\n */\n async exchangeCodeForToken(\n authCode: string,\n codeVerifier: string\n ): Promise<LinearTokens> {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n const tokenUrl = 'https://api.linear.app/oauth/token';\n\n const body = new URLSearchParams({\n grant_type: 'authorization_code',\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n redirect_uri: this.config.redirectUri,\n code: authCode,\n code_verifier: codeVerifier,\n });\n\n const response = await fetch(tokenUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded',\n Accept: 'application/json',\n },\n body: body.toString(),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Token exchange failed: ${response.status} ${errorText}`);\n }\n\n const result = (await response.json()) as LinearAuthResult;\n\n // Calculate expiration time (tokens expire in 24 hours)\n const expiresAt = Date.now() + result.expiresIn * 1000;\n\n const tokens: LinearTokens = {\n accessToken: result.accessToken,\n refreshToken: result.refreshToken,\n expiresAt,\n scope: result.scope.split(' '),\n };\n\n this.saveTokens(tokens);\n return tokens;\n }\n\n /**\n * Refresh access token using refresh token\n */\n async refreshAccessToken(): Promise<LinearTokens> {\n if (!this.config) {\n throw new Error('Linear OAuth configuration not loaded');\n }\n\n const currentTokens = this.loadTokens();\n if (!currentTokens?.refreshToken) {\n throw new Error('No refresh token available');\n }\n\n const tokenUrl = 'https://api.linear.app/oauth/token';\n\n const body = new URLSearchParams({\n grant_type: 'refresh_token',\n client_id: this.config.clientId,\n client_secret: this.config.clientSecret,\n refresh_token: currentTokens.refreshToken,\n });\n\n const response = await fetch(tokenUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded',\n Accept: 'application/json',\n },\n body: body.toString(),\n });\n\n if (!response.ok) {\n const errorText = await response.text();\n throw new Error(`Token refresh failed: ${response.status} ${errorText}`);\n }\n\n const result = (await response.json()) as LinearAuthResult;\n\n const tokens: LinearTokens = {\n accessToken: result.accessToken,\n refreshToken: result.refreshToken || currentTokens.refreshToken,\n expiresAt: Date.now() + result.expiresIn * 1000,\n scope: result.scope.split(' '),\n };\n\n this.saveTokens(tokens);\n return tokens;\n }\n\n /**\n * Get valid access token (refresh if needed)\n */\n async getValidToken(): Promise<string> {\n const tokens = this.loadTokens();\n if (!tokens) {\n throw new Error('No Linear tokens found. Please complete OAuth setup.');\n }\n\n // Check if token expires in next 5 minutes\n const fiveMinutes = 5 * 60 * 1000;\n if (tokens.expiresAt - Date.now() < fiveMinutes) {\n logger.info('Linear token expiring soon, refreshing...');\n const newTokens = await this.refreshAccessToken();\n return newTokens.accessToken;\n }\n\n return tokens.accessToken;\n }\n\n /**\n * Save tokens to file\n */\n private saveTokens(tokens: LinearTokens): void {\n writeFileSync(this.tokensPath, JSON.stringify(tokens, null, 2));\n logger.info('Linear tokens saved');\n }\n\n /**\n * Load tokens from file\n */\n loadTokens(): LinearTokens | null {\n if (!existsSync(this.tokensPath)) {\n return null;\n }\n\n try {\n const tokensData = readFileSync(this.tokensPath, 'utf8');\n return JSON.parse(tokensData);\n } catch (error: unknown) {\n logger.error('Failed to load Linear tokens:', error as Error);\n return null;\n }\n }\n\n /**\n * Clear stored tokens and config\n */\n clearAuth(): void {\n if (existsSync(this.tokensPath)) {\n writeFileSync(this.tokensPath, '');\n }\n if (existsSync(this.configPath)) {\n writeFileSync(this.configPath, '');\n }\n logger.info('Linear authentication cleared');\n }\n}\n\n/**\n * Default Linear OAuth scopes for task management\n */\nexport const DEFAULT_LINEAR_SCOPES = [\n 'read', // Read issues, projects, teams\n 'write', // Create and update issues\n 'admin', // Manage team settings and workflows\n];\n\n/**\n * Linear OAuth setup helper\n */\nexport class LinearOAuthSetup {\n private authManager: LinearAuthManager;\n\n constructor(projectRoot: string) {\n this.authManager = new LinearAuthManager(projectRoot);\n }\n\n /**\n * Interactive setup for Linear OAuth\n */\n async setupInteractive(): Promise<{\n authUrl: string;\n instructions: string[];\n }> {\n // For now, we'll provide manual setup instructions\n // In a full implementation, this could open a browser or use a local server\n\n const config: LinearAuthConfig = {\n clientId: process.env['LINEAR_CLIENT_ID'] || '',\n clientSecret: process.env['LINEAR_CLIENT_SECRET'] || '',\n redirectUri:\n process.env['LINEAR_REDIRECT_URI'] ||\n 'http://localhost:3456/auth/linear/callback',\n scopes: DEFAULT_LINEAR_SCOPES,\n };\n\n if (!config.clientId || !config.clientSecret) {\n return {\n authUrl: '',\n instructions: [\n '1. Create a Linear OAuth application at https://linear.app/settings/api',\n '2. Set redirect URI to: http://localhost:3456/auth/linear/callback',\n '3. Copy your Client ID and Client Secret',\n '4. Set environment variables:',\n ' export LINEAR_CLIENT_ID=\"your_client_id\"',\n ' export LINEAR_CLIENT_SECRET=\"your_client_secret\"',\n '5. Re-run this setup command',\n ],\n };\n }\n\n this.authManager.saveConfig(config);\n\n const { url, codeVerifier } = this.authManager.generateAuthUrl();\n\n // Store code verifier temporarily (in a real app, this would be in a secure session store)\n process.env['_LINEAR_CODE_VERIFIER'] = codeVerifier;\n\n return {\n authUrl: url,\n instructions: [\n '1. Open this URL in your browser:',\n url,\n '',\n '2. Approve the StackMemory integration',\n '3. Copy the authorization code from the redirect URL',\n '4. Run: stackmemory linear authorize <code>',\n ],\n };\n }\n\n /**\n * Complete OAuth flow with authorization code\n */\n async completeAuth(authCode: string): Promise<boolean> {\n try {\n const codeVerifier = process.env['_LINEAR_CODE_VERIFIER'];\n if (!codeVerifier) {\n throw new Error(\n 'Code verifier not found. Please restart the setup process.'\n );\n }\n\n await this.authManager.exchangeCodeForToken(authCode, codeVerifier);\n delete process.env['_LINEAR_CODE_VERIFIER'];\n\n logger.info('Linear OAuth setup completed successfully!');\n return true;\n } catch (error: unknown) {\n logger.error('Failed to complete Linear OAuth setup:', error as Error);\n return false;\n }\n }\n\n /**\n * Test the Linear connection\n */\n async testConnection(): Promise<boolean> {\n try {\n const token = await this.authManager.getValidToken();\n\n // Test with a simple GraphQL query\n const response = await fetch('https://api.linear.app/graphql', {\n method: 'POST',\n headers: {\n Authorization: `Bearer ${token}`,\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n query: 'query { viewer { id name email } }',\n }),\n });\n\n if (response.ok) {\n const result = (await response.json()) as {\n data?: { viewer?: { id: string; name: string; email: string } };\n };\n if (result.data?.viewer) {\n logger.info(\n `Connected to Linear as: ${result.data.viewer.name} (${result.data.viewer.email})`\n );\n return true;\n }\n }\n\n return false;\n } catch (error: unknown) {\n logger.error('Linear connection test failed:', error as Error);\n return false;\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }