@tienne/gestalt 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/CLAUDE.md +203 -0
  2. package/README.md +60 -0
  3. package/agents/closure-completer/AGENT.md +32 -0
  4. package/agents/continuity-judge/AGENT.md +40 -0
  5. package/agents/ground-mapper/AGENT.md +31 -0
  6. package/agents/proximity-worker/AGENT.md +33 -0
  7. package/agents/similarity-crystallizer/AGENT.md +34 -0
  8. package/dist/agents/closure-completer/AGENT.md +32 -0
  9. package/dist/agents/continuity-judge/AGENT.md +40 -0
  10. package/dist/agents/ground-mapper/AGENT.md +31 -0
  11. package/dist/agents/proximity-worker/AGENT.md +33 -0
  12. package/dist/agents/similarity-crystallizer/AGENT.md +34 -0
  13. package/dist/bin/gestalt.d.ts +3 -0
  14. package/dist/bin/gestalt.d.ts.map +1 -0
  15. package/dist/bin/gestalt.js +5 -0
  16. package/dist/bin/gestalt.js.map +1 -0
  17. package/dist/skills/execute/SKILL.md +84 -0
  18. package/dist/skills/interview/SKILL.md +82 -0
  19. package/dist/skills/seed/SKILL.md +92 -0
  20. package/dist/skills/spec/SKILL.md +92 -0
  21. package/dist/src/agent/figural-router.d.ts +48 -0
  22. package/dist/src/agent/figural-router.d.ts.map +1 -0
  23. package/dist/src/agent/figural-router.js +55 -0
  24. package/dist/src/agent/figural-router.js.map +1 -0
  25. package/dist/src/agent/parser.d.ts +3 -0
  26. package/dist/src/agent/parser.d.ts.map +1 -0
  27. package/dist/src/agent/parser.js +32 -0
  28. package/dist/src/agent/parser.js.map +1 -0
  29. package/dist/src/agent/prompt-resolver.d.ts +17 -0
  30. package/dist/src/agent/prompt-resolver.d.ts.map +1 -0
  31. package/dist/src/agent/prompt-resolver.js +31 -0
  32. package/dist/src/agent/prompt-resolver.js.map +1 -0
  33. package/dist/src/agent/registry.d.ts +9 -0
  34. package/dist/src/agent/registry.d.ts.map +1 -0
  35. package/dist/src/agent/registry.js +22 -0
  36. package/dist/src/agent/registry.js.map +1 -0
  37. package/dist/src/cli/commands/interview.d.ts +2 -0
  38. package/dist/src/cli/commands/interview.d.ts.map +1 -0
  39. package/dist/src/cli/commands/interview.js +67 -0
  40. package/dist/src/cli/commands/interview.js.map +1 -0
  41. package/dist/src/cli/commands/seed.d.ts +4 -0
  42. package/dist/src/cli/commands/seed.d.ts.map +1 -0
  43. package/dist/src/cli/commands/seed.js +34 -0
  44. package/dist/src/cli/commands/seed.js.map +1 -0
  45. package/dist/src/cli/commands/serve.d.ts +2 -0
  46. package/dist/src/cli/commands/serve.d.ts.map +1 -0
  47. package/dist/src/cli/commands/serve.js +5 -0
  48. package/dist/src/cli/commands/serve.js.map +1 -0
  49. package/dist/src/cli/commands/spec.d.ts +4 -0
  50. package/dist/src/cli/commands/spec.d.ts.map +1 -0
  51. package/dist/src/cli/commands/spec.js +34 -0
  52. package/dist/src/cli/commands/spec.js.map +1 -0
  53. package/dist/src/cli/commands/status.d.ts +2 -0
  54. package/dist/src/cli/commands/status.d.ts.map +1 -0
  55. package/dist/src/cli/commands/status.js +66 -0
  56. package/dist/src/cli/commands/status.js.map +1 -0
  57. package/dist/src/cli/index.d.ts +3 -0
  58. package/dist/src/cli/index.d.ts.map +1 -0
  59. package/dist/src/cli/index.js +39 -0
  60. package/dist/src/cli/index.js.map +1 -0
  61. package/dist/src/core/config.d.ts +36 -0
  62. package/dist/src/core/config.d.ts.map +1 -0
  63. package/dist/src/core/config.js +65 -0
  64. package/dist/src/core/config.js.map +1 -0
  65. package/dist/src/core/constants.d.ts +34 -0
  66. package/dist/src/core/constants.d.ts.map +1 -0
  67. package/dist/src/core/constants.js +68 -0
  68. package/dist/src/core/constants.js.map +1 -0
  69. package/dist/src/core/errors.d.ts +50 -0
  70. package/dist/src/core/errors.d.ts.map +1 -0
  71. package/dist/src/core/errors.js +99 -0
  72. package/dist/src/core/errors.js.map +1 -0
  73. package/dist/src/core/log.d.ts +5 -0
  74. package/dist/src/core/log.d.ts.map +1 -0
  75. package/dist/src/core/log.js +7 -0
  76. package/dist/src/core/log.js.map +1 -0
  77. package/dist/src/core/result.d.ts +20 -0
  78. package/dist/src/core/result.d.ts.map +1 -0
  79. package/dist/src/core/result.js +43 -0
  80. package/dist/src/core/result.js.map +1 -0
  81. package/dist/src/core/types.d.ts +289 -0
  82. package/dist/src/core/types.d.ts.map +1 -0
  83. package/dist/src/core/types.js +15 -0
  84. package/dist/src/core/types.js.map +1 -0
  85. package/dist/src/events/store.d.ts +22 -0
  86. package/dist/src/events/store.d.ts.map +1 -0
  87. package/dist/src/events/store.js +130 -0
  88. package/dist/src/events/store.js.map +1 -0
  89. package/dist/src/events/types.d.ts +33 -0
  90. package/dist/src/events/types.d.ts.map +1 -0
  91. package/dist/src/events/types.js +35 -0
  92. package/dist/src/events/types.js.map +1 -0
  93. package/dist/src/execute/dag-validator.d.ts +7 -0
  94. package/dist/src/execute/dag-validator.d.ts.map +1 -0
  95. package/dist/src/execute/dag-validator.js +141 -0
  96. package/dist/src/execute/dag-validator.js.map +1 -0
  97. package/dist/src/execute/drift-detector.d.ts +11 -0
  98. package/dist/src/execute/drift-detector.d.ts.map +1 -0
  99. package/dist/src/execute/drift-detector.js +111 -0
  100. package/dist/src/execute/drift-detector.js.map +1 -0
  101. package/dist/src/execute/impact-identifier.d.ts +12 -0
  102. package/dist/src/execute/impact-identifier.d.ts.map +1 -0
  103. package/dist/src/execute/impact-identifier.js +60 -0
  104. package/dist/src/execute/impact-identifier.js.map +1 -0
  105. package/dist/src/execute/passthrough-engine.d.ts +181 -0
  106. package/dist/src/execute/passthrough-engine.d.ts.map +1 -0
  107. package/dist/src/execute/passthrough-engine.js +795 -0
  108. package/dist/src/execute/passthrough-engine.js.map +1 -0
  109. package/dist/src/execute/prompts.d.ts +19 -0
  110. package/dist/src/execute/prompts.d.ts.map +1 -0
  111. package/dist/src/execute/prompts.js +505 -0
  112. package/dist/src/execute/prompts.js.map +1 -0
  113. package/dist/src/execute/repository.d.ts +30 -0
  114. package/dist/src/execute/repository.d.ts.map +1 -0
  115. package/dist/src/execute/repository.js +248 -0
  116. package/dist/src/execute/repository.js.map +1 -0
  117. package/dist/src/execute/session.d.ts +34 -0
  118. package/dist/src/execute/session.d.ts.map +1 -0
  119. package/dist/src/execute/session.js +273 -0
  120. package/dist/src/execute/session.js.map +1 -0
  121. package/dist/src/execute/spec-patch-applier.d.ts +13 -0
  122. package/dist/src/execute/spec-patch-applier.d.ts.map +1 -0
  123. package/dist/src/execute/spec-patch-applier.js +83 -0
  124. package/dist/src/execute/spec-patch-applier.js.map +1 -0
  125. package/dist/src/execute/spec-patch-validator.d.ts +20 -0
  126. package/dist/src/execute/spec-patch-validator.d.ts.map +1 -0
  127. package/dist/src/execute/spec-patch-validator.js +60 -0
  128. package/dist/src/execute/spec-patch-validator.js.map +1 -0
  129. package/dist/src/execute/termination-detector.d.ts +20 -0
  130. package/dist/src/execute/termination-detector.d.ts.map +1 -0
  131. package/dist/src/execute/termination-detector.js +77 -0
  132. package/dist/src/execute/termination-detector.js.map +1 -0
  133. package/dist/src/gestalt/analyzer.d.ts +18 -0
  134. package/dist/src/gestalt/analyzer.d.ts.map +1 -0
  135. package/dist/src/gestalt/analyzer.js +59 -0
  136. package/dist/src/gestalt/analyzer.js.map +1 -0
  137. package/dist/src/gestalt/principles.d.ts +21 -0
  138. package/dist/src/gestalt/principles.d.ts.map +1 -0
  139. package/dist/src/gestalt/principles.js +66 -0
  140. package/dist/src/gestalt/principles.js.map +1 -0
  141. package/dist/src/index.d.ts +22 -0
  142. package/dist/src/index.d.ts.map +1 -0
  143. package/dist/src/index.js +22 -0
  144. package/dist/src/index.js.map +1 -0
  145. package/dist/src/interview/ambiguity.d.ts +8 -0
  146. package/dist/src/interview/ambiguity.d.ts.map +1 -0
  147. package/dist/src/interview/ambiguity.js +69 -0
  148. package/dist/src/interview/ambiguity.js.map +1 -0
  149. package/dist/src/interview/brownfield.d.ts +7 -0
  150. package/dist/src/interview/brownfield.d.ts.map +1 -0
  151. package/dist/src/interview/brownfield.js +28 -0
  152. package/dist/src/interview/brownfield.js.map +1 -0
  153. package/dist/src/interview/engine.d.ts +31 -0
  154. package/dist/src/interview/engine.d.ts.map +1 -0
  155. package/dist/src/interview/engine.js +110 -0
  156. package/dist/src/interview/engine.js.map +1 -0
  157. package/dist/src/interview/passthrough-engine.d.ts +52 -0
  158. package/dist/src/interview/passthrough-engine.d.ts.map +1 -0
  159. package/dist/src/interview/passthrough-engine.js +174 -0
  160. package/dist/src/interview/passthrough-engine.js.map +1 -0
  161. package/dist/src/interview/questions.d.ts +12 -0
  162. package/dist/src/interview/questions.d.ts.map +1 -0
  163. package/dist/src/interview/questions.js +54 -0
  164. package/dist/src/interview/questions.js.map +1 -0
  165. package/dist/src/interview/repository.d.ts +25 -0
  166. package/dist/src/interview/repository.d.ts.map +1 -0
  167. package/dist/src/interview/repository.js +102 -0
  168. package/dist/src/interview/repository.js.map +1 -0
  169. package/dist/src/interview/session.d.ts +22 -0
  170. package/dist/src/interview/session.d.ts.map +1 -0
  171. package/dist/src/interview/session.js +120 -0
  172. package/dist/src/interview/session.js.map +1 -0
  173. package/dist/src/llm/adapter.d.ts +8 -0
  174. package/dist/src/llm/adapter.d.ts.map +1 -0
  175. package/dist/src/llm/adapter.js +42 -0
  176. package/dist/src/llm/adapter.js.map +1 -0
  177. package/dist/src/llm/openai-adapter.d.ts +8 -0
  178. package/dist/src/llm/openai-adapter.d.ts.map +1 -0
  179. package/dist/src/llm/openai-adapter.js +45 -0
  180. package/dist/src/llm/openai-adapter.js.map +1 -0
  181. package/dist/src/llm/prompts.d.ts +15 -0
  182. package/dist/src/llm/prompts.d.ts.map +1 -0
  183. package/dist/src/llm/prompts.js +83 -0
  184. package/dist/src/llm/prompts.js.map +1 -0
  185. package/dist/src/llm/types.d.ts +21 -0
  186. package/dist/src/llm/types.d.ts.map +1 -0
  187. package/dist/src/llm/types.js +2 -0
  188. package/dist/src/llm/types.js.map +1 -0
  189. package/dist/src/mcp/schemas.d.ts +799 -0
  190. package/dist/src/mcp/schemas.d.ts.map +1 -0
  191. package/dist/src/mcp/schemas.js +151 -0
  192. package/dist/src/mcp/schemas.js.map +1 -0
  193. package/dist/src/mcp/server.d.ts +13 -0
  194. package/dist/src/mcp/server.d.ts.map +1 -0
  195. package/dist/src/mcp/server.js +255 -0
  196. package/dist/src/mcp/server.js.map +1 -0
  197. package/dist/src/mcp/tools/execute-passthrough.d.ts +4 -0
  198. package/dist/src/mcp/tools/execute-passthrough.d.ts.map +1 -0
  199. package/dist/src/mcp/tools/execute-passthrough.js +349 -0
  200. package/dist/src/mcp/tools/execute-passthrough.js.map +1 -0
  201. package/dist/src/mcp/tools/index.d.ts +4 -0
  202. package/dist/src/mcp/tools/index.d.ts.map +1 -0
  203. package/dist/src/mcp/tools/index.js +4 -0
  204. package/dist/src/mcp/tools/index.js.map +1 -0
  205. package/dist/src/mcp/tools/interview-passthrough.d.ts +4 -0
  206. package/dist/src/mcp/tools/interview-passthrough.d.ts.map +1 -0
  207. package/dist/src/mcp/tools/interview-passthrough.js +96 -0
  208. package/dist/src/mcp/tools/interview-passthrough.js.map +1 -0
  209. package/dist/src/mcp/tools/interview.d.ts +4 -0
  210. package/dist/src/mcp/tools/interview.d.ts.map +1 -0
  211. package/dist/src/mcp/tools/interview.js +85 -0
  212. package/dist/src/mcp/tools/interview.js.map +1 -0
  213. package/dist/src/mcp/tools/seed-passthrough.d.ts +5 -0
  214. package/dist/src/mcp/tools/seed-passthrough.d.ts.map +1 -0
  215. package/dist/src/mcp/tools/seed-passthrough.js +29 -0
  216. package/dist/src/mcp/tools/seed-passthrough.js.map +1 -0
  217. package/dist/src/mcp/tools/seed.d.ts +5 -0
  218. package/dist/src/mcp/tools/seed.d.ts.map +1 -0
  219. package/dist/src/mcp/tools/seed.js +19 -0
  220. package/dist/src/mcp/tools/seed.js.map +1 -0
  221. package/dist/src/mcp/tools/spec-passthrough.d.ts +5 -0
  222. package/dist/src/mcp/tools/spec-passthrough.d.ts.map +1 -0
  223. package/dist/src/mcp/tools/spec-passthrough.js +29 -0
  224. package/dist/src/mcp/tools/spec-passthrough.js.map +1 -0
  225. package/dist/src/mcp/tools/spec.d.ts +5 -0
  226. package/dist/src/mcp/tools/spec.d.ts.map +1 -0
  227. package/dist/src/mcp/tools/spec.js +19 -0
  228. package/dist/src/mcp/tools/spec.js.map +1 -0
  229. package/dist/src/mcp/tools/status.d.ts +4 -0
  230. package/dist/src/mcp/tools/status.d.ts.map +1 -0
  231. package/dist/src/mcp/tools/status.js +45 -0
  232. package/dist/src/mcp/tools/status.js.map +1 -0
  233. package/dist/src/registry/base-registry.d.ts +26 -0
  234. package/dist/src/registry/base-registry.d.ts.map +1 -0
  235. package/dist/src/registry/base-registry.js +82 -0
  236. package/dist/src/registry/base-registry.js.map +1 -0
  237. package/dist/src/seed/extractor.d.ts +15 -0
  238. package/dist/src/seed/extractor.d.ts.map +1 -0
  239. package/dist/src/seed/extractor.js +88 -0
  240. package/dist/src/seed/extractor.js.map +1 -0
  241. package/dist/src/seed/generator.d.ts +12 -0
  242. package/dist/src/seed/generator.d.ts.map +1 -0
  243. package/dist/src/seed/generator.js +66 -0
  244. package/dist/src/seed/generator.js.map +1 -0
  245. package/dist/src/seed/passthrough-generator.d.ts +31 -0
  246. package/dist/src/seed/passthrough-generator.d.ts.map +1 -0
  247. package/dist/src/seed/passthrough-generator.js +80 -0
  248. package/dist/src/seed/passthrough-generator.js.map +1 -0
  249. package/dist/src/seed/schema.d.ts +145 -0
  250. package/dist/src/seed/schema.d.ts.map +1 -0
  251. package/dist/src/seed/schema.js +37 -0
  252. package/dist/src/seed/schema.js.map +1 -0
  253. package/dist/src/skills/executor.d.ts +14 -0
  254. package/dist/src/skills/executor.d.ts.map +1 -0
  255. package/dist/src/skills/executor.js +17 -0
  256. package/dist/src/skills/executor.js.map +1 -0
  257. package/dist/src/skills/parser.d.ts +3 -0
  258. package/dist/src/skills/parser.d.ts.map +1 -0
  259. package/dist/src/skills/parser.js +37 -0
  260. package/dist/src/skills/parser.js.map +1 -0
  261. package/dist/src/skills/registry.d.ts +8 -0
  262. package/dist/src/skills/registry.d.ts.map +1 -0
  263. package/dist/src/skills/registry.js +19 -0
  264. package/dist/src/skills/registry.js.map +1 -0
  265. package/dist/src/skills/types.d.ts +19 -0
  266. package/dist/src/skills/types.d.ts.map +1 -0
  267. package/dist/src/skills/types.js +2 -0
  268. package/dist/src/skills/types.js.map +1 -0
  269. package/dist/src/spec/extractor.d.ts +15 -0
  270. package/dist/src/spec/extractor.d.ts.map +1 -0
  271. package/dist/src/spec/extractor.js +88 -0
  272. package/dist/src/spec/extractor.js.map +1 -0
  273. package/dist/src/spec/generator.d.ts +12 -0
  274. package/dist/src/spec/generator.d.ts.map +1 -0
  275. package/dist/src/spec/generator.js +66 -0
  276. package/dist/src/spec/generator.js.map +1 -0
  277. package/dist/src/spec/passthrough-generator.d.ts +34 -0
  278. package/dist/src/spec/passthrough-generator.d.ts.map +1 -0
  279. package/dist/src/spec/passthrough-generator.js +86 -0
  280. package/dist/src/spec/passthrough-generator.js.map +1 -0
  281. package/dist/src/spec/schema.d.ts +145 -0
  282. package/dist/src/spec/schema.d.ts.map +1 -0
  283. package/dist/src/spec/schema.js +37 -0
  284. package/dist/src/spec/schema.js.map +1 -0
  285. package/package.json +53 -0
  286. package/skills/execute/SKILL.md +84 -0
  287. package/skills/interview/SKILL.md +82 -0
  288. package/skills/spec/SKILL.md +92 -0
