@stackmemoryai/stackmemory 0.3.16 → 0.3.18
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.
- package/README.md +48 -2
- package/dist/cli/commands/skills.js +15 -2
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/index.js +113 -834
- package/dist/cli/index.js.map +3 -3
- package/dist/core/context/dual-stack-manager.js +1 -1
- package/dist/core/context/dual-stack-manager.js.map +1 -1
- package/dist/core/context/frame-manager.js +3 -0
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/integrations/claude-code/subagent-client.js +106 -3
- package/dist/integrations/claude-code/subagent-client.js.map +2 -2
- package/dist/servers/railway/config.js +51 -0
- package/dist/servers/railway/config.js.map +7 -0
- package/dist/servers/railway/index-enhanced.js +156 -0
- package/dist/servers/railway/index-enhanced.js.map +7 -0
- package/dist/servers/railway/minimal.js +48 -3
- package/dist/servers/railway/minimal.js.map +2 -2
- package/dist/servers/railway/storage-test.js +455 -0
- package/dist/servers/railway/storage-test.js.map +7 -0
- package/dist/skills/claude-skills.js +13 -12
- package/dist/skills/claude-skills.js.map +2 -2
- package/dist/skills/recursive-agent-orchestrator.js +27 -18
- package/dist/skills/recursive-agent-orchestrator.js.map +2 -2
- package/dist/skills/unified-rlm-orchestrator.js.map +2 -2
- package/package.json +6 -18
- package/scripts/README-TESTING.md +186 -0
- package/scripts/analyze-cli-security.js +288 -0
- package/scripts/archive/add-phase-tasks-to-linear.js +163 -0
- package/scripts/archive/analyze-linear-duplicates.js +214 -0
- package/scripts/archive/analyze-remaining-duplicates.js +230 -0
- package/scripts/archive/analyze-sta-duplicates.js +292 -0
- package/scripts/archive/analyze-sta-graphql.js +399 -0
- package/scripts/archive/cancel-duplicate-tasks.ts +246 -0
- package/scripts/archive/check-all-duplicates.ts +419 -0
- package/scripts/archive/clean-duplicate-tasks.js +114 -0
- package/scripts/archive/cleanup-duplicate-tasks.ts +286 -0
- package/scripts/archive/create-phase-tasks.js +387 -0
- package/scripts/archive/delete-linear-duplicates.js +182 -0
- package/scripts/archive/delete-remaining-duplicates.js +158 -0
- package/scripts/archive/delete-sta-duplicates.js +201 -0
- package/scripts/archive/delete-sta-oauth.js +201 -0
- package/scripts/archive/export-sta-tasks.js +62 -0
- package/scripts/archive/install-auto-sync.js +266 -0
- package/scripts/archive/install-chromadb-hooks.sh +133 -0
- package/scripts/archive/install-enhanced-clear-hooks.sh +431 -0
- package/scripts/archive/install-post-task-hooks.sh +289 -0
- package/scripts/archive/install-stackmemory-hooks.sh +420 -0
- package/scripts/archive/merge-linear-duplicates-safe.ts +362 -0
- package/scripts/archive/merge-linear-duplicates.ts +180 -0
- package/scripts/archive/remove-sta-tasks.js +70 -0
- package/scripts/archive/setup-background-sync.sh +168 -0
- package/scripts/archive/setup-claude-auto-triggers.sh +181 -0
- package/scripts/archive/setup-claude-autostart.sh +305 -0
- package/scripts/archive/setup-git-hooks.sh +25 -0
- package/scripts/archive/setup-linear-oauth.sh +46 -0
- package/scripts/archive/setup-mcp.sh +113 -0
- package/scripts/archive/setup-railway-deployment.sh +81 -0
- package/scripts/auto-handoff.sh +262 -0
- package/scripts/background-sync-manager.js +416 -0
- package/scripts/benchmark-performance.ts +57 -0
- package/scripts/check-redis.ts +48 -0
- package/scripts/chromadb-auto-loader.sh +128 -0
- package/scripts/chromadb-context-loader.js +479 -0
- package/scripts/claude-chromadb-hook.js +460 -0
- package/scripts/claude-code-wrapper.sh +66 -0
- package/scripts/claude-linear-skill.js +455 -0
- package/scripts/claude-pre-commit.sh +302 -0
- package/scripts/claude-sm-autostart.js +532 -0
- package/scripts/claude-sm-setup.sh +367 -0
- package/scripts/claude-with-chromadb.sh +69 -0
- package/scripts/claude-worktree-manager.sh +323 -0
- package/scripts/claude-worktree-monitor.sh +371 -0
- package/scripts/claude-worktree-setup.sh +327 -0
- package/scripts/clean-linear-backlog.js +273 -0
- package/scripts/cleanup-old-sessions.sh +57 -0
- package/scripts/codex-wrapper.sh +88 -0
- package/scripts/create-sandbox.sh +269 -0
- package/scripts/debug-linear-update.js +174 -0
- package/scripts/delete-linear-tasks.js +167 -0
- package/scripts/deploy.sh +89 -0
- package/scripts/deployment/railway.sh +352 -0
- package/scripts/deployment/test-deployment.js +194 -0
- package/scripts/detect-and-rehydrate.js +162 -0
- package/scripts/detect-and-rehydrate.mjs +165 -0
- package/scripts/development/create-demo-tasks.js +143 -0
- package/scripts/development/debug-frame-test.js +16 -0
- package/scripts/development/demo-auto-sync.js +128 -0
- package/scripts/development/fix-all-imports.js +213 -0
- package/scripts/development/fix-imports.js +229 -0
- package/scripts/development/fix-lint-loop.cjs +103 -0
- package/scripts/development/fix-project-id.ts +161 -0
- package/scripts/development/fix-strict-mode-issues.ts +291 -0
- package/scripts/development/reorganize-structure.sh +228 -0
- package/scripts/development/test-persistence-direct.js +148 -0
- package/scripts/development/test-persistence.js +114 -0
- package/scripts/development/test-tasks.js +93 -0
- package/scripts/development/update-imports.js +212 -0
- package/scripts/fetch-linear-status.js +125 -0
- package/scripts/git-hooks/README.md +310 -0
- package/scripts/git-hooks/branch-context-manager.sh +342 -0
- package/scripts/git-hooks/post-checkout-stackmemory.sh +63 -0
- package/scripts/git-hooks/post-commit-stackmemory.sh +305 -0
- package/scripts/git-hooks/pre-commit-stackmemory.sh +275 -0
- package/scripts/hooks/cleanup-shell.sh +130 -0
- package/scripts/hooks/task-complete.sh +114 -0
- package/scripts/initialize.ts +129 -0
- package/scripts/install-claude-hooks-auto.js +104 -0
- package/scripts/install-claude-hooks.sh +133 -0
- package/scripts/install-global.sh +296 -0
- package/scripts/install.sh +235 -0
- package/scripts/linear-auto-sync.js +262 -0
- package/scripts/linear-auto-sync.sh +161 -0
- package/scripts/linear-sync-daemon.js +150 -0
- package/scripts/linear-task-review.js +237 -0
- package/scripts/list-linear-tasks.ts +178 -0
- package/scripts/mcp-proxy.js +66 -0
- package/scripts/opencode-wrapper.sh +85 -0
- package/scripts/publish-local.js +74 -0
- package/scripts/query-chromadb.ts +201 -0
- package/scripts/railway-env-setup.sh +39 -0
- package/scripts/reconcile-local-tasks.js +170 -0
- package/scripts/recreate-frames-db.js +89 -0
- package/scripts/setup/claude-integration.js +138 -0
- package/scripts/setup/configure-alias.js +125 -0
- package/scripts/setup/configure-codex-alias.js +161 -0
- package/scripts/setup/configure-opencode-alias.js +175 -0
- package/scripts/setup-claude-integration.js +204 -0
- package/scripts/setup-claude-integration.sh +183 -0
- package/scripts/setup.sh +31 -0
- package/scripts/show-linear-summary.ts +172 -0
- package/scripts/stackmemory-auto-handoff.sh +231 -0
- package/scripts/stackmemory-daemon.sh +40 -0
- package/scripts/start-linear-sync-daemon.sh +141 -0
- package/scripts/start-temporal-paradox.sh +214 -0
- package/scripts/status.ts +159 -0
- package/scripts/sync-and-clean-tasks.js +258 -0
- package/scripts/sync-frames-from-railway.js +228 -0
- package/scripts/sync-linear-graphql.js +303 -0
- package/scripts/sync-linear-tasks.js +186 -0
- package/scripts/test-auto-triggers.sh +57 -0
- package/scripts/test-browser-mcp.js +74 -0
- package/scripts/test-chromadb-full.js +115 -0
- package/scripts/test-chromadb-hooks.sh +28 -0
- package/scripts/test-chromadb-sync.ts +245 -0
- package/scripts/test-cli-security.js +293 -0
- package/scripts/test-hooks-persistence.sh +220 -0
- package/scripts/test-installation-scenarios.sh +359 -0
- package/scripts/test-installation.sh +224 -0
- package/scripts/test-mcp.js +163 -0
- package/scripts/test-pre-publish-quick.sh +75 -0
- package/scripts/test-quality-gates.sh +263 -0
- package/scripts/test-railway-db.js +222 -0
- package/scripts/test-redis-storage.ts +490 -0
- package/scripts/test-rlm-basic.sh +122 -0
- package/scripts/test-rlm-comprehensive.sh +260 -0
- package/scripts/test-rlm-e2e.sh +268 -0
- package/scripts/test-rlm-simple.js +90 -0
- package/scripts/test-rlm.js +110 -0
- package/scripts/test-session-handoff.sh +165 -0
- package/scripts/test-shell-integration.sh +275 -0
- package/scripts/testing/ab-test-runner.ts +508 -0
- package/scripts/testing/collect-metrics.ts +457 -0
- package/scripts/testing/quick-effectiveness-demo.js +187 -0
- package/scripts/testing/real-performance-test.js +422 -0
- package/scripts/testing/run-effectiveness-tests.sh +176 -0
- package/scripts/testing/scripts/testing/ab-test-runner.js +363 -0
- package/scripts/testing/scripts/testing/collect-metrics.js +292 -0
- package/scripts/testing/simple-effectiveness-test.js +310 -0
- package/scripts/testing/src/core/context/context-bridge.js +253 -0
- package/scripts/testing/src/core/context/frame-manager.js +746 -0
- package/scripts/testing/src/core/context/shared-context-layer.js +437 -0
- package/scripts/testing/src/core/database/database-adapter.js +54 -0
- package/scripts/testing/src/core/errors/index.js +291 -0
- package/scripts/testing/src/core/errors/recovery.js +268 -0
- package/scripts/testing/src/core/monitoring/logger.js +145 -0
- package/scripts/testing/src/core/retrieval/context-retriever.js +516 -0
- package/scripts/testing/src/core/session/index.js +1 -0
- package/scripts/testing/src/core/session/session-manager.js +323 -0
- package/scripts/testing/src/core/trace/cli-trace-wrapper.js +140 -0
- package/scripts/testing/src/core/trace/db-trace-wrapper.js +251 -0
- package/scripts/testing/src/core/trace/debug-trace.js +398 -0
- package/scripts/testing/src/core/trace/index.js +120 -0
- package/scripts/testing/src/core/trace/linear-api-wrapper.js +204 -0
- package/scripts/update-linear-status.js +268 -0
- package/scripts/update-linear-tasks-fixed.js +284 -0
- package/templates/claude-hooks/hooks.json +5 -0
- package/templates/claude-hooks/on-clear.js +56 -0
- package/templates/claude-hooks/on-startup.js +56 -0
- package/templates/claude-hooks/tool-use-trace.js +67 -0
- package/dist/features/tui/components/analytics-panel.js +0 -157
- package/dist/features/tui/components/analytics-panel.js.map +0 -7
- package/dist/features/tui/components/frame-visualizer.js +0 -377
- package/dist/features/tui/components/frame-visualizer.js.map +0 -7
- package/dist/features/tui/components/pr-tracker.js +0 -135
- package/dist/features/tui/components/pr-tracker.js.map +0 -7
- package/dist/features/tui/components/session-monitor.js +0 -299
- package/dist/features/tui/components/session-monitor.js.map +0 -7
- package/dist/features/tui/components/subagent-fleet.js +0 -395
- package/dist/features/tui/components/subagent-fleet.js.map +0 -7
- package/dist/features/tui/components/task-board.js +0 -1139
- package/dist/features/tui/components/task-board.js.map +0 -7
- package/dist/features/tui/index.js +0 -408
- package/dist/features/tui/index.js.map +0 -7
- package/dist/features/tui/services/data-service.js +0 -641
- package/dist/features/tui/services/data-service.js.map +0 -7
- package/dist/features/tui/services/linear-task-reader.js +0 -102
- package/dist/features/tui/services/linear-task-reader.js.map +0 -7
- package/dist/features/tui/services/websocket-client.js +0 -162
- package/dist/features/tui/services/websocket-client.js.map +0 -7
- package/dist/features/tui/terminal-compat.js +0 -220
- package/dist/features/tui/terminal-compat.js.map +0 -7
- package/dist/features/tui/types.js +0 -1
- package/dist/features/tui/types.js.map +0 -7
package/dist/cli/index.js.map
CHANGED
|
@@ -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": [
|
|
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 { 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);\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": ";AAOA,QAAQ,IAAI,iBAAiB,IAAI;AAGjC,OAAO;AAGP,SAAS,mBAAmB,aAAa;AACzC,kBAAkB;AAElB,SAAS,eAAe;AACxB,SAAS,cAAc;AACvB,SAAS,oBAAoB;AAC7B,SAAS,gBAAgB,sBAAsB;AAC/C,SAAS,0BAA0B;AACnC,SAAS,qBAAqB;AAC9B,SAAS,uBAAuB;AAChC,SAAS,+BAA+B;AACxC,SAAS,8BAA8B;AACvC,SAAS,6BAA6B;AACtC,SAAS,gCAAgC;AACzC,SAAS,iCAAiC;AAC1C,SAAS,0BAA0B;AACnC,SAAS,2BAA2B;AACpC,SAAS,wBAAwB;AACjC,SAAS,6BAA6B;AACtC,SAAS,2BAA2B;AACpC,SAAS,4BAA4B;AACrC,SAAS,4BAA4B;AACrC,SAAS,2BAA2B;AACpC,SAAS,yBAAyB;AAClC,OAAO,kBAAkB;AACzB,OAAO,2BAA2B;AAClC,OAAO,oBAAoB;AAC3B,OAAO,oBAAoB;AAC3B,SAAS,sBAAsB;AAC/B,OAAO,cAAc;AACrB,SAAS,YAAY;AACrB,SAAS,YAAY,iBAAiB;AAEtC,MAAM,UAAU;AAGhB,cAAc,gBAAgB,SAAS,IAAI,EAAE,MAAM,MAAM;AAEzD,CAAC;AAED,QACG,KAAK,aAAa,EAClB;AAAA,EACC;AACF,EACC,QAAQ,OAAO;AAElB,QACG,QAAQ,MAAM,EACd,YAAY,2CAA2C,EACvD,OAAO,YAAY;AAClB,MAAI;AACF,UAAM,cAAc,QAAQ,IAAI;AAChC,UAAM,QAAQ,KAAK,aAAa,cAAc;AAE9C,QAAI,CAAC,WAAW,KAAK,GAAG;AACtB,gBAAU,OAAO,EAAE,WAAW,KAAK,CAAC;AAAA,IACtC;AAEA,UAAM,SAAS,KAAK,OAAO,YAAY;AACvC,UAAM,KAAK,IAAI,SAAS,MAAM;AAC9B,QAAI,aAAa,IAAI,aAAa;AAElC,WAAO,KAAK,wCAAwC,EAAE,YAAY,CAAC;AACnE,YAAQ,IAAI,qCAAgC,WAAW;AAEvD,OAAG,MAAM;AAAA,EACX,SAAS,OAAgB;AACvB,WAAO,MAAM,oCAAoC,KAAc;AAC/D,YAAQ,MAAM,iCAA6B,MAAgB,OAAO;AAClE,YAAQ,KAAK,CAAC;AAAA,EAChB;AACF,CAAC;AAEH,QACG,QAAQ,QAAQ,EAChB,YAAY,iCAAiC,EAC7C,OAAO,SAAS,wCAAwC,EACxD,OAAO,aAAa,2CAA2C,EAC/D,OAAO,kBAAkB,kCAAkC,EAC3D,OAAO,OAAO,YAAY;AACzB,SAAO,MAAM,QAAQ,sBAAsB,SAAS,YAAY;AAC9D,QAAI;AACF,YAAM,cAAc,QAAQ,IAAI;AAChC,YAAM,SAAS,KAAK,aAAa,gBAAgB,YAAY;AAE7D,UAAI,CAAC,WAAW,MAAM,GAAG;AACvB,gBAAQ;AAAA,UACN;AAAA,QACF;AACA;AAAA,MACF;AAGA,YAAM,cAAc,gBAAgB,OAAO;AAG3C,YAAM,eAAe,WAAW;AAChC,YAAM,mBAAmB,WAAW;AAEpC,YAAM,UAAU,MAAM,eAAe,mBAAmB;AAAA,QACtD,aAAa;AAAA,QACb,WAAW,QAAQ;AAAA,MACrB,CAAC;AAGD,YAAM,mBAAmB,MAAM,mBAAmB,oBAAoB;AAGtE,UACE,iBAAiB,oBACjB,iBAAiB,eAAe,GAChC;AACA,gBAAQ,IAAI;AAAA,oCAAgC;AAC5C,gBAAQ;AAAA,UACN,MAAM,iBAAiB,YAAY;AAAA,QACrC;AAEA,YAAI,iBAAiB,eAAe,SAAS,GAAG;AAC9C,kBAAQ,IAAI,qBAAqB;AACjC,2BAAiB,eAAe,MAAM,GAAG,CAAC,EAAE,QAAQ,CAAC,MAAM;AACzD,oBAAQ;AAAA,cACN,eAAU,EAAE,IAAI,KAAK,EAAE,QAAQ,MAAM,GAAG,EAAE,CAAC,KAAK,EAAE,SAAS;AAAA,YAC7D;AAAA,UACF,CAAC;AAAA,QACH;AAEA,YAAI,iBAAiB,cAAc,SAAS,GAAG;AAC7C,kBAAQ;AAAA,YACN,qBAAqB,iBAAiB,cAAc,CAAC,EAAE,SAAS,MAAM,GAAG,EAAE,CAAC;AAAA,UAC9E;AAAA,QACF;AAAA,MACF;AAEA,YAAM,KAAK,IAAI,SAAS,MAAM;AAC9B,YAAM,eAAe,IAAI,aAAa,IAAI,QAAQ,SAAS;AAG3D,UAAI,QAAQ,KAAK;AACf,qBAAa,aAAa,eAAe,UAAU;AAAA,MACrD,WAAW,QAAQ,SAAS;AAC1B,qBAAa,aAAa,eAAe,cAAc;AAAA,MACzD;AAEA,YAAM,eAAe,aAAa,mBAAmB;AACrD,YAAM,aAAa,aAAa,cAAc;AAG9C,YAAM,aAAa,GAChB;AAAA,QACC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MASF,EACC,IAAI,QAAQ,SAAS;AAOxB,YAAM,eAAe,GAClB;AAAA,QACC;AAAA;AAAA;AAAA,MAGF,EACC,IAAI;AAEP,YAAM,aAAa,GAChB;AAAA,QACC;AAAA;AAAA;AAAA;AAAA;AAAA,MAKF,EACC,IAAI,QAAQ,SAAS;AAExB,cAAQ,IAAI,+BAAwB;AACpC,cAAQ;AAAA,QACN,eAAe,QAAQ,UAAU,MAAM,GAAG,CAAC,CAAC,KAAK,QAAQ,KAAK,KAAK,KAAK,OAAO,KAAK,IAAI,IAAI,QAAQ,aAAa,MAAO,EAAE,CAAC;AAAA,MAC7H;AACA,cAAQ,IAAI,eAAe,QAAQ,SAAS,EAAE;AAC9C,UAAI,QAAQ,QAAQ;AAClB,gBAAQ,IAAI,cAAc,QAAQ,MAAM,EAAE;AAAA,MAC5C;AAGA,cAAQ,IAAI;AAAA,uCAA0C;AACtD,cAAQ;AAAA,QACN,gBAAgB,WAAW,gBAAgB,CAAC,KAAK,WAAW,iBAAiB,CAAC,YAAY,WAAW,iBAAiB,CAAC;AAAA,MACzH;AACA,cAAQ,IAAI,gBAAgB,WAAW,SAAS,CAAC,EAAE;AACnD,cAAQ,IAAI,kBAAkB,WAAW,kBAAkB,CAAC,EAAE;AAC9D,cAAQ;AAAA,QACN,yBAAyB,aAAa,SAAS,CAAC;AAAA,MAClD;AAGA,YAAM,eAAe,GAClB;AAAA,QACC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MAOF,EACC,IAAI,QAAQ,SAAS;AAOxB,UAAI,aAAa,SAAS,GAAG;AAC3B,gBAAQ,IAAI;AAAA,oBAAuB;AACnC,qBAAa,QAAQ,CAAC,MAAM;AAC1B,gBAAM,YAAY,EAAE,UAAU,WAAW,cAAO;AAChD,kBAAQ;AAAA,YACN,QAAQ,SAAS,IAAI,EAAE,IAAI,KAAK,EAAE,IAAI,OAAO,EAAE,OAAO;AAAA,UACxD;AAAA,QACF,CAAC;AAAA,MACH;AAEA,cAAQ,IAAI;AAAA,oBAAuB;AACnC,cAAQ,IAAI,qBAAqB,UAAU,EAAE;AAC7C,cAAQ,IAAI,uBAAuB,aAAa,MAAM,EAAE;AAExD,UAAI,aAAa,SAAS,GAAG;AAC3B,qBAAa,QAAQ,CAAC,OAAO,MAAM;AACjC,gBAAM,SAAS,UAAU,KAAK,OAAO,MAAM,SAAS,CAAC;AACrD,gBAAM,SAAS,MAAM,IAAI,iBAAO;AAChC,kBAAQ,IAAI,GAAG,MAAM,GAAG,MAAM,IAAI,MAAM,IAAI,KAAK,MAAM,IAAI,GAAG;AAAA,QAChE,CAAC;AAAA,MACH;AAGA,UAAI,CAAC,QAAQ,OAAO,CAAC,QAAQ,SAAS;AACpC,cAAM,gBAAgB,MAAM,eAAe,aAAa;AAAA,UACtD,WAAW,QAAQ;AAAA,UACnB,OAAO;AAAA,QACT,CAAC;AAED,cAAM,cAAc,cAAc;AAAA,UAChC,CAAC,MAAM,EAAE,cAAc,QAAQ;AAAA,QACjC;AACA,YAAI,YAAY,SAAS,GAAG;AAC1B,kBAAQ,IAAI;AAAA,yCAA4C;AACxD,sBAAY,QAAQ,CAAC,MAAM;AACzB,kBAAM,MAAM,KAAK;AAAA,eACd,KAAK,IAAI,IAAI,EAAE,gBAAgB,MAAO,KAAK;AAAA,YAC9C;AACA,oBAAQ;AAAA,cACN,UAAU,EAAE,UAAU,MAAM,GAAG,CAAC,CAAC,KAAK,EAAE,UAAU,MAAM,KAAK,GAAG;AAAA,YAClE;AAAA,UACF,CAAC;AACD,kBAAQ,IAAI;AAAA,gDAAmD;AAAA,QACjE;AAAA,MACF;AAEA,SAAG,MAAM;AAAA,IACX,SAAS,OAAgB;AACvB,aAAO,MAAM,wBAAwB,KAAc;AACnD,cAAQ,MAAM,+BAA2B,MAAgB,OAAO;AAChE,cAAQ,KAAK,CAAC;AAAA,IAChB;AAAA,EACF,CAAC;AACH,CAAC;AAEH,QACG,QAAQ,cAAc,EACtB,YAAY,+BAA+B,EAC3C,OAAO,YAAY;AAClB,MAAI;AACF,YAAQ,IAAI,mCAA4B;AACxC,UAAM,cAAc,WAAW,OAAO;AAAA,EACxC,SAAS,OAAgB;AACvB,WAAO,MAAM,uBAAuB,KAAc;AAClD,YAAQ,MAAM,+BAA2B,MAAgB,OAAO;AAChE,YAAQ,KAAK,CAAC;AAAA,EAChB;AACF,CAAC;AAEH,QACG,QAAQ,UAAU,EAClB,YAAY,0CAA0C,EACtD,OAAO,YAAY;AAClB,MAAI;AACF,UAAM,cAAc,QAAQ,IAAI;AAChC,UAAM,SAAS,KAAK,aAAa,gBAAgB,YAAY;AAE7D,QAAI,CAAC,WAAW,MAAM,GAAG;AACvB,cAAQ;AAAA,QACN;AAAA,MACF;AACA;AAAA,IACF;AAEA,UAAM,WAAW,IAAI,gBAAgB,WAAW;AAChD,YAAQ,IAAI,SAAS,WAAW,CAAC;AAAA,EACnC,SAAS,OAAgB;AACvB,WAAO,MAAM,2BAA2B,KAAc;AACtD,YAAQ,MAAM,mCAA+B,MAAgB,OAAO;AACpE,YAAQ,KAAK,CAAC;AAAA,EAChB;AACF,CAAC;AAEH,QACG,QAAQ,YAAY,EACpB,YAAY,iDAAiD,EAC7D,OAAO,wBAAwB,0BAA0B,QAAQ,IAAI,CAAC,EACtE,OAAO,OAAO,YAAY;AACzB,MAAI;AACF,UAAM,EAAE,aAAa,IAAI,MAAM,OAAO,+BAA+B;AAGrE,YAAQ,IAAI,cAAc,IAAI,QAAQ;AAEtC,YAAQ,IAAI,8CAAuC;AACnD,YAAQ,IAAI,eAAe,QAAQ,OAAO,EAAE;AAC5C,YAAQ,IAAI,eAAe,OAAO,EAAE;AAGpC,kBAAc,gBAAgB,SAAS,IAAI,EAAE,MAAM,MAAM;AAAA,IAAC,CAAC;AAG3D,UAAM,aAAa;AAAA,EACrB,SAAS,OAAgB;AACvB,WAAO,MAAM,8BAA8B,KAAc;AACzD,YAAQ,MAAM,6BAAyB,MAAgB,OAAO;AAC9D,YAAQ,KAAK,CAAC;AAAA,EAChB;AACF,CAAC;AAGH,QACG,QAAQ,cAAc,EACtB,YAAY,oDAAoD,EAChE,OAAO,YAAY;AAClB,MAAI;AACF,UAAM,cAAc,QAAQ,IAAI;AAChC,UAAM,SAAS,KAAK,aAAa,gBAAgB,YAAY;AAE7D,QAAI,CAAC,WAAW,MAAM,GAAG;AACvB,cAAQ;AAAA,QACN;AAAA,MACF;AACA;AAAA,IACF;AAEA,UAAM,KAAK,IAAI,SAAS,MAAM;AAC9B,UAAM,eAAe,IAAI,aAAa,IAAI,aAAa;AAGvD,YAAQ,IAAI,2CAAoC;AAEhD,UAAM,YAAY,aAAa,YAAY;AAAA,MACzC,MAAM;AAAA,MACN,MAAM;AAAA,MACN,QAAQ,EAAE,MAAM,MAAM,YAAW,oBAAI,KAAK,GAAE,YAAY,EAAE;AAAA,IAC5D,CAAC;AAED,UAAM,YAAY,aAAa,YAAY;AAAA,MACzC,MAAM;AAAA,MACN,MAAM;AAAA,MACN,QAAQ,EAAE,aAAa,8BAA8B;AAAA,MACrD,eAAe;AAAA,IACjB,CAAC;AAED,UAAM,eAAe,aAAa,YAAY;AAAA,MAC5C,MAAM;AAAA,MACN,MAAM;AAAA,MACN,QAAQ,EAAE,MAAM,CAAC,QAAQ,EAAE;AAAA,MAC3B,eAAe;AAAA,IACjB,CAAC;AAGD,iBAAa;AAAA,MACX;AAAA,MACA;AAAA,QACE,SAAS;AAAA,MACX;AAAA,MACA;AAAA,IACF;AAEA,YAAQ,IAAI,6BAAwB;AACpC,YAAQ,IAAI,0BAAmB,aAAa,cAAc,CAAC,EAAE;AAC7D,YAAQ;AAAA,MACN,4BAAqB,aAAa,mBAAmB,EAAE,MAAM;AAAA,IAC/D;AAGA,iBAAa,WAAW,YAAY;AACpC,YAAQ;AAAA,MACN,kDAA2C,aAAa,cAAc,CAAC;AAAA,IACzE;AAEA,OAAG,MAAM;AAAA,EACX,SAAS,OAAgB;AACvB,WAAO,MAAM,uBAAuB,KAAc;AAClD,YAAQ,MAAM,uBAAmB,MAAgB,OAAO;AACxD,YAAQ,KAAK,CAAC;AAAA,EAChB;AACF,CAAC;AAIH,0BAA0B,OAAO;AACjC,wBAAwB,OAAO;AAC/B,yBAAyB,OAAO;AAGhC,uBAAuB,OAAO;AAG9B,QAAQ,WAAW,sBAAsB,CAAC;AAG1C,QAAQ,WAAW,mBAAmB,CAAC;AACvC,QAAQ,WAAW,oBAAoB,CAAC;AACxC,QAAQ,WAAW,iBAAiB,CAAC;AACrC,QAAQ,WAAW,sBAAsB,CAAC;AAC1C,QAAQ,WAAW,oBAAoB,CAAC;AACxC,QAAQ,WAAW,qBAAqB,CAAC;AACzC,QAAQ,WAAW,qBAAqB,CAAC;AACzC,QAAQ,WAAW,oBAAoB,CAAC;AACxC,QAAQ,WAAW,kBAAkB,CAAC;AACtC,QAAQ,WAAW,YAAY;AAC/B,QAAQ,WAAW,sBAAsB,CAAC;AAC1C,QAAQ,WAAW,cAAc;AACjC,QAAQ,WAAW,cAAc;AAGjC,QACG,QAAQ,WAAW,EACnB,YAAY,0CAA0C,EACtD,OAAO,eAAe,wBAAwB,EAC9C,OAAO,4BAA4B,+BAA+B,GAAG,EACrE,OAAO,OAAO,YAAY;AACzB,QAAM,EAAE,iBAAiB,IAAI,MAAM,OAAO,yBAAyB;AACnE,QAAM,iBAAiB,QAAQ,OAAO;AACxC,CAAC;AAGH,IAAI,QAAQ,KAAK,SAAS,GAAG;AAC3B,QAAM,UAAU,eAAe,YAAY;AAC3C,UAAQ,cAAc,EAAE,MAAM,MAAM;AAAA,EAEpC,CAAC;AACH;AAGA,MAAM,eACJ,YAAY,QAAQ,UAAU,QAAQ,KAAK,CAAC,CAAC,MAC7C,QAAQ,KAAK,CAAC,GAAG,SAAS,cAAc,KACxC,QAAQ,KAAK,CAAC,GAAG,SAAS,UAAU,KACpC,QAAQ,KAAK,CAAC,GAAG,SAAS,KAAK;AAEjC,IAAI,cAAc;AAChB,UAAQ,MAAM;AAChB;",
|
|
6
|
+
"names": []
|
|
7
7
|
}
|
|
@@ -111,7 +111,7 @@ class DualStackManager {
|
|
|
111
111
|
}
|
|
112
112
|
}
|
|
113
113
|
async executeSchemaQuery(sql) {
|
|
114
|
-
logger.
|
|
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
|
}
|
|
@@ -77,6 +77,9 @@ class FrameManager {
|
|
|
77
77
|
runId: this.currentRunId
|
|
78
78
|
});
|
|
79
79
|
try {
|
|
80
|
+
if (!this.db || typeof this.db.exec !== "function") {
|
|
81
|
+
throw new Error("Database not properly initialized. Expected SQLite Database instance with exec() method.");
|
|
82
|
+
}
|
|
80
83
|
this.db.exec(`
|
|
81
84
|
CREATE TABLE IF NOT EXISTS frames (
|
|
82
85
|
frame_id TEXT PRIMARY KEY,
|