@stackmemoryai/stackmemory 0.3.6 → 0.3.8
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 +5 -5
- package/dist/agents/core/agent-task-manager.js.map +2 -2
- package/dist/agents/verifiers/base-verifier.js +2 -2
- package/dist/agents/verifiers/base-verifier.js.map +2 -2
- package/dist/agents/verifiers/formatter-verifier.js.map +2 -2
- package/dist/agents/verifiers/llm-judge.js.map +2 -2
- package/dist/cli/claude-sm.js +13 -13
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/codex-sm.js +13 -13
- package/dist/cli/codex-sm.js.map +2 -2
- package/dist/cli/commands/agent.js.map +2 -2
- package/dist/cli/commands/chromadb.js +261 -46
- package/dist/cli/commands/chromadb.js.map +2 -2
- package/dist/cli/commands/clear.js +10 -3
- package/dist/cli/commands/clear.js.map +2 -2
- package/dist/cli/commands/config.js +43 -33
- package/dist/cli/commands/config.js.map +2 -2
- package/dist/cli/commands/context.js +13 -2
- package/dist/cli/commands/context.js.map +2 -2
- package/dist/cli/commands/dashboard.js +41 -13
- package/dist/cli/commands/dashboard.js.map +2 -2
- package/dist/cli/commands/gc.js +251 -0
- package/dist/cli/commands/gc.js.map +7 -0
- package/dist/cli/commands/handoff.js +12 -1
- package/dist/cli/commands/handoff.js.map +2 -2
- package/dist/cli/commands/infinite-storage.js +92 -40
- package/dist/cli/commands/infinite-storage.js.map +2 -2
- package/dist/cli/commands/linear-create.js +49 -10
- package/dist/cli/commands/linear-create.js.map +2 -2
- package/dist/cli/commands/linear-list.js +45 -11
- package/dist/cli/commands/linear-list.js.map +2 -2
- package/dist/cli/commands/linear-migrate.js +29 -5
- package/dist/cli/commands/linear-migrate.js.map +2 -2
- package/dist/cli/commands/linear-test.js +26 -7
- package/dist/cli/commands/linear-test.js.map +2 -2
- package/dist/cli/commands/linear-unified.js +350 -0
- package/dist/cli/commands/linear-unified.js.map +7 -0
- package/dist/cli/commands/linear.js +17 -6
- package/dist/cli/commands/linear.js.map +2 -2
- package/dist/cli/commands/monitor.js.map +2 -2
- package/dist/cli/commands/onboard.js +35 -8
- package/dist/cli/commands/onboard.js.map +2 -2
- package/dist/cli/commands/quality.js +2 -7
- package/dist/cli/commands/quality.js.map +2 -2
- package/dist/cli/commands/search.js.map +2 -2
- package/dist/cli/commands/session.js +23 -6
- package/dist/cli/commands/session.js.map +2 -2
- package/dist/cli/commands/skills.js +84 -28
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/commands/storage.js +119 -38
- package/dist/cli/commands/storage.js.map +2 -2
- package/dist/cli/commands/tasks.js.map +2 -2
- package/dist/cli/commands/tui.js +13 -2
- package/dist/cli/commands/tui.js.map +2 -2
- package/dist/cli/commands/webhook.js +71 -21
- package/dist/cli/commands/webhook.js.map +2 -2
- package/dist/cli/commands/workflow.js +11 -7
- package/dist/cli/commands/workflow.js.map +2 -2
- package/dist/cli/commands/worktree.js +34 -13
- package/dist/cli/commands/worktree.js.map +2 -2
- package/dist/cli/index.js +7 -5
- package/dist/cli/index.js.map +2 -2
- package/dist/core/config/config-manager.js.map +2 -2
- package/dist/core/config/types.js.map +1 -1
- package/dist/core/context/auto-context.js +10 -6
- package/dist/core/context/auto-context.js.map +2 -2
- 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.map +2 -2
- package/dist/core/context/frame-database.js +13 -3
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-digest.js +7 -5
- package/dist/core/context/frame-digest.js.map +2 -2
- package/dist/core/context/frame-handoff-manager.js.map +2 -2
- package/dist/core/context/frame-manager.js +12 -1
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/context/frame-stack.js +16 -5
- package/dist/core/context/frame-stack.js.map +2 -2
- package/dist/core/context/incremental-gc.js +286 -0
- package/dist/core/context/incremental-gc.js.map +7 -0
- package/dist/core/context/index.js.map +1 -1
- package/dist/core/context/permission-manager.js +12 -1
- package/dist/core/context/permission-manager.js.map +2 -2
- package/dist/core/context/refactored-frame-manager.js +12 -3
- package/dist/core/context/refactored-frame-manager.js.map +2 -2
- package/dist/core/context/shared-context-layer.js +16 -3
- package/dist/core/context/shared-context-layer.js.map +2 -2
- package/dist/core/context/stack-merge-resolver.js.map +2 -2
- package/dist/core/context/validation.js.map +2 -2
- package/dist/core/database/batch-operations.js +112 -86
- package/dist/core/database/batch-operations.js.map +2 -2
- package/dist/core/database/connection-pool.js.map +2 -2
- package/dist/core/database/migration-manager.js.map +2 -2
- package/dist/core/database/paradedb-adapter.js.map +2 -2
- package/dist/core/database/query-cache.js +19 -9
- package/dist/core/database/query-cache.js.map +2 -2
- package/dist/core/database/query-router.js.map +2 -2
- package/dist/core/database/sqlite-adapter.js +1 -1
- package/dist/core/database/sqlite-adapter.js.map +2 -2
- package/dist/core/digest/enhanced-hybrid-digest.js +8 -2
- package/dist/core/digest/enhanced-hybrid-digest.js.map +2 -2
- package/dist/core/errors/recovery.js +9 -2
- package/dist/core/errors/recovery.js.map +2 -2
- package/dist/core/frame/workflow-templates-stub.js.map +1 -1
- package/dist/core/frame/workflow-templates.js +40 -1
- package/dist/core/frame/workflow-templates.js.map +2 -2
- package/dist/core/merge/resolution-engine.js.map +2 -2
- package/dist/core/monitoring/error-handler.js.map +2 -2
- package/dist/core/monitoring/logger.js +19 -3
- package/dist/core/monitoring/logger.js.map +2 -2
- package/dist/core/monitoring/metrics.js +13 -2
- package/dist/core/monitoring/metrics.js.map +2 -2
- package/dist/core/monitoring/progress-tracker.js +12 -1
- package/dist/core/monitoring/progress-tracker.js.map +2 -2
- package/dist/core/monitoring/session-monitor.js.map +2 -2
- package/dist/core/performance/context-cache.js.map +2 -2
- package/dist/core/performance/lazy-context-loader.js +24 -20
- package/dist/core/performance/lazy-context-loader.js.map +2 -2
- package/dist/core/performance/monitor.js.map +2 -2
- package/dist/core/performance/optimized-frame-context.js +27 -12
- package/dist/core/performance/optimized-frame-context.js.map +2 -2
- package/dist/core/performance/performance-benchmark.js +10 -6
- package/dist/core/performance/performance-benchmark.js.map +2 -2
- package/dist/core/performance/performance-profiler.js +63 -15
- package/dist/core/performance/performance-profiler.js.map +2 -2
- package/dist/core/performance/streaming-jsonl-parser.js +5 -1
- package/dist/core/performance/streaming-jsonl-parser.js.map +2 -2
- package/dist/core/persistence/postgres-adapter.js.map +2 -2
- package/dist/core/projects/project-manager.js +14 -20
- package/dist/core/projects/project-manager.js.map +2 -2
- package/dist/core/retrieval/context-retriever.js.map +2 -2
- package/dist/core/retrieval/graph-retrieval.js.map +2 -2
- package/dist/core/retrieval/llm-context-retrieval.js.map +2 -2
- package/dist/core/retrieval/retrieval-benchmarks.js.map +2 -2
- package/dist/core/retrieval/summary-generator.js.map +2 -2
- package/dist/core/session/clear-survival-stub.js +5 -1
- package/dist/core/session/clear-survival-stub.js.map +2 -2
- package/dist/core/session/clear-survival.js +35 -0
- package/dist/core/session/clear-survival.js.map +2 -2
- package/dist/core/session/handoff-generator.js.map +2 -2
- package/dist/core/session/index.js.map +1 -1
- package/dist/core/session/session-manager.js +16 -5
- package/dist/core/session/session-manager.js.map +2 -2
- package/dist/core/skills/skill-storage.js +13 -2
- package/dist/core/skills/skill-storage.js.map +2 -2
- package/dist/core/storage/chromadb-adapter.js +6 -2
- package/dist/core/storage/chromadb-adapter.js.map +2 -2
- package/dist/core/storage/chromadb-simple.js +17 -5
- package/dist/core/storage/chromadb-simple.js.map +2 -2
- package/dist/core/storage/infinite-storage.js +109 -46
- package/dist/core/storage/infinite-storage.js.map +2 -2
- package/dist/core/storage/railway-optimized-storage.js +67 -30
- package/dist/core/storage/railway-optimized-storage.js.map +2 -2
- package/dist/core/storage/remote-storage.js +53 -24
- package/dist/core/storage/remote-storage.js.map +2 -2
- package/dist/core/trace/cli-trace-wrapper.js +25 -7
- package/dist/core/trace/cli-trace-wrapper.js.map +2 -2
- package/dist/core/trace/db-trace-wrapper.js +96 -68
- package/dist/core/trace/db-trace-wrapper.js.map +2 -2
- package/dist/core/trace/debug-trace.js +44 -16
- package/dist/core/trace/debug-trace.js.map +2 -2
- package/dist/core/trace/index.js +50 -35
- package/dist/core/trace/index.js.map +2 -2
- package/dist/core/trace/linear-api-wrapper.js +10 -5
- package/dist/core/trace/linear-api-wrapper.js.map +2 -2
- package/dist/core/trace/trace-demo.js +26 -11
- package/dist/core/trace/trace-demo.js.map +2 -2
- package/dist/core/trace/trace-detector.js +9 -2
- package/dist/core/trace/trace-detector.js.map +2 -2
- package/dist/core/trace/trace-store.js.map +2 -2
- package/dist/core/trace/types.js.map +1 -1
- package/dist/core/utils/compression.js.map +1 -1
- package/dist/core/utils/update-checker.js.map +2 -2
- package/dist/core/worktree/worktree-manager.js +18 -7
- package/dist/core/worktree/worktree-manager.js.map +2 -2
- package/dist/features/analytics/api/analytics-api.js.map +2 -2
- package/dist/features/analytics/core/analytics-service.js +12 -1
- package/dist/features/analytics/core/analytics-service.js.map +2 -2
- package/dist/features/analytics/queries/metrics-queries.js +1 -1
- package/dist/features/analytics/queries/metrics-queries.js.map +2 -2
- package/dist/features/tasks/pebbles-task-store.js.map +2 -2
- package/dist/features/tui/components/analytics-panel.js +36 -15
- package/dist/features/tui/components/analytics-panel.js.map +2 -2
- package/dist/features/tui/components/pr-tracker.js +19 -7
- package/dist/features/tui/components/pr-tracker.js.map +2 -2
- package/dist/features/tui/components/session-monitor.js +22 -9
- package/dist/features/tui/components/session-monitor.js.map +2 -2
- package/dist/features/tui/components/subagent-fleet.js +20 -13
- package/dist/features/tui/components/subagent-fleet.js.map +2 -2
- package/dist/features/tui/components/task-board.js +666 -2
- package/dist/features/tui/components/task-board.js.map +2 -2
- package/dist/features/tui/index.js +16 -5
- package/dist/features/tui/index.js.map +2 -2
- package/dist/features/tui/services/data-service.js +30 -15
- package/dist/features/tui/services/data-service.js.map +2 -2
- package/dist/features/tui/services/linear-task-reader.js +3 -1
- package/dist/features/tui/services/linear-task-reader.js.map +2 -2
- package/dist/features/tui/services/websocket-client.js +16 -3
- package/dist/features/tui/services/websocket-client.js.map +2 -2
- package/dist/features/tui/terminal-compat.js +33 -18
- package/dist/features/tui/terminal-compat.js.map +2 -2
- package/dist/features/web/client/stores/task-store.js.map +2 -2
- package/dist/features/web/server/index.js +31 -12
- package/dist/features/web/server/index.js.map +2 -2
- package/dist/integrations/claude-code/enhanced-pre-clear-hooks.js.map +2 -2
- package/dist/integrations/claude-code/lifecycle-hooks.js.map +2 -2
- package/dist/integrations/claude-code/post-task-hooks.js.map +2 -2
- package/dist/integrations/linear/auth.js +17 -6
- package/dist/integrations/linear/auth.js.map +2 -2
- package/dist/integrations/linear/auto-sync.js.map +2 -2
- package/dist/integrations/linear/client.js.map +2 -2
- package/dist/integrations/linear/config.js.map +2 -2
- package/dist/integrations/linear/migration.js.map +2 -2
- package/dist/integrations/linear/oauth-server.js +13 -2
- package/dist/integrations/linear/oauth-server.js.map +2 -2
- package/dist/integrations/linear/rest-client.js.map +2 -2
- package/dist/integrations/linear/sync-enhanced.js +202 -0
- package/dist/integrations/linear/sync-enhanced.js.map +7 -0
- package/dist/integrations/linear/sync-manager.js.map +2 -2
- package/dist/integrations/linear/sync-service.js +24 -14
- package/dist/integrations/linear/sync-service.js.map +2 -2
- package/dist/integrations/linear/sync.js +196 -3
- package/dist/integrations/linear/sync.js.map +2 -2
- package/dist/integrations/linear/unified-sync.js +560 -0
- package/dist/integrations/linear/unified-sync.js.map +7 -0
- package/dist/integrations/linear/webhook-handler.js +12 -1
- package/dist/integrations/linear/webhook-handler.js.map +2 -2
- package/dist/integrations/linear/webhook-server.js +29 -19
- package/dist/integrations/linear/webhook-server.js.map +2 -2
- package/dist/integrations/linear/webhook.js +12 -1
- package/dist/integrations/linear/webhook.js.map +2 -2
- package/dist/integrations/mcp/handlers/context-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/linear-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/skill-handlers.js +13 -2
- package/dist/integrations/mcp/handlers/skill-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/task-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/trace-handlers.js.map +2 -2
- package/dist/integrations/mcp/middleware/tool-scoring.js.map +2 -2
- package/dist/integrations/mcp/refactored-server.js +15 -4
- package/dist/integrations/mcp/refactored-server.js.map +2 -2
- package/dist/integrations/mcp/server.js +12 -1
- package/dist/integrations/mcp/server.js.map +2 -2
- package/dist/integrations/mcp/tool-definitions.js.map +2 -2
- package/dist/integrations/pg-aiguide/embedding-provider.js +13 -2
- package/dist/integrations/pg-aiguide/embedding-provider.js.map +2 -2
- package/dist/integrations/pg-aiguide/semantic-search.js.map +2 -2
- package/dist/mcp/stackmemory-mcp-server.js +1 -1
- package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
- package/dist/middleware/exponential-rate-limiter.js.map +2 -2
- package/dist/servers/production/auth-middleware.js +13 -2
- package/dist/servers/production/auth-middleware.js.map +2 -2
- package/dist/servers/railway/index.js +22 -11
- package/dist/servers/railway/index.js.map +2 -2
- package/dist/services/config-service.js +6 -7
- package/dist/services/config-service.js.map +2 -2
- package/dist/services/context-service.js +11 -12
- package/dist/services/context-service.js.map +2 -2
- package/dist/skills/claude-skills.js +108 -3
- package/dist/skills/claude-skills.js.map +2 -2
- package/dist/skills/dashboard-launcher.js.map +2 -2
- package/dist/skills/repo-ingestion-skill.js +561 -0
- package/dist/skills/repo-ingestion-skill.js.map +7 -0
- package/dist/utils/env.js +46 -0
- package/dist/utils/env.js.map +7 -0
- package/dist/utils/logger.js +1 -1
- package/dist/utils/logger.js.map +2 -2
- package/package.json +5 -1
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../src/cli/commands/linear-unified.ts"],
|
|
4
|
+
"sourcesContent": ["/**\n * Unified Linear CLI Commands\n * Consolidates all Linear operations into a single, coherent interface\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport {\n UnifiedLinearSync,\n UnifiedSyncConfig,\n SyncStats,\n DEFAULT_UNIFIED_CONFIG,\n} from '../../integrations/linear/unified-sync.js';\nimport { LinearAuthManager } from '../../integrations/linear/auth.js';\nimport { LinearClient } from '../../integrations/linear/client.js';\nimport { PebblesTaskStore } from '../../features/tasks/pebbles-task-store.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, readFileSync } from 'fs';\nimport { logger } from '../../core/monitoring/logger.js';\nimport Table from 'cli-table3';\nimport ora from 'ora';\n\nexport function registerUnifiedLinearCommands(parent: Command) {\n const linear = parent\n .command('linear')\n .description(\n 'Unified Linear integration with duplicate detection and task planning'\n );\n\n // Main sync command with all features\n linear\n .command('sync')\n .description(\n 'Intelligent sync with Linear (duplicate detection, bidirectional, task planning)'\n )\n .option(\n '-d, --direction <dir>',\n 'Sync direction: bidirectional, to_linear, from_linear',\n 'bidirectional'\n )\n .option('-t, --team <id>', 'Linear team ID')\n .option('--no-duplicates', 'Disable duplicate detection')\n .option(\n '--merge-strategy <strategy>',\n 'Duplicate handling: merge_content, skip, create_anyway',\n 'merge_content'\n )\n .option(\n '--conflict <resolution>',\n 'Conflict resolution: newest_wins, linear_wins, local_wins',\n 'newest_wins'\n )\n .option('--task-plan', 'Enable task planning integration')\n .option('--dry-run', 'Preview changes without syncing')\n .option('--daemon', 'Run as background daemon')\n .option('-i, --interval <minutes>', 'Auto-sync interval', '15')\n .option('--verbose', 'Show detailed sync progress')\n .action(async (options) => {\n const spinner = ora('Initializing Linear sync...').start();\n\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n spinner.fail('StackMemory not initialized');\n console.log(chalk.yellow('Run \"stackmemory init\" first'));\n return;\n }\n\n // Initialize components\n const db = new Database(dbPath);\n const taskStore = new PebblesTaskStore(projectRoot, db);\n const authManager = new LinearAuthManager(projectRoot);\n\n // Build config from options\n const config: Partial<UnifiedSyncConfig> = {\n direction: options.direction,\n defaultTeamId: options.team,\n duplicateDetection: options.duplicates !== false,\n mergeStrategy: options.mergeStrategy,\n conflictResolution: options.conflict,\n taskPlanningEnabled: options.taskPlan || false,\n };\n\n // Create unified sync instance\n const unifiedSync = new UnifiedLinearSync(\n taskStore,\n authManager,\n projectRoot,\n config\n );\n\n // Listen to events for progress\n if (options.verbose) {\n unifiedSync.on('sync:started', ({ config }) => {\n spinner.text = `Syncing (${config.direction})...`;\n });\n }\n\n // Initialize\n spinner.text = 'Authenticating with Linear...';\n await unifiedSync.initialize();\n\n if (options.dryRun) {\n spinner.info('Dry run mode - no changes will be made');\n // TODO: Implement dry run preview\n return;\n }\n\n if (options.daemon) {\n // Daemon mode\n spinner.succeed('Starting sync daemon');\n console.log(chalk.cyan(`Sync interval: ${options.interval} minutes`));\n console.log(chalk.gray('Press Ctrl+C to stop\\n'));\n\n // Initial sync\n const stats = await unifiedSync.sync();\n displaySyncStats(stats);\n\n // Schedule periodic syncs\n const interval = setInterval(\n async () => {\n console.log(\n chalk.yellow(\n `\\n[${new Date().toLocaleTimeString()}] Running scheduled sync...`\n )\n );\n try {\n const stats = await unifiedSync.sync();\n displaySyncStats(stats);\n } catch (error: unknown) {\n console.error(\n chalk.red('Sync failed:'),\n (error as Error).message\n );\n }\n },\n parseInt(options.interval) * 60 * 1000\n );\n\n // Handle graceful shutdown\n process.on('SIGINT', () => {\n console.log(chalk.yellow('\\nStopping sync daemon...'));\n clearInterval(interval);\n db.close();\n process.exit(0);\n });\n\n // Keep process alive\n process.stdin.resume();\n } else {\n // Single sync\n spinner.text = 'Syncing tasks...';\n const stats = await unifiedSync.sync();\n\n spinner.succeed('Sync completed');\n displaySyncStats(stats);\n\n // Show task plan if enabled\n if (options.taskPlan) {\n displayTaskPlan(projectRoot);\n }\n\n db.close();\n }\n } catch (error: unknown) {\n spinner.fail('Sync failed');\n console.error(chalk.red('Error:'), (error as Error).message);\n if (options.verbose) {\n console.error(error);\n }\n process.exit(1);\n }\n });\n\n // Quick status command\n linear\n .command('status')\n .description('Show Linear connection status and sync statistics')\n .action(async () => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const hasAuth = authManager.isConfigured();\n\n if (!hasAuth && !process.env.LINEAR_API_KEY) {\n console.log(chalk.yellow('\u26A0 Not connected to Linear'));\n console.log('Run \"stackmemory linear auth\" to connect');\n return;\n }\n\n console.log(chalk.green('\u2713 Linear connection configured'));\n\n // Show last sync stats if available\n const statsFile = join(\n process.cwd(),\n '.stackmemory',\n 'sync-stats.json'\n );\n if (existsSync(statsFile)) {\n const stats = JSON.parse(readFileSync(statsFile, 'utf8'));\n console.log(chalk.cyan('\\nLast sync:'));\n console.log(` Time: ${new Date(stats.timestamp).toLocaleString()}`);\n console.log(` Duration: ${stats.duration}ms`);\n console.log(\n ` To Linear: ${stats.toLinear.created} created, ${stats.toLinear.updated} updated`\n );\n console.log(\n ` From Linear: ${stats.fromLinear.created} created, ${stats.fromLinear.updated} updated`\n );\n\n if (stats.toLinear.duplicatesMerged > 0) {\n console.log(\n chalk.green(\n ` Duplicates prevented: ${stats.toLinear.duplicatesMerged}`\n )\n );\n }\n }\n\n // Show task plan status\n const planFile = join(process.cwd(), '.stackmemory', 'task-plan.md');\n if (existsSync(planFile)) {\n console.log(chalk.cyan('\\n\u2713 Task planning enabled'));\n const reportFile = join(\n process.cwd(),\n '.stackmemory',\n 'task-report.md'\n );\n if (existsSync(reportFile)) {\n console.log(` Report: ${reportFile}`);\n }\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('Status check failed:'),\n (error as Error).message\n );\n }\n });\n\n // Task planning command\n linear\n .command('plan')\n .description('View and manage task planning')\n .option('--generate', 'Generate task plan from current tasks')\n .option('--report', 'Show task report')\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n\n if (options.generate) {\n console.log(chalk.yellow('Generating task plan...'));\n // Run sync with task planning enabled\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n const db = new Database(dbPath);\n const taskStore = new PebblesTaskStore(projectRoot, db);\n const authManager = new LinearAuthManager(projectRoot);\n\n const unifiedSync = new UnifiedLinearSync(\n taskStore,\n authManager,\n projectRoot,\n {\n taskPlanningEnabled: true,\n autoCreateTaskPlan: true,\n }\n );\n\n await unifiedSync.initialize();\n await unifiedSync.sync();\n\n console.log(chalk.green('\u2713 Task plan generated'));\n db.close();\n }\n\n displayTaskPlan(projectRoot, options.report);\n } catch (error: unknown) {\n console.error(\n chalk.red('Plan generation failed:'),\n (error as Error).message\n );\n }\n });\n\n // Duplicate detection command\n linear\n .command('duplicates')\n .description('Check for and manage duplicate Linear issues')\n .option('--check <title>', 'Check if a title would create a duplicate')\n .option('--merge', 'Merge detected duplicates')\n .option('--list', 'List potential duplicates')\n .action(async (options) => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const token = await authManager.getValidToken();\n\n if (!token) {\n console.log(chalk.red('Not authenticated with Linear'));\n return;\n }\n\n const client = new LinearClient({\n apiKey: token,\n useBearer: authManager.isOAuth(),\n });\n\n if (options.check) {\n // Import dynamically to avoid circular dependency\n const { LinearDuplicateDetector } =\n await import('../../integrations/linear/sync-enhanced.js');\n const detector = new LinearDuplicateDetector(client);\n\n console.log(\n chalk.yellow(`Checking for duplicates of: \"${options.check}\"`)\n );\n const result = await detector.checkForDuplicate(options.check);\n\n if (result.isDuplicate && result.existingIssue) {\n console.log(chalk.red('\u26A0 Duplicate detected!'));\n console.log(\n ` Issue: ${result.existingIssue.identifier} - ${result.existingIssue.title}`\n );\n console.log(\n ` Similarity: ${Math.round((result.similarity || 0) * 100)}%`\n );\n console.log(` URL: ${result.existingIssue.url}`);\n } else {\n console.log(chalk.green('\u2713 No duplicates found'));\n }\n } else if (options.list) {\n console.log(chalk.yellow('Scanning for potential duplicates...'));\n // TODO: Implement duplicate scanning\n console.log('Feature coming soon');\n } else {\n console.log('Specify --check, --merge, or --list');\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('Duplicate check failed:'),\n (error as Error).message\n );\n }\n });\n\n // Auth command (simplified)\n linear\n .command('auth')\n .description('Authenticate with Linear')\n .option('--api-key <key>', 'Use API key')\n .option('--oauth', 'Use OAuth flow')\n .action(async (options) => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n\n if (options.apiKey) {\n // Simple API key setup\n process.env.LINEAR_API_KEY = options.apiKey;\n console.log(chalk.green('\u2713 API key configured'));\n\n // Test connection\n const client = new LinearClient({ apiKey: options.apiKey });\n const user = await client.getViewer();\n console.log(chalk.cyan(`Connected as: ${user.name} (${user.email})`));\n } else {\n console.log(chalk.cyan('Linear Authentication'));\n console.log('\\nOption 1: API Key (Recommended for automation)');\n console.log(' 1. Go to: https://linear.app/settings/api');\n console.log(' 2. Create a personal API key');\n console.log(' 3. Run: stackmemory linear auth --api-key YOUR_KEY');\n console.log('\\nOption 2: OAuth (For user-facing apps)');\n console.log(' Configure OAuth app and use linear-oauth-server');\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('Authentication failed:'),\n (error as Error).message\n );\n }\n });\n}\n\n/**\n * Display sync statistics\n */\nfunction displaySyncStats(stats: SyncStats): void {\n const table = new Table({\n head: ['Direction', 'Created', 'Updated', 'Merged', 'Skipped'],\n style: { head: ['cyan'] },\n });\n\n table.push(\n [\n '\u2192 Linear',\n stats.toLinear.created,\n stats.toLinear.updated,\n stats.toLinear.duplicatesMerged,\n stats.toLinear.skipped,\n ],\n [\n '\u2190 Linear',\n stats.fromLinear.created,\n stats.fromLinear.updated,\n '-',\n stats.fromLinear.skipped,\n ]\n );\n\n console.log('\\n' + table.toString());\n\n if (stats.conflicts.length > 0) {\n console.log(chalk.yellow(`\\n\u26A0 Conflicts: ${stats.conflicts.length}`));\n stats.conflicts.slice(0, 5).forEach((c) => {\n console.log(` - ${c.reason}`);\n });\n }\n\n if (stats.errors.length > 0) {\n console.log(chalk.red(`\\n\u274C Errors: ${stats.errors.length}`));\n stats.errors.slice(0, 5).forEach((e: string) => {\n console.log(` - ${e.substring(0, 100)}`);\n });\n }\n\n console.log(chalk.gray(`\\nCompleted in ${stats.duration}ms`));\n}\n\n/**\n * Display task plan\n */\nfunction displayTaskPlan(projectRoot: string, showReport = false): void {\n const reportFile = join(projectRoot, '.stackmemory', 'task-report.md');\n const planFile = join(projectRoot, '.stackmemory', 'task-plan.json');\n\n if (showReport && existsSync(reportFile)) {\n const report = readFileSync(reportFile, 'utf8');\n console.log('\\n' + report);\n } else if (existsSync(planFile)) {\n const plan = JSON.parse(readFileSync(planFile, 'utf8'));\n\n console.log(chalk.cyan('\\n\uD83D\uDCCB Task Plan Overview'));\n console.log(\n `Last updated: ${new Date(plan.lastUpdated).toLocaleString()}\\n`\n );\n\n plan.phases.forEach((phase) => {\n console.log(chalk.yellow(`${phase.name} (${phase.tasks.length})`));\n console.log(chalk.gray(` ${phase.description}`));\n\n if (phase.tasks.length > 0) {\n phase.tasks.slice(0, 5).forEach((task) => {\n const status = task.linearId ? '\uD83D\uDD17' : ' ';\n console.log(` ${status} ${task.title}`);\n });\n\n if (phase.tasks.length > 5) {\n console.log(chalk.gray(` ...and ${phase.tasks.length - 5} more`));\n }\n }\n console.log();\n });\n } else {\n console.log(\n chalk.yellow('No task plan found. Run sync with --task-plan to generate.')\n );\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["config", "stats"]
|
|
7
|
+
}
|
|
@@ -20,6 +20,17 @@ import { join } from "path";
|
|
|
20
20
|
import { existsSync } from "fs";
|
|
21
21
|
import { logger } from "../../core/monitoring/logger.js";
|
|
22
22
|
import Table from "cli-table3";
|
|
23
|
+
function getEnv(key, defaultValue) {
|
|
24
|
+
const value = process.env[key];
|
|
25
|
+
if (value === void 0) {
|
|
26
|
+
if (defaultValue !== void 0) return defaultValue;
|
|
27
|
+
throw new Error(`Environment variable ${key} is required`);
|
|
28
|
+
}
|
|
29
|
+
return value;
|
|
30
|
+
}
|
|
31
|
+
function getOptionalEnv(key) {
|
|
32
|
+
return process.env[key];
|
|
33
|
+
}
|
|
23
34
|
function displaySyncResult(result) {
|
|
24
35
|
if (result.success) {
|
|
25
36
|
console.log(chalk.green("\u2713 Sync completed successfully"));
|
|
@@ -64,7 +75,7 @@ function registerLinearCommands(parent) {
|
|
|
64
75
|
"Filter by status (backlog, started, completed, etc.)"
|
|
65
76
|
).option("--my", "Show only tasks assigned to me").option("--cache", "Show cache stats only").option("--refresh", "Force refresh cache").option("--count", "Show count by status only").action(async (options) => {
|
|
66
77
|
try {
|
|
67
|
-
const apiKey = process.env
|
|
78
|
+
const apiKey = process.env["LINEAR_API_KEY"];
|
|
68
79
|
if (!apiKey) {
|
|
69
80
|
console.log(
|
|
70
81
|
chalk.yellow("\u26A0 Set LINEAR_API_KEY environment variable")
|
|
@@ -138,7 +149,7 @@ ${displayTasks.length} shown, ${tasks.length} total tasks`
|
|
|
138
149
|
linear.command("auth").description("Authenticate with Linear").option("--api-key <key>", "Use API key instead of OAuth").option("--no-browser", "Do not open browser automatically").action(async (options) => {
|
|
139
150
|
try {
|
|
140
151
|
if (options.apiKey) {
|
|
141
|
-
process.env
|
|
152
|
+
process.env["LINEAR_API_KEY"] = options.apiKey;
|
|
142
153
|
console.log(chalk.green("\u2713 Linear API key set"));
|
|
143
154
|
const client = new LinearClient({ apiKey: options.apiKey });
|
|
144
155
|
const user = await client.getViewer();
|
|
@@ -149,8 +160,8 @@ ${displayTasks.length} shown, ${tasks.length} total tasks`
|
|
|
149
160
|
}
|
|
150
161
|
} else {
|
|
151
162
|
const authManager = new LinearAuthManager(process.cwd());
|
|
152
|
-
const clientId = process.env
|
|
153
|
-
const clientSecret = process.env
|
|
163
|
+
const clientId = process.env["LINEAR_CLIENT_ID"];
|
|
164
|
+
const clientSecret = process.env["LINEAR_CLIENT_SECRET"];
|
|
154
165
|
if (!clientId || !clientSecret) {
|
|
155
166
|
console.log(chalk.yellow("\n\u26A0 Linear OAuth app not configured"));
|
|
156
167
|
console.log(chalk.cyan("\n\u{1F4DD} Setup Instructions:"));
|
|
@@ -305,7 +316,7 @@ ${displayTasks.length} shown, ${tasks.length} total tasks`
|
|
|
305
316
|
try {
|
|
306
317
|
const authManager = new LinearAuthManager(process.cwd());
|
|
307
318
|
const tokens = authManager.loadTokens();
|
|
308
|
-
const apiKey = process.env
|
|
319
|
+
const apiKey = process.env["LINEAR_API_KEY"];
|
|
309
320
|
if (!tokens && !apiKey) {
|
|
310
321
|
console.log(chalk.yellow("\u26A0 Not authenticated with Linear"));
|
|
311
322
|
console.log('Run "stackmemory linear auth" to connect');
|
|
@@ -345,7 +356,7 @@ ${displayTasks.length} shown, ${tasks.length} total tasks`
|
|
|
345
356
|
"Filter by status (backlog, started, completed, etc.)"
|
|
346
357
|
).option("--my", "Show only tasks assigned to me").option("--cache", "Show cache stats").option("--refresh", "Force refresh cache").action(async (options) => {
|
|
347
358
|
try {
|
|
348
|
-
const apiKey = process.env
|
|
359
|
+
const apiKey = process.env["LINEAR_API_KEY"];
|
|
349
360
|
if (!apiKey) {
|
|
350
361
|
console.log(
|
|
351
362
|
chalk.yellow("\u26A0 Set LINEAR_API_KEY environment variable")
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/linear.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Linear integration commands\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport {\n LinearAuthManager,\n LinearOAuthSetup,\n} from '../../integrations/linear/auth.js';\nimport { LinearOAuthServer } from '../../integrations/linear/oauth-server.js';\nimport {\n LinearSyncEngine,\n DEFAULT_SYNC_CONFIG,\n} from '../../integrations/linear/sync.js';\nimport {\n LinearSyncManager,\n DEFAULT_SYNC_MANAGER_CONFIG,\n} from '../../integrations/linear/sync-manager.js';\nimport { LinearConfigManager } from '../../integrations/linear/config.js';\nimport { PebblesTaskStore } from '../../features/tasks/pebbles-task-store.js';\nimport { LinearClient } from '../../integrations/linear/client.js';\nimport { LinearRestClient } from '../../integrations/linear/rest-client.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\nimport { logger } from '../../core/monitoring/logger.js';\nimport Table from 'cli-table3';\nimport { SyncResult } from '../../integrations/linear/sync.js';\n\n/**\n * Display sync result in a formatted way\n */\nfunction displaySyncResult(result: SyncResult) {\n if (result.success) {\n console.log(chalk.green('\u2713 Sync completed successfully'));\n } else {\n console.log(chalk.yellow('\u26A0 Sync completed with issues'));\n }\n\n if (\n result.synced.toLinear > 0 ||\n result.synced.fromLinear > 0 ||\n result.synced.updated > 0\n ) {\n console.log(chalk.cyan(' \uD83D\uDCCA Summary:'));\n if (result.synced.toLinear > 0) {\n console.log(` \u2192 Linear: ${result.synced.toLinear} tasks`);\n }\n if (result.synced.fromLinear > 0) {\n console.log(` \u2190 Linear: ${result.synced.fromLinear} tasks`);\n }\n if (result.synced.updated > 0) {\n console.log(` \u2194 Updated: ${result.synced.updated} tasks`);\n }\n }\n\n if (result.conflicts.length > 0) {\n console.log(chalk.yellow(` \u26A0 Conflicts: ${result.conflicts.length}`));\n result.conflicts.slice(0, 3).forEach((conflict) => {\n console.log(` - ${conflict.reason}`);\n });\n if (result.conflicts.length > 3) {\n console.log(` ... and ${result.conflicts.length - 3} more`);\n }\n }\n\n if (result.errors.length > 0) {\n console.log(chalk.red(` \u274C Errors: ${result.errors.length}`));\n result.errors.slice(0, 3).forEach((error) => {\n console.log(` - ${error.substring(0, 80)}`);\n });\n if (result.errors.length > 3) {\n console.log(` ... and ${result.errors.length - 3} more`);\n }\n }\n}\n\nexport function registerLinearCommands(parent: Command) {\n const linear = parent\n .command('linear')\n .description('Linear API integration commands');\n\n // Quick tasks command using memory cache\n linear\n .command('list')\n .alias('ls')\n .description('List Linear tasks (memory-cached)')\n .option('--limit <n>', 'Number of tasks to show', '20')\n .option(\n '--status <status>',\n 'Filter by status (backlog, started, completed, etc.)'\n )\n .option('--my', 'Show only tasks assigned to me')\n .option('--cache', 'Show cache stats only')\n .option('--refresh', 'Force refresh cache')\n .option('--count', 'Show count by status only')\n .action(async (options) => {\n try {\n const apiKey = process.env.LINEAR_API_KEY;\n if (!apiKey) {\n console.log(\n chalk.yellow('\u26A0 Set LINEAR_API_KEY environment variable')\n );\n return;\n }\n\n const restClient = new LinearRestClient(apiKey);\n\n // Show cache stats if requested\n if (options.cache) {\n const stats = restClient.getCacheStats();\n console.log(chalk.cyan('\uD83D\uDCCA Cache Stats:'));\n console.log(` Size: ${stats.size} tasks`);\n console.log(` Age: ${Math.round(stats.age / 1000)}s`);\n console.log(` Fresh: ${stats.fresh ? 'yes' : 'no'}`);\n console.log(\n ` Last sync: ${new Date(stats.lastSync).toLocaleString()}`\n );\n return;\n }\n\n // Show counts only\n if (options.count) {\n const counts = await restClient.getTaskCounts();\n console.log(chalk.cyan('\uD83D\uDCCA Task Counts:'));\n Object.entries(counts)\n .sort(([, a], [, b]) => b - a)\n .forEach(([status, count]) => {\n console.log(` ${status}: ${count}`);\n });\n return;\n }\n\n let tasks;\n if (options.my) {\n tasks = await restClient.getMyTasks();\n } else if (options.status) {\n tasks = await restClient.getTasksByStatus(options.status);\n } else {\n tasks = await restClient.getAllTasks(options.refresh);\n }\n\n if (!tasks || tasks.length === 0) {\n console.log(chalk.gray('No tasks found'));\n return;\n }\n\n // Limit results\n const limit = parseInt(options.limit);\n const displayTasks = tasks.slice(0, limit);\n\n console.log(\n chalk.cyan(\n `\\n\uD83D\uDCCB Linear Tasks (${displayTasks.length}/${tasks.length}):`\n )\n );\n\n displayTasks.forEach((task) => {\n const priority = task.priority ? `P${task.priority}` : '';\n const assignee = task.assignee ? ` @${task.assignee.name}` : '';\n const statusColor =\n task.state.type === 'completed'\n ? chalk.green\n : task.state.type === 'started'\n ? chalk.yellow\n : chalk.gray;\n\n console.log(`${chalk.blue(task.identifier)} ${task.title}`);\n console.log(\n chalk.gray(\n ` ${statusColor(task.state.name)} ${priority}${assignee}`\n )\n );\n });\n\n console.log(\n chalk.gray(\n `\\n${displayTasks.length} shown, ${tasks.length} total tasks`\n )\n );\n } catch (error) {\n console.error(\n chalk.red('Failed to list tasks:'),\n (error as Error).message\n );\n }\n });\n\n // Auth command\n linear\n .command('auth')\n .description('Authenticate with Linear')\n .option('--api-key <key>', 'Use API key instead of OAuth')\n .option('--no-browser', 'Do not open browser automatically')\n .action(async (options) => {\n try {\n if (options.apiKey) {\n // Set API key as environment variable\n process.env.LINEAR_API_KEY = options.apiKey;\n console.log(chalk.green('\u2713 Linear API key set'));\n\n // Test the connection\n const client = new LinearClient({ apiKey: options.apiKey });\n const user = await client.getViewer();\n\n if (user) {\n console.log(\n chalk.cyan(`Connected as: ${user.name} (${user.email})`)\n );\n }\n } else {\n // OAuth flow with callback server\n const authManager = new LinearAuthManager(process.cwd());\n\n // Check if client ID and secret are configured\n const clientId = process.env.LINEAR_CLIENT_ID;\n const clientSecret = process.env.LINEAR_CLIENT_SECRET;\n\n if (!clientId || !clientSecret) {\n console.log(chalk.yellow('\\n\u26A0 Linear OAuth app not configured'));\n console.log(chalk.cyan('\\n\uD83D\uDCDD Setup Instructions:'));\n console.log(\n ' 1. Create a Linear OAuth app at: https://linear.app/settings/api'\n );\n console.log(\n ' 2. Set redirect URI to: http://localhost:3456/auth/linear/callback'\n );\n console.log(' 3. Copy your Client ID and Client Secret');\n console.log(' 4. Set environment variables:');\n console.log(\n chalk.gray(' export LINEAR_CLIENT_ID=\"your_client_id\"')\n );\n console.log(\n chalk.gray(\n ' export LINEAR_CLIENT_SECRET=\"your_client_secret\"'\n )\n );\n console.log(' 5. Run this command again');\n return;\n }\n\n // Save OAuth config\n authManager.saveConfig({\n clientId,\n clientSecret,\n redirectUri: 'http://localhost:3456/auth/linear/callback',\n scopes: ['read', 'write', 'admin'],\n });\n\n // Start OAuth server\n const oauthServer = new LinearOAuthServer(process.cwd());\n const { url } = await oauthServer.start();\n\n // Open browser if not disabled\n if (options.browser !== false) {\n const open = (await import('open')).default;\n await open(url);\n console.log(\n chalk.green('\\n\u2713 Browser opened with authorization page')\n );\n } else {\n console.log(chalk.cyan('\\n\uD83D\uDD17 Open this URL in your browser:'));\n console.log(chalk.underline(url));\n }\n\n console.log(chalk.gray('\\nWaiting for authorization...'));\n console.log(\n chalk.gray(\n 'The server will automatically shut down after authorization.'\n )\n );\n }\n } catch (error) {\n console.error(\n chalk.red('Authentication failed:'),\n (error as Error).message\n );\n process.exit(1);\n }\n });\n\n // Sync command\n linear\n .command('sync')\n .description('Sync tasks with Linear')\n .option(\n '-d, --direction <dir>',\n 'Sync direction: bidirectional, to_linear, from_linear',\n 'bidirectional'\n )\n .option('-t, --team <id>', 'Default Linear team ID')\n .option('--dry-run', 'Preview sync without making changes')\n .option('--daemon', 'Run in daemon mode with periodic sync')\n .option(\n '-i, --interval <minutes>',\n 'Sync interval in minutes (default: 15)'\n )\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(chalk.red('\u274C StackMemory not initialized'));\n return;\n }\n\n const db = new Database(dbPath);\n const taskStore = new PebblesTaskStore(projectRoot, db);\n const authManager = new LinearAuthManager(projectRoot);\n\n const config = {\n ...DEFAULT_SYNC_CONFIG,\n direction: options.direction,\n defaultTeamId: options.team,\n enabled: true,\n };\n\n if (options.daemon) {\n // Run in daemon mode with periodic sync\n const managerConfig = {\n ...DEFAULT_SYNC_MANAGER_CONFIG,\n ...config,\n autoSyncInterval: parseInt(options.interval) || 15,\n };\n\n const syncManager = new LinearSyncManager(\n taskStore,\n authManager,\n managerConfig,\n projectRoot\n );\n\n console.log(chalk.green('\uD83D\uDE80 Starting Linear sync daemon'));\n console.log(\n chalk.cyan(\n ` Sync interval: ${managerConfig.autoSyncInterval} minutes`\n )\n );\n console.log(chalk.cyan(` Direction: ${managerConfig.direction}`));\n console.log(chalk.gray(' Press Ctrl+C to stop\\n'));\n\n // Initial sync\n const initialResult = await syncManager.syncOnStart();\n if (initialResult) {\n displaySyncResult(initialResult);\n }\n\n // Listen for sync events\n syncManager.on('sync:started', ({ trigger }) => {\n console.log(\n chalk.yellow(\n `\\n\uD83D\uDD04 ${new Date().toLocaleTimeString()} - Starting ${trigger} sync...`\n )\n );\n });\n\n syncManager.on('sync:completed', ({ result }) => {\n displaySyncResult(result);\n });\n\n syncManager.on('sync:failed', ({ result }) => {\n console.log(chalk.red('\u274C Sync failed'));\n if (result.errors.length > 0) {\n result.errors.forEach((error: string) => {\n console.log(chalk.red(` - ${error}`));\n });\n }\n });\n\n // Handle graceful shutdown\n process.on('SIGINT', async () => {\n console.log(chalk.yellow('\\n\u23F9 Stopping sync daemon...'));\n await syncManager.syncOnEnd();\n syncManager.stop();\n db.close();\n process.exit(0);\n });\n\n // Keep process alive\n process.stdin.resume();\n } else {\n // Single sync\n const syncEngine = new LinearSyncEngine(\n taskStore,\n authManager,\n config\n );\n\n console.log(chalk.yellow('\uD83D\uDD04 Syncing with Linear...'));\n\n if (options.dryRun) {\n console.log(chalk.gray('(Dry run - no changes will be made)'));\n }\n\n const result = await syncEngine.sync();\n displaySyncResult(result);\n db.close();\n }\n } catch (error) {\n logger.error('Sync failed', error as Error);\n console.error(chalk.red('Sync failed:'), (error as Error).message);\n process.exit(1);\n }\n });\n\n // Status command\n linear\n .command('status')\n .description('Show Linear sync status')\n .action(async () => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const tokens = authManager.loadTokens();\n const apiKey = process.env.LINEAR_API_KEY;\n\n if (!tokens && !apiKey) {\n console.log(chalk.yellow('\u26A0 Not authenticated with Linear'));\n console.log('Run \"stackmemory linear auth\" to connect');\n return;\n }\n\n const client = apiKey\n ? new LinearClient({ apiKey })\n : new LinearClient({\n apiKey: tokens!.accessToken,\n useBearer: true,\n onUnauthorized: async () => {\n const refreshed = await authManager.refreshAccessToken();\n return refreshed.accessToken;\n },\n });\n\n const user = await client.getViewer();\n\n if (user) {\n console.log(chalk.green('\u2713 Connected to Linear'));\n console.log(chalk.cyan(` User: ${user.name} (${user.email})`));\n\n // Show teams\n const teams = await client.getTeams();\n if (teams && teams.length > 0) {\n console.log(chalk.cyan('\\n\uD83D\uDCCB Teams:'));\n teams.forEach((team) => {\n console.log(` - ${team.name} (${team.key})`);\n });\n }\n } else {\n console.log(chalk.red('\u274C Could not connect to Linear'));\n }\n } catch (error) {\n console.error(\n chalk.red('Status check failed:'),\n (error as Error).message\n );\n }\n });\n\n // List tasks command\n linear\n .command('tasks')\n .description('List Linear tasks')\n .option('--limit <n>', 'Number of tasks to show', '50')\n .option(\n '--status <status>',\n 'Filter by status (backlog, started, completed, etc.)'\n )\n .option('--my', 'Show only tasks assigned to me')\n .option('--cache', 'Show cache stats')\n .option('--refresh', 'Force refresh cache')\n .action(async (options) => {\n try {\n const apiKey = process.env.LINEAR_API_KEY;\n if (!apiKey) {\n console.log(\n chalk.yellow('\u26A0 Set LINEAR_API_KEY environment variable')\n );\n return;\n }\n\n const restClient = new LinearRestClient(apiKey);\n\n // Show cache stats if requested\n if (options.cache) {\n const stats = restClient.getCacheStats();\n console.log(chalk.cyan('\uD83D\uDCCA Cache Stats:'));\n console.log(` Size: ${stats.size} tasks`);\n console.log(` Age: ${Math.round(stats.age / 1000)}s`);\n console.log(` Fresh: ${stats.fresh ? 'yes' : 'no'}`);\n console.log(\n ` Last sync: ${new Date(stats.lastSync).toLocaleString()}`\n );\n return;\n }\n\n let tasks;\n if (options.my) {\n tasks = await restClient.getMyTasks();\n } else if (options.status) {\n tasks = await restClient.getTasksByStatus(options.status);\n } else {\n tasks = await restClient.getAllTasks(options.refresh);\n }\n\n if (!tasks || tasks.length === 0) {\n console.log(chalk.gray('No tasks found'));\n return;\n }\n\n // Limit results\n const limit = parseInt(options.limit);\n const displayTasks = tasks.slice(0, limit);\n\n const table = new Table({\n head: ['ID', 'Title', 'State', 'Priority', 'Assignee'],\n style: { head: ['cyan'] },\n });\n\n displayTasks.forEach((task) => {\n table.push([\n task.identifier,\n task.title.substring(0, 40) + (task.title.length > 40 ? '...' : ''),\n task.state?.name || '-',\n task.priority ? `P${task.priority}` : '-',\n task.assignee?.name || '-',\n ]);\n });\n\n console.log(table.toString());\n\n // Show counts by status\n const counts = await restClient.getTaskCounts();\n console.log(chalk.cyan('\\n\uD83D\uDCCA Task Summary:'));\n Object.entries(counts).forEach(([status, count]) => {\n console.log(` ${status}: ${count}`);\n });\n\n console.log(\n chalk.gray(\n `\\nShowing ${displayTasks.length} of ${tasks.length} total tasks`\n )\n );\n\n const cacheStats = restClient.getCacheStats();\n console.log(\n chalk.gray(\n `Cache: ${cacheStats.size} tasks, age: ${Math.round(cacheStats.age / 1000)}s`\n )\n );\n } catch (error) {\n console.error(\n chalk.red('Failed to list tasks:'),\n (error as Error).message\n );\n }\n });\n\n // Update command - update Linear task status\n linear\n .command('update <issueId>')\n .description('Update Linear task status')\n .option(\n '-s, --status <status>',\n 'New status (todo, in-progress, done, canceled)'\n )\n .option('-t, --title <title>', 'Update task title')\n .option('-d, --description <desc>', 'Update task description')\n .option(\n '-p, --priority <priority>',\n 'Set priority (1=urgent, 2=high, 3=medium, 4=low)'\n )\n .action(async (issueId, options) => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const tokens = authManager.loadTokens();\n\n if (!tokens) {\n console.error(\n chalk.red('Not authenticated. Run: stackmemory linear auth')\n );\n return;\n }\n\n const client = new LinearClient({\n apiKey: tokens.accessToken,\n });\n\n // Find the issue first\n let issue = await client.getIssue(issueId);\n if (!issue) {\n // Try finding by identifier\n issue = await client.findIssueByIdentifier(issueId);\n }\n\n if (!issue) {\n console.error(chalk.red(`Issue ${issueId} not found`));\n return;\n }\n\n const updates: any = {};\n\n // Handle status update\n if (options.status) {\n const team = await client.getTeam();\n const states = await client.getWorkflowStates(team.id);\n\n const statusMap: Record<string, string> = {\n todo: 'unstarted',\n 'in-progress': 'started',\n done: 'completed',\n canceled: 'cancelled',\n };\n\n const targetType =\n statusMap[options.status.toLowerCase()] || options.status;\n const targetState = states.find((s) => s.type === targetType);\n\n if (!targetState) {\n console.error(chalk.red(`Invalid status: ${options.status}`));\n console.log(chalk.gray('Available states:'));\n states.forEach((s) =>\n console.log(chalk.gray(` - ${s.name} (${s.type})`))\n );\n return;\n }\n\n updates.stateId = targetState.id;\n }\n\n if (options.title) updates.title = options.title;\n if (options.description) updates.description = options.description;\n if (options.priority) updates.priority = parseInt(options.priority);\n\n // Perform update\n const updatedIssue = await client.updateIssue(issue.id, updates);\n\n console.log(\n chalk.green(\n `\u2713 Updated ${updatedIssue.identifier}: ${updatedIssue.title}`\n )\n );\n if (options.status) {\n console.log(chalk.cyan(` Status: ${updatedIssue.state.name}`));\n }\n console.log(chalk.gray(` ${updatedIssue.url}`));\n } catch (error) {\n console.error(\n chalk.red('Failed to update task:'),\n (error as Error).message\n );\n }\n });\n\n // Config command\n linear\n .command('config')\n .description('Configure Linear sync settings')\n .option('--team <id>', 'Set default team ID')\n .option('--interval <minutes>', 'Auto-sync interval in minutes')\n .option('--direction <dir>', 'Sync direction')\n .option('--conflict <strategy>', 'Conflict resolution strategy')\n .action(async (options) => {\n try {\n const configManager = new LinearConfigManager(process.cwd());\n const config =\n configManager.loadConfig() || configManager.getDefaultConfig();\n\n let updated = false;\n\n if (options.team) {\n // Team ID would need to be stored separately or in a different config\n logger.info('Team ID configuration not yet implemented', {\n teamId: options.team,\n });\n }\n\n if (options.interval) {\n config.interval = parseInt(options.interval);\n updated = true;\n }\n\n if (options.direction) {\n config.direction = options.direction;\n updated = true;\n }\n\n if (options.conflict) {\n config.conflictResolution = options.conflict;\n updated = true;\n }\n\n if (updated) {\n configManager.saveConfig(config);\n console.log(chalk.green('\u2713 Configuration updated'));\n }\n\n // Display current config\n console.log(chalk.cyan('\\n\uD83D\uDCCB Current Configuration:'));\n console.log(` Enabled: ${config.enabled ? 'yes' : 'no'}`);\n console.log(` Interval: ${config.interval} minutes`);\n console.log(` Direction: ${config.direction}`);\n console.log(` Conflicts: ${config.conflictResolution}`);\n } catch (error) {\n console.error(chalk.red('Config failed:'), (error as Error).message);\n }\n });\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["/**\n * Linear integration commands\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport {\n LinearAuthManager,\n LinearOAuthSetup,\n} from '../../integrations/linear/auth.js';\nimport { LinearOAuthServer } from '../../integrations/linear/oauth-server.js';\nimport {\n LinearSyncEngine,\n DEFAULT_SYNC_CONFIG,\n} from '../../integrations/linear/sync.js';\nimport {\n LinearSyncManager,\n DEFAULT_SYNC_MANAGER_CONFIG,\n} from '../../integrations/linear/sync-manager.js';\nimport { LinearConfigManager } from '../../integrations/linear/config.js';\nimport { PebblesTaskStore } from '../../features/tasks/pebbles-task-store.js';\nimport { LinearClient } from '../../integrations/linear/client.js';\nimport { LinearRestClient } from '../../integrations/linear/rest-client.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\nimport { logger } from '../../core/monitoring/logger.js';\nimport Table from 'cli-table3';\nimport { SyncResult } from '../../integrations/linear/sync.js';\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new Error(`Environment variable ${key} is required`);\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\n/**\n * Display sync result in a formatted way\n */\nfunction displaySyncResult(result: SyncResult) {\n if (result.success) {\n console.log(chalk.green('\u2713 Sync completed successfully'));\n } else {\n console.log(chalk.yellow('\u26A0 Sync completed with issues'));\n }\n\n if (\n result.synced.toLinear > 0 ||\n result.synced.fromLinear > 0 ||\n result.synced.updated > 0\n ) {\n console.log(chalk.cyan(' \uD83D\uDCCA Summary:'));\n if (result.synced.toLinear > 0) {\n console.log(` \u2192 Linear: ${result.synced.toLinear} tasks`);\n }\n if (result.synced.fromLinear > 0) {\n console.log(` \u2190 Linear: ${result.synced.fromLinear} tasks`);\n }\n if (result.synced.updated > 0) {\n console.log(` \u2194 Updated: ${result.synced.updated} tasks`);\n }\n }\n\n if (result.conflicts.length > 0) {\n console.log(chalk.yellow(` \u26A0 Conflicts: ${result.conflicts.length}`));\n result.conflicts.slice(0, 3).forEach((conflict) => {\n console.log(` - ${conflict.reason}`);\n });\n if (result.conflicts.length > 3) {\n console.log(` ... and ${result.conflicts.length - 3} more`);\n }\n }\n\n if (result.errors.length > 0) {\n console.log(chalk.red(` \u274C Errors: ${result.errors.length}`));\n result.errors.slice(0, 3).forEach((error) => {\n console.log(` - ${error.substring(0, 80)}`);\n });\n if (result.errors.length > 3) {\n console.log(` ... and ${result.errors.length - 3} more`);\n }\n }\n}\n\nexport function registerLinearCommands(parent: Command) {\n const linear = parent\n .command('linear')\n .description('Linear API integration commands');\n\n // Quick tasks command using memory cache\n linear\n .command('list')\n .alias('ls')\n .description('List Linear tasks (memory-cached)')\n .option('--limit <n>', 'Number of tasks to show', '20')\n .option(\n '--status <status>',\n 'Filter by status (backlog, started, completed, etc.)'\n )\n .option('--my', 'Show only tasks assigned to me')\n .option('--cache', 'Show cache stats only')\n .option('--refresh', 'Force refresh cache')\n .option('--count', 'Show count by status only')\n .action(async (options) => {\n try {\n const apiKey = process.env['LINEAR_API_KEY'];\n if (!apiKey) {\n console.log(\n chalk.yellow('\u26A0 Set LINEAR_API_KEY environment variable')\n );\n return;\n }\n\n const restClient = new LinearRestClient(apiKey);\n\n // Show cache stats if requested\n if (options.cache) {\n const stats = restClient.getCacheStats();\n console.log(chalk.cyan('\uD83D\uDCCA Cache Stats:'));\n console.log(` Size: ${stats.size} tasks`);\n console.log(` Age: ${Math.round(stats.age / 1000)}s`);\n console.log(` Fresh: ${stats.fresh ? 'yes' : 'no'}`);\n console.log(\n ` Last sync: ${new Date(stats.lastSync).toLocaleString()}`\n );\n return;\n }\n\n // Show counts only\n if (options.count) {\n const counts = await restClient.getTaskCounts();\n console.log(chalk.cyan('\uD83D\uDCCA Task Counts:'));\n Object.entries(counts)\n .sort(([, a], [, b]) => b - a)\n .forEach(([status, count]) => {\n console.log(` ${status}: ${count}`);\n });\n return;\n }\n\n let tasks;\n if (options.my) {\n tasks = await restClient.getMyTasks();\n } else if (options.status) {\n tasks = await restClient.getTasksByStatus(options.status);\n } else {\n tasks = await restClient.getAllTasks(options.refresh);\n }\n\n if (!tasks || tasks.length === 0) {\n console.log(chalk.gray('No tasks found'));\n return;\n }\n\n // Limit results\n const limit = parseInt(options.limit);\n const displayTasks = tasks.slice(0, limit);\n\n console.log(\n chalk.cyan(\n `\\n\uD83D\uDCCB Linear Tasks (${displayTasks.length}/${tasks.length}):`\n )\n );\n\n displayTasks.forEach((task) => {\n const priority = task.priority ? `P${task.priority}` : '';\n const assignee = task.assignee ? ` @${task.assignee.name}` : '';\n const statusColor =\n task.state.type === 'completed'\n ? chalk.green\n : task.state.type === 'started'\n ? chalk.yellow\n : chalk.gray;\n\n console.log(`${chalk.blue(task.identifier)} ${task.title}`);\n console.log(\n chalk.gray(\n ` ${statusColor(task.state.name)} ${priority}${assignee}`\n )\n );\n });\n\n console.log(\n chalk.gray(\n `\\n${displayTasks.length} shown, ${tasks.length} total tasks`\n )\n );\n } catch (error: unknown) {\n console.error(\n chalk.red('Failed to list tasks:'),\n (error as Error).message\n );\n }\n });\n\n // Auth command\n linear\n .command('auth')\n .description('Authenticate with Linear')\n .option('--api-key <key>', 'Use API key instead of OAuth')\n .option('--no-browser', 'Do not open browser automatically')\n .action(async (options) => {\n try {\n if (options.apiKey) {\n // Set API key as environment variable\n process.env['LINEAR_API_KEY'] = options.apiKey;\n console.log(chalk.green('\u2713 Linear API key set'));\n\n // Test the connection\n const client = new LinearClient({ apiKey: options.apiKey });\n const user = await client.getViewer();\n\n if (user) {\n console.log(\n chalk.cyan(`Connected as: ${user.name} (${user.email})`)\n );\n }\n } else {\n // OAuth flow with callback server\n const authManager = new LinearAuthManager(process.cwd());\n\n // Check if client ID and secret are configured\n const clientId = process.env['LINEAR_CLIENT_ID'];\n const clientSecret = process.env['LINEAR_CLIENT_SECRET'];\n\n if (!clientId || !clientSecret) {\n console.log(chalk.yellow('\\n\u26A0 Linear OAuth app not configured'));\n console.log(chalk.cyan('\\n\uD83D\uDCDD Setup Instructions:'));\n console.log(\n ' 1. Create a Linear OAuth app at: https://linear.app/settings/api'\n );\n console.log(\n ' 2. Set redirect URI to: http://localhost:3456/auth/linear/callback'\n );\n console.log(' 3. Copy your Client ID and Client Secret');\n console.log(' 4. Set environment variables:');\n console.log(\n chalk.gray(' export LINEAR_CLIENT_ID=\"your_client_id\"')\n );\n console.log(\n chalk.gray(\n ' export LINEAR_CLIENT_SECRET=\"your_client_secret\"'\n )\n );\n console.log(' 5. Run this command again');\n return;\n }\n\n // Save OAuth config\n authManager.saveConfig({\n clientId,\n clientSecret,\n redirectUri: 'http://localhost:3456/auth/linear/callback',\n scopes: ['read', 'write', 'admin'],\n });\n\n // Start OAuth server\n const oauthServer = new LinearOAuthServer(process.cwd());\n const { url } = await oauthServer.start();\n\n // Open browser if not disabled\n if (options.browser !== false) {\n const open = (await import('open')).default;\n await open(url);\n console.log(\n chalk.green('\\n\u2713 Browser opened with authorization page')\n );\n } else {\n console.log(chalk.cyan('\\n\uD83D\uDD17 Open this URL in your browser:'));\n console.log(chalk.underline(url));\n }\n\n console.log(chalk.gray('\\nWaiting for authorization...'));\n console.log(\n chalk.gray(\n 'The server will automatically shut down after authorization.'\n )\n );\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('Authentication failed:'),\n (error as Error).message\n );\n process.exit(1);\n }\n });\n\n // Sync command\n linear\n .command('sync')\n .description('Sync tasks with Linear')\n .option(\n '-d, --direction <dir>',\n 'Sync direction: bidirectional, to_linear, from_linear',\n 'bidirectional'\n )\n .option('-t, --team <id>', 'Default Linear team ID')\n .option('--dry-run', 'Preview sync without making changes')\n .option('--daemon', 'Run in daemon mode with periodic sync')\n .option(\n '-i, --interval <minutes>',\n 'Sync interval in minutes (default: 15)'\n )\n .action(async (options) => {\n try {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(chalk.red('\u274C StackMemory not initialized'));\n return;\n }\n\n const db = new Database(dbPath);\n const taskStore = new PebblesTaskStore(projectRoot, db);\n const authManager = new LinearAuthManager(projectRoot);\n\n const config = {\n ...DEFAULT_SYNC_CONFIG,\n direction: options.direction,\n defaultTeamId: options.team,\n enabled: true,\n };\n\n if (options.daemon) {\n // Run in daemon mode with periodic sync\n const managerConfig = {\n ...DEFAULT_SYNC_MANAGER_CONFIG,\n ...config,\n autoSyncInterval: parseInt(options.interval) || 15,\n };\n\n const syncManager = new LinearSyncManager(\n taskStore,\n authManager,\n managerConfig,\n projectRoot\n );\n\n console.log(chalk.green('\uD83D\uDE80 Starting Linear sync daemon'));\n console.log(\n chalk.cyan(\n ` Sync interval: ${managerConfig.autoSyncInterval} minutes`\n )\n );\n console.log(chalk.cyan(` Direction: ${managerConfig.direction}`));\n console.log(chalk.gray(' Press Ctrl+C to stop\\n'));\n\n // Initial sync\n const initialResult = await syncManager.syncOnStart();\n if (initialResult) {\n displaySyncResult(initialResult);\n }\n\n // Listen for sync events\n syncManager.on('sync:started', ({ trigger }) => {\n console.log(\n chalk.yellow(\n `\\n\uD83D\uDD04 ${new Date().toLocaleTimeString()} - Starting ${trigger} sync...`\n )\n );\n });\n\n syncManager.on('sync:completed', ({ result }) => {\n displaySyncResult(result);\n });\n\n syncManager.on('sync:failed', ({ result }) => {\n console.log(chalk.red('\u274C Sync failed'));\n if (result.errors.length > 0) {\n result.errors.forEach((error: string) => {\n console.log(chalk.red(` - ${error}`));\n });\n }\n });\n\n // Handle graceful shutdown\n process.on('SIGINT', async () => {\n console.log(chalk.yellow('\\n\u23F9 Stopping sync daemon...'));\n await syncManager.syncOnEnd();\n syncManager.stop();\n db.close();\n process.exit(0);\n });\n\n // Keep process alive\n process.stdin.resume();\n } else {\n // Single sync\n const syncEngine = new LinearSyncEngine(\n taskStore,\n authManager,\n config\n );\n\n console.log(chalk.yellow('\uD83D\uDD04 Syncing with Linear...'));\n\n if (options.dryRun) {\n console.log(chalk.gray('(Dry run - no changes will be made)'));\n }\n\n const result = await syncEngine.sync();\n displaySyncResult(result);\n db.close();\n }\n } catch (error: unknown) {\n logger.error('Sync failed', error as Error);\n console.error(chalk.red('Sync failed:'), (error as Error).message);\n process.exit(1);\n }\n });\n\n // Status command\n linear\n .command('status')\n .description('Show Linear sync status')\n .action(async () => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const tokens = authManager.loadTokens();\n const apiKey = process.env['LINEAR_API_KEY'];\n\n if (!tokens && !apiKey) {\n console.log(chalk.yellow('\u26A0 Not authenticated with Linear'));\n console.log('Run \"stackmemory linear auth\" to connect');\n return;\n }\n\n const client = apiKey\n ? new LinearClient({ apiKey })\n : new LinearClient({\n apiKey: tokens!.accessToken,\n useBearer: true,\n onUnauthorized: async () => {\n const refreshed = await authManager.refreshAccessToken();\n return refreshed.accessToken;\n },\n });\n\n const user = await client.getViewer();\n\n if (user) {\n console.log(chalk.green('\u2713 Connected to Linear'));\n console.log(chalk.cyan(` User: ${user.name} (${user.email})`));\n\n // Show teams\n const teams = await client.getTeams();\n if (teams && teams.length > 0) {\n console.log(chalk.cyan('\\n\uD83D\uDCCB Teams:'));\n teams.forEach((team) => {\n console.log(` - ${team.name} (${team.key})`);\n });\n }\n } else {\n console.log(chalk.red('\u274C Could not connect to Linear'));\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('Status check failed:'),\n (error as Error).message\n );\n }\n });\n\n // List tasks command\n linear\n .command('tasks')\n .description('List Linear tasks')\n .option('--limit <n>', 'Number of tasks to show', '50')\n .option(\n '--status <status>',\n 'Filter by status (backlog, started, completed, etc.)'\n )\n .option('--my', 'Show only tasks assigned to me')\n .option('--cache', 'Show cache stats')\n .option('--refresh', 'Force refresh cache')\n .action(async (options) => {\n try {\n const apiKey = process.env['LINEAR_API_KEY'];\n if (!apiKey) {\n console.log(\n chalk.yellow('\u26A0 Set LINEAR_API_KEY environment variable')\n );\n return;\n }\n\n const restClient = new LinearRestClient(apiKey);\n\n // Show cache stats if requested\n if (options.cache) {\n const stats = restClient.getCacheStats();\n console.log(chalk.cyan('\uD83D\uDCCA Cache Stats:'));\n console.log(` Size: ${stats.size} tasks`);\n console.log(` Age: ${Math.round(stats.age / 1000)}s`);\n console.log(` Fresh: ${stats.fresh ? 'yes' : 'no'}`);\n console.log(\n ` Last sync: ${new Date(stats.lastSync).toLocaleString()}`\n );\n return;\n }\n\n let tasks;\n if (options.my) {\n tasks = await restClient.getMyTasks();\n } else if (options.status) {\n tasks = await restClient.getTasksByStatus(options.status);\n } else {\n tasks = await restClient.getAllTasks(options.refresh);\n }\n\n if (!tasks || tasks.length === 0) {\n console.log(chalk.gray('No tasks found'));\n return;\n }\n\n // Limit results\n const limit = parseInt(options.limit);\n const displayTasks = tasks.slice(0, limit);\n\n const table = new Table({\n head: ['ID', 'Title', 'State', 'Priority', 'Assignee'],\n style: { head: ['cyan'] },\n });\n\n displayTasks.forEach((task) => {\n table.push([\n task.identifier,\n task.title.substring(0, 40) + (task.title.length > 40 ? '...' : ''),\n task.state?.name || '-',\n task.priority ? `P${task.priority}` : '-',\n task.assignee?.name || '-',\n ]);\n });\n\n console.log(table.toString());\n\n // Show counts by status\n const counts = await restClient.getTaskCounts();\n console.log(chalk.cyan('\\n\uD83D\uDCCA Task Summary:'));\n Object.entries(counts).forEach(([status, count]) => {\n console.log(` ${status}: ${count}`);\n });\n\n console.log(\n chalk.gray(\n `\\nShowing ${displayTasks.length} of ${tasks.length} total tasks`\n )\n );\n\n const cacheStats = restClient.getCacheStats();\n console.log(\n chalk.gray(\n `Cache: ${cacheStats.size} tasks, age: ${Math.round(cacheStats.age / 1000)}s`\n )\n );\n } catch (error: unknown) {\n console.error(\n chalk.red('Failed to list tasks:'),\n (error as Error).message\n );\n }\n });\n\n // Update command - update Linear task status\n linear\n .command('update <issueId>')\n .description('Update Linear task status')\n .option(\n '-s, --status <status>',\n 'New status (todo, in-progress, done, canceled)'\n )\n .option('-t, --title <title>', 'Update task title')\n .option('-d, --description <desc>', 'Update task description')\n .option(\n '-p, --priority <priority>',\n 'Set priority (1=urgent, 2=high, 3=medium, 4=low)'\n )\n .action(async (issueId, options) => {\n try {\n const authManager = new LinearAuthManager(process.cwd());\n const tokens = authManager.loadTokens();\n\n if (!tokens) {\n console.error(\n chalk.red('Not authenticated. Run: stackmemory linear auth')\n );\n return;\n }\n\n const client = new LinearClient({\n apiKey: tokens.accessToken,\n });\n\n // Find the issue first\n let issue = await client.getIssue(issueId);\n if (!issue) {\n // Try finding by identifier\n issue = await client.findIssueByIdentifier(issueId);\n }\n\n if (!issue) {\n console.error(chalk.red(`Issue ${issueId} not found`));\n return;\n }\n\n const updates: any = {};\n\n // Handle status update\n if (options.status) {\n const team = await client.getTeam();\n const states = await client.getWorkflowStates(team.id);\n\n const statusMap: Record<string, string> = {\n todo: 'unstarted',\n 'in-progress': 'started',\n done: 'completed',\n canceled: 'cancelled',\n };\n\n const targetType =\n statusMap[options.status.toLowerCase()] || options.status;\n const targetState = states.find((s) => s.type === targetType);\n\n if (!targetState) {\n console.error(chalk.red(`Invalid status: ${options.status}`));\n console.log(chalk.gray('Available states:'));\n states.forEach((s) =>\n console.log(chalk.gray(` - ${s.name} (${s.type})`))\n );\n return;\n }\n\n updates.stateId = targetState.id;\n }\n\n if (options.title) updates.title = options.title;\n if (options.description) updates.description = options.description;\n if (options.priority) updates.priority = parseInt(options.priority);\n\n // Perform update\n const updatedIssue = await client.updateIssue(issue.id, updates);\n\n console.log(\n chalk.green(\n `\u2713 Updated ${updatedIssue.identifier}: ${updatedIssue.title}`\n )\n );\n if (options.status) {\n console.log(chalk.cyan(` Status: ${updatedIssue.state.name}`));\n }\n console.log(chalk.gray(` ${updatedIssue.url}`));\n } catch (error: unknown) {\n console.error(\n chalk.red('Failed to update task:'),\n (error as Error).message\n );\n }\n });\n\n // Config command\n linear\n .command('config')\n .description('Configure Linear sync settings')\n .option('--team <id>', 'Set default team ID')\n .option('--interval <minutes>', 'Auto-sync interval in minutes')\n .option('--direction <dir>', 'Sync direction')\n .option('--conflict <strategy>', 'Conflict resolution strategy')\n .action(async (options) => {\n try {\n const configManager = new LinearConfigManager(process.cwd());\n const config =\n configManager.loadConfig() || configManager.getDefaultConfig();\n\n let updated = false;\n\n if (options.team) {\n // Team ID would need to be stored separately or in a different config\n logger.info('Team ID configuration not yet implemented', {\n teamId: options.team,\n });\n }\n\n if (options.interval) {\n config.interval = parseInt(options.interval);\n updated = true;\n }\n\n if (options.direction) {\n config.direction = options.direction;\n updated = true;\n }\n\n if (options.conflict) {\n config.conflictResolution = options.conflict;\n updated = true;\n }\n\n if (updated) {\n configManager.saveConfig(config);\n console.log(chalk.green('\u2713 Configuration updated'));\n }\n\n // Display current config\n console.log(chalk.cyan('\\n\uD83D\uDCCB Current Configuration:'));\n console.log(` Enabled: ${config.enabled ? 'yes' : 'no'}`);\n console.log(` Interval: ${config.interval} minutes`);\n console.log(` Direction: ${config.direction}`);\n console.log(` Conflicts: ${config.conflictResolution}`);\n } catch (error: unknown) {\n console.error(chalk.red('Config failed:'), (error as Error).message);\n }\n });\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/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) {\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(\n projectRoot: string,\n options: any,\n spinner: any\n) {\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) {\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
|
-
"mappings": "AAUA,SAAS,eAAe;AACxB,OAAO,WAAW;AAClB,OAAO,SAAS;AAChB,SAAS,sBAAsB;AAC/B,SAAS,oBAAoB;AAC7B,OAAO,cAAc;AAErB,YAAY,QAAQ;AACpB,YAAY,UAAU;AACtB,SAAS,aAAa;AAEf,SAAS,uBAAgC;AAC9C,QAAM,MAAM,IAAI,QAAQ,SAAS,EAC9B,YAAY,kDAAkD,EAC9D,OAAO,WAAW,yBAAyB,EAC3C,OAAO,UAAU,wBAAwB,EACzC,OAAO,YAAY,sBAAsB,EACzC,OAAO,YAAY,4BAA4B,EAC/C,OAAO,cAAc,2BAA2B,EAChD,OAAO,YAAY,oCAAoC,EACvD,OAAO,gBAAgB,iCAAiC,EACxD,OAAO,wBAAwB,6BAA6B,IAAI,EAChE,OAAO,oBAAoB,2BAA2B,GAAG,EACzD,OAAO,OAAO,YAAY;AACzB,UAAM,UAAU,IAAI;AAEpB,QAAI;AACF,YAAM,cAAc,MAAM,eAAe;AACzC,YAAM,SAAS,KAAK;AAAA,QAClB;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAGA,UAAI;AACF,cAAM,GAAG,OAAO,MAAM;AAAA,MACxB,QAAQ;AACN,gBAAQ,MAAM,MAAM,IAAI,oCAA+B,CAAC;AACxD,gBAAQ,IAAI,MAAM,OAAO,uBAAuB,CAAC;AACjD,gBAAQ,KAAK,CAAC;AAAA,MAChB;AAEA,UAAI,QAAQ,OAAO;AACjB,cAAM,aAAa,aAAa,SAAS,OAAO;AAAA,MAClD,WAAW,QAAQ,MAAM;AACvB,cAAM,YAAY,aAAa,OAAO;AAAA,MACxC,WAAW,QAAQ,QAAQ;AACzB,cAAM,WAAW,WAAW;AAAA,MAC9B,WAAW,QAAQ,QAAQ;AACzB,cAAM,WAAW,WAAW;AAAA,MAC9B,WAAW,QAAQ,UAAU;AAC3B,cAAM,eAAe,WAAW;AAAA,MAClC,WAAW,QAAQ,QAAQ;AACzB,cAAM,UAAU,aAAa,OAAO;AAAA,MACtC,WAAW,QAAQ,YAAY;AAC7B,cAAM,cAAc,aAAa,OAAO;AAAA,MAC1C,OAAO;AAEL,cAAM,WAAW,WAAW;AAAA,MAC9B;AAAA,IACF,SAAS,
|
|
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"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -29,11 +29,16 @@ function registerOnboardingCommand(program) {
|
|
|
29
29
|
try {
|
|
30
30
|
const config = await runOnboarding();
|
|
31
31
|
await applyConfiguration(config);
|
|
32
|
-
console.log(
|
|
32
|
+
console.log(
|
|
33
|
+
chalk.green("\n\u2705 StackMemory setup completed successfully!\n")
|
|
34
|
+
);
|
|
33
35
|
showNextSteps(config);
|
|
34
36
|
} catch (error) {
|
|
35
37
|
logger.error("Onboarding failed", error);
|
|
36
|
-
console.error(
|
|
38
|
+
console.error(
|
|
39
|
+
chalk.red("\n\u274C Setup failed:"),
|
|
40
|
+
error.message
|
|
41
|
+
);
|
|
37
42
|
process.exit(1);
|
|
38
43
|
}
|
|
39
44
|
});
|
|
@@ -156,11 +161,23 @@ async function runOnboarding() {
|
|
|
156
161
|
choices: [
|
|
157
162
|
{ name: "~/Dev", value: join(homedir(), "Dev"), checked: true },
|
|
158
163
|
{ name: "~/dev", value: join(homedir(), "dev"), checked: true },
|
|
159
|
-
{
|
|
160
|
-
|
|
164
|
+
{
|
|
165
|
+
name: "~/Projects",
|
|
166
|
+
value: join(homedir(), "Projects"),
|
|
167
|
+
checked: true
|
|
168
|
+
},
|
|
169
|
+
{
|
|
170
|
+
name: "~/projects",
|
|
171
|
+
value: join(homedir(), "projects"),
|
|
172
|
+
checked: true
|
|
173
|
+
},
|
|
161
174
|
{ name: "~/Work", value: join(homedir(), "Work"), checked: false },
|
|
162
175
|
{ name: "~/code", value: join(homedir(), "code"), checked: true },
|
|
163
|
-
{
|
|
176
|
+
{
|
|
177
|
+
name: "~/Documents/GitHub",
|
|
178
|
+
value: join(homedir(), "Documents/GitHub"),
|
|
179
|
+
checked: false
|
|
180
|
+
}
|
|
164
181
|
],
|
|
165
182
|
when: () => projectDetailAnswers.scanProjects
|
|
166
183
|
}
|
|
@@ -316,7 +333,9 @@ exec stackmemory "$@"
|
|
|
316
333
|
console.log(chalk.green(" \u2713 Created claude-sm command"));
|
|
317
334
|
}
|
|
318
335
|
} catch (error) {
|
|
319
|
-
console.log(
|
|
336
|
+
console.log(
|
|
337
|
+
chalk.yellow(" \u26A0 Could not create claude-sm symlink (may need sudo)")
|
|
338
|
+
);
|
|
320
339
|
}
|
|
321
340
|
}
|
|
322
341
|
function showNextSteps(config) {
|
|
@@ -326,9 +345,17 @@ function showNextSteps(config) {
|
|
|
326
345
|
console.log(chalk.gray(" stackmemory init\n"));
|
|
327
346
|
if (config.enableWorktrees) {
|
|
328
347
|
console.log("2. Create a new worktree:");
|
|
329
|
-
console.log(
|
|
348
|
+
console.log(
|
|
349
|
+
chalk.gray(
|
|
350
|
+
" git worktree add -b feature/new-feature ../project-feature"
|
|
351
|
+
)
|
|
352
|
+
);
|
|
330
353
|
console.log(chalk.gray(" cd ../project-feature"));
|
|
331
|
-
console.log(
|
|
354
|
+
console.log(
|
|
355
|
+
chalk.gray(
|
|
356
|
+
" stackmemory status # Isolated context for this worktree\n"
|
|
357
|
+
)
|
|
358
|
+
);
|
|
332
359
|
}
|
|
333
360
|
console.log("3. Use with Claude:");
|
|
334
361
|
console.log(chalk.gray(" claude-sm # Or use stackmemory directly\n"));
|