@stackmemoryai/stackmemory 0.3.17 → 0.3.19

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 (234) hide show
  1. package/dist/cli/claude-sm.js +51 -5
  2. package/dist/cli/claude-sm.js.map +2 -2
  3. package/dist/cli/codex-sm.js +52 -19
  4. package/dist/cli/codex-sm.js.map +2 -2
  5. package/dist/cli/commands/db.js +143 -0
  6. package/dist/cli/commands/db.js.map +7 -0
  7. package/dist/cli/commands/login.js +50 -0
  8. package/dist/cli/commands/login.js.map +7 -0
  9. package/dist/cli/commands/migrate.js +178 -0
  10. package/dist/cli/commands/migrate.js.map +7 -0
  11. package/dist/cli/commands/onboard.js +158 -2
  12. package/dist/cli/commands/onboard.js.map +2 -2
  13. package/dist/cli/commands/skills.js +15 -2
  14. package/dist/cli/commands/skills.js.map +2 -2
  15. package/dist/cli/index.js +118 -834
  16. package/dist/cli/index.js.map +3 -3
  17. package/dist/core/context/dual-stack-manager.js +1 -1
  18. package/dist/core/context/dual-stack-manager.js.map +1 -1
  19. package/dist/core/context/frame-database.js +1 -0
  20. package/dist/core/context/frame-database.js.map +2 -2
  21. package/dist/core/context/frame-manager.js +59 -2
  22. package/dist/core/context/frame-manager.js.map +2 -2
  23. package/dist/core/database/database-adapter.js +6 -1
  24. package/dist/core/database/database-adapter.js.map +2 -2
  25. package/dist/core/database/sqlite-adapter.js +60 -2
  26. package/dist/core/database/sqlite-adapter.js.map +2 -2
  27. package/dist/integrations/claude-code/subagent-client.js +106 -3
  28. package/dist/integrations/claude-code/subagent-client.js.map +2 -2
  29. package/dist/servers/railway/config.js +51 -0
  30. package/dist/servers/railway/config.js.map +7 -0
  31. package/dist/servers/railway/index-enhanced.js +156 -0
  32. package/dist/servers/railway/index-enhanced.js.map +7 -0
  33. package/dist/servers/railway/index.js +843 -82
  34. package/dist/servers/railway/index.js.map +3 -3
  35. package/dist/servers/railway/minimal.js +48 -3
  36. package/dist/servers/railway/minimal.js.map +2 -2
  37. package/dist/servers/railway/storage-test.js +455 -0
  38. package/dist/servers/railway/storage-test.js.map +7 -0
  39. package/dist/skills/claude-skills.js +13 -12
  40. package/dist/skills/claude-skills.js.map +2 -2
  41. package/dist/skills/recursive-agent-orchestrator.js +27 -18
  42. package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
  43. package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
  44. package/package.json +13 -21
  45. package/scripts/README-TESTING.md +186 -0
  46. package/scripts/analyze-cli-security.js +288 -0
  47. package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
  48. package/scripts/archive/analyze-linear-duplicates.js +214 -0
  49. package/scripts/archive/analyze-remaining-duplicates.js +230 -0
  50. package/scripts/archive/analyze-sta-duplicates.js +292 -0
  51. package/scripts/archive/analyze-sta-graphql.js +399 -0
  52. package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
  53. package/scripts/archive/check-all-duplicates.ts +419 -0
  54. package/scripts/archive/clean-duplicate-tasks.js +114 -0
  55. package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
  56. package/scripts/archive/create-phase-tasks.js +387 -0
  57. package/scripts/archive/delete-linear-duplicates.js +182 -0
  58. package/scripts/archive/delete-remaining-duplicates.js +158 -0
  59. package/scripts/archive/delete-sta-duplicates.js +201 -0
  60. package/scripts/archive/delete-sta-oauth.js +201 -0
  61. package/scripts/archive/export-sta-tasks.js +62 -0
  62. package/scripts/archive/install-auto-sync.js +266 -0
  63. package/scripts/archive/install-chromadb-hooks.sh +133 -0
  64. package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
  65. package/scripts/archive/install-post-task-hooks.sh +289 -0
  66. package/scripts/archive/install-stackmemory-hooks.sh +420 -0
  67. package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
  68. package/scripts/archive/merge-linear-duplicates.ts +180 -0
  69. package/scripts/archive/remove-sta-tasks.js +70 -0
  70. package/scripts/archive/setup-background-sync.sh +168 -0
  71. package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
  72. package/scripts/archive/setup-claude-autostart.sh +305 -0
  73. package/scripts/archive/setup-git-hooks.sh +25 -0
  74. package/scripts/archive/setup-linear-oauth.sh +46 -0
  75. package/scripts/archive/setup-mcp.sh +113 -0
  76. package/scripts/archive/setup-railway-deployment.sh +81 -0
  77. package/scripts/auto-handoff.sh +262 -0
  78. package/scripts/background-sync-manager.js +416 -0
  79. package/scripts/benchmark-performance.ts +57 -0
  80. package/scripts/check-redis.ts +48 -0
  81. package/scripts/chromadb-auto-loader.sh +128 -0
  82. package/scripts/chromadb-context-loader.js +479 -0
  83. package/scripts/claude-chromadb-hook.js +460 -0
  84. package/scripts/claude-code-wrapper.sh +66 -0
  85. package/scripts/claude-linear-skill.js +455 -0
  86. package/scripts/claude-pre-commit.sh +302 -0
  87. package/scripts/claude-sm-autostart.js +532 -0
  88. package/scripts/claude-sm-setup.sh +367 -0
  89. package/scripts/claude-with-chromadb.sh +69 -0
  90. package/scripts/claude-worktree-manager.sh +323 -0
  91. package/scripts/claude-worktree-monitor.sh +371 -0
  92. package/scripts/claude-worktree-setup.sh +327 -0
  93. package/scripts/clean-linear-backlog.js +273 -0
  94. package/scripts/cleanup-old-sessions.sh +57 -0
  95. package/scripts/codex-wrapper.sh +88 -0
  96. package/scripts/create-sandbox.sh +269 -0
  97. package/scripts/debug-linear-update.js +174 -0
  98. package/scripts/delete-linear-tasks.js +167 -0
  99. package/scripts/deploy.sh +89 -0
  100. package/scripts/deployment/railway.sh +352 -0
  101. package/scripts/deployment/test-deployment.js +194 -0
  102. package/scripts/detect-and-rehydrate.js +162 -0
  103. package/scripts/detect-and-rehydrate.mjs +165 -0
  104. package/scripts/development/create-demo-tasks.js +143 -0
  105. package/scripts/development/debug-frame-test.js +16 -0
  106. package/scripts/development/demo-auto-sync.js +128 -0
  107. package/scripts/development/fix-all-imports.js +213 -0
  108. package/scripts/development/fix-imports.js +229 -0
  109. package/scripts/development/fix-lint-loop.cjs +103 -0
  110. package/scripts/development/fix-project-id.ts +161 -0
  111. package/scripts/development/fix-strict-mode-issues.ts +291 -0
  112. package/scripts/development/reorganize-structure.sh +228 -0
  113. package/scripts/development/test-persistence-direct.js +148 -0
  114. package/scripts/development/test-persistence.js +114 -0
  115. package/scripts/development/test-tasks.js +93 -0
  116. package/scripts/development/update-imports.js +212 -0
  117. package/scripts/fetch-linear-status.js +125 -0
  118. package/scripts/git-hooks/README.md +310 -0
  119. package/scripts/git-hooks/branch-context-manager.sh +342 -0
  120. package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
  121. package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
  122. package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
  123. package/scripts/hooks/cleanup-shell.sh +130 -0
  124. package/scripts/hooks/task-complete.sh +114 -0
  125. package/scripts/initialize.ts +129 -0
  126. package/scripts/install-claude-hooks-auto.js +104 -0
  127. package/scripts/install-claude-hooks.sh +133 -0
  128. package/scripts/install-global.sh +296 -0
  129. package/scripts/install.sh +235 -0
  130. package/scripts/linear-auto-sync.js +262 -0
  131. package/scripts/linear-auto-sync.sh +161 -0
  132. package/scripts/linear-sync-daemon.js +150 -0
  133. package/scripts/linear-task-review.js +237 -0
  134. package/scripts/list-linear-tasks.ts +178 -0
  135. package/scripts/mcp-proxy.js +66 -0
  136. package/scripts/opencode-wrapper.sh +85 -0
  137. package/scripts/publish-local.js +74 -0
  138. package/scripts/query-chromadb.ts +201 -0
  139. package/scripts/railway-env-setup.sh +39 -0
  140. package/scripts/reconcile-local-tasks.js +170 -0
  141. package/scripts/recreate-frames-db.js +89 -0
  142. package/scripts/setup/claude-integration.js +138 -0
  143. package/scripts/setup/configure-alias.js +125 -0
  144. package/scripts/setup/configure-codex-alias.js +161 -0
  145. package/scripts/setup/configure-opencode-alias.js +175 -0
  146. package/scripts/setup-claude-integration.js +204 -0
  147. package/scripts/setup-claude-integration.sh +183 -0
  148. package/scripts/setup-railway-deployment.sh +37 -0
  149. package/scripts/setup.sh +31 -0
  150. package/scripts/show-linear-summary.ts +172 -0
  151. package/scripts/stackmemory-auto-handoff.sh +231 -0
  152. package/scripts/stackmemory-daemon.sh +40 -0
  153. package/scripts/start-linear-sync-daemon.sh +141 -0
  154. package/scripts/start-temporal-paradox.sh +214 -0
  155. package/scripts/status.ts +159 -0
  156. package/scripts/sync-and-clean-tasks.js +258 -0
  157. package/scripts/sync-frames-from-railway.js +228 -0
  158. package/scripts/sync-linear-graphql.js +303 -0
  159. package/scripts/sync-linear-tasks.js +186 -0
  160. package/scripts/test-auto-triggers.sh +57 -0
  161. package/scripts/test-browser-mcp.js +74 -0
  162. package/scripts/test-chromadb-full.js +115 -0
  163. package/scripts/test-chromadb-hooks.sh +28 -0
  164. package/scripts/test-chromadb-sync.ts +245 -0
  165. package/scripts/test-cli-security.js +293 -0
  166. package/scripts/test-hooks-persistence.sh +220 -0
  167. package/scripts/test-installation-scenarios.sh +359 -0
  168. package/scripts/test-installation.sh +224 -0
  169. package/scripts/test-mcp.js +163 -0
  170. package/scripts/test-pre-publish-quick.sh +75 -0
  171. package/scripts/test-quality-gates.sh +263 -0
  172. package/scripts/test-railway-db.js +222 -0
  173. package/scripts/test-redis-storage.ts +490 -0
  174. package/scripts/test-rlm-basic.sh +122 -0
  175. package/scripts/test-rlm-comprehensive.sh +260 -0
  176. package/scripts/test-rlm-e2e.sh +268 -0
  177. package/scripts/test-rlm-simple.js +90 -0
  178. package/scripts/test-rlm.js +110 -0
  179. package/scripts/test-session-handoff.sh +165 -0
  180. package/scripts/test-shell-integration.sh +275 -0
  181. package/scripts/testing/ab-test-runner.ts +508 -0
  182. package/scripts/testing/collect-metrics.ts +457 -0
  183. package/scripts/testing/quick-effectiveness-demo.js +187 -0
  184. package/scripts/testing/real-performance-test.js +422 -0
  185. package/scripts/testing/run-effectiveness-tests.sh +176 -0
  186. package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
  187. package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
  188. package/scripts/testing/simple-effectiveness-test.js +310 -0
  189. package/scripts/testing/src/core/context/context-bridge.js +253 -0
  190. package/scripts/testing/src/core/context/frame-manager.js +746 -0
  191. package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
  192. package/scripts/testing/src/core/database/database-adapter.js +54 -0
  193. package/scripts/testing/src/core/errors/index.js +291 -0
  194. package/scripts/testing/src/core/errors/recovery.js +268 -0
  195. package/scripts/testing/src/core/monitoring/logger.js +145 -0
  196. package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
  197. package/scripts/testing/src/core/session/index.js +1 -0
  198. package/scripts/testing/src/core/session/session-manager.js +323 -0
  199. package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
  200. package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
  201. package/scripts/testing/src/core/trace/debug-trace.js +398 -0
  202. package/scripts/testing/src/core/trace/index.js +120 -0
  203. package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
  204. package/scripts/update-linear-status.js +268 -0
  205. package/scripts/update-linear-tasks-fixed.js +284 -0
  206. package/scripts/verify-railway-schema.ts +35 -0
  207. package/templates/claude-hooks/hooks.json +5 -0
  208. package/templates/claude-hooks/on-clear.js +56 -0
  209. package/templates/claude-hooks/on-startup.js +56 -0
  210. package/templates/claude-hooks/tool-use-trace.js +67 -0
  211. package/dist/features/tui/components/analytics-panel.js +0 -157
  212. package/dist/features/tui/components/analytics-panel.js.map +0 -7
  213. package/dist/features/tui/components/frame-visualizer.js +0 -377
  214. package/dist/features/tui/components/frame-visualizer.js.map +0 -7
  215. package/dist/features/tui/components/pr-tracker.js +0 -135
  216. package/dist/features/tui/components/pr-tracker.js.map +0 -7
  217. package/dist/features/tui/components/session-monitor.js +0 -299
  218. package/dist/features/tui/components/session-monitor.js.map +0 -7
  219. package/dist/features/tui/components/subagent-fleet.js +0 -395
  220. package/dist/features/tui/components/subagent-fleet.js.map +0 -7
  221. package/dist/features/tui/components/task-board.js +0 -1139
  222. package/dist/features/tui/components/task-board.js.map +0 -7
  223. package/dist/features/tui/index.js +0 -408
  224. package/dist/features/tui/index.js.map +0 -7
  225. package/dist/features/tui/services/data-service.js +0 -641
  226. package/dist/features/tui/services/data-service.js.map +0 -7
  227. package/dist/features/tui/services/linear-task-reader.js +0 -102
  228. package/dist/features/tui/services/linear-task-reader.js.map +0 -7
  229. package/dist/features/tui/services/websocket-client.js +0 -162
  230. package/dist/features/tui/services/websocket-client.js.map +0 -7
  231. package/dist/features/tui/terminal-compat.js +0 -220
  232. package/dist/features/tui/terminal-compat.js.map +0 -7
  233. package/dist/features/tui/types.js +0 -1
  234. package/dist/features/tui/types.js.map +0 -7
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/cli/index.ts"],
4
- "sourcesContent": ["#!/usr/bin/env node\n/**\n * StackMemory CLI\n * Command-line interface for StackMemory operations\n */\n\n// Set environment flag for CLI usage to skip async context bridge\nprocess.env['STACKMEMORY_CLI'] = 'true';\n\nimport { program } from 'commander';\nimport { logger } from '../core/monitoring/logger.js';\nimport { FrameManager } from '../core/context/frame-manager.js';\nimport { sessionManager, FrameQueryMode } from '../core/session/index.js';\nimport { sharedContextLayer } from '../core/context/shared-context-layer.js';\nimport { LinearTaskManager } from '../features/tasks/linear-task-manager.js';\nimport {\n LinearAuthManager,\n LinearOAuthSetup,\n} from '../integrations/linear/auth.js';\nimport {\n LinearSyncEngine,\n DEFAULT_SYNC_CONFIG,\n} from '../integrations/linear/sync.js';\nimport {\n initializeAutoSync,\n getAutoSyncService,\n stopAutoSync,\n} from '../integrations/linear/auto-sync.js';\nimport { LinearConfigManager } from '../integrations/linear/config.js';\nimport { UpdateChecker } from '../core/utils/update-checker.js';\nimport { ProgressTracker } from '../core/monitoring/progress-tracker.js';\nimport { registerProjectCommands } from './commands/projects.js';\nimport { registerLinearCommands } from './commands/linear.js';\nimport { registerLinearTestCommand } from './commands/linear-test.js';\nimport { registerLinearListCommand } from './commands/linear-list.js';\nimport { registerLinearMigrateCommand } from './commands/linear-migrate.js';\nimport { registerLinearCreateCommand } from './commands/linear-create.js';\nimport { createChromaDBCommand } from './commands/chromadb.js';\nimport { createInfiniteStorageCommand } from './commands/infinite-storage.js';\nimport { createGCCommand } from './commands/gc.js';\nimport { createSessionCommands } from './commands/session.js';\nimport { registerWorktreeCommands } from './commands/worktree.js';\nimport { registerOnboardingCommand } from './commands/onboard.js';\nimport { webhookCommand } from './commands/webhook.js';\nimport { createTaskCommands } from './commands/tasks.js';\nimport { createSearchCommand } from './commands/search.js';\nimport { createLogCommand } from './commands/log.js';\nimport { createContextCommands } from './commands/context.js';\nimport { createConfigCommand } from './commands/config.js';\nimport { createAgentCommand } from './commands/agent.js';\nimport { createHandoffCommand } from './commands/handoff.js';\nimport { createStorageCommand } from './commands/storage.js';\nimport { createSkillsCommand } from './commands/skills.js';\nimport { createTestCommand } from './commands/test.js';\nimport clearCommand from './commands/clear.js';\nimport createWorkflowCommand from './commands/workflow.js';\nimport monitorCommand from './commands/monitor.js';\nimport qualityCommand from './commands/quality.js';\nimport { ProjectManager } from '../core/projects/project-manager.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\n\nconst VERSION = '0.3.14';\n\n// Check for updates on CLI startup\nUpdateChecker.checkForUpdates(VERSION, true).catch(() => {\n // Silently ignore errors\n});\n\nprogram\n .name('stackmemory')\n .description(\n 'Lossless memory runtime for AI coding tools - organizes context as a call stack instead of linear chat logs, with team collaboration and infinite retention'\n )\n .version(VERSION);\n\nprogram\n .command('init')\n .description('Initialize StackMemory in current project')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbDir = join(projectRoot, '.stackmemory');\n\n if (!existsSync(dbDir)) {\n mkdirSync(dbDir, { recursive: true });\n }\n\n const dbPath = join(dbDir, 'context.db');\n const db = new Database(dbPath);\n new FrameManager(db, 'cli-project');\n\n logger.info('StackMemory initialized successfully', { projectRoot });\n console.log('\u2705 StackMemory initialized in', projectRoot);\n\n db.close();\n } catch (error: unknown) {\n logger.error('Failed to initialize StackMemory', error as Error);\n console.error('\u274C Initialization failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('status')\n .description('Show current StackMemory status')\n .option('--all', 'Show all active frames across sessions')\n .option('--project', 'Show all active frames in current project')\n .option('--session <id>', 'Show frames for specific session')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n // Check for updates and display if available\n await UpdateChecker.checkForUpdates(VERSION);\n\n // Initialize session manager and shared context\n await sessionManager.initialize();\n await sharedContextLayer.initialize();\n\n const session = await sessionManager.getOrCreateSession({\n projectPath: projectRoot,\n sessionId: options.session,\n });\n\n // Auto-discover shared context on startup\n const contextDiscovery = await sharedContextLayer.autoDiscoverContext();\n\n // Show context hints if available\n if (\n contextDiscovery.hasSharedContext &&\n contextDiscovery.sessionCount > 1\n ) {\n console.log(`\\n\uD83D\uDCA1 Shared Context Available:`);\n console.log(\n ` ${contextDiscovery.sessionCount} sessions with shared context`\n );\n\n if (contextDiscovery.recentPatterns.length > 0) {\n console.log(` Recent patterns:`);\n contextDiscovery.recentPatterns.slice(0, 3).forEach((p) => {\n console.log(\n ` \u2022 ${p.type}: ${p.pattern.slice(0, 50)} (${p.frequency}x)`\n );\n });\n }\n\n if (contextDiscovery.lastDecisions.length > 0) {\n console.log(\n ` Last decision: ${contextDiscovery.lastDecisions[0].decision.slice(0, 60)}`\n );\n }\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db, session.projectId);\n\n // Set query mode based on options\n if (options.all) {\n frameManager.setQueryMode(FrameQueryMode.ALL_ACTIVE);\n } else if (options.project) {\n frameManager.setQueryMode(FrameQueryMode.PROJECT_ACTIVE);\n }\n\n const activeFrames = frameManager.getActiveFramePath();\n const stackDepth = frameManager.getStackDepth();\n\n // Always get total counts across all sessions\n const totalStats = db\n .prepare(\n `\n SELECT \n COUNT(*) as total_frames,\n SUM(CASE WHEN state = 'active' THEN 1 ELSE 0 END) as active_frames,\n SUM(CASE WHEN state = 'closed' THEN 1 ELSE 0 END) as closed_frames,\n COUNT(DISTINCT run_id) as total_sessions\n FROM frames\n WHERE project_id = ?\n `\n )\n .get(session.projectId) as {\n total_frames: number;\n active_frames: number;\n closed_frames: number;\n total_sessions: number;\n };\n\n const contextCount = db\n .prepare(\n `\n SELECT COUNT(*) as count FROM contexts\n `\n )\n .get() as { count: number };\n\n const eventCount = db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(session.projectId) as { count: number };\n\n console.log('\uD83D\uDCCA StackMemory Status:');\n console.log(\n ` Session: ${session.sessionId.slice(0, 8)} (${session.state}, ${Math.round((Date.now() - session.startedAt) / 1000 / 60)}min old)`\n );\n console.log(` Project: ${session.projectId}`);\n if (session.branch) {\n console.log(` Branch: ${session.branch}`);\n }\n\n // Show total database statistics\n console.log(`\\n Database Statistics (this project):`);\n console.log(\n ` Frames: ${totalStats.total_frames || 0} (${totalStats.active_frames || 0} active, ${totalStats.closed_frames || 0} closed)`\n );\n console.log(` Events: ${eventCount.count || 0}`);\n console.log(` Sessions: ${totalStats.total_sessions || 0}`);\n console.log(` Cached contexts: ${contextCount.count || 0} (global)`);\n\n // Show recent activity\n const recentFrames = db\n .prepare(\n `\n SELECT name, type, state, datetime(created_at, 'unixepoch') as created\n FROM frames\n WHERE project_id = ?\n ORDER BY created_at DESC\n LIMIT 3\n `\n )\n .all(session.projectId) as Array<{\n name: string;\n type: string;\n state: string;\n created: string;\n }>;\n\n if (recentFrames.length > 0) {\n console.log(`\\n Recent Activity:`);\n recentFrames.forEach((f) => {\n const stateIcon = f.state === 'active' ? '\uD83D\uDFE2' : '\u26AB';\n console.log(` ${stateIcon} ${f.name} [${f.type}] - ${f.created}`);\n });\n }\n\n console.log(`\\n Current Session:`);\n console.log(` Stack depth: ${stackDepth}`);\n console.log(` Active frames: ${activeFrames.length}`);\n\n if (activeFrames.length > 0) {\n activeFrames.forEach((frame, i) => {\n const indent = ' ' + ' '.repeat(frame.depth || i);\n const prefix = i === 0 ? '\u2514\u2500' : ' \u2514\u2500';\n console.log(`${indent}${prefix} ${frame.name} [${frame.type}]`);\n });\n }\n\n // Show other sessions if in default mode\n if (!options.all && !options.project) {\n const otherSessions = await sessionManager.listSessions({\n projectId: session.projectId,\n state: 'active',\n });\n\n const otherActive = otherSessions.filter(\n (s) => s.sessionId !== session.sessionId\n );\n if (otherActive.length > 0) {\n console.log(`\\n Other Active Sessions (same project):`);\n otherActive.forEach((s) => {\n const age = Math.round(\n (Date.now() - s.lastActiveAt) / 1000 / 60 / 60\n );\n console.log(\n ` - ${s.sessionId.slice(0, 8)}: ${s.branch || 'main'}, ${age}h old`\n );\n });\n console.log(`\\n Tip: Use --all to see frames across sessions`);\n }\n }\n\n db.close();\n } catch (error: unknown) {\n logger.error('Failed to get status', error as Error);\n console.error('\u274C Status check failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Linear Integration Commands\nconst linearCommand = program\n .command('linear')\n .description('Linear API integration commands');\n\nlinearCommand\n .command('setup')\n .description('Setup Linear OAuth integration')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const linearSetup = new LinearOAuthSetup(projectRoot);\n\n const { authUrl, instructions } = await linearSetup.setupInteractive();\n\n console.log('\uD83D\uDD17 Linear OAuth Setup\\n');\n\n instructions.forEach((instruction) => {\n console.log(instruction);\n });\n\n if (authUrl) {\n console.log('\\n\uD83D\uDCCB Next step: Complete authorization and run:');\n console.log('stackmemory linear authorize <auth-code>');\n }\n } catch (error: unknown) {\n logger.error('Linear setup failed', error as Error);\n console.error('\u274C Setup failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('authorize')\n .description('Complete Linear OAuth authorization')\n .argument('<code>', 'Authorization code from Linear')\n .action(async (authCode: string) => {\n try {\n const projectRoot = process.cwd();\n const linearSetup = new LinearOAuthSetup(projectRoot);\n\n const success = await linearSetup.completeAuth(authCode);\n\n if (success) {\n console.log('\u2705 Linear integration authorized successfully!');\n console.log('\uD83E\uDDEA Testing connection...');\n\n const connectionOk = await linearSetup.testConnection();\n if (connectionOk) {\n console.log('\u2705 Linear connection test passed!');\n console.log('\\n\uD83D\uDE80 You can now use:');\n console.log('- stackmemory linear sync');\n console.log('- stackmemory linear status');\n } else {\n console.log(\n '\u26A0\uFE0F Linear connection test failed. Check your configuration.'\n );\n }\n } else {\n console.error('\u274C Authorization failed. Please try again.');\n process.exit(1);\n }\n } catch (error: unknown) {\n logger.error('Linear authorization failed', error as Error);\n console.error('\u274C Authorization failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('status')\n .description('Show Linear integration status')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const authManager = new LinearAuthManager(projectRoot);\n\n const isConfigured = authManager.isConfigured();\n\n console.log('\uD83D\uDCCA Linear Integration Status:');\n console.log(` Configured: ${isConfigured ? '\u2705' : '\u274C'}`);\n\n if (isConfigured) {\n const config = authManager.loadConfig();\n const tokens = authManager.loadTokens();\n\n console.log(\n ` Client ID: ${config?.clientId ? config.clientId.substring(0, 8) + '...' : 'Not set'}`\n );\n console.log(` Tokens: ${tokens ? '\u2705 Valid' : '\u274C Missing'}`);\n\n if (tokens) {\n const expiresIn = Math.floor(\n (tokens.expiresAt - Date.now()) / 1000 / 60\n );\n console.log(\n ` Token expires: ${expiresIn > 0 ? `${expiresIn} minutes` : 'Expired'}`\n );\n }\n\n // Test connection\n console.log('\\n\uD83E\uDDEA Testing connection...');\n const linearSetup = new LinearOAuthSetup(projectRoot);\n const connectionOk = await linearSetup.testConnection();\n console.log(` Connection: ${connectionOk ? '\u2705 OK' : '\u274C Failed'}`);\n } else {\n console.log('\\n\uD83D\uDCA1 Run \"stackmemory linear setup\" to get started');\n }\n } catch (error: unknown) {\n logger.error('Linear status check failed', error as Error);\n console.error('\u274C Status check failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('sync')\n .description('Sync tasks with Linear')\n .option(\n '-d, --direction <direction>',\n 'Sync direction: bidirectional, to_linear, from_linear',\n 'bidirectional'\n )\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const authManager = new LinearAuthManager(projectRoot);\n\n // Check for API key from environment first\n if (!process.env['LINEAR_API_KEY'] && !authManager.isConfigured()) {\n console.log(\n '\u274C Linear not configured. Set LINEAR_API_KEY environment variable or run \"stackmemory linear setup\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n const taskStore = new LinearTaskManager(projectRoot, db);\n\n const syncConfig = {\n ...DEFAULT_SYNC_CONFIG,\n enabled: true,\n direction: options.direction,\n };\n\n const linearSync = new LinearSyncEngine(\n taskStore,\n authManager,\n syncConfig\n );\n\n console.log(`\uD83D\uDD04 Starting ${options.direction} sync with Linear...`);\n\n const result = await linearSync.sync();\n\n // Track progress\n const progress = new ProgressTracker(projectRoot);\n\n if (result.success) {\n console.log('\u2705 Sync completed successfully!');\n console.log(` To Linear: ${result.synced.toLinear} created`);\n console.log(` From Linear: ${result.synced.fromLinear} created`);\n console.log(` Updated: ${result.synced.updated}`);\n\n // Update progress tracker\n progress.updateLinearStatus({\n lastSync: new Date().toISOString(),\n tasksSynced:\n result.synced.toLinear +\n result.synced.fromLinear +\n result.synced.updated,\n });\n\n if (result.conflicts.length > 0) {\n console.log(`\\n\u26A0\uFE0F Conflicts detected: ${result.conflicts.length}`);\n result.conflicts.forEach((conflict) => {\n console.log(` - ${conflict.taskId}: ${conflict.reason}`);\n });\n }\n } else {\n console.log('\u274C Sync failed');\n if (result.errors.length > 0) {\n result.errors.forEach((error) => {\n console.log(` Error: ${error}`);\n });\n }\n }\n\n db.close();\n } catch (error: unknown) {\n logger.error('Linear sync failed', error as Error);\n console.error('\u274C Sync failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Auto-sync commands\nlinearCommand\n .command('auto-sync')\n .description('Manage automatic synchronization')\n .option('--start', 'Start auto-sync service')\n .option('--stop', 'Stop auto-sync service')\n .option('--status', 'Show auto-sync status')\n .option('--interval <minutes>', 'Set sync interval in minutes', '5')\n .option(\n '--direction <direction>',\n 'Set sync direction: bidirectional, to_linear, from_linear',\n 'bidirectional'\n )\n .option('--quiet-start <hour>', 'Start of quiet hours (0-23)', '22')\n .option('--quiet-end <hour>', 'End of quiet hours (0-23)', '7')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n\n if (options.status) {\n const service = getAutoSyncService();\n if (service) {\n const status = service.getStatus();\n console.log('\uD83D\uDCCA Linear Auto-Sync Status:');\n console.log(` Running: ${status.running ? '\u2705' : '\u274C'}`);\n console.log(` Direction: ${status.config.direction}`);\n console.log(` Interval: ${status.config.interval} minutes`);\n console.log(\n ` Conflict Resolution: ${status.config.conflictResolution}`\n );\n\n if (status.lastSyncTime > 0) {\n const lastSync = new Date(status.lastSyncTime);\n console.log(` Last Sync: ${lastSync.toLocaleString()}`);\n }\n\n if (status.nextSyncTime) {\n const nextSync = new Date(status.nextSyncTime);\n console.log(` Next Sync: ${nextSync.toLocaleString()}`);\n }\n\n if (status.config.quietHours) {\n console.log(\n ` Quiet Hours: ${status.config.quietHours.start}:00 - ${status.config.quietHours.end}:00`\n );\n }\n\n if (status.retryCount > 0) {\n console.log(` \u26A0\uFE0F Retry Count: ${status.retryCount}`);\n }\n } else {\n console.log('\uD83D\uDCCA Linear Auto-Sync Status: \u274C Not running');\n }\n return;\n }\n\n if (options.start) {\n const authManager = new LinearAuthManager(projectRoot);\n if (!authManager.isConfigured()) {\n console.log(\n '\u274C Linear not configured. Run \"stackmemory linear setup\" first.'\n );\n return;\n }\n\n const config = {\n interval: parseInt(options.interval),\n direction: options.direction,\n quietHours: {\n start: parseInt(options.quietStart),\n end: parseInt(options.quietEnd),\n },\n };\n\n const service = initializeAutoSync(projectRoot, config);\n await service.start();\n\n console.log('\u2705 Linear auto-sync started');\n console.log(` Interval: ${config.interval} minutes`);\n console.log(` Direction: ${config.direction}`);\n console.log(\n ` Quiet Hours: ${config.quietHours.start}:00 - ${config.quietHours.end}:00`\n );\n console.log(\n '\\n\uD83D\uDCA1 Use \"stackmemory linear auto-sync --status\" to check status'\n );\n\n // Keep process alive for auto-sync\n process.on('SIGINT', () => {\n console.log('\\n\uD83D\uDED1 Stopping auto-sync service...');\n service.stop();\n process.exit(0);\n });\n\n console.log('\uD83D\uDD04 Auto-sync running... Press Ctrl+C to stop');\n // Keep the process running\n await new Promise(() => {}); // Intentionally never resolves\n }\n\n if (options.stop) {\n stopAutoSync();\n console.log('\uD83D\uDED1 Linear auto-sync stopped');\n }\n\n if (!options.start && !options.stop && !options.status) {\n console.log('\uD83D\uDCA1 Usage:');\n console.log(' --start Start auto-sync service');\n console.log(' --stop Stop auto-sync service');\n console.log(' --status Show current status');\n console.log(\n '\\nExample: stackmemory linear auto-sync --start --interval 10'\n );\n }\n } catch (error: unknown) {\n logger.error('Linear auto-sync command failed', error as Error);\n console.error('\u274C Auto-sync failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('force-sync')\n .description('Force immediate synchronization')\n .action(async () => {\n try {\n const service = getAutoSyncService();\n if (service) {\n console.log('\uD83D\uDD04 Forcing immediate sync...');\n await service.forceSync();\n console.log('\u2705 Sync completed');\n } else {\n console.log(\n '\u274C Auto-sync service not running. Use manual sync instead:'\n );\n console.log(' stackmemory linear sync');\n }\n } catch (error: unknown) {\n logger.error('Force sync failed', error as Error);\n console.error('\u274C Force sync failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('update <issueId>')\n .description('Update Linear task status')\n .option(\n '-s, --status <status>',\n 'New status (todo, in-progress, done, canceled)'\n )\n .option('-t, --title <title>', 'Update task title')\n .option('-d, --description <desc>', 'Update task description')\n .option(\n '-p, --priority <priority>',\n 'Set priority (1=urgent, 2=high, 3=medium, 4=low)'\n )\n .action(async (issueId, options) => {\n try {\n const projectRoot = process.cwd();\n const authManager = new LinearAuthManager(projectRoot);\n const tokens = authManager.loadTokens();\n\n if (!tokens) {\n console.error('\u274C Not authenticated. Run: stackmemory linear setup');\n process.exit(1);\n }\n\n const { LinearClient } = await import('../integrations/linear/client.js');\n const client = new LinearClient({\n apiKey: tokens.accessToken,\n useBearer: true,\n onUnauthorized: async () => {\n const refreshed = await authManager.refreshAccessToken();\n return refreshed.accessToken;\n },\n });\n\n // Find the issue first\n let issue = await client.getIssue(issueId);\n if (!issue) {\n // Try finding by identifier\n issue = await client.findIssueByIdentifier(issueId);\n }\n\n if (!issue) {\n console.error(`\u274C Issue ${issueId} not found`);\n process.exit(1);\n }\n\n const updates: Record<string, unknown> = {};\n\n // Handle status update\n if (options.status) {\n const team = await client.getTeam();\n const states = await client.getWorkflowStates(team.id);\n\n const statusMap: Record<string, string> = {\n todo: 'unstarted',\n 'in-progress': 'started',\n done: 'completed',\n canceled: 'cancelled',\n };\n\n const targetType =\n statusMap[options.status.toLowerCase()] || options.status;\n const targetState = states.find(\n (s: { type: string }) => s.type === targetType\n );\n\n if (!targetState) {\n console.error(`\u274C Invalid status: ${options.status}`);\n console.log('Available states:');\n states.forEach((s: { name: string; type: string }) =>\n console.log(` - ${s.name} (${s.type})`)\n );\n process.exit(1);\n }\n\n updates.stateId = targetState.id;\n }\n\n if (options.title) updates.title = options.title;\n if (options.description) updates.description = options.description;\n if (options.priority) updates.priority = parseInt(options.priority);\n\n // Perform update\n const updatedIssue = await client.updateIssue(issue.id, updates);\n\n console.log(\n `\u2705 Updated ${updatedIssue.identifier}: ${updatedIssue.title}`\n );\n if (options.status) {\n console.log(` Status: ${updatedIssue.state.name}`);\n }\n console.log(` ${updatedIssue.url}`);\n\n // Auto-sync to local tasks after update\n console.log('\\n\uD83D\uDD04 Syncing to local tasks...');\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n if (existsSync(dbPath)) {\n const db = new Database(dbPath);\n const taskStore = new LinearTaskManager(projectRoot, db);\n const { LinearSyncEngine, DEFAULT_SYNC_CONFIG } =\n await import('../integrations/linear/sync.js');\n const syncEngine = new LinearSyncEngine(\n taskStore,\n authManager,\n { ...DEFAULT_SYNC_CONFIG, enabled: true, direction: 'from_linear' },\n projectRoot\n );\n const syncResult = await syncEngine.sync();\n if (syncResult.success) {\n console.log(\n ` \u2705 Local tasks synced (${syncResult.synced.fromLinear} new, ${syncResult.synced.updated} updated)`\n );\n }\n db.close();\n }\n } catch (error: unknown) {\n logger.error('Failed to update Linear task', error as Error);\n console.error('\u274C Failed to update task:', (error as Error).message);\n process.exit(1);\n }\n });\n\nlinearCommand\n .command('config')\n .description('Configure auto-sync settings')\n .option('--show', 'Show current configuration')\n .option('--set-interval <minutes>', 'Set sync interval in minutes')\n .option(\n '--set-direction <direction>',\n 'Set sync direction: bidirectional, to_linear, from_linear'\n )\n .option(\n '--set-conflict-resolution <strategy>',\n 'Set conflict resolution: newest_wins, linear_wins, stackmemory_wins, manual'\n )\n .option('--set-quiet-start <hour>', 'Set start of quiet hours (0-23)')\n .option('--set-quiet-end <hour>', 'Set end of quiet hours (0-23)')\n .option('--enable', 'Enable auto-sync')\n .option('--disable', 'Disable auto-sync')\n .option('--reset', 'Reset to default configuration')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const configManager = new LinearConfigManager(projectRoot);\n\n if (options.reset) {\n configManager.resetConfig();\n console.log('\u2705 Configuration reset to defaults');\n return;\n }\n\n if (options.show) {\n const config = configManager.loadConfig();\n if (config) {\n console.log('\uD83D\uDCCA Linear Auto-Sync Configuration:');\n console.log(` Enabled: ${config.enabled ? '\u2705' : '\u274C'}`);\n console.log(` Interval: ${config.interval} minutes`);\n console.log(` Direction: ${config.direction}`);\n console.log(` Conflict Resolution: ${config.conflictResolution}`);\n console.log(` Retry Attempts: ${config.retryAttempts}`);\n console.log(` Retry Delay: ${config.retryDelay / 1000}s`);\n\n if (config.quietHours) {\n console.log(\n ` Quiet Hours: ${config.quietHours.start}:00 - ${config.quietHours.end}:00`\n );\n }\n\n const lastUpdated = new Date(config.lastUpdated);\n console.log(` Last Updated: ${lastUpdated.toLocaleString()}`);\n } else {\n console.log('\uD83D\uDCCA No configuration found. Using defaults.');\n const defaultConfig = configManager.getDefaultConfig();\n console.log(` Default interval: ${defaultConfig.interval} minutes`);\n console.log(` Default direction: ${defaultConfig.direction}`);\n }\n return;\n }\n\n // Update configuration\n const updates: Record<string, unknown> = {};\n\n if (options.setInterval) {\n const interval = parseInt(options.setInterval);\n if (isNaN(interval) || interval < 1) {\n console.error('\u274C Interval must be a positive number');\n process.exit(1);\n }\n updates.interval = interval;\n console.log(`\u2705 Set interval to ${interval} minutes`);\n }\n\n if (options.setDirection) {\n const validDirections = ['bidirectional', 'to_linear', 'from_linear'];\n if (!validDirections.includes(options.setDirection)) {\n console.error(\n `\u274C Invalid direction. Must be one of: ${validDirections.join(', ')}`\n );\n process.exit(1);\n }\n updates.direction = options.setDirection;\n console.log(`\u2705 Set direction to ${options.setDirection}`);\n }\n\n if (options.setConflictResolution) {\n const validStrategies = [\n 'newest_wins',\n 'linear_wins',\n 'stackmemory_wins',\n 'manual',\n ];\n if (!validStrategies.includes(options.setConflictResolution)) {\n console.error(\n `\u274C Invalid strategy. Must be one of: ${validStrategies.join(', ')}`\n );\n process.exit(1);\n }\n updates.conflictResolution = options.setConflictResolution;\n console.log(\n `\u2705 Set conflict resolution to ${options.setConflictResolution}`\n );\n }\n\n if (options.setQuietStart) {\n const hour = parseInt(options.setQuietStart);\n if (isNaN(hour) || hour < 0 || hour > 23) {\n console.error('\u274C Quiet start hour must be between 0 and 23');\n process.exit(1);\n }\n const currentConfig =\n configManager.loadConfig() || configManager.getDefaultConfig();\n updates.quietHours = {\n start: hour,\n end: currentConfig.quietHours?.end || 7,\n };\n console.log(`\u2705 Set quiet hours start to ${hour}:00`);\n }\n\n if (options.setQuietEnd) {\n const hour = parseInt(options.setQuietEnd);\n if (isNaN(hour) || hour < 0 || hour > 23) {\n console.error('\u274C Quiet end hour must be between 0 and 23');\n process.exit(1);\n }\n const currentConfig =\n configManager.loadConfig() || configManager.getDefaultConfig();\n updates.quietHours = {\n start: currentConfig.quietHours?.start || 22,\n end: hour,\n };\n console.log(`\u2705 Set quiet hours end to ${hour}:00`);\n }\n\n if (options.enable) {\n updates.enabled = true;\n console.log('\u2705 Auto-sync enabled');\n }\n\n if (options.disable) {\n updates.enabled = false;\n console.log('\u274C Auto-sync disabled');\n }\n\n if (Object.keys(updates).length > 0) {\n configManager.saveConfig(updates);\n console.log(\n '\\n\uD83D\uDCA1 Configuration updated. Restart auto-sync service to apply changes.'\n );\n } else if (!options.show) {\n console.log('\uD83D\uDCA1 Use --show to view current configuration');\n console.log('\uD83D\uDCA1 Use --help to see all configuration options');\n }\n } catch (error: unknown) {\n logger.error('Linear config command failed', error as Error);\n console.error('\u274C Config failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('update-check')\n .description('Check for StackMemory updates')\n .action(async () => {\n try {\n console.log('\uD83D\uDD0D Checking for updates...');\n await UpdateChecker.forceCheck(VERSION);\n } catch (error: unknown) {\n logger.error('Update check failed', error as Error);\n console.error('\u274C Update check failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('analytics')\n .description('Launch task analytics dashboard')\n .option('-p, --port <port>', 'Port for dashboard server', '3000')\n .option('-o, --open', 'Open dashboard in browser')\n .option('--export <format>', 'Export metrics (json|csv)')\n .option('--sync', 'Sync with Linear before launching')\n .option('--view', 'Show analytics in terminal')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n if (options.view) {\n const { displayAnalyticsDashboard } = await import('./utils/viewer.js');\n await displayAnalyticsDashboard(projectRoot);\n return;\n }\n\n if (options.export) {\n const { AnalyticsService } =\n await import('../features/analytics/index.js');\n const service = new AnalyticsService(projectRoot);\n\n if (options.sync) {\n console.log('\uD83D\uDD04 Syncing with Linear...');\n await service.syncLinearTasks();\n }\n\n const state = await service.getDashboardState();\n\n if (options.export === 'csv') {\n console.log('\uD83D\uDCCA Exporting metrics as CSV...');\n // Convert to CSV format\n const tasks = state.recentTasks;\n const headers = [\n 'ID',\n 'Title',\n 'State',\n 'Priority',\n 'Created',\n 'Completed',\n ];\n const rows = tasks.map((t) => [\n t.id,\n t.title,\n t.state,\n t.priority,\n t.createdAt.toISOString(),\n t.completedAt?.toISOString() || '',\n ]);\n console.log(headers.join(','));\n rows.forEach((r) => console.log(r.join(',')));\n } else {\n console.log(JSON.stringify(state, null, 2));\n }\n\n service.close();\n return;\n }\n\n // Launch dashboard server\n console.log(\n `\uD83D\uDE80 Launching analytics dashboard on port ${options.port}...`\n );\n\n const express = (await import('express')).default;\n const { AnalyticsAPI } = await import('../features/analytics/index.js');\n const { createServer } = await import('http');\n\n const app = express();\n\n // Add error handling middleware\n app.use(\n (\n err: Error,\n _req: import('express').Request,\n res: import('express').Response,\n _next: import('express').NextFunction\n ) => {\n console.error('Express error:', err);\n res.status(500).json({ error: err.message });\n }\n );\n\n const analyticsAPI = new AnalyticsAPI(projectRoot);\n\n if (options.sync) {\n console.log('\uD83D\uDD04 Syncing with Linear...');\n const service = new (\n await import('../features/analytics/index.js')\n ).AnalyticsService(projectRoot);\n await service.syncLinearTasks();\n service.close();\n }\n\n app.use('/api/analytics', analyticsAPI.getRouter());\n\n // Serve the HTML dashboard\n app.get('/', async (req, res) => {\n // Try multiple paths for the dashboard HTML\n const possiblePaths = [\n join(projectRoot, 'src/features/analytics/dashboard.html'),\n join(projectRoot, 'dist/features/analytics/dashboard.html'),\n ];\n\n for (const dashboardPath of possiblePaths) {\n if (existsSync(dashboardPath)) {\n res.sendFile(dashboardPath);\n return;\n }\n }\n\n // Inline fallback dashboard\n res.send(`<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n <meta charset=\"UTF-8\">\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n <title>StackMemory Analytics</title>\n <script src=\"https://cdn.jsdelivr.net/npm/chart.js\"></script>\n <style>\n * { margin: 0; padding: 0; box-sizing: border-box; }\n body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; background: #1a1a2e; color: #eee; padding: 20px; }\n .container { max-width: 1200px; margin: 0 auto; }\n h1 { color: #667eea; margin-bottom: 20px; }\n .grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 20px; margin-bottom: 30px; }\n .card { background: #16213e; border-radius: 12px; padding: 20px; }\n .metric-value { font-size: 2.5em; font-weight: bold; color: #667eea; }\n .metric-label { color: #888; text-transform: uppercase; font-size: 0.8em; }\n .task-list { max-height: 400px; overflow-y: auto; }\n .task-item { padding: 10px; border-left: 3px solid #667eea; margin-bottom: 8px; background: #1a1a2e; }\n .task-item.completed { border-color: #22c55e; }\n .task-item.in_progress { border-color: #f59e0b; }\n .status { display: inline-block; padding: 2px 8px; border-radius: 4px; font-size: 0.8em; margin-right: 8px; }\n .status.completed { background: #22c55e30; color: #22c55e; }\n .status.in_progress { background: #f59e0b30; color: #f59e0b; }\n .status.todo { background: #667eea30; color: #667eea; }\n </style>\n</head>\n<body>\n <div class=\"container\">\n <h1>\uD83D\uDCCA StackMemory Analytics</h1>\n <div class=\"grid\" id=\"metrics\"></div>\n <div class=\"card\"><h3>Recent Tasks</h3><div class=\"task-list\" id=\"tasks\">Loading...</div></div>\n </div>\n <script>\n async function load() {\n const metrics = await fetch('/api/analytics/metrics').then(r => r.json());\n const tasks = await fetch('/api/analytics/tasks').then(r => r.json());\n \n document.getElementById('metrics').innerHTML = \\`\n <div class=\"card\"><div class=\"metric-label\">Total</div><div class=\"metric-value\">\\${metrics.data.metrics.totalTasks}</div></div>\n <div class=\"card\"><div class=\"metric-label\">Completed</div><div class=\"metric-value\">\\${metrics.data.metrics.completedTasks}</div></div>\n <div class=\"card\"><div class=\"metric-label\">In Progress</div><div class=\"metric-value\">\\${metrics.data.metrics.inProgressTasks}</div></div>\n <div class=\"card\"><div class=\"metric-label\">Completion</div><div class=\"metric-value\">\\${metrics.data.metrics.completionRate.toFixed(0)}%</div></div>\n \\`;\n \n document.getElementById('tasks').innerHTML = tasks.data.tasks.slice(0, 10).map((t: any) => \\`\n <div class=\"task-item \\${t.state}\">\n <span class=\"status \\${t.state}\">\\${t.state}</span>\n <strong>\\${t.title}</strong>\n </div>\n \\`).join('');\n }\n load();\n setInterval(load, 30000);\n </script>\n</body>\n</html>`);\n });\n\n const server = createServer(app);\n analyticsAPI.setupWebSocket(server);\n\n server.listen(options.port, async () => {\n console.log(\n `\u2705 Analytics dashboard running at http://localhost:${options.port}`\n );\n\n if (options.open) {\n const { exec } = await import('child_process');\n const url = `http://localhost:${options.port}`;\n const command =\n process.platform === 'darwin'\n ? `open ${url}`\n : process.platform === 'win32'\n ? `start ${url}`\n : `xdg-open ${url}`;\n exec(command);\n }\n });\n\n process.on('SIGINT', () => {\n console.log('\\n\uD83D\uDC4B Shutting down analytics dashboard...');\n analyticsAPI.close();\n server.close();\n process.exit(0);\n });\n\n // Keep the process alive\n await new Promise(() => {});\n } catch (error: unknown) {\n logger.error('Analytics command failed', error as Error);\n console.error('\u274C Analytics failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('progress')\n .description('Show current progress and recent changes')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const progress = new ProgressTracker(projectRoot);\n console.log(progress.getSummary());\n } catch (error: unknown) {\n logger.error('Failed to show progress', error as Error);\n console.error('\u274C Failed to show progress:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('mcp-server')\n .description('Start StackMemory MCP server for Claude Desktop')\n .option('-p, --project <path>', 'Project root directory', process.cwd())\n .action(async (options) => {\n try {\n const { runMCPServer } = await import('../integrations/mcp/server.js');\n\n // Set project root\n process.env['PROJECT_ROOT'] = options.project;\n\n console.log('\uD83D\uDE80 Starting StackMemory MCP Server...');\n console.log(` Project: ${options.project}`);\n console.log(` Version: ${VERSION}`);\n\n // Check for updates silently\n UpdateChecker.checkForUpdates(VERSION, true).catch(() => {});\n\n // Start the MCP server\n await runMCPServer();\n } catch (error: unknown) {\n logger.error('Failed to start MCP server', error as Error);\n console.error('\u274C MCP server failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Add test context command\nprogram\n .command('context:test')\n .description('Test context persistence by creating sample frames')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db, 'cli-project');\n\n // Create test frames\n console.log('\uD83D\uDCDD Creating test context frames...');\n\n const rootFrame = frameManager.createFrame({\n type: 'task',\n name: 'Test Session',\n inputs: { test: true, timestamp: new Date().toISOString() },\n });\n\n const taskFrame = frameManager.createFrame({\n type: 'subtask',\n name: 'Sample Task',\n inputs: { description: 'Testing context persistence' },\n parentFrameId: rootFrame,\n });\n\n const commandFrame = frameManager.createFrame({\n type: 'tool_scope',\n name: 'test-command',\n inputs: { args: ['--test'] },\n parentFrameId: taskFrame,\n });\n\n // Add some events\n frameManager.addEvent(\n 'observation',\n {\n message: 'Test event recorded',\n },\n commandFrame\n );\n\n console.log('\u2705 Test frames created!');\n console.log(`\uD83D\uDCCA Stack depth: ${frameManager.getStackDepth()}`);\n console.log(\n `\uD83D\uDD04 Active frames: ${frameManager.getActiveFramePath().length}`\n );\n\n // Close one frame to test state changes\n frameManager.closeFrame(commandFrame);\n console.log(\n `\uD83D\uDCCA After closing command frame: depth = ${frameManager.getStackDepth()}`\n );\n\n db.close();\n } catch (error: unknown) {\n logger.error('Test context failed', error as Error);\n console.error('\u274C Test failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Register project management commands\n// Register command modules\nregisterOnboardingCommand(program);\nregisterProjectCommands(program);\nregisterWorktreeCommands(program);\n\n// Register Linear integration commands\nregisterLinearCommands(program);\nregisterLinearTestCommand(program);\nregisterLinearListCommand(program);\nregisterLinearMigrateCommand(program);\nregisterLinearCreateCommand(program);\n\n// Add ChromaDB command\nprogram.addCommand(createChromaDBCommand());\n\n// Add Infinite Storage command\nprogram.addCommand(createInfiniteStorageCommand());\nprogram.addCommand(createGCCommand());\n\n// Register session management commands\nprogram.addCommand(createSessionCommands());\n\n// Register webhook command\nprogram.addCommand(webhookCommand());\n\n// Register enhanced CLI commands\nprogram.addCommand(createTaskCommands());\nprogram.addCommand(createSearchCommand());\nprogram.addCommand(createLogCommand());\nprogram.addCommand(createContextCommands());\nprogram.addCommand(createConfigCommand());\nprogram.addCommand(createAgentCommand());\nprogram.addCommand(createHandoffCommand());\nprogram.addCommand(createStorageCommand());\nprogram.addCommand(createSkillsCommand());\nprogram.addCommand(createTestCommand());\nprogram.addCommand(clearCommand);\nprogram.addCommand(createWorkflowCommand());\nprogram.addCommand(monitorCommand);\nprogram.addCommand(qualityCommand);\n\n// Register dashboard command\nprogram\n .command('dashboard')\n .description('Display monitoring dashboard in terminal')\n .option('-w, --watch', 'Auto-refresh dashboard')\n .option('-i, --interval <seconds>', 'Refresh interval in seconds', '5')\n .action(async (options) => {\n const { dashboardCommand } = await import('./commands/dashboard.js');\n await dashboardCommand.handler(options);\n });\n\n// Register TUI command (advanced terminal UI - requires blessed)\nprogram\n .command('tui')\n .description('Launch interactive TUI monitoring dashboard (requires blessed)')\n .option('-s, --server', 'Start WebSocket server for real-time updates')\n .option('-w, --ws-url <url>', 'WebSocket server URL', 'ws://localhost:8080')\n .option('-r, --refresh <ms>', 'Auto-refresh interval in milliseconds', '2000')\n .action(async (options) => {\n try {\n // Check if blessed is installed by trying to import it\n try {\n await import('blessed');\n } catch {\n console.log(\n '\u274C The TUI requires the blessed package. Install it with:'\n );\n console.log(' npm install blessed blessed-contrib');\n console.log(\n '\\n\uD83D\uDCA1 Alternatively, use \"stackmemory dashboard\" for a simpler view'\n );\n process.exit(1);\n }\n\n const { spawn } = await import('child_process');\n const { fileURLToPath } = await import('url');\n const { dirname, join } = await import('path');\n\n const __filename = fileURLToPath(import.meta.url);\n const __dirname = dirname(__filename);\n\n console.log('\uD83D\uDE80 Launching StackMemory TUI Dashboard...');\n\n // Set environment variables\n process.env['STACKMEMORY_WS_URL'] = options.wsUrl;\n\n // Get TUI module path\n const tuiPath = join(__dirname, '../features/tui/index.js');\n\n // Launch TUI directly\n const tui = spawn('node', [tuiPath], {\n stdio: 'inherit',\n env: {\n ...process.env,\n STACKMEMORY_WS_URL: options.wsUrl,\n },\n });\n\n tui.on('error', (error) => {\n console.error('Failed to launch TUI:', error);\n console.log('\\n\uD83D\uDCA1 Try \"stackmemory dashboard\" instead');\n process.exit(1);\n });\n\n tui.on('exit', (code) => {\n if (code !== 0) {\n console.error(`TUI exited with code ${code}`);\n process.exit(code || 1);\n }\n });\n } catch (error: unknown) {\n console.error('\u274C Failed to launch TUI:', (error as Error).message);\n console.log('\\n\uD83D\uDCA1 Try \"stackmemory dashboard\" for a simpler view');\n process.exit(1);\n }\n });\n\n// Auto-detect current project on startup\nif (process.argv.length > 2) {\n const manager = ProjectManager.getInstance();\n manager.detectProject().catch(() => {\n // Silently fail if not in a project directory\n });\n}\n\n// Only parse when running as main module (not when imported for testing)\nconst isMainModule =\n import.meta.url === `file://${process.argv[1]}` ||\n process.argv[1]?.endsWith('/stackmemory') ||\n process.argv[1]?.endsWith('index.ts') ||\n process.argv[1]?.includes('tsx');\n\nif (isMainModule) {\n program.parse();\n}\n\nexport { program };\n"],
5
- "mappings": "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",
6
- "names": ["LinearSyncEngine", "DEFAULT_SYNC_CONFIG", "join"]
4
+ "sourcesContent": ["#!/usr/bin/env node\n/**\n * StackMemory CLI\n * Command-line interface for StackMemory operations\n */\n\n// Set environment flag for CLI usage to skip async context bridge\nprocess.env['STACKMEMORY_CLI'] = 'true';\n\n// Load environment variables\nimport 'dotenv/config';\n\n// Initialize tracing system early\nimport { initializeTracing, trace } from '../core/trace/index.js';\ninitializeTracing();\n\nimport { program } from 'commander';\nimport { logger } from '../core/monitoring/logger.js';\nimport { FrameManager } from '../core/context/frame-manager.js';\nimport { sessionManager, FrameQueryMode } from '../core/session/index.js';\nimport { sharedContextLayer } from '../core/context/shared-context-layer.js';\nimport { UpdateChecker } from '../core/utils/update-checker.js';\nimport { ProgressTracker } from '../core/monitoring/progress-tracker.js';\nimport { registerProjectCommands } from './commands/projects.js';\nimport { registerLinearCommands } from './commands/linear.js';\nimport { createSessionCommands } from './commands/session.js';\nimport { registerWorktreeCommands } from './commands/worktree.js';\nimport { registerOnboardingCommand } from './commands/onboard.js';\nimport { createTaskCommands } from './commands/tasks.js';\nimport { createSearchCommand } from './commands/search.js';\nimport { createLogCommand } from './commands/log.js';\nimport { createContextCommands } from './commands/context.js';\nimport { createConfigCommand } from './commands/config.js';\nimport { createHandoffCommand } from './commands/handoff.js';\nimport { createStorageCommand } from './commands/storage.js';\nimport { createSkillsCommand } from './commands/skills.js';\nimport { createTestCommand } from './commands/test.js';\nimport clearCommand from './commands/clear.js';\nimport createWorkflowCommand from './commands/workflow.js';\nimport monitorCommand from './commands/monitor.js';\nimport qualityCommand from './commands/quality.js';\nimport { registerLoginCommand } from './commands/login.js';\nimport { registerLogoutCommand, registerDbCommands } from './commands/db.js';\nimport { ProjectManager } from '../core/projects/project-manager.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\n\nconst VERSION = '0.3.17';\n\n// Check for updates on CLI startup\nUpdateChecker.checkForUpdates(VERSION, true).catch(() => {\n // Silently ignore errors\n});\n\nprogram\n .name('stackmemory')\n .description(\n 'Lossless memory runtime for AI coding tools - organizes context as a call stack instead of linear chat logs, with team collaboration and infinite retention'\n )\n .version(VERSION);\n\nprogram\n .command('init')\n .description('Initialize StackMemory in current project')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbDir = join(projectRoot, '.stackmemory');\n\n if (!existsSync(dbDir)) {\n mkdirSync(dbDir, { recursive: true });\n }\n\n const dbPath = join(dbDir, 'context.db');\n const db = new Database(dbPath);\n new FrameManager(db, 'cli-project');\n\n logger.info('StackMemory initialized successfully', { projectRoot });\n console.log('\u2705 StackMemory initialized in', projectRoot);\n\n db.close();\n } catch (error: unknown) {\n logger.error('Failed to initialize StackMemory', error as Error);\n console.error('\u274C Initialization failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('status')\n .description('Show current StackMemory status')\n .option('--all', 'Show all active frames across sessions')\n .option('--project', 'Show all active frames in current project')\n .option('--session <id>', 'Show frames for specific session')\n .action(async (options) => {\n return trace.command('stackmemory-status', options, async () => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n // Check for updates and display if available\n await UpdateChecker.checkForUpdates(VERSION);\n\n // Initialize session manager and shared context\n await sessionManager.initialize();\n await sharedContextLayer.initialize();\n\n const session = await sessionManager.getOrCreateSession({\n projectPath: projectRoot,\n sessionId: options.session,\n });\n\n // Auto-discover shared context on startup\n const contextDiscovery = await sharedContextLayer.autoDiscoverContext();\n\n // Show context hints if available\n if (\n contextDiscovery.hasSharedContext &&\n contextDiscovery.sessionCount > 1\n ) {\n console.log(`\\n\uD83D\uDCA1 Shared Context Available:`);\n console.log(\n ` ${contextDiscovery.sessionCount} sessions with shared context`\n );\n\n if (contextDiscovery.recentPatterns.length > 0) {\n console.log(` Recent patterns:`);\n contextDiscovery.recentPatterns.slice(0, 3).forEach((p) => {\n console.log(\n ` \u2022 ${p.type}: ${p.pattern.slice(0, 50)} (${p.frequency}x)`\n );\n });\n }\n\n if (contextDiscovery.lastDecisions.length > 0) {\n console.log(\n ` Last decision: ${contextDiscovery.lastDecisions[0].decision.slice(0, 60)}`\n );\n }\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db, session.projectId);\n\n // Set query mode based on options\n if (options.all) {\n frameManager.setQueryMode(FrameQueryMode.ALL_ACTIVE);\n } else if (options.project) {\n frameManager.setQueryMode(FrameQueryMode.PROJECT_ACTIVE);\n }\n\n const activeFrames = frameManager.getActiveFramePath();\n const stackDepth = frameManager.getStackDepth();\n\n // Always get total counts across all sessions\n const totalStats = db\n .prepare(\n `\n SELECT \n COUNT(*) as total_frames,\n SUM(CASE WHEN state = 'active' THEN 1 ELSE 0 END) as active_frames,\n SUM(CASE WHEN state = 'closed' THEN 1 ELSE 0 END) as closed_frames,\n COUNT(DISTINCT run_id) as total_sessions\n FROM frames\n WHERE project_id = ?\n `\n )\n .get(session.projectId) as {\n total_frames: number;\n active_frames: number;\n closed_frames: number;\n total_sessions: number;\n };\n\n const contextCount = db\n .prepare(\n `\n SELECT COUNT(*) as count FROM contexts\n `\n )\n .get() as { count: number };\n\n const eventCount = db\n .prepare(\n `\n SELECT COUNT(*) as count FROM events e\n JOIN frames f ON e.frame_id = f.frame_id\n WHERE f.project_id = ?\n `\n )\n .get(session.projectId) as { count: number };\n\n console.log('\uD83D\uDCCA StackMemory Status:');\n console.log(\n ` Session: ${session.sessionId.slice(0, 8)} (${session.state}, ${Math.round((Date.now() - session.startedAt) / 1000 / 60)}min old)`\n );\n console.log(` Project: ${session.projectId}`);\n if (session.branch) {\n console.log(` Branch: ${session.branch}`);\n }\n\n // Show total database statistics\n console.log(`\\n Database Statistics (this project):`);\n console.log(\n ` Frames: ${totalStats.total_frames || 0} (${totalStats.active_frames || 0} active, ${totalStats.closed_frames || 0} closed)`\n );\n console.log(` Events: ${eventCount.count || 0}`);\n console.log(` Sessions: ${totalStats.total_sessions || 0}`);\n console.log(\n ` Cached contexts: ${contextCount.count || 0} (global)`\n );\n\n // Show recent activity\n const recentFrames = db\n .prepare(\n `\n SELECT name, type, state, datetime(created_at, 'unixepoch') as created\n FROM frames\n WHERE project_id = ?\n ORDER BY created_at DESC\n LIMIT 3\n `\n )\n .all(session.projectId) as Array<{\n name: string;\n type: string;\n state: string;\n created: string;\n }>;\n\n if (recentFrames.length > 0) {\n console.log(`\\n Recent Activity:`);\n recentFrames.forEach((f) => {\n const stateIcon = f.state === 'active' ? '\uD83D\uDFE2' : '\u26AB';\n console.log(\n ` ${stateIcon} ${f.name} [${f.type}] - ${f.created}`\n );\n });\n }\n\n console.log(`\\n Current Session:`);\n console.log(` Stack depth: ${stackDepth}`);\n console.log(` Active frames: ${activeFrames.length}`);\n\n if (activeFrames.length > 0) {\n activeFrames.forEach((frame, i) => {\n const indent = ' ' + ' '.repeat(frame.depth || i);\n const prefix = i === 0 ? '\u2514\u2500' : ' \u2514\u2500';\n console.log(`${indent}${prefix} ${frame.name} [${frame.type}]`);\n });\n }\n\n // Show other sessions if in default mode\n if (!options.all && !options.project) {\n const otherSessions = await sessionManager.listSessions({\n projectId: session.projectId,\n state: 'active',\n });\n\n const otherActive = otherSessions.filter(\n (s) => s.sessionId !== session.sessionId\n );\n if (otherActive.length > 0) {\n console.log(`\\n Other Active Sessions (same project):`);\n otherActive.forEach((s) => {\n const age = Math.round(\n (Date.now() - s.lastActiveAt) / 1000 / 60 / 60\n );\n console.log(\n ` - ${s.sessionId.slice(0, 8)}: ${s.branch || 'main'}, ${age}h old`\n );\n });\n console.log(`\\n Tip: Use --all to see frames across sessions`);\n }\n }\n\n db.close();\n } catch (error: unknown) {\n logger.error('Failed to get status', error as Error);\n console.error('\u274C Status check failed:', (error as Error).message);\n process.exit(1);\n }\n });\n });\n\nprogram\n .command('update-check')\n .description('Check for StackMemory updates')\n .action(async () => {\n try {\n console.log('\uD83D\uDD0D Checking for updates...');\n await UpdateChecker.forceCheck(VERSION);\n } catch (error: unknown) {\n logger.error('Update check failed', error as Error);\n console.error('\u274C Update check failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('progress')\n .description('Show current progress and recent changes')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const progress = new ProgressTracker(projectRoot);\n console.log(progress.getSummary());\n } catch (error: unknown) {\n logger.error('Failed to show progress', error as Error);\n console.error('\u274C Failed to show progress:', (error as Error).message);\n process.exit(1);\n }\n });\n\nprogram\n .command('mcp-server')\n .description('Start StackMemory MCP server for Claude Desktop')\n .option('-p, --project <path>', 'Project root directory', process.cwd())\n .action(async (options) => {\n try {\n const { runMCPServer } = await import('../integrations/mcp/server.js');\n\n // Set project root\n process.env['PROJECT_ROOT'] = options.project;\n\n console.log('\uD83D\uDE80 Starting StackMemory MCP Server...');\n console.log(` Project: ${options.project}`);\n console.log(` Version: ${VERSION}`);\n\n // Check for updates silently\n UpdateChecker.checkForUpdates(VERSION, true).catch(() => {});\n\n // Start the MCP server\n await runMCPServer();\n } catch (error: unknown) {\n logger.error('Failed to start MCP server', error as Error);\n console.error('\u274C MCP server failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Add test context command\nprogram\n .command('context:test')\n .description('Test context persistence by creating sample frames')\n .action(async () => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db, 'cli-project');\n\n // Create test frames\n console.log('\uD83D\uDCDD Creating test context frames...');\n\n const rootFrame = frameManager.createFrame({\n type: 'task',\n name: 'Test Session',\n inputs: { test: true, timestamp: new Date().toISOString() },\n });\n\n const taskFrame = frameManager.createFrame({\n type: 'subtask',\n name: 'Sample Task',\n inputs: { description: 'Testing context persistence' },\n parentFrameId: rootFrame,\n });\n\n const commandFrame = frameManager.createFrame({\n type: 'tool_scope',\n name: 'test-command',\n inputs: { args: ['--test'] },\n parentFrameId: taskFrame,\n });\n\n // Add some events\n frameManager.addEvent(\n 'observation',\n {\n message: 'Test event recorded',\n },\n commandFrame\n );\n\n console.log('\u2705 Test frames created!');\n console.log(`\uD83D\uDCCA Stack depth: ${frameManager.getStackDepth()}`);\n console.log(\n `\uD83D\uDD04 Active frames: ${frameManager.getActiveFramePath().length}`\n );\n\n // Close one frame to test state changes\n frameManager.closeFrame(commandFrame);\n console.log(\n `\uD83D\uDCCA After closing command frame: depth = ${frameManager.getStackDepth()}`\n );\n\n db.close();\n } catch (error: unknown) {\n logger.error('Test context failed', error as Error);\n console.error('\u274C Test failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// Register project management commands\n// Register command modules\nregisterOnboardingCommand(program);\nregisterLoginCommand(program);\nregisterLogoutCommand(program);\nregisterDbCommands(program);\nregisterProjectCommands(program);\nregisterWorktreeCommands(program);\n\n// Register Linear integration commands\nregisterLinearCommands(program);\n\n// Register session management commands\nprogram.addCommand(createSessionCommands());\n\n// Register enhanced CLI commands\nprogram.addCommand(createTaskCommands());\nprogram.addCommand(createSearchCommand());\nprogram.addCommand(createLogCommand());\nprogram.addCommand(createContextCommands());\nprogram.addCommand(createConfigCommand());\nprogram.addCommand(createHandoffCommand());\nprogram.addCommand(createStorageCommand());\nprogram.addCommand(createSkillsCommand());\nprogram.addCommand(createTestCommand());\nprogram.addCommand(clearCommand);\nprogram.addCommand(createWorkflowCommand());\nprogram.addCommand(monitorCommand);\nprogram.addCommand(qualityCommand);\n\n// Register dashboard command\nprogram\n .command('dashboard')\n .description('Display monitoring dashboard in terminal')\n .option('-w, --watch', 'Auto-refresh dashboard')\n .option('-i, --interval <seconds>', 'Refresh interval in seconds', '5')\n .action(async (options) => {\n const { dashboardCommand } = await import('./commands/dashboard.js');\n await dashboardCommand.handler(options);\n });\n\n// Auto-detect current project on startup\nif (process.argv.length > 2) {\n const manager = ProjectManager.getInstance();\n manager.detectProject().catch(() => {\n // Silently fail if not in a project directory\n });\n}\n\n// Only parse when running as main module (not when imported for testing)\nconst isMainModule =\n import.meta.url === `file://${process.argv[1]}` ||\n process.argv[1]?.endsWith('/stackmemory') ||\n process.argv[1]?.endsWith('index.ts') ||\n process.argv[1]?.includes('tsx');\n\nif (isMainModule) {\n program.parse();\n}\n\nexport { program };\n"],
5
+ "mappings": "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",
6
+ "names": []
7
7
  }