@@ -0,0 +1,80 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { AmbiguityThresholdError, SeedGenerationError } from '../core/errors.js';
3
+ import { AMBIGUITY_THRESHOLD } from '../core/constants.js';
4
+ import { ok, err } from '../core/result.js';
5
+ import { seedSchema } from './schema.js';
6
+ import { EventType } from '../events/types.js';
7
+ import { INTERVIEW_SYSTEM_PROMPT, buildSeedPrompt } from '../llm/prompts.js';
8
+ // ─── Generator ──────────────────────────────────────────────────
9
+ export class PassthroughSeedGenerator {
10
+ eventStore;
11
+ constructor(eventStore) {
12
+ this.eventStore = eventStore;
13
+ }
14
+ buildSeedContext(session) {
15
+ const answeredRounds = session.rounds
16
+ .filter((r) => r.userResponse !== null)
17
+ .map((r) => ({
18
+ question: r.question,
19
+ response: r.userResponse,
20
+ }));
21
+ const seedPrompt = buildSeedPrompt(session.topic, answeredRounds, session.projectType);
22
+ return {
23
+ systemPrompt: INTERVIEW_SYSTEM_PROMPT,
24
+ seedPrompt,
25
+ allRounds: session.rounds
26
+ .filter((r) => r.userResponse !== null)
27
+ .map((r) => ({
28
+ roundNumber: r.roundNumber,
29
+ question: r.question,
30
+ response: r.userResponse,
31
+ gestaltFocus: r.gestaltFocus,
32
+ })),
33
+ };
34
+ }
35
+ validateAndStore(session, externalSeed, force = false) {
36
+ // Validate ambiguity threshold
37
+ if (!force) {
38
+ const ambiguity = session.ambiguityScore?.overall ?? 1.0;
39
+ if (ambiguity > AMBIGUITY_THRESHOLD) {
40
+ return err(new AmbiguityThresholdError(ambiguity, AMBIGUITY_THRESHOLD));
41
+ }
42
+ }
43
+ if (session.status !== 'completed') {
44
+ return err(new SeedGenerationError('Interview session must be completed before generating a seed'));
45
+ }
46
+ try {
47
+ const seed = {
48
+ version: '1.0.0',
49
+ goal: externalSeed.goal,
50
+ constraints: externalSeed.constraints,
51
+ acceptanceCriteria: externalSeed.acceptanceCriteria,
52
+ ontologySchema: externalSeed.ontologySchema,
53
+ gestaltAnalysis: externalSeed.gestaltAnalysis,
54
+ metadata: {
55
+ seedId: randomUUID(),
56
+ interviewSessionId: session.sessionId,
57
+ ambiguityScore: session.ambiguityScore?.overall ?? 1.0,
58
+ generatedAt: new Date().toISOString(),
59
+ },
60
+ };
61
+ // Validate against schema
62
+ const validation = seedSchema.safeParse(seed);
63
+ if (!validation.success) {
64
+ return err(new SeedGenerationError(`Seed validation failed: ${validation.error.message}`));
65
+ }
66
+ this.eventStore.append('seed', seed.metadata.seedId, EventType.SEED_SPEC_GENERATED, {
67
+ sessionId: session.sessionId,
68
+ goal: seed.goal,
69
+ constraintCount: seed.constraints.length,
70
+ criteriaCount: seed.acceptanceCriteria.length,
71
+ source: 'passthrough',
72
+ });
73
+ return ok(seed);
74
+ }
75
+ catch (e) {
76
+ return err(new SeedGenerationError(`Failed to validate seed: ${e instanceof Error ? e.message : String(e)}`));
77
+ }
78
+ }
79
+ }
80
+ //# sourceMappingURL=passthrough-generator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"passthrough-generator.js","sourceRoot":"","sources":["../../../src/seed/passthrough-generator.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC,OAAO,EAAE,uBAAuB,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AACjF,OAAO,EAAE,mBAAmB,EAAE,MAAM,sBAAsB,CAAC;AAC3D,OAAO,EAAe,EAAE,EAAE,GAAG,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEzC,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAC/C,OAAO,EAAE,uBAAuB,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAkB7E,mEAAmE;AAEnE,MAAM,OAAO,wBAAwB;IACf;IAApB,YAAoB,UAAsB;QAAtB,eAAU,GAAV,UAAU,CAAY;IAAG,CAAC;IAE9C,gBAAgB,CAAC,OAAyB;QACxC,MAAM,cAAc,GAAG,OAAO,CAAC,MAAM;aAClC,MAAM,CAAC,CAAC,CAAC,EAAkD,EAAE,CAAC,CAAC,CAAC,YAAY,KAAK,IAAI,CAAC;aACtF,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACX,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,QAAQ,EAAE,CAAC,CAAC,YAAY;SACzB,CAAC,CAAC,CAAC;QAEN,MAAM,UAAU,GAAG,eAAe,CAAC,OAAO,CAAC,KAAK,EAAE,cAAc,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;QAEvF,OAAO;YACL,YAAY,EAAE,uBAAuB;YACrC,UAAU;YACV,SAAS,EAAE,OAAO,CAAC,MAAM;iBACtB,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,YAAY,KAAK,IAAI,CAAC;iBACtC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACX,WAAW,EAAE,CAAC,CAAC,WAAW;gBAC1B,QAAQ,EAAE,CAAC,CAAC,QAAQ;gBACpB,QAAQ,EAAE,CAAC,CAAC,YAAa;gBACzB,YAAY,EAAE,CAAC,CAAC,YAAY;aAC7B,CAAC,CAAC;SACN,CAAC;IACJ,CAAC;IAED,gBAAgB,CACd,OAAyB,EACzB,YAA0B,EAC1B,KAAK,GAAG,KAAK;QAEb,+BAA+B;QAC/B,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,SAAS,GAAG,OAAO,CAAC,cAAc,EAAE,OAAO,IAAI,GAAG,CAAC;YACzD,IAAI,SAAS,GAAG,mBAAmB,EAAE,CAAC;gBACpC,OAAO,GAAG,CAAC,IAAI,uBAAuB,CAAC,SAAS,EAAE,mBAAmB,CAAC,CAAC,CAAC;YAC1E,CAAC;QACH,CAAC;QAED,IAAI,OAAO,CAAC,MAAM,KAAK,WAAW,EAAE,CAAC;YACnC,OAAO,GAAG,CAAC,IAAI,mBAAmB,CAAC,8DAA8D,CAAC,CAAC,CAAC;QACtG,CAAC;QAED,IAAI,CAAC;YACH,MAAM,IAAI,GAAS;gBACjB,OAAO,EAAE,OAAO;gBAChB,IAAI,EAAE,YAAY,CAAC,IAAI;gBACvB,WAAW,EAAE,YAAY,CAAC,WAAW;gBACrC,kBAAkB,EAAE,YAAY,CAAC,kBAAkB;gBACnD,cAAc,EAAE,YAAY,CAAC,cAAwC;gBACrE,eAAe,EAAE,YAAY,CAAC,eAA0C;gBACxE,QAAQ,EAAE;oBACR,MAAM,EAAE,UAAU,EAAE;oBACpB,kBAAkB,EAAE,OAAO,CAAC,SAAS;oBACrC,cAAc,EAAE,OAAO,CAAC,cAAc,EAAE,OAAO,IAAI,GAAG;oBACtD,WAAW,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;iBACtC;aACF,CAAC;YAEF,0BAA0B;YAC1B,MAAM,UAAU,GAAG,UAAU,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC9C,IAAI,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;gBACxB,OAAO,GAAG,CACR,IAAI,mBAAmB,CACrB,2BAA2B,UAAU,CAAC,KAAK,CAAC,OAAO,EAAE,CACtD,CACF,CAAC;YACJ,CAAC;YAED,IAAI,CAAC,UAAU,CAAC,MAAM,CACpB,MAAM,EACN,IAAI,CAAC,QAAQ,CAAC,MAAM,EACpB,SAAS,CAAC,mBAAmB,EAC7B;gBACE,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,eAAe,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM;gBACxC,aAAa,EAAE,IAAI,CAAC,kBAAkB,CAAC,MAAM;gBAC7C,MAAM,EAAE,aAAa;aACtB,CACF,CAAC;YAEF,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC;QAClB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,GAAG,CACR,IAAI,mBAAmB,CACrB,4BAA4B,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CACzE,CACF,CAAC;QACJ,CAAC;IACH,CAAC;CACF"}
@@ -0,0 +1,145 @@
1
+ import { z } from 'zod';
2
+ import { GestaltPrinciple } from '../core/types.js';
3
+ export declare const seedSchema: z.ZodObject<{
4
+ version: z.ZodString;
5
+ goal: z.ZodString;
6
+ constraints: z.ZodArray<z.ZodString, "many">;
7
+ acceptanceCriteria: z.ZodArray<z.ZodString, "many">;
8
+ ontologySchema: z.ZodObject<{
9
+ entities: z.ZodArray<z.ZodObject<{
10
+ name: z.ZodString;
11
+ description: z.ZodString;
12
+ attributes: z.ZodArray<z.ZodString, "many">;
13
+ }, "strip", z.ZodTypeAny, {
14
+ name: string;
15
+ description: string;
16
+ attributes: string[];
17
+ }, {
18
+ name: string;
19
+ description: string;
20
+ attributes: string[];
21
+ }>, "many">;
22
+ relations: z.ZodArray<z.ZodObject<{
23
+ from: z.ZodString;
24
+ to: z.ZodString;
25
+ type: z.ZodString;
26
+ }, "strip", z.ZodTypeAny, {
27
+ type: string;
28
+ from: string;
29
+ to: string;
30
+ }, {
31
+ type: string;
32
+ from: string;
33
+ to: string;
34
+ }>, "many">;
35
+ }, "strip", z.ZodTypeAny, {
36
+ entities: {
37
+ name: string;
38
+ description: string;
39
+ attributes: string[];
40
+ }[];
41
+ relations: {
42
+ type: string;
43
+ from: string;
44
+ to: string;
45
+ }[];
46
+ }, {
47
+ entities: {
48
+ name: string;
49
+ description: string;
50
+ attributes: string[];
51
+ }[];
52
+ relations: {
53
+ type: string;
54
+ from: string;
55
+ to: string;
56
+ }[];
57
+ }>;
58
+ gestaltAnalysis: z.ZodArray<z.ZodObject<{
59
+ principle: z.ZodNativeEnum<typeof GestaltPrinciple>;
60
+ finding: z.ZodString;
61
+ confidence: z.ZodNumber;
62
+ }, "strip", z.ZodTypeAny, {
63
+ principle: GestaltPrinciple;
64
+ finding: string;
65
+ confidence: number;
66
+ }, {
67
+ principle: GestaltPrinciple;
68
+ finding: string;
69
+ confidence: number;
70
+ }>, "many">;
71
+ metadata: z.ZodObject<{
72
+ seedId: z.ZodString;
73
+ interviewSessionId: z.ZodString;
74
+ ambiguityScore: z.ZodNumber;
75
+ generatedAt: z.ZodString;
76
+ }, "strip", z.ZodTypeAny, {
77
+ seedId: string;
78
+ interviewSessionId: string;
79
+ ambiguityScore: number;
80
+ generatedAt: string;
81
+ }, {
82
+ seedId: string;
83
+ interviewSessionId: string;
84
+ ambiguityScore: number;
85
+ generatedAt: string;
86
+ }>;
87
+ }, "strip", z.ZodTypeAny, {
88
+ goal: string;
89
+ version: string;
90
+ constraints: string[];
91
+ acceptanceCriteria: string[];
92
+ ontologySchema: {
93
+ entities: {
94
+ name: string;
95
+ description: string;
96
+ attributes: string[];
97
+ }[];
98
+ relations: {
99
+ type: string;
100
+ from: string;
101
+ to: string;
102
+ }[];
103
+ };
104
+ gestaltAnalysis: {
105
+ principle: GestaltPrinciple;
106
+ finding: string;
107
+ confidence: number;
108
+ }[];
109
+ metadata: {
110
+ seedId: string;
111
+ interviewSessionId: string;
112
+ ambiguityScore: number;
113
+ generatedAt: string;
114
+ };
115
+ }, {
116
+ goal: string;
117
+ version: string;
118
+ constraints: string[];
119
+ acceptanceCriteria: string[];
120
+ ontologySchema: {
121
+ entities: {
122
+ name: string;
123
+ description: string;
124
+ attributes: string[];
125
+ }[];
126
+ relations: {
127
+ type: string;
128
+ from: string;
129
+ to: string;
130
+ }[];
131
+ };
132
+ gestaltAnalysis: {
133
+ principle: GestaltPrinciple;
134
+ finding: string;
135
+ confidence: number;
136
+ }[];
137
+ metadata: {
138
+ seedId: string;
139
+ interviewSessionId: string;
140
+ ambiguityScore: number;
141
+ generatedAt: string;
142
+ };
143
+ }>;
144
+ export type ValidatedSeed = z.infer<typeof seedSchema>;
145
+ //# sourceMappingURL=schema.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.d.ts","sourceRoot":"","sources":["../../../src/seed/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAgCpD,eAAO,MAAM,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAQrB,CAAC;AAEH,MAAM,MAAM,aAAa,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,UAAU,CAAC,CAAC"}
@@ -0,0 +1,37 @@
1
+ import { z } from 'zod';
2
+ import { GestaltPrinciple } from '../core/types.js';
3
+ const ontologyEntitySchema = z.object({
4
+ name: z.string().min(1),
5
+ description: z.string(),
6
+ attributes: z.array(z.string()),
7
+ });
8
+ const ontologyRelationSchema = z.object({
9
+ from: z.string().min(1),
10
+ to: z.string().min(1),
11
+ type: z.string().min(1),
12
+ });
13
+ const ontologySchemaSchema = z.object({
14
+ entities: z.array(ontologyEntitySchema),
15
+ relations: z.array(ontologyRelationSchema),
16
+ });
17
+ const gestaltAnalysisSchema = z.object({
18
+ principle: z.nativeEnum(GestaltPrinciple),
19
+ finding: z.string(),
20
+ confidence: z.number().min(0).max(1),
21
+ });
22
+ const seedMetadataSchema = z.object({
23
+ seedId: z.string(),
24
+ interviewSessionId: z.string(),
25
+ ambiguityScore: z.number().min(0).max(1),
26
+ generatedAt: z.string(),
27
+ });
28
+ export const seedSchema = z.object({
29
+ version: z.string(),
30
+ goal: z.string().min(1),
31
+ constraints: z.array(z.string()),
32
+ acceptanceCriteria: z.array(z.string()),
33
+ ontologySchema: ontologySchemaSchema,
34
+ gestaltAnalysis: z.array(gestaltAnalysisSchema),
35
+ metadata: seedMetadataSchema,
36
+ });
37
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../../src/seed/schema.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AAEpD,MAAM,oBAAoB,GAAG,CAAC,CAAC,MAAM,CAAC;IACpC,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACvB,WAAW,EAAE,CAAC,CAAC,MAAM,EAAE;IACvB,UAAU,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;CAChC,CAAC,CAAC;AAEH,MAAM,sBAAsB,GAAG,CAAC,CAAC,MAAM,CAAC;IACtC,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACvB,EAAE,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACrB,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;CACxB,CAAC,CAAC;AAEH,MAAM,oBAAoB,GAAG,CAAC,CAAC,MAAM,CAAC;IACpC,QAAQ,EAAE,CAAC,CAAC,KAAK,CAAC,oBAAoB,CAAC;IACvC,SAAS,EAAE,CAAC,CAAC,KAAK,CAAC,sBAAsB,CAAC;CAC3C,CAAC,CAAC;AAEH,MAAM,qBAAqB,GAAG,CAAC,CAAC,MAAM,CAAC;IACrC,SAAS,EAAE,CAAC,CAAC,UAAU,CAAC,gBAAgB,CAAC;IACzC,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE;IACnB,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;CACrC,CAAC,CAAC;AAEH,MAAM,kBAAkB,GAAG,CAAC,CAAC,MAAM,CAAC;IAClC,MAAM,EAAE,CAAC,CAAC,MAAM,EAAE;IAClB,kBAAkB,EAAE,CAAC,CAAC,MAAM,EAAE;IAC9B,cAAc,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IACxC,WAAW,EAAE,CAAC,CAAC,MAAM,EAAE;CACxB,CAAC,CAAC;AAEH,MAAM,CAAC,MAAM,UAAU,GAAG,CAAC,CAAC,MAAM,CAAC;IACjC,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE;IACnB,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACvB,WAAW,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;IAChC,kBAAkB,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;IACvC,cAAc,EAAE,oBAAoB;IACpC,eAAe,EAAE,CAAC,CAAC,KAAK,CAAC,qBAAqB,CAAC;IAC/C,QAAQ,EAAE,kBAAkB;CAC7B,CAAC,CAAC"}
@@ -0,0 +1,14 @@
1
+ import type { SkillDefinition } from './types.js';
2
+ export interface SkillContext {
3
+ inputs: Record<string, unknown>;
4
+ cwd: string;
5
+ }
6
+ export interface SkillResult {
7
+ skillName: string;
8
+ body: string;
9
+ inputs: Record<string, unknown>;
10
+ }
11
+ export declare class SkillExecutor {
12
+ execute(skill: SkillDefinition, context: SkillContext): SkillResult;
13
+ }
14
+ //# sourceMappingURL=executor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.d.ts","sourceRoot":"","sources":["../../../src/skills/executor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAGlD,MAAM,WAAW,YAAY;IAC3B,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAChC,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,WAAW,WAAW;IAC1B,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACjC;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,KAAK,EAAE,eAAe,EAAE,OAAO,EAAE,YAAY,GAAG,WAAW;CAcpE"}
@@ -0,0 +1,17 @@
1
+ import { SkillParseError } from '../core/errors.js';
2
+ export class SkillExecutor {
3
+ execute(skill, context) {
4
+ // Validate required inputs
5
+ for (const [name, def] of Object.entries(skill.frontmatter.inputs)) {
6
+ if (def.required && !(name in context.inputs)) {
7
+ throw new SkillParseError(`Missing required input: ${name}`);
8
+ }
9
+ }
10
+ return {
11
+ skillName: skill.frontmatter.name,
12
+ body: skill.body,
13
+ inputs: context.inputs,
14
+ };
15
+ }
16
+ }
17
+ //# sourceMappingURL=executor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.js","sourceRoot":"","sources":["../../../src/skills/executor.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAapD,MAAM,OAAO,aAAa;IACxB,OAAO,CAAC,KAAsB,EAAE,OAAqB;QACnD,2BAA2B;QAC3B,KAAK,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC;YACnE,IAAI,GAAG,CAAC,QAAQ,IAAI,CAAC,CAAC,IAAI,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC9C,MAAM,IAAI,eAAe,CAAC,2BAA2B,IAAI,EAAE,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;QAED,OAAO;YACL,SAAS,EAAE,KAAK,CAAC,WAAW,CAAC,IAAI;YACjC,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,MAAM,EAAE,OAAO,CAAC,MAAM;SACvB,CAAC;IACJ,CAAC;CACF"}
@@ -0,0 +1,3 @@
1
+ import type { SkillDefinition } from './types.js';
2
+ export declare function parseSkillMd(content: string, filePath: string): SkillDefinition;
3
+ //# sourceMappingURL=parser.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parser.d.ts","sourceRoot":"","sources":["../../../src/skills/parser.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,eAAe,EAAoB,MAAM,YAAY,CAAC;AAiBpE,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,eAAe,CAuB/E"}
@@ -0,0 +1,37 @@
1
+ import matter from 'gray-matter';
2
+ import { z } from 'zod';
3
+ import { SkillParseError } from '../core/errors.js';
4
+ const skillInputSchema = z.object({
5
+ type: z.string(),
6
+ required: z.boolean().default(false),
7
+ description: z.string().default(''),
8
+ });
9
+ const skillFrontmatterSchema = z.object({
10
+ name: z.string().min(1),
11
+ version: z.string().default('1.0.0'),
12
+ description: z.string().default(''),
13
+ triggers: z.array(z.string()).default([]),
14
+ inputs: z.record(z.string(), skillInputSchema).default({}),
15
+ outputs: z.array(z.string()).default([]),
16
+ });
17
+ export function parseSkillMd(content, filePath) {
18
+ try {
19
+ const { data, content: body } = matter(content);
20
+ const result = skillFrontmatterSchema.safeParse(data);
21
+ if (!result.success) {
22
+ const issues = result.error.issues.map((i) => `${i.path.join('.')}: ${i.message}`);
23
+ throw new SkillParseError(`Invalid SKILL.md frontmatter in ${filePath}:\n${issues.join('\n')}`);
24
+ }
25
+ return {
26
+ frontmatter: result.data,
27
+ body: body.trim(),
28
+ filePath,
29
+ };
30
+ }
31
+ catch (e) {
32
+ if (e instanceof SkillParseError)
33
+ throw e;
34
+ throw new SkillParseError(`Failed to parse SKILL.md at ${filePath}: ${e instanceof Error ? e.message : String(e)}`);
35
+ }
36
+ }
37
+ //# sourceMappingURL=parser.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parser.js","sourceRoot":"","sources":["../../../src/skills/parser.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,aAAa,CAAC;AACjC,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGpD,MAAM,gBAAgB,GAAG,CAAC,CAAC,MAAM,CAAC;IAChC,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE;IAChB,QAAQ,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC;IACpC,WAAW,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;CACpC,CAAC,CAAC;AAEH,MAAM,sBAAsB,GAAG,CAAC,CAAC,MAAM,CAAC;IACtC,IAAI,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACvB,OAAO,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC;IACpC,WAAW,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC;IACnC,QAAQ,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC;IACzC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,EAAE,EAAE,gBAAgB,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC;IAC1D,OAAO,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC;CACzC,CAAC,CAAC;AAEH,MAAM,UAAU,YAAY,CAAC,OAAe,EAAE,QAAgB;IAC5D,IAAI,CAAC;QACH,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;QAChD,MAAM,MAAM,GAAG,sBAAsB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAEtD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;YACnF,MAAM,IAAI,eAAe,CACvB,mCAAmC,QAAQ,MAAM,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACrE,CAAC;QACJ,CAAC;QAED,OAAO;YACL,WAAW,EAAE,MAAM,CAAC,IAAwB;YAC5C,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE;YACjB,QAAQ;SACT,CAAC;IACJ,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,IAAI,CAAC,YAAY,eAAe;YAAE,MAAM,CAAC,CAAC;QAC1C,MAAM,IAAI,eAAe,CACvB,+BAA+B,QAAQ,KAAK,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CACzF,CAAC;IACJ,CAAC;AACH,CAAC"}
@@ -0,0 +1,8 @@
1
+ import { BaseRegistry } from '../registry/base-registry.js';
2
+ import type { SkillDefinition } from './types.js';
3
+ export declare class SkillRegistry extends BaseRegistry<SkillDefinition> {
4
+ constructor(skillsDir?: string);
5
+ protected parse(content: string, filePath: string): SkillDefinition;
6
+ protected getName(item: SkillDefinition): string;
7
+ }
8
+ //# sourceMappingURL=registry.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"registry.d.ts","sourceRoot":"","sources":["../../../src/skills/registry.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAE5D,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAGlD,qBAAa,aAAc,SAAQ,YAAY,CAAC,eAAe,CAAC;gBAClD,SAAS,CAAC,EAAE,MAAM;IAQ9B,SAAS,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,eAAe;IAInE,SAAS,CAAC,OAAO,CAAC,IAAI,EAAE,eAAe,GAAG,MAAM;CAGjD"}
@@ -0,0 +1,19 @@
1
+ import { BaseRegistry } from '../registry/base-registry.js';
2
+ import { parseSkillMd } from './parser.js';
3
+ import { SKILLS_DIR } from '../core/constants.js';
4
+ export class SkillRegistry extends BaseRegistry {
5
+ constructor(skillsDir) {
6
+ super({
7
+ dir: skillsDir ?? SKILLS_DIR,
8
+ filename: 'SKILL.md',
9
+ label: 'skill',
10
+ });
11
+ }
12
+ parse(content, filePath) {
13
+ return parseSkillMd(content, filePath);
14
+ }
15
+ getName(item) {
16
+ return item.frontmatter.name;
17
+ }
18
+ }
19
+ //# sourceMappingURL=registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"registry.js","sourceRoot":"","sources":["../../../src/skills/registry.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAElD,MAAM,OAAO,aAAc,SAAQ,YAA6B;IAC9D,YAAY,SAAkB;QAC5B,KAAK,CAAC;YACJ,GAAG,EAAE,SAAS,IAAI,UAAU;YAC5B,QAAQ,EAAE,UAAU;YACpB,KAAK,EAAE,OAAO;SACf,CAAC,CAAC;IACL,CAAC;IAES,KAAK,CAAC,OAAe,EAAE,QAAgB;QAC/C,OAAO,YAAY,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;IACzC,CAAC;IAES,OAAO,CAAC,IAAqB;QACrC,OAAO,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;IAC/B,CAAC;CACF"}
@@ -0,0 +1,19 @@
1
+ export interface SkillFrontmatter {
2
+ name: string;
3
+ version: string;
4
+ description: string;
5
+ triggers: string[];
6
+ inputs: Record<string, SkillInputDef>;
7
+ outputs: string[];
8
+ }
9
+ export interface SkillInputDef {
10
+ type: string;
11
+ required: boolean;
12
+ description: string;
13
+ }
14
+ export interface SkillDefinition {
15
+ frontmatter: SkillFrontmatter;
16
+ body: string;
17
+ filePath: string;
18
+ }
19
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/skills/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;IACtC,OAAO,EAAE,MAAM,EAAE,CAAC;CACnB;AAED,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,OAAO,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,eAAe;IAC9B,WAAW,EAAE,gBAAgB,CAAC;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;CAClB"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../../src/skills/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,15 @@
1
+ import type { InterviewSession, GestaltAnalysis, OntologySchema } from '../core/types.js';
2
+ import type { LLMAdapter } from '../llm/types.js';
3
+ export interface ExtractedData {
4
+ goal: string;
5
+ constraints: string[];
6
+ acceptanceCriteria: string[];
7
+ ontologySchema: OntologySchema;
8
+ gestaltAnalysis: GestaltAnalysis[];
9
+ }
10
+ export declare class SpecExtractor {
11
+ private llm;
12
+ constructor(llm: LLMAdapter);
13
+ extract(session: InterviewSession): Promise<ExtractedData>;
14
+ }
15
+ //# sourceMappingURL=extractor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"extractor.d.ts","sourceRoot":"","sources":["../../../src/spec/extractor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAG1F,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAElD,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,kBAAkB,EAAE,MAAM,EAAE,CAAC;IAC7B,cAAc,EAAE,cAAc,CAAC;IAC/B,eAAe,EAAE,eAAe,EAAE,CAAC;CACpC;AAED,qBAAa,aAAa;IACZ,OAAO,CAAC,GAAG;gBAAH,GAAG,EAAE,UAAU;IAE7B,OAAO,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,aAAa,CAAC;CAsBjE"}
@@ -0,0 +1,88 @@
1
+ import { buildSpecPrompt, INTERVIEW_SYSTEM_PROMPT } from '../llm/prompts.js';
2
+ import { SpecGenerationError } from '../core/errors.js';
3
+ export class SpecExtractor {
4
+ llm;
5
+ constructor(llm) {
6
+ this.llm = llm;
7
+ }
8
+ async extract(session) {
9
+ const completedRounds = session.rounds
10
+ .filter((r) => r.userResponse)
11
+ .map((r) => ({
12
+ question: r.question,
13
+ response: r.userResponse,
14
+ }));
15
+ if (completedRounds.length === 0) {
16
+ throw new SpecGenerationError('No completed interview rounds to extract from');
17
+ }
18
+ const prompt = buildSpecPrompt(session.topic, completedRounds, session.projectType);
19
+ const response = await this.llm.chat({
20
+ system: INTERVIEW_SYSTEM_PROMPT,
21
+ messages: [{ role: 'user', content: prompt }],
22
+ temperature: 0.3,
23
+ });
24
+ return parseSpecResponse(response.content);
25
+ }
26
+ }
27
+ function parseSpecResponse(content) {
28
+ const jsonMatch = content.match(/\{[\s\S]*\}/);
29
+ if (!jsonMatch) {
30
+ throw new SpecGenerationError('Failed to parse LLM response as JSON');
31
+ }
32
+ try {
33
+ const parsed = JSON.parse(jsonMatch[0]);
34
+ return {
35
+ goal: asString(parsed['goal'], 'Undefined goal'),
36
+ constraints: asStringArray(parsed['constraints']),
37
+ acceptanceCriteria: asStringArray(parsed['acceptanceCriteria']),
38
+ ontologySchema: parseOntology(parsed['ontologySchema']),
39
+ gestaltAnalysis: parseGestaltAnalysis(parsed['gestaltAnalysis']),
40
+ };
41
+ }
42
+ catch (e) {
43
+ if (e instanceof SpecGenerationError)
44
+ throw e;
45
+ throw new SpecGenerationError(`Failed to parse spec data: ${e instanceof Error ? e.message : String(e)}`);
46
+ }
47
+ }
48
+ function asString(value, fallback) {
49
+ return typeof value === 'string' && value.length > 0 ? value : fallback;
50
+ }
51
+ function asStringArray(value) {
52
+ if (!Array.isArray(value))
53
+ return [];
54
+ return value.filter((v) => typeof v === 'string');
55
+ }
56
+ function parseOntology(value) {
57
+ if (!value || typeof value !== 'object') {
58
+ return { entities: [], relations: [] };
59
+ }
60
+ const obj = value;
61
+ const entities = Array.isArray(obj['entities'])
62
+ ? obj['entities'].map((e) => ({
63
+ name: asString(e['name'], 'Unknown'),
64
+ description: asString(e['description'], ''),
65
+ attributes: asStringArray(e['attributes']),
66
+ }))
67
+ : [];
68
+ const relations = Array.isArray(obj['relations'])
69
+ ? obj['relations'].map((r) => ({
70
+ from: asString(r['from'], ''),
71
+ to: asString(r['to'], ''),
72
+ type: asString(r['type'], ''),
73
+ }))
74
+ : [];
75
+ return { entities, relations };
76
+ }
77
+ function parseGestaltAnalysis(value) {
78
+ if (!Array.isArray(value))
79
+ return [];
80
+ return value
81
+ .filter((v) => typeof v === 'object' && v !== null)
82
+ .map((v) => ({
83
+ principle: asString(v['principle'], 'closure'),
84
+ finding: asString(v['finding'], ''),
85
+ confidence: typeof v['confidence'] === 'number' ? v['confidence'] : 0.5,
86
+ }));
87
+ }
88
+ //# sourceMappingURL=extractor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"extractor.js","sourceRoot":"","sources":["../../../src/spec/extractor.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,eAAe,EAAE,uBAAuB,EAAE,MAAM,mBAAmB,CAAC;AAC7E,OAAO,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAWxD,MAAM,OAAO,aAAa;IACJ;IAApB,YAAoB,GAAe;QAAf,QAAG,GAAH,GAAG,CAAY;IAAG,CAAC;IAEvC,KAAK,CAAC,OAAO,CAAC,OAAyB;QACrC,MAAM,eAAe,GAAG,OAAO,CAAC,MAAM;aACnC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC;aAC7B,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACX,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,QAAQ,EAAE,CAAC,CAAC,YAAa;SAC1B,CAAC,CAAC,CAAC;QAEN,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,mBAAmB,CAAC,+CAA+C,CAAC,CAAC;QACjF,CAAC;QAED,MAAM,MAAM,GAAG,eAAe,CAAC,OAAO,CAAC,KAAK,EAAE,eAAe,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;QAEpF,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;YACnC,MAAM,EAAE,uBAAuB;YAC/B,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,CAAC;YAC7C,WAAW,EAAE,GAAG;SACjB,CAAC,CAAC;QAEH,OAAO,iBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;IAC7C,CAAC;CACF;AAED,SAAS,iBAAiB,CAAC,OAAe;IACxC,MAAM,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IAC/C,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,MAAM,IAAI,mBAAmB,CAAC,sCAAsC,CAAC,CAAC;IACxE,CAAC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,CAA4B,CAAC;QAEnE,OAAO;YACL,IAAI,EAAE,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,gBAAgB,CAAC;YAChD,WAAW,EAAE,aAAa,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;YACjD,kBAAkB,EAAE,aAAa,CAAC,MAAM,CAAC,oBAAoB,CAAC,CAAC;YAC/D,cAAc,EAAE,aAAa,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;YACvD,eAAe,EAAE,oBAAoB,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAC;SACjE,CAAC;IACJ,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,IAAI,CAAC,YAAY,mBAAmB;YAAE,MAAM,CAAC,CAAC;QAC9C,MAAM,IAAI,mBAAmB,CAC3B,8BAA8B,CAAC,YAAY,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAC3E,CAAC;IACJ,CAAC;AACH,CAAC;AAED,SAAS,QAAQ,CAAC,KAAc,EAAE,QAAgB;IAChD,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC;AAC1E,CAAC;AAED,SAAS,aAAa,CAAC,KAAc;IACnC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IACrC,OAAO,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAe,EAAE,CAAC,OAAO,CAAC,KAAK,QAAQ,CAAC,CAAC;AACjE,CAAC;AAED,SAAS,aAAa,CAAC,KAAc;IACnC,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QACxC,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC;IACzC,CAAC;IAED,MAAM,GAAG,GAAG,KAAgC,CAAC;IAC7C,MAAM,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC7C,CAAC,CAAE,GAAG,CAAC,UAAU,CAAoC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC9D,IAAI,EAAE,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,SAAS,CAAC;YACpC,WAAW,EAAE,QAAQ,CAAC,CAAC,CAAC,aAAa,CAAC,EAAE,EAAE,CAAC;YAC3C,UAAU,EAAE,aAAa,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC;SAC3C,CAAC,CAAC;QACL,CAAC,CAAC,EAAE,CAAC;IAEP,MAAM,SAAS,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;QAC/C,CAAC,CAAE,GAAG,CAAC,WAAW,CAAoC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC/D,IAAI,EAAE,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;YAC7B,EAAE,EAAE,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,CAAC;YACzB,IAAI,EAAE,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;SAC9B,CAAC,CAAC;QACL,CAAC,CAAC,EAAE,CAAC;IAEP,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,CAAC;AACjC,CAAC;AAED,SAAS,oBAAoB,CAAC,KAAc;IAC1C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,EAAE,CAAC;IACrC,OAAO,KAAK;SACT,MAAM,CAAC,CAAC,CAAC,EAAgC,EAAE,CAAC,OAAO,CAAC,KAAK,QAAQ,IAAI,CAAC,KAAK,IAAI,CAAC;SAChF,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACX,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,EAAE,SAAS,CAAiC;QAC9E,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,EAAE,CAAC;QACnC,UAAU,EAAE,OAAO,CAAC,CAAC,YAAY,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,GAAG;KACxE,CAAC,CAAC,CAAC;AACR,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { InterviewSession, Spec } from '../core/types.js';
2
+ import { AmbiguityThresholdError, SpecGenerationError } from '../core/errors.js';
3
+ import { type Result } from '../core/result.js';
4
+ import type { LLMAdapter } from '../llm/types.js';
5
+ import { EventStore } from '../events/store.js';
6
+ export declare class SpecGenerator {
7
+ private eventStore;
8
+ private extractor;
9
+ constructor(llm: LLMAdapter, eventStore: EventStore);
10
+ generate(session: InterviewSession, force?: boolean): Promise<Result<Spec, SpecGenerationError | AmbiguityThresholdError>>;
11
+ }
12
+ //# sourceMappingURL=generator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generator.d.ts","sourceRoot":"","sources":["../../../src/spec/generator.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,gBAAgB,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAC/D,OAAO,EAAE,uBAAuB,EAAE,mBAAmB,EAAE,MAAM,mBAAmB,CAAC;AAEjF,OAAO,EAAE,KAAK,MAAM,EAAW,MAAM,mBAAmB,CAAC;AAGzD,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAClD,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC;AAGhD,qBAAa,aAAa;IAKtB,OAAO,CAAC,UAAU;IAJpB,OAAO,CAAC,SAAS,CAAgB;gBAG/B,GAAG,EAAE,UAAU,EACP,UAAU,EAAE,UAAU;IAK1B,QAAQ,CACZ,OAAO,EAAE,gBAAgB,EACzB,KAAK,UAAQ,GACZ,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,mBAAmB,GAAG,uBAAuB,CAAC,CAAC;CAmExE"}
@@ -0,0 +1,66 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { AmbiguityThresholdError, SpecGenerationError } from '../core/errors.js';
3
+ import { AMBIGUITY_THRESHOLD, MAX_SPEC_RETRIES } from '../core/constants.js';
4
+ import { ok, err } from '../core/result.js';
5
+ import { specSchema } from './schema.js';
6
+ import { SpecExtractor } from './extractor.js';
7
+ import { EventType } from '../events/types.js';
8
+ export class SpecGenerator {
9
+ eventStore;
10
+ extractor;
11
+ constructor(llm, eventStore) {
12
+ this.eventStore = eventStore;
13
+ this.extractor = new SpecExtractor(llm);
14
+ }
15
+ async generate(session, force = false) {
16
+ // Validate ambiguity threshold
17
+ if (!force) {
18
+ const ambiguity = session.ambiguityScore?.overall ?? 1.0;
19
+ if (ambiguity > AMBIGUITY_THRESHOLD) {
20
+ return err(new AmbiguityThresholdError(ambiguity, AMBIGUITY_THRESHOLD));
21
+ }
22
+ }
23
+ if (session.status !== 'completed') {
24
+ return err(new SpecGenerationError('Interview session must be completed before generating a spec'));
25
+ }
26
+ // Retry up to MAX_SPEC_RETRIES
27
+ let lastError = null;
28
+ for (let attempt = 0; attempt < MAX_SPEC_RETRIES; attempt++) {
29
+ try {
30
+ const extracted = await this.extractor.extract(session);
31
+ const spec = {
32
+ version: '1.0.0',
33
+ goal: extracted.goal,
34
+ constraints: extracted.constraints,
35
+ acceptanceCriteria: extracted.acceptanceCriteria,
36
+ ontologySchema: extracted.ontologySchema,
37
+ gestaltAnalysis: extracted.gestaltAnalysis,
38
+ metadata: {
39
+ specId: randomUUID(),
40
+ interviewSessionId: session.sessionId,
41
+ ambiguityScore: session.ambiguityScore?.overall ?? 1.0,
42
+ generatedAt: new Date().toISOString(),
43
+ },
44
+ };
45
+ // Validate against schema
46
+ const validation = specSchema.safeParse(spec);
47
+ if (!validation.success) {
48
+ lastError = new SpecGenerationError(`Spec validation failed: ${validation.error.message}`);
49
+ continue;
50
+ }
51
+ this.eventStore.append('spec', spec.metadata.specId, EventType.SPEC_GENERATED, {
52
+ sessionId: session.sessionId,
53
+ goal: spec.goal,
54
+ constraintCount: spec.constraints.length,
55
+ criteriaCount: spec.acceptanceCriteria.length,
56
+ });
57
+ return ok(spec);
58
+ }
59
+ catch (e) {
60
+ lastError = e instanceof Error ? e : new Error(String(e));
61
+ }
62
+ }
63
+ return err(new SpecGenerationError(`Failed after ${MAX_SPEC_RETRIES} attempts: ${lastError?.message ?? 'unknown error'}`));
64
+ }
65
+ }
66
+ //# sourceMappingURL=generator.js.map