@specsafe/cli 0.7.1 → 2.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (242) hide show
  1. package/README.md +100 -279
  2. package/canonical/personas/bolt-zane.md +29 -0
  3. package/canonical/personas/forge-reva.md +29 -0
  4. package/canonical/personas/herald-cass.md +30 -0
  5. package/canonical/personas/mason-kai.md +30 -0
  6. package/canonical/personas/scout-elena.md +27 -0
  7. package/canonical/personas/warden-lyra.md +30 -0
  8. package/canonical/rules/.cursorrules.mdc +53 -0
  9. package/canonical/rules/.rules +48 -0
  10. package/canonical/rules/AGENTS.md +48 -0
  11. package/canonical/rules/CLAUDE.md +48 -0
  12. package/canonical/rules/CONVENTIONS.md +41 -0
  13. package/canonical/rules/GEMINI.md +50 -0
  14. package/canonical/rules/continue-config.yaml +5 -0
  15. package/canonical/skills/specsafe-archive/SKILL.md +63 -0
  16. package/canonical/skills/specsafe-code/SKILL.md +7 -0
  17. package/canonical/skills/specsafe-code/workflow.md +212 -0
  18. package/canonical/skills/specsafe-complete/SKILL.md +7 -0
  19. package/canonical/skills/specsafe-complete/workflow.md +130 -0
  20. package/canonical/skills/specsafe-doctor/SKILL.md +103 -0
  21. package/canonical/skills/specsafe-explore/SKILL.md +7 -0
  22. package/canonical/skills/specsafe-explore/workflow.md +100 -0
  23. package/canonical/skills/specsafe-init/SKILL.md +119 -0
  24. package/canonical/skills/specsafe-new/SKILL.md +7 -0
  25. package/canonical/skills/specsafe-new/workflow.md +156 -0
  26. package/canonical/skills/specsafe-qa/SKILL.md +7 -0
  27. package/canonical/skills/specsafe-qa/workflow.md +223 -0
  28. package/canonical/skills/specsafe-spec/SKILL.md +7 -0
  29. package/canonical/skills/specsafe-spec/workflow.md +158 -0
  30. package/canonical/skills/specsafe-status/SKILL.md +77 -0
  31. package/canonical/skills/specsafe-test/SKILL.md +7 -0
  32. package/canonical/skills/specsafe-test/workflow.md +210 -0
  33. package/canonical/skills/specsafe-verify/SKILL.md +7 -0
  34. package/canonical/skills/specsafe-verify/workflow.md +143 -0
  35. package/canonical/templates/project-state-template.md +33 -0
  36. package/canonical/templates/qa-report-template.md +55 -0
  37. package/canonical/templates/spec-template.md +52 -0
  38. package/canonical/templates/specsafe-config-template.json +10 -0
  39. package/generators/dist/adapters/aider.d.ts +2 -0
  40. package/generators/dist/adapters/aider.js +23 -0
  41. package/generators/dist/adapters/aider.js.map +1 -0
  42. package/generators/dist/adapters/antigravity.d.ts +2 -0
  43. package/generators/dist/adapters/antigravity.js +33 -0
  44. package/generators/dist/adapters/antigravity.js.map +1 -0
  45. package/generators/dist/adapters/claude-code.d.ts +2 -0
  46. package/generators/dist/adapters/claude-code.js +31 -0
  47. package/generators/dist/adapters/claude-code.js.map +1 -0
  48. package/generators/dist/adapters/continue.d.ts +2 -0
  49. package/generators/dist/adapters/continue.js +33 -0
  50. package/generators/dist/adapters/continue.js.map +1 -0
  51. package/generators/dist/adapters/cursor.d.ts +2 -0
  52. package/generators/dist/adapters/cursor.js +32 -0
  53. package/generators/dist/adapters/cursor.js.map +1 -0
  54. package/generators/dist/adapters/gemini.d.ts +2 -0
  55. package/generators/dist/adapters/gemini.js +36 -0
  56. package/generators/dist/adapters/gemini.js.map +1 -0
  57. package/generators/dist/adapters/index.d.ts +13 -0
  58. package/generators/dist/adapters/index.js +29 -0
  59. package/generators/dist/adapters/index.js.map +1 -0
  60. package/generators/dist/adapters/opencode.d.ts +2 -0
  61. package/generators/dist/adapters/opencode.js +32 -0
  62. package/generators/dist/adapters/opencode.js.map +1 -0
  63. package/generators/dist/adapters/types.d.ts +49 -0
  64. package/generators/dist/adapters/types.js +14 -0
  65. package/generators/dist/adapters/types.js.map +1 -0
  66. package/generators/dist/adapters/utils.d.ts +12 -0
  67. package/generators/dist/adapters/utils.js +68 -0
  68. package/generators/dist/adapters/utils.js.map +1 -0
  69. package/generators/dist/adapters/zed.d.ts +2 -0
  70. package/generators/dist/adapters/zed.js +31 -0
  71. package/generators/dist/adapters/zed.js.map +1 -0
  72. package/generators/dist/doctor.d.ts +10 -0
  73. package/generators/dist/doctor.js +123 -0
  74. package/generators/dist/doctor.js.map +1 -0
  75. package/generators/dist/index.d.ts +2 -0
  76. package/generators/dist/index.js +45 -0
  77. package/generators/dist/index.js.map +1 -0
  78. package/generators/dist/init.d.ts +9 -0
  79. package/generators/dist/init.js +167 -0
  80. package/generators/dist/init.js.map +1 -0
  81. package/generators/dist/install.d.ts +5 -0
  82. package/generators/dist/install.js +66 -0
  83. package/generators/dist/install.js.map +1 -0
  84. package/generators/dist/registry.d.ts +3 -0
  85. package/generators/dist/registry.js +8 -0
  86. package/generators/dist/registry.js.map +1 -0
  87. package/generators/dist/update.d.ts +5 -0
  88. package/generators/dist/update.js +40 -0
  89. package/generators/dist/update.js.map +1 -0
  90. package/package.json +32 -27
  91. package/dist/commands/apply.d.ts +0 -3
  92. package/dist/commands/apply.d.ts.map +0 -1
  93. package/dist/commands/apply.js +0 -182
  94. package/dist/commands/apply.js.map +0 -1
  95. package/dist/commands/archive.d.ts +0 -3
  96. package/dist/commands/archive.d.ts.map +0 -1
  97. package/dist/commands/archive.js +0 -99
  98. package/dist/commands/archive.js.map +0 -1
  99. package/dist/commands/capsule.d.ts +0 -8
  100. package/dist/commands/capsule.d.ts.map +0 -1
  101. package/dist/commands/capsule.js +0 -466
  102. package/dist/commands/capsule.js.map +0 -1
  103. package/dist/commands/complete.d.ts +0 -3
  104. package/dist/commands/complete.d.ts.map +0 -1
  105. package/dist/commands/complete.js +0 -140
  106. package/dist/commands/complete.js.map +0 -1
  107. package/dist/commands/constitution.d.ts +0 -3
  108. package/dist/commands/constitution.d.ts.map +0 -1
  109. package/dist/commands/constitution.js +0 -192
  110. package/dist/commands/constitution.js.map +0 -1
  111. package/dist/commands/create.d.ts +0 -10
  112. package/dist/commands/create.d.ts.map +0 -1
  113. package/dist/commands/create.js +0 -120
  114. package/dist/commands/create.js.map +0 -1
  115. package/dist/commands/delta.d.ts +0 -3
  116. package/dist/commands/delta.d.ts.map +0 -1
  117. package/dist/commands/delta.js +0 -82
  118. package/dist/commands/delta.js.map +0 -1
  119. package/dist/commands/diff.d.ts +0 -3
  120. package/dist/commands/diff.d.ts.map +0 -1
  121. package/dist/commands/diff.js +0 -102
  122. package/dist/commands/diff.js.map +0 -1
  123. package/dist/commands/doctor.d.ts +0 -3
  124. package/dist/commands/doctor.d.ts.map +0 -1
  125. package/dist/commands/doctor.js +0 -204
  126. package/dist/commands/doctor.js.map +0 -1
  127. package/dist/commands/done.d.ts +0 -3
  128. package/dist/commands/done.d.ts.map +0 -1
  129. package/dist/commands/done.js +0 -237
  130. package/dist/commands/done.js.map +0 -1
  131. package/dist/commands/explore.d.ts +0 -3
  132. package/dist/commands/explore.d.ts.map +0 -1
  133. package/dist/commands/explore.js +0 -236
  134. package/dist/commands/explore.js.map +0 -1
  135. package/dist/commands/export.d.ts +0 -7
  136. package/dist/commands/export.d.ts.map +0 -1
  137. package/dist/commands/export.js +0 -179
  138. package/dist/commands/export.js.map +0 -1
  139. package/dist/commands/extend.d.ts +0 -6
  140. package/dist/commands/extend.d.ts.map +0 -1
  141. package/dist/commands/extend.js +0 -167
  142. package/dist/commands/extend.js.map +0 -1
  143. package/dist/commands/init-old.d.ts +0 -3
  144. package/dist/commands/init-old.d.ts.map +0 -1
  145. package/dist/commands/init-old.js +0 -146
  146. package/dist/commands/init-old.js.map +0 -1
  147. package/dist/commands/init.d.ts +0 -3
  148. package/dist/commands/init.d.ts.map +0 -1
  149. package/dist/commands/init.js +0 -298
  150. package/dist/commands/init.js.map +0 -1
  151. package/dist/commands/list.d.ts +0 -3
  152. package/dist/commands/list.d.ts.map +0 -1
  153. package/dist/commands/list.js +0 -122
  154. package/dist/commands/list.js.map +0 -1
  155. package/dist/commands/memory.d.ts +0 -3
  156. package/dist/commands/memory.d.ts.map +0 -1
  157. package/dist/commands/memory.js +0 -166
  158. package/dist/commands/memory.js.map +0 -1
  159. package/dist/commands/new.d.ts +0 -3
  160. package/dist/commands/new.d.ts.map +0 -1
  161. package/dist/commands/new.js +0 -508
  162. package/dist/commands/new.js.map +0 -1
  163. package/dist/commands/qa.d.ts +0 -3
  164. package/dist/commands/qa.d.ts.map +0 -1
  165. package/dist/commands/qa.js +0 -179
  166. package/dist/commands/qa.js.map +0 -1
  167. package/dist/commands/rules.d.ts +0 -6
  168. package/dist/commands/rules.d.ts.map +0 -1
  169. package/dist/commands/rules.js +0 -232
  170. package/dist/commands/rules.js.map +0 -1
  171. package/dist/commands/shard.d.ts +0 -6
  172. package/dist/commands/shard.d.ts.map +0 -1
  173. package/dist/commands/shard.js +0 -199
  174. package/dist/commands/shard.js.map +0 -1
  175. package/dist/commands/spec.d.ts +0 -3
  176. package/dist/commands/spec.d.ts.map +0 -1
  177. package/dist/commands/spec.js +0 -302
  178. package/dist/commands/spec.js.map +0 -1
  179. package/dist/commands/status.d.ts +0 -3
  180. package/dist/commands/status.d.ts.map +0 -1
  181. package/dist/commands/status.js +0 -47
  182. package/dist/commands/status.js.map +0 -1
  183. package/dist/commands/test-apply.d.ts +0 -3
  184. package/dist/commands/test-apply.d.ts.map +0 -1
  185. package/dist/commands/test-apply.js +0 -228
  186. package/dist/commands/test-apply.js.map +0 -1
  187. package/dist/commands/test-create.d.ts +0 -3
  188. package/dist/commands/test-create.d.ts.map +0 -1
  189. package/dist/commands/test-create.js +0 -183
  190. package/dist/commands/test-create.js.map +0 -1
  191. package/dist/commands/test-guide.d.ts +0 -3
  192. package/dist/commands/test-guide.d.ts.map +0 -1
  193. package/dist/commands/test-guide.js +0 -190
  194. package/dist/commands/test-guide.js.map +0 -1
  195. package/dist/commands/test-report.d.ts +0 -3
  196. package/dist/commands/test-report.d.ts.map +0 -1
  197. package/dist/commands/test-report.js +0 -196
  198. package/dist/commands/test-report.js.map +0 -1
  199. package/dist/commands/test-submit.d.ts +0 -6
  200. package/dist/commands/test-submit.d.ts.map +0 -1
  201. package/dist/commands/test-submit.js +0 -236
  202. package/dist/commands/test-submit.js.map +0 -1
  203. package/dist/commands/verify.d.ts +0 -3
  204. package/dist/commands/verify.d.ts.map +0 -1
  205. package/dist/commands/verify.js +0 -288
  206. package/dist/commands/verify.js.map +0 -1
  207. package/dist/config.d.ts +0 -23
  208. package/dist/config.d.ts.map +0 -1
  209. package/dist/config.js +0 -44
  210. package/dist/config.js.map +0 -1
  211. package/dist/index.d.ts +0 -8
  212. package/dist/index.d.ts.map +0 -1
  213. package/dist/index.js +0 -129
  214. package/dist/index.js.map +0 -1
  215. package/dist/rules/downloader.d.ts +0 -40
  216. package/dist/rules/downloader.d.ts.map +0 -1
  217. package/dist/rules/downloader.js +0 -253
  218. package/dist/rules/downloader.js.map +0 -1
  219. package/dist/rules/index.d.ts +0 -8
  220. package/dist/rules/index.d.ts.map +0 -1
  221. package/dist/rules/index.js +0 -8
  222. package/dist/rules/index.js.map +0 -1
  223. package/dist/rules/registry.d.ts +0 -45
  224. package/dist/rules/registry.d.ts.map +0 -1
  225. package/dist/rules/registry.js +0 -158
  226. package/dist/rules/registry.js.map +0 -1
  227. package/dist/rules/types.d.ts +0 -86
  228. package/dist/rules/types.d.ts.map +0 -1
  229. package/dist/rules/types.js +0 -6
  230. package/dist/rules/types.js.map +0 -1
  231. package/dist/utils/detectTools.d.ts +0 -15
  232. package/dist/utils/detectTools.d.ts.map +0 -1
  233. package/dist/utils/detectTools.js +0 -54
  234. package/dist/utils/detectTools.js.map +0 -1
  235. package/dist/utils/generateToolConfig.d.ts +0 -12
  236. package/dist/utils/generateToolConfig.d.ts.map +0 -1
  237. package/dist/utils/generateToolConfig.js +0 -1179
  238. package/dist/utils/generateToolConfig.js.map +0 -1
  239. package/dist/utils/testRunner.d.ts +0 -39
  240. package/dist/utils/testRunner.d.ts.map +0 -1
  241. package/dist/utils/testRunner.js +0 -325
  242. package/dist/utils/testRunner.js.map +0 -1
