@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.
Files changed (234) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +204 -0
  3. package/bin/openspec.js +3 -0
  4. package/dist/cli/index.d.ts +2 -0
  5. package/dist/cli/index.js +482 -0
  6. package/dist/commands/change.d.ts +35 -0
  7. package/dist/commands/change.js +277 -0
  8. package/dist/commands/completion.d.ts +72 -0
  9. package/dist/commands/completion.js +257 -0
  10. package/dist/commands/config.d.ts +8 -0
  11. package/dist/commands/config.js +198 -0
  12. package/dist/commands/feedback.d.ts +9 -0
  13. package/dist/commands/feedback.js +183 -0
  14. package/dist/commands/schema.d.ts +6 -0
  15. package/dist/commands/schema.js +869 -0
  16. package/dist/commands/show.d.ts +14 -0
  17. package/dist/commands/show.js +132 -0
  18. package/dist/commands/spec.d.ts +15 -0
  19. package/dist/commands/spec.js +225 -0
  20. package/dist/commands/validate.d.ts +24 -0
  21. package/dist/commands/validate.js +294 -0
  22. package/dist/commands/workflow/index.d.ts +17 -0
  23. package/dist/commands/workflow/index.js +12 -0
  24. package/dist/commands/workflow/instructions.d.ts +29 -0
  25. package/dist/commands/workflow/instructions.js +381 -0
  26. package/dist/commands/workflow/new-change.d.ts +11 -0
  27. package/dist/commands/workflow/new-change.js +44 -0
  28. package/dist/commands/workflow/schemas.d.ts +10 -0
  29. package/dist/commands/workflow/schemas.js +34 -0
  30. package/dist/commands/workflow/shared.d.ts +52 -0
  31. package/dist/commands/workflow/shared.js +111 -0
  32. package/dist/commands/workflow/status.d.ts +14 -0
  33. package/dist/commands/workflow/status.js +58 -0
  34. package/dist/commands/workflow/templates.d.ts +16 -0
  35. package/dist/commands/workflow/templates.js +68 -0
  36. package/dist/core/archive.d.ts +11 -0
  37. package/dist/core/archive.js +328 -0
  38. package/dist/core/artifact-graph/graph.d.ts +56 -0
  39. package/dist/core/artifact-graph/graph.js +141 -0
  40. package/dist/core/artifact-graph/index.d.ts +7 -0
  41. package/dist/core/artifact-graph/index.js +13 -0
  42. package/dist/core/artifact-graph/instruction-loader.d.ts +143 -0
  43. package/dist/core/artifact-graph/instruction-loader.js +214 -0
  44. package/dist/core/artifact-graph/resolver.d.ts +81 -0
  45. package/dist/core/artifact-graph/resolver.js +257 -0
  46. package/dist/core/artifact-graph/schema.d.ts +13 -0
  47. package/dist/core/artifact-graph/schema.js +108 -0
  48. package/dist/core/artifact-graph/state.d.ts +12 -0
  49. package/dist/core/artifact-graph/state.js +54 -0
  50. package/dist/core/artifact-graph/types.d.ts +45 -0
  51. package/dist/core/artifact-graph/types.js +43 -0
  52. package/dist/core/command-generation/adapters/amazon-q.d.ts +13 -0
  53. package/dist/core/command-generation/adapters/amazon-q.js +26 -0
  54. package/dist/core/command-generation/adapters/antigravity.d.ts +13 -0
  55. package/dist/core/command-generation/adapters/antigravity.js +26 -0
  56. package/dist/core/command-generation/adapters/auggie.d.ts +13 -0
  57. package/dist/core/command-generation/adapters/auggie.js +27 -0
  58. package/dist/core/command-generation/adapters/claude.d.ts +13 -0
  59. package/dist/core/command-generation/adapters/claude.js +50 -0
  60. package/dist/core/command-generation/adapters/cline.d.ts +14 -0
  61. package/dist/core/command-generation/adapters/cline.js +27 -0
  62. package/dist/core/command-generation/adapters/codebuddy.d.ts +13 -0
  63. package/dist/core/command-generation/adapters/codebuddy.js +28 -0
  64. package/dist/core/command-generation/adapters/codex.d.ts +16 -0
  65. package/dist/core/command-generation/adapters/codex.js +39 -0
  66. package/dist/core/command-generation/adapters/continue.d.ts +13 -0
  67. package/dist/core/command-generation/adapters/continue.js +28 -0
  68. package/dist/core/command-generation/adapters/costrict.d.ts +13 -0
  69. package/dist/core/command-generation/adapters/costrict.js +27 -0
  70. package/dist/core/command-generation/adapters/crush.d.ts +13 -0
  71. package/dist/core/command-generation/adapters/crush.js +30 -0
  72. package/dist/core/command-generation/adapters/cursor.d.ts +14 -0
  73. package/dist/core/command-generation/adapters/cursor.js +44 -0
  74. package/dist/core/command-generation/adapters/devagent.d.ts +15 -0
  75. package/dist/core/command-generation/adapters/devagent.js +28 -0
  76. package/dist/core/command-generation/adapters/factory.d.ts +13 -0
  77. package/dist/core/command-generation/adapters/factory.js +27 -0
  78. package/dist/core/command-generation/adapters/gemini.d.ts +13 -0
  79. package/dist/core/command-generation/adapters/gemini.js +26 -0
  80. package/dist/core/command-generation/adapters/github-copilot.d.ts +13 -0
  81. package/dist/core/command-generation/adapters/github-copilot.js +26 -0
  82. package/dist/core/command-generation/adapters/iflow.d.ts +13 -0
  83. package/dist/core/command-generation/adapters/iflow.js +29 -0
  84. package/dist/core/command-generation/adapters/index.d.ts +28 -0
  85. package/dist/core/command-generation/adapters/index.js +28 -0
  86. package/dist/core/command-generation/adapters/kilocode.d.ts +14 -0
  87. package/dist/core/command-generation/adapters/kilocode.js +23 -0
  88. package/dist/core/command-generation/adapters/opencode.d.ts +13 -0
  89. package/dist/core/command-generation/adapters/opencode.js +29 -0
  90. package/dist/core/command-generation/adapters/qoder.d.ts +13 -0
  91. package/dist/core/command-generation/adapters/qoder.js +30 -0
  92. package/dist/core/command-generation/adapters/qwen.d.ts +13 -0
  93. package/dist/core/command-generation/adapters/qwen.js +26 -0
  94. package/dist/core/command-generation/adapters/roocode.d.ts +14 -0
  95. package/dist/core/command-generation/adapters/roocode.js +27 -0
  96. package/dist/core/command-generation/adapters/windsurf.d.ts +14 -0
  97. package/dist/core/command-generation/adapters/windsurf.js +51 -0
  98. package/dist/core/command-generation/generator.d.ts +21 -0
  99. package/dist/core/command-generation/generator.js +27 -0
  100. package/dist/core/command-generation/index.d.ts +22 -0
  101. package/dist/core/command-generation/index.js +24 -0
  102. package/dist/core/command-generation/registry.d.ts +36 -0
  103. package/dist/core/command-generation/registry.js +90 -0
  104. package/dist/core/command-generation/types.d.ts +56 -0
  105. package/dist/core/command-generation/types.js +8 -0
  106. package/dist/core/completions/command-registry.d.ts +7 -0
  107. package/dist/core/completions/command-registry.js +454 -0
  108. package/dist/core/completions/completion-provider.d.ts +60 -0
  109. package/dist/core/completions/completion-provider.js +102 -0
  110. package/dist/core/completions/factory.d.ts +64 -0
  111. package/dist/core/completions/factory.js +75 -0
  112. package/dist/core/completions/generators/bash-generator.d.ts +32 -0
  113. package/dist/core/completions/generators/bash-generator.js +174 -0
  114. package/dist/core/completions/generators/fish-generator.d.ts +32 -0
  115. package/dist/core/completions/generators/fish-generator.js +157 -0
  116. package/dist/core/completions/generators/powershell-generator.d.ts +33 -0
  117. package/dist/core/completions/generators/powershell-generator.js +207 -0
  118. package/dist/core/completions/generators/zsh-generator.d.ts +44 -0
  119. package/dist/core/completions/generators/zsh-generator.js +250 -0
  120. package/dist/core/completions/installers/bash-installer.d.ts +87 -0
  121. package/dist/core/completions/installers/bash-installer.js +318 -0
  122. package/dist/core/completions/installers/fish-installer.d.ts +43 -0
  123. package/dist/core/completions/installers/fish-installer.js +143 -0
  124. package/dist/core/completions/installers/powershell-installer.d.ts +88 -0
  125. package/dist/core/completions/installers/powershell-installer.js +327 -0
  126. package/dist/core/completions/installers/zsh-installer.d.ts +125 -0
  127. package/dist/core/completions/installers/zsh-installer.js +449 -0
  128. package/dist/core/completions/templates/bash-templates.d.ts +6 -0
  129. package/dist/core/completions/templates/bash-templates.js +24 -0
  130. package/dist/core/completions/templates/fish-templates.d.ts +7 -0
  131. package/dist/core/completions/templates/fish-templates.js +39 -0
  132. package/dist/core/completions/templates/powershell-templates.d.ts +6 -0
  133. package/dist/core/completions/templates/powershell-templates.js +25 -0
  134. package/dist/core/completions/templates/zsh-templates.d.ts +6 -0
  135. package/dist/core/completions/templates/zsh-templates.js +36 -0
  136. package/dist/core/completions/types.d.ts +79 -0
  137. package/dist/core/completions/types.js +2 -0
  138. package/dist/core/config-prompts.d.ts +9 -0
  139. package/dist/core/config-prompts.js +34 -0
  140. package/dist/core/config-schema.d.ts +76 -0
  141. package/dist/core/config-schema.js +200 -0
  142. package/dist/core/config.d.ts +17 -0
  143. package/dist/core/config.js +175 -0
  144. package/dist/core/converters/json-converter.d.ts +6 -0
  145. package/dist/core/converters/json-converter.js +51 -0
  146. package/dist/core/global-config.d.ts +39 -0
  147. package/dist/core/global-config.js +115 -0
  148. package/dist/core/index.d.ts +2 -0
  149. package/dist/core/index.js +3 -0
  150. package/dist/core/init.d.ts +32 -0
  151. package/dist/core/init.js +447 -0
  152. package/dist/core/legacy-cleanup.d.ts +162 -0
  153. package/dist/core/legacy-cleanup.js +520 -0
  154. package/dist/core/list.d.ts +9 -0
  155. package/dist/core/list.js +171 -0
  156. package/dist/core/parsers/change-parser.d.ts +13 -0
  157. package/dist/core/parsers/change-parser.js +193 -0
  158. package/dist/core/parsers/markdown-parser.d.ts +22 -0
  159. package/dist/core/parsers/markdown-parser.js +187 -0
  160. package/dist/core/parsers/requirement-blocks.d.ts +37 -0
  161. package/dist/core/parsers/requirement-blocks.js +201 -0
  162. package/dist/core/project-config.d.ts +64 -0
  163. package/dist/core/project-config.js +223 -0
  164. package/dist/core/schemas/base.schema.d.ts +13 -0
  165. package/dist/core/schemas/base.schema.js +13 -0
  166. package/dist/core/schemas/change.schema.d.ts +73 -0
  167. package/dist/core/schemas/change.schema.js +31 -0
  168. package/dist/core/schemas/index.d.ts +4 -0
  169. package/dist/core/schemas/index.js +4 -0
  170. package/dist/core/schemas/spec.schema.d.ts +18 -0
  171. package/dist/core/schemas/spec.schema.js +15 -0
  172. package/dist/core/shared/index.d.ts +8 -0
  173. package/dist/core/shared/index.js +8 -0
  174. package/dist/core/shared/skill-generation.d.ts +42 -0
  175. package/dist/core/shared/skill-generation.js +80 -0
  176. package/dist/core/shared/tool-detection.d.ts +66 -0
  177. package/dist/core/shared/tool-detection.js +140 -0
  178. package/dist/core/specs-apply.d.ts +73 -0
  179. package/dist/core/specs-apply.js +384 -0
  180. package/dist/core/styles/palette.d.ts +7 -0
  181. package/dist/core/styles/palette.js +8 -0
  182. package/dist/core/templates/index.d.ts +8 -0
  183. package/dist/core/templates/index.js +9 -0
  184. package/dist/core/templates/skill-templates.d.ts +122 -0
  185. package/dist/core/templates/skill-templates.js +3437 -0
  186. package/dist/core/update.d.ts +42 -0
  187. package/dist/core/update.js +311 -0
  188. package/dist/core/validation/constants.d.ts +34 -0
  189. package/dist/core/validation/constants.js +40 -0
  190. package/dist/core/validation/types.d.ts +18 -0
  191. package/dist/core/validation/types.js +2 -0
  192. package/dist/core/validation/validator.d.ts +33 -0
  193. package/dist/core/validation/validator.js +409 -0
  194. package/dist/core/view.d.ts +8 -0
  195. package/dist/core/view.js +168 -0
  196. package/dist/index.d.ts +3 -0
  197. package/dist/index.js +3 -0
  198. package/dist/prompts/searchable-multi-select.d.ts +27 -0
  199. package/dist/prompts/searchable-multi-select.js +149 -0
  200. package/dist/telemetry/config.d.ts +32 -0
  201. package/dist/telemetry/config.js +68 -0
  202. package/dist/telemetry/index.d.ts +31 -0
  203. package/dist/telemetry/index.js +145 -0
  204. package/dist/ui/ascii-patterns.d.ts +16 -0
  205. package/dist/ui/ascii-patterns.js +133 -0
  206. package/dist/ui/welcome-screen.d.ts +10 -0
  207. package/dist/ui/welcome-screen.js +146 -0
  208. package/dist/utils/change-metadata.d.ts +51 -0
  209. package/dist/utils/change-metadata.js +147 -0
  210. package/dist/utils/change-utils.d.ts +62 -0
  211. package/dist/utils/change-utils.js +121 -0
  212. package/dist/utils/command-references.d.ts +18 -0
  213. package/dist/utils/command-references.js +20 -0
  214. package/dist/utils/file-system.d.ts +36 -0
  215. package/dist/utils/file-system.js +281 -0
  216. package/dist/utils/index.d.ts +6 -0
  217. package/dist/utils/index.js +9 -0
  218. package/dist/utils/interactive.d.ts +18 -0
  219. package/dist/utils/interactive.js +21 -0
  220. package/dist/utils/item-discovery.d.ts +4 -0
  221. package/dist/utils/item-discovery.js +72 -0
  222. package/dist/utils/match.d.ts +3 -0
  223. package/dist/utils/match.js +22 -0
  224. package/dist/utils/shell-detection.d.ts +20 -0
  225. package/dist/utils/shell-detection.js +41 -0
  226. package/dist/utils/task-progress.d.ts +8 -0
  227. package/dist/utils/task-progress.js +36 -0
  228. package/package.json +83 -0
  229. package/schemas/spec-driven/schema.yaml +151 -0
  230. package/schemas/spec-driven/templates/design.md +21 -0
  231. package/schemas/spec-driven/templates/proposal.md +25 -0
  232. package/schemas/spec-driven/templates/spec.md +10 -0
  233. package/schemas/spec-driven/templates/tasks.md +9 -0
  234. package/scripts/postinstall.js +147 -0
