lean-spec 0.2.1 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (36) hide show
  1. package/dist/{chunk-ER23B6KS.js → chunk-7MCDTSVE.js} +1797 -193
  2. package/dist/chunk-7MCDTSVE.js.map +1 -0
  3. package/dist/cli.js +36 -2
  4. package/dist/cli.js.map +1 -1
  5. package/dist/commands-GRG5UUOF.js +4 -0
  6. package/dist/{commands-ZNL7ZCHU.js.map → commands-GRG5UUOF.js.map} +1 -1
  7. package/dist/mcp-server.js +1 -1
  8. package/package.json +5 -3
  9. package/templates/_shared/agents-components/core-rules-base.md +5 -0
  10. package/templates/_shared/agents-components/core-rules-enterprise.md +5 -0
  11. package/templates/_shared/agents-components/discovery-commands-enterprise.md +10 -0
  12. package/templates/_shared/agents-components/discovery-commands-minimal.md +8 -0
  13. package/templates/_shared/agents-components/discovery-commands-standard.md +9 -0
  14. package/templates/_shared/agents-components/enterprise-approval.md +10 -0
  15. package/templates/_shared/agents-components/enterprise-compliance.md +12 -0
  16. package/templates/_shared/agents-components/enterprise-when-required.md +13 -0
  17. package/templates/_shared/agents-components/frontmatter-enterprise.md +33 -0
  18. package/templates/_shared/agents-components/frontmatter-minimal.md +18 -0
  19. package/templates/_shared/agents-components/frontmatter-standard.md +23 -0
  20. package/templates/_shared/agents-components/quality-standards-base.md +5 -0
  21. package/templates/_shared/agents-components/quality-standards-enterprise.md +6 -0
  22. package/templates/_shared/agents-components/when-to-use-enterprise.md +11 -0
  23. package/templates/_shared/agents-components/when-to-use-minimal.md +9 -0
  24. package/templates/_shared/agents-components/when-to-use-standard.md +9 -0
  25. package/templates/_shared/agents-components/workflow-enterprise.md +8 -0
  26. package/templates/_shared/agents-components/workflow-standard-detailed.md +7 -0
  27. package/templates/_shared/agents-components/workflow-standard.md +5 -0
  28. package/templates/_shared/agents-template.hbs +39 -0
  29. package/templates/enterprise/agents-config.json +15 -0
  30. package/templates/enterprise/files/AGENTS.md +1 -0
  31. package/templates/minimal/agents-config.json +12 -0
  32. package/templates/minimal/files/AGENTS.md +1 -0
  33. package/templates/standard/agents-config.json +12 -0
  34. package/templates/standard/files/AGENTS.md +1 -0
  35. package/dist/chunk-ER23B6KS.js.map +0 -1
  36. package/dist/commands-ZNL7ZCHU.js +0 -4
package/dist/cli.js CHANGED
@@ -1,4 +1,4 @@
1
- import { archiveSpec, backfillTimestamps, boardCommand, checkSpecs, validateCommand, createSpec, depsCommand, filesCommand, ganttCommand, initProject, listSpecs, openCommand, searchCommand, statsCommand, listTemplates, showTemplate, addTemplate, removeTemplate, copyTemplate, timelineCommand, updateSpec, viewCommand, migrateCommand, mcpCommand } from './chunk-ER23B6KS.js';
1
+ import { archiveSpec, backfillTimestamps, boardCommand, checkSpecs, validateCommand, createSpec, depsCommand, filesCommand, ganttCommand, initProject, listSpecs, openCommand, searchCommand, statsCommand, listTemplates, showTemplate, addTemplate, removeTemplate, copyTemplate, timelineCommand, updateSpec, viewCommand, migrateCommand, mcpCommand } from './chunk-7MCDTSVE.js';
2
2
  import './chunk-LVD7ZAVZ.js';
3
3
  import { Command } from 'commander';
4
4
  import { readFileSync } from 'fs';