@@ -111,7 +111,7 @@ class DualStackManager {
111
111
  }
112
112
  }
113
113
  async executeSchemaQuery(sql) {
114
- logger.warn(
114
+ logger.debug(
115
115
  "Using fallback schema creation - implement execute method in adapter"
116
116
  );
117
117
  const rawDb = this.adapter instanceof SQLiteAdapter ? this.adapter.getRawDatabase() : null;
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/context/dual-stack-manager.ts"],
4
- "sourcesContent": ["/**\n * Dual Stack Manager - STA-99\n * Manages both individual and shared team stacks for collaboration\n */\n\nimport type { Frame, Event, Anchor } from './frame-manager.js';\nimport { FrameManager } from './frame-manager.js';\nimport type { DatabaseAdapter } from '../database/database-adapter.js';\nimport { SQLiteAdapter } from '../database/sqlite-adapter.js';\nimport { logger } from '../monitoring/logger.js';\nimport { ValidationError, DatabaseError, ErrorCode } from '../errors/index.js';\nimport {\n validateInput,\n CreateSharedStackSchema,\n SwitchStackSchema,\n type CreateSharedStackInput,\n type SwitchStackInput,\n} from './validation.js';\nimport { PermissionManager } from './permission-manager.js';\n\nexport interface StackContext {\n stackId: string;\n type: 'individual' | 'shared';\n projectId: string;\n ownerId?: string; // For individual stacks\n teamId?: string; // For shared stacks\n permissions: StackPermissions;\n metadata: Record<string, any>;\n createdAt: Date;\n lastActive: Date;\n}\n\nexport interface StackPermissions {\n canRead: boolean;\n canWrite: boolean;\n canHandoff: boolean;\n canMerge: boolean;\n canAdminister: boolean;\n}\n\nexport interface HandoffRequest {\n requestId: string;\n sourceStackId: string;\n targetStackId: string;\n frameIds: string[];\n requesterId: string;\n targetUserId?: string;\n message?: string;\n status: 'pending' | 'accepted' | 'rejected' | 'expired';\n createdAt: Date;\n expiresAt: Date;\n}\n\nexport interface StackSyncResult {\n success: boolean;\n conflictFrames: string[];\n mergedFrames: string[];\n errors: Array<{\n frameId: string;\n error: string;\n resolution?: 'skipped' | 'merged' | 'manual';\n }>;\n}\n\nexport class DualStackManager {\n private adapter: DatabaseAdapter;\n private individualStack: FrameManager;\n private sharedStacks: Map<string, FrameManager> = new Map();\n private activeContext: StackContext;\n private handoffRequests: Map<string, HandoffRequest> = new Map();\n private permissionManager: PermissionManager;\n\n constructor(\n adapter: DatabaseAdapter,\n projectId: string,\n userId: string,\n defaultTeamId?: string\n ) {\n this.adapter = adapter;\n this.permissionManager = new PermissionManager();\n\n // Initialize individual stack\n // Extract raw database for FrameManager which expects SQLite directly\n const rawDb =\n adapter instanceof SQLiteAdapter ? adapter.getRawDatabase() : null;\n if (!rawDb) {\n throw new Error(\n 'DualStackManager requires SQLiteAdapter with connected database'\n );\n }\n\n this.individualStack = new FrameManager(rawDb, projectId, userId);\n\n // Set default active context to individual stack\n this.activeContext = {\n stackId: `individual-${userId}`,\n type: 'individual',\n projectId,\n ownerId: userId,\n permissions: this.getDefaultIndividualPermissions(),\n metadata: {},\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n // Set up initial permissions for the user's individual stack\n this.permissionManager.setStackPermissions(\n userId,\n `individual-${userId}`,\n this.getDefaultIndividualPermissions()\n );\n\n this.initializeSchema();\n }\n\n private async initializeSchema(): Promise<void> {\n try {\n // Create stack_contexts table\n await this.adapter.beginTransaction();\n\n const createStackContextsTable = `\n CREATE TABLE IF NOT EXISTS stack_contexts (\n stack_id TEXT PRIMARY KEY,\n type TEXT NOT NULL CHECK (type IN ('individual', 'shared')),\n project_id TEXT NOT NULL,\n owner_id TEXT,\n team_id TEXT,\n permissions TEXT NOT NULL,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL,\n last_active INTEGER NOT NULL,\n CONSTRAINT valid_ownership CHECK (\n (type = 'individual' AND owner_id IS NOT NULL AND team_id IS NULL) OR\n (type = 'shared' AND team_id IS NOT NULL)\n )\n )\n `;\n\n const createHandoffRequestsTable = `\n CREATE TABLE IF NOT EXISTS handoff_requests (\n request_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n frame_ids TEXT NOT NULL,\n requester_id TEXT NOT NULL,\n target_user_id TEXT,\n message TEXT,\n status TEXT NOT NULL DEFAULT 'pending' CHECK (status IN ('pending', 'accepted', 'rejected', 'expired')),\n created_at INTEGER NOT NULL,\n expires_at INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n const createStackSyncLogTable = `\n CREATE TABLE IF NOT EXISTS stack_sync_log (\n sync_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n operation TEXT NOT NULL CHECK (operation IN ('handoff', 'merge', 'sync')),\n frame_count INTEGER NOT NULL,\n conflicts TEXT DEFAULT '[]',\n resolution TEXT,\n timestamp INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n // Execute schema creation using raw SQL\n if (this.adapter.isConnected()) {\n // Note: This is a temporary workaround - proper schema creation would use adapter methods\n (await (this.adapter as any).execute?.(createStackContextsTable)) ||\n this.executeSchemaQuery(createStackContextsTable);\n (await (this.adapter as any).execute?.(createHandoffRequestsTable)) ||\n this.executeSchemaQuery(createHandoffRequestsTable);\n (await (this.adapter as any).execute?.(createStackSyncLogTable)) ||\n this.executeSchemaQuery(createStackSyncLogTable);\n }\n\n await this.adapter.commitTransaction();\n\n logger.info('Dual stack schema initialized successfully');\n } catch (error: unknown) {\n await this.adapter.rollbackTransaction();\n logger.error('Failed to initialize dual stack schema', error);\n throw new DatabaseError(\n 'Schema initialization failed',\n ErrorCode.DB_SCHEMA_ERROR,\n { adapter: this.adapter.constructor.name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n private async executeSchemaQuery(sql: string): Promise<void> {\n // Fallback for adapters that don't have execute method\n logger.warn(\n 'Using fallback schema creation - implement execute method in adapter'\n );\n\n // Execute using raw SQLite database\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n try {\n rawDb.exec(sql);\n logger.debug('Executed schema query successfully');\n } catch (error: unknown) {\n logger.error('Failed to execute schema query', { sql, error });\n throw error;\n }\n } else {\n throw new Error(\n 'Cannot execute schema query: raw database not available'\n );\n }\n }\n\n private getDefaultIndividualPermissions(): StackPermissions {\n return {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: true,\n canAdminister: true,\n };\n }\n\n private getSharedStackPermissions(\n role: 'member' | 'lead' | 'admin'\n ): StackPermissions {\n const basePermissions = {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: false,\n canAdminister: false,\n };\n\n switch (role) {\n case 'lead':\n return { ...basePermissions, canMerge: true };\n case 'admin':\n return { ...basePermissions, canMerge: true, canAdminister: true };\n default:\n return basePermissions;\n }\n }\n\n /**\n * Switch between individual and shared stacks\n */\n async switchToStack(stackId: string): Promise<void> {\n // Validate input\n const input = validateInput(SwitchStackSchema, { stackId });\n\n try {\n if (input.stackId.startsWith('individual-')) {\n this.activeContext = {\n ...this.activeContext,\n stackId: input.stackId,\n type: 'individual',\n };\n return;\n }\n\n // Load shared stack context\n const stackContext = await this.loadStackContext(input.stackId);\n if (!stackContext) {\n throw new ValidationError(\n `Stack context not found: ${input.stackId}`,\n ErrorCode.STACK_CONTEXT_NOT_FOUND\n );\n }\n\n // Check permission to access the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n input.stackId,\n stackContext\n )\n );\n\n this.activeContext = stackContext;\n\n // Initialize shared stack manager if not already loaded\n if (!this.sharedStacks.has(input.stackId)) {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('Failed to get raw database for shared stack');\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n input.stackId\n );\n this.sharedStacks.set(input.stackId, sharedStack);\n }\n\n // Update last active timestamp\n await this.updateStackActivity(input.stackId);\n\n logger.info(`Switched to stack: ${input.stackId}`, {\n type: stackContext.type,\n });\n } catch (error: unknown) {\n throw new ValidationError(\n `Failed to switch to stack: ${input.stackId}`,\n ErrorCode.OPERATION_FAILED,\n { stackId: input.stackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get the current active stack manager\n */\n getActiveStack(): FrameManager {\n if (this.activeContext.type === 'individual') {\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(this.activeContext.stackId);\n if (!sharedStack) {\n throw new DatabaseError(\n `Active shared stack not initialized: ${this.activeContext.stackId}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n return sharedStack;\n }\n\n /**\n * Create a new shared stack for team collaboration\n */\n async createSharedStack(\n teamId: string,\n name: string,\n ownerId: string,\n permissions?: StackPermissions\n ): Promise<string> {\n // Validate input parameters\n const input = validateInput(CreateSharedStackSchema, {\n teamId,\n name,\n ownerId,\n permissions,\n });\n\n // Check permission to create shared stacks\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n input.ownerId,\n 'administer',\n 'stack',\n `shared-${input.teamId}`,\n this.activeContext\n )\n );\n\n const stackId = `shared-${input.teamId}-${Date.now()}`;\n\n const stackContext: StackContext = {\n stackId,\n type: 'shared',\n projectId: this.activeContext.projectId,\n teamId: input.teamId,\n permissions: input.permissions || this.getSharedStackPermissions('admin'),\n metadata: { name: input.name, ownerId: input.ownerId },\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n try {\n await this.saveStackContext(stackContext);\n\n // Initialize the shared stack manager\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('Failed to get raw database for new shared stack');\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n stackId\n );\n this.sharedStacks.set(stackId, sharedStack);\n\n // Set up permissions for the owner and team\n const stackPermissions = stackContext.permissions;\n this.permissionManager.setStackPermissions(\n input.ownerId,\n stackId,\n stackPermissions\n );\n\n logger.info(`Created shared stack: ${stackId}`, { teamId, name });\n return stackId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to create shared stack`,\n ErrorCode.OPERATION_FAILED,\n { teamId, name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Initiate handoff of frames between stacks\n */\n async initiateHandoff(\n targetStackId: string,\n frameIds: string[],\n targetUserId?: string,\n message?: string\n ): Promise<string> {\n // Check permission to perform handoff from current stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'handoff',\n 'stack',\n this.activeContext.stackId,\n this.activeContext\n )\n );\n\n const requestId = `handoff-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n\n const request: HandoffRequest = {\n requestId,\n sourceStackId: this.activeContext.stackId,\n targetStackId,\n frameIds,\n requesterId: this.activeContext.ownerId!,\n targetUserId,\n message,\n status: 'pending',\n createdAt: new Date(),\n expiresAt: new Date(Date.now() + 24 * 60 * 60 * 1000), // 24 hours\n };\n\n try {\n await this.saveHandoffRequest(request);\n this.handoffRequests.set(requestId, request);\n\n logger.info(`Initiated handoff request: ${requestId}`, {\n sourceStack: this.activeContext.stackId,\n targetStack: targetStackId,\n frameCount: frameIds.length,\n });\n\n return requestId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to initiate handoff`,\n ErrorCode.OPERATION_FAILED,\n { targetStackId, frameIds },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Accept a handoff request and move frames\n */\n async acceptHandoff(requestId: string): Promise<StackSyncResult> {\n logger.debug('acceptHandoff called', { requestId });\n const request = await this.loadHandoffRequest(requestId);\n logger.debug('loadHandoffRequest returned', {\n requestId,\n found: !!request,\n });\n if (!request) {\n logger.error('Handoff request not found', {\n requestId,\n availableRequests: Array.from(this.handoffRequests.keys()),\n });\n throw new DatabaseError(\n `Handoff request not found: ${requestId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n if (request.status !== 'pending') {\n throw new DatabaseError(\n `Handoff request is not pending: ${request.status}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n if (request.expiresAt < new Date()) {\n throw new DatabaseError(\n `Handoff request has expired`,\n ErrorCode.OPERATION_EXPIRED\n );\n }\n\n try {\n // Perform the handoff operation\n logger.debug('Starting moveFramesBetweenStacks', { requestId });\n const syncResult = await this.moveFramesBetweenStacks(\n request.sourceStackId,\n request.targetStackId,\n request.frameIds\n );\n logger.debug('moveFramesBetweenStacks completed', {\n requestId,\n success: syncResult.success,\n });\n\n // Update request status\n logger.debug('Updating request status', { requestId });\n request.status = 'accepted';\n logger.debug('Calling saveHandoffRequest', { requestId });\n await this.saveHandoffRequest(request);\n logger.debug('saveHandoffRequest completed', { requestId });\n\n logger.info(`Accepted handoff request: ${requestId}`, {\n frameCount: request.frameIds.length,\n conflicts: syncResult.conflictFrames.length,\n });\n\n return syncResult;\n } catch (error: unknown) {\n logger.error('acceptHandoff caught error', {\n error: error instanceof Error ? error.message : error,\n });\n // Update request status to rejected on failure\n request.status = 'rejected';\n await this.saveHandoffRequest(request);\n\n throw new DatabaseError(\n `Failed to accept handoff`,\n ErrorCode.OPERATION_FAILED,\n { requestId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Sync frames between individual and shared stacks\n */\n async syncStacks(\n sourceStackId: string,\n targetStackId: string,\n options: {\n frameIds?: string[];\n conflictResolution: 'skip' | 'merge' | 'overwrite';\n dryRun?: boolean;\n }\n ): Promise<StackSyncResult> {\n try {\n const sourceStack = this.getStackManager(sourceStackId);\n const targetStack = this.getStackManager(targetStackId);\n\n // Get frames to sync\n const framesToSync =\n options.frameIds ||\n (await sourceStack.getActiveFrames()).map((f) => f.frame_id);\n\n const result: StackSyncResult = {\n success: true,\n conflictFrames: [],\n mergedFrames: [],\n errors: [],\n };\n\n for (const frameId of framesToSync) {\n try {\n const sourceFrame = await sourceStack.getFrame(frameId);\n if (!sourceFrame) {\n result.errors.push({\n frameId,\n error: 'Source frame not found',\n resolution: 'skipped',\n });\n continue;\n }\n\n const existingFrame = await targetStack.getFrame(frameId);\n\n if (existingFrame) {\n // Handle conflict\n switch (options.conflictResolution) {\n case 'skip':\n result.conflictFrames.push(frameId);\n result.errors.push({\n frameId,\n error: 'Frame already exists',\n resolution: 'skipped',\n });\n continue;\n\n case 'merge':\n if (!options.dryRun) {\n await this.mergeFrames(\n existingFrame,\n sourceFrame,\n targetStack\n );\n }\n result.mergedFrames.push(frameId);\n break;\n\n case 'overwrite':\n if (!options.dryRun) {\n await targetStack.deleteFrame(frameId);\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n break;\n }\n } else {\n // Copy frame to target\n if (!options.dryRun) {\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n }\n } catch (error: unknown) {\n result.errors.push({\n frameId,\n error: error instanceof Error ? error.message : String(error),\n resolution: 'skipped',\n });\n result.success = false;\n }\n }\n\n logger.info(`Stack sync completed`, {\n source: sourceStackId,\n target: targetStackId,\n merged: result.mergedFrames.length,\n conflicts: result.conflictFrames.length,\n errors: result.errors.length,\n });\n\n return result;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Stack sync failed`,\n ErrorCode.OPERATION_FAILED,\n { sourceStackId, targetStackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n getStackManager(stackId: string): FrameManager {\n logger.debug('getStackManager called', {\n stackId,\n availableStacks: Array.from(this.sharedStacks.keys()),\n });\n\n if (stackId.startsWith('individual-')) {\n logger.debug('Returning individual stack', { stackId });\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(stackId);\n if (!sharedStack) {\n logger.error('Stack manager not found', {\n stackId,\n availableSharedStacks: Array.from(this.sharedStacks.keys()),\n message: 'getStackManager could not find shared stack',\n });\n throw new DatabaseError(\n `Stack manager not found: ${stackId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n logger.debug('Returning shared stack', { stackId });\n return sharedStack;\n }\n\n private async moveFramesBetweenStacks(\n sourceStackId: string,\n targetStackId: string,\n frameIds: string[]\n ): Promise<StackSyncResult> {\n const syncResult = await this.syncStacks(sourceStackId, targetStackId, {\n frameIds,\n conflictResolution: 'merge',\n });\n\n // Remove frames from source stack after successful sync\n if (syncResult.success && syncResult.errors.length === 0) {\n const sourceStack = this.getStackManager(sourceStackId);\n for (const frameId of frameIds) {\n try {\n sourceStack.deleteFrame(frameId);\n logger.debug('Deleted frame from source stack', {\n frameId,\n sourceStackId,\n });\n } catch (error: unknown) {\n logger.warn('Failed to delete frame from source stack', {\n frameId,\n error,\n });\n }\n }\n logger.debug('Completed frame cleanup from source stack', {\n frameIds: frameIds.length,\n });\n }\n\n return syncResult;\n }\n\n private async copyFrame(\n frame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Create frame in target stack\n await targetStack.createFrame({\n type: frame.type as any,\n name: frame.name,\n inputs: frame.inputs,\n });\n\n // Copy events\n const events = await this.individualStack.getFrameEvents(frame.frame_id);\n for (const event of events) {\n await targetStack.addEvent(frame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: event.metadata,\n });\n }\n\n // Copy anchors\n const anchors = await this.individualStack.getFrameAnchors(frame.frame_id);\n for (const anchor of anchors) {\n await targetStack.addAnchor(frame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: anchor.metadata,\n });\n }\n }\n\n private async mergeFrames(\n existingFrame: Frame,\n sourceFrame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Simple merge strategy - append events and anchors\n const sourceEvents = await this.individualStack.getFrameEvents(\n sourceFrame.frame_id\n );\n for (const event of sourceEvents) {\n await targetStack.addEvent(existingFrame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: { ...event.metadata, merged: true },\n });\n }\n\n const sourceAnchors = await this.individualStack.getFrameAnchors(\n sourceFrame.frame_id\n );\n for (const anchor of sourceAnchors) {\n await targetStack.addAnchor(existingFrame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: { ...anchor.metadata, merged: true },\n });\n }\n }\n\n private async loadStackContext(\n stackId: string\n ): Promise<StackContext | null> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n return null;\n }\n\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE stack_id = ?\n `);\n\n const row = query.get(stackId) as any;\n if (!row) {\n return null;\n }\n\n return {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n } catch (error: unknown) {\n logger.error('Failed to load stack context', { stackId, error });\n return null;\n }\n }\n\n private async saveStackContext(context: StackContext): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('SQLite database not available for stack context save');\n }\n\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO stack_contexts \n (stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n context.stackId,\n context.type,\n context.projectId,\n context.ownerId || null,\n context.teamId || null,\n JSON.stringify(context.permissions),\n JSON.stringify(context.metadata || {}),\n context.createdAt.getTime(),\n context.lastActive.getTime()\n );\n\n logger.debug('Saved stack context', { stackId: context.stackId });\n } catch (error: unknown) {\n logger.error('Failed to save stack context', {\n stackId: context.stackId,\n error,\n });\n throw error;\n }\n }\n\n private async updateStackActivity(stackId: string): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n logger.warn('SQLite database not available for activity update');\n return;\n }\n\n const query = rawDb.prepare(`\n UPDATE stack_contexts \n SET last_active = ?\n WHERE stack_id = ?\n `);\n\n query.run(Date.now(), stackId);\n logger.debug('Updated stack activity', { stackId });\n } catch (error: unknown) {\n logger.error('Failed to update stack activity', { stackId, error });\n // Don't throw - activity updates are not critical\n }\n }\n\n private async loadHandoffRequest(\n requestId: string\n ): Promise<HandoffRequest | null> {\n // Try in-memory first for fast access\n const memoryRequest = this.handoffRequests.get(requestId);\n if (memoryRequest) {\n return memoryRequest;\n }\n\n // Try loading from database\n try {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT * FROM handoff_requests WHERE request_id = ?\n `);\n\n const row = query.get(requestId) as any;\n if (row) {\n const request: HandoffRequest = {\n requestId: row.request_id,\n sourceStackId: row.source_stack_id,\n targetStackId: row.target_stack_id,\n frameIds: JSON.parse(row.frame_ids),\n status: row.status,\n createdAt: new Date(row.created_at),\n expiresAt: new Date(row.expires_at),\n targetUserId: row.target_user_id,\n message: row.message,\n };\n\n // Cache in memory for future access\n this.handoffRequests.set(requestId, request);\n return request;\n }\n }\n } catch (error: unknown) {\n logger.error('Failed to load handoff request from database', {\n requestId,\n error,\n });\n }\n\n return null;\n }\n\n private async saveHandoffRequest(request: HandoffRequest): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO handoff_requests \n (request_id, source_stack_id, target_stack_id, frame_ids, status, created_at, expires_at, target_user_id, message)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n request.requestId,\n request.sourceStackId,\n request.targetStackId,\n JSON.stringify(request.frameIds),\n request.status,\n request.createdAt.getTime(),\n request.expiresAt.getTime(),\n request.targetUserId || null,\n request.message || null\n );\n\n logger.debug('Saved handoff request to database', {\n requestId: request.requestId,\n });\n }\n\n // Also keep in-memory for fast access\n this.handoffRequests.set(request.requestId, request);\n } catch (error: unknown) {\n logger.error('Failed to save handoff request', {\n requestId: request.requestId,\n error,\n });\n // Fallback to in-memory only\n this.handoffRequests.set(request.requestId, request);\n }\n }\n\n /**\n * Get list of available stacks for the current user\n */\n async getAvailableStacks(): Promise<StackContext[]> {\n try {\n const stacks: StackContext[] = [];\n\n // Always include current individual stack context\n stacks.push(this.activeContext);\n\n // Query database for all shared stacks the user has access to\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE type = 'shared' AND project_id = ?\n `);\n\n const rows = query.all(this.activeContext.projectId) as any[];\n\n for (const row of rows) {\n const context: StackContext = {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n\n // Check if user has permission to access this stack\n try {\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n context.stackId,\n context\n )\n );\n stacks.push(context);\n } catch (permissionError: unknown) {\n // Skip stacks user doesn't have access to\n logger.debug('User lacks access to stack', {\n stackId: context.stackId,\n userId: this.activeContext.ownerId,\n });\n }\n }\n }\n\n return stacks;\n } catch (error: unknown) {\n logger.error('Failed to get available stacks', error);\n // Return at least the current individual stack\n return [this.activeContext];\n }\n }\n\n /**\n * Get pending handoff requests for the current user\n */\n async getPendingHandoffRequests(): Promise<HandoffRequest[]> {\n return Array.from(this.handoffRequests.values()).filter(\n (request) =>\n request.status === 'pending' && request.expiresAt > new Date()\n );\n }\n\n /**\n * Get current stack context\n */\n getCurrentContext(): StackContext {\n return { ...this.activeContext };\n }\n\n /**\n * Get permission manager for external access\n */\n getPermissionManager(): PermissionManager {\n return this.permissionManager;\n }\n\n /**\n * Add user to shared stack with specific permissions\n */\n async addUserToSharedStack(\n stackId: string,\n userId: string,\n permissions: StackPermissions,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Grant permissions to the new user\n this.permissionManager.setStackPermissions(userId, stackId, permissions);\n\n logger.info(`Added user to shared stack`, {\n stackId,\n userId,\n permissions,\n requesterId,\n });\n }\n\n /**\n * Remove user from shared stack\n */\n async removeUserFromSharedStack(\n stackId: string,\n userId: string,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Remove user's permissions\n const userPermissions = this.permissionManager.getUserPermissions(userId);\n userPermissions.delete(stackId);\n\n logger.info(`Removed user from shared stack`, {\n stackId,\n userId,\n requesterId,\n });\n }\n}\n"],
4
+ "sourcesContent": ["/**\n * Dual Stack Manager - STA-99\n * Manages both individual and shared team stacks for collaboration\n */\n\nimport type { Frame, Event, Anchor } from './frame-manager.js';\nimport { FrameManager } from './frame-manager.js';\nimport type { DatabaseAdapter } from '../database/database-adapter.js';\nimport { SQLiteAdapter } from '../database/sqlite-adapter.js';\nimport { logger } from '../monitoring/logger.js';\nimport { ValidationError, DatabaseError, ErrorCode } from '../errors/index.js';\nimport {\n validateInput,\n CreateSharedStackSchema,\n SwitchStackSchema,\n type CreateSharedStackInput,\n type SwitchStackInput,\n} from './validation.js';\nimport { PermissionManager } from './permission-manager.js';\n\nexport interface StackContext {\n stackId: string;\n type: 'individual' | 'shared';\n projectId: string;\n ownerId?: string; // For individual stacks\n teamId?: string; // For shared stacks\n permissions: StackPermissions;\n metadata: Record<string, any>;\n createdAt: Date;\n lastActive: Date;\n}\n\nexport interface StackPermissions {\n canRead: boolean;\n canWrite: boolean;\n canHandoff: boolean;\n canMerge: boolean;\n canAdminister: boolean;\n}\n\nexport interface HandoffRequest {\n requestId: string;\n sourceStackId: string;\n targetStackId: string;\n frameIds: string[];\n requesterId: string;\n targetUserId?: string;\n message?: string;\n status: 'pending' | 'accepted' | 'rejected' | 'expired';\n createdAt: Date;\n expiresAt: Date;\n}\n\nexport interface StackSyncResult {\n success: boolean;\n conflictFrames: string[];\n mergedFrames: string[];\n errors: Array<{\n frameId: string;\n error: string;\n resolution?: 'skipped' | 'merged' | 'manual';\n }>;\n}\n\nexport class DualStackManager {\n private adapter: DatabaseAdapter;\n private individualStack: FrameManager;\n private sharedStacks: Map<string, FrameManager> = new Map();\n private activeContext: StackContext;\n private handoffRequests: Map<string, HandoffRequest> = new Map();\n private permissionManager: PermissionManager;\n\n constructor(\n adapter: DatabaseAdapter,\n projectId: string,\n userId: string,\n defaultTeamId?: string\n ) {\n this.adapter = adapter;\n this.permissionManager = new PermissionManager();\n\n // Initialize individual stack\n // Extract raw database for FrameManager which expects SQLite directly\n const rawDb =\n adapter instanceof SQLiteAdapter ? adapter.getRawDatabase() : null;\n if (!rawDb) {\n throw new Error(\n 'DualStackManager requires SQLiteAdapter with connected database'\n );\n }\n\n this.individualStack = new FrameManager(rawDb, projectId, userId);\n\n // Set default active context to individual stack\n this.activeContext = {\n stackId: `individual-${userId}`,\n type: 'individual',\n projectId,\n ownerId: userId,\n permissions: this.getDefaultIndividualPermissions(),\n metadata: {},\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n // Set up initial permissions for the user's individual stack\n this.permissionManager.setStackPermissions(\n userId,\n `individual-${userId}`,\n this.getDefaultIndividualPermissions()\n );\n\n this.initializeSchema();\n }\n\n private async initializeSchema(): Promise<void> {\n try {\n // Create stack_contexts table\n await this.adapter.beginTransaction();\n\n const createStackContextsTable = `\n CREATE TABLE IF NOT EXISTS stack_contexts (\n stack_id TEXT PRIMARY KEY,\n type TEXT NOT NULL CHECK (type IN ('individual', 'shared')),\n project_id TEXT NOT NULL,\n owner_id TEXT,\n team_id TEXT,\n permissions TEXT NOT NULL,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL,\n last_active INTEGER NOT NULL,\n CONSTRAINT valid_ownership CHECK (\n (type = 'individual' AND owner_id IS NOT NULL AND team_id IS NULL) OR\n (type = 'shared' AND team_id IS NOT NULL)\n )\n )\n `;\n\n const createHandoffRequestsTable = `\n CREATE TABLE IF NOT EXISTS handoff_requests (\n request_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n frame_ids TEXT NOT NULL,\n requester_id TEXT NOT NULL,\n target_user_id TEXT,\n message TEXT,\n status TEXT NOT NULL DEFAULT 'pending' CHECK (status IN ('pending', 'accepted', 'rejected', 'expired')),\n created_at INTEGER NOT NULL,\n expires_at INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n const createStackSyncLogTable = `\n CREATE TABLE IF NOT EXISTS stack_sync_log (\n sync_id TEXT PRIMARY KEY,\n source_stack_id TEXT NOT NULL,\n target_stack_id TEXT NOT NULL,\n operation TEXT NOT NULL CHECK (operation IN ('handoff', 'merge', 'sync')),\n frame_count INTEGER NOT NULL,\n conflicts TEXT DEFAULT '[]',\n resolution TEXT,\n timestamp INTEGER NOT NULL,\n FOREIGN KEY (source_stack_id) REFERENCES stack_contexts(stack_id),\n FOREIGN KEY (target_stack_id) REFERENCES stack_contexts(stack_id)\n )\n `;\n\n // Execute schema creation using raw SQL\n if (this.adapter.isConnected()) {\n // Note: This is a temporary workaround - proper schema creation would use adapter methods\n (await (this.adapter as any).execute?.(createStackContextsTable)) ||\n this.executeSchemaQuery(createStackContextsTable);\n (await (this.adapter as any).execute?.(createHandoffRequestsTable)) ||\n this.executeSchemaQuery(createHandoffRequestsTable);\n (await (this.adapter as any).execute?.(createStackSyncLogTable)) ||\n this.executeSchemaQuery(createStackSyncLogTable);\n }\n\n await this.adapter.commitTransaction();\n\n logger.info('Dual stack schema initialized successfully');\n } catch (error: unknown) {\n await this.adapter.rollbackTransaction();\n logger.error('Failed to initialize dual stack schema', error);\n throw new DatabaseError(\n 'Schema initialization failed',\n ErrorCode.DB_SCHEMA_ERROR,\n { adapter: this.adapter.constructor.name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n private async executeSchemaQuery(sql: string): Promise<void> {\n // Fallback for adapters that don't have execute method\n logger.debug(\n 'Using fallback schema creation - implement execute method in adapter'\n );\n\n // Execute using raw SQLite database\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n try {\n rawDb.exec(sql);\n logger.debug('Executed schema query successfully');\n } catch (error: unknown) {\n logger.error('Failed to execute schema query', { sql, error });\n throw error;\n }\n } else {\n throw new Error(\n 'Cannot execute schema query: raw database not available'\n );\n }\n }\n\n private getDefaultIndividualPermissions(): StackPermissions {\n return {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: true,\n canAdminister: true,\n };\n }\n\n private getSharedStackPermissions(\n role: 'member' | 'lead' | 'admin'\n ): StackPermissions {\n const basePermissions = {\n canRead: true,\n canWrite: true,\n canHandoff: true,\n canMerge: false,\n canAdminister: false,\n };\n\n switch (role) {\n case 'lead':\n return { ...basePermissions, canMerge: true };\n case 'admin':\n return { ...basePermissions, canMerge: true, canAdminister: true };\n default:\n return basePermissions;\n }\n }\n\n /**\n * Switch between individual and shared stacks\n */\n async switchToStack(stackId: string): Promise<void> {\n // Validate input\n const input = validateInput(SwitchStackSchema, { stackId });\n\n try {\n if (input.stackId.startsWith('individual-')) {\n this.activeContext = {\n ...this.activeContext,\n stackId: input.stackId,\n type: 'individual',\n };\n return;\n }\n\n // Load shared stack context\n const stackContext = await this.loadStackContext(input.stackId);\n if (!stackContext) {\n throw new ValidationError(\n `Stack context not found: ${input.stackId}`,\n ErrorCode.STACK_CONTEXT_NOT_FOUND\n );\n }\n\n // Check permission to access the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n input.stackId,\n stackContext\n )\n );\n\n this.activeContext = stackContext;\n\n // Initialize shared stack manager if not already loaded\n if (!this.sharedStacks.has(input.stackId)) {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('Failed to get raw database for shared stack');\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n input.stackId\n );\n this.sharedStacks.set(input.stackId, sharedStack);\n }\n\n // Update last active timestamp\n await this.updateStackActivity(input.stackId);\n\n logger.info(`Switched to stack: ${input.stackId}`, {\n type: stackContext.type,\n });\n } catch (error: unknown) {\n throw new ValidationError(\n `Failed to switch to stack: ${input.stackId}`,\n ErrorCode.OPERATION_FAILED,\n { stackId: input.stackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get the current active stack manager\n */\n getActiveStack(): FrameManager {\n if (this.activeContext.type === 'individual') {\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(this.activeContext.stackId);\n if (!sharedStack) {\n throw new DatabaseError(\n `Active shared stack not initialized: ${this.activeContext.stackId}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n return sharedStack;\n }\n\n /**\n * Create a new shared stack for team collaboration\n */\n async createSharedStack(\n teamId: string,\n name: string,\n ownerId: string,\n permissions?: StackPermissions\n ): Promise<string> {\n // Validate input parameters\n const input = validateInput(CreateSharedStackSchema, {\n teamId,\n name,\n ownerId,\n permissions,\n });\n\n // Check permission to create shared stacks\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n input.ownerId,\n 'administer',\n 'stack',\n `shared-${input.teamId}`,\n this.activeContext\n )\n );\n\n const stackId = `shared-${input.teamId}-${Date.now()}`;\n\n const stackContext: StackContext = {\n stackId,\n type: 'shared',\n projectId: this.activeContext.projectId,\n teamId: input.teamId,\n permissions: input.permissions || this.getSharedStackPermissions('admin'),\n metadata: { name: input.name, ownerId: input.ownerId },\n createdAt: new Date(),\n lastActive: new Date(),\n };\n\n try {\n await this.saveStackContext(stackContext);\n\n // Initialize the shared stack manager\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('Failed to get raw database for new shared stack');\n }\n\n const sharedStack = new FrameManager(\n rawDb,\n stackContext.projectId,\n stackId\n );\n this.sharedStacks.set(stackId, sharedStack);\n\n // Set up permissions for the owner and team\n const stackPermissions = stackContext.permissions;\n this.permissionManager.setStackPermissions(\n input.ownerId,\n stackId,\n stackPermissions\n );\n\n logger.info(`Created shared stack: ${stackId}`, { teamId, name });\n return stackId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to create shared stack`,\n ErrorCode.OPERATION_FAILED,\n { teamId, name },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Initiate handoff of frames between stacks\n */\n async initiateHandoff(\n targetStackId: string,\n frameIds: string[],\n targetUserId?: string,\n message?: string\n ): Promise<string> {\n // Check permission to perform handoff from current stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'handoff',\n 'stack',\n this.activeContext.stackId,\n this.activeContext\n )\n );\n\n const requestId = `handoff-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n\n const request: HandoffRequest = {\n requestId,\n sourceStackId: this.activeContext.stackId,\n targetStackId,\n frameIds,\n requesterId: this.activeContext.ownerId!,\n targetUserId,\n message,\n status: 'pending',\n createdAt: new Date(),\n expiresAt: new Date(Date.now() + 24 * 60 * 60 * 1000), // 24 hours\n };\n\n try {\n await this.saveHandoffRequest(request);\n this.handoffRequests.set(requestId, request);\n\n logger.info(`Initiated handoff request: ${requestId}`, {\n sourceStack: this.activeContext.stackId,\n targetStack: targetStackId,\n frameCount: frameIds.length,\n });\n\n return requestId;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to initiate handoff`,\n ErrorCode.OPERATION_FAILED,\n { targetStackId, frameIds },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Accept a handoff request and move frames\n */\n async acceptHandoff(requestId: string): Promise<StackSyncResult> {\n logger.debug('acceptHandoff called', { requestId });\n const request = await this.loadHandoffRequest(requestId);\n logger.debug('loadHandoffRequest returned', {\n requestId,\n found: !!request,\n });\n if (!request) {\n logger.error('Handoff request not found', {\n requestId,\n availableRequests: Array.from(this.handoffRequests.keys()),\n });\n throw new DatabaseError(\n `Handoff request not found: ${requestId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n if (request.status !== 'pending') {\n throw new DatabaseError(\n `Handoff request is not pending: ${request.status}`,\n ErrorCode.INVALID_STATE\n );\n }\n\n if (request.expiresAt < new Date()) {\n throw new DatabaseError(\n `Handoff request has expired`,\n ErrorCode.OPERATION_EXPIRED\n );\n }\n\n try {\n // Perform the handoff operation\n logger.debug('Starting moveFramesBetweenStacks', { requestId });\n const syncResult = await this.moveFramesBetweenStacks(\n request.sourceStackId,\n request.targetStackId,\n request.frameIds\n );\n logger.debug('moveFramesBetweenStacks completed', {\n requestId,\n success: syncResult.success,\n });\n\n // Update request status\n logger.debug('Updating request status', { requestId });\n request.status = 'accepted';\n logger.debug('Calling saveHandoffRequest', { requestId });\n await this.saveHandoffRequest(request);\n logger.debug('saveHandoffRequest completed', { requestId });\n\n logger.info(`Accepted handoff request: ${requestId}`, {\n frameCount: request.frameIds.length,\n conflicts: syncResult.conflictFrames.length,\n });\n\n return syncResult;\n } catch (error: unknown) {\n logger.error('acceptHandoff caught error', {\n error: error instanceof Error ? error.message : error,\n });\n // Update request status to rejected on failure\n request.status = 'rejected';\n await this.saveHandoffRequest(request);\n\n throw new DatabaseError(\n `Failed to accept handoff`,\n ErrorCode.OPERATION_FAILED,\n { requestId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Sync frames between individual and shared stacks\n */\n async syncStacks(\n sourceStackId: string,\n targetStackId: string,\n options: {\n frameIds?: string[];\n conflictResolution: 'skip' | 'merge' | 'overwrite';\n dryRun?: boolean;\n }\n ): Promise<StackSyncResult> {\n try {\n const sourceStack = this.getStackManager(sourceStackId);\n const targetStack = this.getStackManager(targetStackId);\n\n // Get frames to sync\n const framesToSync =\n options.frameIds ||\n (await sourceStack.getActiveFrames()).map((f) => f.frame_id);\n\n const result: StackSyncResult = {\n success: true,\n conflictFrames: [],\n mergedFrames: [],\n errors: [],\n };\n\n for (const frameId of framesToSync) {\n try {\n const sourceFrame = await sourceStack.getFrame(frameId);\n if (!sourceFrame) {\n result.errors.push({\n frameId,\n error: 'Source frame not found',\n resolution: 'skipped',\n });\n continue;\n }\n\n const existingFrame = await targetStack.getFrame(frameId);\n\n if (existingFrame) {\n // Handle conflict\n switch (options.conflictResolution) {\n case 'skip':\n result.conflictFrames.push(frameId);\n result.errors.push({\n frameId,\n error: 'Frame already exists',\n resolution: 'skipped',\n });\n continue;\n\n case 'merge':\n if (!options.dryRun) {\n await this.mergeFrames(\n existingFrame,\n sourceFrame,\n targetStack\n );\n }\n result.mergedFrames.push(frameId);\n break;\n\n case 'overwrite':\n if (!options.dryRun) {\n await targetStack.deleteFrame(frameId);\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n break;\n }\n } else {\n // Copy frame to target\n if (!options.dryRun) {\n await this.copyFrame(sourceFrame, targetStack);\n }\n result.mergedFrames.push(frameId);\n }\n } catch (error: unknown) {\n result.errors.push({\n frameId,\n error: error instanceof Error ? error.message : String(error),\n resolution: 'skipped',\n });\n result.success = false;\n }\n }\n\n logger.info(`Stack sync completed`, {\n source: sourceStackId,\n target: targetStackId,\n merged: result.mergedFrames.length,\n conflicts: result.conflictFrames.length,\n errors: result.errors.length,\n });\n\n return result;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Stack sync failed`,\n ErrorCode.OPERATION_FAILED,\n { sourceStackId, targetStackId },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n getStackManager(stackId: string): FrameManager {\n logger.debug('getStackManager called', {\n stackId,\n availableStacks: Array.from(this.sharedStacks.keys()),\n });\n\n if (stackId.startsWith('individual-')) {\n logger.debug('Returning individual stack', { stackId });\n return this.individualStack;\n }\n\n const sharedStack = this.sharedStacks.get(stackId);\n if (!sharedStack) {\n logger.error('Stack manager not found', {\n stackId,\n availableSharedStacks: Array.from(this.sharedStacks.keys()),\n message: 'getStackManager could not find shared stack',\n });\n throw new DatabaseError(\n `Stack manager not found: ${stackId}`,\n ErrorCode.RESOURCE_NOT_FOUND\n );\n }\n\n logger.debug('Returning shared stack', { stackId });\n return sharedStack;\n }\n\n private async moveFramesBetweenStacks(\n sourceStackId: string,\n targetStackId: string,\n frameIds: string[]\n ): Promise<StackSyncResult> {\n const syncResult = await this.syncStacks(sourceStackId, targetStackId, {\n frameIds,\n conflictResolution: 'merge',\n });\n\n // Remove frames from source stack after successful sync\n if (syncResult.success && syncResult.errors.length === 0) {\n const sourceStack = this.getStackManager(sourceStackId);\n for (const frameId of frameIds) {\n try {\n sourceStack.deleteFrame(frameId);\n logger.debug('Deleted frame from source stack', {\n frameId,\n sourceStackId,\n });\n } catch (error: unknown) {\n logger.warn('Failed to delete frame from source stack', {\n frameId,\n error,\n });\n }\n }\n logger.debug('Completed frame cleanup from source stack', {\n frameIds: frameIds.length,\n });\n }\n\n return syncResult;\n }\n\n private async copyFrame(\n frame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Create frame in target stack\n await targetStack.createFrame({\n type: frame.type as any,\n name: frame.name,\n inputs: frame.inputs,\n });\n\n // Copy events\n const events = await this.individualStack.getFrameEvents(frame.frame_id);\n for (const event of events) {\n await targetStack.addEvent(frame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: event.metadata,\n });\n }\n\n // Copy anchors\n const anchors = await this.individualStack.getFrameAnchors(frame.frame_id);\n for (const anchor of anchors) {\n await targetStack.addAnchor(frame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: anchor.metadata,\n });\n }\n }\n\n private async mergeFrames(\n existingFrame: Frame,\n sourceFrame: Frame,\n targetStack: FrameManager\n ): Promise<void> {\n // Simple merge strategy - append events and anchors\n const sourceEvents = await this.individualStack.getFrameEvents(\n sourceFrame.frame_id\n );\n for (const event of sourceEvents) {\n await targetStack.addEvent(existingFrame.frame_id, {\n type: event.type as any,\n text: event.text,\n metadata: { ...event.metadata, merged: true },\n });\n }\n\n const sourceAnchors = await this.individualStack.getFrameAnchors(\n sourceFrame.frame_id\n );\n for (const anchor of sourceAnchors) {\n await targetStack.addAnchor(existingFrame.frame_id, {\n type: anchor.type as any,\n text: anchor.text,\n priority: anchor.priority,\n metadata: { ...anchor.metadata, merged: true },\n });\n }\n }\n\n private async loadStackContext(\n stackId: string\n ): Promise<StackContext | null> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n return null;\n }\n\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE stack_id = ?\n `);\n\n const row = query.get(stackId) as any;\n if (!row) {\n return null;\n }\n\n return {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n } catch (error: unknown) {\n logger.error('Failed to load stack context', { stackId, error });\n return null;\n }\n }\n\n private async saveStackContext(context: StackContext): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n throw new Error('SQLite database not available for stack context save');\n }\n\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO stack_contexts \n (stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n context.stackId,\n context.type,\n context.projectId,\n context.ownerId || null,\n context.teamId || null,\n JSON.stringify(context.permissions),\n JSON.stringify(context.metadata || {}),\n context.createdAt.getTime(),\n context.lastActive.getTime()\n );\n\n logger.debug('Saved stack context', { stackId: context.stackId });\n } catch (error: unknown) {\n logger.error('Failed to save stack context', {\n stackId: context.stackId,\n error,\n });\n throw error;\n }\n }\n\n private async updateStackActivity(stackId: string): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (!rawDb) {\n logger.warn('SQLite database not available for activity update');\n return;\n }\n\n const query = rawDb.prepare(`\n UPDATE stack_contexts \n SET last_active = ?\n WHERE stack_id = ?\n `);\n\n query.run(Date.now(), stackId);\n logger.debug('Updated stack activity', { stackId });\n } catch (error: unknown) {\n logger.error('Failed to update stack activity', { stackId, error });\n // Don't throw - activity updates are not critical\n }\n }\n\n private async loadHandoffRequest(\n requestId: string\n ): Promise<HandoffRequest | null> {\n // Try in-memory first for fast access\n const memoryRequest = this.handoffRequests.get(requestId);\n if (memoryRequest) {\n return memoryRequest;\n }\n\n // Try loading from database\n try {\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT * FROM handoff_requests WHERE request_id = ?\n `);\n\n const row = query.get(requestId) as any;\n if (row) {\n const request: HandoffRequest = {\n requestId: row.request_id,\n sourceStackId: row.source_stack_id,\n targetStackId: row.target_stack_id,\n frameIds: JSON.parse(row.frame_ids),\n status: row.status,\n createdAt: new Date(row.created_at),\n expiresAt: new Date(row.expires_at),\n targetUserId: row.target_user_id,\n message: row.message,\n };\n\n // Cache in memory for future access\n this.handoffRequests.set(requestId, request);\n return request;\n }\n }\n } catch (error: unknown) {\n logger.error('Failed to load handoff request from database', {\n requestId,\n error,\n });\n }\n\n return null;\n }\n\n private async saveHandoffRequest(request: HandoffRequest): Promise<void> {\n try {\n // Use raw database for direct query\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n INSERT OR REPLACE INTO handoff_requests \n (request_id, source_stack_id, target_stack_id, frame_ids, status, created_at, expires_at, target_user_id, message)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n query.run(\n request.requestId,\n request.sourceStackId,\n request.targetStackId,\n JSON.stringify(request.frameIds),\n request.status,\n request.createdAt.getTime(),\n request.expiresAt.getTime(),\n request.targetUserId || null,\n request.message || null\n );\n\n logger.debug('Saved handoff request to database', {\n requestId: request.requestId,\n });\n }\n\n // Also keep in-memory for fast access\n this.handoffRequests.set(request.requestId, request);\n } catch (error: unknown) {\n logger.error('Failed to save handoff request', {\n requestId: request.requestId,\n error,\n });\n // Fallback to in-memory only\n this.handoffRequests.set(request.requestId, request);\n }\n }\n\n /**\n * Get list of available stacks for the current user\n */\n async getAvailableStacks(): Promise<StackContext[]> {\n try {\n const stacks: StackContext[] = [];\n\n // Always include current individual stack context\n stacks.push(this.activeContext);\n\n // Query database for all shared stacks the user has access to\n const rawDb =\n this.adapter instanceof SQLiteAdapter\n ? this.adapter.getRawDatabase()\n : null;\n if (rawDb) {\n const query = rawDb.prepare(`\n SELECT stack_id, type, project_id, owner_id, team_id, permissions, metadata, created_at, last_active\n FROM stack_contexts \n WHERE type = 'shared' AND project_id = ?\n `);\n\n const rows = query.all(this.activeContext.projectId) as any[];\n\n for (const row of rows) {\n const context: StackContext = {\n stackId: row.stack_id,\n type: row.type,\n projectId: row.project_id,\n ownerId: row.owner_id,\n teamId: row.team_id,\n permissions: JSON.parse(row.permissions),\n metadata: JSON.parse(row.metadata || '{}'),\n createdAt: new Date(row.created_at),\n lastActive: new Date(row.last_active),\n };\n\n // Check if user has permission to access this stack\n try {\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n this.activeContext.ownerId || 'unknown',\n 'read',\n 'stack',\n context.stackId,\n context\n )\n );\n stacks.push(context);\n } catch (permissionError: unknown) {\n // Skip stacks user doesn't have access to\n logger.debug('User lacks access to stack', {\n stackId: context.stackId,\n userId: this.activeContext.ownerId,\n });\n }\n }\n }\n\n return stacks;\n } catch (error: unknown) {\n logger.error('Failed to get available stacks', error);\n // Return at least the current individual stack\n return [this.activeContext];\n }\n }\n\n /**\n * Get pending handoff requests for the current user\n */\n async getPendingHandoffRequests(): Promise<HandoffRequest[]> {\n return Array.from(this.handoffRequests.values()).filter(\n (request) =>\n request.status === 'pending' && request.expiresAt > new Date()\n );\n }\n\n /**\n * Get current stack context\n */\n getCurrentContext(): StackContext {\n return { ...this.activeContext };\n }\n\n /**\n * Get permission manager for external access\n */\n getPermissionManager(): PermissionManager {\n return this.permissionManager;\n }\n\n /**\n * Add user to shared stack with specific permissions\n */\n async addUserToSharedStack(\n stackId: string,\n userId: string,\n permissions: StackPermissions,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Grant permissions to the new user\n this.permissionManager.setStackPermissions(userId, stackId, permissions);\n\n logger.info(`Added user to shared stack`, {\n stackId,\n userId,\n permissions,\n requesterId,\n });\n }\n\n /**\n * Remove user from shared stack\n */\n async removeUserFromSharedStack(\n stackId: string,\n userId: string,\n requesterId: string\n ): Promise<void> {\n // Check if requester has admin permissions on the stack\n await this.permissionManager.enforcePermission(\n this.permissionManager.createContext(\n requesterId,\n 'administer',\n 'stack',\n stackId\n )\n );\n\n // Remove user's permissions\n const userPermissions = this.permissionManager.getUserPermissions(userId);\n userPermissions.delete(stackId);\n\n logger.info(`Removed user from shared stack`, {\n stackId,\n userId,\n requesterId,\n });\n }\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -11,6 +11,7 @@ class FrameDatabase {
11
11
  try {
12
12
  this.db.pragma("journal_mode = WAL");
13
13
  this.db.pragma("synchronous = NORMAL");
14
+ this.db.pragma("foreign_keys = ON");
14
15
  this.db.exec(`
15
16
  CREATE TABLE IF NOT EXISTS frames (
16
17
  frame_id TEXT PRIMARY KEY,
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/core/context/frame-database.ts"],
4
- "sourcesContent": ["/**\n * Frame Database Operations\n * Handles all database interactions for frames, events, and anchors\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Event, Anchor } from './frame-types.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode } from '../errors/index.js';\n\nexport class FrameDatabase {\n constructor(private db: Database.Database) {}\n\n /**\n * Initialize database schema\n */\n initSchema(): void {\n try {\n // Enable WAL mode for better concurrency\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('synchronous = NORMAL');\n\n // Create frames table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT NOT NULL DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n closed_at INTEGER,\n FOREIGN KEY (parent_frame_id) REFERENCES frames(frame_id)\n );\n `);\n\n // Create events table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n run_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n ts INTEGER NOT NULL DEFAULT (unixepoch() * 1000),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create anchors table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER NOT NULL DEFAULT 5,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create indexes for performance\n this.db.exec(`\n CREATE INDEX IF NOT EXISTS idx_frames_project_state ON frames(project_id, state);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_frame_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame_priority ON anchors(frame_id, priority DESC);\n `);\n\n logger.info('Frame database schema initialized');\n } catch (error: unknown) {\n throw new DatabaseError(\n 'Failed to initialize frame database schema',\n ErrorCode.DB_SCHEMA_ERROR,\n { operation: 'initSchema' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert new frame\n */\n insertFrame(frame: Omit<Frame, 'created_at' | 'closed_at'>): Frame {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO frames (frame_id, run_id, project_id, parent_frame_id, depth, type, name, state, inputs, outputs, digest_json)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n frame.frame_id,\n frame.run_id,\n frame.project_id,\n frame.parent_frame_id || null,\n frame.depth,\n frame.type,\n frame.name,\n frame.state,\n JSON.stringify(frame.inputs),\n JSON.stringify(frame.outputs),\n JSON.stringify(frame.digest_json)\n );\n\n if (result.changes === 0) {\n throw new Error('Frame insertion failed - no rows affected');\n }\n\n // Return the created frame with timestamp\n const createdFrame = this.getFrame(frame.frame_id);\n if (!createdFrame) {\n throw new Error('Failed to retrieve created frame');\n }\n\n return createdFrame;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert frame: ${frame.frame_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n frameId: frame.frame_id,\n frameName: frame.name,\n operation: 'insertFrame',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frame by ID\n */\n getFrame(frameId: string): Frame | undefined {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as any;\n\n if (!row) return undefined;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n } catch (error: unknown) {\n logger.warn(`Failed to get frame: ${frameId}`, { error });\n return undefined;\n }\n }\n\n /**\n * Update frame state and outputs\n */\n updateFrame(frameId: string, updates: Partial<Frame>): void {\n try {\n const setClauses: string[] = [];\n const values: any[] = [];\n\n if (updates.state !== undefined) {\n setClauses.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n setClauses.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n setClauses.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n setClauses.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n setClauses.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (setClauses.length === 0) {\n return; // No updates to apply\n }\n\n values.push(frameId);\n\n const stmt = this.db.prepare(`\n UPDATE frames SET ${setClauses.join(', ')} WHERE frame_id = ?\n `);\n\n const result = stmt.run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to update frame: ${frameId}`,\n ErrorCode.DB_UPDATE_FAILED,\n { frameId, updates, operation: 'updateFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frames by project and state\n */\n getFramesByProject(projectId: string, state?: 'active' | 'closed'): Frame[] {\n try {\n const query = state\n ? 'SELECT * FROM frames WHERE project_id = ? AND state = ? ORDER BY created_at'\n : 'SELECT * FROM frames WHERE project_id = ? ORDER BY created_at';\n\n const params = state ? [projectId, state] : [projectId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get frames for project: ${projectId}`,\n ErrorCode.DB_QUERY_FAILED,\n { projectId, state, operation: 'getFramesByProject' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert event\n */\n insertEvent(event: Omit<Event, 'ts'>): Event {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO events (event_id, frame_id, run_id, seq, event_type, payload)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n event.event_id,\n event.frame_id,\n event.run_id,\n event.seq,\n event.event_type,\n JSON.stringify(event.payload)\n );\n\n if (result.changes === 0) {\n throw new Error('Event insertion failed');\n }\n\n // Return the created event with timestamp\n const createdEvent = this.db\n .prepare('SELECT * FROM events WHERE event_id = ?')\n .get(event.event_id) as any;\n\n return {\n ...createdEvent,\n payload: JSON.parse(createdEvent.payload),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert event: ${event.event_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n eventId: event.event_id,\n frameId: event.frame_id,\n operation: 'insertEvent',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get events for a frame\n */\n getFrameEvents(frameId: string, limit?: number): Event[] {\n try {\n const query = limit\n ? 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n : 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq ASC';\n\n const params = limit ? [frameId, limit] : [frameId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n payload: JSON.parse(row.payload),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get events for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, limit, operation: 'getFrameEvents' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get next event sequence number\n */\n getNextEventSequence(frameId: string): number {\n try {\n const result = this.db\n .prepare('SELECT MAX(seq) as max_seq FROM events WHERE frame_id = ?')\n .get(frameId) as { max_seq: number | null };\n\n return (result.max_seq || 0) + 1;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get next event sequence for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getNextEventSequence' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert anchor\n */\n insertAnchor(anchor: Omit<Anchor, 'created_at'>): Anchor {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO anchors (anchor_id, frame_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n anchor.anchor_id,\n anchor.frame_id,\n anchor.type,\n anchor.text,\n anchor.priority,\n JSON.stringify(anchor.metadata)\n );\n\n if (result.changes === 0) {\n throw new Error('Anchor insertion failed');\n }\n\n // Return the created anchor with timestamp\n const createdAnchor = this.db\n .prepare('SELECT * FROM anchors WHERE anchor_id = ?')\n .get(anchor.anchor_id) as any;\n\n return {\n ...createdAnchor,\n metadata: JSON.parse(createdAnchor.metadata || '{}'),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert anchor: ${anchor.anchor_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n anchorId: anchor.anchor_id,\n frameId: anchor.frame_id,\n operation: 'insertAnchor',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get anchors for a frame\n */\n getFrameAnchors(frameId: string): Anchor[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get anchors for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getFrameAnchors' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Delete frame and all related data\n */\n deleteFrame(frameId: string): void {\n try {\n // Delete in order due to foreign keys\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n\n logger.info('Frame deleted', { frameId });\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to delete frame: ${frameId}`,\n ErrorCode.DB_DELETE_FAILED,\n { frameId, operation: 'deleteFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get database statistics\n */\n getStatistics(): Record<string, number> {\n try {\n const frameCount = this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as { count: number };\n const eventCount = this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as { count: number };\n const anchorCount = this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as { count: number };\n const activeFrames = this.db\n .prepare(\"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\")\n .get() as { count: number };\n\n return {\n totalFrames: frameCount.count,\n totalEvents: eventCount.count,\n totalAnchors: anchorCount.count,\n activeFrames: activeFrames.count,\n };\n } catch (error: unknown) {\n logger.warn('Failed to get database statistics', {\n error: error instanceof Error ? error.message : String(error),\n });\n return {};\n }\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Frame Database Operations\n * Handles all database interactions for frames, events, and anchors\n */\n\nimport Database from 'better-sqlite3';\nimport { Frame, Event, Anchor } from './frame-types.js';\nimport { logger } from '../monitoring/logger.js';\nimport { DatabaseError, ErrorCode } from '../errors/index.js';\n\nexport class FrameDatabase {\n constructor(private db: Database.Database) {}\n\n /**\n * Initialize database schema\n */\n initSchema(): void {\n try {\n // Enable WAL mode for better concurrency\n this.db.pragma('journal_mode = WAL');\n this.db.pragma('synchronous = NORMAL');\n // Enforce referential integrity\n this.db.pragma('foreign_keys = ON');\n\n // Create frames table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS frames (\n frame_id TEXT PRIMARY KEY,\n run_id TEXT NOT NULL,\n project_id TEXT NOT NULL,\n parent_frame_id TEXT,\n depth INTEGER NOT NULL DEFAULT 0,\n type TEXT NOT NULL,\n name TEXT NOT NULL,\n state TEXT NOT NULL DEFAULT 'active',\n inputs TEXT DEFAULT '{}',\n outputs TEXT DEFAULT '{}',\n digest_text TEXT,\n digest_json TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n closed_at INTEGER,\n FOREIGN KEY (parent_frame_id) REFERENCES frames(frame_id)\n );\n `);\n\n // Create events table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS events (\n event_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n run_id TEXT NOT NULL,\n seq INTEGER NOT NULL,\n event_type TEXT NOT NULL,\n payload TEXT NOT NULL DEFAULT '{}',\n ts INTEGER NOT NULL DEFAULT (unixepoch() * 1000),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create anchors table\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS anchors (\n anchor_id TEXT PRIMARY KEY,\n frame_id TEXT NOT NULL,\n type TEXT NOT NULL,\n text TEXT NOT NULL,\n priority INTEGER NOT NULL DEFAULT 5,\n metadata TEXT DEFAULT '{}',\n created_at INTEGER NOT NULL DEFAULT (unixepoch()),\n FOREIGN KEY (frame_id) REFERENCES frames(frame_id) ON DELETE CASCADE\n );\n `);\n\n // Create indexes for performance\n this.db.exec(`\n CREATE INDEX IF NOT EXISTS idx_frames_project_state ON frames(project_id, state);\n CREATE INDEX IF NOT EXISTS idx_frames_parent ON frames(parent_frame_id);\n CREATE INDEX IF NOT EXISTS idx_events_frame_seq ON events(frame_id, seq);\n CREATE INDEX IF NOT EXISTS idx_anchors_frame_priority ON anchors(frame_id, priority DESC);\n `);\n\n logger.info('Frame database schema initialized');\n } catch (error: unknown) {\n throw new DatabaseError(\n 'Failed to initialize frame database schema',\n ErrorCode.DB_SCHEMA_ERROR,\n { operation: 'initSchema' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert new frame\n */\n insertFrame(frame: Omit<Frame, 'created_at' | 'closed_at'>): Frame {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO frames (frame_id, run_id, project_id, parent_frame_id, depth, type, name, state, inputs, outputs, digest_json)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n frame.frame_id,\n frame.run_id,\n frame.project_id,\n frame.parent_frame_id || null,\n frame.depth,\n frame.type,\n frame.name,\n frame.state,\n JSON.stringify(frame.inputs),\n JSON.stringify(frame.outputs),\n JSON.stringify(frame.digest_json)\n );\n\n if (result.changes === 0) {\n throw new Error('Frame insertion failed - no rows affected');\n }\n\n // Return the created frame with timestamp\n const createdFrame = this.getFrame(frame.frame_id);\n if (!createdFrame) {\n throw new Error('Failed to retrieve created frame');\n }\n\n return createdFrame;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert frame: ${frame.frame_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n frameId: frame.frame_id,\n frameName: frame.name,\n operation: 'insertFrame',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frame by ID\n */\n getFrame(frameId: string): Frame | undefined {\n try {\n const row = this.db\n .prepare('SELECT * FROM frames WHERE frame_id = ?')\n .get(frameId) as any;\n\n if (!row) return undefined;\n\n return {\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n };\n } catch (error: unknown) {\n logger.warn(`Failed to get frame: ${frameId}`, { error });\n return undefined;\n }\n }\n\n /**\n * Update frame state and outputs\n */\n updateFrame(frameId: string, updates: Partial<Frame>): void {\n try {\n const setClauses: string[] = [];\n const values: any[] = [];\n\n if (updates.state !== undefined) {\n setClauses.push('state = ?');\n values.push(updates.state);\n }\n\n if (updates.outputs !== undefined) {\n setClauses.push('outputs = ?');\n values.push(JSON.stringify(updates.outputs));\n }\n\n if (updates.digest_text !== undefined) {\n setClauses.push('digest_text = ?');\n values.push(updates.digest_text);\n }\n\n if (updates.digest_json !== undefined) {\n setClauses.push('digest_json = ?');\n values.push(JSON.stringify(updates.digest_json));\n }\n\n if (updates.closed_at !== undefined) {\n setClauses.push('closed_at = ?');\n values.push(updates.closed_at);\n }\n\n if (setClauses.length === 0) {\n return; // No updates to apply\n }\n\n values.push(frameId);\n\n const stmt = this.db.prepare(`\n UPDATE frames SET ${setClauses.join(', ')} WHERE frame_id = ?\n `);\n\n const result = stmt.run(...values);\n\n if (result.changes === 0) {\n throw new Error(`Frame not found: ${frameId}`);\n }\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to update frame: ${frameId}`,\n ErrorCode.DB_UPDATE_FAILED,\n { frameId, updates, operation: 'updateFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get frames by project and state\n */\n getFramesByProject(projectId: string, state?: 'active' | 'closed'): Frame[] {\n try {\n const query = state\n ? 'SELECT * FROM frames WHERE project_id = ? AND state = ? ORDER BY created_at'\n : 'SELECT * FROM frames WHERE project_id = ? ORDER BY created_at';\n\n const params = state ? [projectId, state] : [projectId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n inputs: JSON.parse(row.inputs || '{}'),\n outputs: JSON.parse(row.outputs || '{}'),\n digest_json: JSON.parse(row.digest_json || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get frames for project: ${projectId}`,\n ErrorCode.DB_QUERY_FAILED,\n { projectId, state, operation: 'getFramesByProject' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert event\n */\n insertEvent(event: Omit<Event, 'ts'>): Event {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO events (event_id, frame_id, run_id, seq, event_type, payload)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n event.event_id,\n event.frame_id,\n event.run_id,\n event.seq,\n event.event_type,\n JSON.stringify(event.payload)\n );\n\n if (result.changes === 0) {\n throw new Error('Event insertion failed');\n }\n\n // Return the created event with timestamp\n const createdEvent = this.db\n .prepare('SELECT * FROM events WHERE event_id = ?')\n .get(event.event_id) as any;\n\n return {\n ...createdEvent,\n payload: JSON.parse(createdEvent.payload),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert event: ${event.event_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n eventId: event.event_id,\n frameId: event.frame_id,\n operation: 'insertEvent',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get events for a frame\n */\n getFrameEvents(frameId: string, limit?: number): Event[] {\n try {\n const query = limit\n ? 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq DESC LIMIT ?'\n : 'SELECT * FROM events WHERE frame_id = ? ORDER BY seq ASC';\n\n const params = limit ? [frameId, limit] : [frameId];\n const rows = this.db.prepare(query).all(...params) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n payload: JSON.parse(row.payload),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get events for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, limit, operation: 'getFrameEvents' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get next event sequence number\n */\n getNextEventSequence(frameId: string): number {\n try {\n const result = this.db\n .prepare('SELECT MAX(seq) as max_seq FROM events WHERE frame_id = ?')\n .get(frameId) as { max_seq: number | null };\n\n return (result.max_seq || 0) + 1;\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get next event sequence for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getNextEventSequence' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Insert anchor\n */\n insertAnchor(anchor: Omit<Anchor, 'created_at'>): Anchor {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO anchors (anchor_id, frame_id, type, text, priority, metadata)\n VALUES (?, ?, ?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n anchor.anchor_id,\n anchor.frame_id,\n anchor.type,\n anchor.text,\n anchor.priority,\n JSON.stringify(anchor.metadata)\n );\n\n if (result.changes === 0) {\n throw new Error('Anchor insertion failed');\n }\n\n // Return the created anchor with timestamp\n const createdAnchor = this.db\n .prepare('SELECT * FROM anchors WHERE anchor_id = ?')\n .get(anchor.anchor_id) as any;\n\n return {\n ...createdAnchor,\n metadata: JSON.parse(createdAnchor.metadata || '{}'),\n };\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to insert anchor: ${anchor.anchor_id}`,\n ErrorCode.DB_INSERT_FAILED,\n {\n anchorId: anchor.anchor_id,\n frameId: anchor.frame_id,\n operation: 'insertAnchor',\n },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get anchors for a frame\n */\n getFrameAnchors(frameId: string): Anchor[] {\n try {\n const rows = this.db\n .prepare(\n 'SELECT * FROM anchors WHERE frame_id = ? ORDER BY priority DESC, created_at ASC'\n )\n .all(frameId) as any[];\n\n return rows.map((row: any) => ({\n ...row,\n metadata: JSON.parse(row.metadata || '{}'),\n }));\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to get anchors for frame: ${frameId}`,\n ErrorCode.DB_QUERY_FAILED,\n { frameId, operation: 'getFrameAnchors' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Delete frame and all related data\n */\n deleteFrame(frameId: string): void {\n try {\n // Delete in order due to foreign keys\n this.db.prepare('DELETE FROM anchors WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM events WHERE frame_id = ?').run(frameId);\n this.db.prepare('DELETE FROM frames WHERE frame_id = ?').run(frameId);\n\n logger.info('Frame deleted', { frameId });\n } catch (error: unknown) {\n throw new DatabaseError(\n `Failed to delete frame: ${frameId}`,\n ErrorCode.DB_DELETE_FAILED,\n { frameId, operation: 'deleteFrame' },\n error instanceof Error ? error : undefined\n );\n }\n }\n\n /**\n * Get database statistics\n */\n getStatistics(): Record<string, number> {\n try {\n const frameCount = this.db\n .prepare('SELECT COUNT(*) as count FROM frames')\n .get() as { count: number };\n const eventCount = this.db\n .prepare('SELECT COUNT(*) as count FROM events')\n .get() as { count: number };\n const anchorCount = this.db\n .prepare('SELECT COUNT(*) as count FROM anchors')\n .get() as { count: number };\n const activeFrames = this.db\n .prepare(\"SELECT COUNT(*) as count FROM frames WHERE state = 'active'\")\n .get() as { count: number };\n\n return {\n totalFrames: frameCount.count,\n totalEvents: eventCount.count,\n totalAnchors: anchorCount.count,\n activeFrames: activeFrames.count,\n };\n } catch (error: unknown) {\n logger.warn('Failed to get database statistics', {\n error: error instanceof Error ? error.message : String(error),\n });\n return {};\n }\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }