@superkou/openspec 1.4.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 (373) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +213 -0
  3. package/bin/openspec.js +5 -0
  4. package/dist/cli/index.d.ts +5 -0
  5. package/dist/cli/index.js +544 -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 +264 -0
  10. package/dist/commands/config.d.ts +36 -0
  11. package/dist/commands/config.js +611 -0
  12. package/dist/commands/context-store.d.ts +3 -0
  13. package/dist/commands/context-store.js +475 -0
  14. package/dist/commands/feedback.d.ts +9 -0
  15. package/dist/commands/feedback.js +183 -0
  16. package/dist/commands/initiative.d.ts +13 -0
  17. package/dist/commands/initiative.js +318 -0
  18. package/dist/commands/schema.d.ts +6 -0
  19. package/dist/commands/schema.js +869 -0
  20. package/dist/commands/show.d.ts +14 -0
  21. package/dist/commands/show.js +132 -0
  22. package/dist/commands/spec.d.ts +15 -0
  23. package/dist/commands/spec.js +225 -0
  24. package/dist/commands/validate.d.ts +24 -0
  25. package/dist/commands/validate.js +294 -0
  26. package/dist/commands/workflow/index.d.ts +19 -0
  27. package/dist/commands/workflow/index.js +13 -0
  28. package/dist/commands/workflow/initiative-link.d.ts +24 -0
  29. package/dist/commands/workflow/initiative-link.js +47 -0
  30. package/dist/commands/workflow/instructions.d.ts +29 -0
  31. package/dist/commands/workflow/instructions.js +344 -0
  32. package/dist/commands/workflow/new-change.d.ts +17 -0
  33. package/dist/commands/workflow/new-change.js +141 -0
  34. package/dist/commands/workflow/schemas.d.ts +10 -0
  35. package/dist/commands/workflow/schemas.js +34 -0
  36. package/dist/commands/workflow/set-change.d.ts +13 -0
  37. package/dist/commands/workflow/set-change.js +87 -0
  38. package/dist/commands/workflow/shared.d.ts +59 -0
  39. package/dist/commands/workflow/shared.js +116 -0
  40. package/dist/commands/workflow/status.d.ts +14 -0
  41. package/dist/commands/workflow/status.js +90 -0
  42. package/dist/commands/workflow/templates.d.ts +16 -0
  43. package/dist/commands/workflow/templates.js +69 -0
  44. package/dist/commands/workspace/context-status.d.ts +4 -0
  45. package/dist/commands/workspace/context-status.js +59 -0
  46. package/dist/commands/workspace/open-target-selection.d.ts +13 -0
  47. package/dist/commands/workspace/open-target-selection.js +146 -0
  48. package/dist/commands/workspace/open-view.d.ts +62 -0
  49. package/dist/commands/workspace/open-view.js +249 -0
  50. package/dist/commands/workspace/open.d.ts +37 -0
  51. package/dist/commands/workspace/open.js +110 -0
  52. package/dist/commands/workspace/opener-selection.d.ts +11 -0
  53. package/dist/commands/workspace/opener-selection.js +98 -0
  54. package/dist/commands/workspace/operations.d.ts +29 -0
  55. package/dist/commands/workspace/operations.js +543 -0
  56. package/dist/commands/workspace/prompt-theme.d.ts +29 -0
  57. package/dist/commands/workspace/prompt-theme.js +24 -0
  58. package/dist/commands/workspace/registration.d.ts +13 -0
  59. package/dist/commands/workspace/registration.js +84 -0
  60. package/dist/commands/workspace/selection.d.ts +8 -0
  61. package/dist/commands/workspace/selection.js +104 -0
  62. package/dist/commands/workspace/setup-prompts.d.ts +13 -0
  63. package/dist/commands/workspace/setup-prompts.js +121 -0
  64. package/dist/commands/workspace/types.d.ts +103 -0
  65. package/dist/commands/workspace/types.js +36 -0
  66. package/dist/commands/workspace.d.ts +5 -0
  67. package/dist/commands/workspace.js +577 -0
  68. package/dist/core/archive.d.ts +11 -0
  69. package/dist/core/archive.js +318 -0
  70. package/dist/core/artifact-graph/graph.d.ts +56 -0
  71. package/dist/core/artifact-graph/graph.js +141 -0
  72. package/dist/core/artifact-graph/index.d.ts +9 -0
  73. package/dist/core/artifact-graph/index.js +14 -0
  74. package/dist/core/artifact-graph/instruction-loader.d.ts +183 -0
  75. package/dist/core/artifact-graph/instruction-loader.js +256 -0
  76. package/dist/core/artifact-graph/outputs.d.ts +14 -0
  77. package/dist/core/artifact-graph/outputs.js +39 -0
  78. package/dist/core/artifact-graph/resolver.d.ts +81 -0
  79. package/dist/core/artifact-graph/resolver.js +257 -0
  80. package/dist/core/artifact-graph/schema.d.ts +13 -0
  81. package/dist/core/artifact-graph/schema.js +108 -0
  82. package/dist/core/artifact-graph/state.d.ts +12 -0
  83. package/dist/core/artifact-graph/state.js +31 -0
  84. package/dist/core/artifact-graph/types.d.ts +40 -0
  85. package/dist/core/artifact-graph/types.js +29 -0
  86. package/dist/core/available-tools.d.ts +17 -0
  87. package/dist/core/available-tools.js +43 -0
  88. package/dist/core/change-metadata/index.d.ts +2 -0
  89. package/dist/core/change-metadata/index.js +2 -0
  90. package/dist/core/change-metadata/schema.d.ts +18 -0
  91. package/dist/core/change-metadata/schema.js +28 -0
  92. package/dist/core/change-status-policy.d.ts +50 -0
  93. package/dist/core/change-status-policy.js +70 -0
  94. package/dist/core/collections/index.d.ts +3 -0
  95. package/dist/core/collections/index.js +3 -0
  96. package/dist/core/collections/initiatives/collection.d.ts +4 -0
  97. package/dist/core/collections/initiatives/collection.js +17 -0
  98. package/dist/core/collections/initiatives/index.d.ts +6 -0
  99. package/dist/core/collections/initiatives/index.js +6 -0
  100. package/dist/core/collections/initiatives/operations.d.ts +49 -0
  101. package/dist/core/collections/initiatives/operations.js +175 -0
  102. package/dist/core/collections/initiatives/resolution.d.ts +87 -0
  103. package/dist/core/collections/initiatives/resolution.js +374 -0
  104. package/dist/core/collections/initiatives/schema.d.ts +41 -0
  105. package/dist/core/collections/initiatives/schema.js +134 -0
  106. package/dist/core/collections/initiatives/templates.d.ts +12 -0
  107. package/dist/core/collections/initiatives/templates.js +90 -0
  108. package/dist/core/collections/runtime.d.ts +46 -0
  109. package/dist/core/collections/runtime.js +194 -0
  110. package/dist/core/command-generation/adapters/amazon-q.d.ts +13 -0
  111. package/dist/core/command-generation/adapters/amazon-q.js +26 -0
  112. package/dist/core/command-generation/adapters/antigravity.d.ts +13 -0
  113. package/dist/core/command-generation/adapters/antigravity.js +26 -0
  114. package/dist/core/command-generation/adapters/auggie.d.ts +13 -0
  115. package/dist/core/command-generation/adapters/auggie.js +27 -0
  116. package/dist/core/command-generation/adapters/bob.d.ts +14 -0
  117. package/dist/core/command-generation/adapters/bob.js +45 -0
  118. package/dist/core/command-generation/adapters/claude.d.ts +13 -0
  119. package/dist/core/command-generation/adapters/claude.js +50 -0
  120. package/dist/core/command-generation/adapters/cline.d.ts +14 -0
  121. package/dist/core/command-generation/adapters/cline.js +27 -0
  122. package/dist/core/command-generation/adapters/codebuddy.d.ts +13 -0
  123. package/dist/core/command-generation/adapters/codebuddy.js +28 -0
  124. package/dist/core/command-generation/adapters/codex.d.ts +16 -0
  125. package/dist/core/command-generation/adapters/codex.js +39 -0
  126. package/dist/core/command-generation/adapters/continue.d.ts +13 -0
  127. package/dist/core/command-generation/adapters/continue.js +28 -0
  128. package/dist/core/command-generation/adapters/costrict.d.ts +13 -0
  129. package/dist/core/command-generation/adapters/costrict.js +27 -0
  130. package/dist/core/command-generation/adapters/crush.d.ts +13 -0
  131. package/dist/core/command-generation/adapters/crush.js +30 -0
  132. package/dist/core/command-generation/adapters/cursor.d.ts +14 -0
  133. package/dist/core/command-generation/adapters/cursor.js +44 -0
  134. package/dist/core/command-generation/adapters/factory.d.ts +13 -0
  135. package/dist/core/command-generation/adapters/factory.js +27 -0
  136. package/dist/core/command-generation/adapters/gemini.d.ts +13 -0
  137. package/dist/core/command-generation/adapters/gemini.js +26 -0
  138. package/dist/core/command-generation/adapters/github-copilot.d.ts +13 -0
  139. package/dist/core/command-generation/adapters/github-copilot.js +26 -0
  140. package/dist/core/command-generation/adapters/iflow.d.ts +13 -0
  141. package/dist/core/command-generation/adapters/iflow.js +29 -0
  142. package/dist/core/command-generation/adapters/index.d.ts +32 -0
  143. package/dist/core/command-generation/adapters/index.js +32 -0
  144. package/dist/core/command-generation/adapters/junie.d.ts +13 -0
  145. package/dist/core/command-generation/adapters/junie.js +26 -0
  146. package/dist/core/command-generation/adapters/kilocode.d.ts +14 -0
  147. package/dist/core/command-generation/adapters/kilocode.js +23 -0
  148. package/dist/core/command-generation/adapters/kiro.d.ts +13 -0
  149. package/dist/core/command-generation/adapters/kiro.js +26 -0
  150. package/dist/core/command-generation/adapters/lingma.d.ts +13 -0
  151. package/dist/core/command-generation/adapters/lingma.js +30 -0
  152. package/dist/core/command-generation/adapters/opencode.d.ts +13 -0
  153. package/dist/core/command-generation/adapters/opencode.js +29 -0
  154. package/dist/core/command-generation/adapters/pi.d.ts +18 -0
  155. package/dist/core/command-generation/adapters/pi.js +55 -0
  156. package/dist/core/command-generation/adapters/qoder.d.ts +13 -0
  157. package/dist/core/command-generation/adapters/qoder.js +30 -0
  158. package/dist/core/command-generation/adapters/qwen.d.ts +13 -0
  159. package/dist/core/command-generation/adapters/qwen.js +26 -0
  160. package/dist/core/command-generation/adapters/roocode.d.ts +14 -0
  161. package/dist/core/command-generation/adapters/roocode.js +27 -0
  162. package/dist/core/command-generation/adapters/windsurf.d.ts +14 -0
  163. package/dist/core/command-generation/adapters/windsurf.js +51 -0
  164. package/dist/core/command-generation/generator.d.ts +21 -0
  165. package/dist/core/command-generation/generator.js +27 -0
  166. package/dist/core/command-generation/index.d.ts +22 -0
  167. package/dist/core/command-generation/index.js +24 -0
  168. package/dist/core/command-generation/registry.d.ts +36 -0
  169. package/dist/core/command-generation/registry.js +98 -0
  170. package/dist/core/command-generation/types.d.ts +56 -0
  171. package/dist/core/command-generation/types.js +8 -0
  172. package/dist/core/completions/command-registry.d.ts +3 -0
  173. package/dist/core/completions/command-registry.js +961 -0
  174. package/dist/core/completions/completion-provider.d.ts +71 -0
  175. package/dist/core/completions/completion-provider.js +129 -0
  176. package/dist/core/completions/factory.d.ts +64 -0
  177. package/dist/core/completions/factory.js +75 -0
  178. package/dist/core/completions/generators/bash-generator.d.ts +35 -0
  179. package/dist/core/completions/generators/bash-generator.js +230 -0
  180. package/dist/core/completions/generators/fish-generator.d.ts +32 -0
  181. package/dist/core/completions/generators/fish-generator.js +160 -0
  182. package/dist/core/completions/generators/powershell-generator.d.ts +36 -0
  183. package/dist/core/completions/generators/powershell-generator.js +266 -0
  184. package/dist/core/completions/generators/zsh-generator.d.ts +47 -0
  185. package/dist/core/completions/generators/zsh-generator.js +274 -0
  186. package/dist/core/completions/installers/bash-installer.d.ts +87 -0
  187. package/dist/core/completions/installers/bash-installer.js +318 -0
  188. package/dist/core/completions/installers/fish-installer.d.ts +43 -0
  189. package/dist/core/completions/installers/fish-installer.js +143 -0
  190. package/dist/core/completions/installers/powershell-installer.d.ts +102 -0
  191. package/dist/core/completions/installers/powershell-installer.js +387 -0
  192. package/dist/core/completions/installers/zsh-installer.d.ts +117 -0
  193. package/dist/core/completions/installers/zsh-installer.js +421 -0
  194. package/dist/core/completions/shared-flags.d.ts +12 -0
  195. package/dist/core/completions/shared-flags.js +28 -0
  196. package/dist/core/completions/templates/bash-templates.d.ts +6 -0
  197. package/dist/core/completions/templates/bash-templates.js +30 -0
  198. package/dist/core/completions/templates/fish-templates.d.ts +7 -0
  199. package/dist/core/completions/templates/fish-templates.js +45 -0
  200. package/dist/core/completions/templates/powershell-templates.d.ts +6 -0
  201. package/dist/core/completions/templates/powershell-templates.js +34 -0
  202. package/dist/core/completions/templates/zsh-templates.d.ts +6 -0
  203. package/dist/core/completions/templates/zsh-templates.js +45 -0
  204. package/dist/core/completions/types.d.ts +101 -0
  205. package/dist/core/completions/types.js +2 -0
  206. package/dist/core/config-prompts.d.ts +9 -0
  207. package/dist/core/config-prompts.js +34 -0
  208. package/dist/core/config-schema.d.ts +86 -0
  209. package/dist/core/config-schema.js +213 -0
  210. package/dist/core/config.d.ts +18 -0
  211. package/dist/core/config.js +39 -0
  212. package/dist/core/context-store/binding.d.ts +53 -0
  213. package/dist/core/context-store/binding.js +197 -0
  214. package/dist/core/context-store/errors.d.ts +20 -0
  215. package/dist/core/context-store/errors.js +22 -0
  216. package/dist/core/context-store/foundation.d.ts +55 -0
  217. package/dist/core/context-store/foundation.js +321 -0
  218. package/dist/core/context-store/index.d.ts +6 -0
  219. package/dist/core/context-store/index.js +6 -0
  220. package/dist/core/context-store/operations.d.ts +85 -0
  221. package/dist/core/context-store/operations.js +528 -0
  222. package/dist/core/context-store/registry.d.ts +45 -0
  223. package/dist/core/context-store/registry.js +229 -0
  224. package/dist/core/converters/json-converter.d.ts +6 -0
  225. package/dist/core/converters/json-converter.js +51 -0
  226. package/dist/core/global-config.d.ts +49 -0
  227. package/dist/core/global-config.js +124 -0
  228. package/dist/core/index.d.ts +6 -0
  229. package/dist/core/index.js +7 -0
  230. package/dist/core/init.d.ts +37 -0
  231. package/dist/core/init.js +593 -0
  232. package/dist/core/legacy-cleanup.d.ts +162 -0
  233. package/dist/core/legacy-cleanup.js +514 -0
  234. package/dist/core/list.d.ts +9 -0
  235. package/dist/core/list.js +171 -0
  236. package/dist/core/migration.d.ts +23 -0
  237. package/dist/core/migration.js +108 -0
  238. package/dist/core/parsers/change-parser.d.ts +13 -0
  239. package/dist/core/parsers/change-parser.js +197 -0
  240. package/dist/core/parsers/markdown-parser.d.ts +26 -0
  241. package/dist/core/parsers/markdown-parser.js +227 -0
  242. package/dist/core/parsers/requirement-blocks.d.ts +37 -0
  243. package/dist/core/parsers/requirement-blocks.js +201 -0
  244. package/dist/core/parsers/spec-structure.d.ts +9 -0
  245. package/dist/core/parsers/spec-structure.js +88 -0
  246. package/dist/core/planning-home.d.ts +21 -0
  247. package/dist/core/planning-home.js +108 -0
  248. package/dist/core/profile-sync-drift.d.ts +38 -0
  249. package/dist/core/profile-sync-drift.js +200 -0
  250. package/dist/core/profiles.d.ts +26 -0
  251. package/dist/core/profiles.js +40 -0
  252. package/dist/core/project-config.d.ts +64 -0
  253. package/dist/core/project-config.js +223 -0
  254. package/dist/core/schemas/base.schema.d.ts +13 -0
  255. package/dist/core/schemas/base.schema.js +13 -0
  256. package/dist/core/schemas/change.schema.d.ts +73 -0
  257. package/dist/core/schemas/change.schema.js +31 -0
  258. package/dist/core/schemas/index.d.ts +4 -0
  259. package/dist/core/schemas/index.js +4 -0
  260. package/dist/core/schemas/spec.schema.d.ts +18 -0
  261. package/dist/core/schemas/spec.schema.js +15 -0
  262. package/dist/core/shared/index.d.ts +8 -0
  263. package/dist/core/shared/index.js +8 -0
  264. package/dist/core/shared/skill-generation.d.ts +49 -0
  265. package/dist/core/shared/skill-generation.js +96 -0
  266. package/dist/core/shared/tool-detection.d.ts +71 -0
  267. package/dist/core/shared/tool-detection.js +158 -0
  268. package/dist/core/specs-apply.d.ts +73 -0
  269. package/dist/core/specs-apply.js +392 -0
  270. package/dist/core/styles/palette.d.ts +7 -0
  271. package/dist/core/styles/palette.js +8 -0
  272. package/dist/core/templates/index.d.ts +8 -0
  273. package/dist/core/templates/index.js +9 -0
  274. package/dist/core/templates/skill-templates.d.ts +19 -0
  275. package/dist/core/templates/skill-templates.js +18 -0
  276. package/dist/core/templates/types.d.ts +19 -0
  277. package/dist/core/templates/types.js +5 -0
  278. package/dist/core/templates/workflows/apply-change.d.ts +10 -0
  279. package/dist/core/templates/workflows/apply-change.js +314 -0
  280. package/dist/core/templates/workflows/archive-change.d.ts +10 -0
  281. package/dist/core/templates/workflows/archive-change.js +277 -0
  282. package/dist/core/templates/workflows/bulk-archive-change.d.ts +10 -0
  283. package/dist/core/templates/workflows/bulk-archive-change.js +492 -0
  284. package/dist/core/templates/workflows/continue-change.d.ts +10 -0
  285. package/dist/core/templates/workflows/continue-change.js +234 -0
  286. package/dist/core/templates/workflows/explore.d.ts +10 -0
  287. package/dist/core/templates/workflows/explore.js +459 -0
  288. package/dist/core/templates/workflows/feedback.d.ts +9 -0
  289. package/dist/core/templates/workflows/feedback.js +108 -0
  290. package/dist/core/templates/workflows/ff-change.d.ts +10 -0
  291. package/dist/core/templates/workflows/ff-change.js +200 -0
  292. package/dist/core/templates/workflows/new-change.d.ts +10 -0
  293. package/dist/core/templates/workflows/new-change.js +143 -0
  294. package/dist/core/templates/workflows/onboard.d.ts +10 -0
  295. package/dist/core/templates/workflows/onboard.js +563 -0
  296. package/dist/core/templates/workflows/propose.d.ts +10 -0
  297. package/dist/core/templates/workflows/propose.js +218 -0
  298. package/dist/core/templates/workflows/sync-specs.d.ts +10 -0
  299. package/dist/core/templates/workflows/sync-specs.js +290 -0
  300. package/dist/core/templates/workflows/verify-change.d.ts +10 -0
  301. package/dist/core/templates/workflows/verify-change.js +338 -0
  302. package/dist/core/update.d.ts +82 -0
  303. package/dist/core/update.js +557 -0
  304. package/dist/core/validation/constants.d.ts +34 -0
  305. package/dist/core/validation/constants.js +40 -0
  306. package/dist/core/validation/types.d.ts +18 -0
  307. package/dist/core/validation/types.js +2 -0
  308. package/dist/core/validation/validator.d.ts +43 -0
  309. package/dist/core/validation/validator.js +435 -0
  310. package/dist/core/view.d.ts +8 -0
  311. package/dist/core/view.js +168 -0
  312. package/dist/core/workspace/foundation.d.ts +67 -0
  313. package/dist/core/workspace/foundation.js +295 -0
  314. package/dist/core/workspace/index.d.ts +8 -0
  315. package/dist/core/workspace/index.js +8 -0
  316. package/dist/core/workspace/legacy-state.d.ts +28 -0
  317. package/dist/core/workspace/legacy-state.js +200 -0
  318. package/dist/core/workspace/link-input.d.ts +9 -0
  319. package/dist/core/workspace/link-input.js +32 -0
  320. package/dist/core/workspace/open-surface.d.ts +45 -0
  321. package/dist/core/workspace/open-surface.js +215 -0
  322. package/dist/core/workspace/openers.d.ts +21 -0
  323. package/dist/core/workspace/openers.js +124 -0
  324. package/dist/core/workspace/registry.d.ts +24 -0
  325. package/dist/core/workspace/registry.js +146 -0
  326. package/dist/core/workspace/skills.d.ts +57 -0
  327. package/dist/core/workspace/skills.js +334 -0
  328. package/dist/core/workspace/state-io.d.ts +10 -0
  329. package/dist/core/workspace/state-io.js +121 -0
  330. package/dist/index.d.ts +3 -0
  331. package/dist/index.js +3 -0
  332. package/dist/prompts/searchable-multi-select.d.ts +28 -0
  333. package/dist/prompts/searchable-multi-select.js +159 -0
  334. package/dist/telemetry/config.d.ts +38 -0
  335. package/dist/telemetry/config.js +136 -0
  336. package/dist/telemetry/index.d.ts +31 -0
  337. package/dist/telemetry/index.js +164 -0
  338. package/dist/ui/ascii-patterns.d.ts +16 -0
  339. package/dist/ui/ascii-patterns.js +133 -0
  340. package/dist/ui/welcome-screen.d.ts +10 -0
  341. package/dist/ui/welcome-screen.js +146 -0
  342. package/dist/utils/change-metadata.d.ts +54 -0
  343. package/dist/utils/change-metadata.js +141 -0
  344. package/dist/utils/change-utils.d.ts +71 -0
  345. package/dist/utils/change-utils.js +123 -0
  346. package/dist/utils/command-references.d.ts +18 -0
  347. package/dist/utils/command-references.js +20 -0
  348. package/dist/utils/file-system.d.ts +41 -0
  349. package/dist/utils/file-system.js +301 -0
  350. package/dist/utils/index.d.ts +6 -0
  351. package/dist/utils/index.js +9 -0
  352. package/dist/utils/interactive.d.ts +18 -0
  353. package/dist/utils/interactive.js +21 -0
  354. package/dist/utils/item-discovery.d.ts +4 -0
  355. package/dist/utils/item-discovery.js +72 -0
  356. package/dist/utils/match.d.ts +3 -0
  357. package/dist/utils/match.js +22 -0
  358. package/dist/utils/shell-detection.d.ts +20 -0
  359. package/dist/utils/shell-detection.js +41 -0
  360. package/dist/utils/task-progress.d.ts +8 -0
  361. package/dist/utils/task-progress.js +36 -0
  362. package/package.json +85 -0
  363. package/schemas/spec-driven/schema.yaml +151 -0
  364. package/schemas/spec-driven/templates/design.md +19 -0
  365. package/schemas/spec-driven/templates/proposal.md +23 -0
  366. package/schemas/spec-driven/templates/spec.md +8 -0
  367. package/schemas/spec-driven/templates/tasks.md +9 -0
  368. package/schemas/workspace-planning/schema.yaml +72 -0
  369. package/schemas/workspace-planning/templates/design.md +33 -0
  370. package/schemas/workspace-planning/templates/proposal.md +28 -0
  371. package/schemas/workspace-planning/templates/spec.md +9 -0
  372. package/schemas/workspace-planning/templates/tasks.md +15 -0
  373. package/scripts/postinstall.js +83 -0
