@stackmemoryai/stackmemory 0.5.33 → 0.5.34
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/dist/agents/core/agent-task-manager.js.map +1 -1
- package/dist/cli/commands/clear.js +1 -1
- package/dist/cli/commands/clear.js.map +1 -1
- package/dist/cli/commands/context.js +1 -1
- package/dist/cli/commands/context.js.map +1 -1
- package/dist/cli/commands/dashboard.js.map +1 -1
- package/dist/cli/commands/discovery.js +1 -1
- package/dist/cli/commands/discovery.js.map +1 -1
- package/dist/cli/commands/handoff.js +1 -1
- package/dist/cli/commands/handoff.js.map +1 -1
- package/dist/cli/commands/monitor.js +1 -1
- package/dist/cli/commands/monitor.js.map +1 -1
- package/dist/cli/commands/quality.js +1 -1
- package/dist/cli/commands/quality.js.map +1 -1
- package/dist/cli/commands/skills.js +1 -1
- package/dist/cli/commands/skills.js.map +1 -1
- package/dist/cli/commands/workflow.js +1 -1
- package/dist/cli/commands/workflow.js.map +1 -1
- package/dist/cli/commands/worktree.js +1 -1
- package/dist/cli/commands/worktree.js.map +1 -1
- package/dist/cli/index.js +1 -1
- package/dist/cli/index.js.map +1 -1
- package/dist/core/context/auto-context.js.map +1 -1
- package/dist/core/context/compaction-handler.js.map +2 -2
- package/dist/core/context/context-bridge.js.map +2 -2
- 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/enhanced-rehydration.js.map +1 -1
- package/dist/core/context/frame-database.js +8 -0
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-handoff-manager.js.map +1 -1
- package/dist/core/context/frame-lifecycle-hooks.js +119 -0
- package/dist/core/context/frame-lifecycle-hooks.js.map +7 -0
- package/dist/core/context/frame-stack.js +29 -0
- package/dist/core/context/frame-stack.js.map +2 -2
- package/dist/core/context/incremental-gc.js.map +2 -2
- package/dist/core/context/index.js +4 -22
- package/dist/core/context/index.js.map +2 -2
- package/dist/core/context/refactored-frame-manager.js +140 -34
- package/dist/core/context/refactored-frame-manager.js.map +3 -3
- package/dist/core/context/shared-context-layer.js.map +1 -1
- package/dist/core/context/stack-merge-resolver.js.map +1 -1
- package/dist/core/database/database-adapter.js.map +1 -1
- package/dist/core/database/paradedb-adapter.js.map +1 -1
- package/dist/core/database/query-router.js.map +1 -1
- package/dist/core/database/sqlite-adapter.js.map +1 -1
- package/dist/core/digest/frame-digest-integration.js.map +1 -1
- package/dist/core/digest/hybrid-digest-generator.js.map +1 -1
- package/dist/core/digest/types.js.map +1 -1
- package/dist/core/errors/index.js +249 -0
- package/dist/core/errors/index.js.map +2 -2
- package/dist/core/frame/workflow-templates.js.map +2 -2
- package/dist/core/merge/conflict-detector.js.map +1 -1
- package/dist/core/merge/resolution-engine.js.map +1 -1
- package/dist/core/merge/stack-diff.js.map +1 -1
- package/dist/core/models/model-router.js +10 -1
- package/dist/core/models/model-router.js.map +2 -2
- package/dist/core/monitoring/error-handler.js +37 -270
- package/dist/core/monitoring/error-handler.js.map +3 -3
- package/dist/core/monitoring/session-monitor.js.map +1 -1
- package/dist/core/performance/lazy-context-loader.js.map +1 -1
- package/dist/core/performance/optimized-frame-context.js.map +1 -1
- package/dist/core/retrieval/context-retriever.js.map +1 -1
- package/dist/core/retrieval/graph-retrieval.js.map +1 -1
- package/dist/core/retrieval/hierarchical-retrieval.js.map +1 -1
- package/dist/core/retrieval/llm-context-retrieval.js.map +1 -1
- package/dist/core/retrieval/retrieval-benchmarks.js.map +1 -1
- package/dist/core/retrieval/summary-generator.js.map +1 -1
- package/dist/core/retrieval/types.js.map +1 -1
- package/dist/core/storage/chromadb-adapter.js.map +1 -1
- package/dist/core/storage/infinite-storage.js.map +1 -1
- package/dist/core/storage/two-tier-storage.js.map +1 -1
- package/dist/features/tasks/task-aware-context.js.map +1 -1
- package/dist/features/web/server/index.js +1 -1
- package/dist/features/web/server/index.js.map +1 -1
- package/dist/hooks/schemas.js +50 -0
- package/dist/hooks/schemas.js.map +2 -2
- package/dist/hooks/sms-action-runner.js +4 -0
- package/dist/hooks/sms-action-runner.js.map +2 -2
- package/dist/hooks/whatsapp-commands.js +142 -2
- package/dist/hooks/whatsapp-commands.js.map +2 -2
- package/dist/hooks/whatsapp-sync.js +34 -0
- package/dist/hooks/whatsapp-sync.js.map +2 -2
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/integrations/mcp/handlers/context-handlers.js.map +1 -1
- package/dist/integrations/mcp/handlers/discovery-handlers.js.map +1 -1
- package/dist/integrations/mcp/server.js +1 -1
- package/dist/integrations/mcp/server.js.map +1 -1
- package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js +1 -1
- package/dist/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +1 -1
- package/dist/integrations/ralph/context/stackmemory-context-loader.js +1 -1
- package/dist/integrations/ralph/context/stackmemory-context-loader.js.map +1 -1
- package/dist/integrations/ralph/learning/pattern-learner.js +1 -1
- package/dist/integrations/ralph/learning/pattern-learner.js.map +1 -1
- package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js +1 -1
- package/dist/integrations/ralph/orchestration/multi-loop-orchestrator.js.map +1 -1
- package/dist/integrations/ralph/swarm/swarm-coordinator.js +1 -1
- package/dist/integrations/ralph/swarm/swarm-coordinator.js.map +1 -1
- package/dist/integrations/ralph/visualization/ralph-debugger.js +1 -1
- package/dist/integrations/ralph/visualization/ralph-debugger.js.map +1 -1
- package/dist/mcp/stackmemory-mcp-server.js +1 -1
- package/dist/mcp/stackmemory-mcp-server.js.map +1 -1
- package/dist/skills/claude-skills.js.map +1 -1
- package/dist/skills/recursive-agent-orchestrator.js.map +1 -1
- package/dist/skills/unified-rlm-orchestrator.js.map +1 -1
- package/package.json +1 -1
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/handoff.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Handoff command - Commits work and generates a prompt for the next session\n */\n\nimport { Command } from 'commander';\nimport { execSync, execFileSync } from 'child_process';\nimport {\n existsSync,\n readFileSync,\n writeFileSync,\n mkdirSync,\n readdirSync,\n unlinkSync,\n} from 'fs';\nimport { join } from 'path';\nimport Database from 'better-sqlite3';\nimport { z } from 'zod';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport { LinearTaskManager } from '../../features/tasks/linear-task-manager.js';\nimport { logger } from '../../core/monitoring/logger.js';\nimport { EnhancedHandoffGenerator } from '../../core/session/enhanced-handoff.js';\n\n// Handoff versioning - keep last N handoffs\nconst MAX_HANDOFF_VERSIONS = 10;\n\nfunction saveVersionedHandoff(\n projectRoot: string,\n branch: string,\n content: string\n): string {\n const handoffsDir = join(projectRoot, '.stackmemory', 'handoffs');\n if (!existsSync(handoffsDir)) {\n mkdirSync(handoffsDir, { recursive: true });\n }\n\n // Generate versioned filename: YYYY-MM-DD-HH-mm-branch.md\n const now = new Date();\n const timestamp = now.toISOString().slice(0, 16).replace(/[T:]/g, '-');\n const safeBranch = branch.replace(/[^a-zA-Z0-9-]/g, '-').slice(0, 30);\n const filename = `${timestamp}-${safeBranch}.md`;\n const versionedPath = join(handoffsDir, filename);\n\n // Save versioned handoff\n writeFileSync(versionedPath, content);\n\n // Clean up old handoffs (keep last N)\n try {\n const files = readdirSync(handoffsDir)\n .filter((f) => f.endsWith('.md'))\n .sort()\n .reverse();\n\n for (const oldFile of files.slice(MAX_HANDOFF_VERSIONS)) {\n unlinkSync(join(handoffsDir, oldFile));\n }\n } catch {\n // Cleanup failed, not critical\n }\n\n return versionedPath;\n}\n\n// Input validation schemas\nconst CommitMessageSchema = z\n .string()\n .min(1, 'Commit message cannot be empty')\n .max(200, 'Commit message too long')\n .regex(\n /^[a-zA-Z0-9\\s\\-_.,:()\\/\\[\\]]+$/,\n 'Commit message contains invalid characters'\n )\n .refine(\n (msg) => !msg.includes('\\n'),\n 'Commit message cannot contain newlines'\n )\n .refine(\n (msg) => !msg.includes('\"'),\n 'Commit message cannot contain double quotes'\n )\n .refine(\n (msg) => !msg.includes('`'),\n 'Commit message cannot contain backticks'\n );\n\nexport function createHandoffCommand(): Command {\n const cmd = new Command('handoff');\n\n cmd.description('Session handoff for continuity between Claude sessions');\n\n // Default action - capture handoff\n cmd\n .command('capture', { isDefault: true })\n .description('Commit current work and generate a handoff prompt')\n .option('-m, --message <message>', 'Custom commit message')\n .option('--no-commit', 'Skip git commit')\n .option('--copy', 'Copy the handoff prompt to clipboard')\n .option('--basic', 'Use basic handoff format instead of enhanced')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n // 1. Check git status\n let gitStatus = '';\n let hasChanges = false;\n\n try {\n gitStatus = execSync('git status --short', {\n encoding: 'utf-8',\n cwd: projectRoot,\n });\n hasChanges = gitStatus.trim().length > 0;\n } catch {\n console.log('\u26A0\uFE0F Not in a git repository');\n }\n\n // 2. Commit if there are changes and not skipped\n if (hasChanges && options.commit !== false) {\n try {\n // Get current branch\n const currentBranch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n // Stage all changes\n execSync('git add -A', { cwd: projectRoot });\n\n // Generate or use custom commit message\n let commitMessage =\n options.message ||\n `chore: handoff checkpoint on ${currentBranch}`;\n\n // Validate commit message\n try {\n commitMessage = CommitMessageSchema.parse(commitMessage);\n } catch (validationError) {\n console.error(\n '\u274C Invalid commit message:',\n (validationError as Error).message\n );\n return;\n }\n\n // Commit using execFileSync for safety\n execFileSync('git', ['commit', '-m', commitMessage], {\n cwd: projectRoot,\n stdio: 'inherit',\n });\n\n console.log(`\u2705 Committed changes: \"${commitMessage}\"`);\n console.log(` Branch: ${currentBranch}`);\n } catch (err: unknown) {\n console.error(\n '\u274C Failed to commit changes:',\n (err as Error).message\n );\n }\n } else if (!hasChanges) {\n console.log('\u2139\uFE0F No changes to commit');\n }\n\n // 3. Gather context for handoff prompt\n let contextSummary = '';\n let tasksSummary = '';\n let recentWork = '';\n\n if (existsSync(dbPath)) {\n const db = new Database(dbPath);\n\n // Get recent context\n const frameManager = new FrameManager(db, 'cli-project');\n const activeFrames = frameManager.getActiveFramePath();\n\n if (activeFrames.length > 0) {\n contextSummary = 'Active context frames:\\n';\n activeFrames.forEach((frame) => {\n contextSummary += ` - ${frame.name} [${frame.type}]\\n`;\n });\n }\n\n // Get task status\n const taskStore = new LinearTaskManager(projectRoot, db);\n const activeTasks = taskStore.getActiveTasks();\n\n const inProgress = activeTasks.filter(\n (t: any) => t.status === 'in_progress'\n );\n const todo = activeTasks.filter((t: any) => t.status === 'pending');\n const recentlyCompleted = activeTasks\n .filter((t: any) => t.status === 'completed' && t.completed_at)\n .sort(\n (a: any, b: any) => (b.completed_at || 0) - (a.completed_at || 0)\n )\n .slice(0, 3);\n\n if (inProgress.length > 0 || todo.length > 0) {\n tasksSummary = '\\nTasks:\\n';\n\n if (inProgress.length > 0) {\n tasksSummary += 'In Progress:\\n';\n inProgress.forEach((t: any) => {\n const externalId = t.external_refs?.linear?.id;\n tasksSummary += ` - ${t.title}${externalId ? ` [${externalId}]` : ''}\\n`;\n });\n }\n\n if (todo.length > 0) {\n tasksSummary += 'TODO:\\n';\n todo.slice(0, 5).forEach((t: any) => {\n const externalId = t.external_refs?.linear?.id;\n tasksSummary += ` - ${t.title}${externalId ? ` [${externalId}]` : ''}\\n`;\n });\n if (todo.length > 5) {\n tasksSummary += ` ... and ${todo.length - 5} more\\n`;\n }\n }\n }\n\n if (recentlyCompleted.length > 0) {\n recentWork = '\\nRecently Completed:\\n';\n recentlyCompleted.forEach((t: any) => {\n recentWork += ` \u2713 ${t.title}\\n`;\n });\n }\n\n // Get recent events\n const recentEvents = db\n .prepare(\n `\n SELECT event_type as type, payload as data, datetime(ts, 'unixepoch') as time\n FROM events\n ORDER BY ts DESC\n LIMIT 5\n `\n )\n .all() as any[];\n\n if (recentEvents.length > 0) {\n recentWork += '\\nRecent Activity:\\n';\n recentEvents.forEach((event) => {\n const data = JSON.parse(event.data);\n recentWork += ` - ${event.type}: ${data.message || data.name || 'activity'}\\n`;\n });\n }\n\n db.close();\n }\n\n // 4. Get current git info\n let gitInfo = '';\n try {\n const branch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n const lastCommit = execSync('git log -1 --oneline', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n gitInfo = `\\nGit Status:\\n Branch: ${branch}\\n Last commit: ${lastCommit}\\n`;\n } catch {\n // Ignore git errors\n }\n\n // 5. Check for any blockers or notes\n let notes = '';\n const notesPath = join(projectRoot, '.stackmemory', 'handoff.md');\n if (existsSync(notesPath)) {\n const handoffNotes = readFileSync(notesPath, 'utf-8');\n if (handoffNotes.trim()) {\n notes = `\\nNotes from previous handoff:\\n${handoffNotes}\\n`;\n }\n }\n\n // 6. Generate the handoff prompt\n let handoffPrompt: string;\n\n if (options.basic) {\n // Use basic handoff format\n const timestamp = new Date().toISOString();\n handoffPrompt = `# Session Handoff - ${timestamp}\n\n## Project: ${projectRoot.split('/').pop()}\n\n${gitInfo}\n${contextSummary}\n${tasksSummary}\n${recentWork}\n${notes}\n\n## Continue from here:\n\n1. Run \\`stackmemory status\\` to check the current state\n2. Review any in-progress tasks above\n3. Check for any uncommitted changes with \\`git status\\`\n4. Resume work on the active context\n\n## Quick Commands:\n- \\`stackmemory context load --recent\\` - Load recent context\n- \\`stackmemory task list --state in_progress\\` - Show in-progress tasks\n- \\`stackmemory linear sync\\` - Sync with Linear if configured\n- \\`stackmemory log recent\\` - View recent activity\n\n---\nGenerated by stackmemory handoff at ${timestamp}\n`;\n } else {\n // Use high-efficacy enhanced handoff generator (default)\n const enhancedGenerator = new EnhancedHandoffGenerator(projectRoot);\n const enhancedHandoff = await enhancedGenerator.generate();\n handoffPrompt = enhancedGenerator.toMarkdown(enhancedHandoff);\n console.log(`Estimated tokens: ~${enhancedHandoff.estimatedTokens}`);\n }\n\n // 7. Save handoff prompt (both latest and versioned)\n const handoffPath = join(\n projectRoot,\n '.stackmemory',\n 'last-handoff.md'\n );\n writeFileSync(handoffPath, handoffPrompt);\n\n // Save versioned copy\n let branch = 'unknown';\n try {\n branch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n } catch {\n // Not a git repo\n }\n const versionedPath = saveVersionedHandoff(\n projectRoot,\n branch,\n handoffPrompt\n );\n console.log(\n `Versioned: ${versionedPath.split('/').slice(-2).join('/')}`\n );\n\n // 8. Display the prompt\n console.log('\\n' + '='.repeat(60));\n console.log(handoffPrompt);\n console.log('='.repeat(60));\n\n // 9. Copy to clipboard if requested\n if (options.copy) {\n try {\n // Use execFileSync with predefined commands for safety\n if (process.platform === 'darwin') {\n execFileSync('pbcopy', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else if (process.platform === 'win32') {\n execFileSync('clip', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else {\n execFileSync('xclip', ['-selection', 'clipboard'], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n }\n\n console.log('\\n\u2705 Handoff prompt copied to clipboard!');\n } catch {\n console.log('\\n\u26A0\uFE0F Could not copy to clipboard');\n }\n }\n\n console.log(`\\n\uD83D\uDCBE Handoff saved to: ${handoffPath}`);\n console.log('\uD83D\uDCCB Use this prompt when starting your next session');\n } catch (error: unknown) {\n logger.error('Handoff command failed', error as Error);\n console.error('\u274C Handoff failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n // Restore command\n cmd\n .command('restore')\n .description('Restore context from last handoff')\n .option('--no-copy', 'Do not copy prompt to clipboard')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const handoffPath = join(\n projectRoot,\n '.stackmemory',\n 'last-handoff.md'\n );\n const metaPath = join(\n process.env['HOME'] || '~',\n '.stackmemory',\n 'handoffs',\n 'last-handoff-meta.json'\n );\n\n if (!existsSync(handoffPath)) {\n console.log('\u274C No handoff found in this project');\n console.log('\uD83D\uDCA1 Run \"stackmemory handoff\" to create one');\n return;\n }\n\n // Read handoff prompt\n const handoffPrompt = readFileSync(handoffPath, 'utf-8');\n\n // Display the prompt\n console.log('\\n' + '='.repeat(60));\n console.log('\uD83D\uDCCB RESTORED HANDOFF');\n console.log('='.repeat(60));\n console.log(handoffPrompt);\n console.log('='.repeat(60));\n\n // Check for metadata\n if (existsSync(metaPath)) {\n const metadata = JSON.parse(readFileSync(metaPath, 'utf-8'));\n console.log('\\n\uD83D\uDCCA Session Metadata:');\n console.log(` Timestamp: ${metadata.timestamp}`);\n console.log(` Reason: ${metadata.reason}`);\n console.log(` Duration: ${metadata.session_duration}s`);\n console.log(` Command: ${metadata.command}`);\n }\n\n // Check current git status\n try {\n const gitStatus = execSync('git status --short', {\n encoding: 'utf-8',\n }).trim();\n if (gitStatus) {\n console.log('\\n\u26A0\uFE0F Current uncommitted changes:');\n console.log(gitStatus);\n }\n } catch {\n // Not a git repo\n }\n\n // Copy to clipboard unless disabled\n if (options.copy !== false) {\n try {\n // Use execFileSync with predefined commands for safety\n if (process.platform === 'darwin') {\n execFileSync('pbcopy', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else if (process.platform === 'win32') {\n execFileSync('clip', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else {\n execFileSync('xclip', ['-selection', 'clipboard'], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n }\n\n console.log('\\n\u2705 Handoff prompt copied to clipboard!');\n } catch {\n console.log('\\n\u26A0\uFE0F Could not copy to clipboard');\n }\n }\n\n console.log('\\n\uD83D\uDE80 Ready to continue where you left off!');\n } catch (error: unknown) {\n logger.error('Handoff restore failed', error as Error);\n console.error('\u274C Restore failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n // Auto command - enable auto-capture\n cmd\n .command('auto')\n .description('Enable auto-capture on session termination')\n .option('--command <command>', 'Command to wrap with auto-handoff')\n .action(async (options) => {\n const scriptPath = join(\n __dirname,\n '..',\n '..',\n '..',\n 'scripts',\n 'stackmemory-auto-handoff.sh'\n );\n\n if (!existsSync(scriptPath)) {\n console.error('\u274C Auto-handoff script not found');\n console.log('\uD83D\uDCE6 Please ensure StackMemory is properly installed');\n return;\n }\n\n console.log('\uD83D\uDEE1\uFE0F StackMemory Auto-Handoff');\n console.log('\u2500'.repeat(50));\n\n if (options.command) {\n // Validate and wrap specific command\n const commandSchema = z\n .string()\n .min(1, 'Command cannot be empty')\n .max(200, 'Command too long')\n .regex(\n /^[a-zA-Z0-9\\s\\-_./:]+$/,\n 'Command contains invalid characters'\n )\n .refine((cmd) => !cmd.includes(';'), 'Command cannot contain \";\"')\n .refine((cmd) => !cmd.includes('&'), 'Command cannot contain \"&\"')\n .refine((cmd) => !cmd.includes('|'), 'Command cannot contain \"|\"')\n .refine((cmd) => !cmd.includes('$'), 'Command cannot contain \"$\"')\n .refine((cmd) => !cmd.includes('`'), 'Command cannot contain \"`\"');\n\n try {\n const validatedCommand = commandSchema.parse(options.command);\n console.log(`Wrapping command: ${validatedCommand}`);\n execFileSync(scriptPath, [validatedCommand], {\n stdio: 'inherit',\n env: { ...process.env, AUTO_CAPTURE_ON_EXIT: 'true' },\n });\n } catch (validationError) {\n if (validationError instanceof z.ZodError) {\n console.error('\u274C Invalid command:');\n validationError.errors.forEach((err) => {\n console.error(` ${err.message}`);\n });\n } else {\n console.error(\n '\u274C Failed to execute command:',\n (validationError as Error).message\n );\n }\n return;\n }\n } else {\n // Interactive mode\n console.log('To enable auto-handoff for your current session:');\n console.log('');\n console.log(' For bash/zsh:');\n console.log(` source <(${scriptPath} --shell)`);\n console.log('');\n console.log(' Or wrap a command:');\n console.log(` ${scriptPath} claude`);\n console.log(` ${scriptPath} npm run dev`);\n console.log('');\n console.log(' Add to your shell profile for permanent setup:');\n console.log(\n ` echo 'alias claude=\"${scriptPath} claude\"' >> ~/.bashrc`\n );\n }\n });\n\n return cmd;\n}\n"],
|
|
4
|
+
"sourcesContent": ["/**\n * Handoff command - Commits work and generates a prompt for the next session\n */\n\nimport { Command } from 'commander';\nimport { execSync, execFileSync } from 'child_process';\nimport {\n existsSync,\n readFileSync,\n writeFileSync,\n mkdirSync,\n readdirSync,\n unlinkSync,\n} from 'fs';\nimport { join } from 'path';\nimport Database from 'better-sqlite3';\nimport { z } from 'zod';\nimport { FrameManager } from '../../core/context/index.js';\nimport { LinearTaskManager } from '../../features/tasks/linear-task-manager.js';\nimport { logger } from '../../core/monitoring/logger.js';\nimport { EnhancedHandoffGenerator } from '../../core/session/enhanced-handoff.js';\n\n// Handoff versioning - keep last N handoffs\nconst MAX_HANDOFF_VERSIONS = 10;\n\nfunction saveVersionedHandoff(\n projectRoot: string,\n branch: string,\n content: string\n): string {\n const handoffsDir = join(projectRoot, '.stackmemory', 'handoffs');\n if (!existsSync(handoffsDir)) {\n mkdirSync(handoffsDir, { recursive: true });\n }\n\n // Generate versioned filename: YYYY-MM-DD-HH-mm-branch.md\n const now = new Date();\n const timestamp = now.toISOString().slice(0, 16).replace(/[T:]/g, '-');\n const safeBranch = branch.replace(/[^a-zA-Z0-9-]/g, '-').slice(0, 30);\n const filename = `${timestamp}-${safeBranch}.md`;\n const versionedPath = join(handoffsDir, filename);\n\n // Save versioned handoff\n writeFileSync(versionedPath, content);\n\n // Clean up old handoffs (keep last N)\n try {\n const files = readdirSync(handoffsDir)\n .filter((f) => f.endsWith('.md'))\n .sort()\n .reverse();\n\n for (const oldFile of files.slice(MAX_HANDOFF_VERSIONS)) {\n unlinkSync(join(handoffsDir, oldFile));\n }\n } catch {\n // Cleanup failed, not critical\n }\n\n return versionedPath;\n}\n\n// Input validation schemas\nconst CommitMessageSchema = z\n .string()\n .min(1, 'Commit message cannot be empty')\n .max(200, 'Commit message too long')\n .regex(\n /^[a-zA-Z0-9\\s\\-_.,:()\\/\\[\\]]+$/,\n 'Commit message contains invalid characters'\n )\n .refine(\n (msg) => !msg.includes('\\n'),\n 'Commit message cannot contain newlines'\n )\n .refine(\n (msg) => !msg.includes('\"'),\n 'Commit message cannot contain double quotes'\n )\n .refine(\n (msg) => !msg.includes('`'),\n 'Commit message cannot contain backticks'\n );\n\nexport function createHandoffCommand(): Command {\n const cmd = new Command('handoff');\n\n cmd.description('Session handoff for continuity between Claude sessions');\n\n // Default action - capture handoff\n cmd\n .command('capture', { isDefault: true })\n .description('Commit current work and generate a handoff prompt')\n .option('-m, --message <message>', 'Custom commit message')\n .option('--no-commit', 'Skip git commit')\n .option('--copy', 'Copy the handoff prompt to clipboard')\n .option('--basic', 'Use basic handoff format instead of enhanced')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n // 1. Check git status\n let gitStatus = '';\n let hasChanges = false;\n\n try {\n gitStatus = execSync('git status --short', {\n encoding: 'utf-8',\n cwd: projectRoot,\n });\n hasChanges = gitStatus.trim().length > 0;\n } catch {\n console.log('\u26A0\uFE0F Not in a git repository');\n }\n\n // 2. Commit if there are changes and not skipped\n if (hasChanges && options.commit !== false) {\n try {\n // Get current branch\n const currentBranch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n // Stage all changes\n execSync('git add -A', { cwd: projectRoot });\n\n // Generate or use custom commit message\n let commitMessage =\n options.message ||\n `chore: handoff checkpoint on ${currentBranch}`;\n\n // Validate commit message\n try {\n commitMessage = CommitMessageSchema.parse(commitMessage);\n } catch (validationError) {\n console.error(\n '\u274C Invalid commit message:',\n (validationError as Error).message\n );\n return;\n }\n\n // Commit using execFileSync for safety\n execFileSync('git', ['commit', '-m', commitMessage], {\n cwd: projectRoot,\n stdio: 'inherit',\n });\n\n console.log(`\u2705 Committed changes: \"${commitMessage}\"`);\n console.log(` Branch: ${currentBranch}`);\n } catch (err: unknown) {\n console.error(\n '\u274C Failed to commit changes:',\n (err as Error).message\n );\n }\n } else if (!hasChanges) {\n console.log('\u2139\uFE0F No changes to commit');\n }\n\n // 3. Gather context for handoff prompt\n let contextSummary = '';\n let tasksSummary = '';\n let recentWork = '';\n\n if (existsSync(dbPath)) {\n const db = new Database(dbPath);\n\n // Get recent context\n const frameManager = new FrameManager(db, 'cli-project');\n const activeFrames = frameManager.getActiveFramePath();\n\n if (activeFrames.length > 0) {\n contextSummary = 'Active context frames:\\n';\n activeFrames.forEach((frame) => {\n contextSummary += ` - ${frame.name} [${frame.type}]\\n`;\n });\n }\n\n // Get task status\n const taskStore = new LinearTaskManager(projectRoot, db);\n const activeTasks = taskStore.getActiveTasks();\n\n const inProgress = activeTasks.filter(\n (t: any) => t.status === 'in_progress'\n );\n const todo = activeTasks.filter((t: any) => t.status === 'pending');\n const recentlyCompleted = activeTasks\n .filter((t: any) => t.status === 'completed' && t.completed_at)\n .sort(\n (a: any, b: any) => (b.completed_at || 0) - (a.completed_at || 0)\n )\n .slice(0, 3);\n\n if (inProgress.length > 0 || todo.length > 0) {\n tasksSummary = '\\nTasks:\\n';\n\n if (inProgress.length > 0) {\n tasksSummary += 'In Progress:\\n';\n inProgress.forEach((t: any) => {\n const externalId = t.external_refs?.linear?.id;\n tasksSummary += ` - ${t.title}${externalId ? ` [${externalId}]` : ''}\\n`;\n });\n }\n\n if (todo.length > 0) {\n tasksSummary += 'TODO:\\n';\n todo.slice(0, 5).forEach((t: any) => {\n const externalId = t.external_refs?.linear?.id;\n tasksSummary += ` - ${t.title}${externalId ? ` [${externalId}]` : ''}\\n`;\n });\n if (todo.length > 5) {\n tasksSummary += ` ... and ${todo.length - 5} more\\n`;\n }\n }\n }\n\n if (recentlyCompleted.length > 0) {\n recentWork = '\\nRecently Completed:\\n';\n recentlyCompleted.forEach((t: any) => {\n recentWork += ` \u2713 ${t.title}\\n`;\n });\n }\n\n // Get recent events\n const recentEvents = db\n .prepare(\n `\n SELECT event_type as type, payload as data, datetime(ts, 'unixepoch') as time\n FROM events\n ORDER BY ts DESC\n LIMIT 5\n `\n )\n .all() as any[];\n\n if (recentEvents.length > 0) {\n recentWork += '\\nRecent Activity:\\n';\n recentEvents.forEach((event) => {\n const data = JSON.parse(event.data);\n recentWork += ` - ${event.type}: ${data.message || data.name || 'activity'}\\n`;\n });\n }\n\n db.close();\n }\n\n // 4. Get current git info\n let gitInfo = '';\n try {\n const branch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n const lastCommit = execSync('git log -1 --oneline', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n\n gitInfo = `\\nGit Status:\\n Branch: ${branch}\\n Last commit: ${lastCommit}\\n`;\n } catch {\n // Ignore git errors\n }\n\n // 5. Check for any blockers or notes\n let notes = '';\n const notesPath = join(projectRoot, '.stackmemory', 'handoff.md');\n if (existsSync(notesPath)) {\n const handoffNotes = readFileSync(notesPath, 'utf-8');\n if (handoffNotes.trim()) {\n notes = `\\nNotes from previous handoff:\\n${handoffNotes}\\n`;\n }\n }\n\n // 6. Generate the handoff prompt\n let handoffPrompt: string;\n\n if (options.basic) {\n // Use basic handoff format\n const timestamp = new Date().toISOString();\n handoffPrompt = `# Session Handoff - ${timestamp}\n\n## Project: ${projectRoot.split('/').pop()}\n\n${gitInfo}\n${contextSummary}\n${tasksSummary}\n${recentWork}\n${notes}\n\n## Continue from here:\n\n1. Run \\`stackmemory status\\` to check the current state\n2. Review any in-progress tasks above\n3. Check for any uncommitted changes with \\`git status\\`\n4. Resume work on the active context\n\n## Quick Commands:\n- \\`stackmemory context load --recent\\` - Load recent context\n- \\`stackmemory task list --state in_progress\\` - Show in-progress tasks\n- \\`stackmemory linear sync\\` - Sync with Linear if configured\n- \\`stackmemory log recent\\` - View recent activity\n\n---\nGenerated by stackmemory handoff at ${timestamp}\n`;\n } else {\n // Use high-efficacy enhanced handoff generator (default)\n const enhancedGenerator = new EnhancedHandoffGenerator(projectRoot);\n const enhancedHandoff = await enhancedGenerator.generate();\n handoffPrompt = enhancedGenerator.toMarkdown(enhancedHandoff);\n console.log(`Estimated tokens: ~${enhancedHandoff.estimatedTokens}`);\n }\n\n // 7. Save handoff prompt (both latest and versioned)\n const handoffPath = join(\n projectRoot,\n '.stackmemory',\n 'last-handoff.md'\n );\n writeFileSync(handoffPath, handoffPrompt);\n\n // Save versioned copy\n let branch = 'unknown';\n try {\n branch = execSync('git rev-parse --abbrev-ref HEAD', {\n encoding: 'utf-8',\n cwd: projectRoot,\n }).trim();\n } catch {\n // Not a git repo\n }\n const versionedPath = saveVersionedHandoff(\n projectRoot,\n branch,\n handoffPrompt\n );\n console.log(\n `Versioned: ${versionedPath.split('/').slice(-2).join('/')}`\n );\n\n // 8. Display the prompt\n console.log('\\n' + '='.repeat(60));\n console.log(handoffPrompt);\n console.log('='.repeat(60));\n\n // 9. Copy to clipboard if requested\n if (options.copy) {\n try {\n // Use execFileSync with predefined commands for safety\n if (process.platform === 'darwin') {\n execFileSync('pbcopy', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else if (process.platform === 'win32') {\n execFileSync('clip', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else {\n execFileSync('xclip', ['-selection', 'clipboard'], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n }\n\n console.log('\\n\u2705 Handoff prompt copied to clipboard!');\n } catch {\n console.log('\\n\u26A0\uFE0F Could not copy to clipboard');\n }\n }\n\n console.log(`\\n\uD83D\uDCBE Handoff saved to: ${handoffPath}`);\n console.log('\uD83D\uDCCB Use this prompt when starting your next session');\n } catch (error: unknown) {\n logger.error('Handoff command failed', error as Error);\n console.error('\u274C Handoff failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n // Restore command\n cmd\n .command('restore')\n .description('Restore context from last handoff')\n .option('--no-copy', 'Do not copy prompt to clipboard')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const handoffPath = join(\n projectRoot,\n '.stackmemory',\n 'last-handoff.md'\n );\n const metaPath = join(\n process.env['HOME'] || '~',\n '.stackmemory',\n 'handoffs',\n 'last-handoff-meta.json'\n );\n\n if (!existsSync(handoffPath)) {\n console.log('\u274C No handoff found in this project');\n console.log('\uD83D\uDCA1 Run \"stackmemory handoff\" to create one');\n return;\n }\n\n // Read handoff prompt\n const handoffPrompt = readFileSync(handoffPath, 'utf-8');\n\n // Display the prompt\n console.log('\\n' + '='.repeat(60));\n console.log('\uD83D\uDCCB RESTORED HANDOFF');\n console.log('='.repeat(60));\n console.log(handoffPrompt);\n console.log('='.repeat(60));\n\n // Check for metadata\n if (existsSync(metaPath)) {\n const metadata = JSON.parse(readFileSync(metaPath, 'utf-8'));\n console.log('\\n\uD83D\uDCCA Session Metadata:');\n console.log(` Timestamp: ${metadata.timestamp}`);\n console.log(` Reason: ${metadata.reason}`);\n console.log(` Duration: ${metadata.session_duration}s`);\n console.log(` Command: ${metadata.command}`);\n }\n\n // Check current git status\n try {\n const gitStatus = execSync('git status --short', {\n encoding: 'utf-8',\n }).trim();\n if (gitStatus) {\n console.log('\\n\u26A0\uFE0F Current uncommitted changes:');\n console.log(gitStatus);\n }\n } catch {\n // Not a git repo\n }\n\n // Copy to clipboard unless disabled\n if (options.copy !== false) {\n try {\n // Use execFileSync with predefined commands for safety\n if (process.platform === 'darwin') {\n execFileSync('pbcopy', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else if (process.platform === 'win32') {\n execFileSync('clip', [], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n } else {\n execFileSync('xclip', ['-selection', 'clipboard'], {\n input: handoffPrompt,\n cwd: projectRoot,\n });\n }\n\n console.log('\\n\u2705 Handoff prompt copied to clipboard!');\n } catch {\n console.log('\\n\u26A0\uFE0F Could not copy to clipboard');\n }\n }\n\n console.log('\\n\uD83D\uDE80 Ready to continue where you left off!');\n } catch (error: unknown) {\n logger.error('Handoff restore failed', error as Error);\n console.error('\u274C Restore failed:', (error as Error).message);\n process.exit(1);\n }\n });\n\n // Auto command - enable auto-capture\n cmd\n .command('auto')\n .description('Enable auto-capture on session termination')\n .option('--command <command>', 'Command to wrap with auto-handoff')\n .action(async (options) => {\n const scriptPath = join(\n __dirname,\n '..',\n '..',\n '..',\n 'scripts',\n 'stackmemory-auto-handoff.sh'\n );\n\n if (!existsSync(scriptPath)) {\n console.error('\u274C Auto-handoff script not found');\n console.log('\uD83D\uDCE6 Please ensure StackMemory is properly installed');\n return;\n }\n\n console.log('\uD83D\uDEE1\uFE0F StackMemory Auto-Handoff');\n console.log('\u2500'.repeat(50));\n\n if (options.command) {\n // Validate and wrap specific command\n const commandSchema = z\n .string()\n .min(1, 'Command cannot be empty')\n .max(200, 'Command too long')\n .regex(\n /^[a-zA-Z0-9\\s\\-_./:]+$/,\n 'Command contains invalid characters'\n )\n .refine((cmd) => !cmd.includes(';'), 'Command cannot contain \";\"')\n .refine((cmd) => !cmd.includes('&'), 'Command cannot contain \"&\"')\n .refine((cmd) => !cmd.includes('|'), 'Command cannot contain \"|\"')\n .refine((cmd) => !cmd.includes('$'), 'Command cannot contain \"$\"')\n .refine((cmd) => !cmd.includes('`'), 'Command cannot contain \"`\"');\n\n try {\n const validatedCommand = commandSchema.parse(options.command);\n console.log(`Wrapping command: ${validatedCommand}`);\n execFileSync(scriptPath, [validatedCommand], {\n stdio: 'inherit',\n env: { ...process.env, AUTO_CAPTURE_ON_EXIT: 'true' },\n });\n } catch (validationError) {\n if (validationError instanceof z.ZodError) {\n console.error('\u274C Invalid command:');\n validationError.errors.forEach((err) => {\n console.error(` ${err.message}`);\n });\n } else {\n console.error(\n '\u274C Failed to execute command:',\n (validationError as Error).message\n );\n }\n return;\n }\n } else {\n // Interactive mode\n console.log('To enable auto-handoff for your current session:');\n console.log('');\n console.log(' For bash/zsh:');\n console.log(` source <(${scriptPath} --shell)`);\n console.log('');\n console.log(' Or wrap a command:');\n console.log(` ${scriptPath} claude`);\n console.log(` ${scriptPath} npm run dev`);\n console.log('');\n console.log(' Add to your shell profile for permanent setup:');\n console.log(\n ` echo 'alias claude=\"${scriptPath} claude\"' >> ~/.bashrc`\n );\n }\n });\n\n return cmd;\n}\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": ["branch", "cmd"]
|
|
7
7
|
}
|
|
@@ -6,7 +6,7 @@ import { Command } from "commander";
|
|
|
6
6
|
import chalk from "chalk";
|
|
7
7
|
import ora from "ora";
|
|
8
8
|
import { SessionMonitor } from "../../core/monitoring/session-monitor.js";
|
|
9
|
-
import { FrameManager } from "../../core/context/
|
|
9
|
+
import { FrameManager } from "../../core/context/index.js";
|
|
10
10
|
import Database from "better-sqlite3";
|
|
11
11
|
import * as fs from "fs/promises";
|
|
12
12
|
import * as path from "path";
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/monitor.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Monitor command for StackMemory\n * Real-time monitoring of context and frame activity\n */\n\n/**\n * Monitor command for StackMemory\n * Runs background monitoring daemon for automatic triggers\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport { SessionMonitor } from '../../core/monitoring/session-monitor.js';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport Database from 'better-sqlite3';\n// getProjectRoot function will be defined below\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { spawn } from 'child_process';\n\nexport function createMonitorCommand(): Command {\n const cmd = new Command('monitor')\n .description('Run background monitoring for automatic triggers')\n .option('--start', 'Start monitoring daemon')\n .option('--stop', 'Stop monitoring daemon')\n .option('--status', 'Check monitor status')\n .option('--config', 'Show monitor configuration')\n .option('--activity', 'Update activity timestamp')\n .option('--daemon', 'Run as daemon (background process)')\n .option('--foreground', 'Run in foreground (for testing)')\n .option('--interval <seconds>', 'Check interval in seconds', '30')\n .option('--idle <minutes>', 'Idle timeout in minutes', '5')\n .action(async (options) => {\n const spinner = ora();\n\n try {\n const projectRoot = await getProjectRoot();\n const dbPath = path.join(\n projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n\n // Check if StackMemory is initialized\n try {\n await fs.access(dbPath);\n } catch {\n console.error(chalk.red('\u2717 StackMemory not initialized'));\n console.log(chalk.yellow('Run: stackmemory init'));\n process.exit(1);\n }\n\n if (options.start) {\n await startMonitor(projectRoot, options, spinner);\n } else if (options.stop) {\n await stopMonitor(projectRoot, spinner);\n } else if (options.status) {\n await showStatus(projectRoot);\n } else if (options.config) {\n await showConfig(projectRoot);\n } else if (options.activity) {\n await updateActivity(projectRoot);\n } else if (options.daemon) {\n await runDaemon(projectRoot, options);\n } else if (options.foreground) {\n await runForeground(projectRoot, options);\n } else {\n // Default: show status\n await showStatus(projectRoot);\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Error: ${error}`));\n process.exit(1);\n }\n });\n\n return cmd;\n}\n\n/**\n * Start monitoring daemon\n */\nasync function startMonitor(projectRoot: string, options: any, spinner: any) {\n spinner.start('Starting monitor daemon...');\n\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n\n // Check if already running\n try {\n const pid = await fs.readFile(pidFile, 'utf-8');\n // Check if process is actually running\n try {\n process.kill(parseInt(pid), 0);\n spinner.fail(chalk.yellow('Monitor already running'));\n console.log(chalk.gray(`PID: ${pid}`));\n return;\n } catch {\n // Process not running, clean up stale PID file\n await fs.unlink(pidFile).catch(() => {});\n }\n } catch {\n // No PID file\n }\n\n // Spawn daemon process\n const daemon = spawn(\n process.execPath,\n [\n process.argv[1],\n 'monitor',\n '--daemon',\n '--interval',\n options.interval || '30',\n '--idle',\n options.idle || '5',\n ],\n {\n cwd: projectRoot,\n detached: true,\n stdio: ['ignore', 'ignore', 'ignore'],\n }\n );\n\n // Store PID\n await fs.writeFile(pidFile, daemon.pid!.toString(), 'utf-8');\n\n daemon.unref();\n\n spinner.succeed(chalk.green('\u2705 Monitor daemon started'));\n console.log(chalk.gray(`PID: ${daemon.pid}`));\n console.log(chalk.gray(`Check interval: ${options.interval || 30}s`));\n console.log(chalk.gray(`Idle timeout: ${options.idle || 5}min`));\n\n console.log(chalk.bold('\\n\uD83D\uDD0D Monitoring:'));\n console.log(' \u2022 Context usage (warns at 60%, saves at 85%)');\n console.log(' \u2022 Idle detection (handoff after 5min)');\n console.log(' \u2022 Session end (auto-save on exit)');\n\n console.log(chalk.gray('\\nStop with: stackmemory monitor --stop'));\n}\n\n/**\n * Stop monitoring daemon\n */\nasync function stopMonitor(projectRoot: string, spinner: ora.Ora) {\n spinner.start('Stopping monitor daemon...');\n\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n\n try {\n const pid = parseInt(await fs.readFile(pidFile, 'utf-8'));\n\n // Send termination signal\n process.kill(pid, 'SIGTERM');\n\n // Clean up PID file\n await fs.unlink(pidFile);\n\n spinner.succeed(chalk.green('\u2705 Monitor daemon stopped'));\n } catch (error: unknown) {\n spinner.fail(chalk.yellow('Monitor not running'));\n }\n}\n\n/**\n * Show monitor status\n */\nasync function showStatus(projectRoot: string) {\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n const statusFile = path.join(projectRoot, '.stackmemory', 'monitor.status');\n\n console.log(chalk.bold('\\n\uD83D\uDCCA Monitor Status\\n'));\n\n // Check if daemon is running\n let isRunning = false;\n let pid: number | undefined;\n\n try {\n pid = parseInt(await fs.readFile(pidFile, 'utf-8'));\n process.kill(pid, 0);\n isRunning = true;\n } catch {\n // Not running\n }\n\n if (isRunning && pid) {\n console.log(chalk.green('\u2705 Monitor is running'));\n console.log(chalk.gray(`PID: ${pid}`));\n\n // Try to read status file\n try {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n\n console.log(chalk.bold('\\nLast Check:'));\n console.log(` Time: ${new Date(status.lastCheck).toLocaleString()}`);\n console.log(` Context: ${Math.round(status.contextPercentage * 100)}%`);\n console.log(\n ` Status: ${getStatusEmoji(status.contextStatus)} ${status.contextStatus}`\n );\n\n if (status.lastActivity) {\n const idleMinutes = Math.round(\n (Date.now() - new Date(status.lastActivity).getTime()) / 60000\n );\n console.log(` Idle: ${idleMinutes} minutes`);\n }\n\n if (status.lastLedgerSave) {\n console.log(chalk.bold('\\nLast Ledger Save:'));\n console.log(` ${new Date(status.lastLedgerSave).toLocaleString()}`);\n }\n\n if (status.lastHandoff) {\n console.log(chalk.bold('\\nLast Handoff:'));\n console.log(` ${new Date(status.lastHandoff).toLocaleString()}`);\n }\n } catch {\n // No status file or invalid\n }\n } else {\n console.log(chalk.yellow('\u26A0\uFE0F Monitor is not running'));\n console.log(chalk.gray('Start with: stackmemory monitor --start'));\n }\n}\n\n/**\n * Show monitor configuration\n */\nasync function showConfig(projectRoot: string) {\n const configFile = path.join(projectRoot, '.stackmemory', 'config.json');\n\n console.log(chalk.bold('\\n\u2699\uFE0F Monitor Configuration\\n'));\n\n try {\n const config = JSON.parse(await fs.readFile(configFile, 'utf-8'));\n const monitorConfig = config.monitor || {};\n\n console.log('Context Thresholds:');\n console.log(\n ` Warning: ${(monitorConfig.contextWarningThreshold || 0.6) * 100}%`\n );\n console.log(\n ` Critical: ${(monitorConfig.contextCriticalThreshold || 0.7) * 100}%`\n );\n console.log(\n ` Auto-save: ${(monitorConfig.contextAutoSaveThreshold || 0.85) * 100}%`\n );\n\n console.log('\\nTimings:');\n console.log(\n ` Check interval: ${monitorConfig.checkIntervalSeconds || 30}s`\n );\n console.log(` Idle timeout: ${monitorConfig.idleTimeoutMinutes || 5}min`);\n\n console.log('\\nAuto Actions:');\n console.log(\n ` Auto-save ledger: ${monitorConfig.autoSaveLedger !== false ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Auto-generate handoff: ${monitorConfig.autoGenerateHandoff !== false ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Session-end handoff: ${monitorConfig.sessionEndHandoff !== false ? '\u2705' : '\u274C'}`\n );\n } catch {\n console.log(chalk.gray('Using default configuration'));\n console.log('\\nDefaults:');\n console.log(' Warning at 60%, Critical at 70%, Auto-save at 85%');\n console.log(' Check every 30s, Idle timeout 5min');\n console.log(' All auto-actions enabled');\n }\n}\n\n/**\n * Update activity timestamp\n */\nasync function updateActivity(projectRoot: string) {\n const activityFile = path.join(\n projectRoot,\n '.stackmemory',\n 'monitor.activity'\n );\n await fs.mkdir(path.dirname(activityFile), { recursive: true });\n await fs.writeFile(activityFile, new Date().toISOString(), 'utf-8');\n // Silent update - no output\n}\n\n/**\n * Run as daemon (background process)\n */\nasync function runDaemon(projectRoot: string, options: any) {\n const dbPath = path.join(projectRoot, '.stackmemory', 'db', 'stackmemory.db');\n const db = new Database(dbPath);\n\n const frameManager = new FrameManager(db, 'current');\n\n const monitor = new SessionMonitor(frameManager, db, projectRoot, {\n checkIntervalSeconds: parseInt(options.interval) || 30,\n idleTimeoutMinutes: parseInt(options.idle) || 5,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n sessionEndHandoff: true,\n });\n\n // Write status periodically\n const statusFile = path.join(projectRoot, '.stackmemory', 'monitor.status');\n const activityFile = path.join(\n projectRoot,\n '.stackmemory',\n 'monitor.activity'\n );\n\n monitor.on('context:usage', async (data) => {\n // Check for activity file updates\n try {\n const activityTime = await fs.readFile(activityFile, 'utf-8');\n monitor.updateActivity();\n } catch {\n // No activity file\n }\n\n // Write status\n const status = {\n lastCheck: new Date().toISOString(),\n contextPercentage: data.percentage,\n contextStatus: data.status,\n lastActivity: monitor.getStatus().lastActivity,\n };\n\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n monitor.on('context:ledger_saved', async () => {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n status.lastLedgerSave = new Date().toISOString();\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n monitor.on('handoff:generated', async () => {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n status.lastHandoff = new Date().toISOString();\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n // Start monitoring\n await monitor.start();\n\n // Keep process alive\n process.on('SIGTERM', async () => {\n await monitor.stop();\n process.exit(0);\n });\n}\n\n/**\n * Run in foreground (for testing)\n */\nasync function runForeground(projectRoot: string, options: any) {\n console.log(chalk.bold('\uD83D\uDD0D Running monitor in foreground...\\n'));\n\n const dbPath = path.join(projectRoot, '.stackmemory', 'db', 'stackmemory.db');\n const db = new Database(dbPath);\n\n const frameManager = new FrameManager(db, 'current');\n\n const monitor = new SessionMonitor(frameManager, db, projectRoot, {\n checkIntervalSeconds: parseInt(options.interval) || 30,\n idleTimeoutMinutes: parseInt(options.idle) || 5,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n sessionEndHandoff: true,\n });\n\n // Log all events\n monitor.on('context:usage', (data) => {\n console.log(\n `[${new Date().toLocaleTimeString()}] Context: ${Math.round(data.percentage * 100)}% (${data.status})`\n );\n });\n\n monitor.on('context:warning', () => {\n console.log(chalk.yellow('\u26A0\uFE0F Context warning threshold reached'));\n });\n\n monitor.on('context:high', () => {\n console.log(chalk.yellow('\uD83D\uDFE1 Context high - considering auto-save'));\n });\n\n monitor.on('context:ledger_saved', (data) => {\n console.log(\n chalk.green(`\u2705 Ledger saved (${data.compression}x compression)`)\n );\n });\n\n monitor.on('handoff:generated', (data) => {\n console.log(chalk.green(`\uD83D\uDCCB Handoff generated (${data.trigger})`));\n });\n\n // Start monitoring\n await monitor.start();\n\n console.log('Press Ctrl+C to stop\\n');\n\n // Handle exit\n process.on('SIGINT', async () => {\n console.log('\\nStopping monitor...');\n await monitor.stop();\n process.exit(0);\n });\n}\n\n// Helper functions\n\nasync function getProjectRoot(): Promise<string> {\n return process.cwd();\n}\n\nfunction getStatusEmoji(status: string): string {\n switch (status) {\n case 'ok':\n return '\uD83D\uDFE2';\n case 'warning':\n return '\uD83D\uDFE1';\n case 'high':\n return '\uD83D\uDFE0';\n case 'critical':\n return '\uD83D\uDD34';\n default:\n return '\u26AB';\n }\n}\n\n// Export for use in main CLI\nexport default createMonitorCommand();\n"],
|
|
4
|
+
"sourcesContent": ["/**\n * Monitor command for StackMemory\n * Real-time monitoring of context and frame activity\n */\n\n/**\n * Monitor command for StackMemory\n * Runs background monitoring daemon for automatic triggers\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport { SessionMonitor } from '../../core/monitoring/session-monitor.js';\nimport { FrameManager } from '../../core/context/index.js';\nimport Database from 'better-sqlite3';\n// getProjectRoot function will be defined below\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { spawn } from 'child_process';\n\nexport function createMonitorCommand(): Command {\n const cmd = new Command('monitor')\n .description('Run background monitoring for automatic triggers')\n .option('--start', 'Start monitoring daemon')\n .option('--stop', 'Stop monitoring daemon')\n .option('--status', 'Check monitor status')\n .option('--config', 'Show monitor configuration')\n .option('--activity', 'Update activity timestamp')\n .option('--daemon', 'Run as daemon (background process)')\n .option('--foreground', 'Run in foreground (for testing)')\n .option('--interval <seconds>', 'Check interval in seconds', '30')\n .option('--idle <minutes>', 'Idle timeout in minutes', '5')\n .action(async (options) => {\n const spinner = ora();\n\n try {\n const projectRoot = await getProjectRoot();\n const dbPath = path.join(\n projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n\n // Check if StackMemory is initialized\n try {\n await fs.access(dbPath);\n } catch {\n console.error(chalk.red('\u2717 StackMemory not initialized'));\n console.log(chalk.yellow('Run: stackmemory init'));\n process.exit(1);\n }\n\n if (options.start) {\n await startMonitor(projectRoot, options, spinner);\n } else if (options.stop) {\n await stopMonitor(projectRoot, spinner);\n } else if (options.status) {\n await showStatus(projectRoot);\n } else if (options.config) {\n await showConfig(projectRoot);\n } else if (options.activity) {\n await updateActivity(projectRoot);\n } else if (options.daemon) {\n await runDaemon(projectRoot, options);\n } else if (options.foreground) {\n await runForeground(projectRoot, options);\n } else {\n // Default: show status\n await showStatus(projectRoot);\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Error: ${error}`));\n process.exit(1);\n }\n });\n\n return cmd;\n}\n\n/**\n * Start monitoring daemon\n */\nasync function startMonitor(projectRoot: string, options: any, spinner: any) {\n spinner.start('Starting monitor daemon...');\n\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n\n // Check if already running\n try {\n const pid = await fs.readFile(pidFile, 'utf-8');\n // Check if process is actually running\n try {\n process.kill(parseInt(pid), 0);\n spinner.fail(chalk.yellow('Monitor already running'));\n console.log(chalk.gray(`PID: ${pid}`));\n return;\n } catch {\n // Process not running, clean up stale PID file\n await fs.unlink(pidFile).catch(() => {});\n }\n } catch {\n // No PID file\n }\n\n // Spawn daemon process\n const daemon = spawn(\n process.execPath,\n [\n process.argv[1],\n 'monitor',\n '--daemon',\n '--interval',\n options.interval || '30',\n '--idle',\n options.idle || '5',\n ],\n {\n cwd: projectRoot,\n detached: true,\n stdio: ['ignore', 'ignore', 'ignore'],\n }\n );\n\n // Store PID\n await fs.writeFile(pidFile, daemon.pid!.toString(), 'utf-8');\n\n daemon.unref();\n\n spinner.succeed(chalk.green('\u2705 Monitor daemon started'));\n console.log(chalk.gray(`PID: ${daemon.pid}`));\n console.log(chalk.gray(`Check interval: ${options.interval || 30}s`));\n console.log(chalk.gray(`Idle timeout: ${options.idle || 5}min`));\n\n console.log(chalk.bold('\\n\uD83D\uDD0D Monitoring:'));\n console.log(' \u2022 Context usage (warns at 60%, saves at 85%)');\n console.log(' \u2022 Idle detection (handoff after 5min)');\n console.log(' \u2022 Session end (auto-save on exit)');\n\n console.log(chalk.gray('\\nStop with: stackmemory monitor --stop'));\n}\n\n/**\n * Stop monitoring daemon\n */\nasync function stopMonitor(projectRoot: string, spinner: ora.Ora) {\n spinner.start('Stopping monitor daemon...');\n\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n\n try {\n const pid = parseInt(await fs.readFile(pidFile, 'utf-8'));\n\n // Send termination signal\n process.kill(pid, 'SIGTERM');\n\n // Clean up PID file\n await fs.unlink(pidFile);\n\n spinner.succeed(chalk.green('\u2705 Monitor daemon stopped'));\n } catch (error: unknown) {\n spinner.fail(chalk.yellow('Monitor not running'));\n }\n}\n\n/**\n * Show monitor status\n */\nasync function showStatus(projectRoot: string) {\n const pidFile = path.join(projectRoot, '.stackmemory', 'monitor.pid');\n const statusFile = path.join(projectRoot, '.stackmemory', 'monitor.status');\n\n console.log(chalk.bold('\\n\uD83D\uDCCA Monitor Status\\n'));\n\n // Check if daemon is running\n let isRunning = false;\n let pid: number | undefined;\n\n try {\n pid = parseInt(await fs.readFile(pidFile, 'utf-8'));\n process.kill(pid, 0);\n isRunning = true;\n } catch {\n // Not running\n }\n\n if (isRunning && pid) {\n console.log(chalk.green('\u2705 Monitor is running'));\n console.log(chalk.gray(`PID: ${pid}`));\n\n // Try to read status file\n try {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n\n console.log(chalk.bold('\\nLast Check:'));\n console.log(` Time: ${new Date(status.lastCheck).toLocaleString()}`);\n console.log(` Context: ${Math.round(status.contextPercentage * 100)}%`);\n console.log(\n ` Status: ${getStatusEmoji(status.contextStatus)} ${status.contextStatus}`\n );\n\n if (status.lastActivity) {\n const idleMinutes = Math.round(\n (Date.now() - new Date(status.lastActivity).getTime()) / 60000\n );\n console.log(` Idle: ${idleMinutes} minutes`);\n }\n\n if (status.lastLedgerSave) {\n console.log(chalk.bold('\\nLast Ledger Save:'));\n console.log(` ${new Date(status.lastLedgerSave).toLocaleString()}`);\n }\n\n if (status.lastHandoff) {\n console.log(chalk.bold('\\nLast Handoff:'));\n console.log(` ${new Date(status.lastHandoff).toLocaleString()}`);\n }\n } catch {\n // No status file or invalid\n }\n } else {\n console.log(chalk.yellow('\u26A0\uFE0F Monitor is not running'));\n console.log(chalk.gray('Start with: stackmemory monitor --start'));\n }\n}\n\n/**\n * Show monitor configuration\n */\nasync function showConfig(projectRoot: string) {\n const configFile = path.join(projectRoot, '.stackmemory', 'config.json');\n\n console.log(chalk.bold('\\n\u2699\uFE0F Monitor Configuration\\n'));\n\n try {\n const config = JSON.parse(await fs.readFile(configFile, 'utf-8'));\n const monitorConfig = config.monitor || {};\n\n console.log('Context Thresholds:');\n console.log(\n ` Warning: ${(monitorConfig.contextWarningThreshold || 0.6) * 100}%`\n );\n console.log(\n ` Critical: ${(monitorConfig.contextCriticalThreshold || 0.7) * 100}%`\n );\n console.log(\n ` Auto-save: ${(monitorConfig.contextAutoSaveThreshold || 0.85) * 100}%`\n );\n\n console.log('\\nTimings:');\n console.log(\n ` Check interval: ${monitorConfig.checkIntervalSeconds || 30}s`\n );\n console.log(` Idle timeout: ${monitorConfig.idleTimeoutMinutes || 5}min`);\n\n console.log('\\nAuto Actions:');\n console.log(\n ` Auto-save ledger: ${monitorConfig.autoSaveLedger !== false ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Auto-generate handoff: ${monitorConfig.autoGenerateHandoff !== false ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Session-end handoff: ${monitorConfig.sessionEndHandoff !== false ? '\u2705' : '\u274C'}`\n );\n } catch {\n console.log(chalk.gray('Using default configuration'));\n console.log('\\nDefaults:');\n console.log(' Warning at 60%, Critical at 70%, Auto-save at 85%');\n console.log(' Check every 30s, Idle timeout 5min');\n console.log(' All auto-actions enabled');\n }\n}\n\n/**\n * Update activity timestamp\n */\nasync function updateActivity(projectRoot: string) {\n const activityFile = path.join(\n projectRoot,\n '.stackmemory',\n 'monitor.activity'\n );\n await fs.mkdir(path.dirname(activityFile), { recursive: true });\n await fs.writeFile(activityFile, new Date().toISOString(), 'utf-8');\n // Silent update - no output\n}\n\n/**\n * Run as daemon (background process)\n */\nasync function runDaemon(projectRoot: string, options: any) {\n const dbPath = path.join(projectRoot, '.stackmemory', 'db', 'stackmemory.db');\n const db = new Database(dbPath);\n\n const frameManager = new FrameManager(db, 'current');\n\n const monitor = new SessionMonitor(frameManager, db, projectRoot, {\n checkIntervalSeconds: parseInt(options.interval) || 30,\n idleTimeoutMinutes: parseInt(options.idle) || 5,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n sessionEndHandoff: true,\n });\n\n // Write status periodically\n const statusFile = path.join(projectRoot, '.stackmemory', 'monitor.status');\n const activityFile = path.join(\n projectRoot,\n '.stackmemory',\n 'monitor.activity'\n );\n\n monitor.on('context:usage', async (data) => {\n // Check for activity file updates\n try {\n const activityTime = await fs.readFile(activityFile, 'utf-8');\n monitor.updateActivity();\n } catch {\n // No activity file\n }\n\n // Write status\n const status = {\n lastCheck: new Date().toISOString(),\n contextPercentage: data.percentage,\n contextStatus: data.status,\n lastActivity: monitor.getStatus().lastActivity,\n };\n\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n monitor.on('context:ledger_saved', async () => {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n status.lastLedgerSave = new Date().toISOString();\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n monitor.on('handoff:generated', async () => {\n const status = JSON.parse(await fs.readFile(statusFile, 'utf-8'));\n status.lastHandoff = new Date().toISOString();\n await fs.writeFile(statusFile, JSON.stringify(status, null, 2), 'utf-8');\n });\n\n // Start monitoring\n await monitor.start();\n\n // Keep process alive\n process.on('SIGTERM', async () => {\n await monitor.stop();\n process.exit(0);\n });\n}\n\n/**\n * Run in foreground (for testing)\n */\nasync function runForeground(projectRoot: string, options: any) {\n console.log(chalk.bold('\uD83D\uDD0D Running monitor in foreground...\\n'));\n\n const dbPath = path.join(projectRoot, '.stackmemory', 'db', 'stackmemory.db');\n const db = new Database(dbPath);\n\n const frameManager = new FrameManager(db, 'current');\n\n const monitor = new SessionMonitor(frameManager, db, projectRoot, {\n checkIntervalSeconds: parseInt(options.interval) || 30,\n idleTimeoutMinutes: parseInt(options.idle) || 5,\n autoSaveLedger: true,\n autoGenerateHandoff: true,\n sessionEndHandoff: true,\n });\n\n // Log all events\n monitor.on('context:usage', (data) => {\n console.log(\n `[${new Date().toLocaleTimeString()}] Context: ${Math.round(data.percentage * 100)}% (${data.status})`\n );\n });\n\n monitor.on('context:warning', () => {\n console.log(chalk.yellow('\u26A0\uFE0F Context warning threshold reached'));\n });\n\n monitor.on('context:high', () => {\n console.log(chalk.yellow('\uD83D\uDFE1 Context high - considering auto-save'));\n });\n\n monitor.on('context:ledger_saved', (data) => {\n console.log(\n chalk.green(`\u2705 Ledger saved (${data.compression}x compression)`)\n );\n });\n\n monitor.on('handoff:generated', (data) => {\n console.log(chalk.green(`\uD83D\uDCCB Handoff generated (${data.trigger})`));\n });\n\n // Start monitoring\n await monitor.start();\n\n console.log('Press Ctrl+C to stop\\n');\n\n // Handle exit\n process.on('SIGINT', async () => {\n console.log('\\nStopping monitor...');\n await monitor.stop();\n process.exit(0);\n });\n}\n\n// Helper functions\n\nasync function getProjectRoot(): Promise<string> {\n return process.cwd();\n}\n\nfunction getStatusEmoji(status: string): string {\n switch (status) {\n case 'ok':\n return '\uD83D\uDFE2';\n case 'warning':\n return '\uD83D\uDFE1';\n case 'high':\n return '\uD83D\uDFE0';\n case 'critical':\n return '\uD83D\uDD34';\n default:\n return '\u26AB';\n }\n}\n\n// Export for use in main CLI\nexport default createMonitorCommand();\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -10,7 +10,7 @@ import inquirer from "inquirer";
|
|
|
10
10
|
import {
|
|
11
11
|
PostTaskHooks
|
|
12
12
|
} from "../../integrations/claude-code/post-task-hooks.js";
|
|
13
|
-
import { FrameManager } from "../../core/context/
|
|
13
|
+
import { FrameManager } from "../../core/context/index.js";
|
|
14
14
|
import Database from "better-sqlite3";
|
|
15
15
|
import * as fs from "fs/promises";
|
|
16
16
|
import * as path from "path";
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/quality.ts"],
|
|
4
|
-
"sourcesContent": ["#!/usr/bin/env node\n\n/**\n * Quality command for StackMemory\n * Manages post-task quality gates and code review automation\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport inquirer from 'inquirer';\nimport {\n PostTaskHooks,\n PostTaskConfig,\n QualityGateResult,\n} from '../../integrations/claude-code/post-task-hooks.js';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport Database from 'better-sqlite3';\n// getProjectRoot function will be defined below\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport function createQualityCommand(): Command {\n const cmd = new Command('quality')\n .description('Manage post-task quality gates and automation')\n .option('--enable', 'Enable quality gates')\n .option('--disable', 'Disable quality gates')\n .option('--status', 'Show quality gate status')\n .option('--config', 'Configure quality gates')\n .option('--run', 'Run quality gates manually')\n .option('--history', 'Show quality gate history')\n .option('--setup', 'Interactive setup wizard')\n .action(async (options) => {\n const spinner = ora();\n\n try {\n const projectRoot = await getProjectRoot();\n const dbPath = path.join(\n projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n\n // Check if StackMemory is initialized\n try {\n await fs.access(dbPath);\n } catch {\n console.error(chalk.red('\u2717 StackMemory not initialized'));\n console.log(chalk.yellow('Run: stackmemory init'));\n process.exit(1);\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db);\n\n if (options.enable) {\n await enableQualityGates(projectRoot, frameManager, db, spinner);\n } else if (options.disable) {\n await disableQualityGates(projectRoot, spinner);\n } else if (options.status) {\n await showStatus(projectRoot, frameManager, db);\n } else if (options.config) {\n await configureQualityGates(projectRoot);\n } else if (options.run) {\n await runQualityGates(projectRoot, frameManager, db, spinner);\n } else if (options.history) {\n await showHistory(frameManager);\n } else if (options.setup) {\n await setupWizard(projectRoot, frameManager, db);\n } else {\n // Default: show status\n await showStatus(projectRoot, frameManager, db);\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Error: ${error}`));\n process.exit(1);\n }\n });\n\n return cmd;\n}\n\n/**\n * Enable quality gates\n */\nasync function enableQualityGates(\n projectRoot: string,\n frameManager: FrameManager,\n db: any,\n spinner: any\n) {\n spinner.start('Enabling quality gates...');\n\n try {\n // Load or create config\n const config = await loadConfig(projectRoot);\n config.qualityGates = {\n ...config.qualityGates,\n runTests: true,\n runCodeReview: true,\n runLinter: true,\n };\n\n // Initialize hooks\n const hooks = new PostTaskHooks(frameManager, db, config);\n await hooks.initialize();\n\n // Save config\n await saveConfig(projectRoot, config);\n\n // Create systemd-style service file for persistence\n await createServiceFile(projectRoot);\n\n spinner.succeed(chalk.green('\u2705 Quality gates enabled'));\n\n console.log(chalk.bold('\\n\uD83D\uDD0D Quality Gates Active:'));\n console.log(' \u2022 Auto-run tests after code changes');\n console.log(' \u2022 Auto-run linter on file saves');\n console.log(' \u2022 Auto-trigger code review on task completion');\n console.log(' \u2022 Real-time file change monitoring');\n\n console.log(chalk.gray('\\nDisable with: stackmemory quality --disable'));\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Failed to enable quality gates: ${error}`));\n }\n}\n\n/**\n * Disable quality gates\n */\nasync function disableQualityGates(projectRoot: string, spinner: ora.Ora) {\n spinner.start('Disabling quality gates...');\n\n try {\n const config = await loadConfig(projectRoot);\n config.qualityGates = {\n ...config.qualityGates,\n runTests: false,\n runCodeReview: false,\n runLinter: false,\n };\n\n await saveConfig(projectRoot, config);\n\n // Remove service file\n const serviceFile = path.join(\n projectRoot,\n '.stackmemory',\n 'quality.service'\n );\n try {\n await fs.unlink(serviceFile);\n } catch {\n // Service file doesn't exist\n }\n\n spinner.succeed(chalk.green('\u2705 Quality gates disabled'));\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Failed to disable quality gates: ${error}`));\n }\n}\n\n/**\n * Show quality gate status\n */\nasync function showStatus(\n projectRoot: string,\n frameManager: FrameManager,\n db: Database\n) {\n console.log(chalk.bold('\\n\uD83D\uDCCA Quality Gates Status\\n'));\n\n try {\n const config = await loadConfig(projectRoot);\n\n // Overall status\n const isEnabled =\n config.qualityGates?.runTests ||\n config.qualityGates?.runCodeReview ||\n config.qualityGates?.runLinter;\n console.log(\n `Status: ${isEnabled ? chalk.green('\u2705 Enabled') : chalk.yellow('\u26A0\uFE0F Disabled')}`\n );\n\n // Individual gates\n console.log('\\nGates:');\n console.log(\n ` Tests: ${config.qualityGates?.runTests ? '\u2705' : '\u274C'} ${getTestCommand(config)}`\n );\n console.log(\n ` Linter: ${config.qualityGates?.runLinter ? '\u2705' : '\u274C'} ${getLintCommand(config)}`\n );\n console.log(\n ` Code Review: ${config.qualityGates?.runCodeReview ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Coverage: ${config.qualityGates?.requireTestCoverage ? '\u2705' : '\u274C'}`\n );\n\n // Detected frameworks\n if (config.testFrameworks?.detected.length) {\n console.log('\\nDetected Frameworks:');\n config.testFrameworks.detected.forEach((fw) => {\n console.log(` \u2022 ${fw}`);\n });\n }\n\n // Recent quality results\n const recentResults = await getRecentQualityResults(frameManager);\n if (recentResults.length > 0) {\n console.log('\\nRecent Results:');\n recentResults.slice(0, 3).forEach((result) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n console.log(\n ` ${icon} ${result.frameName} (${new Date(result.timestamp).toLocaleString()})`\n );\n });\n }\n } catch (error: unknown) {\n console.error(chalk.red('Failed to get status:', error));\n }\n}\n\n/**\n * Configure quality gates\n */\nasync function configureQualityGates(projectRoot: string) {\n console.log(chalk.bold('\\n\u2699\uFE0F Quality Gates Configuration\\n'));\n\n try {\n const config = await loadConfig(projectRoot);\n\n const answers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'runTests',\n message: 'Auto-run tests after code changes?',\n default: config.qualityGates?.runTests ?? true,\n },\n {\n type: 'confirm',\n name: 'runLinter',\n message: 'Auto-run linter on file changes?',\n default: config.qualityGates?.runLinter ?? true,\n },\n {\n type: 'confirm',\n name: 'runCodeReview',\n message: 'Auto-trigger code review on task completion?',\n default: config.qualityGates?.runCodeReview ?? true,\n },\n {\n type: 'confirm',\n name: 'requireTestCoverage',\n message: 'Require test coverage checks?',\n default: config.qualityGates?.requireTestCoverage ?? false,\n },\n {\n type: 'confirm',\n name: 'blockOnFailure',\n message: 'Block further work when quality gates fail?',\n default: config.qualityGates?.blockOnFailure ?? false,\n },\n {\n type: 'input',\n name: 'testCommand',\n message: 'Custom test command (leave empty for auto-detect):',\n default: config.testFrameworks?.testCommand || '',\n },\n {\n type: 'input',\n name: 'lintCommand',\n message: 'Custom lint command (leave empty for auto-detect):',\n default: config.testFrameworks?.lintCommand || '',\n },\n ]);\n\n // Update config\n config.qualityGates = {\n runTests: answers.runTests,\n runLinter: answers.runLinter,\n runCodeReview: answers.runCodeReview,\n requireTestCoverage: answers.requireTestCoverage,\n blockOnFailure: answers.blockOnFailure,\n };\n\n config.testFrameworks = {\n ...config.testFrameworks,\n testCommand: answers.testCommand || config.testFrameworks?.testCommand,\n lintCommand: answers.lintCommand || config.testFrameworks?.lintCommand,\n };\n\n await saveConfig(projectRoot, config);\n\n console.log(chalk.green('\\n\u2705 Configuration saved'));\n console.log(chalk.gray('Enable with: stackmemory quality --enable'));\n } catch (error: unknown) {\n console.error(chalk.red('Configuration failed:', error));\n }\n}\n\n/**\n * Run quality gates manually\n */\nasync function runQualityGates(\n projectRoot: string,\n frameManager: FrameManager,\n db: any,\n spinner: any\n) {\n spinner.start('Running quality gates...');\n\n try {\n const config = await loadConfig(projectRoot);\n const hooks = new PostTaskHooks(frameManager, db, config);\n\n // Simulate a task completion event\n const mockEvent = {\n taskType: 'task_complete' as const,\n frameId: 'manual-run',\n frameName: 'Manual quality gate run',\n files: await getRecentlyModifiedFiles(projectRoot),\n changes: { added: 0, removed: 0, modified: 1 },\n metadata: { trigger: 'manual' },\n };\n\n // Run quality gates\n const results = await (hooks as any).runQualityGates(mockEvent);\n\n spinner.succeed(chalk.green('\u2705 Quality gates completed'));\n\n // Show results\n console.log(chalk.bold('\\n\uD83D\uDCCB Quality Gate Results:\\n'));\n\n results.forEach((result: QualityGateResult) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n const duration =\n result.duration < 1000\n ? `${result.duration}ms`\n : `${Math.round(result.duration / 1000)}s`;\n\n console.log(`${icon} ${result.gate} (${duration})`);\n\n if (!result.passed && result.issues) {\n result.issues.slice(0, 3).forEach((issue) => {\n console.log(\n ` ${issue.severity === 'error' ? '\u274C' : '\u26A0\uFE0F'} ${issue.message}`\n );\n });\n if (result.issues.length > 3) {\n console.log(\n chalk.gray(` ... and ${result.issues.length - 3} more issues`)\n );\n }\n }\n });\n\n const allPassed = results.every((r: QualityGateResult) => r.passed);\n if (allPassed) {\n console.log(chalk.green('\\n\uD83C\uDF89 All quality gates passed!'));\n } else {\n console.log(\n chalk.yellow('\\n\u26A0\uFE0F Some quality gates failed. See details above.')\n );\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Quality gates failed: ${error}`));\n }\n}\n\n/**\n * Show quality gate history\n */\nasync function showHistory(frameManager: FrameManager) {\n console.log(chalk.bold('\\n\uD83D\uDCC8 Quality Gate History\\n'));\n\n try {\n const results = await getRecentQualityResults(frameManager);\n\n if (results.length === 0) {\n console.log(chalk.gray('No quality gate history found'));\n return;\n }\n\n results.slice(0, 10).forEach((result, index) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n const date = new Date(result.timestamp).toLocaleDateString();\n const time = new Date(result.timestamp).toLocaleTimeString();\n\n console.log(`${icon} ${result.frameName}`);\n console.log(chalk.gray(` ${date} ${time} - ${result.duration}ms`));\n\n if (result.gates) {\n result.gates.forEach((gate: any) => {\n const gateIcon = gate.passed ? ' \u2713' : ' \u2717';\n console.log(chalk.gray(`${gateIcon} ${gate.gate}`));\n });\n }\n\n if (index < results.length - 1) console.log('');\n });\n } catch (error: unknown) {\n console.error(chalk.red('Failed to get history:', error));\n }\n}\n\n/**\n * Setup wizard\n */\nasync function setupWizard(\n projectRoot: string,\n frameManager: FrameManager,\n db: Database\n) {\n console.log(chalk.bold('\uD83E\uDDD9 Quality Gates Setup Wizard\\n'));\n\n console.log(\n 'This wizard will help you configure automatic quality gates for your project.'\n );\n console.log(\n 'Quality gates run automatically after Claude completes tasks.\\n'\n );\n\n // Detect project type\n const packageJsonPath = path.join(projectRoot, 'package.json');\n let projectType = 'unknown';\n const detectedFrameworks: string[] = [];\n\n try {\n const packageJson = JSON.parse(await fs.readFile(packageJsonPath, 'utf-8'));\n const deps = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n };\n\n if (deps.react) projectType = 'React';\n else if (deps.vue) projectType = 'Vue';\n else if (deps.angular) projectType = 'Angular';\n else if (deps.express) projectType = 'Node.js API';\n else if (deps.nextjs) projectType = 'Next.js';\n\n if (deps.jest) detectedFrameworks.push('Jest');\n if (deps.vitest) detectedFrameworks.push('Vitest');\n if (deps.playwright) detectedFrameworks.push('Playwright');\n if (deps.eslint) detectedFrameworks.push('ESLint');\n } catch {\n // No package.json or invalid\n }\n\n console.log(`\uD83D\uDCE6 Detected project: ${chalk.cyan(projectType)}`);\n if (detectedFrameworks.length) {\n console.log(\n `\uD83D\uDD27 Detected tools: ${chalk.cyan(detectedFrameworks.join(', '))}`\n );\n }\n console.log('');\n\n const { proceed } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'proceed',\n message: 'Continue with setup?',\n default: true,\n },\n ]);\n\n if (!proceed) {\n console.log(chalk.gray('Setup cancelled'));\n return;\n }\n\n // Configure quality gates\n await configureQualityGates(projectRoot);\n\n // Enable quality gates\n const { enable } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enable',\n message: 'Enable quality gates now?',\n default: true,\n },\n ]);\n\n if (enable) {\n const spinner = ora();\n await enableQualityGates(projectRoot, frameManager, db, spinner);\n }\n\n console.log(chalk.bold('\\n\uD83C\uDF89 Setup Complete!\\n'));\n console.log(\n 'Quality gates will now run automatically after Claude completes tasks.'\n );\n console.log('Check status with: stackmemory quality --status');\n}\n\n// Helper functions\n\nasync function getProjectRoot(): Promise<string> {\n return process.cwd();\n}\n\nasync function loadConfig(\n projectRoot: string\n): Promise<Partial<PostTaskConfig>> {\n const configPath = path.join(projectRoot, '.stackmemory', 'config.json');\n\n try {\n const content = await fs.readFile(configPath, 'utf-8');\n return JSON.parse(content);\n } catch {\n return {};\n }\n}\n\nasync function saveConfig(\n projectRoot: string,\n config: Partial<PostTaskConfig>\n): Promise<void> {\n const configPath = path.join(projectRoot, '.stackmemory', 'config.json');\n await fs.mkdir(path.dirname(configPath), { recursive: true });\n await fs.writeFile(configPath, JSON.stringify(config, null, 2), 'utf-8');\n}\n\nfunction getTestCommand(config: Partial<PostTaskConfig>): string {\n if (config.testFrameworks?.testCommand) {\n return chalk.gray(`(${config.testFrameworks.testCommand})`);\n }\n return chalk.gray('(auto-detect)');\n}\n\nfunction getLintCommand(config: Partial<PostTaskConfig>): string {\n if (config.testFrameworks?.lintCommand) {\n return chalk.gray(`(${config.testFrameworks.lintCommand})`);\n }\n return chalk.gray('(auto-detect)');\n}\n\nasync function getRecentQualityResults(\n frameManager: FrameManager\n): Promise<any[]> {\n // This would query the frame metadata for quality gate results\n // For now, return empty array\n return [];\n}\n\nasync function getRecentlyModifiedFiles(\n projectRoot: string\n): Promise<string[]> {\n try {\n const { execFileSync } = await import('child_process');\n const output = execFileSync('git', ['diff', '--name-only', 'HEAD~1'], {\n cwd: projectRoot,\n encoding: 'utf-8',\n });\n return output\n .trim()\n .split('\\n')\n .filter((f) => f.length > 0);\n } catch {\n return [];\n }\n}\n\nasync function createServiceFile(projectRoot: string): Promise<void> {\n const serviceContent = `# StackMemory Quality Gates Service\n# This file indicates quality gates are enabled\n# Created: ${new Date().toISOString()}\n`;\n\n const servicePath = path.join(projectRoot, '.stackmemory', 'quality.service');\n await fs.writeFile(servicePath, serviceContent, 'utf-8');\n}\n\n// Export for use in main CLI\nexport default createQualityCommand();\n"],
|
|
4
|
+
"sourcesContent": ["#!/usr/bin/env node\n\n/**\n * Quality command for StackMemory\n * Manages post-task quality gates and code review automation\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport inquirer from 'inquirer';\nimport {\n PostTaskHooks,\n PostTaskConfig,\n QualityGateResult,\n} from '../../integrations/claude-code/post-task-hooks.js';\nimport { FrameManager } from '../../core/context/index.js';\nimport Database from 'better-sqlite3';\n// getProjectRoot function will be defined below\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport function createQualityCommand(): Command {\n const cmd = new Command('quality')\n .description('Manage post-task quality gates and automation')\n .option('--enable', 'Enable quality gates')\n .option('--disable', 'Disable quality gates')\n .option('--status', 'Show quality gate status')\n .option('--config', 'Configure quality gates')\n .option('--run', 'Run quality gates manually')\n .option('--history', 'Show quality gate history')\n .option('--setup', 'Interactive setup wizard')\n .action(async (options) => {\n const spinner = ora();\n\n try {\n const projectRoot = await getProjectRoot();\n const dbPath = path.join(\n projectRoot,\n '.stackmemory',\n 'db',\n 'stackmemory.db'\n );\n\n // Check if StackMemory is initialized\n try {\n await fs.access(dbPath);\n } catch {\n console.error(chalk.red('\u2717 StackMemory not initialized'));\n console.log(chalk.yellow('Run: stackmemory init'));\n process.exit(1);\n }\n\n const db = new Database(dbPath);\n const frameManager = new FrameManager(db);\n\n if (options.enable) {\n await enableQualityGates(projectRoot, frameManager, db, spinner);\n } else if (options.disable) {\n await disableQualityGates(projectRoot, spinner);\n } else if (options.status) {\n await showStatus(projectRoot, frameManager, db);\n } else if (options.config) {\n await configureQualityGates(projectRoot);\n } else if (options.run) {\n await runQualityGates(projectRoot, frameManager, db, spinner);\n } else if (options.history) {\n await showHistory(frameManager);\n } else if (options.setup) {\n await setupWizard(projectRoot, frameManager, db);\n } else {\n // Default: show status\n await showStatus(projectRoot, frameManager, db);\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Error: ${error}`));\n process.exit(1);\n }\n });\n\n return cmd;\n}\n\n/**\n * Enable quality gates\n */\nasync function enableQualityGates(\n projectRoot: string,\n frameManager: FrameManager,\n db: any,\n spinner: any\n) {\n spinner.start('Enabling quality gates...');\n\n try {\n // Load or create config\n const config = await loadConfig(projectRoot);\n config.qualityGates = {\n ...config.qualityGates,\n runTests: true,\n runCodeReview: true,\n runLinter: true,\n };\n\n // Initialize hooks\n const hooks = new PostTaskHooks(frameManager, db, config);\n await hooks.initialize();\n\n // Save config\n await saveConfig(projectRoot, config);\n\n // Create systemd-style service file for persistence\n await createServiceFile(projectRoot);\n\n spinner.succeed(chalk.green('\u2705 Quality gates enabled'));\n\n console.log(chalk.bold('\\n\uD83D\uDD0D Quality Gates Active:'));\n console.log(' \u2022 Auto-run tests after code changes');\n console.log(' \u2022 Auto-run linter on file saves');\n console.log(' \u2022 Auto-trigger code review on task completion');\n console.log(' \u2022 Real-time file change monitoring');\n\n console.log(chalk.gray('\\nDisable with: stackmemory quality --disable'));\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Failed to enable quality gates: ${error}`));\n }\n}\n\n/**\n * Disable quality gates\n */\nasync function disableQualityGates(projectRoot: string, spinner: ora.Ora) {\n spinner.start('Disabling quality gates...');\n\n try {\n const config = await loadConfig(projectRoot);\n config.qualityGates = {\n ...config.qualityGates,\n runTests: false,\n runCodeReview: false,\n runLinter: false,\n };\n\n await saveConfig(projectRoot, config);\n\n // Remove service file\n const serviceFile = path.join(\n projectRoot,\n '.stackmemory',\n 'quality.service'\n );\n try {\n await fs.unlink(serviceFile);\n } catch {\n // Service file doesn't exist\n }\n\n spinner.succeed(chalk.green('\u2705 Quality gates disabled'));\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Failed to disable quality gates: ${error}`));\n }\n}\n\n/**\n * Show quality gate status\n */\nasync function showStatus(\n projectRoot: string,\n frameManager: FrameManager,\n db: Database\n) {\n console.log(chalk.bold('\\n\uD83D\uDCCA Quality Gates Status\\n'));\n\n try {\n const config = await loadConfig(projectRoot);\n\n // Overall status\n const isEnabled =\n config.qualityGates?.runTests ||\n config.qualityGates?.runCodeReview ||\n config.qualityGates?.runLinter;\n console.log(\n `Status: ${isEnabled ? chalk.green('\u2705 Enabled') : chalk.yellow('\u26A0\uFE0F Disabled')}`\n );\n\n // Individual gates\n console.log('\\nGates:');\n console.log(\n ` Tests: ${config.qualityGates?.runTests ? '\u2705' : '\u274C'} ${getTestCommand(config)}`\n );\n console.log(\n ` Linter: ${config.qualityGates?.runLinter ? '\u2705' : '\u274C'} ${getLintCommand(config)}`\n );\n console.log(\n ` Code Review: ${config.qualityGates?.runCodeReview ? '\u2705' : '\u274C'}`\n );\n console.log(\n ` Coverage: ${config.qualityGates?.requireTestCoverage ? '\u2705' : '\u274C'}`\n );\n\n // Detected frameworks\n if (config.testFrameworks?.detected.length) {\n console.log('\\nDetected Frameworks:');\n config.testFrameworks.detected.forEach((fw) => {\n console.log(` \u2022 ${fw}`);\n });\n }\n\n // Recent quality results\n const recentResults = await getRecentQualityResults(frameManager);\n if (recentResults.length > 0) {\n console.log('\\nRecent Results:');\n recentResults.slice(0, 3).forEach((result) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n console.log(\n ` ${icon} ${result.frameName} (${new Date(result.timestamp).toLocaleString()})`\n );\n });\n }\n } catch (error: unknown) {\n console.error(chalk.red('Failed to get status:', error));\n }\n}\n\n/**\n * Configure quality gates\n */\nasync function configureQualityGates(projectRoot: string) {\n console.log(chalk.bold('\\n\u2699\uFE0F Quality Gates Configuration\\n'));\n\n try {\n const config = await loadConfig(projectRoot);\n\n const answers = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'runTests',\n message: 'Auto-run tests after code changes?',\n default: config.qualityGates?.runTests ?? true,\n },\n {\n type: 'confirm',\n name: 'runLinter',\n message: 'Auto-run linter on file changes?',\n default: config.qualityGates?.runLinter ?? true,\n },\n {\n type: 'confirm',\n name: 'runCodeReview',\n message: 'Auto-trigger code review on task completion?',\n default: config.qualityGates?.runCodeReview ?? true,\n },\n {\n type: 'confirm',\n name: 'requireTestCoverage',\n message: 'Require test coverage checks?',\n default: config.qualityGates?.requireTestCoverage ?? false,\n },\n {\n type: 'confirm',\n name: 'blockOnFailure',\n message: 'Block further work when quality gates fail?',\n default: config.qualityGates?.blockOnFailure ?? false,\n },\n {\n type: 'input',\n name: 'testCommand',\n message: 'Custom test command (leave empty for auto-detect):',\n default: config.testFrameworks?.testCommand || '',\n },\n {\n type: 'input',\n name: 'lintCommand',\n message: 'Custom lint command (leave empty for auto-detect):',\n default: config.testFrameworks?.lintCommand || '',\n },\n ]);\n\n // Update config\n config.qualityGates = {\n runTests: answers.runTests,\n runLinter: answers.runLinter,\n runCodeReview: answers.runCodeReview,\n requireTestCoverage: answers.requireTestCoverage,\n blockOnFailure: answers.blockOnFailure,\n };\n\n config.testFrameworks = {\n ...config.testFrameworks,\n testCommand: answers.testCommand || config.testFrameworks?.testCommand,\n lintCommand: answers.lintCommand || config.testFrameworks?.lintCommand,\n };\n\n await saveConfig(projectRoot, config);\n\n console.log(chalk.green('\\n\u2705 Configuration saved'));\n console.log(chalk.gray('Enable with: stackmemory quality --enable'));\n } catch (error: unknown) {\n console.error(chalk.red('Configuration failed:', error));\n }\n}\n\n/**\n * Run quality gates manually\n */\nasync function runQualityGates(\n projectRoot: string,\n frameManager: FrameManager,\n db: any,\n spinner: any\n) {\n spinner.start('Running quality gates...');\n\n try {\n const config = await loadConfig(projectRoot);\n const hooks = new PostTaskHooks(frameManager, db, config);\n\n // Simulate a task completion event\n const mockEvent = {\n taskType: 'task_complete' as const,\n frameId: 'manual-run',\n frameName: 'Manual quality gate run',\n files: await getRecentlyModifiedFiles(projectRoot),\n changes: { added: 0, removed: 0, modified: 1 },\n metadata: { trigger: 'manual' },\n };\n\n // Run quality gates\n const results = await (hooks as any).runQualityGates(mockEvent);\n\n spinner.succeed(chalk.green('\u2705 Quality gates completed'));\n\n // Show results\n console.log(chalk.bold('\\n\uD83D\uDCCB Quality Gate Results:\\n'));\n\n results.forEach((result: QualityGateResult) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n const duration =\n result.duration < 1000\n ? `${result.duration}ms`\n : `${Math.round(result.duration / 1000)}s`;\n\n console.log(`${icon} ${result.gate} (${duration})`);\n\n if (!result.passed && result.issues) {\n result.issues.slice(0, 3).forEach((issue) => {\n console.log(\n ` ${issue.severity === 'error' ? '\u274C' : '\u26A0\uFE0F'} ${issue.message}`\n );\n });\n if (result.issues.length > 3) {\n console.log(\n chalk.gray(` ... and ${result.issues.length - 3} more issues`)\n );\n }\n }\n });\n\n const allPassed = results.every((r: QualityGateResult) => r.passed);\n if (allPassed) {\n console.log(chalk.green('\\n\uD83C\uDF89 All quality gates passed!'));\n } else {\n console.log(\n chalk.yellow('\\n\u26A0\uFE0F Some quality gates failed. See details above.')\n );\n }\n } catch (error: unknown) {\n spinner.fail(chalk.red(`Quality gates failed: ${error}`));\n }\n}\n\n/**\n * Show quality gate history\n */\nasync function showHistory(frameManager: FrameManager) {\n console.log(chalk.bold('\\n\uD83D\uDCC8 Quality Gate History\\n'));\n\n try {\n const results = await getRecentQualityResults(frameManager);\n\n if (results.length === 0) {\n console.log(chalk.gray('No quality gate history found'));\n return;\n }\n\n results.slice(0, 10).forEach((result, index) => {\n const icon = result.passed ? '\u2705' : '\u274C';\n const date = new Date(result.timestamp).toLocaleDateString();\n const time = new Date(result.timestamp).toLocaleTimeString();\n\n console.log(`${icon} ${result.frameName}`);\n console.log(chalk.gray(` ${date} ${time} - ${result.duration}ms`));\n\n if (result.gates) {\n result.gates.forEach((gate: any) => {\n const gateIcon = gate.passed ? ' \u2713' : ' \u2717';\n console.log(chalk.gray(`${gateIcon} ${gate.gate}`));\n });\n }\n\n if (index < results.length - 1) console.log('');\n });\n } catch (error: unknown) {\n console.error(chalk.red('Failed to get history:', error));\n }\n}\n\n/**\n * Setup wizard\n */\nasync function setupWizard(\n projectRoot: string,\n frameManager: FrameManager,\n db: Database\n) {\n console.log(chalk.bold('\uD83E\uDDD9 Quality Gates Setup Wizard\\n'));\n\n console.log(\n 'This wizard will help you configure automatic quality gates for your project.'\n );\n console.log(\n 'Quality gates run automatically after Claude completes tasks.\\n'\n );\n\n // Detect project type\n const packageJsonPath = path.join(projectRoot, 'package.json');\n let projectType = 'unknown';\n const detectedFrameworks: string[] = [];\n\n try {\n const packageJson = JSON.parse(await fs.readFile(packageJsonPath, 'utf-8'));\n const deps = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n };\n\n if (deps.react) projectType = 'React';\n else if (deps.vue) projectType = 'Vue';\n else if (deps.angular) projectType = 'Angular';\n else if (deps.express) projectType = 'Node.js API';\n else if (deps.nextjs) projectType = 'Next.js';\n\n if (deps.jest) detectedFrameworks.push('Jest');\n if (deps.vitest) detectedFrameworks.push('Vitest');\n if (deps.playwright) detectedFrameworks.push('Playwright');\n if (deps.eslint) detectedFrameworks.push('ESLint');\n } catch {\n // No package.json or invalid\n }\n\n console.log(`\uD83D\uDCE6 Detected project: ${chalk.cyan(projectType)}`);\n if (detectedFrameworks.length) {\n console.log(\n `\uD83D\uDD27 Detected tools: ${chalk.cyan(detectedFrameworks.join(', '))}`\n );\n }\n console.log('');\n\n const { proceed } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'proceed',\n message: 'Continue with setup?',\n default: true,\n },\n ]);\n\n if (!proceed) {\n console.log(chalk.gray('Setup cancelled'));\n return;\n }\n\n // Configure quality gates\n await configureQualityGates(projectRoot);\n\n // Enable quality gates\n const { enable } = await inquirer.prompt([\n {\n type: 'confirm',\n name: 'enable',\n message: 'Enable quality gates now?',\n default: true,\n },\n ]);\n\n if (enable) {\n const spinner = ora();\n await enableQualityGates(projectRoot, frameManager, db, spinner);\n }\n\n console.log(chalk.bold('\\n\uD83C\uDF89 Setup Complete!\\n'));\n console.log(\n 'Quality gates will now run automatically after Claude completes tasks.'\n );\n console.log('Check status with: stackmemory quality --status');\n}\n\n// Helper functions\n\nasync function getProjectRoot(): Promise<string> {\n return process.cwd();\n}\n\nasync function loadConfig(\n projectRoot: string\n): Promise<Partial<PostTaskConfig>> {\n const configPath = path.join(projectRoot, '.stackmemory', 'config.json');\n\n try {\n const content = await fs.readFile(configPath, 'utf-8');\n return JSON.parse(content);\n } catch {\n return {};\n }\n}\n\nasync function saveConfig(\n projectRoot: string,\n config: Partial<PostTaskConfig>\n): Promise<void> {\n const configPath = path.join(projectRoot, '.stackmemory', 'config.json');\n await fs.mkdir(path.dirname(configPath), { recursive: true });\n await fs.writeFile(configPath, JSON.stringify(config, null, 2), 'utf-8');\n}\n\nfunction getTestCommand(config: Partial<PostTaskConfig>): string {\n if (config.testFrameworks?.testCommand) {\n return chalk.gray(`(${config.testFrameworks.testCommand})`);\n }\n return chalk.gray('(auto-detect)');\n}\n\nfunction getLintCommand(config: Partial<PostTaskConfig>): string {\n if (config.testFrameworks?.lintCommand) {\n return chalk.gray(`(${config.testFrameworks.lintCommand})`);\n }\n return chalk.gray('(auto-detect)');\n}\n\nasync function getRecentQualityResults(\n frameManager: FrameManager\n): Promise<any[]> {\n // This would query the frame metadata for quality gate results\n // For now, return empty array\n return [];\n}\n\nasync function getRecentlyModifiedFiles(\n projectRoot: string\n): Promise<string[]> {\n try {\n const { execFileSync } = await import('child_process');\n const output = execFileSync('git', ['diff', '--name-only', 'HEAD~1'], {\n cwd: projectRoot,\n encoding: 'utf-8',\n });\n return output\n .trim()\n .split('\\n')\n .filter((f) => f.length > 0);\n } catch {\n return [];\n }\n}\n\nasync function createServiceFile(projectRoot: string): Promise<void> {\n const serviceContent = `# StackMemory Quality Gates Service\n# This file indicates quality gates are enabled\n# Created: ${new Date().toISOString()}\n`;\n\n const servicePath = path.join(projectRoot, '.stackmemory', 'quality.service');\n await fs.writeFile(servicePath, serviceContent, 'utf-8');\n}\n\n// Export for use in main CLI\nexport default createQualityCommand();\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -11,7 +11,7 @@ import {
|
|
|
11
11
|
} from "../../skills/unified-rlm-orchestrator.js";
|
|
12
12
|
import { DualStackManager } from "../../core/context/dual-stack-manager.js";
|
|
13
13
|
import { FrameHandoffManager } from "../../core/context/frame-handoff-manager.js";
|
|
14
|
-
import { FrameManager } from "../../core/context/
|
|
14
|
+
import { FrameManager } from "../../core/context/index.js";
|
|
15
15
|
import { ContextRetriever } from "../../core/retrieval/context-retriever.js";
|
|
16
16
|
import { SQLiteAdapter } from "../../core/database/sqlite-adapter.js";
|
|
17
17
|
import { LinearTaskManager } from "../../features/tasks/linear-task-manager.js";
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/skills.ts"],
|
|
4
|
-
"sourcesContent": ["#!/usr/bin/env node\n/**\n * Claude Skills CLI Commands\n * Integrates Claude skills into the stackmemory CLI\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport {\n ClaudeSkillsManager,\n type SkillContext,\n} from '../../skills/claude-skills.js';\nimport {\n UnifiedRLMOrchestrator,\n initializeUnifiedOrchestrator,\n} from '../../skills/unified-rlm-orchestrator.js';\nimport { DualStackManager } from '../../core/context/dual-stack-manager.js';\nimport { FrameHandoffManager } from '../../core/context/frame-handoff-manager.js';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport { ContextRetriever } from '../../core/retrieval/context-retriever.js';\nimport { SQLiteAdapter } from '../../core/database/sqlite-adapter.js';\nimport { LinearTaskManager } from '../../features/tasks/linear-task-manager.js';\nimport { ConfigManager } from '../../core/config/config-manager.js';\nimport * as path from 'path';\nimport * as os from 'os';\nimport {\n SystemError,\n DatabaseError,\n ErrorCode,\n} from '../../core/errors/index.js';\n\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new SystemError(\n `Environment variable ${key} is required`,\n ErrorCode.CONFIGURATION_ERROR,\n { variable: key }\n );\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\nasync function initializeSkillContext(): Promise<{\n context: SkillContext;\n unifiedOrchestrator: UnifiedRLMOrchestrator;\n}> {\n const config = ConfigManager.getInstance();\n const projectId = config.get('project.id') || 'default-project';\n const userId = config.get('user.id') || process.env['USER'] || 'default';\n\n const dbPath = path.join(\n os.homedir(),\n '.stackmemory',\n 'data',\n projectId,\n 'stackmemory.db'\n );\n\n const database = new SQLiteAdapter(projectId, { dbPath });\n await database.connect();\n\n // Get raw database for FrameManager\n const rawDatabase = database.getRawDatabase();\n if (!rawDatabase) {\n throw new DatabaseError(\n 'Failed to get raw database connection',\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' }\n );\n }\n\n // Validate database has required methods\n if (typeof rawDatabase.exec !== 'function') {\n throw new DatabaseError(\n `Invalid database instance: missing exec() method. Got: ${typeof rawDatabase.exec}`,\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' }\n );\n }\n\n // Test database connectivity\n try {\n rawDatabase.exec('SELECT 1');\n } catch (err) {\n throw new DatabaseError(\n `Database connection test failed: ${(err as Error).message}`,\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' },\n err as Error\n );\n }\n\n const dualStackManager = new DualStackManager(database, projectId, userId);\n const handoffManager = new FrameHandoffManager(dualStackManager);\n const contextRetriever = new ContextRetriever(database);\n const frameManager = new FrameManager(rawDatabase, projectId);\n const taskStore = new LinearTaskManager();\n\n const context: SkillContext = {\n projectId,\n userId,\n dualStackManager,\n handoffManager,\n contextRetriever,\n database,\n frameManager,\n };\n\n // Initialize unified RLM orchestrator\n const unifiedOrchestrator = initializeUnifiedOrchestrator(\n frameManager,\n dualStackManager,\n contextRetriever,\n taskStore,\n context\n );\n\n return { context, unifiedOrchestrator };\n}\n\nexport function createSkillsCommand(): Command {\n const skillsCmd = new Command('skills').description(\n 'Execute Claude skills for enhanced workflow'\n );\n\n // Handoff skill command\n skillsCmd\n .command('handoff <targetUser> <message>')\n .description('Streamline frame handoffs between team members')\n .option(\n '-p, --priority <level>',\n 'Set priority (low, medium, high, critical)',\n 'medium'\n )\n .option('-f, --frames <frames...>', 'Specific frames to handoff')\n .option('--no-auto-detect', 'Disable auto-detection of frames')\n .action(async (targetUser, message, options) => {\n const spinner = ora('Initiating handoff...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n // Use unified RLM orchestrator for RLM-first execution\n const result = await unifiedOrchestrator.executeSkill(\n 'handoff',\n [targetUser, message],\n {\n priority: options.priority,\n frames: options.frames,\n autoDetect: options.autoDetect !== false,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nHandoff Details:'));\n console.log(` ID: ${result.data.handoffId}`);\n console.log(` Frames: ${result.data.frameCount}`);\n console.log(` Priority: ${result.data.priority}`);\n if (result.data.actionItems?.length > 0) {\n console.log(chalk.yellow('\\n Action Items:'));\n result.data.actionItems.forEach((item) => {\n console.log(` \u2022 ${item}`);\n });\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Checkpoint skill commands\n const checkpointCmd = skillsCmd\n .command('checkpoint')\n .description('Create and manage recovery points');\n\n checkpointCmd\n .command('create <description>')\n .description('Create a new checkpoint')\n .option('--files <files...>', 'Include specific files in checkpoint')\n .option('--auto-detect-risky', 'Auto-detect risky operations')\n .action(async (description, options) => {\n const spinner = ora('Creating checkpoint...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill(\n 'checkpoint',\n ['create', description],\n {\n includeFiles: options.files,\n autoDetectRisky: options.autoDetectRisky,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nCheckpoint Info:'));\n console.log(` ID: ${result.data.checkpointId}`);\n console.log(` Time: ${result.data.timestamp}`);\n console.log(` Frames: ${result.data.frameCount}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('restore <checkpointId>')\n .description('Restore from a checkpoint')\n .action(async (checkpointId) => {\n const spinner = ora('Restoring checkpoint...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('checkpoint', [\n 'restore',\n checkpointId,\n ]);\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nRestored:'));\n console.log(` Frames: ${result.data.frameCount}`);\n console.log(` Files: ${result.data.filesRestored}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('list')\n .description('List available checkpoints')\n .option('-l, --limit <number>', 'Limit number of results', '10')\n .option('-s, --since <date>', 'Show checkpoints since date')\n .action(async (options) => {\n const spinner = ora('Loading checkpoints...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill(\n 'checkpoint',\n ['list'],\n {\n limit: parseInt(options.limit),\n since: options.since ? new Date(options.since) : undefined,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.cyan('Available Checkpoints:\\n'));\n if (result.data && result.data.length > 0) {\n result.data.forEach((cp: any) => {\n const riskIndicator = cp.risky ? chalk.yellow(' [RISKY]') : '';\n console.log(`${chalk.bold(cp.id)}${riskIndicator}`);\n console.log(` ${cp.description}`);\n console.log(\n chalk.gray(` ${cp.timestamp} (${cp.frameCount} frames)\\n`)\n );\n });\n } else {\n console.log(chalk.gray('No checkpoints found'));\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('diff <checkpoint1> <checkpoint2>')\n .description('Show differences between two checkpoints')\n .action(async (checkpoint1, checkpoint2) => {\n const spinner = ora('Comparing checkpoints...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('checkpoint', [\n 'diff',\n checkpoint1,\n checkpoint2,\n ]);\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.cyan('Checkpoint Diff:\\n'));\n if (result.data) {\n console.log(` Time difference: ${result.data.timeDiff}`);\n console.log(` Frame difference: ${result.data.framesDiff}`);\n console.log(` New frames: ${result.data.newFrames}`);\n console.log(` Removed frames: ${result.data.removedFrames}`);\n console.log(` Modified frames: ${result.data.modifiedFrames}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Context Archaeologist skill command\n skillsCmd\n .command('dig <query>')\n .description('Deep historical context retrieval')\n .option(\n '-d, --depth <depth>',\n 'Search depth (e.g., 30days, 6months, all)',\n '30days'\n )\n .option('--patterns', 'Extract patterns from results')\n .option('--decisions', 'Extract key decisions')\n .option('--timeline', 'Generate activity timeline')\n .action(async (query, options) => {\n const spinner = ora('Digging through context...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('dig', [query], {\n depth: options.depth,\n patterns: options.patterns,\n decisions: options.decisions,\n timeline: options.timeline,\n });\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n\n if (result.data) {\n console.log(\n chalk.cyan(\n `\\nSearched ${result.data.timeRange.from} to ${result.data.timeRange.to}`\n )\n );\n\n if (result.data.summary) {\n console.log('\\n' + result.data.summary);\n } else {\n // Display top results\n if (result.data.topResults?.length > 0) {\n console.log(chalk.cyan('\\nTop Results:'));\n result.data.topResults.forEach((r: any) => {\n console.log(\n ` ${chalk.yellow(`[${r.score.toFixed(2)}]`)} ${r.summary}`\n );\n });\n }\n\n // Display patterns if found\n if (result.data.patterns?.length > 0) {\n console.log(chalk.cyan('\\nDetected Patterns:'));\n result.data.patterns.forEach((p: any) => {\n console.log(` ${p.name}: ${p.count} occurrences`);\n });\n }\n\n // Display decisions if found\n if (result.data.decisions?.length > 0) {\n console.log(chalk.cyan('\\nKey Decisions:'));\n result.data.decisions.slice(0, 5).forEach((d: any) => {\n console.log(\n ` ${chalk.gray(new Date(d.timestamp).toLocaleDateString())}: ${d.decision}`\n );\n });\n }\n\n // Display timeline if generated\n if (result.data.timeline?.length > 0) {\n console.log(chalk.cyan('\\nActivity Timeline:'));\n result.data.timeline.slice(0, 5).forEach((t: any) => {\n console.log(` ${t.date}: ${t.itemCount} activities`);\n });\n }\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // RLM (Recursive Language Model) skill command\n skillsCmd\n .command('rlm <task>')\n .description('Execute complex tasks with recursive agent orchestration')\n .option('--max-parallel <number>', 'Maximum concurrent subagents', '5')\n .option('--max-recursion <number>', 'Maximum recursion depth', '4')\n .option(\n '--max-tokens-per-agent <number>',\n 'Token budget per subagent',\n '30000'\n )\n .option('--review-stages <number>', 'Number of review iterations', '3')\n .option(\n '--quality-threshold <number>',\n 'Target quality score (0-1)',\n '0.85'\n )\n .option(\n '--test-mode <mode>',\n 'Test generation mode (unit/integration/e2e/all)',\n 'all'\n )\n .option('--verbose', 'Show all recursive operations', false)\n .option(\n '--share-context-realtime',\n 'Share discoveries between agents',\n true\n )\n .option('--retry-failed-agents', 'Retry on failure', true)\n .option('--timeout-per-agent <number>', 'Timeout in seconds', '300')\n .action(async (task, options) => {\n const spinner = ora('Initializing RLM orchestrator...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n spinner.text = 'Decomposing task...';\n\n const result = await unifiedOrchestrator.executeSkill('rlm', [task], {\n maxParallel: parseInt(options.maxParallel),\n maxRecursionDepth: parseInt(options.maxRecursion),\n maxTokensPerAgent: parseInt(options.maxTokensPerAgent),\n reviewStages: parseInt(options.reviewStages),\n qualityThreshold: parseFloat(options.qualityThreshold),\n testGenerationMode: options.testMode,\n verboseLogging: options.verbose,\n shareContextRealtime: options.shareContextRealtime,\n retryFailedAgents: options.retryFailedAgents,\n timeoutPerAgent: parseInt(options.timeoutPerAgent) * 1000,\n });\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), 'RLM execution completed');\n\n if (result.data) {\n console.log(chalk.cyan('\\nExecution Summary:'));\n console.log(` Total tokens: ${result.data.totalTokens}`);\n console.log(\n ` Estimated cost: $${result.data.totalCost.toFixed(2)}`\n );\n console.log(` Duration: ${result.data.duration}ms`);\n console.log(` Tests generated: ${result.data.testsGenerated}`);\n console.log(` Issues found: ${result.data.issuesFound}`);\n console.log(` Issues fixed: ${result.data.issuesFixed}`);\n\n if (result.data.improvements?.length > 0) {\n console.log(chalk.cyan('\\nImprovements:'));\n result.data.improvements.forEach((imp: string) => {\n console.log(` \u2022 ${imp}`);\n });\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Help command for skills\n skillsCmd\n .command('help [skill]')\n .description('Show help for a specific skill')\n .action(async (skill) => {\n if (skill) {\n // Show specific skill help\n switch (skill) {\n case 'lint':\n console.log(`\nlint (RLM-Orchestrated)\nPrimary Agent: linting\nSecondary Agents: improve\n\nComprehensive linting of code: Check syntax, types, formatting, security, performance, and dead code. Provide fixes.\n\nThis skill is executed through RLM orchestration for:\n- Automatic task decomposition\n- Parallel agent execution\n- Multi-stage quality review\n- Comprehensive result aggregation\n\nUsage:\n stackmemory skills lint # Lint current directory\n stackmemory skills lint src/ # Lint specific directory\n stackmemory skills lint src/file.ts # Lint specific file\n\nOptions:\n --fix Automatically fix issues where possible\n --format Focus on formatting issues\n --security Focus on security vulnerabilities\n --performance Focus on performance issues\n --verbose Show detailed output\n`);\n break;\n default:\n console.log(\n `Unknown skill: ${skill}. Use \"stackmemory skills help\" to see all available skills.`\n );\n }\n } else {\n console.log(\n chalk.cyan('Available Claude Skills (RLM-Orchestrated):\\n')\n );\n console.log(\n ' handoff - Streamline frame handoffs between team members'\n );\n console.log(' checkpoint - Create and manage recovery points');\n console.log(' dig - Deep historical context retrieval');\n console.log(\n ' lint - Comprehensive code linting and quality checks'\n );\n console.log(' test - Generate comprehensive test suites');\n console.log(' review - Multi-stage code review and improvements');\n console.log(' refactor - Refactor code for better architecture');\n console.log(' publish - Prepare and execute releases');\n console.log(' rlm - Direct recursive agent orchestration\\n');\n console.log(\n chalk.yellow(\n '\\nAll skills now use RLM orchestration for intelligent task decomposition'\n )\n );\n console.log(\n 'Use \"stackmemory skills help <skill>\" for detailed help on each skill'\n );\n }\n });\n\n return skillsCmd;\n}\n"],
|
|
4
|
+
"sourcesContent": ["#!/usr/bin/env node\n/**\n * Claude Skills CLI Commands\n * Integrates Claude skills into the stackmemory CLI\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport {\n ClaudeSkillsManager,\n type SkillContext,\n} from '../../skills/claude-skills.js';\nimport {\n UnifiedRLMOrchestrator,\n initializeUnifiedOrchestrator,\n} from '../../skills/unified-rlm-orchestrator.js';\nimport { DualStackManager } from '../../core/context/dual-stack-manager.js';\nimport { FrameHandoffManager } from '../../core/context/frame-handoff-manager.js';\nimport { FrameManager } from '../../core/context/index.js';\nimport { ContextRetriever } from '../../core/retrieval/context-retriever.js';\nimport { SQLiteAdapter } from '../../core/database/sqlite-adapter.js';\nimport { LinearTaskManager } from '../../features/tasks/linear-task-manager.js';\nimport { ConfigManager } from '../../core/config/config-manager.js';\nimport * as path from 'path';\nimport * as os from 'os';\nimport {\n SystemError,\n DatabaseError,\n ErrorCode,\n} from '../../core/errors/index.js';\n\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new SystemError(\n `Environment variable ${key} is required`,\n ErrorCode.CONFIGURATION_ERROR,\n { variable: key }\n );\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\nasync function initializeSkillContext(): Promise<{\n context: SkillContext;\n unifiedOrchestrator: UnifiedRLMOrchestrator;\n}> {\n const config = ConfigManager.getInstance();\n const projectId = config.get('project.id') || 'default-project';\n const userId = config.get('user.id') || process.env['USER'] || 'default';\n\n const dbPath = path.join(\n os.homedir(),\n '.stackmemory',\n 'data',\n projectId,\n 'stackmemory.db'\n );\n\n const database = new SQLiteAdapter(projectId, { dbPath });\n await database.connect();\n\n // Get raw database for FrameManager\n const rawDatabase = database.getRawDatabase();\n if (!rawDatabase) {\n throw new DatabaseError(\n 'Failed to get raw database connection',\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' }\n );\n }\n\n // Validate database has required methods\n if (typeof rawDatabase.exec !== 'function') {\n throw new DatabaseError(\n `Invalid database instance: missing exec() method. Got: ${typeof rawDatabase.exec}`,\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' }\n );\n }\n\n // Test database connectivity\n try {\n rawDatabase.exec('SELECT 1');\n } catch (err) {\n throw new DatabaseError(\n `Database connection test failed: ${(err as Error).message}`,\n ErrorCode.DB_CONNECTION_FAILED,\n { projectId, operation: 'initializeSkillContext' },\n err as Error\n );\n }\n\n const dualStackManager = new DualStackManager(database, projectId, userId);\n const handoffManager = new FrameHandoffManager(dualStackManager);\n const contextRetriever = new ContextRetriever(database);\n const frameManager = new FrameManager(rawDatabase, projectId);\n const taskStore = new LinearTaskManager();\n\n const context: SkillContext = {\n projectId,\n userId,\n dualStackManager,\n handoffManager,\n contextRetriever,\n database,\n frameManager,\n };\n\n // Initialize unified RLM orchestrator\n const unifiedOrchestrator = initializeUnifiedOrchestrator(\n frameManager,\n dualStackManager,\n contextRetriever,\n taskStore,\n context\n );\n\n return { context, unifiedOrchestrator };\n}\n\nexport function createSkillsCommand(): Command {\n const skillsCmd = new Command('skills').description(\n 'Execute Claude skills for enhanced workflow'\n );\n\n // Handoff skill command\n skillsCmd\n .command('handoff <targetUser> <message>')\n .description('Streamline frame handoffs between team members')\n .option(\n '-p, --priority <level>',\n 'Set priority (low, medium, high, critical)',\n 'medium'\n )\n .option('-f, --frames <frames...>', 'Specific frames to handoff')\n .option('--no-auto-detect', 'Disable auto-detection of frames')\n .action(async (targetUser, message, options) => {\n const spinner = ora('Initiating handoff...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n // Use unified RLM orchestrator for RLM-first execution\n const result = await unifiedOrchestrator.executeSkill(\n 'handoff',\n [targetUser, message],\n {\n priority: options.priority,\n frames: options.frames,\n autoDetect: options.autoDetect !== false,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nHandoff Details:'));\n console.log(` ID: ${result.data.handoffId}`);\n console.log(` Frames: ${result.data.frameCount}`);\n console.log(` Priority: ${result.data.priority}`);\n if (result.data.actionItems?.length > 0) {\n console.log(chalk.yellow('\\n Action Items:'));\n result.data.actionItems.forEach((item) => {\n console.log(` \u2022 ${item}`);\n });\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Checkpoint skill commands\n const checkpointCmd = skillsCmd\n .command('checkpoint')\n .description('Create and manage recovery points');\n\n checkpointCmd\n .command('create <description>')\n .description('Create a new checkpoint')\n .option('--files <files...>', 'Include specific files in checkpoint')\n .option('--auto-detect-risky', 'Auto-detect risky operations')\n .action(async (description, options) => {\n const spinner = ora('Creating checkpoint...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill(\n 'checkpoint',\n ['create', description],\n {\n includeFiles: options.files,\n autoDetectRisky: options.autoDetectRisky,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nCheckpoint Info:'));\n console.log(` ID: ${result.data.checkpointId}`);\n console.log(` Time: ${result.data.timestamp}`);\n console.log(` Frames: ${result.data.frameCount}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('restore <checkpointId>')\n .description('Restore from a checkpoint')\n .action(async (checkpointId) => {\n const spinner = ora('Restoring checkpoint...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('checkpoint', [\n 'restore',\n checkpointId,\n ]);\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n if (result.data) {\n console.log(chalk.cyan('\\nRestored:'));\n console.log(` Frames: ${result.data.frameCount}`);\n console.log(` Files: ${result.data.filesRestored}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('list')\n .description('List available checkpoints')\n .option('-l, --limit <number>', 'Limit number of results', '10')\n .option('-s, --since <date>', 'Show checkpoints since date')\n .action(async (options) => {\n const spinner = ora('Loading checkpoints...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill(\n 'checkpoint',\n ['list'],\n {\n limit: parseInt(options.limit),\n since: options.since ? new Date(options.since) : undefined,\n }\n );\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.cyan('Available Checkpoints:\\n'));\n if (result.data && result.data.length > 0) {\n result.data.forEach((cp: any) => {\n const riskIndicator = cp.risky ? chalk.yellow(' [RISKY]') : '';\n console.log(`${chalk.bold(cp.id)}${riskIndicator}`);\n console.log(` ${cp.description}`);\n console.log(\n chalk.gray(` ${cp.timestamp} (${cp.frameCount} frames)\\n`)\n );\n });\n } else {\n console.log(chalk.gray('No checkpoints found'));\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n checkpointCmd\n .command('diff <checkpoint1> <checkpoint2>')\n .description('Show differences between two checkpoints')\n .action(async (checkpoint1, checkpoint2) => {\n const spinner = ora('Comparing checkpoints...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('checkpoint', [\n 'diff',\n checkpoint1,\n checkpoint2,\n ]);\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.cyan('Checkpoint Diff:\\n'));\n if (result.data) {\n console.log(` Time difference: ${result.data.timeDiff}`);\n console.log(` Frame difference: ${result.data.framesDiff}`);\n console.log(` New frames: ${result.data.newFrames}`);\n console.log(` Removed frames: ${result.data.removedFrames}`);\n console.log(` Modified frames: ${result.data.modifiedFrames}`);\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Context Archaeologist skill command\n skillsCmd\n .command('dig <query>')\n .description('Deep historical context retrieval')\n .option(\n '-d, --depth <depth>',\n 'Search depth (e.g., 30days, 6months, all)',\n '30days'\n )\n .option('--patterns', 'Extract patterns from results')\n .option('--decisions', 'Extract key decisions')\n .option('--timeline', 'Generate activity timeline')\n .action(async (query, options) => {\n const spinner = ora('Digging through context...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n const result = await unifiedOrchestrator.executeSkill('dig', [query], {\n depth: options.depth,\n patterns: options.patterns,\n decisions: options.decisions,\n timeline: options.timeline,\n });\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), result.message);\n\n if (result.data) {\n console.log(\n chalk.cyan(\n `\\nSearched ${result.data.timeRange.from} to ${result.data.timeRange.to}`\n )\n );\n\n if (result.data.summary) {\n console.log('\\n' + result.data.summary);\n } else {\n // Display top results\n if (result.data.topResults?.length > 0) {\n console.log(chalk.cyan('\\nTop Results:'));\n result.data.topResults.forEach((r: any) => {\n console.log(\n ` ${chalk.yellow(`[${r.score.toFixed(2)}]`)} ${r.summary}`\n );\n });\n }\n\n // Display patterns if found\n if (result.data.patterns?.length > 0) {\n console.log(chalk.cyan('\\nDetected Patterns:'));\n result.data.patterns.forEach((p: any) => {\n console.log(` ${p.name}: ${p.count} occurrences`);\n });\n }\n\n // Display decisions if found\n if (result.data.decisions?.length > 0) {\n console.log(chalk.cyan('\\nKey Decisions:'));\n result.data.decisions.slice(0, 5).forEach((d: any) => {\n console.log(\n ` ${chalk.gray(new Date(d.timestamp).toLocaleDateString())}: ${d.decision}`\n );\n });\n }\n\n // Display timeline if generated\n if (result.data.timeline?.length > 0) {\n console.log(chalk.cyan('\\nActivity Timeline:'));\n result.data.timeline.slice(0, 5).forEach((t: any) => {\n console.log(` ${t.date}: ${t.itemCount} activities`);\n });\n }\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // RLM (Recursive Language Model) skill command\n skillsCmd\n .command('rlm <task>')\n .description('Execute complex tasks with recursive agent orchestration')\n .option('--max-parallel <number>', 'Maximum concurrent subagents', '5')\n .option('--max-recursion <number>', 'Maximum recursion depth', '4')\n .option(\n '--max-tokens-per-agent <number>',\n 'Token budget per subagent',\n '30000'\n )\n .option('--review-stages <number>', 'Number of review iterations', '3')\n .option(\n '--quality-threshold <number>',\n 'Target quality score (0-1)',\n '0.85'\n )\n .option(\n '--test-mode <mode>',\n 'Test generation mode (unit/integration/e2e/all)',\n 'all'\n )\n .option('--verbose', 'Show all recursive operations', false)\n .option(\n '--share-context-realtime',\n 'Share discoveries between agents',\n true\n )\n .option('--retry-failed-agents', 'Retry on failure', true)\n .option('--timeout-per-agent <number>', 'Timeout in seconds', '300')\n .action(async (task, options) => {\n const spinner = ora('Initializing RLM orchestrator...').start();\n\n try {\n const { context, unifiedOrchestrator } = await initializeSkillContext();\n\n spinner.text = 'Decomposing task...';\n\n const result = await unifiedOrchestrator.executeSkill('rlm', [task], {\n maxParallel: parseInt(options.maxParallel),\n maxRecursionDepth: parseInt(options.maxRecursion),\n maxTokensPerAgent: parseInt(options.maxTokensPerAgent),\n reviewStages: parseInt(options.reviewStages),\n qualityThreshold: parseFloat(options.qualityThreshold),\n testGenerationMode: options.testMode,\n verboseLogging: options.verbose,\n shareContextRealtime: options.shareContextRealtime,\n retryFailedAgents: options.retryFailedAgents,\n timeoutPerAgent: parseInt(options.timeoutPerAgent) * 1000,\n });\n\n spinner.stop();\n\n if (result.success) {\n console.log(chalk.green('\u2713'), 'RLM execution completed');\n\n if (result.data) {\n console.log(chalk.cyan('\\nExecution Summary:'));\n console.log(` Total tokens: ${result.data.totalTokens}`);\n console.log(\n ` Estimated cost: $${result.data.totalCost.toFixed(2)}`\n );\n console.log(` Duration: ${result.data.duration}ms`);\n console.log(` Tests generated: ${result.data.testsGenerated}`);\n console.log(` Issues found: ${result.data.issuesFound}`);\n console.log(` Issues fixed: ${result.data.issuesFixed}`);\n\n if (result.data.improvements?.length > 0) {\n console.log(chalk.cyan('\\nImprovements:'));\n result.data.improvements.forEach((imp: string) => {\n console.log(` \u2022 ${imp}`);\n });\n }\n }\n } else {\n console.log(chalk.red('\u2717'), result.message);\n }\n\n await context.database.disconnect();\n } catch (error: unknown) {\n spinner.stop();\n console.error(chalk.red('Error:'), error.message);\n process.exit(1);\n }\n });\n\n // Help command for skills\n skillsCmd\n .command('help [skill]')\n .description('Show help for a specific skill')\n .action(async (skill) => {\n if (skill) {\n // Show specific skill help\n switch (skill) {\n case 'lint':\n console.log(`\nlint (RLM-Orchestrated)\nPrimary Agent: linting\nSecondary Agents: improve\n\nComprehensive linting of code: Check syntax, types, formatting, security, performance, and dead code. Provide fixes.\n\nThis skill is executed through RLM orchestration for:\n- Automatic task decomposition\n- Parallel agent execution\n- Multi-stage quality review\n- Comprehensive result aggregation\n\nUsage:\n stackmemory skills lint # Lint current directory\n stackmemory skills lint src/ # Lint specific directory\n stackmemory skills lint src/file.ts # Lint specific file\n\nOptions:\n --fix Automatically fix issues where possible\n --format Focus on formatting issues\n --security Focus on security vulnerabilities\n --performance Focus on performance issues\n --verbose Show detailed output\n`);\n break;\n default:\n console.log(\n `Unknown skill: ${skill}. Use \"stackmemory skills help\" to see all available skills.`\n );\n }\n } else {\n console.log(\n chalk.cyan('Available Claude Skills (RLM-Orchestrated):\\n')\n );\n console.log(\n ' handoff - Streamline frame handoffs between team members'\n );\n console.log(' checkpoint - Create and manage recovery points');\n console.log(' dig - Deep historical context retrieval');\n console.log(\n ' lint - Comprehensive code linting and quality checks'\n );\n console.log(' test - Generate comprehensive test suites');\n console.log(' review - Multi-stage code review and improvements');\n console.log(' refactor - Refactor code for better architecture');\n console.log(' publish - Prepare and execute releases');\n console.log(' rlm - Direct recursive agent orchestration\\n');\n console.log(\n chalk.yellow(\n '\\nAll skills now use RLM orchestration for intelligent task decomposition'\n )\n );\n console.log(\n 'Use \"stackmemory skills help <skill>\" for detailed help on each skill'\n );\n }\n });\n\n return skillsCmd;\n}\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -7,7 +7,7 @@ import chalk from "chalk";
|
|
|
7
7
|
import * as path from "path";
|
|
8
8
|
import Database from "better-sqlite3";
|
|
9
9
|
import { existsSync } from "fs";
|
|
10
|
-
import { FrameManager } from "../../core/context/
|
|
10
|
+
import { FrameManager } from "../../core/context/index.js";
|
|
11
11
|
import { workflowTemplates } from "../../core/frame/workflow-templates.js";
|
|
12
12
|
import { sessionManager } from "../../core/session/session-manager.js";
|
|
13
13
|
function createWorkflowCommand() {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/workflow.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Workflow command for StackMemory\n * Manages workflow templates and execution\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport * as path from 'path';\nimport Database from 'better-sqlite3';\nimport { existsSync } from 'fs';\nimport { FrameManager } from '../../core/context/
|
|
4
|
+
"sourcesContent": ["/**\n * Workflow command for StackMemory\n * Manages workflow templates and execution\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport * as path from 'path';\nimport Database from 'better-sqlite3';\nimport { existsSync } from 'fs';\nimport { FrameManager } from '../../core/context/index.js';\nimport { workflowTemplates } from '../../core/frame/workflow-templates.js';\nimport { sessionManager } from '../../core/session/session-manager.js';\nimport { getEnv, getOptionalEnv } from '../../utils/env.js';\n\nexport function createWorkflowCommand(): Command {\n const cmd = new Command('workflow')\n .description('Manage structured workflow templates')\n .option('-l, --list', 'List available workflow templates')\n .option('-s, --start <template>', 'Start a new workflow from template')\n .option('--status', 'Show status of active workflow')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = path.join(projectRoot, '.stackmemory', 'context.db');\n\n // Check if StackMemory is initialized\n if (!existsSync(dbPath)) {\n console.error(chalk.red('\u2717 StackMemory not initialized'));\n console.log(chalk.yellow('Run: stackmemory init'));\n if (process.env['NODE_ENV'] !== 'test') {\n process.exit(1);\n }\n return;\n }\n\n if (options.list) {\n await listWorkflows();\n } else if (options.start) {\n await startWorkflow(options.start, dbPath);\n } else if (options.status) {\n await showWorkflowStatus(dbPath);\n } else {\n // Default: list workflows\n await listWorkflows();\n }\n } catch (error: unknown) {\n console.error(chalk.red('Error: ' + (error as Error).message));\n if (process.env['NODE_ENV'] !== 'test') {\n process.exit(1);\n }\n }\n });\n\n return cmd;\n}\n\nasync function listWorkflows(): Promise<void> {\n console.log(chalk.bold('\\n\uD83D\uDCCB Available Workflows'));\n console.log('\u2500'.repeat(40));\n\n Object.entries(workflowTemplates).forEach(([key, template]) => {\n console.log(chalk.cyan(`\\n${key}:`));\n console.log(` ${template.description}`);\n console.log(chalk.gray(` Phases: ${template.phases.length}`));\n\n // Show first few phases\n template.phases.slice(0, 3).forEach((phase) => {\n console.log(` \u2022 ${phase.name}`);\n });\n if (template.phases.length > 3) {\n console.log(` ... and ${template.phases.length - 3} more`);\n }\n });\n\n console.log(\n chalk.gray('\\nStart a workflow: stackmemory workflow --start <name>')\n );\n}\n\nasync function startWorkflow(\n workflowName: string,\n dbPath: string\n): Promise<void> {\n const template =\n workflowTemplates[workflowName as keyof typeof workflowTemplates];\n\n if (!template) {\n console.error(chalk.red(`Unknown workflow: ${workflowName}`));\n console.log(chalk.yellow('Use --list to see available workflows'));\n if (process.env['NODE_ENV'] !== 'test') {\n process.exit(1);\n }\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n // Initialize session\n await sessionManager.initialize();\n const session = await sessionManager.getOrCreateSession({\n projectPath: process.cwd(),\n });\n\n const frameManager = new FrameManager(db, session.projectId);\n\n // Create root frame for workflow\n const workflowId = await frameManager.createFrame({\n type: 'workflow',\n name: `${template.name} Workflow`,\n metadata: {\n workflow: workflowName,\n phases: template.phases.map((p: any) => p.name),\n currentPhase: 0,\n startTime: Date.now(),\n },\n });\n\n console.log(chalk.green(`\u2713 Started ${workflowName} workflow`));\n console.log(chalk.cyan(`Workflow ID: ${workflowId}`));\n console.log('\\nPhases:');\n\n template.phases.forEach((phase, index) => {\n const marker = index === 0 ? '\u2192' : ' ';\n console.log(`${marker} ${index + 1}. ${phase.name}`);\n });\n\n console.log(chalk.gray('\\nTrack progress: stackmemory workflow --status'));\n } finally {\n db.close();\n }\n}\n\nasync function showWorkflowStatus(dbPath: string): Promise<void> {\n const db = new Database(dbPath);\n\n try {\n // Get active workflow frames\n const workflows = db\n .prepare(\n `\n SELECT * FROM frames \n WHERE type = 'workflow' \n AND state = 'active'\n ORDER BY created_at DESC\n `\n )\n .all() as any[];\n\n if (workflows.length === 0) {\n console.log(chalk.yellow('No active workflows'));\n return;\n }\n\n console.log(chalk.bold('\\n\uD83D\uDD04 Active Workflows'));\n console.log('\u2500'.repeat(40));\n\n workflows.forEach((workflow) => {\n const metadata = workflow.metadata ? JSON.parse(workflow.metadata) : {};\n const elapsed = Date.now() - workflow.created_at;\n const minutes = Math.floor(elapsed / 60000);\n\n console.log(chalk.cyan(`\\n${workflow.name}`));\n console.log(` ID: ${workflow.frame_id}`);\n console.log(` Duration: ${minutes} minutes`);\n\n if (metadata.phases) {\n const current = metadata.currentPhase || 0;\n console.log(` Phase: ${current + 1}/${metadata.phases.length}`);\n console.log(` Current: ${metadata.phases[current]}`);\n }\n });\n } finally {\n db.close();\n }\n}\n\nexport default createWorkflowCommand;\n"],
|
|
5
5
|
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -5,7 +5,7 @@ const __filename = __fileURLToPath(import.meta.url);
|
|
|
5
5
|
const __dirname = __pathDirname(__filename);
|
|
6
6
|
import { WorktreeManager } from "../../core/worktree/worktree-manager.js";
|
|
7
7
|
import { ProjectManager } from "../../core/projects/project-manager.js";
|
|
8
|
-
import { FrameManager } from "../../core/context/
|
|
8
|
+
import { FrameManager } from "../../core/context/index.js";
|
|
9
9
|
import chalk from "chalk";
|
|
10
10
|
import Table from "cli-table3";
|
|
11
11
|
import { existsSync } from "fs";
|