synarcx 0.1.0

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 (288) hide show
  1. package/LICENSE +23 -0
  2. package/README.md +90 -0
  3. package/bin/synarcx.js +3 -0
  4. package/dist/cli/index.d.ts +2 -0
  5. package/dist/cli/index.js +474 -0
  6. package/dist/commands/change.d.ts +35 -0
  7. package/dist/commands/change.js +278 -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 +552 -0
  12. package/dist/commands/feedback.d.ts +9 -0
  13. package/dist/commands/feedback.js +170 -0
  14. package/dist/commands/schema.d.ts +6 -0
  15. package/dist/commands/schema.js +870 -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 +226 -0
  20. package/dist/commands/validate.d.ts +24 -0
  21. package/dist/commands/validate.js +295 -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 +327 -0
  26. package/dist/commands/workflow/new-change.d.ts +11 -0
  27. package/dist/commands/workflow/new-change.js +45 -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 +57 -0
  31. package/dist/commands/workflow/shared.js +117 -0
  32. package/dist/commands/workflow/status.d.ts +14 -0
  33. package/dist/commands/workflow/status.js +75 -0
  34. package/dist/commands/workflow/templates.d.ts +16 -0
  35. package/dist/commands/workflow/templates.js +69 -0
  36. package/dist/commands/workspace/open.d.ts +29 -0
  37. package/dist/commands/workspace/open.js +84 -0
  38. package/dist/commands/workspace/operations.d.ts +18 -0
  39. package/dist/commands/workspace/operations.js +461 -0
  40. package/dist/commands/workspace/selection.d.ts +5 -0
  41. package/dist/commands/workspace/selection.js +90 -0
  42. package/dist/commands/workspace/types.d.ts +83 -0
  43. package/dist/commands/workspace/types.js +36 -0
  44. package/dist/commands/workspace.d.ts +3 -0
  45. package/dist/commands/workspace.js +635 -0
  46. package/dist/core/archive.d.ts +11 -0
  47. package/dist/core/archive.js +319 -0
  48. package/dist/core/artifact-graph/graph.d.ts +56 -0
  49. package/dist/core/artifact-graph/graph.js +141 -0
  50. package/dist/core/artifact-graph/index.d.ts +8 -0
  51. package/dist/core/artifact-graph/index.js +14 -0
  52. package/dist/core/artifact-graph/instruction-loader.d.ts +143 -0
  53. package/dist/core/artifact-graph/instruction-loader.js +217 -0
  54. package/dist/core/artifact-graph/outputs.d.ts +14 -0
  55. package/dist/core/artifact-graph/outputs.js +39 -0
  56. package/dist/core/artifact-graph/resolver.d.ts +81 -0
  57. package/dist/core/artifact-graph/resolver.js +258 -0
  58. package/dist/core/artifact-graph/schema.d.ts +13 -0
  59. package/dist/core/artifact-graph/schema.js +108 -0
  60. package/dist/core/artifact-graph/state.d.ts +12 -0
  61. package/dist/core/artifact-graph/state.js +31 -0
  62. package/dist/core/artifact-graph/types.d.ts +45 -0
  63. package/dist/core/artifact-graph/types.js +43 -0
  64. package/dist/core/available-tools.d.ts +17 -0
  65. package/dist/core/available-tools.js +43 -0
  66. package/dist/core/command-generation/adapters/amazon-q.d.ts +13 -0
  67. package/dist/core/command-generation/adapters/amazon-q.js +26 -0
  68. package/dist/core/command-generation/adapters/antigravity.d.ts +13 -0
  69. package/dist/core/command-generation/adapters/antigravity.js +26 -0
  70. package/dist/core/command-generation/adapters/auggie.d.ts +13 -0
  71. package/dist/core/command-generation/adapters/auggie.js +27 -0
  72. package/dist/core/command-generation/adapters/bob.d.ts +14 -0
  73. package/dist/core/command-generation/adapters/bob.js +45 -0
  74. package/dist/core/command-generation/adapters/claude.d.ts +13 -0
  75. package/dist/core/command-generation/adapters/claude.js +50 -0
  76. package/dist/core/command-generation/adapters/cline.d.ts +14 -0
  77. package/dist/core/command-generation/adapters/cline.js +27 -0
  78. package/dist/core/command-generation/adapters/codebuddy.d.ts +13 -0
  79. package/dist/core/command-generation/adapters/codebuddy.js +28 -0
  80. package/dist/core/command-generation/adapters/codex.d.ts +16 -0
  81. package/dist/core/command-generation/adapters/codex.js +39 -0
  82. package/dist/core/command-generation/adapters/continue.d.ts +13 -0
  83. package/dist/core/command-generation/adapters/continue.js +28 -0
  84. package/dist/core/command-generation/adapters/costrict.d.ts +13 -0
  85. package/dist/core/command-generation/adapters/costrict.js +27 -0
  86. package/dist/core/command-generation/adapters/crush.d.ts +13 -0
  87. package/dist/core/command-generation/adapters/crush.js +30 -0
  88. package/dist/core/command-generation/adapters/cursor.d.ts +14 -0
  89. package/dist/core/command-generation/adapters/cursor.js +44 -0
  90. package/dist/core/command-generation/adapters/factory.d.ts +13 -0
  91. package/dist/core/command-generation/adapters/factory.js +27 -0
  92. package/dist/core/command-generation/adapters/gemini.d.ts +13 -0
  93. package/dist/core/command-generation/adapters/gemini.js +26 -0
  94. package/dist/core/command-generation/adapters/github-copilot.d.ts +13 -0
  95. package/dist/core/command-generation/adapters/github-copilot.js +26 -0
  96. package/dist/core/command-generation/adapters/iflow.d.ts +13 -0
  97. package/dist/core/command-generation/adapters/iflow.js +29 -0
  98. package/dist/core/command-generation/adapters/index.d.ts +32 -0
  99. package/dist/core/command-generation/adapters/index.js +32 -0
  100. package/dist/core/command-generation/adapters/junie.d.ts +13 -0
  101. package/dist/core/command-generation/adapters/junie.js +26 -0
  102. package/dist/core/command-generation/adapters/kilocode.d.ts +14 -0
  103. package/dist/core/command-generation/adapters/kilocode.js +23 -0
  104. package/dist/core/command-generation/adapters/kiro.d.ts +13 -0
  105. package/dist/core/command-generation/adapters/kiro.js +26 -0
  106. package/dist/core/command-generation/adapters/lingma.d.ts +13 -0
  107. package/dist/core/command-generation/adapters/lingma.js +30 -0
  108. package/dist/core/command-generation/adapters/opencode.d.ts +13 -0
  109. package/dist/core/command-generation/adapters/opencode.js +27 -0
  110. package/dist/core/command-generation/adapters/pi.d.ts +18 -0
  111. package/dist/core/command-generation/adapters/pi.js +55 -0
  112. package/dist/core/command-generation/adapters/qoder.d.ts +13 -0
  113. package/dist/core/command-generation/adapters/qoder.js +30 -0
  114. package/dist/core/command-generation/adapters/qwen.d.ts +13 -0
  115. package/dist/core/command-generation/adapters/qwen.js +26 -0
  116. package/dist/core/command-generation/adapters/roocode.d.ts +14 -0
  117. package/dist/core/command-generation/adapters/roocode.js +27 -0
  118. package/dist/core/command-generation/adapters/windsurf.d.ts +14 -0
  119. package/dist/core/command-generation/adapters/windsurf.js +51 -0
  120. package/dist/core/command-generation/generator.d.ts +21 -0
  121. package/dist/core/command-generation/generator.js +27 -0
  122. package/dist/core/command-generation/index.d.ts +22 -0
  123. package/dist/core/command-generation/index.js +24 -0
  124. package/dist/core/command-generation/registry.d.ts +36 -0
  125. package/dist/core/command-generation/registry.js +98 -0
  126. package/dist/core/command-generation/types.d.ts +56 -0
  127. package/dist/core/command-generation/types.js +8 -0
  128. package/dist/core/completions/command-registry.d.ts +7 -0
  129. package/dist/core/completions/command-registry.js +596 -0
  130. package/dist/core/completions/completion-provider.d.ts +71 -0
  131. package/dist/core/completions/completion-provider.js +129 -0
  132. package/dist/core/completions/factory.d.ts +64 -0
  133. package/dist/core/completions/factory.js +75 -0
  134. package/dist/core/completions/generators/bash-generator.d.ts +35 -0
  135. package/dist/core/completions/generators/bash-generator.js +230 -0
  136. package/dist/core/completions/generators/fish-generator.d.ts +32 -0
  137. package/dist/core/completions/generators/fish-generator.js +160 -0
  138. package/dist/core/completions/generators/powershell-generator.d.ts +36 -0
  139. package/dist/core/completions/generators/powershell-generator.js +266 -0
  140. package/dist/core/completions/generators/zsh-generator.d.ts +47 -0
  141. package/dist/core/completions/generators/zsh-generator.js +274 -0
  142. package/dist/core/completions/installers/bash-installer.d.ts +87 -0
  143. package/dist/core/completions/installers/bash-installer.js +318 -0
  144. package/dist/core/completions/installers/fish-installer.d.ts +43 -0
  145. package/dist/core/completions/installers/fish-installer.js +143 -0
  146. package/dist/core/completions/installers/powershell-installer.d.ts +102 -0
  147. package/dist/core/completions/installers/powershell-installer.js +387 -0
  148. package/dist/core/completions/installers/zsh-installer.d.ts +117 -0
  149. package/dist/core/completions/installers/zsh-installer.js +421 -0
  150. package/dist/core/completions/templates/bash-templates.d.ts +6 -0
  151. package/dist/core/completions/templates/bash-templates.js +30 -0
  152. package/dist/core/completions/templates/fish-templates.d.ts +7 -0
  153. package/dist/core/completions/templates/fish-templates.js +45 -0
  154. package/dist/core/completions/templates/powershell-templates.d.ts +6 -0
  155. package/dist/core/completions/templates/powershell-templates.js +34 -0
  156. package/dist/core/completions/templates/zsh-templates.d.ts +6 -0
  157. package/dist/core/completions/templates/zsh-templates.js +45 -0
  158. package/dist/core/completions/types.d.ts +101 -0
  159. package/dist/core/completions/types.js +2 -0
  160. package/dist/core/config-prompts.d.ts +9 -0
  161. package/dist/core/config-prompts.js +34 -0
  162. package/dist/core/config-schema.d.ts +86 -0
  163. package/dist/core/config-schema.js +213 -0
  164. package/dist/core/config.d.ts +19 -0
  165. package/dist/core/config.js +38 -0
  166. package/dist/core/converters/json-converter.d.ts +6 -0
  167. package/dist/core/converters/json-converter.js +51 -0
  168. package/dist/core/global-config.d.ts +49 -0
  169. package/dist/core/global-config.js +124 -0
  170. package/dist/core/index.d.ts +3 -0
  171. package/dist/core/index.js +4 -0
  172. package/dist/core/init.d.ts +37 -0
  173. package/dist/core/init.js +585 -0
  174. package/dist/core/legacy-cleanup.d.ts +169 -0
  175. package/dist/core/legacy-cleanup.js +578 -0
  176. package/dist/core/list.d.ts +9 -0
  177. package/dist/core/list.js +172 -0
  178. package/dist/core/migration.d.ts +23 -0
  179. package/dist/core/migration.js +108 -0
  180. package/dist/core/parsers/change-parser.d.ts +13 -0
  181. package/dist/core/parsers/change-parser.js +197 -0
  182. package/dist/core/parsers/markdown-parser.d.ts +26 -0
  183. package/dist/core/parsers/markdown-parser.js +227 -0
  184. package/dist/core/parsers/requirement-blocks.d.ts +37 -0
  185. package/dist/core/parsers/requirement-blocks.js +201 -0
  186. package/dist/core/parsers/spec-structure.d.ts +9 -0
  187. package/dist/core/parsers/spec-structure.js +88 -0
  188. package/dist/core/profile-sync-drift.d.ts +38 -0
  189. package/dist/core/profile-sync-drift.js +197 -0
  190. package/dist/core/profiles.d.ts +26 -0
  191. package/dist/core/profiles.js +37 -0
  192. package/dist/core/project-config.d.ts +64 -0
  193. package/dist/core/project-config.js +224 -0
  194. package/dist/core/schemas/base.schema.d.ts +13 -0
  195. package/dist/core/schemas/base.schema.js +13 -0
  196. package/dist/core/schemas/change.schema.d.ts +73 -0
  197. package/dist/core/schemas/change.schema.js +31 -0
  198. package/dist/core/schemas/index.d.ts +4 -0
  199. package/dist/core/schemas/index.js +4 -0
  200. package/dist/core/schemas/spec.schema.d.ts +18 -0
  201. package/dist/core/schemas/spec.schema.js +15 -0
  202. package/dist/core/shared/index.d.ts +8 -0
  203. package/dist/core/shared/index.js +8 -0
  204. package/dist/core/shared/skill-generation.d.ts +49 -0
  205. package/dist/core/shared/skill-generation.js +90 -0
  206. package/dist/core/shared/tool-detection.d.ts +71 -0
  207. package/dist/core/shared/tool-detection.js +152 -0
  208. package/dist/core/specs-apply.d.ts +73 -0
  209. package/dist/core/specs-apply.js +393 -0
  210. package/dist/core/styles/palette.d.ts +7 -0
  211. package/dist/core/styles/palette.js +8 -0
  212. package/dist/core/templates/index.d.ts +8 -0
  213. package/dist/core/templates/index.js +9 -0
  214. package/dist/core/templates/skill-templates.d.ts +15 -0
  215. package/dist/core/templates/skill-templates.js +14 -0
  216. package/dist/core/templates/types.d.ts +19 -0
  217. package/dist/core/templates/types.js +5 -0
  218. package/dist/core/templates/workflows/analyze.d.ts +4 -0
  219. package/dist/core/templates/workflows/analyze.js +101 -0
  220. package/dist/core/templates/workflows/apply-change.d.ts +10 -0
  221. package/dist/core/templates/workflows/apply-change.js +308 -0
  222. package/dist/core/templates/workflows/archive-change.d.ts +10 -0
  223. package/dist/core/templates/workflows/archive-change.js +271 -0
  224. package/dist/core/templates/workflows/clarify.d.ts +4 -0
  225. package/dist/core/templates/workflows/clarify.js +108 -0
  226. package/dist/core/templates/workflows/debug.d.ts +4 -0
  227. package/dist/core/templates/workflows/debug.js +117 -0
  228. package/dist/core/templates/workflows/explore.d.ts +10 -0
  229. package/dist/core/templates/workflows/explore.js +479 -0
  230. package/dist/core/templates/workflows/propose.d.ts +10 -0
  231. package/dist/core/templates/workflows/propose.js +216 -0
  232. package/dist/core/templates/workflows/sync.d.ts +4 -0
  233. package/dist/core/templates/workflows/sync.js +108 -0
  234. package/dist/core/update.d.ts +82 -0
  235. package/dist/core/update.js +555 -0
  236. package/dist/core/validation/constants.d.ts +34 -0
  237. package/dist/core/validation/constants.js +40 -0
  238. package/dist/core/validation/types.d.ts +18 -0
  239. package/dist/core/validation/types.js +2 -0
  240. package/dist/core/validation/validator.d.ts +33 -0
  241. package/dist/core/validation/validator.js +418 -0
  242. package/dist/core/view.d.ts +8 -0
  243. package/dist/core/view.js +169 -0
  244. package/dist/core/workspace/foundation.d.ts +79 -0
  245. package/dist/core/workspace/foundation.js +367 -0
  246. package/dist/core/workspace/index.d.ts +5 -0
  247. package/dist/core/workspace/index.js +5 -0
  248. package/dist/core/workspace/link-input.d.ts +9 -0
  249. package/dist/core/workspace/link-input.js +32 -0
  250. package/dist/core/workspace/open-surface.d.ts +24 -0
  251. package/dist/core/workspace/open-surface.js +137 -0
  252. package/dist/core/workspace/openers.d.ts +21 -0
  253. package/dist/core/workspace/openers.js +119 -0
  254. package/dist/index.d.ts +3 -0
  255. package/dist/index.js +3 -0
  256. package/dist/prompts/searchable-multi-select.d.ts +28 -0
  257. package/dist/prompts/searchable-multi-select.js +159 -0
  258. package/dist/ui/ascii-patterns.d.ts +25 -0
  259. package/dist/ui/ascii-patterns.js +140 -0
  260. package/dist/ui/welcome-screen.d.ts +10 -0
  261. package/dist/ui/welcome-screen.js +144 -0
  262. package/dist/utils/change-metadata.d.ts +51 -0
  263. package/dist/utils/change-metadata.js +147 -0
  264. package/dist/utils/change-utils.d.ts +62 -0
  265. package/dist/utils/change-utils.js +122 -0
  266. package/dist/utils/command-references.d.ts +18 -0
  267. package/dist/utils/command-references.js +20 -0
  268. package/dist/utils/file-system.d.ts +41 -0
  269. package/dist/utils/file-system.js +301 -0
  270. package/dist/utils/index.d.ts +6 -0
  271. package/dist/utils/index.js +9 -0
  272. package/dist/utils/interactive.d.ts +18 -0
  273. package/dist/utils/interactive.js +21 -0
  274. package/dist/utils/item-discovery.d.ts +4 -0
  275. package/dist/utils/item-discovery.js +73 -0
  276. package/dist/utils/match.d.ts +3 -0
  277. package/dist/utils/match.js +22 -0
  278. package/dist/utils/shell-detection.d.ts +20 -0
  279. package/dist/utils/shell-detection.js +41 -0
  280. package/dist/utils/task-progress.d.ts +8 -0
  281. package/dist/utils/task-progress.js +36 -0
  282. package/package.json +76 -0
  283. package/schemas/synarcx/schema.yaml +153 -0
  284. package/schemas/synarcx/templates/design.md +19 -0
  285. package/schemas/synarcx/templates/proposal.md +23 -0
  286. package/schemas/synarcx/templates/spec.md +8 -0
  287. package/schemas/synarcx/templates/tasks.md +9 -0
  288. package/scripts/postinstall.js +83 -0