@@ -0,0 +1,318 @@
1
+ import { promises as fs } from 'fs';
2
+ import path from 'path';
3
+ import { getTaskProgressForChange, formatTaskStatus } from '../utils/task-progress.js';
4
+ import { Validator } from './validation/validator.js';
5
+ import chalk from 'chalk';
6
+ import { findSpecUpdates, buildUpdatedSpec, writeUpdatedSpec, } from './specs-apply.js';
7
+ /**
8
+ * Recursively copy a directory. Used when fs.rename fails (e.g. EPERM on Windows).
9
+ */
10
+ async function copyDirRecursive(src, dest) {
11
+ await fs.mkdir(dest, { recursive: true });
12
+ const entries = await fs.readdir(src, { withFileTypes: true });
13
+ for (const entry of entries) {
14
+ const srcPath = path.join(src, entry.name);
15
+ const destPath = path.join(dest, entry.name);
16
+ if (entry.isDirectory()) {
17
+ await copyDirRecursive(srcPath, destPath);
18
+ }
19
+ else {
20
+ await fs.copyFile(srcPath, destPath);
21
+ }
22
+ }
23
+ }
24
+ /**
25
+ * Move a directory from src to dest. On Windows, fs.rename() often fails with
26
+ * EPERM when the directory is non-empty or another process has it open (IDE,
27
+ * file watcher, antivirus). Fall back to copy-then-remove when rename fails
28
+ * with EPERM or EXDEV.
29
+ */
30
+ async function moveDirectory(src, dest) {
31
+ try {
32
+ await fs.rename(src, dest);
33
+ }
34
+ catch (err) {
35
+ const code = err?.code;
36
+ if (code === 'EPERM' || code === 'EXDEV') {
37
+ await copyDirRecursive(src, dest);
38
+ await fs.rm(src, { recursive: true, force: true });
39
+ }
40
+ else {
41
+ throw err;
42
+ }
43
+ }
44
+ }
45
+ export class ArchiveCommand {
46
+ async execute(changeName, options = {}) {
47
+ const targetPath = '.';
48
+ const changesDir = path.join(targetPath, 'openspec', 'changes');
49
+ const archiveDir = path.join(changesDir, 'archive');
50
+ const mainSpecsDir = path.join(targetPath, 'openspec', 'specs');
51
+ // Check if changes directory exists
52
+ try {
53
+ await fs.access(changesDir);
54
+ }
55
+ catch {
56
+ throw new Error("No OpenSpec changes directory found. Run 'openspec init' first.");
57
+ }
58
+ // Get change name interactively if not provided
59
+ if (!changeName) {
60
+ const selectedChange = await this.selectChange(changesDir);
61
+ if (!selectedChange) {
62
+ console.log('未选择变更,已中止。');
63
+ return;
64
+ }
65
+ changeName = selectedChange;
66
+ }
67
+ const changeDir = path.join(changesDir, changeName);
68
+ // Verify change exists
69
+ try {
70
+ const stat = await fs.stat(changeDir);
71
+ if (!stat.isDirectory()) {
72
+ throw new Error(`Change '${changeName}' not found.`);
73
+ }
74
+ }
75
+ catch {
76
+ throw new Error(`Change '${changeName}' not found.`);
77
+ }
78
+ const skipValidation = options.validate === false || options.noValidate === true;
79
+ // Validate specs and change before archiving
80
+ if (!skipValidation) {
81
+ const validator = new Validator();
82
+ let hasValidationErrors = false;
83
+ // Validate proposal.md (non-blocking unless strict mode desired in future)
84
+ const changeFile = path.join(changeDir, 'proposal.md');
85
+ try {
86
+ await fs.access(changeFile);
87
+ const changeReport = await validator.validateChange(changeFile);
88
+ // Proposal validation is informative only (do not block archive)
89
+ if (!changeReport.valid) {
90
+ console.log(chalk.yellow(`\nproposal.md 中的提案警告(非阻塞):`));
91
+ for (const issue of changeReport.issues) {
92
+ const symbol = issue.level === 'ERROR' ? '⚠' : (issue.level === 'WARNING' ? '⚠' : 'ℹ');
93
+ console.log(chalk.yellow(` ${symbol} ${issue.message}`));
94
+ }
95
+ }
96
+ }
97
+ catch {
98
+ // Change file doesn't exist, skip validation
99
+ }
100
+ // Validate delta-formatted spec files under the change directory if present
101
+ const changeSpecsDir = path.join(changeDir, 'specs');
102
+ let hasDeltaSpecs = false;
103
+ try {
104
+ const candidates = await fs.readdir(changeSpecsDir, { withFileTypes: true });
105
+ for (const c of candidates) {
106
+ if (c.isDirectory()) {
107
+ try {
108
+ const candidatePath = path.join(changeSpecsDir, c.name, 'spec.md');
109
+ await fs.access(candidatePath);
110
+ const content = await fs.readFile(candidatePath, 'utf-8');
111
+ if (/^##\s+(ADDED|MODIFIED|REMOVED|RENAMED)\s+Requirements/m.test(content)) {
112
+ hasDeltaSpecs = true;
113
+ break;
114
+ }
115
+ }
116
+ catch { }
117
+ }
118
+ }
119
+ }
120
+ catch { }
121
+ if (hasDeltaSpecs) {
122
+ const deltaReport = await validator.validateChangeDeltaSpecs(changeDir);
123
+ if (!deltaReport.valid) {
124
+ hasValidationErrors = true;
125
+ console.log(chalk.red(`\n变更 delta 规格中的验证错误:`));
126
+ for (const issue of deltaReport.issues) {
127
+ if (issue.level === 'ERROR') {
128
+ console.log(chalk.red(` ✗ ${issue.message}`));
129
+ }
130
+ else if (issue.level === 'WARNING') {
131
+ console.log(chalk.yellow(` ⚠ ${issue.message}`));
132
+ }
133
+ }
134
+ }
135
+ }
136
+ if (hasValidationErrors) {
137
+ console.log(chalk.red('\n验证失败。请在归档前修复错误。'));
138
+ console.log(chalk.yellow('要跳过验证(不推荐),请使用 --no-validate 标志。'));
139
+ return;
140
+ }
141
+ }
142
+ else {
143
+ // Log warning when validation is skipped
144
+ const timestamp = new Date().toISOString();
145
+ if (!options.yes) {
146
+ const { confirm } = await import('@inquirer/prompts');
147
+ const proceed = await confirm({
148
+ message: chalk.yellow('⚠️ 警告:跳过验证可能导致归档无效规格。是否继续?(y/N)'),
149
+ default: false
150
+ });
151
+ if (!proceed) {
152
+ console.log('归档已取消。');
153
+ return;
154
+ }
155
+ }
156
+ else {
157
+ console.log(chalk.yellow(`\n⚠️ 警告:跳过验证可能导致归档无效规格。`));
158
+ }
159
+ console.log(chalk.yellow(`[${timestamp}] 已跳过变更验证: ${changeName}`));
160
+ console.log(chalk.yellow(`受影响文件: ${changeDir}`));
161
+ }
162
+ // Show progress and check for incomplete tasks
163
+ const progress = await getTaskProgressForChange(changesDir, changeName);
164
+ const status = formatTaskStatus(progress);
165
+ console.log(`任务状态: ${status}`);
166
+ const incompleteTasks = Math.max(progress.total - progress.completed, 0);
167
+ if (incompleteTasks > 0) {
168
+ if (!options.yes) {
169
+ const { confirm } = await import('@inquirer/prompts');
170
+ const proceed = await confirm({
171
+ message: `警告:发现 ${incompleteTasks} 个未完成任务。是否继续?`,
172
+ default: false
173
+ });
174
+ if (!proceed) {
175
+ console.log('归档已取消。');
176
+ return;
177
+ }
178
+ }
179
+ else {
180
+ console.log(`警告:发现 ${incompleteTasks} 个未完成任务。因 --yes 标志继续。`);
181
+ }
182
+ }
183
+ // Handle spec updates unless skipSpecs flag is set
184
+ if (options.skipSpecs) {
185
+ console.log('跳过规格更新(已提供 --skip-specs 标志)。');
186
+ }
187
+ else {
188
+ // Find specs to update
189
+ const specUpdates = await findSpecUpdates(changeDir, mainSpecsDir);
190
+ if (specUpdates.length > 0) {
191
+ console.log('\n要更新的规格:');
192
+ for (const update of specUpdates) {
193
+ const status = update.exists ? 'update' : 'create';
194
+ const capability = path.basename(path.dirname(update.target));
195
+ console.log(` ${capability}: ${status}`);
196
+ }
197
+ let shouldUpdateSpecs = true;
198
+ if (!options.yes) {
199
+ const { confirm } = await import('@inquirer/prompts');
200
+ shouldUpdateSpecs = await confirm({
201
+ message: '是否继续更新规格?',
202
+ default: true
203
+ });
204
+ if (!shouldUpdateSpecs) {
205
+ console.log('跳过规格更新,继续归档。');
206
+ }
207
+ }
208
+ if (shouldUpdateSpecs) {
209
+ // Prepare all updates first (validation pass, no writes)
210
+ const prepared = [];
211
+ try {
212
+ for (const update of specUpdates) {
213
+ const built = await buildUpdatedSpec(update, changeName);
214
+ prepared.push({ update, rebuilt: built.rebuilt, counts: built.counts });
215
+ }
216
+ }
217
+ catch (err) {
218
+ console.log(String(err.message || err));
219
+ console.log('已中止。未修改任何文件。');
220
+ return;
221
+ }
222
+ // All validations passed; pre-validate rebuilt full spec and then write files and display counts
223
+ let totals = { added: 0, modified: 0, removed: 0, renamed: 0 };
224
+ for (const p of prepared) {
225
+ const specName = path.basename(path.dirname(p.update.target));
226
+ if (!skipValidation) {
227
+ const report = await new Validator().validateSpecContent(specName, p.rebuilt);
228
+ if (!report.valid) {
229
+ console.log(chalk.red(`\n重建规格 ${specName} 的验证错误(不会写入更改):`));
230
+ for (const issue of report.issues) {
231
+ if (issue.level === 'ERROR')
232
+ console.log(chalk.red(` ✗ ${issue.message}`));
233
+ else if (issue.level === 'WARNING')
234
+ console.log(chalk.yellow(` ⚠ ${issue.message}`));
235
+ }
236
+ console.log('已中止。未修改任何文件。');
237
+ return;
238
+ }
239
+ }
240
+ await writeUpdatedSpec(p.update, p.rebuilt, p.counts);
241
+ totals.added += p.counts.added;
242
+ totals.modified += p.counts.modified;
243
+ totals.removed += p.counts.removed;
244
+ totals.renamed += p.counts.renamed;
245
+ }
246
+ console.log(`总计:+ ${totals.added} 新增, ~ ${totals.modified} 修改, - ${totals.removed} 删除, → ${totals.renamed} 重命名`);
247
+ console.log('规格更新成功。');
248
+ }
249
+ }
250
+ }
251
+ // Create archive directory with date prefix
252
+ const archiveName = `${this.getArchiveDate()}-${changeName}`;
253
+ const archivePath = path.join(archiveDir, archiveName);
254
+ // Check if archive already exists
255
+ try {
256
+ await fs.access(archivePath);
257
+ throw new Error(`Archive '${archiveName}' already exists.`);
258
+ }
259
+ catch (error) {
260
+ if (error.code !== 'ENOENT') {
261
+ throw error;
262
+ }
263
+ }
264
+ // Create archive directory if needed
265
+ await fs.mkdir(archiveDir, { recursive: true });
266
+ // Move change to archive (uses copy+remove on EPERM/EXDEV, e.g. Windows)
267
+ await moveDirectory(changeDir, archivePath);
268
+ console.log(`变更 '${changeName}' 已归档为 '${archiveName}'。`);
269
+ }
270
+ async selectChange(changesDir) {
271
+ const { select } = await import('@inquirer/prompts');
272
+ // Get all directories in changes (excluding archive)
273
+ const entries = await fs.readdir(changesDir, { withFileTypes: true });
274
+ const changeDirs = entries
275
+ .filter(entry => entry.isDirectory() && entry.name !== 'archive')
276
+ .map(entry => entry.name)
277
+ .sort();
278
+ if (changeDirs.length === 0) {
279
+ console.log('未找到活跃变更。');
280
+ return null;
281
+ }
282
+ // Build choices with progress inline to avoid duplicate lists
283
+ let choices = changeDirs.map(name => ({ name, value: name }));
284
+ try {
285
+ const progressList = [];
286
+ for (const id of changeDirs) {
287
+ const progress = await getTaskProgressForChange(changesDir, id);
288
+ const status = formatTaskStatus(progress);
289
+ progressList.push({ id, status });
290
+ }
291
+ const nameWidth = Math.max(...progressList.map(p => p.id.length));
292
+ choices = progressList.map(p => ({
293
+ name: `${p.id.padEnd(nameWidth)} ${p.status}`,
294
+ value: p.id
295
+ }));
296
+ }
297
+ catch {
298
+ // If anything fails, fall back to simple names
299
+ choices = changeDirs.map(name => ({ name, value: name }));
300
+ }
301
+ try {
302
+ const answer = await select({
303
+ message: 'Select a change to archive',
304
+ choices
305
+ });
306
+ return answer;
307
+ }
308
+ catch (error) {
309
+ // User cancelled (Ctrl+C)
310
+ return null;
311
+ }
312
+ }
313
+ getArchiveDate() {
314
+ // Returns date in YYYY-MM-DD format
315
+ return new Date().toISOString().split('T')[0];
316
+ }
317
+ }
318
+ //# sourceMappingURL=archive.js.map
@@ -0,0 +1,56 @@
1
+ import type { Artifact, SchemaYaml, CompletedSet, BlockedArtifacts } from './types.js';
2
+ /**
3
+ * Represents an artifact dependency graph.
4
+ * Provides methods for querying build order, ready artifacts, and completion status.
5
+ */
6
+ export declare class ArtifactGraph {
7
+ private artifacts;
8
+ private schema;
9
+ private constructor();
10
+ /**
11
+ * Creates an ArtifactGraph from a YAML file path.
12
+ */
13
+ static fromYaml(filePath: string): ArtifactGraph;
14
+ /**
15
+ * Creates an ArtifactGraph from YAML content string.
16
+ */
17
+ static fromYamlContent(yamlContent: string): ArtifactGraph;
18
+ /**
19
+ * Creates an ArtifactGraph from a pre-validated schema object.
20
+ */
21
+ static fromSchema(schema: SchemaYaml): ArtifactGraph;
22
+ /**
23
+ * Gets a single artifact by ID.
24
+ */
25
+ getArtifact(id: string): Artifact | undefined;
26
+ /**
27
+ * Gets all artifacts in the graph.
28
+ */
29
+ getAllArtifacts(): Artifact[];
30
+ /**
31
+ * Gets the schema name.
32
+ */
33
+ getName(): string;
34
+ /**
35
+ * Gets the schema version.
36
+ */
37
+ getVersion(): number;
38
+ /**
39
+ * Computes the topological build order using Kahn's algorithm.
40
+ * Returns artifact IDs in the order they should be built.
41
+ */
42
+ getBuildOrder(): string[];
43
+ /**
44
+ * Gets artifacts that are ready to be created (all dependencies completed).
45
+ */
46
+ getNextArtifacts(completed: CompletedSet): string[];
47
+ /**
48
+ * Checks if all artifacts in the graph are completed.
49
+ */
50
+ isComplete(completed: CompletedSet): boolean;
51
+ /**
52
+ * Gets blocked artifacts and their unmet dependencies.
53
+ */
54
+ getBlocked(completed: CompletedSet): BlockedArtifacts;
55
+ }
56
+ //# sourceMappingURL=graph.d.ts.map
@@ -0,0 +1,141 @@
1
+ import { loadSchema, parseSchema } from './schema.js';
2
+ /**
3
+ * Represents an artifact dependency graph.
4
+ * Provides methods for querying build order, ready artifacts, and completion status.
5
+ */
6
+ export class ArtifactGraph {
7
+ artifacts;
8
+ schema;
9
+ constructor(schema) {
10
+ this.schema = schema;
11
+ this.artifacts = new Map(schema.artifacts.map(a => [a.id, a]));
12
+ }
13
+ /**
14
+ * Creates an ArtifactGraph from a YAML file path.
15
+ */
16
+ static fromYaml(filePath) {
17
+ const schema = loadSchema(filePath);
18
+ return new ArtifactGraph(schema);
19
+ }
20
+ /**
21
+ * Creates an ArtifactGraph from YAML content string.
22
+ */
23
+ static fromYamlContent(yamlContent) {
24
+ const schema = parseSchema(yamlContent);
25
+ return new ArtifactGraph(schema);
26
+ }
27
+ /**
28
+ * Creates an ArtifactGraph from a pre-validated schema object.
29
+ */
30
+ static fromSchema(schema) {
31
+ return new ArtifactGraph(schema);
32
+ }
33
+ /**
34
+ * Gets a single artifact by ID.
35
+ */
36
+ getArtifact(id) {
37
+ return this.artifacts.get(id);
38
+ }
39
+ /**
40
+ * Gets all artifacts in the graph.
41
+ */
42
+ getAllArtifacts() {
43
+ return Array.from(this.artifacts.values());
44
+ }
45
+ /**
46
+ * Gets the schema name.
47
+ */
48
+ getName() {
49
+ return this.schema.name;
50
+ }
51
+ /**
52
+ * Gets the schema version.
53
+ */
54
+ getVersion() {
55
+ return this.schema.version;
56
+ }
57
+ /**
58
+ * Computes the topological build order using Kahn's algorithm.
59
+ * Returns artifact IDs in the order they should be built.
60
+ */
61
+ getBuildOrder() {
62
+ const inDegree = new Map();
63
+ const dependents = new Map();
64
+ // Initialize all artifacts
65
+ for (const artifact of this.artifacts.values()) {
66
+ inDegree.set(artifact.id, artifact.requires.length);
67
+ dependents.set(artifact.id, []);
68
+ }
69
+ // Build reverse adjacency (who depends on whom)
70
+ for (const artifact of this.artifacts.values()) {
71
+ for (const req of artifact.requires) {
72
+ dependents.get(req).push(artifact.id);
73
+ }
74
+ }
75
+ // Start with roots (in-degree 0), sorted for determinism
76
+ const queue = [...this.artifacts.keys()]
77
+ .filter(id => inDegree.get(id) === 0)
78
+ .sort();
79
+ const result = [];
80
+ while (queue.length > 0) {
81
+ const current = queue.shift();
82
+ result.push(current);
83
+ // Collect newly ready artifacts, then sort before adding
84
+ const newlyReady = [];
85
+ for (const dep of dependents.get(current)) {
86
+ const newDegree = inDegree.get(dep) - 1;
87
+ inDegree.set(dep, newDegree);
88
+ if (newDegree === 0) {
89
+ newlyReady.push(dep);
90
+ }
91
+ }
92
+ queue.push(...newlyReady.sort());
93
+ }
94
+ return result;
95
+ }
96
+ /**
97
+ * Gets artifacts that are ready to be created (all dependencies completed).
98
+ */
99
+ getNextArtifacts(completed) {
100
+ const ready = [];
101
+ for (const artifact of this.artifacts.values()) {
102
+ if (completed.has(artifact.id)) {
103
+ continue; // Already completed
104
+ }
105
+ const allDepsCompleted = artifact.requires.every(req => completed.has(req));
106
+ if (allDepsCompleted) {
107
+ ready.push(artifact.id);
108
+ }
109
+ }
110
+ // Sort for deterministic ordering
111
+ return ready.sort();
112
+ }
113
+ /**
114
+ * Checks if all artifacts in the graph are completed.
115
+ */
116
+ isComplete(completed) {
117
+ for (const artifact of this.artifacts.values()) {
118
+ if (!completed.has(artifact.id)) {
119
+ return false;
120
+ }
121
+ }
122
+ return true;
123
+ }
124
+ /**
125
+ * Gets blocked artifacts and their unmet dependencies.
126
+ */
127
+ getBlocked(completed) {
128
+ const blocked = {};
129
+ for (const artifact of this.artifacts.values()) {
130
+ if (completed.has(artifact.id)) {
131
+ continue; // Already completed
132
+ }
133
+ const unmetDeps = artifact.requires.filter(req => !completed.has(req));
134
+ if (unmetDeps.length > 0) {
135
+ blocked[artifact.id] = unmetDeps.sort();
136
+ }
137
+ }
138
+ return blocked;
139
+ }
140
+ }
141
+ //# sourceMappingURL=graph.js.map
@@ -0,0 +1,9 @@
1
+ export { ArtifactSchema, SchemaYamlSchema, type Artifact, type SchemaYaml, type CompletedSet, type BlockedArtifacts, } from './types.js';
2
+ export { loadSchema, parseSchema, SchemaValidationError } from './schema.js';
3
+ export { ArtifactGraph } from './graph.js';
4
+ export { detectCompleted } from './state.js';
5
+ export { artifactOutputExists, isGlobPattern, resolveArtifactOutputs } from './outputs.js';
6
+ export { resolveSchema, listSchemas, listSchemasWithInfo, getSchemaDir, getPackageSchemasDir, getUserSchemasDir, SchemaLoadError, type SchemaInfo, } from './resolver.js';
7
+ export { loadTemplate, loadChangeContext, generateInstructions, formatChangeStatus, TemplateLoadError, type ChangeContext, type LoadChangeContextOptions, type ArtifactInstructions, type DependencyInfo, type ArtifactStatus, type ChangeStatus, type ArtifactPathSummary, } from './instruction-loader.js';
8
+ export type { PlanningHomeSummary, AffectedAreasSummary, ActionContext, } from '../change-status-policy.js';
9
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1,14 @@
1
+ // Types
2
+ export { ArtifactSchema, SchemaYamlSchema, } from './types.js';
3
+ // Schema loading and validation
4
+ export { loadSchema, parseSchema, SchemaValidationError } from './schema.js';
5
+ // Graph operations
6
+ export { ArtifactGraph } from './graph.js';
7
+ // State detection
8
+ export { detectCompleted } from './state.js';
9
+ export { artifactOutputExists, isGlobPattern, resolveArtifactOutputs } from './outputs.js';
10
+ // Schema resolution
11
+ export { resolveSchema, listSchemas, listSchemasWithInfo, getSchemaDir, getPackageSchemasDir, getUserSchemasDir, SchemaLoadError, } from './resolver.js';
12
+ // Instruction loading
13
+ export { loadTemplate, loadChangeContext, generateInstructions, formatChangeStatus, TemplateLoadError, } from './instruction-loader.js';
14
+ //# sourceMappingURL=index.js.map