@@ -0,0 +1,49 @@
1
+ /**
2
+ * Canonical skill definition parsed from SKILL.md
3
+ */
4
+ export interface CanonicalSkill {
5
+ name: string;
6
+ description: string;
7
+ disableModelInvocation: boolean;
8
+ content: string;
9
+ workflowContent?: string;
10
+ directory: string;
11
+ }
12
+ /**
13
+ * Result of generating files for a tool
14
+ */
15
+ export interface GeneratedFile {
16
+ path: string;
17
+ content: string;
18
+ }
19
+ /**
20
+ * Each tool adapter transforms canonical skills into tool-specific files
21
+ */
22
+ export interface ToolAdapter {
23
+ /** Tool identifier (e.g., "claude-code", "opencode") */
24
+ name: string;
25
+ /** Human-readable tool name */
26
+ displayName: string;
27
+ /** Detect if this tool is present in the project */
28
+ detect(projectRoot: string): Promise<boolean>;
29
+ /** Generate all files for this tool from canonical skills */
30
+ generate(skills: CanonicalSkill[], projectRoot: string): Promise<GeneratedFile[]>;
31
+ }
32
+ /**
33
+ * SpecSafe project configuration (specsafe.config.json)
34
+ */
35
+ export interface SpecSafeConfig {
36
+ project: string;
37
+ version: string;
38
+ tools: string[];
39
+ testFramework: string;
40
+ testCommand: string;
41
+ coverageCommand: string;
42
+ language: string;
43
+ specsafeVersion: string;
44
+ }
45
+ /**
46
+ * Tool registry - maps tool names to their adapters
47
+ */
48
+ export declare const TOOL_NAMES: readonly ["claude-code", "opencode", "cursor", "continue", "aider", "zed", "gemini", "antigravity"];
49
+ export type ToolName = typeof TOOL_NAMES[number];
@@ -0,0 +1,14 @@
1
+ /**
2
+ * Tool registry - maps tool names to their adapters
3
+ */
4
+ export const TOOL_NAMES = [
5
+ 'claude-code',
6
+ 'opencode',
7
+ 'cursor',
8
+ 'continue',
9
+ 'aider',
10
+ 'zed',
11
+ 'gemini',
12
+ 'antigravity',
13
+ ];
14
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/adapters/types.ts"],"names":[],"mappings":"AAmDA;;GAEG;AACH,MAAM,CAAC,MAAM,UAAU,GAAG;IACxB,aAAa;IACb,UAAU;IACV,QAAQ;IACR,UAAU;IACV,OAAO;IACP,KAAK;IACL,QAAQ;IACR,aAAa;CACL,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { CanonicalSkill } from './types.js';
2
+ /** Parse YAML frontmatter from markdown content */
3
+ export declare function parseFrontmatter(content: string): {
4
+ frontmatter: Record<string, string>;
5
+ body: string;
6
+ };
7
+ /** Load all canonical skills from the canonical/skills directory */
8
+ export declare function loadCanonicalSkills(canonicalDir: string): CanonicalSkill[];
9
+ /** Read a canonical rule file, returning its content or empty string if missing */
10
+ export declare function readCanonicalRule(canonicalDir: string, filename: string): string;
11
+ /** Reconstruct full SKILL.md content (frontmatter + body) from a CanonicalSkill */
12
+ export declare function reconstructSkillMd(skill: CanonicalSkill): string;
@@ -0,0 +1,68 @@
1
+ import { readFileSync, readdirSync, existsSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ /** Parse YAML frontmatter from markdown content */
4
+ export function parseFrontmatter(content) {
5
+ // Normalize line endings to handle Windows \r\n
6
+ const normalized = content.replace(/\r\n/g, '\n');
7
+ const match = normalized.match(/^---\n([\s\S]*?)\n---\n([\s\S]*)$/);
8
+ if (!match)
9
+ return { frontmatter: {}, body: content };
10
+ const frontmatter = {};
11
+ for (const line of match[1].split('\n')) {
12
+ const colonIdx = line.indexOf(':');
13
+ if (colonIdx > 0) {
14
+ const key = line.slice(0, colonIdx).trim();
15
+ let value = line.slice(colonIdx + 1).trim();
16
+ // Remove quotes
17
+ if ((value.startsWith('"') && value.endsWith('"')) || (value.startsWith("'") && value.endsWith("'"))) {
18
+ value = value.slice(1, -1);
19
+ }
20
+ frontmatter[key] = value;
21
+ }
22
+ }
23
+ return { frontmatter, body: match[2] };
24
+ }
25
+ /** Load all canonical skills from the canonical/skills directory */
26
+ export function loadCanonicalSkills(canonicalDir) {
27
+ const skillsDir = join(canonicalDir, 'skills');
28
+ const skills = [];
29
+ for (const dir of readdirSync(skillsDir, { withFileTypes: true })) {
30
+ if (!dir.isDirectory())
31
+ continue;
32
+ const skillPath = join(skillsDir, dir.name, 'SKILL.md');
33
+ if (!existsSync(skillPath))
34
+ continue;
35
+ const content = readFileSync(skillPath, 'utf-8');
36
+ const { frontmatter, body } = parseFrontmatter(content);
37
+ const workflowPath = join(skillsDir, dir.name, 'workflow.md');
38
+ const workflowContent = existsSync(workflowPath)
39
+ ? readFileSync(workflowPath, 'utf-8')
40
+ : undefined;
41
+ skills.push({
42
+ name: frontmatter.name || dir.name,
43
+ description: frontmatter.description || '',
44
+ disableModelInvocation: frontmatter['disable-model-invocation'] === 'true',
45
+ content: body,
46
+ workflowContent,
47
+ directory: dir.name,
48
+ });
49
+ }
50
+ return skills;
51
+ }
52
+ /** Read a canonical rule file, returning its content or empty string if missing */
53
+ export function readCanonicalRule(canonicalDir, filename) {
54
+ const rulePath = join(canonicalDir, 'rules', filename);
55
+ return existsSync(rulePath) ? readFileSync(rulePath, 'utf-8') : '';
56
+ }
57
+ /** Reconstruct full SKILL.md content (frontmatter + body) from a CanonicalSkill */
58
+ export function reconstructSkillMd(skill) {
59
+ let fm = '---\n';
60
+ fm += `name: ${skill.name}\n`;
61
+ fm += `description: ${skill.description}\n`;
62
+ if (skill.disableModelInvocation) {
63
+ fm += `disable-model-invocation: true\n`;
64
+ }
65
+ fm += '---\n';
66
+ return fm + skill.content;
67
+ }
68
+ //# sourceMappingURL=utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/adapters/utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAChE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC,mDAAmD;AACnD,MAAM,UAAU,gBAAgB,CAAC,OAAe;IAC9C,gDAAgD;IAChD,MAAM,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAClD,MAAM,KAAK,GAAG,UAAU,CAAC,KAAK,CAAC,mCAAmC,CAAC,CAAC;IACpE,IAAI,CAAC,KAAK;QAAE,OAAO,EAAE,WAAW,EAAE,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;IACtD,MAAM,WAAW,GAA2B,EAAE,CAAC;IAC/C,KAAK,MAAM,IAAI,IAAI,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;QACxC,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACnC,IAAI,QAAQ,GAAG,CAAC,EAAE,CAAC;YACjB,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,IAAI,EAAE,CAAC;YAC3C,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;YAC5C,gBAAgB;YAChB,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;gBACrG,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YAC7B,CAAC;YACD,WAAW,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;QAC3B,CAAC;IACH,CAAC;IACD,OAAO,EAAE,WAAW,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;AACzC,CAAC;AAED,oEAAoE;AACpE,MAAM,UAAU,mBAAmB,CAAC,YAAoB;IACtD,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;IAC/C,MAAM,MAAM,GAAqB,EAAE,CAAC;IAEpC,KAAK,MAAM,GAAG,IAAI,WAAW,CAAC,SAAS,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;QAClE,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE;YAAE,SAAS;QACjC,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QACxD,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;YAAE,SAAS;QAErC,MAAM,OAAO,GAAG,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QACjD,MAAM,EAAE,WAAW,EAAE,IAAI,EAAE,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC;QAExD,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;QAC9D,MAAM,eAAe,GAAG,UAAU,CAAC,YAAY,CAAC;YAC9C,CAAC,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC;YACrC,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,CAAC,IAAI,CAAC;YACV,IAAI,EAAE,WAAW,CAAC,IAAI,IAAI,GAAG,CAAC,IAAI;YAClC,WAAW,EAAE,WAAW,CAAC,WAAW,IAAI,EAAE;YAC1C,sBAAsB,EAAE,WAAW,CAAC,0BAA0B,CAAC,KAAK,MAAM;YAC1E,OAAO,EAAE,IAAI;YACb,eAAe;YACf,SAAS,EAAE,GAAG,CAAC,IAAI;SACpB,CAAC,CAAC;IACL,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,mFAAmF;AACnF,MAAM,UAAU,iBAAiB,CAAC,YAAoB,EAAE,QAAgB;IACtE,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC;IACvD,OAAO,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;AACrE,CAAC;AAED,mFAAmF;AACnF,MAAM,UAAU,kBAAkB,CAAC,KAAqB;IACtD,IAAI,EAAE,GAAG,OAAO,CAAC;IACjB,EAAE,IAAI,SAAS,KAAK,CAAC,IAAI,IAAI,CAAC;IAC9B,EAAE,IAAI,gBAAgB,KAAK,CAAC,WAAW,IAAI,CAAC;IAC5C,IAAI,KAAK,CAAC,sBAAsB,EAAE,CAAC;QACjC,EAAE,IAAI,kCAAkC,CAAC;IAC3C,CAAC;IACD,EAAE,IAAI,OAAO,CAAC;IACd,OAAO,EAAE,GAAG,KAAK,CAAC,OAAO,CAAC;AAC5B,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { ToolAdapter } from './types.js';
2
+ export declare const zedAdapter: ToolAdapter;
@@ -0,0 +1,31 @@
1
+ import { existsSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ import { readCanonicalRule } from './utils.js';
4
+ export const zedAdapter = {
5
+ name: 'zed',
6
+ displayName: 'Zed',
7
+ async detect(projectRoot) {
8
+ return existsSync(join(projectRoot, '.zed'));
9
+ },
10
+ async generate(_skills, canonicalDir) {
11
+ const files = [];
12
+ const rules = readCanonicalRule(canonicalDir, '.rules');
13
+ if (rules) {
14
+ files.push({ path: '.rules', content: rules });
15
+ }
16
+ files.push({
17
+ path: '.zed/settings.json',
18
+ content: JSON.stringify({
19
+ assistant: {
20
+ version: '2',
21
+ default_model: {
22
+ provider: 'anthropic',
23
+ model: 'claude-sonnet-4-20250514',
24
+ },
25
+ },
26
+ }, null, 2) + '\n',
27
+ });
28
+ return files;
29
+ },
30
+ };
31
+ //# sourceMappingURL=zed.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"zed.js","sourceRoot":"","sources":["../../src/adapters/zed.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,OAAO,EAAE,iBAAiB,EAAE,MAAM,YAAY,CAAC;AAE/C,MAAM,CAAC,MAAM,UAAU,GAAgB;IACrC,IAAI,EAAE,KAAK;IACX,WAAW,EAAE,KAAK;IAElB,KAAK,CAAC,MAAM,CAAC,WAAmB;QAC9B,OAAO,UAAU,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC,CAAC;IAC/C,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,OAAyB,EAAE,YAAoB;QAC5D,MAAM,KAAK,GAAoB,EAAE,CAAC;QAElC,MAAM,KAAK,GAAG,iBAAiB,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;QACxD,IAAI,KAAK,EAAE,CAAC;YACV,KAAK,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC;QACjD,CAAC;QAED,KAAK,CAAC,IAAI,CAAC;YACT,IAAI,EAAE,oBAAoB;YAC1B,OAAO,EAAE,IAAI,CAAC,SAAS,CACrB;gBACE,SAAS,EAAE;oBACT,OAAO,EAAE,GAAG;oBACZ,aAAa,EAAE;wBACb,QAAQ,EAAE,WAAW;wBACrB,KAAK,EAAE,0BAA0B;qBAClC;iBACF;aACF,EACD,IAAI,EACJ,CAAC,CACF,GAAG,IAAI;SACT,CAAC,CAAC;QAEH,OAAO,KAAK,CAAC;IACf,CAAC;CACF,CAAC"}
@@ -0,0 +1,10 @@
1
+ export interface DoctorOptions {
2
+ cwd?: string;
3
+ }
4
+ interface Check {
5
+ label: string;
6
+ status: 'OK' | 'WARNING' | 'ERROR';
7
+ message?: string;
8
+ }
9
+ export declare function doctor(opts?: DoctorOptions): Promise<Check[]>;
10
+ export {};
@@ -0,0 +1,123 @@
1
+ import { readFile, access, readdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import Table from 'cli-table3';
4
+ import c from 'ansis';
5
+ import { getAdapter } from './registry.js';
6
+ export async function doctor(opts = {}) {
7
+ const cwd = opts.cwd ?? process.cwd();
8
+ const checks = [];
9
+ // Check specsafe.config.json
10
+ let config = null;
11
+ const configPath = join(cwd, 'specsafe.config.json');
12
+ try {
13
+ const raw = await readFile(configPath, 'utf-8');
14
+ config = JSON.parse(raw);
15
+ // Validate required keys
16
+ const requiredKeys = ['project', 'version', 'tools', 'specsafeVersion'];
17
+ const missing = requiredKeys.filter(k => !(k in config));
18
+ if (missing.length > 0) {
19
+ checks.push({ label: 'specsafe.config.json', status: 'ERROR', message: `Missing keys: ${missing.join(', ')}` });
20
+ }
21
+ else {
22
+ checks.push({ label: 'specsafe.config.json', status: 'OK' });
23
+ }
24
+ }
25
+ catch (err) {
26
+ if (err.code === 'ENOENT') {
27
+ checks.push({ label: 'specsafe.config.json', status: 'ERROR', message: 'File not found' });
28
+ }
29
+ else {
30
+ checks.push({ label: 'specsafe.config.json', status: 'ERROR', message: 'Invalid JSON' });
31
+ }
32
+ }
33
+ // Check PROJECT_STATE.md
34
+ if (await fileExists(join(cwd, 'PROJECT_STATE.md'))) {
35
+ checks.push({ label: 'PROJECT_STATE.md', status: 'OK' });
36
+ }
37
+ else {
38
+ checks.push({ label: 'PROJECT_STATE.md', status: 'ERROR', message: 'File not found' });
39
+ }
40
+ // Check directory structure
41
+ const specDirs = ['specs/active', 'specs/completed', 'specs/archive'];
42
+ for (const dir of specDirs) {
43
+ if (await dirExists(join(cwd, dir))) {
44
+ checks.push({ label: dir, status: 'OK' });
45
+ }
46
+ else {
47
+ checks.push({ label: dir, status: 'WARNING', message: 'Directory missing' });
48
+ }
49
+ }
50
+ // Check installed tools
51
+ if (config?.tools) {
52
+ for (const tool of config.tools) {
53
+ const adapter = getAdapter(tool);
54
+ if (adapter) {
55
+ const detected = await adapter.detect(cwd);
56
+ checks.push({
57
+ label: `tool: ${tool}`,
58
+ status: detected ? 'OK' : 'WARNING',
59
+ message: detected ? undefined : 'Tool files not detected',
60
+ });
61
+ }
62
+ else {
63
+ checks.push({ label: `tool: ${tool}`, status: 'WARNING', message: 'Adapter not loaded' });
64
+ }
65
+ }
66
+ }
67
+ // Print formatted report
68
+ const table = new Table({
69
+ head: [c.bold('Status'), c.bold('Check'), c.bold('Detail')],
70
+ style: { head: [], border: [] },
71
+ });
72
+ for (const check of checks) {
73
+ let statusLabel;
74
+ let detail = check.message ?? '';
75
+ if (check.status === 'OK') {
76
+ statusLabel = c.green('\u2713 OK');
77
+ if (!detail)
78
+ detail = c.green('Valid');
79
+ }
80
+ else if (check.status === 'WARNING') {
81
+ statusLabel = c.yellow('\u26A0 WARN');
82
+ detail = c.yellow(detail);
83
+ }
84
+ else {
85
+ statusLabel = c.red('\u2717 ERR');
86
+ detail = c.red(detail);
87
+ }
88
+ table.push([statusLabel, check.label, detail]);
89
+ }
90
+ console.log(table.toString());
91
+ const errorCount = checks.filter(ch => ch.status === 'ERROR').length;
92
+ const warnCount = checks.filter(ch => ch.status === 'WARNING').length;
93
+ if (errorCount > 0) {
94
+ console.log(c.red(`\n${errorCount} error(s)${warnCount > 0 ? `, ${warnCount} warning(s)` : ''} found. Run \`specsafe init\` to fix.`));
95
+ process.exitCode = 1;
96
+ }
97
+ else if (warnCount > 0) {
98
+ console.log(c.yellow(`\n${warnCount} warning(s), but project looks healthy.`));
99
+ }
100
+ else {
101
+ console.log(c.green('\nProject looks healthy!'));
102
+ }
103
+ return checks;
104
+ }
105
+ async function fileExists(path) {
106
+ try {
107
+ await access(path);
108
+ return true;
109
+ }
110
+ catch {
111
+ return false;
112
+ }
113
+ }
114
+ async function dirExists(path) {
115
+ try {
116
+ await readdir(path);
117
+ return true;
118
+ }
119
+ catch {
120
+ return false;
121
+ }
122
+ }
123
+ //# sourceMappingURL=doctor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"doctor.js","sourceRoot":"","sources":["../src/doctor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
@@ -0,0 +1,45 @@
1
+ #!/usr/bin/env node
2
+ import { Command } from 'commander';
3
+ import { readFileSync } from 'node:fs';
4
+ import { fileURLToPath } from 'node:url';
5
+ import { dirname, join } from 'node:path';
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = dirname(__filename);
8
+ const pkg = JSON.parse(readFileSync(join(__dirname, '..', 'package.json'), 'utf-8'));
9
+ const program = new Command();
10
+ program
11
+ .name('specsafe')
12
+ .description('SpecSafe — Skills-first TDD framework for AI-assisted development')
13
+ .version(pkg.version);
14
+ program
15
+ .command('init')
16
+ .description('Initialize a new SpecSafe project')
17
+ .argument('[name]', 'Project name (defaults to directory name)')
18
+ .action(async (name) => {
19
+ const { init } = await import('./init.js');
20
+ await init(name, { interactive: true });
21
+ });
22
+ program
23
+ .command('install')
24
+ .description('Install SpecSafe skills for a specific AI tool')
25
+ .argument('<tool>', 'Tool name (claude-code, opencode, cursor, continue, aider, zed, gemini, antigravity)')
26
+ .action(async (tool) => {
27
+ const { install } = await import('./install.js');
28
+ await install(tool);
29
+ });
30
+ program
31
+ .command('update')
32
+ .description('Regenerate all tool files from canonical skills')
33
+ .action(async () => {
34
+ const { update } = await import('./update.js');
35
+ await update();
36
+ });
37
+ program
38
+ .command('doctor')
39
+ .description('Validate SpecSafe project health')
40
+ .action(async () => {
41
+ const { doctor } = await import('./doctor.js');
42
+ await doctor();
43
+ });
44
+ program.parse();
45
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAE1C,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClD,MAAM,SAAS,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;AACtC,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,cAAc,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AAErF,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,UAAU,CAAC;KAChB,WAAW,CAAC,mEAAmE,CAAC;KAChF,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;AAExB,OAAO;KACJ,OAAO,CAAC,MAAM,CAAC;KACf,WAAW,CAAC,mCAAmC,CAAC;KAChD,QAAQ,CAAC,QAAQ,EAAE,2CAA2C,CAAC;KAC/D,MAAM,CAAC,KAAK,EAAE,IAAa,EAAE,EAAE;IAC9B,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,MAAM,CAAC,WAAW,CAAC,CAAC;IAC3C,MAAM,IAAI,CAAC,IAAI,EAAE,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC;AAC1C,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,SAAS,CAAC;KAClB,WAAW,CAAC,gDAAgD,CAAC;KAC7D,QAAQ,CAAC,QAAQ,EAAE,sFAAsF,CAAC;KAC1G,MAAM,CAAC,KAAK,EAAE,IAAY,EAAE,EAAE;IAC7B,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,CAAC;IACjD,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;AACtB,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,iDAAiD,CAAC;KAC9D,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,CAAC;IAC/C,MAAM,MAAM,EAAE,CAAC;AACjB,CAAC,CAAC,CAAC;AAEL,OAAO;KACJ,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,kCAAkC,CAAC;KAC/C,MAAM,CAAC,KAAK,IAAI,EAAE;IACjB,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,aAAa,CAAC,CAAC;IAC/C,MAAM,MAAM,EAAE,CAAC;AACjB,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,EAAE,CAAC"}
@@ -0,0 +1,9 @@
1
+ export interface InitOptions {
2
+ cwd?: string;
3
+ canonicalDir?: string;
4
+ interactive?: boolean;
5
+ tools?: string[];
6
+ testFramework?: string;
7
+ language?: string;
8
+ }
9
+ export declare function init(name?: string, opts?: InitOptions): Promise<void>;
@@ -0,0 +1,167 @@
1
+ import { mkdir, readFile, writeFile, access } from 'node:fs/promises';
2
+ import { basename, join, resolve } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+ import * as p from '@clack/prompts';
5
+ import c from 'ansis';
6
+ const __filename = fileURLToPath(import.meta.url);
7
+ const __dirname = import.meta.dirname ?? resolve(__filename, '..');
8
+ function defaultCanonicalDir() {
9
+ return resolve(__dirname, '..', '..', 'canonical');
10
+ }
11
+ const TOOL_DETECT_MAP = {
12
+ 'claude-code': '.claude/',
13
+ cursor: '.cursor/',
14
+ opencode: '.opencode/',
15
+ gemini: '.gemini/',
16
+ antigravity: '.agent/',
17
+ zed: '.zed/',
18
+ continue: '.continue/',
19
+ aider: '.aider.conf.yml',
20
+ };
21
+ async function detectTools(cwd) {
22
+ const detected = [];
23
+ for (const [tool, marker] of Object.entries(TOOL_DETECT_MAP)) {
24
+ if (await fileExists(join(cwd, marker))) {
25
+ detected.push(tool);
26
+ }
27
+ }
28
+ return detected;
29
+ }
30
+ export async function init(name, opts = {}) {
31
+ const cwd = opts.cwd ?? process.cwd();
32
+ const canonicalDir = opts.canonicalDir ?? defaultCanonicalDir();
33
+ const interactive = opts.interactive ?? false;
34
+ // Check if already initialized
35
+ const configPath = join(cwd, 'specsafe.config.json');
36
+ if (await fileExists(configPath)) {
37
+ if (interactive) {
38
+ p.log.warn('SpecSafe is already initialized in this directory. Run `specsafe doctor` to check project health.');
39
+ }
40
+ else {
41
+ console.log('SpecSafe is already initialized in this directory. Run `specsafe doctor` to check project health.');
42
+ }
43
+ return;
44
+ }
45
+ let projectName = name ?? basename(cwd);
46
+ let selectedTools = opts.tools ?? [];
47
+ let testFramework = opts.testFramework ?? 'vitest';
48
+ let language = opts.language ?? 'typescript';
49
+ if (interactive) {
50
+ p.intro(c.cyan('SpecSafe — Initialize Project'));
51
+ const result = await p.group({
52
+ projectName: () => p.text({
53
+ message: 'Project name?',
54
+ placeholder: basename(cwd),
55
+ defaultValue: basename(cwd),
56
+ }),
57
+ tools: async () => {
58
+ const detected = await detectTools(cwd);
59
+ const allTools = Object.keys(TOOL_DETECT_MAP);
60
+ return p.multiselect({
61
+ message: 'Which tools to install?',
62
+ options: allTools.map(t => ({
63
+ value: t,
64
+ label: t,
65
+ hint: detected.includes(t) ? 'detected' : undefined,
66
+ })),
67
+ initialValues: detected,
68
+ required: false,
69
+ });
70
+ },
71
+ testFramework: () => p.select({
72
+ message: 'Test framework?',
73
+ options: [
74
+ { value: 'vitest', label: 'vitest' },
75
+ { value: 'jest', label: 'jest' },
76
+ { value: 'pytest', label: 'pytest' },
77
+ { value: 'go test', label: 'go test' },
78
+ ],
79
+ }),
80
+ language: () => p.select({
81
+ message: 'Language?',
82
+ options: [
83
+ { value: 'typescript', label: 'typescript' },
84
+ { value: 'python', label: 'python' },
85
+ { value: 'go', label: 'go' },
86
+ { value: 'rust', label: 'rust' },
87
+ { value: 'other', label: 'other' },
88
+ ],
89
+ }),
90
+ }, {
91
+ onCancel: () => {
92
+ p.cancel('Setup cancelled.');
93
+ process.exit(0);
94
+ },
95
+ });
96
+ projectName = result.projectName || basename(cwd);
97
+ selectedTools = result.tools ?? [];
98
+ testFramework = result.testFramework;
99
+ language = result.language;
100
+ const s = p.spinner();
101
+ s.start('Creating project files...');
102
+ await createProjectFiles(cwd, canonicalDir, projectName, testFramework, language);
103
+ s.stop('Project files created.');
104
+ if (selectedTools.length > 0) {
105
+ const si = p.spinner();
106
+ si.start(`Installing ${selectedTools.length} tool(s)...`);
107
+ const { install } = await import('./install.js');
108
+ for (const tool of selectedTools) {
109
+ await install(tool, { cwd, canonicalDir });
110
+ }
111
+ si.stop(`Installed ${selectedTools.length} tool(s).`);
112
+ }
113
+ p.outro(c.green('Project initialized! Run /specsafe-new to create your first spec.'));
114
+ }
115
+ else {
116
+ await createProjectFiles(cwd, canonicalDir, projectName, testFramework, language);
117
+ console.log(`SpecSafe initialized for project: ${projectName}
118
+
119
+ Created:
120
+ specs/active/
121
+ specs/completed/
122
+ specs/archive/
123
+ specsafe.config.json
124
+ PROJECT_STATE.md
125
+ specs/template.md
126
+
127
+ Next steps:
128
+ 1. Review specsafe.config.json and fill in any missing fields
129
+ 2. Run \`specsafe install <tool>\` to install skills for your AI tool`);
130
+ }
131
+ }
132
+ async function createProjectFiles(cwd, canonicalDir, projectName, testFramework, language) {
133
+ // Create directories
134
+ const dirs = ['specs/active', 'specs/completed', 'specs/archive'];
135
+ for (const dir of dirs) {
136
+ await mkdir(join(cwd, dir), { recursive: true });
137
+ }
138
+ // Copy and populate config template
139
+ const configTemplate = await readFile(join(canonicalDir, 'templates', 'specsafe-config-template.json'), 'utf-8');
140
+ const config = configTemplate.replace(/\{\{project-name\}\}/g, projectName);
141
+ const configPath = join(cwd, 'specsafe.config.json');
142
+ // Parse config to inject testFramework and language
143
+ const configObj = JSON.parse(config);
144
+ configObj.testFramework = testFramework;
145
+ configObj.language = language;
146
+ await writeFile(configPath, JSON.stringify(configObj, null, 2) + '\n', 'utf-8');
147
+ // Copy and populate PROJECT_STATE.md template
148
+ const stateTemplate = await readFile(join(canonicalDir, 'templates', 'project-state-template.md'), 'utf-8');
149
+ const state = stateTemplate
150
+ .replace(/\{\{project-name\}\}/g, projectName)
151
+ .replace(/\{\{version\}\}/g, '1.0.0')
152
+ .replace(/\{\{timestamp\}\}/g, new Date().toISOString().split('T')[0]);
153
+ await writeFile(join(cwd, 'PROJECT_STATE.md'), state, 'utf-8');
154
+ // Copy spec template
155
+ const specTemplate = await readFile(join(canonicalDir, 'templates', 'spec-template.md'), 'utf-8');
156
+ await writeFile(join(cwd, 'specs', 'template.md'), specTemplate, 'utf-8');
157
+ }
158
+ async function fileExists(path) {
159
+ try {
160
+ await access(path);
161
+ return true;
162
+ }
163
+ catch {
164
+ return false;
165
+ }
166
+ }
167
+ //# sourceMappingURL=init.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init.js","sourceRoot":"","sources":["../src/init.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ export interface InstallOptions {
2
+ cwd?: string;
3
+ canonicalDir?: string;
4
+ }
5
+ export declare function install(tool: string, opts?: InstallOptions): Promise<void>;