@bobby_z/openspec 0.0.1
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/LICENSE +22 -0
- package/README.md +204 -0
- package/bin/openspec.js +3 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +482 -0
- package/dist/commands/change.d.ts +35 -0
- package/dist/commands/change.js +277 -0
- package/dist/commands/completion.d.ts +72 -0
- package/dist/commands/completion.js +257 -0
- package/dist/commands/config.d.ts +8 -0
- package/dist/commands/config.js +198 -0
- package/dist/commands/feedback.d.ts +9 -0
- package/dist/commands/feedback.js +183 -0
- package/dist/commands/schema.d.ts +6 -0
- package/dist/commands/schema.js +869 -0
- package/dist/commands/show.d.ts +14 -0
- package/dist/commands/show.js +132 -0
- package/dist/commands/spec.d.ts +15 -0
- package/dist/commands/spec.js +225 -0
- package/dist/commands/validate.d.ts +24 -0
- package/dist/commands/validate.js +294 -0
- package/dist/commands/workflow/index.d.ts +17 -0
- package/dist/commands/workflow/index.js +12 -0
- package/dist/commands/workflow/instructions.d.ts +29 -0
- package/dist/commands/workflow/instructions.js +381 -0
- package/dist/commands/workflow/new-change.d.ts +11 -0
- package/dist/commands/workflow/new-change.js +44 -0
- package/dist/commands/workflow/schemas.d.ts +10 -0
- package/dist/commands/workflow/schemas.js +34 -0
- package/dist/commands/workflow/shared.d.ts +52 -0
- package/dist/commands/workflow/shared.js +111 -0
- package/dist/commands/workflow/status.d.ts +14 -0
- package/dist/commands/workflow/status.js +58 -0
- package/dist/commands/workflow/templates.d.ts +16 -0
- package/dist/commands/workflow/templates.js +68 -0
- package/dist/core/archive.d.ts +11 -0
- package/dist/core/archive.js +328 -0
- package/dist/core/artifact-graph/graph.d.ts +56 -0
- package/dist/core/artifact-graph/graph.js +141 -0
- package/dist/core/artifact-graph/index.d.ts +7 -0
- package/dist/core/artifact-graph/index.js +13 -0
- package/dist/core/artifact-graph/instruction-loader.d.ts +143 -0
- package/dist/core/artifact-graph/instruction-loader.js +214 -0
- package/dist/core/artifact-graph/resolver.d.ts +81 -0
- package/dist/core/artifact-graph/resolver.js +257 -0
- package/dist/core/artifact-graph/schema.d.ts +13 -0
- package/dist/core/artifact-graph/schema.js +108 -0
- package/dist/core/artifact-graph/state.d.ts +12 -0
- package/dist/core/artifact-graph/state.js +54 -0
- package/dist/core/artifact-graph/types.d.ts +45 -0
- package/dist/core/artifact-graph/types.js +43 -0
- package/dist/core/command-generation/adapters/amazon-q.d.ts +13 -0
- package/dist/core/command-generation/adapters/amazon-q.js +26 -0
- package/dist/core/command-generation/adapters/antigravity.d.ts +13 -0
- package/dist/core/command-generation/adapters/antigravity.js +26 -0
- package/dist/core/command-generation/adapters/auggie.d.ts +13 -0
- package/dist/core/command-generation/adapters/auggie.js +27 -0
- package/dist/core/command-generation/adapters/claude.d.ts +13 -0
- package/dist/core/command-generation/adapters/claude.js +50 -0
- package/dist/core/command-generation/adapters/cline.d.ts +14 -0
- package/dist/core/command-generation/adapters/cline.js +27 -0
- package/dist/core/command-generation/adapters/codebuddy.d.ts +13 -0
- package/dist/core/command-generation/adapters/codebuddy.js +28 -0
- package/dist/core/command-generation/adapters/codex.d.ts +16 -0
- package/dist/core/command-generation/adapters/codex.js +39 -0
- package/dist/core/command-generation/adapters/continue.d.ts +13 -0
- package/dist/core/command-generation/adapters/continue.js +28 -0
- package/dist/core/command-generation/adapters/costrict.d.ts +13 -0
- package/dist/core/command-generation/adapters/costrict.js +27 -0
- package/dist/core/command-generation/adapters/crush.d.ts +13 -0
- package/dist/core/command-generation/adapters/crush.js +30 -0
- package/dist/core/command-generation/adapters/cursor.d.ts +14 -0
- package/dist/core/command-generation/adapters/cursor.js +44 -0
- package/dist/core/command-generation/adapters/devagent.d.ts +15 -0
- package/dist/core/command-generation/adapters/devagent.js +28 -0
- package/dist/core/command-generation/adapters/factory.d.ts +13 -0
- package/dist/core/command-generation/adapters/factory.js +27 -0
- package/dist/core/command-generation/adapters/gemini.d.ts +13 -0
- package/dist/core/command-generation/adapters/gemini.js +26 -0
- package/dist/core/command-generation/adapters/github-copilot.d.ts +13 -0
- package/dist/core/command-generation/adapters/github-copilot.js +26 -0
- package/dist/core/command-generation/adapters/iflow.d.ts +13 -0
- package/dist/core/command-generation/adapters/iflow.js +29 -0
- package/dist/core/command-generation/adapters/index.d.ts +28 -0
- package/dist/core/command-generation/adapters/index.js +28 -0
- package/dist/core/command-generation/adapters/kilocode.d.ts +14 -0
- package/dist/core/command-generation/adapters/kilocode.js +23 -0
- package/dist/core/command-generation/adapters/opencode.d.ts +13 -0
- package/dist/core/command-generation/adapters/opencode.js +29 -0
- package/dist/core/command-generation/adapters/qoder.d.ts +13 -0
- package/dist/core/command-generation/adapters/qoder.js +30 -0
- package/dist/core/command-generation/adapters/qwen.d.ts +13 -0
- package/dist/core/command-generation/adapters/qwen.js +26 -0
- package/dist/core/command-generation/adapters/roocode.d.ts +14 -0
- package/dist/core/command-generation/adapters/roocode.js +27 -0
- package/dist/core/command-generation/adapters/windsurf.d.ts +14 -0
- package/dist/core/command-generation/adapters/windsurf.js +51 -0
- package/dist/core/command-generation/generator.d.ts +21 -0
- package/dist/core/command-generation/generator.js +27 -0
- package/dist/core/command-generation/index.d.ts +22 -0
- package/dist/core/command-generation/index.js +24 -0
- package/dist/core/command-generation/registry.d.ts +36 -0
- package/dist/core/command-generation/registry.js +90 -0
- package/dist/core/command-generation/types.d.ts +56 -0
- package/dist/core/command-generation/types.js +8 -0
- package/dist/core/completions/command-registry.d.ts +7 -0
- package/dist/core/completions/command-registry.js +454 -0
- package/dist/core/completions/completion-provider.d.ts +60 -0
- package/dist/core/completions/completion-provider.js +102 -0
- package/dist/core/completions/factory.d.ts +64 -0
- package/dist/core/completions/factory.js +75 -0
- package/dist/core/completions/generators/bash-generator.d.ts +32 -0
- package/dist/core/completions/generators/bash-generator.js +174 -0
- package/dist/core/completions/generators/fish-generator.d.ts +32 -0
- package/dist/core/completions/generators/fish-generator.js +157 -0
- package/dist/core/completions/generators/powershell-generator.d.ts +33 -0
- package/dist/core/completions/generators/powershell-generator.js +207 -0
- package/dist/core/completions/generators/zsh-generator.d.ts +44 -0
- package/dist/core/completions/generators/zsh-generator.js +250 -0
- package/dist/core/completions/installers/bash-installer.d.ts +87 -0
- package/dist/core/completions/installers/bash-installer.js +318 -0
- package/dist/core/completions/installers/fish-installer.d.ts +43 -0
- package/dist/core/completions/installers/fish-installer.js +143 -0
- package/dist/core/completions/installers/powershell-installer.d.ts +88 -0
- package/dist/core/completions/installers/powershell-installer.js +327 -0
- package/dist/core/completions/installers/zsh-installer.d.ts +125 -0
- package/dist/core/completions/installers/zsh-installer.js +449 -0
- package/dist/core/completions/templates/bash-templates.d.ts +6 -0
- package/dist/core/completions/templates/bash-templates.js +24 -0
- package/dist/core/completions/templates/fish-templates.d.ts +7 -0
- package/dist/core/completions/templates/fish-templates.js +39 -0
- package/dist/core/completions/templates/powershell-templates.d.ts +6 -0
- package/dist/core/completions/templates/powershell-templates.js +25 -0
- package/dist/core/completions/templates/zsh-templates.d.ts +6 -0
- package/dist/core/completions/templates/zsh-templates.js +36 -0
- package/dist/core/completions/types.d.ts +79 -0
- package/dist/core/completions/types.js +2 -0
- package/dist/core/config-prompts.d.ts +9 -0
- package/dist/core/config-prompts.js +34 -0
- package/dist/core/config-schema.d.ts +76 -0
- package/dist/core/config-schema.js +200 -0
- package/dist/core/config.d.ts +17 -0
- package/dist/core/config.js +175 -0
- package/dist/core/converters/json-converter.d.ts +6 -0
- package/dist/core/converters/json-converter.js +51 -0
- package/dist/core/global-config.d.ts +39 -0
- package/dist/core/global-config.js +115 -0
- package/dist/core/index.d.ts +2 -0
- package/dist/core/index.js +3 -0
- package/dist/core/init.d.ts +32 -0
- package/dist/core/init.js +447 -0
- package/dist/core/legacy-cleanup.d.ts +162 -0
- package/dist/core/legacy-cleanup.js +520 -0
- package/dist/core/list.d.ts +9 -0
- package/dist/core/list.js +171 -0
- package/dist/core/parsers/change-parser.d.ts +13 -0
- package/dist/core/parsers/change-parser.js +193 -0
- package/dist/core/parsers/markdown-parser.d.ts +22 -0
- package/dist/core/parsers/markdown-parser.js +187 -0
- package/dist/core/parsers/requirement-blocks.d.ts +37 -0
- package/dist/core/parsers/requirement-blocks.js +201 -0
- package/dist/core/project-config.d.ts +64 -0
- package/dist/core/project-config.js +223 -0
- package/dist/core/schemas/base.schema.d.ts +13 -0
- package/dist/core/schemas/base.schema.js +13 -0
- package/dist/core/schemas/change.schema.d.ts +73 -0
- package/dist/core/schemas/change.schema.js +31 -0
- package/dist/core/schemas/index.d.ts +4 -0
- package/dist/core/schemas/index.js +4 -0
- package/dist/core/schemas/spec.schema.d.ts +18 -0
- package/dist/core/schemas/spec.schema.js +15 -0
- package/dist/core/shared/index.d.ts +8 -0
- package/dist/core/shared/index.js +8 -0
- package/dist/core/shared/skill-generation.d.ts +42 -0
- package/dist/core/shared/skill-generation.js +80 -0
- package/dist/core/shared/tool-detection.d.ts +66 -0
- package/dist/core/shared/tool-detection.js +140 -0
- package/dist/core/specs-apply.d.ts +73 -0
- package/dist/core/specs-apply.js +384 -0
- package/dist/core/styles/palette.d.ts +7 -0
- package/dist/core/styles/palette.js +8 -0
- package/dist/core/templates/index.d.ts +8 -0
- package/dist/core/templates/index.js +9 -0
- package/dist/core/templates/skill-templates.d.ts +122 -0
- package/dist/core/templates/skill-templates.js +3437 -0
- package/dist/core/update.d.ts +42 -0
- package/dist/core/update.js +311 -0
- package/dist/core/validation/constants.d.ts +34 -0
- package/dist/core/validation/constants.js +40 -0
- package/dist/core/validation/types.d.ts +18 -0
- package/dist/core/validation/types.js +2 -0
- package/dist/core/validation/validator.d.ts +33 -0
- package/dist/core/validation/validator.js +409 -0
- package/dist/core/view.d.ts +8 -0
- package/dist/core/view.js +168 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.js +3 -0
- package/dist/prompts/searchable-multi-select.d.ts +27 -0
- package/dist/prompts/searchable-multi-select.js +149 -0
- package/dist/telemetry/config.d.ts +32 -0
- package/dist/telemetry/config.js +68 -0
- package/dist/telemetry/index.d.ts +31 -0
- package/dist/telemetry/index.js +145 -0
- package/dist/ui/ascii-patterns.d.ts +16 -0
- package/dist/ui/ascii-patterns.js +133 -0
- package/dist/ui/welcome-screen.d.ts +10 -0
- package/dist/ui/welcome-screen.js +146 -0
- package/dist/utils/change-metadata.d.ts +51 -0
- package/dist/utils/change-metadata.js +147 -0
- package/dist/utils/change-utils.d.ts +62 -0
- package/dist/utils/change-utils.js +121 -0
- package/dist/utils/command-references.d.ts +18 -0
- package/dist/utils/command-references.js +20 -0
- package/dist/utils/file-system.d.ts +36 -0
- package/dist/utils/file-system.js +281 -0
- package/dist/utils/index.d.ts +6 -0
- package/dist/utils/index.js +9 -0
- package/dist/utils/interactive.d.ts +18 -0
- package/dist/utils/interactive.js +21 -0
- package/dist/utils/item-discovery.d.ts +4 -0
- package/dist/utils/item-discovery.js +72 -0
- package/dist/utils/match.d.ts +3 -0
- package/dist/utils/match.js +22 -0
- package/dist/utils/shell-detection.d.ts +20 -0
- package/dist/utils/shell-detection.js +41 -0
- package/dist/utils/task-progress.d.ts +8 -0
- package/dist/utils/task-progress.js +36 -0
- package/package.json +83 -0
- package/schemas/spec-driven/schema.yaml +151 -0
- package/schemas/spec-driven/templates/design.md +21 -0
- package/schemas/spec-driven/templates/proposal.md +25 -0
- package/schemas/spec-driven/templates/spec.md +10 -0
- package/schemas/spec-driven/templates/tasks.md +9 -0
- package/scripts/postinstall.js +147 -0
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { promises as fs } from 'fs';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import { getTaskProgressForChange, formatTaskStatus } from '../utils/task-progress.js';
|
|
4
|
+
import { readFileSync } from 'fs';
|
|
5
|
+
import { join } from 'path';
|
|
6
|
+
import { MarkdownParser } from './parsers/markdown-parser.js';
|
|
7
|
+
/**
|
|
8
|
+
* Get the most recent modification time of any file in a directory (recursive).
|
|
9
|
+
* Falls back to the directory's own mtime if no files are found.
|
|
10
|
+
*/
|
|
11
|
+
async function getLastModified(dirPath) {
|
|
12
|
+
let latest = null;
|
|
13
|
+
async function walk(dir) {
|
|
14
|
+
const entries = await fs.readdir(dir, { withFileTypes: true });
|
|
15
|
+
for (const entry of entries) {
|
|
16
|
+
const fullPath = path.join(dir, entry.name);
|
|
17
|
+
if (entry.isDirectory()) {
|
|
18
|
+
await walk(fullPath);
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
const stat = await fs.stat(fullPath);
|
|
22
|
+
if (latest === null || stat.mtime > latest) {
|
|
23
|
+
latest = stat.mtime;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
await walk(dirPath);
|
|
29
|
+
// If no files found, use the directory's own modification time
|
|
30
|
+
if (latest === null) {
|
|
31
|
+
const dirStat = await fs.stat(dirPath);
|
|
32
|
+
return dirStat.mtime;
|
|
33
|
+
}
|
|
34
|
+
return latest;
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Format a date as relative time (e.g., "2 hours ago", "3 days ago")
|
|
38
|
+
*/
|
|
39
|
+
function formatRelativeTime(date) {
|
|
40
|
+
const now = new Date();
|
|
41
|
+
const diffMs = now.getTime() - date.getTime();
|
|
42
|
+
const diffSecs = Math.floor(diffMs / 1000);
|
|
43
|
+
const diffMins = Math.floor(diffSecs / 60);
|
|
44
|
+
const diffHours = Math.floor(diffMins / 60);
|
|
45
|
+
const diffDays = Math.floor(diffHours / 24);
|
|
46
|
+
if (diffDays > 30) {
|
|
47
|
+
return date.toLocaleDateString();
|
|
48
|
+
}
|
|
49
|
+
else if (diffDays > 0) {
|
|
50
|
+
return `${diffDays}d ago`;
|
|
51
|
+
}
|
|
52
|
+
else if (diffHours > 0) {
|
|
53
|
+
return `${diffHours}h ago`;
|
|
54
|
+
}
|
|
55
|
+
else if (diffMins > 0) {
|
|
56
|
+
return `${diffMins}m ago`;
|
|
57
|
+
}
|
|
58
|
+
else {
|
|
59
|
+
return 'just now';
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
export class ListCommand {
|
|
63
|
+
async execute(targetPath = '.', mode = 'changes', options = {}) {
|
|
64
|
+
const { sort = 'recent', json = false } = options;
|
|
65
|
+
if (mode === 'changes') {
|
|
66
|
+
const changesDir = path.join(targetPath, 'openspec', 'changes');
|
|
67
|
+
// Check if changes directory exists
|
|
68
|
+
try {
|
|
69
|
+
await fs.access(changesDir);
|
|
70
|
+
}
|
|
71
|
+
catch {
|
|
72
|
+
throw new Error("No OpenSpec changes directory found. Run 'openspec init' first.");
|
|
73
|
+
}
|
|
74
|
+
// Get all directories in changes (excluding archive)
|
|
75
|
+
const entries = await fs.readdir(changesDir, { withFileTypes: true });
|
|
76
|
+
const changeDirs = entries
|
|
77
|
+
.filter(entry => entry.isDirectory() && entry.name !== 'archive')
|
|
78
|
+
.map(entry => entry.name);
|
|
79
|
+
if (changeDirs.length === 0) {
|
|
80
|
+
if (json) {
|
|
81
|
+
console.log(JSON.stringify({ changes: [] }));
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
console.log('No active changes found.');
|
|
85
|
+
}
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
// Collect information about each change
|
|
89
|
+
const changes = [];
|
|
90
|
+
for (const changeDir of changeDirs) {
|
|
91
|
+
const progress = await getTaskProgressForChange(changesDir, changeDir);
|
|
92
|
+
const changePath = path.join(changesDir, changeDir);
|
|
93
|
+
const lastModified = await getLastModified(changePath);
|
|
94
|
+
changes.push({
|
|
95
|
+
name: changeDir,
|
|
96
|
+
completedTasks: progress.completed,
|
|
97
|
+
totalTasks: progress.total,
|
|
98
|
+
lastModified
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
// Sort by preference (default: recent first)
|
|
102
|
+
if (sort === 'recent') {
|
|
103
|
+
changes.sort((a, b) => b.lastModified.getTime() - a.lastModified.getTime());
|
|
104
|
+
}
|
|
105
|
+
else {
|
|
106
|
+
changes.sort((a, b) => a.name.localeCompare(b.name));
|
|
107
|
+
}
|
|
108
|
+
// JSON output for programmatic use
|
|
109
|
+
if (json) {
|
|
110
|
+
const jsonOutput = changes.map(c => ({
|
|
111
|
+
name: c.name,
|
|
112
|
+
completedTasks: c.completedTasks,
|
|
113
|
+
totalTasks: c.totalTasks,
|
|
114
|
+
lastModified: c.lastModified.toISOString(),
|
|
115
|
+
status: c.totalTasks === 0 ? 'no-tasks' : c.completedTasks === c.totalTasks ? 'complete' : 'in-progress'
|
|
116
|
+
}));
|
|
117
|
+
console.log(JSON.stringify({ changes: jsonOutput }, null, 2));
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
// Display results
|
|
121
|
+
console.log('Changes:');
|
|
122
|
+
const padding = ' ';
|
|
123
|
+
const nameWidth = Math.max(...changes.map(c => c.name.length));
|
|
124
|
+
for (const change of changes) {
|
|
125
|
+
const paddedName = change.name.padEnd(nameWidth);
|
|
126
|
+
const status = formatTaskStatus({ total: change.totalTasks, completed: change.completedTasks });
|
|
127
|
+
const timeAgo = formatRelativeTime(change.lastModified);
|
|
128
|
+
console.log(`${padding}${paddedName} ${status.padEnd(12)} ${timeAgo}`);
|
|
129
|
+
}
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
// specs mode
|
|
133
|
+
const specsDir = path.join(targetPath, 'openspec', 'specs');
|
|
134
|
+
try {
|
|
135
|
+
await fs.access(specsDir);
|
|
136
|
+
}
|
|
137
|
+
catch {
|
|
138
|
+
console.log('No specs found.');
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
const entries = await fs.readdir(specsDir, { withFileTypes: true });
|
|
142
|
+
const specDirs = entries.filter(e => e.isDirectory()).map(e => e.name);
|
|
143
|
+
if (specDirs.length === 0) {
|
|
144
|
+
console.log('No specs found.');
|
|
145
|
+
return;
|
|
146
|
+
}
|
|
147
|
+
const specs = [];
|
|
148
|
+
for (const id of specDirs) {
|
|
149
|
+
const specPath = join(specsDir, id, 'spec.md');
|
|
150
|
+
try {
|
|
151
|
+
const content = readFileSync(specPath, 'utf-8');
|
|
152
|
+
const parser = new MarkdownParser(content);
|
|
153
|
+
const spec = parser.parseSpec(id);
|
|
154
|
+
specs.push({ id, requirementCount: spec.requirements.length });
|
|
155
|
+
}
|
|
156
|
+
catch {
|
|
157
|
+
// If spec cannot be read or parsed, include with 0 count
|
|
158
|
+
specs.push({ id, requirementCount: 0 });
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
specs.sort((a, b) => a.id.localeCompare(b.id));
|
|
162
|
+
console.log('Specs:');
|
|
163
|
+
const padding = ' ';
|
|
164
|
+
const nameWidth = Math.max(...specs.map(s => s.id.length));
|
|
165
|
+
for (const spec of specs) {
|
|
166
|
+
const padded = spec.id.padEnd(nameWidth);
|
|
167
|
+
console.log(`${padding}${padded} requirements ${spec.requirementCount}`);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
//# sourceMappingURL=list.js.map
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { MarkdownParser } from './markdown-parser.js';
|
|
2
|
+
import { Change } from '../schemas/index.js';
|
|
3
|
+
export declare class ChangeParser extends MarkdownParser {
|
|
4
|
+
private changeDir;
|
|
5
|
+
constructor(content: string, changeDir: string);
|
|
6
|
+
parseChangeWithDeltas(name: string): Promise<Change>;
|
|
7
|
+
private parseDeltaSpecs;
|
|
8
|
+
private parseSpecDeltas;
|
|
9
|
+
private parseRenames;
|
|
10
|
+
private parseSectionsFromContent;
|
|
11
|
+
private getContentUntilNextHeaderFromLines;
|
|
12
|
+
}
|
|
13
|
+
//# sourceMappingURL=change-parser.d.ts.map
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
import { MarkdownParser } from './markdown-parser.js';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import { promises as fs } from 'fs';
|
|
4
|
+
export class ChangeParser extends MarkdownParser {
|
|
5
|
+
changeDir;
|
|
6
|
+
constructor(content, changeDir) {
|
|
7
|
+
super(content);
|
|
8
|
+
this.changeDir = changeDir;
|
|
9
|
+
}
|
|
10
|
+
async parseChangeWithDeltas(name) {
|
|
11
|
+
const sections = this.parseSections();
|
|
12
|
+
const why = this.findSection(sections, 'Why')?.content || '';
|
|
13
|
+
const whatChanges = this.findSection(sections, 'What Changes')?.content || '';
|
|
14
|
+
if (!why) {
|
|
15
|
+
throw new Error('Change must have a Why section');
|
|
16
|
+
}
|
|
17
|
+
if (!whatChanges) {
|
|
18
|
+
throw new Error('Change must have a What Changes section');
|
|
19
|
+
}
|
|
20
|
+
// Parse deltas from the What Changes section (simple format)
|
|
21
|
+
const simpleDeltas = this.parseDeltas(whatChanges);
|
|
22
|
+
// Check if there are spec files with delta format
|
|
23
|
+
const specsDir = path.join(this.changeDir, 'specs');
|
|
24
|
+
const deltaDeltas = await this.parseDeltaSpecs(specsDir);
|
|
25
|
+
// Combine both types of deltas, preferring delta format if available
|
|
26
|
+
const deltas = deltaDeltas.length > 0 ? deltaDeltas : simpleDeltas;
|
|
27
|
+
return {
|
|
28
|
+
name,
|
|
29
|
+
why: why.trim(),
|
|
30
|
+
whatChanges: whatChanges.trim(),
|
|
31
|
+
deltas,
|
|
32
|
+
metadata: {
|
|
33
|
+
version: '1.0.0',
|
|
34
|
+
format: 'openspec-change',
|
|
35
|
+
},
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
async parseDeltaSpecs(specsDir) {
|
|
39
|
+
const deltas = [];
|
|
40
|
+
try {
|
|
41
|
+
const specDirs = await fs.readdir(specsDir, { withFileTypes: true });
|
|
42
|
+
for (const dir of specDirs) {
|
|
43
|
+
if (!dir.isDirectory())
|
|
44
|
+
continue;
|
|
45
|
+
const specName = dir.name;
|
|
46
|
+
const specFile = path.join(specsDir, specName, 'spec.md');
|
|
47
|
+
try {
|
|
48
|
+
const content = await fs.readFile(specFile, 'utf-8');
|
|
49
|
+
const specDeltas = this.parseSpecDeltas(specName, content);
|
|
50
|
+
deltas.push(...specDeltas);
|
|
51
|
+
}
|
|
52
|
+
catch (error) {
|
|
53
|
+
// Spec file might not exist, which is okay
|
|
54
|
+
continue;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
catch (error) {
|
|
59
|
+
// Specs directory might not exist, which is okay
|
|
60
|
+
return [];
|
|
61
|
+
}
|
|
62
|
+
return deltas;
|
|
63
|
+
}
|
|
64
|
+
parseSpecDeltas(specName, content) {
|
|
65
|
+
const deltas = [];
|
|
66
|
+
const sections = this.parseSectionsFromContent(content);
|
|
67
|
+
// Parse ADDED requirements
|
|
68
|
+
const addedSection = this.findSection(sections, 'ADDED Requirements');
|
|
69
|
+
if (addedSection) {
|
|
70
|
+
const requirements = this.parseRequirements(addedSection);
|
|
71
|
+
requirements.forEach(req => {
|
|
72
|
+
deltas.push({
|
|
73
|
+
spec: specName,
|
|
74
|
+
operation: 'ADDED',
|
|
75
|
+
description: `Add requirement: ${req.text}`,
|
|
76
|
+
// Provide both single and plural forms for compatibility
|
|
77
|
+
requirement: req,
|
|
78
|
+
requirements: [req],
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
// Parse MODIFIED requirements
|
|
83
|
+
const modifiedSection = this.findSection(sections, 'MODIFIED Requirements');
|
|
84
|
+
if (modifiedSection) {
|
|
85
|
+
const requirements = this.parseRequirements(modifiedSection);
|
|
86
|
+
requirements.forEach(req => {
|
|
87
|
+
deltas.push({
|
|
88
|
+
spec: specName,
|
|
89
|
+
operation: 'MODIFIED',
|
|
90
|
+
description: `Modify requirement: ${req.text}`,
|
|
91
|
+
requirement: req,
|
|
92
|
+
requirements: [req],
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
// Parse REMOVED requirements
|
|
97
|
+
const removedSection = this.findSection(sections, 'REMOVED Requirements');
|
|
98
|
+
if (removedSection) {
|
|
99
|
+
const requirements = this.parseRequirements(removedSection);
|
|
100
|
+
requirements.forEach(req => {
|
|
101
|
+
deltas.push({
|
|
102
|
+
spec: specName,
|
|
103
|
+
operation: 'REMOVED',
|
|
104
|
+
description: `Remove requirement: ${req.text}`,
|
|
105
|
+
requirement: req,
|
|
106
|
+
requirements: [req],
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
// Parse RENAMED requirements
|
|
111
|
+
const renamedSection = this.findSection(sections, 'RENAMED Requirements');
|
|
112
|
+
if (renamedSection) {
|
|
113
|
+
const renames = this.parseRenames(renamedSection.content);
|
|
114
|
+
renames.forEach(rename => {
|
|
115
|
+
deltas.push({
|
|
116
|
+
spec: specName,
|
|
117
|
+
operation: 'RENAMED',
|
|
118
|
+
description: `Rename requirement from "${rename.from}" to "${rename.to}"`,
|
|
119
|
+
rename,
|
|
120
|
+
});
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
return deltas;
|
|
124
|
+
}
|
|
125
|
+
parseRenames(content) {
|
|
126
|
+
const renames = [];
|
|
127
|
+
const lines = ChangeParser.normalizeContent(content).split('\n');
|
|
128
|
+
let currentRename = {};
|
|
129
|
+
for (const line of lines) {
|
|
130
|
+
const fromMatch = line.match(/^\s*-?\s*FROM:\s*`?###\s*Requirement:\s*(.+?)`?\s*$/);
|
|
131
|
+
const toMatch = line.match(/^\s*-?\s*TO:\s*`?###\s*Requirement:\s*(.+?)`?\s*$/);
|
|
132
|
+
if (fromMatch) {
|
|
133
|
+
currentRename.from = fromMatch[1].trim();
|
|
134
|
+
}
|
|
135
|
+
else if (toMatch) {
|
|
136
|
+
currentRename.to = toMatch[1].trim();
|
|
137
|
+
if (currentRename.from && currentRename.to) {
|
|
138
|
+
renames.push({
|
|
139
|
+
from: currentRename.from,
|
|
140
|
+
to: currentRename.to,
|
|
141
|
+
});
|
|
142
|
+
currentRename = {};
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
return renames;
|
|
147
|
+
}
|
|
148
|
+
parseSectionsFromContent(content) {
|
|
149
|
+
const normalizedContent = ChangeParser.normalizeContent(content);
|
|
150
|
+
const lines = normalizedContent.split('\n');
|
|
151
|
+
const sections = [];
|
|
152
|
+
const stack = [];
|
|
153
|
+
for (let i = 0; i < lines.length; i++) {
|
|
154
|
+
const line = lines[i];
|
|
155
|
+
const headerMatch = line.match(/^(#{1,6})\s+(.+)$/);
|
|
156
|
+
if (headerMatch) {
|
|
157
|
+
const level = headerMatch[1].length;
|
|
158
|
+
const title = headerMatch[2].trim();
|
|
159
|
+
const contentLines = this.getContentUntilNextHeaderFromLines(lines, i + 1, level);
|
|
160
|
+
const section = {
|
|
161
|
+
level,
|
|
162
|
+
title,
|
|
163
|
+
content: contentLines.join('\n').trim(),
|
|
164
|
+
children: [],
|
|
165
|
+
};
|
|
166
|
+
while (stack.length > 0 && stack[stack.length - 1].level >= level) {
|
|
167
|
+
stack.pop();
|
|
168
|
+
}
|
|
169
|
+
if (stack.length === 0) {
|
|
170
|
+
sections.push(section);
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
stack[stack.length - 1].children.push(section);
|
|
174
|
+
}
|
|
175
|
+
stack.push(section);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
return sections;
|
|
179
|
+
}
|
|
180
|
+
getContentUntilNextHeaderFromLines(lines, startLine, currentLevel) {
|
|
181
|
+
const contentLines = [];
|
|
182
|
+
for (let i = startLine; i < lines.length; i++) {
|
|
183
|
+
const line = lines[i];
|
|
184
|
+
const headerMatch = line.match(/^(#{1,6})\s+/);
|
|
185
|
+
if (headerMatch && headerMatch[1].length <= currentLevel) {
|
|
186
|
+
break;
|
|
187
|
+
}
|
|
188
|
+
contentLines.push(line);
|
|
189
|
+
}
|
|
190
|
+
return contentLines;
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
//# sourceMappingURL=change-parser.js.map
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { Spec, Change, Requirement, Scenario, Delta } from '../schemas/index.js';
|
|
2
|
+
export interface Section {
|
|
3
|
+
level: number;
|
|
4
|
+
title: string;
|
|
5
|
+
content: string;
|
|
6
|
+
children: Section[];
|
|
7
|
+
}
|
|
8
|
+
export declare class MarkdownParser {
|
|
9
|
+
private lines;
|
|
10
|
+
private currentLine;
|
|
11
|
+
constructor(content: string);
|
|
12
|
+
protected static normalizeContent(content: string): string;
|
|
13
|
+
parseSpec(name: string): Spec;
|
|
14
|
+
parseChange(name: string): Change;
|
|
15
|
+
protected parseSections(): Section[];
|
|
16
|
+
protected getContentUntilNextHeader(startLine: number, currentLevel: number): string;
|
|
17
|
+
protected findSection(sections: Section[], title: string): Section | undefined;
|
|
18
|
+
protected parseRequirements(section: Section): Requirement[];
|
|
19
|
+
protected parseScenarios(requirementSection: Section): Scenario[];
|
|
20
|
+
protected parseDeltas(content: string): Delta[];
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=markdown-parser.d.ts.map
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
export class MarkdownParser {
|
|
2
|
+
lines;
|
|
3
|
+
currentLine;
|
|
4
|
+
constructor(content) {
|
|
5
|
+
const normalized = MarkdownParser.normalizeContent(content);
|
|
6
|
+
this.lines = normalized.split('\n');
|
|
7
|
+
this.currentLine = 0;
|
|
8
|
+
}
|
|
9
|
+
static normalizeContent(content) {
|
|
10
|
+
return content.replace(/\r\n?/g, '\n');
|
|
11
|
+
}
|
|
12
|
+
parseSpec(name) {
|
|
13
|
+
const sections = this.parseSections();
|
|
14
|
+
const purpose = this.findSection(sections, 'Purpose')?.content || '';
|
|
15
|
+
const requirementsSection = this.findSection(sections, 'Requirements');
|
|
16
|
+
if (!purpose) {
|
|
17
|
+
throw new Error('Spec must have a Purpose section');
|
|
18
|
+
}
|
|
19
|
+
if (!requirementsSection) {
|
|
20
|
+
throw new Error('Spec must have a Requirements section');
|
|
21
|
+
}
|
|
22
|
+
const requirements = this.parseRequirements(requirementsSection);
|
|
23
|
+
return {
|
|
24
|
+
name,
|
|
25
|
+
overview: purpose.trim(),
|
|
26
|
+
requirements,
|
|
27
|
+
metadata: {
|
|
28
|
+
version: '1.0.0',
|
|
29
|
+
format: 'openspec',
|
|
30
|
+
},
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
parseChange(name) {
|
|
34
|
+
const sections = this.parseSections();
|
|
35
|
+
const why = this.findSection(sections, 'Why')?.content || '';
|
|
36
|
+
const whatChanges = this.findSection(sections, 'What Changes')?.content || '';
|
|
37
|
+
if (!why) {
|
|
38
|
+
throw new Error('Change must have a Why section');
|
|
39
|
+
}
|
|
40
|
+
if (!whatChanges) {
|
|
41
|
+
throw new Error('Change must have a What Changes section');
|
|
42
|
+
}
|
|
43
|
+
const deltas = this.parseDeltas(whatChanges);
|
|
44
|
+
return {
|
|
45
|
+
name,
|
|
46
|
+
why: why.trim(),
|
|
47
|
+
whatChanges: whatChanges.trim(),
|
|
48
|
+
deltas,
|
|
49
|
+
metadata: {
|
|
50
|
+
version: '1.0.0',
|
|
51
|
+
format: 'openspec-change',
|
|
52
|
+
},
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
parseSections() {
|
|
56
|
+
const sections = [];
|
|
57
|
+
const stack = [];
|
|
58
|
+
for (let i = 0; i < this.lines.length; i++) {
|
|
59
|
+
const line = this.lines[i];
|
|
60
|
+
const headerMatch = line.match(/^(#{1,6})\s+(.+)$/);
|
|
61
|
+
if (headerMatch) {
|
|
62
|
+
const level = headerMatch[1].length;
|
|
63
|
+
const title = headerMatch[2].trim();
|
|
64
|
+
const content = this.getContentUntilNextHeader(i + 1, level);
|
|
65
|
+
const section = {
|
|
66
|
+
level,
|
|
67
|
+
title,
|
|
68
|
+
content,
|
|
69
|
+
children: [],
|
|
70
|
+
};
|
|
71
|
+
while (stack.length > 0 && stack[stack.length - 1].level >= level) {
|
|
72
|
+
stack.pop();
|
|
73
|
+
}
|
|
74
|
+
if (stack.length === 0) {
|
|
75
|
+
sections.push(section);
|
|
76
|
+
}
|
|
77
|
+
else {
|
|
78
|
+
stack[stack.length - 1].children.push(section);
|
|
79
|
+
}
|
|
80
|
+
stack.push(section);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
return sections;
|
|
84
|
+
}
|
|
85
|
+
getContentUntilNextHeader(startLine, currentLevel) {
|
|
86
|
+
const contentLines = [];
|
|
87
|
+
for (let i = startLine; i < this.lines.length; i++) {
|
|
88
|
+
const line = this.lines[i];
|
|
89
|
+
const headerMatch = line.match(/^(#{1,6})\s+/);
|
|
90
|
+
if (headerMatch && headerMatch[1].length <= currentLevel) {
|
|
91
|
+
break;
|
|
92
|
+
}
|
|
93
|
+
contentLines.push(line);
|
|
94
|
+
}
|
|
95
|
+
return contentLines.join('\n').trim();
|
|
96
|
+
}
|
|
97
|
+
findSection(sections, title) {
|
|
98
|
+
for (const section of sections) {
|
|
99
|
+
if (section.title.toLowerCase() === title.toLowerCase()) {
|
|
100
|
+
return section;
|
|
101
|
+
}
|
|
102
|
+
const child = this.findSection(section.children, title);
|
|
103
|
+
if (child) {
|
|
104
|
+
return child;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return undefined;
|
|
108
|
+
}
|
|
109
|
+
parseRequirements(section) {
|
|
110
|
+
const requirements = [];
|
|
111
|
+
for (const child of section.children) {
|
|
112
|
+
// Extract requirement text from first non-empty content line, fall back to heading
|
|
113
|
+
let text = child.title;
|
|
114
|
+
// Get content before any child sections (scenarios)
|
|
115
|
+
if (child.content.trim()) {
|
|
116
|
+
// Split content into lines and find content before any child headers
|
|
117
|
+
const lines = child.content.split('\n');
|
|
118
|
+
const contentBeforeChildren = [];
|
|
119
|
+
for (const line of lines) {
|
|
120
|
+
// Stop at child headers (scenarios start with ####)
|
|
121
|
+
if (line.trim().startsWith('#')) {
|
|
122
|
+
break;
|
|
123
|
+
}
|
|
124
|
+
contentBeforeChildren.push(line);
|
|
125
|
+
}
|
|
126
|
+
// Find first non-empty line
|
|
127
|
+
const directContent = contentBeforeChildren.join('\n').trim();
|
|
128
|
+
if (directContent) {
|
|
129
|
+
const firstLine = directContent.split('\n').find(l => l.trim());
|
|
130
|
+
if (firstLine) {
|
|
131
|
+
text = firstLine.trim();
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
const scenarios = this.parseScenarios(child);
|
|
136
|
+
requirements.push({
|
|
137
|
+
text,
|
|
138
|
+
scenarios,
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
return requirements;
|
|
142
|
+
}
|
|
143
|
+
parseScenarios(requirementSection) {
|
|
144
|
+
const scenarios = [];
|
|
145
|
+
for (const scenarioSection of requirementSection.children) {
|
|
146
|
+
// Store the raw text content of the scenario section
|
|
147
|
+
if (scenarioSection.content.trim()) {
|
|
148
|
+
scenarios.push({
|
|
149
|
+
rawText: scenarioSection.content
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
return scenarios;
|
|
154
|
+
}
|
|
155
|
+
parseDeltas(content) {
|
|
156
|
+
const deltas = [];
|
|
157
|
+
const lines = content.split('\n');
|
|
158
|
+
for (const line of lines) {
|
|
159
|
+
// Match both formats: **spec:** and **spec**:
|
|
160
|
+
const deltaMatch = line.match(/^\s*-\s*\*\*([^*:]+)(?::\*\*|\*\*:)\s*(.+)$/);
|
|
161
|
+
if (deltaMatch) {
|
|
162
|
+
const specName = deltaMatch[1].trim();
|
|
163
|
+
const description = deltaMatch[2].trim();
|
|
164
|
+
let operation = 'MODIFIED';
|
|
165
|
+
const lowerDesc = description.toLowerCase();
|
|
166
|
+
// Use word boundaries to avoid false matches (e.g., "address" matching "add")
|
|
167
|
+
// Check RENAMED first since it's more specific than patterns containing "new"
|
|
168
|
+
if (/\brename(s|d|ing)?\b/.test(lowerDesc) || /\brenamed\s+(to|from)\b/.test(lowerDesc)) {
|
|
169
|
+
operation = 'RENAMED';
|
|
170
|
+
}
|
|
171
|
+
else if (/\badd(s|ed|ing)?\b/.test(lowerDesc) || /\bcreate(s|d|ing)?\b/.test(lowerDesc) || /\bnew\b/.test(lowerDesc)) {
|
|
172
|
+
operation = 'ADDED';
|
|
173
|
+
}
|
|
174
|
+
else if (/\bremove(s|d|ing)?\b/.test(lowerDesc) || /\bdelete(s|d|ing)?\b/.test(lowerDesc)) {
|
|
175
|
+
operation = 'REMOVED';
|
|
176
|
+
}
|
|
177
|
+
deltas.push({
|
|
178
|
+
spec: specName,
|
|
179
|
+
operation,
|
|
180
|
+
description,
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
return deltas;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
//# sourceMappingURL=markdown-parser.js.map
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
export interface RequirementBlock {
|
|
2
|
+
headerLine: string;
|
|
3
|
+
name: string;
|
|
4
|
+
raw: string;
|
|
5
|
+
}
|
|
6
|
+
export interface RequirementsSectionParts {
|
|
7
|
+
before: string;
|
|
8
|
+
headerLine: string;
|
|
9
|
+
preamble: string;
|
|
10
|
+
bodyBlocks: RequirementBlock[];
|
|
11
|
+
after: string;
|
|
12
|
+
}
|
|
13
|
+
export declare function normalizeRequirementName(name: string): string;
|
|
14
|
+
/**
|
|
15
|
+
* Extracts the Requirements section from a spec file and parses requirement blocks.
|
|
16
|
+
*/
|
|
17
|
+
export declare function extractRequirementsSection(content: string): RequirementsSectionParts;
|
|
18
|
+
export interface DeltaPlan {
|
|
19
|
+
added: RequirementBlock[];
|
|
20
|
+
modified: RequirementBlock[];
|
|
21
|
+
removed: string[];
|
|
22
|
+
renamed: Array<{
|
|
23
|
+
from: string;
|
|
24
|
+
to: string;
|
|
25
|
+
}>;
|
|
26
|
+
sectionPresence: {
|
|
27
|
+
added: boolean;
|
|
28
|
+
modified: boolean;
|
|
29
|
+
removed: boolean;
|
|
30
|
+
renamed: boolean;
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Parse a delta-formatted spec change file content into a DeltaPlan with raw blocks.
|
|
35
|
+
*/
|
|
36
|
+
export declare function parseDeltaSpec(content: string): DeltaPlan;
|
|
37
|
+
//# sourceMappingURL=requirement-blocks.d.ts.map
|