@@ -170,8 +170,42 @@ program.command("search <query>").description("Full-text search with metadata fi
170
170
  program.command("stats").description("Show aggregate statistics (default: simplified view)").option("--tag <tag>", "Filter by tag").option("--assignee <name>", "Filter by assignee").option("--full", "Show full detailed analytics (all sections)").option("--timeline", "Show only timeline section").option("--velocity", "Show only velocity section").option("--json", "Output as JSON").action(async (options) => {
171
171
  await statsCommand(options);
172
172
  });
173
+ program.command("analyze <spec>").description("Analyze spec complexity and structure (spec 059)").option("--json", "Output as JSON for AI agents").option("--verbose", "Include detailed section breakdown").action(async (specPath, options) => {
174
+ const { analyzeCommand } = await import('./commands-GRG5UUOF.js');
175
+ await analyzeCommand(specPath, options);
176
+ });
177
+ program.command("split <spec>").description("Split spec into multiple files by line ranges (spec 059)").option("--output <file:lines>", "Output file with line range (e.g., README.md:1-150)", collectOutputs, []).option("--update-refs", "Update cross-references in README.md").option("--dry-run", "Show what would be created without making changes").option("--force", "Overwrite existing files").action(async (specPath, options) => {
178
+ const { splitCommand } = await import('./commands-GRG5UUOF.js');
179
+ const outputs = options.output.map((opt) => {
180
+ const [file, lines] = opt.split(":");
181
+ if (!file || !lines) {
182
+ throw new Error(`Invalid --output format: ${opt}. Expected: file.md:1-150`);
183
+ }
184
+ return { file, lines };
185
+ });
186
+ await splitCommand(specPath, {
187
+ outputs,
188
+ updateRefs: options.updateRefs,
189
+ dryRun: options.dryRun,
190
+ force: options.force
191
+ });
192
+ });
193
+ function collectOutputs(value, previous) {
194
+ return previous.concat([value]);
195
+ }
196
+ function collectRemoves(value, previous) {
197
+ return previous.concat([value]);
198
+ }
199
+ program.command("compact <spec>").description("Remove specified line ranges from spec (spec 059)").option("--remove <lines>", "Line range to remove (e.g., 145-153)", collectRemoves, []).option("--dry-run", "Show what would be removed without making changes").option("--force", "Skip confirmation").action(async (specPath, options) => {
200
+ const { compactCommand } = await import('./commands-GRG5UUOF.js');
201
+ await compactCommand(specPath, {
202
+ removes: options.remove,
203
+ dryRun: options.dryRun,
204
+ force: options.force
205
+ });
206
+ });
173
207
  program.command("tokens [spec]").description("Count tokens in spec(s) for LLM context management").option("--detailed", "Show content type breakdown (code, prose, tables)").option("--include-sub-specs", "Count all sub-spec files (DESIGN.md, etc.)").option("--all", "Show all specs (when [spec] is omitted)").option("--sort-by <field>", "Sort by: tokens, lines, name (default: tokens)").option("--json", "Output as JSON").action(async (specPath, options) => {
174
- const { tokensCommand, tokensAllCommand } = await import('./commands-ZNL7ZCHU.js');
208
+ const { tokensCommand, tokensAllCommand } = await import('./commands-GRG5UUOF.js');
175
209
  if (specPath) {
176
210
  await tokensCommand(specPath, options);
177
211
  } else {
package/dist/cli.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/utils/cli-helpers.ts","../src/cli.ts"],"names":[],"mappings":"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","file":"cli.js","sourcesContent":["/**\n * Parse custom fields from CLI --field options\n * @param fieldOptions Array of \"name=value\" strings\n * @returns Record of parsed field names and values\n */\nexport function parseCustomFieldOptions(fieldOptions?: string[]): Record<string, unknown> {\n const customFields: Record<string, unknown> = {};\n \n if (!fieldOptions) {\n return customFields;\n }\n \n for (const field of fieldOptions) {\n const [key, ...valueParts] = field.split('=');\n if (key && valueParts.length > 0) {\n const value = valueParts.join('='); // Handle values with '=' in them\n customFields[key.trim()] = value.trim();\n }\n }\n \n return customFields;\n}\n","import { Command } from 'commander';\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\nimport { dirname, join } from 'path';\nimport {\n createSpec,\n archiveSpec,\n listSpecs,\n updateSpec,\n checkSpecs,\n backfillTimestamps,\n listTemplates,\n showTemplate,\n addTemplate,\n removeTemplate,\n copyTemplate,\n initProject,\n statsCommand,\n boardCommand,\n timelineCommand,\n depsCommand,\n searchCommand,\n ganttCommand,\n filesCommand,\n viewCommand,\n openCommand,\n validateCommand,\n mcpCommand,\n migrateCommand,\n} from './commands/index.js';\nimport { parseCustomFieldOptions } from './utils/cli-helpers.js';\nimport type { SpecStatus, SpecPriority } from './frontmatter.js';\n\n// Get version from package.json\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\nconst packageJson = JSON.parse(\n readFileSync(join(__dirname, '../package.json'), 'utf-8')\n);\n\nconst program = new Command();\n\nprogram\n .name('lean-spec')\n .description('Manage LeanSpec documents')\n .version(packageJson.version);\n\n// Add custom help text with grouped commands\nprogram.addHelpText('after', `\nCommand Groups:\n \n Core Commands:\n init Initialize LeanSpec in current directory\n create <name> Create new spec in folder structure\n list List all specs\n update <spec> Update spec metadata\n archive <spec> Move spec to archived/\n backfill [specs...] Backfill timestamps from git history\n migrate <input-path> Migrate specs from other SDD tools\n \n Viewing & Navigation:\n view <spec> View spec content\n open <spec> Open spec in editor\n search <query> Full-text search with metadata filters\n files <spec> List files in a spec\n \n Project & Analytics:\n board Show Kanban-style board view\n stats Show aggregate statistics\n timeline Show creation/completion over time\n gantt Show timeline with dependencies\n deps <spec> Show dependency graph for a spec\n \n Maintenance:\n check Check for sequence conflicts\n validate [specs...] Validate specs for quality issues\n templates Manage spec templates\n \n Server:\n mcp Start MCP server for AI assistants\n\nExamples:\n $ lean-spec init\n $ lean-spec create my-feature --priority high\n $ lean-spec list --status in-progress\n $ lean-spec view 042\n $ lean-spec backfill --dry-run\n $ lean-spec migrate ./docs/adr\n $ lean-spec migrate ./docs/rfcs --with copilot\n $ lean-spec board --tag backend\n $ lean-spec search \"authentication\"\n $ lean-spec validate\n $ lean-spec validate --verbose\n $ lean-spec validate --quiet --rule max-lines\n $ lean-spec validate 018 --max-lines 500\n`);\n\n// archive command\nprogram\n .command('archive <spec>')\n .description('Move spec to archived/')\n .action(async (specPath: string) => {\n await archiveSpec(specPath);\n });\n\n// backfill command\nprogram\n .command('backfill [specs...]')\n .description('Backfill timestamps from git history')\n .option('--dry-run', 'Show what would be updated without making changes')\n .option('--force', 'Overwrite existing timestamp values')\n .option('--assignee', 'Include assignee from first commit author')\n .option('--transitions', 'Include full status transition history')\n .option('--all', 'Include all optional fields (assignee + transitions)')\n .action(async (specs: string[] | undefined, options: {\n dryRun?: boolean;\n force?: boolean;\n assignee?: boolean;\n transitions?: boolean;\n all?: boolean;\n }) => {\n await backfillTimestamps({\n dryRun: options.dryRun,\n force: options.force,\n includeAssignee: options.assignee || options.all,\n includeTransitions: options.transitions || options.all,\n specs: specs && specs.length > 0 ? specs : undefined,\n });\n });\n\n// board command\nprogram\n .command('board')\n .description('Show Kanban-style board view with project completion summary')\n .option('--complete', 'Include complete specs (default: hidden)')\n .option('--simple', 'Hide completion summary (kanban only)')\n .option('--completion-only', 'Show only completion summary (no kanban)')\n .option('--tag <tag>', 'Filter by tag')\n .option('--assignee <name>', 'Filter by assignee')\n .action(async (options: {\n showComplete?: boolean;\n simple?: boolean;\n completionOnly?: boolean;\n tag?: string;\n assignee?: string;\n }) => {\n await boardCommand(options);\n });\n\n// check command\nprogram\n .command('check')\n .description('Check for sequence conflicts')\n .option('-q, --quiet', 'Brief output')\n .action(async (options: {\n quiet?: boolean;\n }) => {\n const hasNoConflicts = await checkSpecs(options);\n // Exit with 0 (success) if no conflicts, 1 (error) if conflicts found\n process.exit(hasNoConflicts ? 0 : 1);\n });\n\n// validate command\nprogram\n .command('validate [specs...]')\n .description('Validate specs for quality issues')\n .option('--max-lines <number>', 'Custom line limit (default: 400)', parseInt)\n .option('--verbose', 'Show passing specs')\n .option('--quiet', 'Suppress warnings, only show errors')\n .option('--format <format>', 'Output format: default, json, compact', 'default')\n .option('--rule <rule>', 'Filter by specific rule name (e.g., max-lines, frontmatter)')\n .action(async (specs: string[] | undefined, options: {\n maxLines?: number;\n verbose?: boolean;\n quiet?: boolean;\n format?: 'default' | 'json' | 'compact';\n rule?: string;\n }) => {\n const passed = await validateCommand({\n maxLines: options.maxLines,\n specs: specs && specs.length > 0 ? specs : undefined,\n verbose: options.verbose,\n quiet: options.quiet,\n format: options.format,\n rule: options.rule,\n });\n // Exit with 0 (success) if all passed, 1 (error) if any failed\n process.exit(passed ? 0 : 1);\n });\n\n// create command\nprogram\n .command('create <name>')\n .description('Create new spec in folder structure')\n .option('--title <title>', 'Set custom title')\n .option('--description <desc>', 'Set initial description')\n .option('--tags <tags>', 'Set tags (comma-separated)')\n .option('--priority <priority>', 'Set priority (low, medium, high, critical)')\n .option('--assignee <name>', 'Set assignee')\n .option('--template <template>', 'Use a specific template')\n .option('--field <name=value...>', 'Set custom field (can specify multiple)')\n .option('--no-prefix', 'Skip date prefix even if configured')\n .action(async (name: string, options: {\n title?: string;\n description?: string;\n tags?: string;\n priority?: SpecPriority;\n assignee?: string;\n template?: string;\n field?: string[];\n prefix?: boolean;\n }) => {\n // Parse custom fields from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const createOptions: {\n title?: string;\n description?: string;\n tags?: string[];\n priority?: SpecPriority;\n assignee?: string;\n template?: string;\n customFields?: Record<string, unknown>;\n noPrefix?: boolean;\n } = {\n title: options.title,\n description: options.description,\n tags: options.tags ? options.tags.split(',').map(t => t.trim()) : undefined,\n priority: options.priority,\n assignee: options.assignee,\n template: options.template,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n noPrefix: options.prefix === false,\n };\n await createSpec(name, createOptions);\n });\n\n// deps command\nprogram\n .command('deps <spec>')\n .description('Show dependency graph for a spec. Related specs (⟷) are shown bidirectionally, depends_on (→) are directional.')\n .option('--depth <n>', 'Show N levels deep (default: 3)', parseInt)\n .option('--graph', 'ASCII graph visualization')\n .option('--json', 'Output as JSON')\n .action(async (specPath: string, options: {\n depth?: number;\n graph?: boolean;\n json?: boolean;\n }) => {\n await depsCommand(specPath, options);\n });\n\n// files command\nprogram\n .command('files <spec>')\n .description('List files in a spec')\n .option('--type <type>', 'Filter by type: docs, assets')\n .option('--tree', 'Show tree structure')\n .action(async (specPath: string, options: {\n type?: 'docs' | 'assets';\n tree?: boolean;\n }) => {\n await filesCommand(specPath, options);\n });\n\n// gantt command\nprogram\n .command('gantt')\n .description('Show timeline with dependencies')\n .option('--weeks <n>', 'Show N weeks (default: 4)', parseInt)\n .option('--show-complete', 'Include completed specs')\n .option('--critical-path', 'Highlight critical path')\n .action(async (options: {\n weeks?: number;\n showComplete?: boolean;\n criticalPath?: boolean;\n }) => {\n await ganttCommand(options);\n });\n\n// init command\nprogram\n .command('init')\n .description('Initialize LeanSpec in current directory')\n .action(async () => {\n await initProject();\n });\n\n// list command\nprogram\n .command('list')\n .description('List all specs')\n .option('--archived', 'Include archived specs')\n .option('--status <status>', 'Filter by status (planned, in-progress, complete, archived)')\n .option('--tag <tag...>', 'Filter by tag (can specify multiple)')\n .option('--priority <priority>', 'Filter by priority (low, medium, high, critical)')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--field <name=value...>', 'Filter by custom field (can specify multiple)')\n .option('--sort <field>', 'Sort by field (id, created, name, status, priority)', 'id')\n .option('--order <order>', 'Sort order (asc, desc)', 'desc')\n .action(async (options: {\n archived?: boolean;\n status?: SpecStatus;\n tag?: string[];\n priority?: SpecPriority;\n assignee?: string;\n field?: string[];\n sort?: string;\n order?: string;\n }) => {\n // Parse custom field filters from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const listOptions: {\n showArchived?: boolean;\n status?: SpecStatus;\n tags?: string[];\n priority?: SpecPriority;\n assignee?: string;\n customFields?: Record<string, unknown>;\n sortBy?: string;\n sortOrder?: 'asc' | 'desc';\n } = {\n showArchived: options.archived,\n status: options.status,\n tags: options.tag,\n priority: options.priority,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n sortBy: options.sort || 'id',\n sortOrder: (options.order as 'asc' | 'desc') || 'desc',\n };\n await listSpecs(listOptions);\n });\n\n// open command\nprogram\n .command('open <spec>')\n .description('Open spec in editor')\n .option('--editor <editor>', 'Specify editor command')\n .action(async (specPath: string, options: {\n editor?: string;\n }) => {\n try {\n await openCommand(specPath, {\n editor: options.editor,\n });\n } catch (error) {\n console.error('\\x1b[31mError:\\x1b[0m', error instanceof Error ? error.message : String(error));\n process.exit(1);\n }\n });\n\n// search command\nprogram\n .command('search <query>')\n .description('Full-text search with metadata filters')\n .option('--status <status>', 'Filter by status')\n .option('--tag <tag>', 'Filter by tag')\n .option('--priority <priority>', 'Filter by priority')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--field <name=value...>', 'Filter by custom field (can specify multiple)')\n .action(async (query: string, options: {\n status?: SpecStatus;\n tag?: string;\n priority?: SpecPriority;\n assignee?: string;\n field?: string[];\n }) => {\n // Parse custom field filters from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n await searchCommand(query, {\n status: options.status,\n tag: options.tag,\n priority: options.priority,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n });\n });\n\n// stats command\nprogram\n .command('stats')\n .description('Show aggregate statistics (default: simplified view)')\n .option('--tag <tag>', 'Filter by tag')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--full', 'Show full detailed analytics (all sections)')\n .option('--timeline', 'Show only timeline section')\n .option('--velocity', 'Show only velocity section')\n .option('--json', 'Output as JSON')\n .action(async (options: {\n tag?: string;\n assignee?: string;\n full?: boolean;\n timeline?: boolean;\n velocity?: boolean;\n json?: boolean;\n }) => {\n await statsCommand(options);\n });\n\n// tokens command\nprogram\n .command('tokens [spec]')\n .description('Count tokens in spec(s) for LLM context management')\n .option('--detailed', 'Show content type breakdown (code, prose, tables)')\n .option('--include-sub-specs', 'Count all sub-spec files (DESIGN.md, etc.)')\n .option('--all', 'Show all specs (when [spec] is omitted)')\n .option('--sort-by <field>', 'Sort by: tokens, lines, name (default: tokens)')\n .option('--json', 'Output as JSON')\n .action(async (specPath: string | undefined, options: {\n detailed?: boolean;\n includeSubSpecs?: boolean;\n all?: boolean;\n sortBy?: 'tokens' | 'lines' | 'name';\n json?: boolean;\n }) => {\n const { tokensCommand, tokensAllCommand } = await import('./commands/index.js');\n \n if (specPath) {\n // Count specific spec\n await tokensCommand(specPath, options);\n } else {\n // Count all specs\n await tokensAllCommand(options);\n }\n });\n\n// templates command and subcommands\nconst templatesCmd = program\n .command('templates')\n .description('Manage spec templates');\n\ntemplatesCmd\n .command('list')\n .description('List available templates')\n .action(async () => {\n await listTemplates();\n });\n\ntemplatesCmd\n .command('show <name>')\n .description('Show template content')\n .action(async (name: string) => {\n await showTemplate(name);\n });\n\ntemplatesCmd\n .command('add <name> <file>')\n .description('Register a template')\n .action(async (name: string, file: string) => {\n await addTemplate(name, file);\n });\n\ntemplatesCmd\n .command('remove <name>')\n .description('Unregister a template')\n .action(async (name: string) => {\n await removeTemplate(name);\n });\n\ntemplatesCmd\n .command('copy <source> <target>')\n .description('Copy a template to create a new one')\n .action(async (source: string, target: string) => {\n await copyTemplate(source, target);\n });\n\n// Default action for templates (list)\ntemplatesCmd\n .action(async () => {\n await listTemplates();\n });\n\n// timeline command\nprogram\n .command('timeline')\n .description('Show creation/completion over time')\n .option('--days <n>', 'Show last N days (default: 30)', parseInt)\n .option('--by-tag', 'Group by tag')\n .option('--by-assignee', 'Group by assignee')\n .action(async (options: {\n days?: number;\n byTag?: boolean;\n byAssignee?: boolean;\n }) => {\n await timelineCommand(options);\n });\n\n// update command\nprogram\n .command('update <spec>')\n .description('Update spec metadata')\n .option('--status <status>', 'Set status (planned, in-progress, complete, archived)')\n .option('--priority <priority>', 'Set priority (low, medium, high, critical)')\n .option('--tags <tags>', 'Set tags (comma-separated)')\n .option('--assignee <name>', 'Set assignee')\n .option('--field <name=value...>', 'Set custom field (can specify multiple)')\n .action(async (specPath: string, options: {\n status?: SpecStatus;\n priority?: SpecPriority;\n tags?: string;\n assignee?: string;\n field?: string[];\n }) => {\n // Parse custom fields from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const updates: {\n status?: SpecStatus;\n priority?: SpecPriority;\n tags?: string[];\n assignee?: string;\n customFields?: Record<string, unknown>;\n } = {\n status: options.status,\n priority: options.priority,\n tags: options.tags ? options.tags.split(',').map(t => t.trim()) : undefined,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n };\n \n // Filter out undefined values\n Object.keys(updates).forEach(key => {\n if (updates[key as keyof typeof updates] === undefined) {\n delete updates[key as keyof typeof updates];\n }\n });\n \n if (Object.keys(updates).length === 0) {\n console.error('Error: At least one update option required (--status, --priority, --tags, --assignee, --field)');\n process.exit(1);\n }\n \n await updateSpec(specPath, updates);\n });\n\n// view command (primary viewer)\nprogram\n .command('view <spec>')\n .description('View spec content (supports sub-specs like \"045/DESIGN.md\")')\n .option('--raw', 'Output raw markdown (for piping/scripting)')\n .option('--json', 'Output as JSON')\n .option('--no-color', 'Disable colors')\n .action(async (specPath: string, options: {\n raw?: boolean;\n json?: boolean;\n color?: boolean;\n }) => {\n try {\n await viewCommand(specPath, {\n raw: options.raw,\n json: options.json,\n noColor: options.color === false,\n });\n } catch (error) {\n console.error('\\x1b[31mError:\\x1b[0m', error instanceof Error ? error.message : String(error));\n process.exit(1);\n }\n });\n\n// migrate command\nprogram\n .command('migrate <input-path>')\n .description('Migrate specs from other SDD tools (ADR, RFC, OpenSpec, spec-kit, etc.)')\n .option('--with <provider>', 'AI-assisted migration (copilot, claude, gemini)')\n .option('--dry-run', 'Preview without making changes')\n .option('--batch-size <n>', 'Process N docs at a time', parseInt)\n .option('--skip-validation', \"Don't validate after migration\")\n .option('--backfill', 'Auto-run backfill after migration')\n .action(async (inputPath: string, options: {\n with?: string;\n dryRun?: boolean;\n batchSize?: number;\n skipValidation?: boolean;\n backfill?: boolean;\n }) => {\n // Validate AI provider if specified\n if (options.with && !['copilot', 'claude', 'gemini'].includes(options.with)) {\n console.error('\\x1b[31m❌ Error:\\x1b[0m Invalid AI provider. Use: copilot, claude, or gemini');\n process.exit(1);\n }\n \n await migrateCommand(inputPath, {\n aiProvider: options.with as 'copilot' | 'claude' | 'gemini' | undefined,\n dryRun: options.dryRun,\n batchSize: options.batchSize,\n skipValidation: options.skipValidation,\n backfill: options.backfill,\n });\n });\n\n// mcp command\nprogram\n .command('mcp')\n .description('Start MCP server for AI assistants (Claude Desktop, Cline, etc.)')\n .action(async () => {\n await mcpCommand();\n });\n\n// Parse and execute\nprogram.parse();\n"]}
1
+ {"version":3,"sources":["../src/utils/cli-helpers.ts","../src/cli.ts"],"names":[],"mappings":";;;;;;;;AAKO,SAAS,wBAAwB,YAAA,EAAkD;AACxF,EAAA,MAAM,eAAwC,EAAC;AAE/C,EAAA,IAAI,CAAC,YAAA,EAAc;AACjB,IAAA,OAAO,YAAA;AAAA,EACT;AAEA,EAAA,KAAA,MAAW,SAAS,YAAA,EAAc;AAChC,IAAA,MAAM,CAAC,GAAA,EAAK,GAAG,UAAU,CAAA,GAAI,KAAA,CAAM,MAAM,GAAG,CAAA;AAC5C,IAAA,IAAI,GAAA,IAAO,UAAA,CAAW,MAAA,GAAS,CAAA,EAAG;AAChC,MAAA,MAAM,KAAA,GAAQ,UAAA,CAAW,IAAA,CAAK,GAAG,CAAA;AACjC,MAAA,YAAA,CAAa,GAAA,CAAI,IAAA,EAAM,CAAA,GAAI,MAAM,IAAA,EAAK;AAAA,IACxC;AAAA,EACF;AAEA,EAAA,OAAO,YAAA;AACT;;;ACaA,IAAM,UAAA,GAAa,aAAA,CAAc,MAAA,CAAA,IAAA,CAAY,GAAG,CAAA;AAChD,IAAM,SAAA,GAAY,QAAQ,UAAU,CAAA;AACpC,IAAM,cAAc,IAAA,CAAK,KAAA;AAAA,EACvB,YAAA,CAAa,IAAA,CAAK,SAAA,EAAW,iBAAiB,GAAG,OAAO;AAC1D,CAAA;AAEA,IAAM,OAAA,GAAU,IAAI,OAAA,EAAQ;AAE5B,OAAA,CACG,IAAA,CAAK,WAAW,CAAA,CAChB,WAAA,CAAY,2BAA2B,CAAA,CACvC,OAAA,CAAQ,YAAY,OAAO,CAAA;AAG9B,OAAA,CAAQ,YAAY,OAAA,EAAS;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,CA+C5B,CAAA;AAGD,OAAA,CACG,OAAA,CAAQ,gBAAgB,CAAA,CACxB,WAAA,CAAY,wBAAwB,CAAA,CACpC,MAAA,CAAO,OAAO,QAAA,KAAqB;AAClC,EAAA,MAAM,YAAY,QAAQ,CAAA;AAC5B,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,qBAAqB,CAAA,CAC7B,WAAA,CAAY,sCAAsC,CAAA,CAClD,MAAA,CAAO,WAAA,EAAa,mDAAmD,CAAA,CACvE,MAAA,CAAO,SAAA,EAAW,qCAAqC,CAAA,CACvD,MAAA,CAAO,YAAA,EAAc,2CAA2C,CAAA,CAChE,MAAA,CAAO,eAAA,EAAiB,wCAAwC,CAAA,CAChE,MAAA,CAAO,OAAA,EAAS,sDAAsD,CAAA,CACtE,MAAA,CAAO,OAAO,OAA6B,OAAA,KAMtC;AACJ,EAAA,MAAM,kBAAA,CAAmB;AAAA,IACvB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,OAAO,OAAA,CAAQ,KAAA;AAAA,IACf,eAAA,EAAiB,OAAA,CAAQ,QAAA,IAAY,OAAA,CAAQ,GAAA;AAAA,IAC7C,kBAAA,EAAoB,OAAA,CAAQ,WAAA,IAAe,OAAA,CAAQ,GAAA;AAAA,IACnD,KAAA,EAAO,KAAA,IAAS,KAAA,CAAM,MAAA,GAAS,IAAI,KAAA,GAAQ;AAAA,GAC5C,CAAA;AACH,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,OAAO,CAAA,CACb,WAAA,CAAY,8DAA8D,CAAA,CAC5E,MAAA,CAAO,YAAA,EAAc,0CAA0C,CAAA,CAC/D,MAAA,CAAO,UAAA,EAAY,uCAAuC,CAAA,CAC1D,MAAA,CAAO,mBAAA,EAAqB,0CAA0C,CAAA,CACtE,MAAA,CAAO,aAAA,EAAe,eAAe,CAAA,CACrC,MAAA,CAAO,mBAAA,EAAqB,oBAAoB,CAAA,CAChD,MAAA,CAAO,OAAO,OAAA,KAMT;AACJ,EAAA,MAAM,aAAa,OAAO,CAAA;AAC5B,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,OAAO,CAAA,CACf,WAAA,CAAY,8BAA8B,CAAA,CAC1C,MAAA,CAAO,aAAA,EAAe,cAAc,CAAA,CACpC,MAAA,CAAO,OAAO,OAAA,KAET;AACJ,EAAA,MAAM,cAAA,GAAiB,MAAM,UAAA,CAAW,OAAO,CAAA;AAE/C,EAAA,OAAA,CAAQ,IAAA,CAAK,cAAA,GAAiB,CAAA,GAAI,CAAC,CAAA;AACrC,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,qBAAqB,CAAA,CAC7B,WAAA,CAAY,mCAAmC,CAAA,CAC/C,MAAA,CAAO,sBAAA,EAAwB,kCAAA,EAAoC,QAAQ,CAAA,CAC3E,MAAA,CAAO,aAAa,oBAAoB,CAAA,CACxC,MAAA,CAAO,SAAA,EAAW,qCAAqC,CAAA,CACvD,MAAA,CAAO,mBAAA,EAAqB,yCAAyC,SAAS,CAAA,CAC9E,MAAA,CAAO,eAAA,EAAiB,6DAA6D,CAAA,CACrF,MAAA,CAAO,OAAO,OAA6B,OAAA,KAMtC;AACJ,EAAA,MAAM,MAAA,GAAS,MAAM,eAAA,CAAgB;AAAA,IACnC,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,KAAA,EAAO,KAAA,IAAS,KAAA,CAAM,MAAA,GAAS,IAAI,KAAA,GAAQ,MAAA;AAAA,IAC3C,SAAS,OAAA,CAAQ,OAAA;AAAA,IACjB,OAAO,OAAA,CAAQ,KAAA;AAAA,IACf,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,MAAM,OAAA,CAAQ;AAAA,GACf,CAAA;AAED,EAAA,OAAA,CAAQ,IAAA,CAAK,MAAA,GAAS,CAAA,GAAI,CAAC,CAAA;AAC7B,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,eAAe,CAAA,CACvB,YAAY,qCAAqC,CAAA,CACjD,OAAO,iBAAA,EAAmB,kBAAkB,CAAA,CAC5C,MAAA,CAAO,wBAAwB,yBAAyB,CAAA,CACxD,OAAO,eAAA,EAAiB,4BAA4B,EACpD,MAAA,CAAO,uBAAA,EAAyB,4CAA4C,CAAA,CAC5E,OAAO,mBAAA,EAAqB,cAAc,EAC1C,MAAA,CAAO,uBAAA,EAAyB,yBAAyB,CAAA,CACzD,MAAA,CAAO,2BAA2B,yCAAyC,CAAA,CAC3E,OAAO,aAAA,EAAe,qCAAqC,EAC3D,MAAA,CAAO,OAAO,MAAc,OAAA,KASvB;AAEJ,EAAA,MAAM,YAAA,GAAe,uBAAA,CAAwB,OAAA,CAAQ,KAAK,CAAA;AAE1D,EAAA,MAAM,aAAA,GASF;AAAA,IACF,OAAO,OAAA,CAAQ,KAAA;AAAA,IACf,aAAa,OAAA,CAAQ,WAAA;AAAA,IACrB,IAAA,EAAM,OAAA,CAAQ,IAAA,GAAO,OAAA,CAAQ,IAAA,CAAK,KAAA,CAAM,GAAG,CAAA,CAAE,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,IAAA,EAAM,CAAA,GAAI,MAAA;AAAA,IAClE,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,cAAc,MAAA,CAAO,IAAA,CAAK,YAAY,CAAA,CAAE,MAAA,GAAS,IAAI,YAAA,GAAe,MAAA;AAAA,IACpE,QAAA,EAAU,QAAQ,MAAA,KAAW;AAAA,GAC/B;AACA,EAAA,MAAM,UAAA,CAAW,MAAM,aAAa,CAAA;AACtC,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,aAAa,CAAA,CACrB,WAAA,CAAY,0HAAgH,CAAA,CAC5H,MAAA,CAAO,aAAA,EAAe,iCAAA,EAAmC,QAAQ,CAAA,CACjE,OAAO,SAAA,EAAW,2BAA2B,EAC7C,MAAA,CAAO,QAAA,EAAU,gBAAgB,CAAA,CACjC,MAAA,CAAO,OAAO,QAAA,EAAkB,OAAA,KAI3B;AACJ,EAAA,MAAM,WAAA,CAAY,UAAU,OAAO,CAAA;AACrC,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,cAAc,CAAA,CACtB,WAAA,CAAY,sBAAsB,EAClC,MAAA,CAAO,eAAA,EAAiB,8BAA8B,CAAA,CACtD,OAAO,QAAA,EAAU,qBAAqB,EACtC,MAAA,CAAO,OAAO,UAAkB,OAAA,KAG3B;AACJ,EAAA,MAAM,YAAA,CAAa,UAAU,OAAO,CAAA;AACtC,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,OAAO,CAAA,CACf,WAAA,CAAY,iCAAiC,CAAA,CAC7C,MAAA,CAAO,eAAe,2BAAA,EAA6B,QAAQ,EAC3D,MAAA,CAAO,iBAAA,EAAmB,yBAAyB,CAAA,CACnD,MAAA,CAAO,mBAAmB,yBAAyB,CAAA,CACnD,MAAA,CAAO,OAAO,OAAA,KAIT;AACJ,EAAA,MAAM,aAAa,OAAO,CAAA;AAC5B,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,MAAM,CAAA,CACd,YAAY,0CAA0C,CAAA,CACtD,OAAO,YAAY;AAClB,EAAA,MAAM,WAAA,EAAY;AACpB,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,MAAM,CAAA,CACd,YAAY,gBAAgB,CAAA,CAC5B,OAAO,YAAA,EAAc,wBAAwB,EAC7C,MAAA,CAAO,mBAAA,EAAqB,6DAA6D,CAAA,CACzF,MAAA,CAAO,kBAAkB,sCAAsC,CAAA,CAC/D,OAAO,uBAAA,EAAyB,kDAAkD,CAAA,CAClF,MAAA,CAAO,qBAAqB,oBAAoB,CAAA,CAChD,OAAO,yBAAA,EAA2B,+CAA+C,EACjF,MAAA,CAAO,gBAAA,EAAkB,uDAAuD,IAAI,CAAA,CACpF,OAAO,iBAAA,EAAmB,wBAAA,EAA0B,MAAM,CAAA,CAC1D,MAAA,CAAO,OAAO,OAAA,KAST;AAEJ,EAAA,MAAM,YAAA,GAAe,uBAAA,CAAwB,OAAA,CAAQ,KAAK,CAAA;AAE1D,EAAA,MAAM,WAAA,GASF;AAAA,IACF,cAAc,OAAA,CAAQ,QAAA;AAAA,IACtB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,MAAM,OAAA,CAAQ,GAAA;AAAA,IACd,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,cAAc,MAAA,CAAO,IAAA,CAAK,YAAY,CAAA,CAAE,MAAA,GAAS,IAAI,YAAA,GAAe,MAAA;AAAA,IACpE,MAAA,EAAQ,QAAQ,IAAA,IAAQ,IAAA;AAAA,IACxB,SAAA,EAAY,QAAQ,KAAA,IAA4B;AAAA,GAClD;AACA,EAAA,MAAM,UAAU,WAAW,CAAA;AAC7B,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,aAAa,CAAA,CACrB,WAAA,CAAY,qBAAqB,CAAA,CACjC,MAAA,CAAO,mBAAA,EAAqB,wBAAwB,CAAA,CACpD,MAAA,CAAO,OAAO,UAAkB,OAAA,KAE3B;AACJ,EAAA,IAAI;AACF,IAAA,MAAM,YAAY,QAAA,EAAU;AAAA,MAC1B,QAAQ,OAAA,CAAQ;AAAA,KACjB,CAAA;AAAA,EACH,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,KAAA,CAAM,yBAAyB,KAAA,YAAiB,KAAA,GAAQ,MAAM,OAAA,GAAU,MAAA,CAAO,KAAK,CAAC,CAAA;AAC7F,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAChB;AACF,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,gBAAgB,CAAA,CACxB,WAAA,CAAY,wCAAwC,CAAA,CACpD,MAAA,CAAO,mBAAA,EAAqB,kBAAkB,CAAA,CAC9C,MAAA,CAAO,aAAA,EAAe,eAAe,CAAA,CACrC,MAAA,CAAO,uBAAA,EAAyB,oBAAoB,CAAA,CACpD,MAAA,CAAO,mBAAA,EAAqB,oBAAoB,CAAA,CAChD,MAAA,CAAO,yBAAA,EAA2B,+CAA+C,CAAA,CACjF,MAAA,CAAO,OAAO,OAAe,OAAA,KAMxB;AAEJ,EAAA,MAAM,YAAA,GAAe,uBAAA,CAAwB,OAAA,CAAQ,KAAK,CAAA;AAE1D,EAAA,MAAM,cAAc,KAAA,EAAO;AAAA,IACzB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,KAAK,OAAA,CAAQ,GAAA;AAAA,IACb,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,cAAc,MAAA,CAAO,IAAA,CAAK,YAAY,CAAA,CAAE,MAAA,GAAS,IAAI,YAAA,GAAe;AAAA,GACrE,CAAA;AACH,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,OAAO,CAAA,CACf,WAAA,CAAY,sDAAsD,CAAA,CAClE,MAAA,CAAO,aAAA,EAAe,eAAe,CAAA,CACrC,MAAA,CAAO,mBAAA,EAAqB,oBAAoB,EAChD,MAAA,CAAO,QAAA,EAAU,6CAA6C,CAAA,CAC9D,MAAA,CAAO,YAAA,EAAc,4BAA4B,CAAA,CACjD,OAAO,YAAA,EAAc,4BAA4B,CAAA,CACjD,MAAA,CAAO,QAAA,EAAU,gBAAgB,CAAA,CACjC,MAAA,CAAO,OAAO,OAAA,KAOT;AACJ,EAAA,MAAM,aAAa,OAAO,CAAA;AAC5B,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,gBAAgB,CAAA,CACxB,WAAA,CAAY,kDAAkD,EAC9D,MAAA,CAAO,QAAA,EAAU,8BAA8B,CAAA,CAC/C,OAAO,WAAA,EAAa,oCAAoC,EACxD,MAAA,CAAO,OAAO,UAAkB,OAAA,KAAmD;AAClF,EAAA,MAAM,EAAE,cAAA,EAAe,GAAI,MAAM,OAAO,wBAAqB,CAAA;AAC7D,EAAA,MAAM,cAAA,CAAe,UAAU,OAAO,CAAA;AACxC,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,cAAc,CAAA,CACtB,WAAA,CAAY,0DAA0D,CAAA,CACtE,MAAA,CAAO,uBAAA,EAAyB,qDAAA,EAAuD,cAAA,EAAgB,EAAE,CAAA,CACzG,MAAA,CAAO,eAAA,EAAiB,sCAAsC,CAAA,CAC9D,MAAA,CAAO,WAAA,EAAa,mDAAmD,CAAA,CACvE,MAAA,CAAO,SAAA,EAAW,0BAA0B,CAAA,CAC5C,MAAA,CAAO,OAAO,QAAA,EAAkB,OAAA,KAK3B;AACJ,EAAA,MAAM,EAAE,YAAA,EAAa,GAAI,MAAM,OAAO,wBAAqB,CAAA;AAG3D,EAAA,MAAM,OAAA,GAAU,OAAA,CAAQ,MAAA,CAAO,GAAA,CAAI,CAAC,GAAA,KAAgB;AAClD,IAAA,MAAM,CAAC,IAAA,EAAM,KAAK,CAAA,GAAI,GAAA,CAAI,MAAM,GAAG,CAAA;AACnC,IAAA,IAAI,CAAC,IAAA,IAAQ,CAAC,KAAA,EAAO;AACnB,MAAA,MAAM,IAAI,KAAA,CAAM,CAAA,yBAAA,EAA4B,GAAG,CAAA,yBAAA,CAA2B,CAAA;AAAA,IAC5E;AACA,IAAA,OAAO,EAAE,MAAM,KAAA,EAAM;AAAA,EACvB,CAAC,CAAA;AAED,EAAA,MAAM,aAAa,QAAA,EAAU;AAAA,IAC3B,OAAA;AAAA,IACA,YAAY,OAAA,CAAQ,UAAA;AAAA,IACpB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,OAAO,OAAA,CAAQ;AAAA,GAChB,CAAA;AACH,CAAC,CAAA;AAGH,SAAS,cAAA,CAAe,OAAe,QAAA,EAA8B;AACnE,EAAA,OAAO,QAAA,CAAS,MAAA,CAAO,CAAC,KAAK,CAAC,CAAA;AAChC;AAGA,SAAS,cAAA,CAAe,OAAe,QAAA,EAA8B;AACnE,EAAA,OAAO,QAAA,CAAS,MAAA,CAAO,CAAC,KAAK,CAAC,CAAA;AAChC;AAGA,OAAA,CACG,OAAA,CAAQ,gBAAgB,CAAA,CACxB,WAAA,CAAY,mDAAmD,EAC/D,MAAA,CAAO,kBAAA,EAAoB,sCAAA,EAAwC,cAAA,EAAgB,EAAE,EACrF,MAAA,CAAO,WAAA,EAAa,mDAAmD,CAAA,CACvE,MAAA,CAAO,SAAA,EAAW,mBAAmB,CAAA,CACrC,MAAA,CAAO,OAAO,QAAA,EAAkB,OAAA,KAI3B;AACJ,EAAA,MAAM,EAAE,cAAA,EAAe,GAAI,MAAM,OAAO,wBAAqB,CAAA;AAE7D,EAAA,MAAM,eAAe,QAAA,EAAU;AAAA,IAC7B,SAAS,OAAA,CAAQ,MAAA;AAAA,IACjB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,OAAO,OAAA,CAAQ;AAAA,GAChB,CAAA;AACH,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,eAAe,CAAA,CACvB,WAAA,CAAY,oDAAoD,CAAA,CAChE,MAAA,CAAO,YAAA,EAAc,mDAAmD,CAAA,CACxE,MAAA,CAAO,qBAAA,EAAuB,4CAA4C,CAAA,CAC1E,MAAA,CAAO,OAAA,EAAS,yCAAyC,CAAA,CACzD,MAAA,CAAO,mBAAA,EAAqB,gDAAgD,CAAA,CAC5E,MAAA,CAAO,QAAA,EAAU,gBAAgB,CAAA,CACjC,MAAA,CAAO,OAAO,UAA8B,OAAA,KAMvC;AACJ,EAAA,MAAM,EAAE,aAAA,EAAe,gBAAA,EAAiB,GAAI,MAAM,OAAO,wBAAqB,CAAA;AAE9E,EAAA,IAAI,QAAA,EAAU;AAEZ,IAAA,MAAM,aAAA,CAAc,UAAU,OAAO,CAAA;AAAA,EACvC,CAAA,MAAO;AAEL,IAAA,MAAM,iBAAiB,OAAO,CAAA;AAAA,EAChC;AACF,CAAC,CAAA;AAGH,IAAM,eAAe,OAAA,CAClB,OAAA,CAAQ,WAAW,CAAA,CACnB,YAAY,uBAAuB,CAAA;AAEtC,YAAA,CACG,QAAQ,MAAM,CAAA,CACd,YAAY,0BAA0B,CAAA,CACtC,OAAO,YAAY;AAClB,EAAA,MAAM,aAAA,EAAc;AACtB,CAAC,CAAA;AAEH,YAAA,CACG,OAAA,CAAQ,aAAa,CAAA,CACrB,WAAA,CAAY,uBAAuB,CAAA,CACnC,MAAA,CAAO,OAAO,IAAA,KAAiB;AAC9B,EAAA,MAAM,aAAa,IAAI,CAAA;AACzB,CAAC,CAAA;AAEH,YAAA,CACG,OAAA,CAAQ,mBAAmB,CAAA,CAC3B,WAAA,CAAY,qBAAqB,CAAA,CACjC,MAAA,CAAO,OAAO,IAAA,EAAc,IAAA,KAAiB;AAC5C,EAAA,MAAM,WAAA,CAAY,MAAM,IAAI,CAAA;AAC9B,CAAC,CAAA;AAEH,YAAA,CACG,OAAA,CAAQ,eAAe,CAAA,CACvB,WAAA,CAAY,uBAAuB,CAAA,CACnC,MAAA,CAAO,OAAO,IAAA,KAAiB;AAC9B,EAAA,MAAM,eAAe,IAAI,CAAA;AAC3B,CAAC,CAAA;AAEH,YAAA,CACG,OAAA,CAAQ,wBAAwB,CAAA,CAChC,WAAA,CAAY,qCAAqC,CAAA,CACjD,MAAA,CAAO,OAAO,MAAA,EAAgB,MAAA,KAAmB;AAChD,EAAA,MAAM,YAAA,CAAa,QAAQ,MAAM,CAAA;AACnC,CAAC,CAAA;AAGH,YAAA,CACG,OAAO,YAAY;AAClB,EAAA,MAAM,aAAA,EAAc;AACtB,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,UAAU,CAAA,CAClB,WAAA,CAAY,oCAAoC,CAAA,CAChD,MAAA,CAAO,cAAc,gCAAA,EAAkC,QAAQ,EAC/D,MAAA,CAAO,UAAA,EAAY,cAAc,CAAA,CACjC,MAAA,CAAO,iBAAiB,mBAAmB,CAAA,CAC3C,MAAA,CAAO,OAAO,OAAA,KAIT;AACJ,EAAA,MAAM,gBAAgB,OAAO,CAAA;AAC/B,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,eAAe,CAAA,CACvB,WAAA,CAAY,sBAAsB,CAAA,CAClC,MAAA,CAAO,mBAAA,EAAqB,uDAAuD,CAAA,CACnF,MAAA,CAAO,uBAAA,EAAyB,4CAA4C,CAAA,CAC5E,MAAA,CAAO,eAAA,EAAiB,4BAA4B,CAAA,CACpD,MAAA,CAAO,mBAAA,EAAqB,cAAc,CAAA,CAC1C,MAAA,CAAO,yBAAA,EAA2B,yCAAyC,CAAA,CAC3E,MAAA,CAAO,OAAO,UAAkB,OAAA,KAM3B;AAEJ,EAAA,MAAM,YAAA,GAAe,uBAAA,CAAwB,OAAA,CAAQ,KAAK,CAAA;AAE1D,EAAA,MAAM,OAAA,GAMF;AAAA,IACF,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,IAAA,EAAM,OAAA,CAAQ,IAAA,GAAO,OAAA,CAAQ,IAAA,CAAK,KAAA,CAAM,GAAG,CAAA,CAAE,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,IAAA,EAAM,CAAA,GAAI,MAAA;AAAA,IAClE,UAAU,OAAA,CAAQ,QAAA;AAAA,IAClB,cAAc,MAAA,CAAO,IAAA,CAAK,YAAY,CAAA,CAAE,MAAA,GAAS,IAAI,YAAA,GAAe;AAAA,GACtE;AAGA,EAAA,MAAA,CAAO,IAAA,CAAK,OAAO,CAAA,CAAE,OAAA,CAAQ,CAAA,GAAA,KAAO;AAClC,IAAA,IAAI,OAAA,CAAQ,GAA2B,CAAA,KAAM,MAAA,EAAW;AACtD,MAAA,OAAO,QAAQ,GAA2B,CAAA;AAAA,IAC5C;AAAA,EACF,CAAC,CAAA;AAED,EAAA,IAAI,MAAA,CAAO,IAAA,CAAK,OAAO,CAAA,CAAE,WAAW,CAAA,EAAG;AACrC,IAAA,OAAA,CAAQ,MAAM,gGAAgG,CAAA;AAC9G,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAChB;AAEA,EAAA,MAAM,UAAA,CAAW,UAAU,OAAO,CAAA;AACpC,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,aAAa,CAAA,CACrB,WAAA,CAAY,6DAA6D,CAAA,CACzE,MAAA,CAAO,SAAS,4CAA4C,CAAA,CAC5D,OAAO,QAAA,EAAU,gBAAgB,EACjC,MAAA,CAAO,YAAA,EAAc,gBAAgB,CAAA,CACrC,MAAA,CAAO,OAAO,QAAA,EAAkB,OAAA,KAI3B;AACJ,EAAA,IAAI;AACF,IAAA,MAAM,YAAY,QAAA,EAAU;AAAA,MAC1B,KAAK,OAAA,CAAQ,GAAA;AAAA,MACb,MAAM,OAAA,CAAQ,IAAA;AAAA,MACd,OAAA,EAAS,QAAQ,KAAA,KAAU;AAAA,KAC5B,CAAA;AAAA,EACH,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,KAAA,CAAM,yBAAyB,KAAA,YAAiB,KAAA,GAAQ,MAAM,OAAA,GAAU,MAAA,CAAO,KAAK,CAAC,CAAA;AAC7F,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAChB;AACF,CAAC,CAAA;AAGH,OAAA,CACG,OAAA,CAAQ,sBAAsB,CAAA,CAC9B,WAAA,CAAY,yEAAyE,CAAA,CACrF,MAAA,CAAO,mBAAA,EAAqB,iDAAiD,CAAA,CAC7E,MAAA,CAAO,WAAA,EAAa,gCAAgC,CAAA,CACpD,MAAA,CAAO,kBAAA,EAAoB,0BAAA,EAA4B,QAAQ,CAAA,CAC/D,MAAA,CAAO,mBAAA,EAAqB,gCAAgC,CAAA,CAC5D,MAAA,CAAO,YAAA,EAAc,mCAAmC,CAAA,CACxD,MAAA,CAAO,OAAO,WAAmB,OAAA,KAM5B;AAEJ,EAAA,IAAI,OAAA,CAAQ,IAAA,IAAQ,CAAC,CAAC,SAAA,EAAW,QAAA,EAAU,QAAQ,CAAA,CAAE,QAAA,CAAS,OAAA,CAAQ,IAAI,CAAA,EAAG;AAC3E,IAAA,OAAA,CAAQ,MAAM,mFAA8E,CAAA;AAC5F,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAChB;AAEA,EAAA,MAAM,eAAe,SAAA,EAAW;AAAA,IAC9B,YAAY,OAAA,CAAQ,IAAA;AAAA,IACpB,QAAQ,OAAA,CAAQ,MAAA;AAAA,IAChB,WAAW,OAAA,CAAQ,SAAA;AAAA,IACnB,gBAAgB,OAAA,CAAQ,cAAA;AAAA,IACxB,UAAU,OAAA,CAAQ;AAAA,GACnB,CAAA;AACH,CAAC,CAAA;AAGH,OAAA,CACG,QAAQ,KAAK,CAAA,CACb,YAAY,kEAAkE,CAAA,CAC9E,OAAO,YAAY;AAClB,EAAA,MAAM,UAAA,EAAW;AACnB,CAAC,CAAA;AAGH,OAAA,CAAQ,KAAA,EAAM","file":"cli.js","sourcesContent":["/**\n * Parse custom fields from CLI --field options\n * @param fieldOptions Array of \"name=value\" strings\n * @returns Record of parsed field names and values\n */\nexport function parseCustomFieldOptions(fieldOptions?: string[]): Record<string, unknown> {\n const customFields: Record<string, unknown> = {};\n \n if (!fieldOptions) {\n return customFields;\n }\n \n for (const field of fieldOptions) {\n const [key, ...valueParts] = field.split('=');\n if (key && valueParts.length > 0) {\n const value = valueParts.join('='); // Handle values with '=' in them\n customFields[key.trim()] = value.trim();\n }\n }\n \n return customFields;\n}\n","import { Command } from 'commander';\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\nimport { dirname, join } from 'path';\nimport {\n createSpec,\n archiveSpec,\n listSpecs,\n updateSpec,\n checkSpecs,\n backfillTimestamps,\n listTemplates,\n showTemplate,\n addTemplate,\n removeTemplate,\n copyTemplate,\n initProject,\n statsCommand,\n boardCommand,\n timelineCommand,\n depsCommand,\n searchCommand,\n ganttCommand,\n filesCommand,\n viewCommand,\n openCommand,\n validateCommand,\n mcpCommand,\n migrateCommand,\n} from './commands/index.js';\nimport { parseCustomFieldOptions } from './utils/cli-helpers.js';\nimport type { SpecStatus, SpecPriority } from './frontmatter.js';\n\n// Get version from package.json\nconst __filename = fileURLToPath(import.meta.url);\nconst __dirname = dirname(__filename);\nconst packageJson = JSON.parse(\n readFileSync(join(__dirname, '../package.json'), 'utf-8')\n);\n\nconst program = new Command();\n\nprogram\n .name('lean-spec')\n .description('Manage LeanSpec documents')\n .version(packageJson.version);\n\n// Add custom help text with grouped commands\nprogram.addHelpText('after', `\nCommand Groups:\n \n Core Commands:\n init Initialize LeanSpec in current directory\n create <name> Create new spec in folder structure\n list List all specs\n update <spec> Update spec metadata\n archive <spec> Move spec to archived/\n backfill [specs...] Backfill timestamps from git history\n migrate <input-path> Migrate specs from other SDD tools\n \n Viewing & Navigation:\n view <spec> View spec content\n open <spec> Open spec in editor\n search <query> Full-text search with metadata filters\n files <spec> List files in a spec\n \n Project & Analytics:\n board Show Kanban-style board view\n stats Show aggregate statistics\n timeline Show creation/completion over time\n gantt Show timeline with dependencies\n deps <spec> Show dependency graph for a spec\n \n Maintenance:\n check Check for sequence conflicts\n validate [specs...] Validate specs for quality issues\n templates Manage spec templates\n \n Server:\n mcp Start MCP server for AI assistants\n\nExamples:\n $ lean-spec init\n $ lean-spec create my-feature --priority high\n $ lean-spec list --status in-progress\n $ lean-spec view 042\n $ lean-spec backfill --dry-run\n $ lean-spec migrate ./docs/adr\n $ lean-spec migrate ./docs/rfcs --with copilot\n $ lean-spec board --tag backend\n $ lean-spec search \"authentication\"\n $ lean-spec validate\n $ lean-spec validate --verbose\n $ lean-spec validate --quiet --rule max-lines\n $ lean-spec validate 018 --max-lines 500\n`);\n\n// archive command\nprogram\n .command('archive <spec>')\n .description('Move spec to archived/')\n .action(async (specPath: string) => {\n await archiveSpec(specPath);\n });\n\n// backfill command\nprogram\n .command('backfill [specs...]')\n .description('Backfill timestamps from git history')\n .option('--dry-run', 'Show what would be updated without making changes')\n .option('--force', 'Overwrite existing timestamp values')\n .option('--assignee', 'Include assignee from first commit author')\n .option('--transitions', 'Include full status transition history')\n .option('--all', 'Include all optional fields (assignee + transitions)')\n .action(async (specs: string[] | undefined, options: {\n dryRun?: boolean;\n force?: boolean;\n assignee?: boolean;\n transitions?: boolean;\n all?: boolean;\n }) => {\n await backfillTimestamps({\n dryRun: options.dryRun,\n force: options.force,\n includeAssignee: options.assignee || options.all,\n includeTransitions: options.transitions || options.all,\n specs: specs && specs.length > 0 ? specs : undefined,\n });\n });\n\n// board command\nprogram\n .command('board')\n .description('Show Kanban-style board view with project completion summary')\n .option('--complete', 'Include complete specs (default: hidden)')\n .option('--simple', 'Hide completion summary (kanban only)')\n .option('--completion-only', 'Show only completion summary (no kanban)')\n .option('--tag <tag>', 'Filter by tag')\n .option('--assignee <name>', 'Filter by assignee')\n .action(async (options: {\n showComplete?: boolean;\n simple?: boolean;\n completionOnly?: boolean;\n tag?: string;\n assignee?: string;\n }) => {\n await boardCommand(options);\n });\n\n// check command\nprogram\n .command('check')\n .description('Check for sequence conflicts')\n .option('-q, --quiet', 'Brief output')\n .action(async (options: {\n quiet?: boolean;\n }) => {\n const hasNoConflicts = await checkSpecs(options);\n // Exit with 0 (success) if no conflicts, 1 (error) if conflicts found\n process.exit(hasNoConflicts ? 0 : 1);\n });\n\n// validate command\nprogram\n .command('validate [specs...]')\n .description('Validate specs for quality issues')\n .option('--max-lines <number>', 'Custom line limit (default: 400)', parseInt)\n .option('--verbose', 'Show passing specs')\n .option('--quiet', 'Suppress warnings, only show errors')\n .option('--format <format>', 'Output format: default, json, compact', 'default')\n .option('--rule <rule>', 'Filter by specific rule name (e.g., max-lines, frontmatter)')\n .action(async (specs: string[] | undefined, options: {\n maxLines?: number;\n verbose?: boolean;\n quiet?: boolean;\n format?: 'default' | 'json' | 'compact';\n rule?: string;\n }) => {\n const passed = await validateCommand({\n maxLines: options.maxLines,\n specs: specs && specs.length > 0 ? specs : undefined,\n verbose: options.verbose,\n quiet: options.quiet,\n format: options.format,\n rule: options.rule,\n });\n // Exit with 0 (success) if all passed, 1 (error) if any failed\n process.exit(passed ? 0 : 1);\n });\n\n// create command\nprogram\n .command('create <name>')\n .description('Create new spec in folder structure')\n .option('--title <title>', 'Set custom title')\n .option('--description <desc>', 'Set initial description')\n .option('--tags <tags>', 'Set tags (comma-separated)')\n .option('--priority <priority>', 'Set priority (low, medium, high, critical)')\n .option('--assignee <name>', 'Set assignee')\n .option('--template <template>', 'Use a specific template')\n .option('--field <name=value...>', 'Set custom field (can specify multiple)')\n .option('--no-prefix', 'Skip date prefix even if configured')\n .action(async (name: string, options: {\n title?: string;\n description?: string;\n tags?: string;\n priority?: SpecPriority;\n assignee?: string;\n template?: string;\n field?: string[];\n prefix?: boolean;\n }) => {\n // Parse custom fields from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const createOptions: {\n title?: string;\n description?: string;\n tags?: string[];\n priority?: SpecPriority;\n assignee?: string;\n template?: string;\n customFields?: Record<string, unknown>;\n noPrefix?: boolean;\n } = {\n title: options.title,\n description: options.description,\n tags: options.tags ? options.tags.split(',').map(t => t.trim()) : undefined,\n priority: options.priority,\n assignee: options.assignee,\n template: options.template,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n noPrefix: options.prefix === false,\n };\n await createSpec(name, createOptions);\n });\n\n// deps command\nprogram\n .command('deps <spec>')\n .description('Show dependency graph for a spec. Related specs (⟷) are shown bidirectionally, depends_on (→) are directional.')\n .option('--depth <n>', 'Show N levels deep (default: 3)', parseInt)\n .option('--graph', 'ASCII graph visualization')\n .option('--json', 'Output as JSON')\n .action(async (specPath: string, options: {\n depth?: number;\n graph?: boolean;\n json?: boolean;\n }) => {\n await depsCommand(specPath, options);\n });\n\n// files command\nprogram\n .command('files <spec>')\n .description('List files in a spec')\n .option('--type <type>', 'Filter by type: docs, assets')\n .option('--tree', 'Show tree structure')\n .action(async (specPath: string, options: {\n type?: 'docs' | 'assets';\n tree?: boolean;\n }) => {\n await filesCommand(specPath, options);\n });\n\n// gantt command\nprogram\n .command('gantt')\n .description('Show timeline with dependencies')\n .option('--weeks <n>', 'Show N weeks (default: 4)', parseInt)\n .option('--show-complete', 'Include completed specs')\n .option('--critical-path', 'Highlight critical path')\n .action(async (options: {\n weeks?: number;\n showComplete?: boolean;\n criticalPath?: boolean;\n }) => {\n await ganttCommand(options);\n });\n\n// init command\nprogram\n .command('init')\n .description('Initialize LeanSpec in current directory')\n .action(async () => {\n await initProject();\n });\n\n// list command\nprogram\n .command('list')\n .description('List all specs')\n .option('--archived', 'Include archived specs')\n .option('--status <status>', 'Filter by status (planned, in-progress, complete, archived)')\n .option('--tag <tag...>', 'Filter by tag (can specify multiple)')\n .option('--priority <priority>', 'Filter by priority (low, medium, high, critical)')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--field <name=value...>', 'Filter by custom field (can specify multiple)')\n .option('--sort <field>', 'Sort by field (id, created, name, status, priority)', 'id')\n .option('--order <order>', 'Sort order (asc, desc)', 'desc')\n .action(async (options: {\n archived?: boolean;\n status?: SpecStatus;\n tag?: string[];\n priority?: SpecPriority;\n assignee?: string;\n field?: string[];\n sort?: string;\n order?: string;\n }) => {\n // Parse custom field filters from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const listOptions: {\n showArchived?: boolean;\n status?: SpecStatus;\n tags?: string[];\n priority?: SpecPriority;\n assignee?: string;\n customFields?: Record<string, unknown>;\n sortBy?: string;\n sortOrder?: 'asc' | 'desc';\n } = {\n showArchived: options.archived,\n status: options.status,\n tags: options.tag,\n priority: options.priority,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n sortBy: options.sort || 'id',\n sortOrder: (options.order as 'asc' | 'desc') || 'desc',\n };\n await listSpecs(listOptions);\n });\n\n// open command\nprogram\n .command('open <spec>')\n .description('Open spec in editor')\n .option('--editor <editor>', 'Specify editor command')\n .action(async (specPath: string, options: {\n editor?: string;\n }) => {\n try {\n await openCommand(specPath, {\n editor: options.editor,\n });\n } catch (error) {\n console.error('\\x1b[31mError:\\x1b[0m', error instanceof Error ? error.message : String(error));\n process.exit(1);\n }\n });\n\n// search command\nprogram\n .command('search <query>')\n .description('Full-text search with metadata filters')\n .option('--status <status>', 'Filter by status')\n .option('--tag <tag>', 'Filter by tag')\n .option('--priority <priority>', 'Filter by priority')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--field <name=value...>', 'Filter by custom field (can specify multiple)')\n .action(async (query: string, options: {\n status?: SpecStatus;\n tag?: string;\n priority?: SpecPriority;\n assignee?: string;\n field?: string[];\n }) => {\n // Parse custom field filters from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n await searchCommand(query, {\n status: options.status,\n tag: options.tag,\n priority: options.priority,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n });\n });\n\n// stats command\nprogram\n .command('stats')\n .description('Show aggregate statistics (default: simplified view)')\n .option('--tag <tag>', 'Filter by tag')\n .option('--assignee <name>', 'Filter by assignee')\n .option('--full', 'Show full detailed analytics (all sections)')\n .option('--timeline', 'Show only timeline section')\n .option('--velocity', 'Show only velocity section')\n .option('--json', 'Output as JSON')\n .action(async (options: {\n tag?: string;\n assignee?: string;\n full?: boolean;\n timeline?: boolean;\n velocity?: boolean;\n json?: boolean;\n }) => {\n await statsCommand(options);\n });\n\n// analyze command - for spec 059\nprogram\n .command('analyze <spec>')\n .description('Analyze spec complexity and structure (spec 059)')\n .option('--json', 'Output as JSON for AI agents')\n .option('--verbose', 'Include detailed section breakdown')\n .action(async (specPath: string, options: { json?: boolean; verbose?: boolean }) => {\n const { analyzeCommand } = await import('./commands/index.js');\n await analyzeCommand(specPath, options);\n });\n\n// split command - for spec 059\nprogram\n .command('split <spec>')\n .description('Split spec into multiple files by line ranges (spec 059)')\n .option('--output <file:lines>', 'Output file with line range (e.g., README.md:1-150)', collectOutputs, [])\n .option('--update-refs', 'Update cross-references in README.md')\n .option('--dry-run', 'Show what would be created without making changes')\n .option('--force', 'Overwrite existing files')\n .action(async (specPath: string, options: { \n output: Array<string>;\n updateRefs?: boolean;\n dryRun?: boolean;\n force?: boolean;\n }) => {\n const { splitCommand } = await import('./commands/index.js');\n \n // Parse outputs into structured format\n const outputs = options.output.map((opt: string) => {\n const [file, lines] = opt.split(':');\n if (!file || !lines) {\n throw new Error(`Invalid --output format: ${opt}. Expected: file.md:1-150`);\n }\n return { file, lines };\n });\n \n await splitCommand(specPath, {\n outputs,\n updateRefs: options.updateRefs,\n dryRun: options.dryRun,\n force: options.force,\n });\n });\n\n// Helper function to collect multiple --output options\nfunction collectOutputs(value: string, previous: string[]): string[] {\n return previous.concat([value]);\n}\n\n// Helper function to collect multiple --remove options\nfunction collectRemoves(value: string, previous: string[]): string[] {\n return previous.concat([value]);\n}\n\n// compact command - for spec 059\nprogram\n .command('compact <spec>')\n .description('Remove specified line ranges from spec (spec 059)')\n .option('--remove <lines>', 'Line range to remove (e.g., 145-153)', collectRemoves, [])\n .option('--dry-run', 'Show what would be removed without making changes')\n .option('--force', 'Skip confirmation')\n .action(async (specPath: string, options: { \n remove: string[];\n dryRun?: boolean;\n force?: boolean;\n }) => {\n const { compactCommand } = await import('./commands/index.js');\n \n await compactCommand(specPath, {\n removes: options.remove,\n dryRun: options.dryRun,\n force: options.force,\n });\n });\n\n// tokens command\nprogram\n .command('tokens [spec]')\n .description('Count tokens in spec(s) for LLM context management')\n .option('--detailed', 'Show content type breakdown (code, prose, tables)')\n .option('--include-sub-specs', 'Count all sub-spec files (DESIGN.md, etc.)')\n .option('--all', 'Show all specs (when [spec] is omitted)')\n .option('--sort-by <field>', 'Sort by: tokens, lines, name (default: tokens)')\n .option('--json', 'Output as JSON')\n .action(async (specPath: string | undefined, options: {\n detailed?: boolean;\n includeSubSpecs?: boolean;\n all?: boolean;\n sortBy?: 'tokens' | 'lines' | 'name';\n json?: boolean;\n }) => {\n const { tokensCommand, tokensAllCommand } = await import('./commands/index.js');\n \n if (specPath) {\n // Count specific spec\n await tokensCommand(specPath, options);\n } else {\n // Count all specs\n await tokensAllCommand(options);\n }\n });\n\n// templates command and subcommands\nconst templatesCmd = program\n .command('templates')\n .description('Manage spec templates');\n\ntemplatesCmd\n .command('list')\n .description('List available templates')\n .action(async () => {\n await listTemplates();\n });\n\ntemplatesCmd\n .command('show <name>')\n .description('Show template content')\n .action(async (name: string) => {\n await showTemplate(name);\n });\n\ntemplatesCmd\n .command('add <name> <file>')\n .description('Register a template')\n .action(async (name: string, file: string) => {\n await addTemplate(name, file);\n });\n\ntemplatesCmd\n .command('remove <name>')\n .description('Unregister a template')\n .action(async (name: string) => {\n await removeTemplate(name);\n });\n\ntemplatesCmd\n .command('copy <source> <target>')\n .description('Copy a template to create a new one')\n .action(async (source: string, target: string) => {\n await copyTemplate(source, target);\n });\n\n// Default action for templates (list)\ntemplatesCmd\n .action(async () => {\n await listTemplates();\n });\n\n// timeline command\nprogram\n .command('timeline')\n .description('Show creation/completion over time')\n .option('--days <n>', 'Show last N days (default: 30)', parseInt)\n .option('--by-tag', 'Group by tag')\n .option('--by-assignee', 'Group by assignee')\n .action(async (options: {\n days?: number;\n byTag?: boolean;\n byAssignee?: boolean;\n }) => {\n await timelineCommand(options);\n });\n\n// update command\nprogram\n .command('update <spec>')\n .description('Update spec metadata')\n .option('--status <status>', 'Set status (planned, in-progress, complete, archived)')\n .option('--priority <priority>', 'Set priority (low, medium, high, critical)')\n .option('--tags <tags>', 'Set tags (comma-separated)')\n .option('--assignee <name>', 'Set assignee')\n .option('--field <name=value...>', 'Set custom field (can specify multiple)')\n .action(async (specPath: string, options: {\n status?: SpecStatus;\n priority?: SpecPriority;\n tags?: string;\n assignee?: string;\n field?: string[];\n }) => {\n // Parse custom fields from --field options\n const customFields = parseCustomFieldOptions(options.field);\n \n const updates: {\n status?: SpecStatus;\n priority?: SpecPriority;\n tags?: string[];\n assignee?: string;\n customFields?: Record<string, unknown>;\n } = {\n status: options.status,\n priority: options.priority,\n tags: options.tags ? options.tags.split(',').map(t => t.trim()) : undefined,\n assignee: options.assignee,\n customFields: Object.keys(customFields).length > 0 ? customFields : undefined,\n };\n \n // Filter out undefined values\n Object.keys(updates).forEach(key => {\n if (updates[key as keyof typeof updates] === undefined) {\n delete updates[key as keyof typeof updates];\n }\n });\n \n if (Object.keys(updates).length === 0) {\n console.error('Error: At least one update option required (--status, --priority, --tags, --assignee, --field)');\n process.exit(1);\n }\n \n await updateSpec(specPath, updates);\n });\n\n// view command (primary viewer)\nprogram\n .command('view <spec>')\n .description('View spec content (supports sub-specs like \"045/DESIGN.md\")')\n .option('--raw', 'Output raw markdown (for piping/scripting)')\n .option('--json', 'Output as JSON')\n .option('--no-color', 'Disable colors')\n .action(async (specPath: string, options: {\n raw?: boolean;\n json?: boolean;\n color?: boolean;\n }) => {\n try {\n await viewCommand(specPath, {\n raw: options.raw,\n json: options.json,\n noColor: options.color === false,\n });\n } catch (error) {\n console.error('\\x1b[31mError:\\x1b[0m', error instanceof Error ? error.message : String(error));\n process.exit(1);\n }\n });\n\n// migrate command\nprogram\n .command('migrate <input-path>')\n .description('Migrate specs from other SDD tools (ADR, RFC, OpenSpec, spec-kit, etc.)')\n .option('--with <provider>', 'AI-assisted migration (copilot, claude, gemini)')\n .option('--dry-run', 'Preview without making changes')\n .option('--batch-size <n>', 'Process N docs at a time', parseInt)\n .option('--skip-validation', \"Don't validate after migration\")\n .option('--backfill', 'Auto-run backfill after migration')\n .action(async (inputPath: string, options: {\n with?: string;\n dryRun?: boolean;\n batchSize?: number;\n skipValidation?: boolean;\n backfill?: boolean;\n }) => {\n // Validate AI provider if specified\n if (options.with && !['copilot', 'claude', 'gemini'].includes(options.with)) {\n console.error('\\x1b[31m❌ Error:\\x1b[0m Invalid AI provider. Use: copilot, claude, or gemini');\n process.exit(1);\n }\n \n await migrateCommand(inputPath, {\n aiProvider: options.with as 'copilot' | 'claude' | 'gemini' | undefined,\n dryRun: options.dryRun,\n batchSize: options.batchSize,\n skipValidation: options.skipValidation,\n backfill: options.backfill,\n });\n });\n\n// mcp command\nprogram\n .command('mcp')\n .description('Start MCP server for AI assistants (Claude Desktop, Cline, etc.)')\n .action(async () => {\n await mcpCommand();\n });\n\n// Parse and execute\nprogram.parse();\n"]}
@@ -0,0 +1,4 @@
1
+ export { addTemplate, analyzeCommand, archiveSpec, backfillTimestamps, boardCommand, checkSpecs, compactCommand, copyTemplate, createSpec, depsCommand, filesCommand, ganttCommand, initProject, listSpecs, listTemplates, mcpCommand, migrateCommand, openCommand, removeTemplate, searchCommand, showTemplate, splitCommand, statsCommand, timelineCommand, tokensAllCommand, tokensCommand, updateSpec, validateCommand, viewCommand } from './chunk-7MCDTSVE.js';
2
+ import './chunk-LVD7ZAVZ.js';
3
+ //# sourceMappingURL=commands-GRG5UUOF.js.map
4
+ //# sourceMappingURL=commands-GRG5UUOF.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"commands-ZNL7ZCHU.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"commands-GRG5UUOF.js"}
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env node
2
- export { createMcpServer } from './chunk-ER23B6KS.js';
2
+ export { createMcpServer } from './chunk-7MCDTSVE.js';
3
3
  import './chunk-LVD7ZAVZ.js';
4
4
  //# sourceMappingURL=mcp-server.js.map
5
5
  //# sourceMappingURL=mcp-server.js.map
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "lean-spec",
3
- "version": "0.2.1",
3
+ "version": "0.2.2",
4
4
  "description": "Lightweight spec methodology for AI-powered development",
5
5
  "type": "module",
6
6
  "bin": {
@@ -40,7 +40,6 @@
40
40
  "CHANGELOG.md"
41
41
  ],
42
42
  "dependencies": {
43
- "@leanspec/core": "workspace:*",
44
43
  "@inquirer/prompts": "^7.9.0",
45
44
  "@modelcontextprotocol/sdk": "^1.21.0",
46
45
  "chalk": "^5.3.0",
@@ -49,8 +48,8 @@
49
48
  "commander": "^14.0.2",
50
49
  "dayjs": "^1.11.19",
51
50
  "gray-matter": "^4.0.3",
51
+ "handlebars": "^4.7.8",
52
52
  "ink": "^6.4.0",
53
- "ink-box": "^2.0.0",
54
53
  "ink-gradient": "^3.0.0",
55
54
  "ink-progress-bar": "^3.0.0",
56
55
  "ink-select-input": "^6.2.0",
@@ -63,10 +62,13 @@
63
62
  "ora": "^9.0.0",
64
63
  "react": "^19.2.0",
65
64
  "strip-ansi": "^7.1.2",
65
+ "tiktoken": "^1.0.22",
66
66
  "tokenx": "^1.2.1",
67
67
  "zod": "^3.25.76"
68
68
  },
69
69
  "devDependencies": {
70
+ "@leanspec/core": "workspace:*",
71
+ "@types/handlebars": "^4.1.0",
70
72
  "@types/js-yaml": "^4.0.9",
71
73
  "@types/marked-terminal": "^6.1.1",
72
74
  "@types/node": "^20.10.0",
@@ -0,0 +1,5 @@
1
+ 1. **Read README.md first** - Understand project context
2
+ 2. **Check specs/** - Review existing specs before starting
3
+ 3. **Follow LeanSpec principles** - Clarity over documentation
4
+ 4. **Keep it minimal** - If it doesn't add clarity, cut it
5
+ 5. **Never use nested code blocks** - Markdown doesn't support code blocks within code blocks. Use indentation or describe the structure instead
@@ -0,0 +1,5 @@
1
+ 1. **Security first** - Follow security standards, no shortcuts
2
+ 2. **Read specs/** - Check for compliance requirements
3
+ 3. **Follow LeanSpec** - Lean doesn't mean skipping governance
4
+ 4. **Document decisions** - Especially security and compliance choices
5
+ 5. **Never use nested code blocks** - Markdown doesn't support code blocks within code blocks. Use indentation or describe the structure instead
@@ -0,0 +1,10 @@
1
+ Before starting work, understand project context and dependencies:
2
+
3
+ - `lean-spec stats` - See work distribution across specs
4
+ - `lean-spec board` - View specs organized by status
5
+ - `lean-spec list --tag=<tag>` - Find specs by tag (e.g., `--tag=security`)
6
+ - `lean-spec search "<query>"` - Full-text search across specs
7
+ - `lean-spec deps <spec>` - Check dependencies before starting work
8
+ - `lean-spec gantt` - View project timeline and milestones
9
+
10
+ These commands help you understand what exists, what's in progress, and what depends on what.
@@ -0,0 +1,8 @@
1
+ Before starting work, understand project context:
2
+
3
+ - `lean-spec stats` - See work distribution
4
+ - `lean-spec board` - View specs by status
5
+ - `lean-spec search "<query>"` - Find relevant work
6
+ - `lean-spec list` - List all specs
7
+
8
+ These help you understand what exists and what's in progress.
@@ -0,0 +1,9 @@
1
+ Before starting work, understand project context:
2
+
3
+ - `lean-spec stats` - See work distribution across specs
4
+ - `lean-spec board` - View specs organized by status
5
+ - `lean-spec list --tag=<tag>` - Find specs by tag (e.g., `--tag=api`)
6
+ - `lean-spec search "<query>"` - Full-text search across specs
7
+ - `lean-spec deps <spec>` - Check dependencies before starting work
8
+
9
+ These commands help you understand what exists, what's in progress, and what depends on what.
@@ -0,0 +1,10 @@
1
+ ## Approval Workflow
2
+
3
+ 1. **Discover context** - Run `lean-spec stats`, `lean-spec board`, `lean-spec deps`
4
+ 2. **Create spec** - Include all required frontmatter fields
5
+ 3. **Technical review** - Assign reviewer in frontmatter
6
+ 4. **Security team review** - For security/compliance-tagged specs
7
+ 5. **Stakeholder sign-off** - Update status to `planned`
8
+ 6. **Implementation** - Update status to `in-progress`, keep spec in sync
9
+ 7. **Final review** - Before deployment
10
+ 8. **Complete & archive** - Mark `complete`, then archive
@@ -0,0 +1,12 @@
1
+ ## Security & Compliance in Frontmatter
2
+
3
+ Tag specs with relevant compliance standards:
4
+ ```yaml
5
+ tags: [security, api, pii]
6
+ compliance: [SOC2, GDPR]
7
+ ```
8
+
9
+ This helps with:
10
+ - Compliance audits and reporting
11
+ - Security review prioritization
12
+ - Regulatory requirement tracking
@@ -0,0 +1,13 @@
1
+ ## When Specs Are Required
2
+
3
+ Required for:
4
+ - Features touching PII or sensitive data
5
+ - Changes to authentication/authorization
6
+ - New external integrations
7
+ - Breaking changes to APIs
8
+ - Infrastructure changes
9
+
10
+ Optional for:
11
+ - Internal refactors
12
+ - Bug fixes (unless security-related)
13
+ - Minor UI changes
@@ -0,0 +1,33 @@
1
+ When creating or updating specs, include YAML frontmatter with all relevant fields:
2
+
3
+ ```yaml
4
+ ---
5
+ status: draft|planned|in-progress|complete|blocked|cancelled
6
+ created: YYYY-MM-DD
7
+ tags: [security, api, compliance] # for discovery
8
+ priority: low|medium|high|critical
9
+ assignee: username
10
+ reviewer: reviewer-username # required for review
11
+ issue: JIRA-123 # link to issue tracker
12
+ epic: EPIC-456 # link to epic
13
+ compliance: [SOC2, GDPR, HIPAA] # applicable standards
14
+ depends_on:
15
+ - path/to/other/spec
16
+ ---
17
+ ```
18
+
19
+ **Required fields:**
20
+ - `status`, `created` - basic tracking
21
+ - `tags`, `priority` - planning and discovery
22
+ - `assignee`, `reviewer` - accountability
23
+ - `compliance` - regulatory requirements (if applicable)
24
+
25
+ **Integration fields:**
26
+ - `issue`, `epic` - link to external systems
27
+ - `depends_on` - explicit dependencies
28
+
29
+ **Update with:**
30
+ ```bash
31
+ lean-spec update <spec> --status in-progress --assignee yourname
32
+ # or edit frontmatter directly
33
+ ```
@@ -0,0 +1,18 @@
1
+ When creating or updating specs, add YAML frontmatter at the top:
2
+
3
+ ```yaml
4
+ ---
5
+ status: draft|planned|in-progress|complete|blocked|cancelled
6
+ created: YYYY-MM-DD
7
+ ---
8
+ ```
9
+
10
+ **Keep it simple:**
11
+ - Just `status` and `created` fields
12
+ - Other fields are optional - only add if helpful
13
+ - Update `status` as work progresses
14
+
15
+ **Update status with:**
16
+ ```bash
17
+ lean-spec update <spec> --status in-progress
18
+ ```
@@ -0,0 +1,23 @@
1
+ When creating or updating specs, include YAML frontmatter at the top:
2
+
3
+ ```yaml
4
+ ---
5
+ status: draft|planned|in-progress|complete|blocked|cancelled
6
+ created: YYYY-MM-DD
7
+ tags: [tag1, tag2] # helps with discovery
8
+ priority: low|medium|high # helps with planning
9
+ assignee: username # for team coordination
10
+ ---
11
+ ```
12
+
13
+ **Core fields:**
14
+ - `status` and `created` are required
15
+ - `tags` help with discovery and organization
16
+ - `priority` helps teams plan work
17
+ - `assignee` shows who's working on what
18
+
19
+ **Update status with:**
20
+ ```bash
21
+ lean-spec update <spec> --status in-progress --assignee yourname
22
+ # or edit frontmatter directly
23
+ ```
@@ -0,0 +1,5 @@
1
+ - Code is clear and maintainable
2
+ - Tests cover critical paths
3
+ - No unnecessary complexity
4
+ - Documentation where needed (not everywhere)
5
+ - Specs stay in sync with implementation
@@ -0,0 +1,6 @@
1
+ - Security requirements verified
2
+ - Tests include security scenarios
3
+ - Documentation complete
4
+ - Compliance checklist completed
5
+ - Code is clear and maintainable
6
+ - Specs stay in sync with implementation
@@ -0,0 +1,11 @@
1
+ Required for:
2
+ - Features touching PII or sensitive data
3
+ - Changes to authentication/authorization
4
+ - New external integrations
5
+ - Breaking changes to APIs
6
+ - Infrastructure changes
7
+
8
+ Optional for:
9
+ - Internal refactors
10
+ - Bug fixes (unless security-related)
11
+ - Minor UI changes
@@ -0,0 +1,9 @@
1
+ - Features that affect multiple parts of the system
2
+ - Breaking changes or significant refactors
3
+ - Design decisions that need thinking through
4
+ - Complex features that benefit from upfront planning
5
+
6
+ Skip specs for:
7
+ - Bug fixes
8
+ - Trivial changes
9
+ - Self-explanatory refactors
@@ -0,0 +1,9 @@
1
+ - Features that affect multiple parts of the system
2
+ - Breaking changes or significant refactors
3
+ - Design decisions that need team alignment
4
+ - Complex features that benefit from upfront thinking
5
+
6
+ Skip specs for:
7
+ - Bug fixes
8
+ - Trivial changes
9
+ - Self-explanatory refactors
@@ -0,0 +1,8 @@
1
+ 1. **Discover context** - Run `lean-spec stats`, `lean-spec board`, or `lean-spec gantt`
2
+ 2. **Search existing specs** - Use `lean-spec search` or `lean-spec list --tag=<relevant>`
3
+ 3. **Check dependencies** - Run `lean-spec deps <spec>` to understand dependencies
4
+ 4. **Create or update spec** - Add complete frontmatter with compliance tags
5
+ 5. **Get reviews** - Assign reviewer, tag for security review if needed
6
+ 6. **Implement changes** - Keep spec in sync, update status appropriately
7
+ 7. **Update status** - Mark progress through workflow states
8
+ 8. **Archive when done** - `lean-spec archive <spec>` after completion
@@ -0,0 +1,7 @@
1
+ 1. **Discover context** - Run `lean-spec stats` or `lean-spec board` to see current state
2
+ 2. **Search existing specs** - Use `lean-spec search` or `lean-spec list` to find relevant work
3
+ 3. **Check dependencies** - Run `lean-spec deps <spec>` if working on existing spec
4
+ 4. **Create or update spec** - Add frontmatter with required fields and helpful metadata
5
+ 5. **Implement changes** - Keep spec in sync as you learn
6
+ 6. **Update status** - Mark progress: `draft` → `in-progress` → `complete`
7
+ 7. **Archive when done** - `lean-spec archive <spec>` moves to archive
@@ -0,0 +1,5 @@
1
+ 1. **Check existing work** - Run `lean-spec board` or `lean-spec search`
2
+ 2. **Create or update spec** - Add frontmatter with `status` and `created`
3
+ 3. **Implement changes** - Keep spec in sync as you learn
4
+ 4. **Update status** - Mark progress: `draft` → `in-progress` → `complete`
5
+ 5. **Archive when done** - `lean-spec archive <spec>` moves to archive
@@ -0,0 +1,39 @@
1
+ # AI Agent Instructions
2
+
3
+ ## Project: {{{project_name}}}
4
+
5
+ {{{description}}}
6
+
7
+ ## Core Rules
8
+
9
+ {{{coreRules}}}
10
+ {{#if whenToUseEarly}}
11
+
12
+ ## When to Use Specs
13
+
14
+ {{{whenToUse}}}
15
+ {{/if}}
16
+
17
+ ## Discovery Commands
18
+
19
+ {{{discoveryCommands}}}
20
+
21
+ ## Spec Frontmatter
22
+
23
+ {{{frontmatter}}}
24
+ {{#each additionalSections}}
25
+
26
+ {{{this}}}
27
+ {{/each}}
28
+
29
+ ## Workflow
30
+
31
+ {{{workflow}}}
32
+
33
+ ## Quality Standards
34
+
35
+ {{{qualityStandards}}}
36
+
37
+ ---
38
+
39
+ {{{closingNote}}}
@@ -0,0 +1,15 @@
1
+ {
2
+ "description": "Enterprise-grade development with security and compliance requirements.",
3
+ "coreRules": "core-rules-enterprise.md",
4
+ "whenToUseEarly": false,
5
+ "discoveryCommands": "discovery-commands-enterprise.md",
6
+ "frontmatter": "frontmatter-enterprise.md",
7
+ "workflow": "workflow-enterprise.md",
8
+ "qualityStandards": "quality-standards-enterprise.md",
9
+ "closingNote": "**Remember**: Enterprise doesn't mean heavy. Keep specs lean while meeting governance needs.",
10
+ "additionalSections": [
11
+ "enterprise-compliance.md",
12
+ "enterprise-when-required.md",
13
+ "enterprise-approval.md"
14
+ ]
15
+ }
@@ -10,6 +10,7 @@ Enterprise-grade development with security and compliance requirements.
10
10
  2. **Read specs/** - Check for compliance requirements
11
11
  3. **Follow LeanSpec** - Lean doesn't mean skipping governance
12
12
  4. **Document decisions** - Especially security and compliance choices
13
+ 5. **Never use nested code blocks** - Markdown doesn't support code blocks within code blocks. Use indentation or describe the structure instead
13
14
 
14
15
  ## Discovery Commands
15
16
 
@@ -0,0 +1,12 @@
1
+ {
2
+ "description": "Lightweight spec methodology for AI-powered development.",
3
+ "coreRules": "core-rules-base.md",
4
+ "whenToUse": "when-to-use-minimal.md",
5
+ "whenToUseEarly": true,
6
+ "discoveryCommands": "discovery-commands-minimal.md",
7
+ "frontmatter": "frontmatter-minimal.md",
8
+ "workflow": "workflow-standard.md",
9
+ "qualityStandards": "quality-standards-base.md",
10
+ "closingNote": "**Remember**: LeanSpec is a mindset. Adapt these guidelines to what actually helps.",
11
+ "additionalSections": []
12
+ }
@@ -10,6 +10,7 @@ Lightweight spec methodology for AI-powered development.
10
10
  2. **Check specs/** - Review existing specs before starting
11
11
  3. **Follow LeanSpec principles** - Clarity over documentation
12
12
  4. **Keep it minimal** - If it doesn't add clarity, cut it
13
+ 5. **Never use nested code blocks** - Markdown doesn't support code blocks within code blocks. Use indentation or describe the structure instead
13
14
 
14
15
  ## When to Use Specs
15
16
 
@@ -0,0 +1,12 @@
1
+ {
2
+ "description": "Lightweight spec methodology for AI-powered development.",
3
+ "coreRules": "core-rules-base.md",
4
+ "whenToUse": "when-to-use-standard.md",
5
+ "whenToUseEarly": true,
6
+ "discoveryCommands": "discovery-commands-standard.md",
7
+ "frontmatter": "frontmatter-standard.md",
8
+ "workflow": "workflow-standard-detailed.md",
9
+ "qualityStandards": "quality-standards-base.md",
10
+ "closingNote": "**Remember**: LeanSpec is a mindset. Adapt these guidelines to what actually helps.",
11
+ "additionalSections": []
12
+ }