@stackmemoryai/stackmemory 0.5.31 → 0.5.33
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/cli/claude-sm.js +199 -16
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/commands/context.js +0 -11
- package/dist/cli/commands/context.js.map +2 -2
- package/dist/cli/commands/linear.js +1 -14
- package/dist/cli/commands/linear.js.map +2 -2
- package/dist/cli/commands/login.js +32 -10
- package/dist/cli/commands/login.js.map +2 -2
- package/dist/cli/commands/migrate.js +80 -22
- package/dist/cli/commands/migrate.js.map +2 -2
- package/dist/cli/commands/model.js +533 -0
- package/dist/cli/commands/model.js.map +7 -0
- package/dist/cli/commands/ralph.js +93 -28
- package/dist/cli/commands/ralph.js.map +2 -2
- package/dist/cli/commands/service.js +10 -3
- package/dist/cli/commands/service.js.map +2 -2
- package/dist/cli/commands/skills.js +60 -10
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/commands/sms-notify.js +342 -22
- package/dist/cli/commands/sms-notify.js.map +3 -3
- package/dist/cli/index.js +2 -0
- package/dist/cli/index.js.map +2 -2
- package/dist/core/context/dual-stack-manager.js +23 -7
- package/dist/core/context/dual-stack-manager.js.map +2 -2
- package/dist/core/context/frame-database.js +33 -5
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-digest.js +6 -1
- package/dist/core/context/frame-digest.js.map +2 -2
- package/dist/core/context/frame-manager.js +56 -9
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/context/permission-manager.js +0 -11
- package/dist/core/context/permission-manager.js.map +2 -2
- package/dist/core/context/recursive-context-manager.js +15 -9
- package/dist/core/context/recursive-context-manager.js.map +2 -2
- package/dist/core/context/shared-context-layer.js +0 -11
- package/dist/core/context/shared-context-layer.js.map +2 -2
- package/dist/core/context/validation.js +6 -1
- package/dist/core/context/validation.js.map +2 -2
- package/dist/core/models/fallback-monitor.js +229 -0
- package/dist/core/models/fallback-monitor.js.map +7 -0
- package/dist/core/models/model-router.js +331 -0
- package/dist/core/models/model-router.js.map +7 -0
- package/dist/hooks/claude-code-whatsapp-hook.js +197 -0
- package/dist/hooks/claude-code-whatsapp-hook.js.map +7 -0
- package/dist/hooks/linear-task-picker.js +1 -1
- package/dist/hooks/linear-task-picker.js.map +2 -2
- package/dist/hooks/schemas.js +55 -1
- package/dist/hooks/schemas.js.map +2 -2
- package/dist/hooks/session-summary.js +5 -1
- package/dist/hooks/session-summary.js.map +2 -2
- package/dist/hooks/sms-action-runner.js +12 -1
- package/dist/hooks/sms-action-runner.js.map +2 -2
- package/dist/hooks/sms-notify.js +4 -2
- package/dist/hooks/sms-notify.js.map +2 -2
- package/dist/hooks/sms-webhook.js +23 -2
- package/dist/hooks/sms-webhook.js.map +2 -2
- package/dist/hooks/whatsapp-commands.js +376 -0
- package/dist/hooks/whatsapp-commands.js.map +7 -0
- package/dist/hooks/whatsapp-scheduler.js +317 -0
- package/dist/hooks/whatsapp-scheduler.js.map +7 -0
- package/dist/hooks/whatsapp-sync.js +375 -0
- package/dist/hooks/whatsapp-sync.js.map +7 -0
- package/package.json +2 -3
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/context.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Context Commands for StackMemory CLI\n * Manage context stack (show, push, pop)\n */\n\nimport { Command } from 'commander';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\nimport { FrameManager, FrameType } from '../../core/context/frame-manager.js';\nimport { createContextRehydrateCommand } from './context-rehydrate.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\nexport function createContextCommands(): Command {\n const context = new Command('context')\n .alias('ctx')\n .description('Manage context stack');\n\n // Show current context\n context\n .command('show')\n .alias('status')\n .description('Show current context stack')\n .option('-v, --verbose', 'Show detailed information')\n .action(async (options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n // Get project ID - try metadata table, fallback to default\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(\n `\n SELECT value FROM metadata WHERE key = 'project_id'\n `\n )\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {\n // metadata table doesn't exist, use default\n }\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const depth = frameManager.getStackDepth();\n const activePath = frameManager.getActiveFramePath();\n\n console.log(`\\n\uD83D\uDCDA Context Stack\\n`);\n console.log(`Project: ${projectId}`);\n console.log(`Depth: ${depth}`);\n console.log(`Active frames: ${activePath.length}\\n`);\n\n if (activePath.length === 0) {\n console.log('No active context frames.\\n');\n console.log('Use \"stackmemory context push\" to create one.');\n } else {\n const typeIcon: Record<string, string> = {\n session: '\uD83D\uDD37',\n task: '\uD83D\uDCCB',\n command: '\u26A1',\n file: '\uD83D\uDCC4',\n decision: '\uD83D\uDCA1',\n };\n\n console.log('Stack (bottom to top):');\n activePath.forEach((frame, i) => {\n const icon = typeIcon[frame.type] || '\uD83D\uDCE6';\n const indent = ' '.repeat(i);\n console.log(\n `${indent}${i === activePath.length - 1 ? '\u2514\u2500' : '\u251C\u2500'} ${icon} ${frame.name || frame.frame_id.slice(0, 10)}`\n );\n\n if (options.verbose) {\n console.log(`${indent} ID: ${frame.frame_id}`);\n console.log(`${indent} Type: ${frame.type}`);\n console.log(\n `${indent} Created: ${new Date(frame.created_at * 1000).toLocaleString()}`\n );\n }\n });\n }\n\n console.log('');\n } catch (error: unknown) {\n console.error('\u274C Failed to show context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Push new context frame\n context\n .command('push <name>')\n .description('Push a new context frame onto the stack')\n .option(\n '-t, --type <type>',\n 'Frame type (session, task, command, file, decision)',\n 'task'\n )\n .option('-m, --metadata <json>', 'Additional metadata as JSON')\n .action(async (name, options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n // Get current top frame as parent\n const activePath = frameManager.getActiveFramePath();\n const parentId =\n activePath.length > 0\n ? activePath[activePath.length - 1].frame_id\n : undefined;\n\n // Parse metadata if provided\n let inputs = {};\n if (options.metadata) {\n try {\n inputs = JSON.parse(options.metadata);\n } catch {\n console.log('\u26A0\uFE0F Invalid metadata JSON, ignoring');\n }\n }\n\n const frameId = frameManager.createFrame({\n type: options.type as FrameType,\n name,\n inputs,\n parentFrameId: parentId,\n });\n\n console.log(`\u2705 Pushed context frame: ${name}`);\n console.log(` ID: ${frameId.slice(0, 10)}`);\n console.log(` Type: ${options.type}`);\n console.log(` Depth: ${frameManager.getStackDepth()}`);\n } catch (error: unknown) {\n console.error('\u274C Failed to push context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Pop context frame\n context\n .command('pop')\n .description('Pop the top context frame from the stack')\n .option('-a, --all', 'Pop all frames (clear stack)')\n .action(async (options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const activePath = frameManager.getActiveFramePath();\n\n if (activePath.length === 0) {\n console.log('\uD83D\uDCDA Stack is already empty.');\n return;\n }\n\n if (options.all) {\n // Close all frames from top to bottom\n for (let i = activePath.length - 1; i >= 0; i--) {\n frameManager.closeFrame(activePath[i].frame_id);\n }\n console.log(`\u2705 Cleared all ${activePath.length} context frames.`);\n } else {\n // Close just the top frame\n const topFrame = activePath[activePath.length - 1];\n frameManager.closeFrame(topFrame.frame_id);\n console.log(\n `\u2705 Popped: ${topFrame.name || topFrame.frame_id.slice(0, 10)}`\n );\n console.log(` Depth: ${frameManager.getStackDepth()}`);\n }\n } catch (error: unknown) {\n console.error('\u274C Failed to pop context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Add event to current context\n context\n .command('add <type> <message>')\n .description(\n 'Add an event to current context (types: observation, decision, error)'\n )\n .action(async (type, message) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const activePath = frameManager.getActiveFramePath();\n\n if (activePath.length === 0) {\n console.log('\u26A0\uFE0F No active context frame. Creating one...');\n frameManager.createFrame({\n type: 'task',\n name: 'cli-session',\n inputs: {},\n });\n }\n\n const currentFrame = frameManager.getActiveFramePath().slice(-1)[0];\n\n const validTypes = [\n 'observation',\n 'decision',\n 'error',\n 'action',\n 'result',\n ];\n if (!validTypes.includes(type)) {\n console.log(`\u26A0\uFE0F Unknown event type \"${type}\". Using \"observation\".`);\n type = 'observation';\n }\n\n frameManager.addEvent(\n type,\n { message, content: message },\n currentFrame.frame_id\n );\n\n console.log(\n `\u2705 Added ${type}: ${message.slice(0, 50)}${message.length > 50 ? '...' : ''}`\n );\n } catch (error: unknown) {\n console.error('\u274C Failed to add event:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Worktree integration commands\n context\n .command('worktree [action]')\n .description('Manage Claude worktree contexts')\n .option('-i, --instance <id>', 'Instance ID')\n .option('-b, --branch <name>', 'Branch name')\n .option('-l, --list', 'List worktree contexts')\n .action(async (action, options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n if (options.list || action === 'list') {\n // List all worktree contexts\n const worktreeFrames = db\n .prepare(\n `\n SELECT * FROM frames \n WHERE project_id = ? \n AND type = 'session' \n AND inputs LIKE '%worktree%'\n ORDER BY created_at DESC\n LIMIT 10\n `\n )\n .all(projectId) as any[];\n\n console.log('\\n\uD83C\uDF33 Worktree Contexts\\n');\n if (worktreeFrames.length === 0) {\n console.log('No worktree contexts found.');\n } else {\n worktreeFrames.forEach((frame) => {\n const inputs = JSON.parse(frame.inputs || '{}');\n const instanceId = inputs.instanceId || 'unknown';\n const branch = inputs.branch || 'unknown';\n const created = new Date(\n frame.created_at * 1000\n ).toLocaleString();\n console.log(`\uD83D\uDCCD ${frame.name || frame.frame_id.slice(0, 10)}`);\n console.log(` Instance: ${instanceId}`);\n console.log(` Branch: ${branch}`);\n console.log(` Created: ${created}`);\n console.log('');\n });\n }\n } else if (action === 'save') {\n // Save current worktree context\n const instanceId =\n options.instance || process.env['CLAUDE_INSTANCE_ID'];\n const branch = options.branch || 'unknown';\n\n if (!instanceId) {\n console.log('\u26A0\uFE0F No instance ID provided or detected.');\n return;\n }\n\n const frameId = frameManager.createFrame({\n type: 'task',\n name: `worktree-${branch}`,\n inputs: {\n worktree: true,\n instanceId,\n branch,\n path: process.cwd(),\n },\n });\n\n console.log(`\u2705 Saved worktree context for ${branch}`);\n console.log(` Instance: ${instanceId}`);\n console.log(` Frame ID: ${frameId.slice(0, 10)}`);\n } else if (action === 'load') {\n // Load worktree context\n const instanceId =\n options.instance || process.env['CLAUDE_INSTANCE_ID'];\n\n if (!instanceId) {\n console.log('\u26A0\uFE0F No instance ID provided.');\n return;\n }\n\n const worktreeFrame = db\n .prepare(\n `\n SELECT * FROM frames \n WHERE project_id = ? \n AND type = 'session' \n AND inputs LIKE ?\n ORDER BY created_at DESC\n LIMIT 1\n `\n )\n .get(projectId, `%\"instanceId\":\"${instanceId}\"%`) as any;\n\n if (worktreeFrame) {\n const inputs = JSON.parse(worktreeFrame.inputs || '{}');\n console.log(`\u2705 Loaded worktree context`);\n console.log(` Branch: ${inputs.branch}`);\n console.log(` Instance: ${inputs.instanceId}`);\n console.log(` Path: ${inputs.path}`);\n } else {\n console.log('\u26A0\uFE0F No worktree context found for this instance.');\n }\n } else {\n console.log('Usage: stackmemory context worktree [save|load|list]');\n }\n } catch (error: unknown) {\n console.error(\n '\u274C Failed to manage worktree context:',\n (error as Error).message\n );\n } finally {\n db.close();\n }\n });\n\n // Add rehydrate subcommand\n context.addCommand(createContextRehydrateCommand());\n\n return context;\n}\n"],
|
|
5
|
-
"mappings": ";;;;AAKA,SAAS,eAAe;AACxB,OAAO,cAAc;AACrB,SAAS,YAAY;AACrB,SAAS,kBAAkB;AAC3B,SAAS,oBAA+B;AACxC,SAAS,qCAAqC;
|
|
4
|
+
"sourcesContent": ["/**\n * Context Commands for StackMemory CLI\n * Manage context stack (show, push, pop)\n */\n\nimport { Command } from 'commander';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\nimport { FrameManager, FrameType } from '../../core/context/frame-manager.js';\nimport { createContextRehydrateCommand } from './context-rehydrate.js';\n\n// Type-safe environment variable access\n\nexport function createContextCommands(): Command {\n const context = new Command('context')\n .alias('ctx')\n .description('Manage context stack');\n\n // Show current context\n context\n .command('show')\n .alias('status')\n .description('Show current context stack')\n .option('-v, --verbose', 'Show detailed information')\n .action(async (options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n // Get project ID - try metadata table, fallback to default\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(\n `\n SELECT value FROM metadata WHERE key = 'project_id'\n `\n )\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {\n // metadata table doesn't exist, use default\n }\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const depth = frameManager.getStackDepth();\n const activePath = frameManager.getActiveFramePath();\n\n console.log(`\\n\uD83D\uDCDA Context Stack\\n`);\n console.log(`Project: ${projectId}`);\n console.log(`Depth: ${depth}`);\n console.log(`Active frames: ${activePath.length}\\n`);\n\n if (activePath.length === 0) {\n console.log('No active context frames.\\n');\n console.log('Use \"stackmemory context push\" to create one.');\n } else {\n const typeIcon: Record<string, string> = {\n session: '\uD83D\uDD37',\n task: '\uD83D\uDCCB',\n command: '\u26A1',\n file: '\uD83D\uDCC4',\n decision: '\uD83D\uDCA1',\n };\n\n console.log('Stack (bottom to top):');\n activePath.forEach((frame, i) => {\n const icon = typeIcon[frame.type] || '\uD83D\uDCE6';\n const indent = ' '.repeat(i);\n console.log(\n `${indent}${i === activePath.length - 1 ? '\u2514\u2500' : '\u251C\u2500'} ${icon} ${frame.name || frame.frame_id.slice(0, 10)}`\n );\n\n if (options.verbose) {\n console.log(`${indent} ID: ${frame.frame_id}`);\n console.log(`${indent} Type: ${frame.type}`);\n console.log(\n `${indent} Created: ${new Date(frame.created_at * 1000).toLocaleString()}`\n );\n }\n });\n }\n\n console.log('');\n } catch (error: unknown) {\n console.error('\u274C Failed to show context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Push new context frame\n context\n .command('push <name>')\n .description('Push a new context frame onto the stack')\n .option(\n '-t, --type <type>',\n 'Frame type (session, task, command, file, decision)',\n 'task'\n )\n .option('-m, --metadata <json>', 'Additional metadata as JSON')\n .action(async (name, options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n // Get current top frame as parent\n const activePath = frameManager.getActiveFramePath();\n const parentId =\n activePath.length > 0\n ? activePath[activePath.length - 1].frame_id\n : undefined;\n\n // Parse metadata if provided\n let inputs = {};\n if (options.metadata) {\n try {\n inputs = JSON.parse(options.metadata);\n } catch {\n console.log('\u26A0\uFE0F Invalid metadata JSON, ignoring');\n }\n }\n\n const frameId = frameManager.createFrame({\n type: options.type as FrameType,\n name,\n inputs,\n parentFrameId: parentId,\n });\n\n console.log(`\u2705 Pushed context frame: ${name}`);\n console.log(` ID: ${frameId.slice(0, 10)}`);\n console.log(` Type: ${options.type}`);\n console.log(` Depth: ${frameManager.getStackDepth()}`);\n } catch (error: unknown) {\n console.error('\u274C Failed to push context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Pop context frame\n context\n .command('pop')\n .description('Pop the top context frame from the stack')\n .option('-a, --all', 'Pop all frames (clear stack)')\n .action(async (options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const activePath = frameManager.getActiveFramePath();\n\n if (activePath.length === 0) {\n console.log('\uD83D\uDCDA Stack is already empty.');\n return;\n }\n\n if (options.all) {\n // Close all frames from top to bottom\n for (let i = activePath.length - 1; i >= 0; i--) {\n frameManager.closeFrame(activePath[i].frame_id);\n }\n console.log(`\u2705 Cleared all ${activePath.length} context frames.`);\n } else {\n // Close just the top frame\n const topFrame = activePath[activePath.length - 1];\n frameManager.closeFrame(topFrame.frame_id);\n console.log(\n `\u2705 Popped: ${topFrame.name || topFrame.frame_id.slice(0, 10)}`\n );\n console.log(` Depth: ${frameManager.getStackDepth()}`);\n }\n } catch (error: unknown) {\n console.error('\u274C Failed to pop context:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Add event to current context\n context\n .command('add <type> <message>')\n .description(\n 'Add an event to current context (types: observation, decision, error)'\n )\n .action(async (type, message) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n const activePath = frameManager.getActiveFramePath();\n\n if (activePath.length === 0) {\n console.log('\u26A0\uFE0F No active context frame. Creating one...');\n frameManager.createFrame({\n type: 'task',\n name: 'cli-session',\n inputs: {},\n });\n }\n\n const currentFrame = frameManager.getActiveFramePath().slice(-1)[0];\n\n const validTypes = [\n 'observation',\n 'decision',\n 'error',\n 'action',\n 'result',\n ];\n if (!validTypes.includes(type)) {\n console.log(`\u26A0\uFE0F Unknown event type \"${type}\". Using \"observation\".`);\n type = 'observation';\n }\n\n frameManager.addEvent(\n type,\n { message, content: message },\n currentFrame.frame_id\n );\n\n console.log(\n `\u2705 Added ${type}: ${message.slice(0, 50)}${message.length > 50 ? '...' : ''}`\n );\n } catch (error: unknown) {\n console.error('\u274C Failed to add event:', (error as Error).message);\n } finally {\n db.close();\n }\n });\n\n // Worktree integration commands\n context\n .command('worktree [action]')\n .description('Manage Claude worktree contexts')\n .option('-i, --instance <id>', 'Instance ID')\n .option('-b, --branch <name>', 'Branch name')\n .option('-l, --list', 'List worktree contexts')\n .action(async (action, options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const db = new Database(dbPath);\n\n try {\n let projectId = 'default';\n try {\n const projectRow = db\n .prepare(`SELECT value FROM metadata WHERE key = 'project_id'`)\n .get() as any;\n if (projectRow?.value) projectId = projectRow.value;\n } catch {}\n\n const frameManager = new FrameManager(db, projectId, {\n skipContextBridge: true,\n });\n\n if (options.list || action === 'list') {\n // List all worktree contexts\n const worktreeFrames = db\n .prepare(\n `\n SELECT * FROM frames \n WHERE project_id = ? \n AND type = 'session' \n AND inputs LIKE '%worktree%'\n ORDER BY created_at DESC\n LIMIT 10\n `\n )\n .all(projectId) as any[];\n\n console.log('\\n\uD83C\uDF33 Worktree Contexts\\n');\n if (worktreeFrames.length === 0) {\n console.log('No worktree contexts found.');\n } else {\n worktreeFrames.forEach((frame) => {\n const inputs = JSON.parse(frame.inputs || '{}');\n const instanceId = inputs.instanceId || 'unknown';\n const branch = inputs.branch || 'unknown';\n const created = new Date(\n frame.created_at * 1000\n ).toLocaleString();\n console.log(`\uD83D\uDCCD ${frame.name || frame.frame_id.slice(0, 10)}`);\n console.log(` Instance: ${instanceId}`);\n console.log(` Branch: ${branch}`);\n console.log(` Created: ${created}`);\n console.log('');\n });\n }\n } else if (action === 'save') {\n // Save current worktree context\n const instanceId =\n options.instance || process.env['CLAUDE_INSTANCE_ID'];\n const branch = options.branch || 'unknown';\n\n if (!instanceId) {\n console.log('\u26A0\uFE0F No instance ID provided or detected.');\n return;\n }\n\n const frameId = frameManager.createFrame({\n type: 'task',\n name: `worktree-${branch}`,\n inputs: {\n worktree: true,\n instanceId,\n branch,\n path: process.cwd(),\n },\n });\n\n console.log(`\u2705 Saved worktree context for ${branch}`);\n console.log(` Instance: ${instanceId}`);\n console.log(` Frame ID: ${frameId.slice(0, 10)}`);\n } else if (action === 'load') {\n // Load worktree context\n const instanceId =\n options.instance || process.env['CLAUDE_INSTANCE_ID'];\n\n if (!instanceId) {\n console.log('\u26A0\uFE0F No instance ID provided.');\n return;\n }\n\n const worktreeFrame = db\n .prepare(\n `\n SELECT * FROM frames \n WHERE project_id = ? \n AND type = 'session' \n AND inputs LIKE ?\n ORDER BY created_at DESC\n LIMIT 1\n `\n )\n .get(projectId, `%\"instanceId\":\"${instanceId}\"%`) as any;\n\n if (worktreeFrame) {\n const inputs = JSON.parse(worktreeFrame.inputs || '{}');\n console.log(`\u2705 Loaded worktree context`);\n console.log(` Branch: ${inputs.branch}`);\n console.log(` Instance: ${inputs.instanceId}`);\n console.log(` Path: ${inputs.path}`);\n } else {\n console.log('\u26A0\uFE0F No worktree context found for this instance.');\n }\n } else {\n console.log('Usage: stackmemory context worktree [save|load|list]');\n }\n } catch (error: unknown) {\n console.error(\n '\u274C Failed to manage worktree context:',\n (error as Error).message\n );\n } finally {\n db.close();\n }\n });\n\n // Add rehydrate subcommand\n context.addCommand(createContextRehydrateCommand());\n\n return context;\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -3,9 +3,7 @@ import { dirname as __pathDirname } from 'path';
|
|
|
3
3
|
const __filename = __fileURLToPath(import.meta.url);
|
|
4
4
|
const __dirname = __pathDirname(__filename);
|
|
5
5
|
import chalk from "chalk";
|
|
6
|
-
import {
|
|
7
|
-
LinearAuthManager
|
|
8
|
-
} from "../../integrations/linear/auth.js";
|
|
6
|
+
import { LinearAuthManager } from "../../integrations/linear/auth.js";
|
|
9
7
|
import { LinearOAuthServer } from "../../integrations/linear/oauth-server.js";
|
|
10
8
|
import {
|
|
11
9
|
LinearSyncEngine,
|
|
@@ -24,17 +22,6 @@ import { join } from "path";
|
|
|
24
22
|
import { existsSync } from "fs";
|
|
25
23
|
import { logger } from "../../core/monitoring/logger.js";
|
|
26
24
|
import Table from "cli-table3";
|
|
27
|
-
function getEnv(key, defaultValue) {
|
|
28
|
-
const value = process.env[key];
|
|
29
|
-
if (value === void 0) {
|
|
30
|
-
if (defaultValue !== void 0) return defaultValue;
|
|
31
|
-
throw new Error(`Environment variable ${key} is required`);
|
|
32
|
-
}
|
|
33
|
-
return value;
|
|
34
|
-
}
|
|
35
|
-
function getOptionalEnv(key) {
|
|
36
|
-
return process.env[key];
|
|
37
|
-
}
|
|
38
25
|
function displaySyncResult(result) {
|
|
39
26
|
if (result.success) {
|
|
40
27
|
console.log(chalk.green("\u2713 Sync completed successfully"));
|
|
@@ -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 { LinearTaskManager } from '../../features/tasks/linear-task-manager.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 LinearTaskManager(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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",
|
|
4
|
+
"sourcesContent": ["/**\n * Linear integration commands\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport { LinearAuthManager } 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 { LinearTaskManager } from '../../features/tasks/linear-task-manager.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 } from 'fs';\nimport { logger } from '../../core/monitoring/logger.js';\nimport Table from 'cli-table3';\nimport { SyncResult } from '../../integrations/linear/sync.js';\n\n// Type-safe environment variable access\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 LinearTaskManager(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
|
}
|
|
@@ -8,6 +8,7 @@ import { homedir } from "os";
|
|
|
8
8
|
import { join } from "path";
|
|
9
9
|
import { existsSync, mkdirSync, writeFileSync, readFileSync } from "fs";
|
|
10
10
|
import open from "open";
|
|
11
|
+
import { IntegrationError, ErrorCode } from "../../core/errors/index.js";
|
|
11
12
|
function registerLoginCommand(program) {
|
|
12
13
|
program.command("login").description("Login to hosted StackMemory service").option("--api-url <url>", "Custom API URL", "https://api.stackmemory.ai").option("--email <email>", "Email address for login").option("--password <password>", "Password (not recommended in CLI)").action(async (options) => {
|
|
13
14
|
const cfgDir = join(homedir(), ".stackmemory");
|
|
@@ -50,7 +51,9 @@ function registerLoginCommand(program) {
|
|
|
50
51
|
const data = await response.json();
|
|
51
52
|
if (!response.ok || !data.success) {
|
|
52
53
|
if (response.status === 404) {
|
|
53
|
-
console.log(
|
|
54
|
+
console.log(
|
|
55
|
+
chalk.yellow("\n\u26A0\uFE0F Hosted API not available. Would you like to:")
|
|
56
|
+
);
|
|
54
57
|
const { choice } = await inquirer.prompt([
|
|
55
58
|
{
|
|
56
59
|
type: "list",
|
|
@@ -80,7 +83,8 @@ function registerLoginCommand(program) {
|
|
|
80
83
|
const cfgPath2 = join(cfgDir, "config.json");
|
|
81
84
|
let cfg2 = {};
|
|
82
85
|
try {
|
|
83
|
-
if (existsSync(cfgPath2))
|
|
86
|
+
if (existsSync(cfgPath2))
|
|
87
|
+
cfg2 = JSON.parse(readFileSync(cfgPath2, "utf-8"));
|
|
84
88
|
} catch {
|
|
85
89
|
}
|
|
86
90
|
cfg2.database = { mode: "hosted", url: databaseUrl };
|
|
@@ -92,7 +96,8 @@ function registerLoginCommand(program) {
|
|
|
92
96
|
const cfgPath2 = join(cfgDir, "config.json");
|
|
93
97
|
let cfg2 = {};
|
|
94
98
|
try {
|
|
95
|
-
if (existsSync(cfgPath2))
|
|
99
|
+
if (existsSync(cfgPath2))
|
|
100
|
+
cfg2 = JSON.parse(readFileSync(cfgPath2, "utf-8"));
|
|
96
101
|
} catch {
|
|
97
102
|
}
|
|
98
103
|
cfg2.database = { mode: "local" };
|
|
@@ -104,12 +109,17 @@ function registerLoginCommand(program) {
|
|
|
104
109
|
return;
|
|
105
110
|
}
|
|
106
111
|
}
|
|
107
|
-
throw new
|
|
112
|
+
throw new IntegrationError(
|
|
113
|
+
data.error || "Authentication failed",
|
|
114
|
+
ErrorCode.LINEAR_AUTH_FAILED,
|
|
115
|
+
{ email: credentials.email, apiUrl }
|
|
116
|
+
);
|
|
108
117
|
}
|
|
109
118
|
const cfgPath = join(cfgDir, "config.json");
|
|
110
119
|
let cfg = {};
|
|
111
120
|
try {
|
|
112
|
-
if (existsSync(cfgPath))
|
|
121
|
+
if (existsSync(cfgPath))
|
|
122
|
+
cfg = JSON.parse(readFileSync(cfgPath, "utf-8"));
|
|
113
123
|
} catch {
|
|
114
124
|
}
|
|
115
125
|
cfg.auth = {
|
|
@@ -132,14 +142,26 @@ ${data.databaseUrl ? `DATABASE_URL=${data.databaseUrl}` : ""}
|
|
|
132
142
|
`;
|
|
133
143
|
writeFileSync(envFile, envContent);
|
|
134
144
|
console.log(chalk.green("\n\u2705 Successfully logged in to StackMemory"));
|
|
135
|
-
console.log(
|
|
145
|
+
console.log(
|
|
146
|
+
chalk.green(`\u2713 Configuration saved to ~/.stackmemory/config.json`)
|
|
147
|
+
);
|
|
136
148
|
console.log(chalk.gray("\nYou can now use:"));
|
|
137
|
-
console.log(
|
|
138
|
-
|
|
139
|
-
|
|
149
|
+
console.log(
|
|
150
|
+
chalk.cyan(" stackmemory sync ") + chalk.gray("- Sync your context to the cloud")
|
|
151
|
+
);
|
|
152
|
+
console.log(
|
|
153
|
+
chalk.cyan(" stackmemory db status") + chalk.gray("- Check database connection")
|
|
154
|
+
);
|
|
155
|
+
console.log(
|
|
156
|
+
chalk.cyan(" stackmemory context ") + chalk.gray("- Manage your contexts")
|
|
157
|
+
);
|
|
140
158
|
} catch (error) {
|
|
141
159
|
console.error(chalk.red("\n\u274C Login failed:"), error.message);
|
|
142
|
-
console.log(
|
|
160
|
+
console.log(
|
|
161
|
+
chalk.yellow(
|
|
162
|
+
"\nTip: Visit https://stackmemory.ai/signup to create an account"
|
|
163
|
+
)
|
|
164
|
+
);
|
|
143
165
|
process.exit(1);
|
|
144
166
|
}
|
|
145
167
|
});
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/login.ts"],
|
|
4
|
-
"sourcesContent": ["import { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync, readFileSync } from 'fs';\nimport open from 'open';\n\ninterface ConfigShape {\n version?: string;\n setupCompleted?: string;\n features?: any;\n paths?: any;\n database?: { mode?: 'local' | 'hosted'; url?: string };\n auth?: {
|
|
5
|
-
"mappings": ";;;;AACA,OAAO,cAAc;AACrB,OAAO,WAAW;AAClB,SAAS,eAAe;AACxB,SAAS,YAAY;AACrB,SAAS,YAAY,WAAW,eAAe,oBAAoB;AACnE,OAAO,UAAU;
|
|
4
|
+
"sourcesContent": ["import { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync, readFileSync } from 'fs';\nimport open from 'open';\nimport { IntegrationError, ErrorCode } from '../../core/errors/index.js';\n\ninterface ConfigShape {\n version?: string;\n setupCompleted?: string;\n features?: any;\n paths?: any;\n database?: { mode?: 'local' | 'hosted'; url?: string };\n auth?: {\n apiKey?: string;\n apiUrl?: string;\n email?: string;\n };\n}\n\ninterface AuthResponse {\n success: boolean;\n apiKey?: string;\n databaseUrl?: string;\n email?: string;\n error?: string;\n}\n\nexport function registerLoginCommand(program: Command): void {\n program\n .command('login')\n .description('Login to hosted StackMemory service')\n .option('--api-url <url>', 'Custom API URL', 'https://api.stackmemory.ai')\n .option('--email <email>', 'Email address for login')\n .option('--password <password>', 'Password (not recommended in CLI)')\n .action(async (options) => {\n const cfgDir = join(homedir(), '.stackmemory');\n if (!existsSync(cfgDir)) mkdirSync(cfgDir, { recursive: true });\n\n console.log(chalk.cyan('\uD83D\uDD10 StackMemory Hosted Service Login\\n'));\n\n // Prompt for credentials\n const credentials = await inquirer.prompt([\n {\n type: 'input',\n name: 'email',\n message: 'Email:',\n default: options.email,\n validate: (input: string) => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n return emailRegex.test(input) ? true : 'Please enter a valid email';\n },\n },\n {\n type: 'password',\n name: 'password',\n message: 'Password:',\n default: options.password,\n mask: '*',\n validate: (input: string) =>\n input.length >= 6 ? true : 'Password must be at least 6 characters',\n },\n ]);\n\n console.log(chalk.gray('\\nAuthenticating with StackMemory API...'));\n\n try {\n // Authenticate with the hosted API\n const apiUrl =\n options.apiUrl ||\n process.env.STACKMEMORY_API_URL ||\n 'https://api.stackmemory.ai';\n const response = await fetch(`${apiUrl}/auth/login`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'User-Agent': 'StackMemory-CLI/0.3.19',\n },\n body: JSON.stringify({\n email: credentials.email,\n password: credentials.password,\n }),\n });\n\n const data: AuthResponse = await response.json();\n\n if (!response.ok || !data.success) {\n if (response.status === 404) {\n // Fallback to Railway server if hosted API not available\n console.log(\n chalk.yellow('\\n\u26A0\uFE0F Hosted API not available. Would you like to:')\n );\n const { choice } = await inquirer.prompt([\n {\n type: 'list',\n name: 'choice',\n message: 'Select an option:',\n choices: [\n { name: 'Open signup page in browser', value: 'signup' },\n { name: 'Configure database URL manually', value: 'manual' },\n { name: 'Use local database', value: 'local' },\n { name: 'Cancel', value: 'cancel' },\n ],\n },\n ]);\n\n if (choice === 'signup') {\n await open('https://stackmemory.ai/signup');\n console.log(chalk.cyan('Opening signup page in browser...'));\n return;\n } else if (choice === 'manual') {\n const { databaseUrl } = await inquirer.prompt([\n {\n type: 'password',\n name: 'databaseUrl',\n message: 'Enter your DATABASE_URL (postgres://...):',\n validate: (input: string) =>\n input.startsWith('postgres://') ||\n input.startsWith('postgresql://')\n ? true\n : 'Must start with postgres:// or postgresql://',\n },\n ]);\n\n // Save manual configuration\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.database = { mode: 'hosted', url: databaseUrl };\n cfg.auth = { email: credentials.email };\n\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n console.log(chalk.green('\u2713 Database configured successfully'));\n return;\n } else if (choice === 'local') {\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.database = { mode: 'local' };\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n console.log(chalk.green('\u2713 Switched to local database mode'));\n return;\n } else {\n console.log(chalk.gray('Login cancelled'));\n return;\n }\n }\n\n throw new IntegrationError(\n data.error || 'Authentication failed',\n ErrorCode.LINEAR_AUTH_FAILED,\n { email: credentials.email, apiUrl }\n );\n }\n\n // Save configuration\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.auth = {\n apiKey: data.apiKey,\n apiUrl: apiUrl,\n email: credentials.email,\n };\n\n if (data.databaseUrl) {\n cfg.database = {\n mode: 'hosted',\n url: data.databaseUrl,\n };\n }\n\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n\n // Save environment variables\n const envFile = join(cfgDir, 'stackmemory.env');\n const envContent = `# StackMemory Authentication\nSTACKMEMORY_API_KEY=${data.apiKey}\nSTACKMEMORY_API_URL=${apiUrl}\n${data.databaseUrl ? `DATABASE_URL=${data.databaseUrl}` : ''}\n`;\n writeFileSync(envFile, envContent);\n\n console.log(chalk.green('\\n\u2705 Successfully logged in to StackMemory'));\n console.log(\n chalk.green(`\u2713 Configuration saved to ~/.stackmemory/config.json`)\n );\n console.log(chalk.gray('\\nYou can now use:'));\n console.log(\n chalk.cyan(' stackmemory sync ') +\n chalk.gray('- Sync your context to the cloud')\n );\n console.log(\n chalk.cyan(' stackmemory db status') +\n chalk.gray('- Check database connection')\n );\n console.log(\n chalk.cyan(' stackmemory context ') +\n chalk.gray('- Manage your contexts')\n );\n } catch (error: any) {\n console.error(chalk.red('\\n\u274C Login failed:'), error.message);\n console.log(\n chalk.yellow(\n '\\nTip: Visit https://stackmemory.ai/signup to create an account'\n )\n );\n process.exit(1);\n }\n });\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["cfgPath", "cfg"]
|
|
7
7
|
}
|