package/LICENSE ADDED
@@ -0,0 +1,23 @@
1
+ MIT License
2
+
3
+ Original work Copyright (c) 2024 OpenSpec-Powered Contributors
4
+ Modified work Copyright (c) 2025 Adhi Rahmadian
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining a copy
7
+ of this software and associated documentation files (the "Software"), to deal
8
+ in the Software without restriction, including without limitation the rights
9
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
+ copies of the Software, and to permit persons to whom the Software is
11
+ furnished to do so, subject to the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be included in all
14
+ copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22
+ SOFTWARE.
23
+
package/README.md ADDED
@@ -0,0 +1,90 @@
1
+ # SynArcX — Synapse Architecture Code Extension
2
+
3
+ Spec-driven development workflow for AI coding agents.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ npm install -g synarcx
9
+ ```
10
+
11
+ Or with pnpm:
12
+
13
+ ```bash
14
+ pnpm add -g synarcx
15
+ ```
16
+
17
+ Run `synarcx --help` to verify.
18
+
19
+ ## Quick Start
20
+
21
+ ```bash
22
+ cd your-project
23
+ synarcx init
24
+ ```
25
+
26
+ Then in your AI tool:
27
+
28
+ 1. `/syn:sync` — generate project context (run once, re-run when the project shifts significantly)
29
+ 2. `/syn:explore "your idea"` — think through the problem, then follow the suggestion to `/syn:propose`
30
+ 3. `/syn:propose my-feature` — create proposal, specs, design, and tasks in one step
31
+ 4. `/syn:clarify` — sharpen the artifacts with targeted questions
32
+ 5. `/syn:analyze` — cross-artifact consistency check
33
+ 6. `/syn:apply` — implement the tasks
34
+ 7. `/syn:archive` — archive when done
35
+
36
+ For bugs, use `/syn:debug` instead of `/syn:explore` — same flow, starting from a known error.
37
+
38
+ ## Commands
39
+
40
+ | Command | Description |
41
+ |---------|-------------|
42
+ | `/syn:sync` | Generate or update the project constitution from README, AGENTS.md, package.json |
43
+ | `/syn:explore` | Thinking partner — explore ideas, investigate problems, clarify requirements |
44
+ | `/syn:debug` | Investigation tool — diagnose a known error and hand off findings to `/syn:propose` |
45
+ | `/syn:propose` | Create a new change with proposal, specs, design, and tasks in one step |
46
+ | `/syn:clarify` | Ask up to 5 targeted questions to sharpen artifacts before implementation |
47
+ | `/syn:analyze` | Cross-artifact consistency check across proposal, specs, design, and tasks |
48
+ | `/syn:apply` | Implement tasks from a change's task list |
49
+ | `/syn:archive` | Archive a completed change and sync specs |
50
+
51
+ ## How It Works
52
+
53
+ AI coding assistants are powerful but lose context fast when requirements live only in chat history. SynArcX adds a lightweight spec layer so you and your AI agree on what to build before any code is written.
54
+
55
+ ```
56
+ syn:sync (once)
57
+
58
+ explore ──┐
59
+ ├──► propose ──► clarify ──► analyze ──► apply ──► archive
60
+ debug ──┘
61
+ ```
62
+
63
+ Each step suggests the next — you decide when to advance.
64
+
65
+ Each change gets its own folder under `synspec/changes/` with:
66
+ - `proposal.md` — what and why
67
+ - `specs/` — what the system shall do
68
+ - `design.md` — how to build it
69
+ - `tasks.md` — implementation checklist
70
+
71
+ ## Supported Tools
72
+
73
+ Works with 25+ AI coding assistants: Claude Code, OpenCode, Cursor, Gemini, GitHub Copilot, Cline, Windsurf, Codex, and more. Slash commands are generated per tool on `synarcx init`.
74
+
75
+ ## Development
76
+
77
+ ```bash
78
+ git clone https://github.com/funara/synarcx
79
+ cd synarcx
80
+ pnpm install
81
+ pnpm build
82
+ pnpm link --global
83
+
84
+ # rebuild after edits:
85
+ pnpm build
86
+ ```
87
+
88
+ ## License
89
+
90
+ MIT
package/bin/synarcx.js ADDED
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+
3
+ import '../dist/cli/index.js';
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1,474 @@
1
+ import { Command } from 'commander';
2
+ import { createRequire } from 'module';
3
+ import ora from 'ora';
4
+ import path from 'path';
5
+ import { promises as fs } from 'fs';
6
+ import { AI_TOOLS } from '../core/config.js';
7
+ import { UpdateCommand } from '../core/update.js';
8
+ import { ListCommand } from '../core/list.js';
9
+ import { ArchiveCommand } from '../core/archive.js';
10
+ import { ViewCommand } from '../core/view.js';
11
+ import { registerSpecCommand } from '../commands/spec.js';
12
+ import { ChangeCommand } from '../commands/change.js';
13
+ import { ValidateCommand } from '../commands/validate.js';
14
+ import { ShowCommand } from '../commands/show.js';
15
+ import { CompletionCommand } from '../commands/completion.js';
16
+ import { FeedbackCommand } from '../commands/feedback.js';
17
+ import { registerConfigCommand } from '../commands/config.js';
18
+ import { registerSchemaCommand } from '../commands/schema.js';
19
+ import { registerWorkspaceCommand } from '../commands/workspace.js';
20
+ import { statusCommand, instructionsCommand, applyInstructionsCommand, templatesCommand, schemasCommand, newChangeCommand, DEFAULT_SCHEMA, } from '../commands/workflow/index.js';
21
+ const program = new Command();
22
+ const require = createRequire(import.meta.url);
23
+ const { version } = require('../../package.json');
24
+ /**
25
+ * Get the full command path for nested commands.
26
+ * For example: 'change show' -> 'change:show'
27
+ */
28
+ function getCommandPath(command) {
29
+ const names = [];
30
+ let current = command;
31
+ while (current) {
32
+ const name = current.name();
33
+ // Skip the root 'synarcx' command
34
+ if (name && name !== 'synarcx') {
35
+ names.unshift(name);
36
+ }
37
+ current = current.parent;
38
+ }
39
+ return names.join(':') || 'synarcx';
40
+ }
41
+ program
42
+ .name('synarcx')
43
+ .description('Spec-driven development workflow for AI coding agents')
44
+ .version(version);
45
+ // Global options
46
+ program.option('--no-color', 'Disable color output');
47
+ // Apply global flags before any command runs
48
+ // Note: preAction receives (thisCommand, actionCommand) where:
49
+ // - thisCommand: the command where hook was added (root program)
50
+ // - actionCommand: the command actually being executed (subcommand)
51
+ program.hook('preAction', async (thisCommand, _actionCommand) => {
52
+ const opts = thisCommand.opts();
53
+ if (opts.color === false) {
54
+ process.env.NO_COLOR = '1';
55
+ }
56
+ });
57
+ const availableToolIds = AI_TOOLS.filter((tool) => tool.skillsDir).map((tool) => tool.value);
58
+ const toolsOptionDescription = `Configure AI tools non-interactively. Use "all", "none", or a comma-separated list of: ${availableToolIds.join(', ')}`;
59
+ program
60
+ .command('init [path]')
61
+ .description('Initialize SynArcX in your project')
62
+ .option('--tools <tools>', toolsOptionDescription)
63
+ .option('--force', 'Auto-cleanup legacy files without prompting')
64
+ .option('--profile <profile>', 'Override global config profile (core or custom)')
65
+ .action(async (targetPath = '.', options) => {
66
+ try {
67
+ // Validate that the path is a valid directory
68
+ const resolvedPath = path.resolve(targetPath);
69
+ try {
70
+ const stats = await fs.stat(resolvedPath);
71
+ if (!stats.isDirectory()) {
72
+ throw new Error(`Path "${targetPath}" is not a directory`);
73
+ }
74
+ }
75
+ catch (error) {
76
+ if (error.code === 'ENOENT') {
77
+ // Directory doesn't exist, but we can create it
78
+ console.log(`Directory "${targetPath}" doesn't exist, it will be created.`);
79
+ }
80
+ else if (error.message && error.message.includes('not a directory')) {
81
+ throw error;
82
+ }
83
+ else {
84
+ throw new Error(`Cannot access path "${targetPath}": ${error.message}`);
85
+ }
86
+ }
87
+ const { InitCommand } = await import('../core/init.js');
88
+ const initCommand = new InitCommand({
89
+ tools: options?.tools,
90
+ force: options?.force,
91
+ profile: options?.profile,
92
+ });
93
+ await initCommand.execute(targetPath);
94
+ }
95
+ catch (error) {
96
+ console.log(); // Empty line for spacing
97
+ ora().fail(`Error: ${error.message}`);
98
+ process.exit(1);
99
+ }
100
+ });
101
+ // Hidden alias: 'experimental' -> 'init' for backwards compatibility
102
+ program
103
+ .command('experimental', { hidden: true })
104
+ .description('Alias for init (deprecated)')
105
+ .option('--tool <tool-id>', 'Target AI tool (maps to --tools)')
106
+ .option('--no-interactive', 'Disable interactive prompts')
107
+ .action(async (options) => {
108
+ try {
109
+ console.log('Note: "synarcx experimental" is deprecated. Use "synarcx init" instead.');
110
+ const { InitCommand } = await import('../core/init.js');
111
+ const initCommand = new InitCommand({
112
+ tools: options?.tool,
113
+ interactive: options?.noInteractive === true ? false : undefined,
114
+ });
115
+ await initCommand.execute('.');
116
+ }
117
+ catch (error) {
118
+ console.log();
119
+ ora().fail(`Error: ${error.message}`);
120
+ process.exit(1);
121
+ }
122
+ });
123
+ program
124
+ .command('update [path]')
125
+ .description('Update SynArcX instruction files')
126
+ .option('--force', 'Force update even when tools are up to date')
127
+ .action(async (targetPath = '.', options) => {
128
+ try {
129
+ const resolvedPath = path.resolve(targetPath);
130
+ const updateCommand = new UpdateCommand({ force: options?.force });
131
+ await updateCommand.execute(resolvedPath);
132
+ }
133
+ catch (error) {
134
+ console.log(); // Empty line for spacing
135
+ ora().fail(`Error: ${error.message}`);
136
+ process.exit(1);
137
+ }
138
+ });
139
+ program
140
+ .command('list')
141
+ .description('List items (changes by default). Use --specs to list specs.')
142
+ .option('--specs', 'List specs instead of changes')
143
+ .option('--changes', 'List changes explicitly (default)')
144
+ .option('--sort <order>', 'Sort order: "recent" (default) or "name"', 'recent')
145
+ .option('--json', 'Output as JSON (for programmatic use)')
146
+ .action(async (options) => {
147
+ try {
148
+ const listCommand = new ListCommand();
149
+ const mode = options?.specs ? 'specs' : 'changes';
150
+ const sort = options?.sort === 'name' ? 'name' : 'recent';
151
+ await listCommand.execute('.', mode, { sort, json: options?.json });
152
+ }
153
+ catch (error) {
154
+ console.log(); // Empty line for spacing
155
+ ora().fail(`Error: ${error.message}`);
156
+ process.exit(1);
157
+ }
158
+ });
159
+ program
160
+ .command('view')
161
+ .description('Display an interactive dashboard of specs and changes')
162
+ .action(async () => {
163
+ try {
164
+ const viewCommand = new ViewCommand();
165
+ await viewCommand.execute('.');
166
+ }
167
+ catch (error) {
168
+ console.log(); // Empty line for spacing
169
+ ora().fail(`Error: ${error.message}`);
170
+ process.exit(1);
171
+ }
172
+ });
173
+ // Change command with subcommands
174
+ const changeCmd = program
175
+ .command('change')
176
+ .description('Manage SynArcX change proposals');
177
+ // Deprecation notice for noun-based commands
178
+ changeCmd.hook('preAction', () => {
179
+ console.error('Warning: The "synarcx change ..." commands are deprecated. Prefer verb-first commands (e.g., "synarcx list", "synarcx validate --changes").');
180
+ });
181
+ changeCmd
182
+ .command('show [change-name]')
183
+ .description('Show a change proposal in JSON or markdown format')
184
+ .option('--json', 'Output as JSON')
185
+ .option('--deltas-only', 'Show only deltas (JSON only)')
186
+ .option('--requirements-only', 'Alias for --deltas-only (deprecated)')
187
+ .option('--no-interactive', 'Disable interactive prompts')
188
+ .action(async (changeName, options) => {
189
+ try {
190
+ const changeCommand = new ChangeCommand();
191
+ await changeCommand.show(changeName, options);
192
+ }
193
+ catch (error) {
194
+ console.error(`Error: ${error.message}`);
195
+ process.exitCode = 1;
196
+ }
197
+ });
198
+ changeCmd
199
+ .command('list')
200
+ .description('List all active changes (DEPRECATED: use "synarcx list" instead)')
201
+ .option('--json', 'Output as JSON')
202
+ .option('--long', 'Show id and title with counts')
203
+ .action(async (options) => {
204
+ try {
205
+ console.error('Warning: "synarcx change list" is deprecated. Use "synarcx list".');
206
+ const changeCommand = new ChangeCommand();
207
+ await changeCommand.list(options);
208
+ }
209
+ catch (error) {
210
+ console.error(`Error: ${error.message}`);
211
+ process.exitCode = 1;
212
+ }
213
+ });
214
+ changeCmd
215
+ .command('validate [change-name]')
216
+ .description('Validate a change proposal')
217
+ .option('--strict', 'Enable strict validation mode')
218
+ .option('--json', 'Output validation report as JSON')
219
+ .option('--no-interactive', 'Disable interactive prompts')
220
+ .action(async (changeName, options) => {
221
+ try {
222
+ const changeCommand = new ChangeCommand();
223
+ await changeCommand.validate(changeName, options);
224
+ if (typeof process.exitCode === 'number' && process.exitCode !== 0) {
225
+ process.exit(process.exitCode);
226
+ }
227
+ }
228
+ catch (error) {
229
+ console.error(`Error: ${error.message}`);
230
+ process.exitCode = 1;
231
+ }
232
+ });
233
+ program
234
+ .command('archive [change-name]')
235
+ .description('Archive a completed change and update main specs')
236
+ .option('-y, --yes', 'Skip confirmation prompts')
237
+ .option('--skip-specs', 'Skip spec update operations (useful for infrastructure, tooling, or doc-only changes)')
238
+ .option('--no-validate', 'Skip validation (not recommended, requires confirmation)')
239
+ .action(async (changeName, options) => {
240
+ try {
241
+ const archiveCommand = new ArchiveCommand();
242
+ await archiveCommand.execute(changeName, options);
243
+ }
244
+ catch (error) {
245
+ console.log(); // Empty line for spacing
246
+ ora().fail(`Error: ${error.message}`);
247
+ process.exit(1);
248
+ }
249
+ });
250
+ registerSpecCommand(program);
251
+ registerConfigCommand(program);
252
+ registerSchemaCommand(program);
253
+ registerWorkspaceCommand(program);
254
+ // Top-level validate command
255
+ program
256
+ .command('validate [item-name]')
257
+ .description('Validate changes and specs')
258
+ .option('--all', 'Validate all changes and specs')
259
+ .option('--changes', 'Validate all changes')
260
+ .option('--specs', 'Validate all specs')
261
+ .option('--type <type>', 'Specify item type when ambiguous: change|spec')
262
+ .option('--strict', 'Enable strict validation mode')
263
+ .option('--json', 'Output validation results as JSON')
264
+ .option('--concurrency <n>', 'Max concurrent validations (defaults to env SYNARCX_CONCURRENCY or 6)')
265
+ .option('--no-interactive', 'Disable interactive prompts')
266
+ .action(async (itemName, options) => {
267
+ try {
268
+ const validateCommand = new ValidateCommand();
269
+ await validateCommand.execute(itemName, options);
270
+ }
271
+ catch (error) {
272
+ console.log();
273
+ ora().fail(`Error: ${error.message}`);
274
+ process.exit(1);
275
+ }
276
+ });
277
+ // Top-level show command
278
+ program
279
+ .command('show [item-name]')
280
+ .description('Show a change or spec')
281
+ .option('--json', 'Output as JSON')
282
+ .option('--type <type>', 'Specify item type when ambiguous: change|spec')
283
+ .option('--no-interactive', 'Disable interactive prompts')
284
+ // change-only flags
285
+ .option('--deltas-only', 'Show only deltas (JSON only, change)')
286
+ .option('--requirements-only', 'Alias for --deltas-only (deprecated, change)')
287
+ // spec-only flags
288
+ .option('--requirements', 'JSON only: Show only requirements (exclude scenarios)')
289
+ .option('--no-scenarios', 'JSON only: Exclude scenario content')
290
+ .option('-r, --requirement <id>', 'JSON only: Show specific requirement by ID (1-based)')
291
+ // allow unknown options to pass-through to underlying command implementation
292
+ .allowUnknownOption(true)
293
+ .action(async (itemName, options) => {
294
+ try {
295
+ const showCommand = new ShowCommand();
296
+ await showCommand.execute(itemName, options ?? {});
297
+ }
298
+ catch (error) {
299
+ console.log();
300
+ ora().fail(`Error: ${error.message}`);
301
+ process.exit(1);
302
+ }
303
+ });
304
+ // Feedback command
305
+ program
306
+ .command('feedback <message>')
307
+ .description('Submit feedback')
308
+ .option('--body <text>', 'Detailed description for the feedback')
309
+ .action(async (message, options) => {
310
+ try {
311
+ const feedbackCommand = new FeedbackCommand();
312
+ await feedbackCommand.execute(message, options);
313
+ }
314
+ catch (error) {
315
+ console.log();
316
+ ora().fail(`Error: ${error.message}`);
317
+ process.exit(1);
318
+ }
319
+ });
320
+ // Completion command with subcommands
321
+ const completionCmd = program
322
+ .command('completion')
323
+ .description('Manage shell completions for SynArcX CLI');
324
+ completionCmd
325
+ .command('generate [shell]')
326
+ .description('Generate completion script for a shell (outputs to stdout)')
327
+ .action(async (shell) => {
328
+ try {
329
+ const completionCommand = new CompletionCommand();
330
+ await completionCommand.generate({ shell });
331
+ }
332
+ catch (error) {
333
+ console.log();
334
+ ora().fail(`Error: ${error.message}`);
335
+ process.exit(1);
336
+ }
337
+ });
338
+ completionCmd
339
+ .command('install [shell]')
340
+ .description('Install completion script for a shell')
341
+ .option('--verbose', 'Show detailed installation output')
342
+ .action(async (shell, options) => {
343
+ try {
344
+ const completionCommand = new CompletionCommand();
345
+ await completionCommand.install({ shell, verbose: options?.verbose });
346
+ }
347
+ catch (error) {
348
+ console.log();
349
+ ora().fail(`Error: ${error.message}`);
350
+ process.exit(1);
351
+ }
352
+ });
353
+ completionCmd
354
+ .command('uninstall [shell]')
355
+ .description('Uninstall completion script for a shell')
356
+ .option('-y, --yes', 'Skip confirmation prompts')
357
+ .action(async (shell, options) => {
358
+ try {
359
+ const completionCommand = new CompletionCommand();
360
+ await completionCommand.uninstall({ shell, yes: options?.yes });
361
+ }
362
+ catch (error) {
363
+ console.log();
364
+ ora().fail(`Error: ${error.message}`);
365
+ process.exit(1);
366
+ }
367
+ });
368
+ // Hidden command for machine-readable completion data
369
+ program
370
+ .command('__complete <type>', { hidden: true })
371
+ .description('Output completion data in machine-readable format (internal use)')
372
+ .action(async (type) => {
373
+ try {
374
+ const completionCommand = new CompletionCommand();
375
+ await completionCommand.complete({ type });
376
+ }
377
+ catch (error) {
378
+ // Silently fail for graceful shell completion experience
379
+ process.exitCode = 1;
380
+ }
381
+ });
382
+ // ═══════════════════════════════════════════════════════════
383
+ // Workflow Commands (formerly experimental)
384
+ // ═══════════════════════════════════════════════════════════
385
+ // Status command
386
+ program
387
+ .command('status')
388
+ .description('Display artifact completion status for a change')
389
+ .option('--change <id>', 'Change name to show status for')
390
+ .option('--schema <name>', 'Schema override (auto-detected from config.yaml)')
391
+ .option('--json', 'Output as JSON')
392
+ .action(async (options) => {
393
+ try {
394
+ await statusCommand(options);
395
+ }
396
+ catch (error) {
397
+ console.log();
398
+ ora().fail(`Error: ${error.message}`);
399
+ process.exit(1);
400
+ }
401
+ });
402
+ // Instructions command
403
+ program
404
+ .command('instructions [artifact]')
405
+ .description('Output enriched instructions for creating an artifact or applying tasks')
406
+ .option('--change <id>', 'Change name')
407
+ .option('--schema <name>', 'Schema override (auto-detected from config.yaml)')
408
+ .option('--json', 'Output as JSON')
409
+ .action(async (artifactId, options) => {
410
+ try {
411
+ // Special case: "apply" is not an artifact, but a command to get apply instructions
412
+ if (artifactId === 'apply') {
413
+ await applyInstructionsCommand(options);
414
+ }
415
+ else {
416
+ await instructionsCommand(artifactId, options);
417
+ }
418
+ }
419
+ catch (error) {
420
+ console.log();
421
+ ora().fail(`Error: ${error.message}`);
422
+ process.exit(1);
423
+ }
424
+ });
425
+ // Templates command
426
+ program
427
+ .command('templates')
428
+ .description('Show resolved template paths for all artifacts in a schema')
429
+ .option('--schema <name>', `Schema to use (default: ${DEFAULT_SCHEMA})`)
430
+ .option('--json', 'Output as JSON mapping artifact IDs to template paths')
431
+ .action(async (options) => {
432
+ try {
433
+ await templatesCommand(options);
434
+ }
435
+ catch (error) {
436
+ console.log();
437
+ ora().fail(`Error: ${error.message}`);
438
+ process.exit(1);
439
+ }
440
+ });
441
+ // Schemas command
442
+ program
443
+ .command('schemas')
444
+ .description('List available workflow schemas with descriptions')
445
+ .option('--json', 'Output as JSON (for agent use)')
446
+ .action(async (options) => {
447
+ try {
448
+ await schemasCommand(options);
449
+ }
450
+ catch (error) {
451
+ console.log();
452
+ ora().fail(`Error: ${error.message}`);
453
+ process.exit(1);
454
+ }
455
+ });
456
+ // New command group with change subcommand
457
+ const newCmd = program.command('new').description('Create new items');
458
+ newCmd
459
+ .command('change <name>')
460
+ .description('Create a new change directory')
461
+ .option('--description <text>', 'Description to add to README.md')
462
+ .option('--schema <name>', `Workflow schema to use (default: ${DEFAULT_SCHEMA})`)
463
+ .action(async (name, options) => {
464
+ try {
465
+ await newChangeCommand(name, options);
466
+ }
467
+ catch (error) {
468
+ console.log();
469
+ ora().fail(`Error: ${error.message}`);
470
+ process.exit(1);
471
+ }
472
+ });
473
+ program.parse();
474
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1,35 @@
1
+ export declare class ChangeCommand {
2
+ private converter;
3
+ constructor();
4
+ /**
5
+ * Show a change proposal.
6
+ * - Text mode: raw markdown passthrough (no filters)
7
+ * - JSON mode: minimal object with deltas; --deltas-only returns same object with filtered deltas
8
+ * Note: --requirements-only is deprecated alias for --deltas-only
9
+ */
10
+ show(changeName?: string, options?: {
11
+ json?: boolean;
12
+ requirementsOnly?: boolean;
13
+ deltasOnly?: boolean;
14
+ noInteractive?: boolean;
15
+ }): Promise<void>;
16
+ /**
17
+ * List active changes.
18
+ * - Text default: IDs only; --long prints minimal details (title, counts)
19
+ * - JSON: array of { id, title, deltaCount, taskStatus }, sorted by id
20
+ */
21
+ list(options?: {
22
+ json?: boolean;
23
+ long?: boolean;
24
+ }): Promise<void>;
25
+ validate(changeName?: string, options?: {
26
+ strict?: boolean;
27
+ json?: boolean;
28
+ noInteractive?: boolean;
29
+ }): Promise<void>;
30
+ private getActiveChanges;
31
+ private extractTitle;
32
+ private countTasks;
33
+ private printNextSteps;
34
+ }
35
+ //# sourceMappingURL=change.d.ts.map