@@ -0,0 +1,257 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ import { getGlobalDataDir } from '../global-config.js';
5
+ import { parseSchema, SchemaValidationError } from './schema.js';
6
+ /**
7
+ * Error thrown when loading a schema fails.
8
+ */
9
+ export class SchemaLoadError extends Error {
10
+ schemaPath;
11
+ cause;
12
+ constructor(message, schemaPath, cause) {
13
+ super(message);
14
+ this.schemaPath = schemaPath;
15
+ this.cause = cause;
16
+ this.name = 'SchemaLoadError';
17
+ }
18
+ }
19
+ /**
20
+ * Gets the package's built-in schemas directory path.
21
+ * Uses import.meta.url to resolve relative to the current module.
22
+ */
23
+ export function getPackageSchemasDir() {
24
+ const currentFile = fileURLToPath(import.meta.url);
25
+ // Navigate from dist/core/artifact-graph/ to package root's schemas/
26
+ return path.join(path.dirname(currentFile), '..', '..', '..', 'schemas');
27
+ }
28
+ /**
29
+ * Gets the user's schema override directory path.
30
+ */
31
+ export function getUserSchemasDir() {
32
+ return path.join(getGlobalDataDir(), 'schemas');
33
+ }
34
+ /**
35
+ * Gets the project-local schemas directory path.
36
+ * @param projectRoot - The project root directory
37
+ * @returns The path to the project's schemas directory
38
+ */
39
+ export function getProjectSchemasDir(projectRoot) {
40
+ return path.join(projectRoot, 'openspec', 'schemas');
41
+ }
42
+ /**
43
+ * Resolves a schema name to its directory path.
44
+ *
45
+ * Resolution order (when projectRoot is provided):
46
+ * 1. Project-local: <projectRoot>/openspec/schemas/<name>/schema.yaml
47
+ * 2. User override: ${XDG_DATA_HOME}/openspec/schemas/<name>/schema.yaml
48
+ * 3. Package built-in: <package>/schemas/<name>/schema.yaml
49
+ *
50
+ * When projectRoot is not provided, only user override and package built-in are checked
51
+ * (backward compatible behavior).
52
+ *
53
+ * @param name - Schema name (e.g., "spec-driven")
54
+ * @param projectRoot - Optional project root directory for project-local schema resolution
55
+ * @returns The path to the schema directory, or null if not found
56
+ */
57
+ export function getSchemaDir(name, projectRoot) {
58
+ // 1. Check project-local directory (if projectRoot provided)
59
+ if (projectRoot) {
60
+ const projectDir = path.join(getProjectSchemasDir(projectRoot), name);
61
+ const projectSchemaPath = path.join(projectDir, 'schema.yaml');
62
+ if (fs.existsSync(projectSchemaPath)) {
63
+ return projectDir;
64
+ }
65
+ }
66
+ // 2. Check user override directory
67
+ const userDir = path.join(getUserSchemasDir(), name);
68
+ const userSchemaPath = path.join(userDir, 'schema.yaml');
69
+ if (fs.existsSync(userSchemaPath)) {
70
+ return userDir;
71
+ }
72
+ // 3. Check package built-in directory
73
+ const packageDir = path.join(getPackageSchemasDir(), name);
74
+ const packageSchemaPath = path.join(packageDir, 'schema.yaml');
75
+ if (fs.existsSync(packageSchemaPath)) {
76
+ return packageDir;
77
+ }
78
+ return null;
79
+ }
80
+ /**
81
+ * Resolves a schema name to a SchemaYaml object.
82
+ *
83
+ * Resolution order (when projectRoot is provided):
84
+ * 1. Project-local: <projectRoot>/openspec/schemas/<name>/schema.yaml
85
+ * 2. User override: ${XDG_DATA_HOME}/openspec/schemas/<name>/schema.yaml
86
+ * 3. Package built-in: <package>/schemas/<name>/schema.yaml
87
+ *
88
+ * When projectRoot is not provided, only user override and package built-in are checked
89
+ * (backward compatible behavior).
90
+ *
91
+ * @param name - Schema name (e.g., "spec-driven")
92
+ * @param projectRoot - Optional project root directory for project-local schema resolution
93
+ * @returns The resolved schema object
94
+ * @throws Error if schema is not found in any location
95
+ */
96
+ export function resolveSchema(name, projectRoot) {
97
+ // Normalize name (remove .yaml extension if provided)
98
+ const normalizedName = name.replace(/\.ya?ml$/, '');
99
+ const schemaDir = getSchemaDir(normalizedName, projectRoot);
100
+ if (!schemaDir) {
101
+ const availableSchemas = listSchemas(projectRoot);
102
+ throw new Error(`Schema '${normalizedName}' not found. Available schemas: ${availableSchemas.join(', ')}`);
103
+ }
104
+ const schemaPath = path.join(schemaDir, 'schema.yaml');
105
+ // Load and parse the schema
106
+ let content;
107
+ try {
108
+ content = fs.readFileSync(schemaPath, 'utf-8');
109
+ }
110
+ catch (err) {
111
+ const ioError = err instanceof Error ? err : new Error(String(err));
112
+ throw new SchemaLoadError(`Failed to read schema at '${schemaPath}': ${ioError.message}`, schemaPath, ioError);
113
+ }
114
+ try {
115
+ return parseSchema(content);
116
+ }
117
+ catch (err) {
118
+ if (err instanceof SchemaValidationError) {
119
+ throw new SchemaLoadError(`Invalid schema at '${schemaPath}': ${err.message}`, schemaPath, err);
120
+ }
121
+ const parseError = err instanceof Error ? err : new Error(String(err));
122
+ throw new SchemaLoadError(`Failed to parse schema at '${schemaPath}': ${parseError.message}`, schemaPath, parseError);
123
+ }
124
+ }
125
+ /**
126
+ * Lists all available schema names.
127
+ * Combines project-local, user override, and package built-in schemas.
128
+ *
129
+ * @param projectRoot - Optional project root directory for project-local schema resolution
130
+ */
131
+ export function listSchemas(projectRoot) {
132
+ const schemas = new Set();
133
+ // Add package built-in schemas
134
+ const packageDir = getPackageSchemasDir();
135
+ if (fs.existsSync(packageDir)) {
136
+ for (const entry of fs.readdirSync(packageDir, { withFileTypes: true })) {
137
+ if (entry.isDirectory()) {
138
+ const schemaPath = path.join(packageDir, entry.name, 'schema.yaml');
139
+ if (fs.existsSync(schemaPath)) {
140
+ schemas.add(entry.name);
141
+ }
142
+ }
143
+ }
144
+ }
145
+ // Add user override schemas (may override package schemas)
146
+ const userDir = getUserSchemasDir();
147
+ if (fs.existsSync(userDir)) {
148
+ for (const entry of fs.readdirSync(userDir, { withFileTypes: true })) {
149
+ if (entry.isDirectory()) {
150
+ const schemaPath = path.join(userDir, entry.name, 'schema.yaml');
151
+ if (fs.existsSync(schemaPath)) {
152
+ schemas.add(entry.name);
153
+ }
154
+ }
155
+ }
156
+ }
157
+ // Add project-local schemas (if projectRoot provided)
158
+ if (projectRoot) {
159
+ const projectDir = getProjectSchemasDir(projectRoot);
160
+ if (fs.existsSync(projectDir)) {
161
+ for (const entry of fs.readdirSync(projectDir, { withFileTypes: true })) {
162
+ if (entry.isDirectory()) {
163
+ const schemaPath = path.join(projectDir, entry.name, 'schema.yaml');
164
+ if (fs.existsSync(schemaPath)) {
165
+ schemas.add(entry.name);
166
+ }
167
+ }
168
+ }
169
+ }
170
+ }
171
+ return Array.from(schemas).sort();
172
+ }
173
+ /**
174
+ * Lists all available schemas with their descriptions and artifact lists.
175
+ * Useful for agent skills to present schema selection to users.
176
+ *
177
+ * @param projectRoot - Optional project root directory for project-local schema resolution
178
+ */
179
+ export function listSchemasWithInfo(projectRoot) {
180
+ const schemas = [];
181
+ const seenNames = new Set();
182
+ // Add project-local schemas first (highest priority, if projectRoot provided)
183
+ if (projectRoot) {
184
+ const projectDir = getProjectSchemasDir(projectRoot);
185
+ if (fs.existsSync(projectDir)) {
186
+ for (const entry of fs.readdirSync(projectDir, { withFileTypes: true })) {
187
+ if (entry.isDirectory()) {
188
+ const schemaPath = path.join(projectDir, entry.name, 'schema.yaml');
189
+ if (fs.existsSync(schemaPath)) {
190
+ try {
191
+ const schema = parseSchema(fs.readFileSync(schemaPath, 'utf-8'));
192
+ schemas.push({
193
+ name: entry.name,
194
+ description: schema.description || '',
195
+ artifacts: schema.artifacts.map((a) => a.id),
196
+ source: 'project',
197
+ });
198
+ seenNames.add(entry.name);
199
+ }
200
+ catch {
201
+ // Skip invalid schemas
202
+ }
203
+ }
204
+ }
205
+ }
206
+ }
207
+ }
208
+ // Add user override schemas (if not overridden by project)
209
+ const userDir = getUserSchemasDir();
210
+ if (fs.existsSync(userDir)) {
211
+ for (const entry of fs.readdirSync(userDir, { withFileTypes: true })) {
212
+ if (entry.isDirectory() && !seenNames.has(entry.name)) {
213
+ const schemaPath = path.join(userDir, entry.name, 'schema.yaml');
214
+ if (fs.existsSync(schemaPath)) {
215
+ try {
216
+ const schema = parseSchema(fs.readFileSync(schemaPath, 'utf-8'));
217
+ schemas.push({
218
+ name: entry.name,
219
+ description: schema.description || '',
220
+ artifacts: schema.artifacts.map((a) => a.id),
221
+ source: 'user',
222
+ });
223
+ seenNames.add(entry.name);
224
+ }
225
+ catch {
226
+ // Skip invalid schemas
227
+ }
228
+ }
229
+ }
230
+ }
231
+ }
232
+ // Add package built-in schemas (if not overridden by project or user)
233
+ const packageDir = getPackageSchemasDir();
234
+ if (fs.existsSync(packageDir)) {
235
+ for (const entry of fs.readdirSync(packageDir, { withFileTypes: true })) {
236
+ if (entry.isDirectory() && !seenNames.has(entry.name)) {
237
+ const schemaPath = path.join(packageDir, entry.name, 'schema.yaml');
238
+ if (fs.existsSync(schemaPath)) {
239
+ try {
240
+ const schema = parseSchema(fs.readFileSync(schemaPath, 'utf-8'));
241
+ schemas.push({
242
+ name: entry.name,
243
+ description: schema.description || '',
244
+ artifacts: schema.artifacts.map((a) => a.id),
245
+ source: 'package',
246
+ });
247
+ }
248
+ catch {
249
+ // Skip invalid schemas
250
+ }
251
+ }
252
+ }
253
+ }
254
+ }
255
+ return schemas.sort((a, b) => a.name.localeCompare(b.name));
256
+ }
257
+ //# sourceMappingURL=resolver.js.map
@@ -0,0 +1,13 @@
1
+ import { type SchemaYaml } from './types.js';
2
+ export declare class SchemaValidationError extends Error {
3
+ constructor(message: string);
4
+ }
5
+ /**
6
+ * Loads and validates an artifact schema from a YAML file.
7
+ */
8
+ export declare function loadSchema(filePath: string): SchemaYaml;
9
+ /**
10
+ * Parses and validates an artifact schema from YAML content.
11
+ */
12
+ export declare function parseSchema(yamlContent: string): SchemaYaml;
13
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1,108 @@
1
+ import * as fs from 'node:fs';
2
+ import { parse as parseYaml } from 'yaml';
3
+ import { SchemaYamlSchema } from './types.js';
4
+ export class SchemaValidationError extends Error {
5
+ constructor(message) {
6
+ super(message);
7
+ this.name = 'SchemaValidationError';
8
+ }
9
+ }
10
+ /**
11
+ * Loads and validates an artifact schema from a YAML file.
12
+ */
13
+ export function loadSchema(filePath) {
14
+ const content = fs.readFileSync(filePath, 'utf-8');
15
+ return parseSchema(content);
16
+ }
17
+ /**
18
+ * Parses and validates an artifact schema from YAML content.
19
+ */
20
+ export function parseSchema(yamlContent) {
21
+ const parsed = parseYaml(yamlContent);
22
+ // Validate with Zod
23
+ const result = SchemaYamlSchema.safeParse(parsed);
24
+ if (!result.success) {
25
+ const errors = result.error.issues.map(e => `${e.path.join('.')}: ${e.message}`).join(', ');
26
+ throw new SchemaValidationError(`Invalid schema: ${errors}`);
27
+ }
28
+ const schema = result.data;
29
+ // Check for duplicate artifact IDs
30
+ validateNoDuplicateIds(schema.artifacts);
31
+ // Check that all requires references are valid
32
+ validateRequiresReferences(schema.artifacts);
33
+ // Check for cycles
34
+ validateNoCycles(schema.artifacts);
35
+ return schema;
36
+ }
37
+ /**
38
+ * Validates that there are no duplicate artifact IDs.
39
+ */
40
+ function validateNoDuplicateIds(artifacts) {
41
+ const seen = new Set();
42
+ for (const artifact of artifacts) {
43
+ if (seen.has(artifact.id)) {
44
+ throw new SchemaValidationError(`Duplicate artifact ID: ${artifact.id}`);
45
+ }
46
+ seen.add(artifact.id);
47
+ }
48
+ }
49
+ /**
50
+ * Validates that all `requires` references point to valid artifact IDs.
51
+ */
52
+ function validateRequiresReferences(artifacts) {
53
+ const validIds = new Set(artifacts.map(a => a.id));
54
+ for (const artifact of artifacts) {
55
+ for (const req of artifact.requires) {
56
+ if (!validIds.has(req)) {
57
+ throw new SchemaValidationError(`Invalid dependency reference in artifact '${artifact.id}': '${req}' does not exist`);
58
+ }
59
+ }
60
+ }
61
+ }
62
+ /**
63
+ * Validates that there are no cyclic dependencies.
64
+ * Uses DFS to detect cycles and reports the full cycle path.
65
+ */
66
+ function validateNoCycles(artifacts) {
67
+ const artifactMap = new Map(artifacts.map(a => [a.id, a]));
68
+ const visited = new Set();
69
+ const inStack = new Set();
70
+ const parent = new Map();
71
+ function dfs(id) {
72
+ visited.add(id);
73
+ inStack.add(id);
74
+ const artifact = artifactMap.get(id);
75
+ if (!artifact)
76
+ return null;
77
+ for (const dep of artifact.requires) {
78
+ if (!visited.has(dep)) {
79
+ parent.set(dep, id);
80
+ const cycle = dfs(dep);
81
+ if (cycle)
82
+ return cycle;
83
+ }
84
+ else if (inStack.has(dep)) {
85
+ // Found a cycle - reconstruct the path
86
+ const cyclePath = [dep];
87
+ let current = id;
88
+ while (current !== dep) {
89
+ cyclePath.unshift(current);
90
+ current = parent.get(current);
91
+ }
92
+ cyclePath.unshift(dep);
93
+ return cyclePath.join(' → ');
94
+ }
95
+ }
96
+ inStack.delete(id);
97
+ return null;
98
+ }
99
+ for (const artifact of artifacts) {
100
+ if (!visited.has(artifact.id)) {
101
+ const cycle = dfs(artifact.id);
102
+ if (cycle) {
103
+ throw new SchemaValidationError(`Cyclic dependency detected: ${cycle}`);
104
+ }
105
+ }
106
+ }
107
+ }
108
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1,12 @@
1
+ import type { CompletedSet } from './types.js';
2
+ import type { ArtifactGraph } from './graph.js';
3
+ /**
4
+ * Detects which artifacts are completed by checking file existence in the change directory.
5
+ * Returns a Set of completed artifact IDs.
6
+ *
7
+ * @param graph - The artifact graph to check
8
+ * @param changeDir - The change directory to scan for files
9
+ * @returns Set of artifact IDs whose generated files exist
10
+ */
11
+ export declare function detectCompleted(graph: ArtifactGraph, changeDir: string): CompletedSet;
12
+ //# sourceMappingURL=state.d.ts.map
@@ -0,0 +1,54 @@
1
+ import * as fs from 'node:fs';
2
+ import * as path from 'node:path';
3
+ import fg from 'fast-glob';
4
+ import { FileSystemUtils } from '../../utils/file-system.js';
5
+ /**
6
+ * Detects which artifacts are completed by checking file existence in the change directory.
7
+ * Returns a Set of completed artifact IDs.
8
+ *
9
+ * @param graph - The artifact graph to check
10
+ * @param changeDir - The change directory to scan for files
11
+ * @returns Set of artifact IDs whose generated files exist
12
+ */
13
+ export function detectCompleted(graph, changeDir) {
14
+ const completed = new Set();
15
+ // Handle missing change directory gracefully
16
+ if (!fs.existsSync(changeDir)) {
17
+ return completed;
18
+ }
19
+ for (const artifact of graph.getAllArtifacts()) {
20
+ if (isArtifactComplete(artifact.generates, changeDir)) {
21
+ completed.add(artifact.id);
22
+ }
23
+ }
24
+ return completed;
25
+ }
26
+ /**
27
+ * Checks if an artifact is complete by checking if its generated file(s) exist.
28
+ * Supports both simple paths and glob patterns.
29
+ */
30
+ function isArtifactComplete(generates, changeDir) {
31
+ const fullPattern = path.join(changeDir, generates);
32
+ // Check if it's a glob pattern
33
+ if (isGlobPattern(generates)) {
34
+ return hasGlobMatches(fullPattern);
35
+ }
36
+ // Simple file path - check if file exists
37
+ return fs.existsSync(fullPattern);
38
+ }
39
+ /**
40
+ * Checks if a path contains glob pattern characters.
41
+ */
42
+ function isGlobPattern(pattern) {
43
+ return pattern.includes('*') || pattern.includes('?') || pattern.includes('[');
44
+ }
45
+ /**
46
+ * Checks if a glob pattern has any matches.
47
+ * Normalizes Windows backslashes to forward slashes for cross-platform glob compatibility.
48
+ */
49
+ function hasGlobMatches(pattern) {
50
+ const normalizedPattern = FileSystemUtils.toPosixPath(pattern);
51
+ const matches = fg.sync(normalizedPattern, { onlyFiles: true });
52
+ return matches.length > 0;
53
+ }
54
+ //# sourceMappingURL=state.js.map
@@ -0,0 +1,45 @@
1
+ import { z } from 'zod';
2
+ export declare const ArtifactSchema: z.ZodObject<{
3
+ id: z.ZodString;
4
+ generates: z.ZodString;
5
+ description: z.ZodString;
6
+ template: z.ZodString;
7
+ instruction: z.ZodOptional<z.ZodString>;
8
+ requires: z.ZodDefault<z.ZodArray<z.ZodString>>;
9
+ }, z.core.$strip>;
10
+ export declare const ApplyPhaseSchema: z.ZodObject<{
11
+ requires: z.ZodArray<z.ZodString>;
12
+ tracks: z.ZodOptional<z.ZodNullable<z.ZodString>>;
13
+ instruction: z.ZodOptional<z.ZodString>;
14
+ }, z.core.$strip>;
15
+ export declare const SchemaYamlSchema: z.ZodObject<{
16
+ name: z.ZodString;
17
+ version: z.ZodNumber;
18
+ description: z.ZodOptional<z.ZodString>;
19
+ artifacts: z.ZodArray<z.ZodObject<{
20
+ id: z.ZodString;
21
+ generates: z.ZodString;
22
+ description: z.ZodString;
23
+ template: z.ZodString;
24
+ instruction: z.ZodOptional<z.ZodString>;
25
+ requires: z.ZodDefault<z.ZodArray<z.ZodString>>;
26
+ }, z.core.$strip>>;
27
+ apply: z.ZodOptional<z.ZodObject<{
28
+ requires: z.ZodArray<z.ZodString>;
29
+ tracks: z.ZodOptional<z.ZodNullable<z.ZodString>>;
30
+ instruction: z.ZodOptional<z.ZodString>;
31
+ }, z.core.$strip>>;
32
+ }, z.core.$strip>;
33
+ export type Artifact = z.infer<typeof ArtifactSchema>;
34
+ export type ApplyPhase = z.infer<typeof ApplyPhaseSchema>;
35
+ export type SchemaYaml = z.infer<typeof SchemaYamlSchema>;
36
+ export declare const ChangeMetadataSchema: z.ZodObject<{
37
+ schema: z.ZodString;
38
+ created: z.ZodOptional<z.ZodString>;
39
+ }, z.core.$strip>;
40
+ export type ChangeMetadata = z.infer<typeof ChangeMetadataSchema>;
41
+ export type CompletedSet = Set<string>;
42
+ export interface BlockedArtifacts {
43
+ [artifactId: string]: string[];
44
+ }
45
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1,43 @@
1
+ import { z } from 'zod';
2
+ // Artifact definition schema
3
+ export const ArtifactSchema = z.object({
4
+ id: z.string().min(1, { error: 'Artifact ID is required' }),
5
+ generates: z.string().min(1, { error: 'generates field is required' }),
6
+ description: z.string(),
7
+ template: z.string().min(1, { error: 'template field is required' }),
8
+ instruction: z.string().optional(),
9
+ requires: z.array(z.string()).default([]),
10
+ });
11
+ // Apply phase configuration for schema-aware apply instructions
12
+ export const ApplyPhaseSchema = z.object({
13
+ // Artifact IDs that must exist before apply is available
14
+ requires: z.array(z.string()).min(1, { error: 'At least one required artifact' }),
15
+ // Path to file with checkboxes for progress (relative to change dir), or null if no tracking
16
+ tracks: z.string().nullable().optional(),
17
+ // Custom guidance for the apply phase
18
+ instruction: z.string().optional(),
19
+ });
20
+ // Full schema YAML structure
21
+ export const SchemaYamlSchema = z.object({
22
+ name: z.string().min(1, { error: 'Schema name is required' }),
23
+ version: z.number().int().positive({ error: 'Version must be a positive integer' }),
24
+ description: z.string().optional(),
25
+ artifacts: z.array(ArtifactSchema).min(1, { error: 'At least one artifact required' }),
26
+ // Optional apply phase configuration (for schema-aware apply instructions)
27
+ apply: ApplyPhaseSchema.optional(),
28
+ });
29
+ // Per-change metadata schema
30
+ // Note: schema field is validated at parse time against available schemas
31
+ // using a lazy import to avoid circular dependencies
32
+ export const ChangeMetadataSchema = z.object({
33
+ // Required: which workflow schema this change uses
34
+ schema: z.string().min(1, { message: 'schema is required' }),
35
+ // Optional: creation timestamp (ISO date string)
36
+ created: z
37
+ .string()
38
+ .regex(/^\d{4}-\d{2}-\d{2}$/, {
39
+ message: 'created must be YYYY-MM-DD format',
40
+ })
41
+ .optional(),
42
+ });
43
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Amazon Q Developer Command Adapter
3
+ *
4
+ * Formats commands for Amazon Q Developer following its frontmatter specification.
5
+ */
6
+ import type { ToolCommandAdapter } from '../types.js';
7
+ /**
8
+ * Amazon Q adapter for command generation.
9
+ * File path: .amazonq/prompts/opsx-<id>.md
10
+ * Frontmatter: description
11
+ */
12
+ export declare const amazonQAdapter: ToolCommandAdapter;
13
+ //# sourceMappingURL=amazon-q.d.ts.map
@@ -0,0 +1,26 @@
1
+ /**
2
+ * Amazon Q Developer Command Adapter
3
+ *
4
+ * Formats commands for Amazon Q Developer following its frontmatter specification.
5
+ */
6
+ import path from 'path';
7
+ /**
8
+ * Amazon Q adapter for command generation.
9
+ * File path: .amazonq/prompts/opsx-<id>.md
10
+ * Frontmatter: description
11
+ */
12
+ export const amazonQAdapter = {
13
+ toolId: 'amazon-q',
14
+ getFilePath(commandId) {
15
+ return path.join('.amazonq', 'prompts', `opsx-${commandId}.md`);
16
+ },
17
+ formatFile(content) {
18
+ return `---
19
+ description: ${content.description}
20
+ ---
21
+
22
+ ${content.body}
23
+ `;
24
+ },
25
+ };
26
+ //# sourceMappingURL=amazon-q.js.map
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Antigravity Command Adapter
3
+ *
4
+ * Formats commands for Antigravity following its frontmatter specification.
5
+ */
6
+ import type { ToolCommandAdapter } from '../types.js';
7
+ /**
8
+ * Antigravity adapter for command generation.
9
+ * File path: .agent/workflows/opsx-<id>.md
10
+ * Frontmatter: description
11
+ */
12
+ export declare const antigravityAdapter: ToolCommandAdapter;
13
+ //# sourceMappingURL=antigravity.d.ts.map
@@ -0,0 +1,26 @@
1
+ /**
2
+ * Antigravity Command Adapter
3
+ *
4
+ * Formats commands for Antigravity following its frontmatter specification.
5
+ */
6
+ import path from 'path';
7
+ /**
8
+ * Antigravity adapter for command generation.
9
+ * File path: .agent/workflows/opsx-<id>.md
10
+ * Frontmatter: description
11
+ */
12
+ export const antigravityAdapter = {
13
+ toolId: 'antigravity',
14
+ getFilePath(commandId) {
15
+ return path.join('.agent', 'workflows', `opsx-${commandId}.md`);
16
+ },
17
+ formatFile(content) {
18
+ return `---
19
+ description: ${content.description}
20
+ ---
21
+
22
+ ${content.body}
23
+ `;
24
+ },
25
+ };
26
+ //# sourceMappingURL=antigravity.js.map
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Auggie (Augment CLI) Command Adapter
3
+ *
4
+ * Formats commands for Auggie following its frontmatter specification.
5
+ */
6
+ import type { ToolCommandAdapter } from '../types.js';
7
+ /**
8
+ * Auggie adapter for command generation.
9
+ * File path: .augment/commands/opsx-<id>.md
10
+ * Frontmatter: description, argument-hint
11
+ */
12
+ export declare const auggieAdapter: ToolCommandAdapter;
13
+ //# sourceMappingURL=auggie.d.ts.map
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Auggie (Augment CLI) Command Adapter
3
+ *
4
+ * Formats commands for Auggie following its frontmatter specification.
5
+ */
6
+ import path from 'path';
7
+ /**
8
+ * Auggie adapter for command generation.
9
+ * File path: .augment/commands/opsx-<id>.md
10
+ * Frontmatter: description, argument-hint
11
+ */
12
+ export const auggieAdapter = {
13
+ toolId: 'auggie',
14
+ getFilePath(commandId) {
15
+ return path.join('.augment', 'commands', `opsx-${commandId}.md`);
16
+ },
17
+ formatFile(content) {
18
+ return `---
19
+ description: ${content.description}
20
+ argument-hint: command arguments
21
+ ---
22
+
23
+ ${content.body}
24
+ `;
25
+ },
26
+ };
27
+ //# sourceMappingURL=auggie.js.map
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Claude Code Command Adapter
3
+ *
4
+ * Formats commands for Claude Code following its frontmatter specification.
5
+ */
6
+ import type { ToolCommandAdapter } from '../types.js';
7
+ /**
8
+ * Claude Code adapter for command generation.
9
+ * File path: .claude/commands/opsx/<id>.md
10
+ * Frontmatter: name, description, category, tags
11
+ */
12
+ export declare const claudeAdapter: ToolCommandAdapter;
13
+ //# sourceMappingURL=claude.d.ts.map