@contractspec/bundle.workspace 1.44.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 (425) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +79 -0
  3. package/dist/_virtual/rolldown_runtime.js +36 -0
  4. package/dist/adapters/ai.d.ts +12 -0
  5. package/dist/adapters/ai.d.ts.map +1 -0
  6. package/dist/adapters/ai.js +83 -0
  7. package/dist/adapters/ai.js.map +1 -0
  8. package/dist/adapters/factory.d.ts +29 -0
  9. package/dist/adapters/factory.d.ts.map +1 -0
  10. package/dist/adapters/factory.js +37 -0
  11. package/dist/adapters/factory.js.map +1 -0
  12. package/dist/adapters/fs.d.ts +11 -0
  13. package/dist/adapters/fs.d.ts.map +1 -0
  14. package/dist/adapters/fs.js +131 -0
  15. package/dist/adapters/fs.js.map +1 -0
  16. package/dist/adapters/git.d.ts +11 -0
  17. package/dist/adapters/git.d.ts.map +1 -0
  18. package/dist/adapters/git.js +55 -0
  19. package/dist/adapters/git.js.map +1 -0
  20. package/dist/adapters/index.d.ts +7 -0
  21. package/dist/adapters/index.js +7 -0
  22. package/dist/adapters/logger.d.ts +18 -0
  23. package/dist/adapters/logger.d.ts.map +1 -0
  24. package/dist/adapters/logger.js +81 -0
  25. package/dist/adapters/logger.js.map +1 -0
  26. package/dist/adapters/watcher.d.ts +11 -0
  27. package/dist/adapters/watcher.d.ts.map +1 -0
  28. package/dist/adapters/watcher.js +74 -0
  29. package/dist/adapters/watcher.js.map +1 -0
  30. package/dist/adapters/workspace.d.ts +148 -0
  31. package/dist/adapters/workspace.d.ts.map +1 -0
  32. package/dist/adapters/workspace.js +275 -0
  33. package/dist/adapters/workspace.js.map +1 -0
  34. package/dist/ai/agents/claude-code-agent.d.ts +22 -0
  35. package/dist/ai/agents/claude-code-agent.d.ts.map +1 -0
  36. package/dist/ai/agents/claude-code-agent.js +182 -0
  37. package/dist/ai/agents/claude-code-agent.js.map +1 -0
  38. package/dist/ai/agents/cursor-agent.d.ts +68 -0
  39. package/dist/ai/agents/cursor-agent.d.ts.map +1 -0
  40. package/dist/ai/agents/cursor-agent.js +436 -0
  41. package/dist/ai/agents/cursor-agent.js.map +1 -0
  42. package/dist/ai/agents/index.js +5 -0
  43. package/dist/ai/agents/openai-codex-agent.d.ts +22 -0
  44. package/dist/ai/agents/openai-codex-agent.d.ts.map +1 -0
  45. package/dist/ai/agents/openai-codex-agent.js +167 -0
  46. package/dist/ai/agents/openai-codex-agent.js.map +1 -0
  47. package/dist/ai/agents/orchestrator.d.ts +50 -0
  48. package/dist/ai/agents/orchestrator.d.ts.map +1 -0
  49. package/dist/ai/agents/orchestrator.js +143 -0
  50. package/dist/ai/agents/orchestrator.js.map +1 -0
  51. package/dist/ai/agents/simple-agent.d.ts +17 -0
  52. package/dist/ai/agents/simple-agent.d.ts.map +1 -0
  53. package/dist/ai/agents/simple-agent.js +92 -0
  54. package/dist/ai/agents/simple-agent.js.map +1 -0
  55. package/dist/ai/agents/types.d.ts +36 -0
  56. package/dist/ai/agents/types.d.ts.map +1 -0
  57. package/dist/ai/client.d.ts +83 -0
  58. package/dist/ai/client.d.ts.map +1 -0
  59. package/dist/ai/client.js +163 -0
  60. package/dist/ai/client.js.map +1 -0
  61. package/dist/ai/index.d.ts +17 -0
  62. package/dist/ai/index.d.ts.map +1 -0
  63. package/dist/ai/index.js +28 -0
  64. package/dist/ai/index.js.map +1 -0
  65. package/dist/ai/prompts/code-generation.d.ts +26 -0
  66. package/dist/ai/prompts/code-generation.d.ts.map +1 -0
  67. package/dist/ai/prompts/code-generation.js +143 -0
  68. package/dist/ai/prompts/code-generation.js.map +1 -0
  69. package/dist/ai/prompts/index.d.ts +10 -0
  70. package/dist/ai/prompts/index.d.ts.map +1 -0
  71. package/dist/ai/prompts/index.js +13 -0
  72. package/dist/ai/prompts/index.js.map +1 -0
  73. package/dist/ai/prompts/spec-creation.d.ts +29 -0
  74. package/dist/ai/prompts/spec-creation.d.ts.map +1 -0
  75. package/dist/ai/prompts/spec-creation.js +111 -0
  76. package/dist/ai/prompts/spec-creation.js.map +1 -0
  77. package/dist/ai/providers.d.ts +29 -0
  78. package/dist/ai/providers.d.ts.map +1 -0
  79. package/dist/ai/providers.js +39 -0
  80. package/dist/ai/providers.js.map +1 -0
  81. package/dist/formatters/index.d.ts +11 -0
  82. package/dist/formatters/index.d.ts.map +1 -0
  83. package/dist/formatters/index.js +19 -0
  84. package/dist/formatters/index.js.map +1 -0
  85. package/dist/formatters/json.d.ts +89 -0
  86. package/dist/formatters/json.d.ts.map +1 -0
  87. package/dist/formatters/json.js +72 -0
  88. package/dist/formatters/json.js.map +1 -0
  89. package/dist/formatters/sarif.d.ts +101 -0
  90. package/dist/formatters/sarif.d.ts.map +1 -0
  91. package/dist/formatters/sarif.js +163 -0
  92. package/dist/formatters/sarif.js.map +1 -0
  93. package/dist/formatters/text.d.ts +35 -0
  94. package/dist/formatters/text.d.ts.map +1 -0
  95. package/dist/formatters/text.js +209 -0
  96. package/dist/formatters/text.js.map +1 -0
  97. package/dist/index.d.ts +82 -0
  98. package/dist/index.d.ts.map +1 -0
  99. package/dist/index.js +204 -0
  100. package/dist/index.js.map +1 -0
  101. package/dist/ports/ai.d.ts +59 -0
  102. package/dist/ports/ai.d.ts.map +1 -0
  103. package/dist/ports/fs.d.ts +81 -0
  104. package/dist/ports/fs.d.ts.map +1 -0
  105. package/dist/ports/git.d.ts +33 -0
  106. package/dist/ports/git.d.ts.map +1 -0
  107. package/dist/ports/index.d.ts +5 -0
  108. package/dist/ports/logger.d.ts +88 -0
  109. package/dist/ports/logger.d.ts.map +1 -0
  110. package/dist/ports/watcher.d.ts +52 -0
  111. package/dist/ports/watcher.d.ts.map +1 -0
  112. package/dist/services/agent-guide/adapters/claude-code.d.ts +35 -0
  113. package/dist/services/agent-guide/adapters/claude-code.d.ts.map +1 -0
  114. package/dist/services/agent-guide/adapters/claude-code.js +144 -0
  115. package/dist/services/agent-guide/adapters/claude-code.js.map +1 -0
  116. package/dist/services/agent-guide/adapters/cursor-cli.d.ts +39 -0
  117. package/dist/services/agent-guide/adapters/cursor-cli.d.ts.map +1 -0
  118. package/dist/services/agent-guide/adapters/cursor-cli.js +135 -0
  119. package/dist/services/agent-guide/adapters/cursor-cli.js.map +1 -0
  120. package/dist/services/agent-guide/adapters/generic-mcp.d.ts +53 -0
  121. package/dist/services/agent-guide/adapters/generic-mcp.d.ts.map +1 -0
  122. package/dist/services/agent-guide/adapters/generic-mcp.js +159 -0
  123. package/dist/services/agent-guide/adapters/generic-mcp.js.map +1 -0
  124. package/dist/services/agent-guide/adapters/index.d.ts +23 -0
  125. package/dist/services/agent-guide/adapters/index.d.ts.map +1 -0
  126. package/dist/services/agent-guide/adapters/index.js +31 -0
  127. package/dist/services/agent-guide/adapters/index.js.map +1 -0
  128. package/dist/services/agent-guide/agent-guide-service.d.ts +56 -0
  129. package/dist/services/agent-guide/agent-guide-service.d.ts.map +1 -0
  130. package/dist/services/agent-guide/agent-guide-service.js +147 -0
  131. package/dist/services/agent-guide/agent-guide-service.js.map +1 -0
  132. package/dist/services/agent-guide/index.d.ts +6 -0
  133. package/dist/services/agent-guide/index.js +5 -0
  134. package/dist/services/agent-guide/types.d.ts +58 -0
  135. package/dist/services/agent-guide/types.d.ts.map +1 -0
  136. package/dist/services/build.d.ts +59 -0
  137. package/dist/services/build.d.ts.map +1 -0
  138. package/dist/services/build.js +140 -0
  139. package/dist/services/build.js.map +1 -0
  140. package/dist/services/ci-check/ci-check-service.d.ts +16 -0
  141. package/dist/services/ci-check/ci-check-service.d.ts.map +1 -0
  142. package/dist/services/ci-check/ci-check-service.js +392 -0
  143. package/dist/services/ci-check/ci-check-service.js.map +1 -0
  144. package/dist/services/ci-check/index.d.ts +2 -0
  145. package/dist/services/ci-check/index.js +2 -0
  146. package/dist/services/ci-check/types.d.ts +143 -0
  147. package/dist/services/ci-check/types.d.ts.map +1 -0
  148. package/dist/services/ci-check/types.js +29 -0
  149. package/dist/services/ci-check/types.js.map +1 -0
  150. package/dist/services/clean.d.ts +41 -0
  151. package/dist/services/clean.d.ts.map +1 -0
  152. package/dist/services/clean.js +72 -0
  153. package/dist/services/clean.js.map +1 -0
  154. package/dist/services/config.d.ts +26 -0
  155. package/dist/services/config.d.ts.map +1 -0
  156. package/dist/services/config.js +77 -0
  157. package/dist/services/config.js.map +1 -0
  158. package/dist/services/deps.d.ts +53 -0
  159. package/dist/services/deps.d.ts.map +1 -0
  160. package/dist/services/deps.js +62 -0
  161. package/dist/services/deps.js.map +1 -0
  162. package/dist/services/diff.d.ts +34 -0
  163. package/dist/services/diff.d.ts.map +1 -0
  164. package/dist/services/diff.js +34 -0
  165. package/dist/services/diff.js.map +1 -0
  166. package/dist/services/doctor/checks/ai.js +119 -0
  167. package/dist/services/doctor/checks/ai.js.map +1 -0
  168. package/dist/services/doctor/checks/cli.js +147 -0
  169. package/dist/services/doctor/checks/cli.js.map +1 -0
  170. package/dist/services/doctor/checks/config.js +171 -0
  171. package/dist/services/doctor/checks/config.js.map +1 -0
  172. package/dist/services/doctor/checks/deps.js +247 -0
  173. package/dist/services/doctor/checks/deps.js.map +1 -0
  174. package/dist/services/doctor/checks/index.js +6 -0
  175. package/dist/services/doctor/checks/mcp.js +145 -0
  176. package/dist/services/doctor/checks/mcp.js.map +1 -0
  177. package/dist/services/doctor/checks/workspace.js +244 -0
  178. package/dist/services/doctor/checks/workspace.js.map +1 -0
  179. package/dist/services/doctor/doctor-service.d.ts +24 -0
  180. package/dist/services/doctor/doctor-service.d.ts.map +1 -0
  181. package/dist/services/doctor/doctor-service.js +116 -0
  182. package/dist/services/doctor/doctor-service.js.map +1 -0
  183. package/dist/services/doctor/index.d.ts +2 -0
  184. package/dist/services/doctor/index.js +2 -0
  185. package/dist/services/doctor/types.d.ts +118 -0
  186. package/dist/services/doctor/types.d.ts.map +1 -0
  187. package/dist/services/doctor/types.js +27 -0
  188. package/dist/services/doctor/types.js.map +1 -0
  189. package/dist/services/impact/formatters.d.ts +27 -0
  190. package/dist/services/impact/formatters.d.ts.map +1 -0
  191. package/dist/services/impact/formatters.js +111 -0
  192. package/dist/services/impact/formatters.js.map +1 -0
  193. package/dist/services/impact/impact-detection-service.d.ts +22 -0
  194. package/dist/services/impact/impact-detection-service.d.ts.map +1 -0
  195. package/dist/services/impact/impact-detection-service.js +96 -0
  196. package/dist/services/impact/impact-detection-service.js.map +1 -0
  197. package/dist/services/impact/index.d.ts +11 -0
  198. package/dist/services/impact/index.d.ts.map +1 -0
  199. package/dist/services/impact/index.js +16 -0
  200. package/dist/services/impact/index.js.map +1 -0
  201. package/dist/services/impact/types.d.ts +58 -0
  202. package/dist/services/impact/types.d.ts.map +1 -0
  203. package/dist/services/implementation/discovery.d.ts +30 -0
  204. package/dist/services/implementation/discovery.d.ts.map +1 -0
  205. package/dist/services/implementation/discovery.js +144 -0
  206. package/dist/services/implementation/discovery.js.map +1 -0
  207. package/dist/services/implementation/index.d.ts +3 -0
  208. package/dist/services/implementation/index.js +2 -0
  209. package/dist/services/implementation/resolver.d.ts +44 -0
  210. package/dist/services/implementation/resolver.d.ts.map +1 -0
  211. package/dist/services/implementation/resolver.js +224 -0
  212. package/dist/services/implementation/resolver.js.map +1 -0
  213. package/dist/services/implementation/types.d.ts +79 -0
  214. package/dist/services/implementation/types.d.ts.map +1 -0
  215. package/dist/services/index.d.ts +60 -0
  216. package/dist/services/index.js +57 -0
  217. package/dist/services/integrity-diagram.d.ts +36 -0
  218. package/dist/services/integrity-diagram.d.ts.map +1 -0
  219. package/dist/services/integrity-diagram.js +275 -0
  220. package/dist/services/integrity-diagram.js.map +1 -0
  221. package/dist/services/integrity.d.ts +134 -0
  222. package/dist/services/integrity.d.ts.map +1 -0
  223. package/dist/services/integrity.js +272 -0
  224. package/dist/services/integrity.js.map +1 -0
  225. package/dist/services/list.d.ts +31 -0
  226. package/dist/services/list.d.ts.map +1 -0
  227. package/dist/services/list.js +36 -0
  228. package/dist/services/list.js.map +1 -0
  229. package/dist/services/openapi/export-service.d.ts +53 -0
  230. package/dist/services/openapi/export-service.d.ts.map +1 -0
  231. package/dist/services/openapi/export-service.js +50 -0
  232. package/dist/services/openapi/export-service.js.map +1 -0
  233. package/dist/services/openapi/import-service.d.ts +17 -0
  234. package/dist/services/openapi/import-service.d.ts.map +1 -0
  235. package/dist/services/openapi/import-service.js +74 -0
  236. package/dist/services/openapi/import-service.js.map +1 -0
  237. package/dist/services/openapi/index.d.ts +5 -0
  238. package/dist/services/openapi/index.js +4 -0
  239. package/dist/services/openapi/sync-service.d.ts +17 -0
  240. package/dist/services/openapi/sync-service.d.ts.map +1 -0
  241. package/dist/services/openapi/sync-service.js +120 -0
  242. package/dist/services/openapi/sync-service.js.map +1 -0
  243. package/dist/services/openapi/types.d.ts +162 -0
  244. package/dist/services/openapi/types.d.ts.map +1 -0
  245. package/dist/services/openapi/validate-service.d.ts +16 -0
  246. package/dist/services/openapi/validate-service.d.ts.map +1 -0
  247. package/dist/services/openapi/validate-service.js +130 -0
  248. package/dist/services/openapi/validate-service.js.map +1 -0
  249. package/dist/services/quickstart/dependencies.d.ts +31 -0
  250. package/dist/services/quickstart/dependencies.d.ts.map +1 -0
  251. package/dist/services/quickstart/dependencies.js +57 -0
  252. package/dist/services/quickstart/dependencies.js.map +1 -0
  253. package/dist/services/quickstart/index.js +2 -0
  254. package/dist/services/quickstart/quickstart-service.d.ts +20 -0
  255. package/dist/services/quickstart/quickstart-service.d.ts.map +1 -0
  256. package/dist/services/quickstart/quickstart-service.js +196 -0
  257. package/dist/services/quickstart/quickstart-service.js.map +1 -0
  258. package/dist/services/quickstart/types.d.ts +81 -0
  259. package/dist/services/quickstart/types.d.ts.map +1 -0
  260. package/dist/services/regenerator.d.ts +18 -0
  261. package/dist/services/regenerator.d.ts.map +1 -0
  262. package/dist/services/regenerator.js +23 -0
  263. package/dist/services/regenerator.js.map +1 -0
  264. package/dist/services/registry.d.ts +53 -0
  265. package/dist/services/registry.d.ts.map +1 -0
  266. package/dist/services/registry.js +74 -0
  267. package/dist/services/registry.js.map +1 -0
  268. package/dist/services/setup/config-generators.d.ts +42 -0
  269. package/dist/services/setup/config-generators.d.ts.map +1 -0
  270. package/dist/services/setup/config-generators.js +238 -0
  271. package/dist/services/setup/config-generators.js.map +1 -0
  272. package/dist/services/setup/file-merger.d.ts +27 -0
  273. package/dist/services/setup/file-merger.d.ts.map +1 -0
  274. package/dist/services/setup/file-merger.js +61 -0
  275. package/dist/services/setup/file-merger.js.map +1 -0
  276. package/dist/services/setup/index.js +4 -0
  277. package/dist/services/setup/setup-service.d.ts +12 -0
  278. package/dist/services/setup/setup-service.d.ts.map +1 -0
  279. package/dist/services/setup/setup-service.js +96 -0
  280. package/dist/services/setup/setup-service.js.map +1 -0
  281. package/dist/services/setup/targets/agents-md.js +47 -0
  282. package/dist/services/setup/targets/agents-md.js.map +1 -0
  283. package/dist/services/setup/targets/cli-config.js +60 -0
  284. package/dist/services/setup/targets/cli-config.js.map +1 -0
  285. package/dist/services/setup/targets/cursor-rules.js +48 -0
  286. package/dist/services/setup/targets/cursor-rules.js.map +1 -0
  287. package/dist/services/setup/targets/mcp-claude.js +60 -0
  288. package/dist/services/setup/targets/mcp-claude.js.map +1 -0
  289. package/dist/services/setup/targets/mcp-cursor.js +59 -0
  290. package/dist/services/setup/targets/mcp-cursor.js.map +1 -0
  291. package/dist/services/setup/targets/vscode-settings.js +63 -0
  292. package/dist/services/setup/targets/vscode-settings.js.map +1 -0
  293. package/dist/services/setup/types.d.ts +85 -0
  294. package/dist/services/setup/types.d.ts.map +1 -0
  295. package/dist/services/setup/types.js +27 -0
  296. package/dist/services/setup/types.js.map +1 -0
  297. package/dist/services/sync.d.ts +41 -0
  298. package/dist/services/sync.d.ts.map +1 -0
  299. package/dist/services/sync.js +63 -0
  300. package/dist/services/sync.js.map +1 -0
  301. package/dist/services/test.d.ts +15 -0
  302. package/dist/services/test.d.ts.map +1 -0
  303. package/dist/services/test.js +30 -0
  304. package/dist/services/test.js.map +1 -0
  305. package/dist/services/validate-implementation.d.ts +32 -0
  306. package/dist/services/validate-implementation.d.ts.map +1 -0
  307. package/dist/services/validate-implementation.js +64 -0
  308. package/dist/services/validate-implementation.js.map +1 -0
  309. package/dist/services/validate.d.ts +41 -0
  310. package/dist/services/validate.d.ts.map +1 -0
  311. package/dist/services/validate.js +48 -0
  312. package/dist/services/validate.js.map +1 -0
  313. package/dist/services/verification-cache/adapters/filesystem.d.ts +46 -0
  314. package/dist/services/verification-cache/adapters/filesystem.d.ts.map +1 -0
  315. package/dist/services/verification-cache/adapters/filesystem.js +120 -0
  316. package/dist/services/verification-cache/adapters/filesystem.js.map +1 -0
  317. package/dist/services/verification-cache/adapters/in-memory.d.ts +27 -0
  318. package/dist/services/verification-cache/adapters/in-memory.d.ts.map +1 -0
  319. package/dist/services/verification-cache/adapters/in-memory.js +46 -0
  320. package/dist/services/verification-cache/adapters/in-memory.js.map +1 -0
  321. package/dist/services/verification-cache/adapters/index.d.ts +3 -0
  322. package/dist/services/verification-cache/adapters/index.js +3 -0
  323. package/dist/services/verification-cache/adapters/workspace-state.d.ts +49 -0
  324. package/dist/services/verification-cache/adapters/workspace-state.d.ts.map +1 -0
  325. package/dist/services/verification-cache/adapters/workspace-state.js +91 -0
  326. package/dist/services/verification-cache/adapters/workspace-state.js.map +1 -0
  327. package/dist/services/verification-cache/cache-service.d.ts +70 -0
  328. package/dist/services/verification-cache/cache-service.d.ts.map +1 -0
  329. package/dist/services/verification-cache/cache-service.js +256 -0
  330. package/dist/services/verification-cache/cache-service.js.map +1 -0
  331. package/dist/services/verification-cache/index.d.ts +6 -0
  332. package/dist/services/verification-cache/index.js +6 -0
  333. package/dist/services/verification-cache/types.d.ts +124 -0
  334. package/dist/services/verification-cache/types.d.ts.map +1 -0
  335. package/dist/services/verification-cache/types.js +16 -0
  336. package/dist/services/verification-cache/types.js.map +1 -0
  337. package/dist/services/verify/ai-verifier.d.ts +25 -0
  338. package/dist/services/verify/ai-verifier.d.ts.map +1 -0
  339. package/dist/services/verify/ai-verifier.js +403 -0
  340. package/dist/services/verify/ai-verifier.js.map +1 -0
  341. package/dist/services/verify/behavior-verifier.d.ts +12 -0
  342. package/dist/services/verify/behavior-verifier.d.ts.map +1 -0
  343. package/dist/services/verify/behavior-verifier.js +186 -0
  344. package/dist/services/verify/behavior-verifier.js.map +1 -0
  345. package/dist/services/verify/index.d.ts +5 -0
  346. package/dist/services/verify/index.js +4 -0
  347. package/dist/services/verify/structure-verifier.d.ts +12 -0
  348. package/dist/services/verify/structure-verifier.d.ts.map +1 -0
  349. package/dist/services/verify/structure-verifier.js +196 -0
  350. package/dist/services/verify/structure-verifier.js.map +1 -0
  351. package/dist/services/verify/types.d.ts +137 -0
  352. package/dist/services/verify/types.d.ts.map +1 -0
  353. package/dist/services/verify/verify-service.d.ts +60 -0
  354. package/dist/services/verify/verify-service.d.ts.map +1 -0
  355. package/dist/services/verify/verify-service.js +204 -0
  356. package/dist/services/verify/verify-service.js.map +1 -0
  357. package/dist/services/watch.d.ts +25 -0
  358. package/dist/services/watch.d.ts.map +1 -0
  359. package/dist/services/watch.js +32 -0
  360. package/dist/services/watch.js.map +1 -0
  361. package/dist/services/workspace-info.d.ts +62 -0
  362. package/dist/services/workspace-info.d.ts.map +1 -0
  363. package/dist/services/workspace-info.js +103 -0
  364. package/dist/services/workspace-info.js.map +1 -0
  365. package/dist/templates/app-config.template.d.ts +7 -0
  366. package/dist/templates/app-config.template.d.ts.map +1 -0
  367. package/dist/templates/app-config.template.js +107 -0
  368. package/dist/templates/app-config.template.js.map +1 -0
  369. package/dist/templates/data-view.template.d.ts +7 -0
  370. package/dist/templates/data-view.template.d.ts.map +1 -0
  371. package/dist/templates/data-view.template.js +70 -0
  372. package/dist/templates/data-view.template.js.map +1 -0
  373. package/dist/templates/event.template.d.ts +11 -0
  374. package/dist/templates/event.template.d.ts.map +1 -0
  375. package/dist/templates/event.template.js +40 -0
  376. package/dist/templates/event.template.js.map +1 -0
  377. package/dist/templates/experiment.template.d.ts +7 -0
  378. package/dist/templates/experiment.template.d.ts.map +1 -0
  379. package/dist/templates/experiment.template.js +89 -0
  380. package/dist/templates/experiment.template.js.map +1 -0
  381. package/dist/templates/handler.template.d.ts +16 -0
  382. package/dist/templates/handler.template.d.ts.map +1 -0
  383. package/dist/templates/handler.template.js +100 -0
  384. package/dist/templates/handler.template.js.map +1 -0
  385. package/dist/templates/index.d.ts +21 -0
  386. package/dist/templates/index.d.ts.map +1 -0
  387. package/dist/templates/index.js +37 -0
  388. package/dist/templates/index.js.map +1 -0
  389. package/dist/templates/integration.template.d.ts +7 -0
  390. package/dist/templates/integration.template.d.ts.map +1 -0
  391. package/dist/templates/integration.template.js +160 -0
  392. package/dist/templates/integration.template.js.map +1 -0
  393. package/dist/templates/knowledge.template.d.ts +7 -0
  394. package/dist/templates/knowledge.template.d.ts.map +1 -0
  395. package/dist/templates/knowledge.template.js +75 -0
  396. package/dist/templates/knowledge.template.js.map +1 -0
  397. package/dist/templates/migration.template.d.ts +7 -0
  398. package/dist/templates/migration.template.d.ts.map +1 -0
  399. package/dist/templates/migration.template.js +62 -0
  400. package/dist/templates/migration.template.js.map +1 -0
  401. package/dist/templates/operation.template.d.ts +11 -0
  402. package/dist/templates/operation.template.d.ts.map +1 -0
  403. package/dist/templates/operation.template.js +105 -0
  404. package/dist/templates/operation.template.js.map +1 -0
  405. package/dist/templates/presentation.template.d.ts +11 -0
  406. package/dist/templates/presentation.template.d.ts.map +1 -0
  407. package/dist/templates/presentation.template.js +80 -0
  408. package/dist/templates/presentation.template.js.map +1 -0
  409. package/dist/templates/telemetry.template.d.ts +7 -0
  410. package/dist/templates/telemetry.template.d.ts.map +1 -0
  411. package/dist/templates/telemetry.template.js +91 -0
  412. package/dist/templates/telemetry.template.js.map +1 -0
  413. package/dist/templates/workflow-runner.template.d.ts +16 -0
  414. package/dist/templates/workflow-runner.template.d.ts.map +1 -0
  415. package/dist/templates/workflow-runner.template.js +50 -0
  416. package/dist/templates/workflow-runner.template.js.map +1 -0
  417. package/dist/templates/workflow.template.d.ts +7 -0
  418. package/dist/templates/workflow.template.d.ts.map +1 -0
  419. package/dist/templates/workflow.template.js +69 -0
  420. package/dist/templates/workflow.template.js.map +1 -0
  421. package/dist/types/config.d.ts +34 -0
  422. package/dist/types/config.d.ts.map +1 -0
  423. package/dist/types.d.ts +324 -0
  424. package/dist/types.d.ts.map +1 -0
  425. package/package.json +68 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"impact-detection-service.js","names":["baseSnapshot: ContractSnapshot","specs: { path: string; content: string }[]","diffs: ReturnType<typeof computeIoDiff>"],"sources":["../../../src/services/impact/impact-detection-service.ts"],"sourcesContent":["/**\n * Impact detection service.\n *\n * Orchestrates contract snapshot generation, diff computation,\n * and impact classification for CI/CD pipelines.\n */\n\n// Note: Using internal imports from the bundle which re-exports from module\nimport {\n classifyImpact,\n computeIoDiff,\n generateSnapshot,\n type ContractSnapshot,\n type SpecSnapshot,\n} from '../../index';\nimport type { FsAdapter } from '../../ports/fs';\nimport type { GitAdapter } from '../../ports/git';\nimport type { LoggerAdapter } from '../../ports/logger';\nimport type { ImpactDetectionOptions, ImpactDetectionResult } from './types';\n\n/**\n * Detect the impact of contract changes between baseline and current state.\n *\n * @param adapters - Required adapters (fs, git, logger)\n * @param options - Detection options\n * @returns Impact detection result\n */\nexport async function detectImpact(\n adapters: { fs: FsAdapter; git: GitAdapter; logger: LoggerAdapter },\n options: ImpactDetectionOptions = {}\n): Promise<ImpactDetectionResult> {\n const { fs, git, logger } = adapters;\n const workspaceRoot = options.workspaceRoot ?? process.cwd();\n\n logger.info('Starting impact detection...', { baseline: options.baseline });\n\n // Discover spec files\n const files = await fs.glob({\n pattern: options.pattern ?? '**/*.{operation,event}.ts',\n cwd: workspaceRoot,\n });\n\n const specFiles = files.filter(\n (f) =>\n !f.includes('.test.') &&\n !f.includes('.spec.') &&\n !f.includes('node_modules')\n );\n\n logger.debug(`Found ${specFiles.length} spec files`);\n\n // Load current (head) specs\n const headSpecs = await loadSpecs(fs, specFiles, workspaceRoot);\n const headSnapshot = generateSnapshot(headSpecs);\n\n // Load baseline specs if specified\n let baseSnapshot: ContractSnapshot;\n if (options.baseline) {\n const baseSpecs = await loadBaselineSpecs(\n fs,\n git,\n specFiles,\n options.baseline,\n workspaceRoot\n );\n baseSnapshot = generateSnapshot(baseSpecs);\n } else {\n // No baseline means all specs are \"new\"\n baseSnapshot = { version: 1, generatedAt: '', specs: [], hash: '' };\n }\n\n // Compute diffs between snapshots\n const diffs = computeSnapshotDiffs(baseSnapshot.specs, headSnapshot.specs);\n\n // Classify the impact\n const impactResult = classifyImpact(\n baseSnapshot.specs,\n headSnapshot.specs,\n diffs\n );\n\n logger.info('Impact detection complete', {\n status: impactResult.status,\n breaking: impactResult.summary.breaking,\n nonBreaking: impactResult.summary.nonBreaking,\n });\n\n return {\n ...impactResult,\n workspaceRoot,\n specsAnalyzed: specFiles.length,\n baseRef: options.baseline,\n };\n}\n\n/**\n * Load specs from current filesystem.\n */\nasync function loadSpecs(\n fs: FsAdapter,\n files: string[],\n _workspaceRoot: string\n): Promise<{ path: string; content: string }[]> {\n const specs: { path: string; content: string }[] = [];\n\n for (const file of files) {\n const content = await fs.readFile(file);\n specs.push({ path: file, content });\n }\n\n return specs;\n}\n\n/**\n * Load specs from git baseline.\n */\nasync function loadBaselineSpecs(\n _fs: FsAdapter,\n git: GitAdapter,\n files: string[],\n baseline: string,\n _workspaceRoot: string\n): Promise<{ path: string; content: string }[]> {\n const specs: { path: string; content: string }[] = [];\n\n for (const file of files) {\n try {\n const content = await git.showFile(baseline, file);\n specs.push({ path: file, content });\n } catch {\n // File doesn't exist in baseline - skip (it's new)\n }\n }\n\n return specs;\n}\n\n/**\n * Compute diffs between two sets of spec snapshots.\n */\nfunction computeSnapshotDiffs(\n baseSpecs: SpecSnapshot[],\n headSpecs: SpecSnapshot[]\n): ReturnType<typeof computeIoDiff> {\n const diffs: ReturnType<typeof computeIoDiff> = [];\n\n const baseMap = new Map(baseSpecs.map((s) => [`${s.key}@${s.version}`, s]));\n const headMap = new Map(headSpecs.map((s) => [`${s.key}@${s.version}`, s]));\n\n // Compare specs that exist in both\n for (const [key, headSpec] of headMap) {\n const baseSpec = baseMap.get(key);\n if (\n baseSpec &&\n headSpec.type === 'operation' &&\n baseSpec.type === 'operation'\n ) {\n const ioDiffs = computeIoDiff(baseSpec.io, headSpec.io);\n diffs.push(...ioDiffs);\n }\n }\n\n return diffs;\n}\n"],"mappings":";;;;;;;;;;;;;;;;AA2BA,eAAsB,aACpB,UACA,UAAkC,EAAE,EACJ;CAChC,MAAM,EAAE,IAAI,KAAK,WAAW;CAC5B,MAAM,gBAAgB,QAAQ,iBAAiB,QAAQ,KAAK;AAE5D,QAAO,KAAK,gCAAgC,EAAE,UAAU,QAAQ,UAAU,CAAC;CAQ3E,MAAM,aALQ,MAAM,GAAG,KAAK;EAC1B,SAAS,QAAQ,WAAW;EAC5B,KAAK;EACN,CAAC,EAEsB,QACrB,MACC,CAAC,EAAE,SAAS,SAAS,IACrB,CAAC,EAAE,SAAS,SAAS,IACrB,CAAC,EAAE,SAAS,eAAe,CAC9B;AAED,QAAO,MAAM,SAAS,UAAU,OAAO,aAAa;CAIpD,MAAM,iDADY,MAAM,UAAU,IAAI,WAAW,cAAc,CACf;CAGhD,IAAIA;AACJ,KAAI,QAAQ,SAQV,kDAPkB,MAAM,kBACtB,IACA,KACA,WACA,QAAQ,UACR,cACD,CACyC;KAG1C,gBAAe;EAAE,SAAS;EAAG,aAAa;EAAI,OAAO,EAAE;EAAE,MAAM;EAAI;CAIrE,MAAM,QAAQ,qBAAqB,aAAa,OAAO,aAAa,MAAM;CAG1E,MAAM,+CACJ,aAAa,OACb,aAAa,OACb,MACD;AAED,QAAO,KAAK,6BAA6B;EACvC,QAAQ,aAAa;EACrB,UAAU,aAAa,QAAQ;EAC/B,aAAa,aAAa,QAAQ;EACnC,CAAC;AAEF,QAAO;EACL,GAAG;EACH;EACA,eAAe,UAAU;EACzB,SAAS,QAAQ;EAClB;;;;;AAMH,eAAe,UACb,IACA,OACA,gBAC8C;CAC9C,MAAMC,QAA6C,EAAE;AAErD,MAAK,MAAM,QAAQ,OAAO;EACxB,MAAM,UAAU,MAAM,GAAG,SAAS,KAAK;AACvC,QAAM,KAAK;GAAE,MAAM;GAAM;GAAS,CAAC;;AAGrC,QAAO;;;;;AAMT,eAAe,kBACb,KACA,KACA,OACA,UACA,gBAC8C;CAC9C,MAAMA,QAA6C,EAAE;AAErD,MAAK,MAAM,QAAQ,MACjB,KAAI;EACF,MAAM,UAAU,MAAM,IAAI,SAAS,UAAU,KAAK;AAClD,QAAM,KAAK;GAAE,MAAM;GAAM;GAAS,CAAC;SAC7B;AAKV,QAAO;;;;;AAMT,SAAS,qBACP,WACA,WACkC;CAClC,MAAMC,QAA0C,EAAE;CAElD,MAAM,UAAU,IAAI,IAAI,UAAU,KAAK,MAAM,CAAC,GAAG,EAAE,IAAI,GAAG,EAAE,WAAW,EAAE,CAAC,CAAC;CAC3E,MAAM,UAAU,IAAI,IAAI,UAAU,KAAK,MAAM,CAAC,GAAG,EAAE,IAAI,GAAG,EAAE,WAAW,EAAE,CAAC,CAAC;AAG3E,MAAK,MAAM,CAAC,KAAK,aAAa,SAAS;EACrC,MAAM,WAAW,QAAQ,IAAI,IAAI;AACjC,MACE,YACA,SAAS,SAAS,eAClB,SAAS,SAAS,aAClB;GACA,MAAM,yCAAwB,SAAS,IAAI,SAAS,GAAG;AACvD,SAAM,KAAK,GAAG,QAAQ;;;AAI1B,QAAO"}
@@ -0,0 +1,11 @@
1
+ import { CheckRunAnnotation, CheckRunPayload, ImpactDetectionOptions, ImpactDetectionResult, ImpactResult, PrCommentOptions } from "./types.js";
2
+ import { detectImpact } from "./impact-detection-service.js";
3
+ import { formatCheckRun, formatJson, formatMinimalComment, formatPrComment } from "./formatters.js";
4
+
5
+ //#region src/services/impact/index.d.ts
6
+ declare namespace index_d_exports {
7
+ export { CheckRunAnnotation, CheckRunPayload, ImpactDetectionOptions, ImpactDetectionResult, ImpactResult, PrCommentOptions, detectImpact, formatCheckRun, formatJson, formatMinimalComment, formatPrComment };
8
+ }
9
+ //#endregion
10
+ export { index_d_exports };
11
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/services/impact/index.ts"],"sourcesContent":[],"mappings":""}
@@ -0,0 +1,16 @@
1
+ import { __exportAll } from "../../_virtual/rolldown_runtime.js";
2
+ import { detectImpact } from "./impact-detection-service.js";
3
+ import { formatCheckRun, formatJson, formatMinimalComment, formatPrComment } from "./formatters.js";
4
+
5
+ //#region src/services/impact/index.ts
6
+ var impact_exports = /* @__PURE__ */ __exportAll({
7
+ detectImpact: () => detectImpact,
8
+ formatCheckRun: () => formatCheckRun,
9
+ formatJson: () => formatJson,
10
+ formatMinimalComment: () => formatMinimalComment,
11
+ formatPrComment: () => formatPrComment
12
+ });
13
+
14
+ //#endregion
15
+ export { impact_exports };
16
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":[],"sources":["../../../src/services/impact/index.ts"],"sourcesContent":["/**\n * Impact detection service module.\n */\n\nexport * from './types';\nexport { detectImpact } from './impact-detection-service';\nexport {\n formatPrComment,\n formatMinimalComment,\n formatCheckRun,\n formatJson,\n} from './formatters';\n"],"mappings":""}
@@ -0,0 +1,58 @@
1
+ import { ImpactResult } from "@contractspec/module.workspace";
2
+
3
+ //#region src/services/impact/types.d.ts
4
+
5
+ /** Options for impact detection */
6
+ interface ImpactDetectionOptions {
7
+ /** Git ref to compare against (branch, tag, commit) */
8
+ baseline?: string;
9
+ /** Glob pattern for spec discovery */
10
+ pattern?: string;
11
+ /** Workspace root directory */
12
+ workspaceRoot?: string;
13
+ }
14
+ /** Impact detection result with additional context */
15
+ interface ImpactDetectionResult extends ImpactResult {
16
+ /** Working directory */
17
+ workspaceRoot: string;
18
+ /** Number of specs analyzed */
19
+ specsAnalyzed: number;
20
+ }
21
+ /** PR comment options */
22
+ interface PrCommentOptions {
23
+ /** Template style */
24
+ template: 'minimal' | 'detailed';
25
+ /** Include links to files */
26
+ includeLinks?: boolean;
27
+ /** Repository URL for links */
28
+ repositoryUrl?: string;
29
+ /** Commit SHA for links */
30
+ commitSha?: string;
31
+ }
32
+ /** GitHub check run payload */
33
+ interface CheckRunPayload {
34
+ /** Check run name */
35
+ name: string;
36
+ /** Head SHA */
37
+ headSha: string;
38
+ /** Conclusion */
39
+ conclusion: 'success' | 'failure' | 'neutral';
40
+ /** Output title */
41
+ title: string;
42
+ /** Output summary */
43
+ summary: string;
44
+ /** Annotations */
45
+ annotations?: CheckRunAnnotation[];
46
+ }
47
+ /** Check run annotation */
48
+ interface CheckRunAnnotation {
49
+ path: string;
50
+ startLine: number;
51
+ endLine: number;
52
+ annotationLevel: 'notice' | 'warning' | 'failure';
53
+ message: string;
54
+ title?: string;
55
+ }
56
+ //#endregion
57
+ export { CheckRunAnnotation, CheckRunPayload, ImpactDetectionOptions, ImpactDetectionResult, type ImpactResult, PrCommentOptions };
58
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","names":[],"sources":["../../../src/services/impact/types.ts"],"sourcesContent":[],"mappings":";;;;AA4BA;AAYiB,UA9BA,sBAAA,CA0CD;EAIC;;;;;;;;UApCA,qBAAA,SAA8B;;;;;;;UAQ9B,gBAAA;;;;;;;;;;;UAYA,eAAA;;;;;;;;;;;;gBAYD;;;UAIC,kBAAA"}
@@ -0,0 +1,30 @@
1
+ import { FsAdapter } from "../../ports/fs.js";
2
+ import { DiscoveryOptions, SpecReferenceMatch } from "./types.js";
3
+ import { ImplementationType } from "@contractspec/lib.contracts";
4
+
5
+ //#region src/services/implementation/discovery.d.ts
6
+
7
+ /**
8
+ * Infer implementation type from file path.
9
+ */
10
+ declare function inferImplementationType(filePath: string): ImplementationType;
11
+ /**
12
+ * Extract spec references from source code.
13
+ */
14
+ declare function extractSpecReferences(code: string, filePath: string): SpecReferenceMatch[];
15
+ /**
16
+ * Discover implementations that reference a specific spec.
17
+ */
18
+ declare function discoverImplementationsForSpec(specKey: string, adapters: {
19
+ fs: FsAdapter;
20
+ }, options?: DiscoveryOptions): Promise<SpecReferenceMatch[]>;
21
+ /**
22
+ * Discover all spec references in the workspace.
23
+ * Returns a map of spec key to implementation references.
24
+ */
25
+ declare function discoverAllImplementations(adapters: {
26
+ fs: FsAdapter;
27
+ }, options?: DiscoveryOptions): Promise<Map<string, SpecReferenceMatch[]>>;
28
+ //#endregion
29
+ export { discoverAllImplementations, discoverImplementationsForSpec, extractSpecReferences, inferImplementationType };
30
+ //# sourceMappingURL=discovery.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"discovery.d.ts","names":[],"sources":["../../../src/services/implementation/discovery.ts"],"sourcesContent":[],"mappings":";;;;;;;;;AAsLU,iBAnIM,uBAAA,CAmIN,QAAA,EAAA,MAAA,CAAA,EAnIiD,kBAmIjD;AAkCV;;;AAGuB,iBAhJP,qBAAA,CAgJO,IAAA,EAAA,MAAA,EAAA,QAAA,EAAA,MAAA,CAAA,EA7IpB,kBA6IoB,EAAA;;;;iBAzCD,8BAAA;MAEJ;aACP,mBACR,QAAQ;;;;;iBAkCW,0BAAA;MACJ;aACP,mBACR,QAAQ,YAAY"}
@@ -0,0 +1,144 @@
1
+ //#region src/services/implementation/discovery.ts
2
+ /**
3
+ * Patterns for detecting spec references in source code.
4
+ */
5
+ const SPEC_REFERENCE_PATTERNS = {
6
+ namedImport: /import\s*\{[^}]*\b(\w+(?:Spec|Contract|Command|Query))\b[^}]*\}\s*from/g,
7
+ defaultImport: /import\s+(\w+(?:Spec|Contract|Command|Query))\s+from/g,
8
+ contractHandler: /ContractHandler\s*<\s*typeof\s+(\w+)\s*>/g,
9
+ typeofSpec: /typeof\s+(\w+(?:Spec|Contract|Command|Query))\b/g,
10
+ specAssignment: /(?:spec|contract)\s*[:=]\s*(\w+(?:Spec|Contract|Command|Query))\b/gi
11
+ };
12
+ /**
13
+ * File patterns that indicate implementation types.
14
+ */
15
+ const IMPLEMENTATION_TYPE_PATTERNS = {
16
+ ".handler.ts": "handler",
17
+ ".handler.tsx": "handler",
18
+ ".service.ts": "service",
19
+ ".service.tsx": "service",
20
+ ".test.ts": "test",
21
+ ".test.tsx": "test",
22
+ ".spec.ts": "test",
23
+ ".spec.tsx": "test",
24
+ ".component.tsx": "component",
25
+ ".tsx": "component",
26
+ ".form.tsx": "form",
27
+ ".hook.ts": "hook",
28
+ ".hook.tsx": "hook"
29
+ };
30
+ /**
31
+ * Infer implementation type from file path.
32
+ */
33
+ function inferImplementationType(filePath) {
34
+ const lowerPath = filePath.toLowerCase();
35
+ for (const [pattern, type] of Object.entries(IMPLEMENTATION_TYPE_PATTERNS)) if (lowerPath.endsWith(pattern)) return type;
36
+ if (lowerPath.includes("/handlers/")) return "handler";
37
+ if (lowerPath.includes("/services/")) return "service";
38
+ if (lowerPath.includes("/components/")) return "component";
39
+ if (lowerPath.includes("/forms/")) return "form";
40
+ if (lowerPath.includes("/hooks/")) return "hook";
41
+ if (lowerPath.includes("/__tests__/")) return "test";
42
+ return "other";
43
+ }
44
+ /**
45
+ * Extract spec references from source code.
46
+ */
47
+ function extractSpecReferences(code, filePath) {
48
+ const matches = [];
49
+ const seenSpecs = /* @__PURE__ */ new Set();
50
+ const addMatch = (specKey, referenceType, lineNumber) => {
51
+ const key = `${specKey}:${referenceType}`;
52
+ if (seenSpecs.has(key)) return;
53
+ seenSpecs.add(key);
54
+ matches.push({
55
+ filePath,
56
+ specKey,
57
+ referenceType,
58
+ lineNumber,
59
+ inferredType: inferImplementationType(filePath)
60
+ });
61
+ };
62
+ const getLineNumber = (position) => {
63
+ return code.substring(0, position).split("\n").length;
64
+ };
65
+ let match;
66
+ const handlerPattern = new RegExp(SPEC_REFERENCE_PATTERNS.contractHandler);
67
+ while ((match = handlerPattern.exec(code)) !== null) if (match[1]) addMatch(match[1], "handler", getLineNumber(match.index));
68
+ const typeofPattern = new RegExp(SPEC_REFERENCE_PATTERNS.typeofSpec);
69
+ while ((match = typeofPattern.exec(code)) !== null) if (match[1]) addMatch(match[1], "typeof", getLineNumber(match.index));
70
+ const namedPattern = new RegExp(SPEC_REFERENCE_PATTERNS.namedImport);
71
+ while ((match = namedPattern.exec(code)) !== null) {
72
+ const specNames = match[0].match(/\b(\w+(?:Spec|Contract|Command|Query))\b/g);
73
+ if (specNames) for (const name of specNames) addMatch(name, "import", getLineNumber(match.index));
74
+ }
75
+ const defaultPattern = new RegExp(SPEC_REFERENCE_PATTERNS.defaultImport);
76
+ while ((match = defaultPattern.exec(code)) !== null) if (match[1]) addMatch(match[1], "import", getLineNumber(match.index));
77
+ const assignPattern = new RegExp(SPEC_REFERENCE_PATTERNS.specAssignment);
78
+ while ((match = assignPattern.exec(code)) !== null) if (match[1]) addMatch(match[1], "unknown", getLineNumber(match.index));
79
+ return matches;
80
+ }
81
+ /**
82
+ * Default glob patterns for implementation files.
83
+ */
84
+ const DEFAULT_INCLUDE_PATTERNS = ["**/*.ts", "**/*.tsx"];
85
+ const DEFAULT_EXCLUDE_PATTERNS = [
86
+ "**/node_modules/**",
87
+ "**/dist/**",
88
+ "**/.git/**",
89
+ "**/*.d.ts",
90
+ "**/*.operation.ts",
91
+ "**/*.spec.ts",
92
+ "**/*.feature.ts",
93
+ "**/*.event.ts",
94
+ "**/*.presentation.ts"
95
+ ];
96
+ /**
97
+ * Discover implementations that reference a specific spec.
98
+ */
99
+ async function discoverImplementationsForSpec(specKey, adapters, options = {}) {
100
+ const { fs } = adapters;
101
+ const includePatterns = options.includePatterns ?? DEFAULT_INCLUDE_PATTERNS;
102
+ const excludePatterns = options.excludePatterns ?? DEFAULT_EXCLUDE_PATTERNS;
103
+ const allMatches = [];
104
+ for (const pattern of includePatterns) {
105
+ const files = await fs.glob({
106
+ pattern,
107
+ ignore: excludePatterns
108
+ });
109
+ for (const filePath of files) try {
110
+ const matchingRefs = extractSpecReferences(await fs.readFile(filePath), filePath).filter((ref) => ref.specKey === specKey);
111
+ allMatches.push(...matchingRefs);
112
+ } catch {}
113
+ }
114
+ return allMatches;
115
+ }
116
+ /**
117
+ * Discover all spec references in the workspace.
118
+ * Returns a map of spec key to implementation references.
119
+ */
120
+ async function discoverAllImplementations(adapters, options = {}) {
121
+ const { fs } = adapters;
122
+ const includePatterns = options.includePatterns ?? DEFAULT_INCLUDE_PATTERNS;
123
+ const excludePatterns = options.excludePatterns ?? DEFAULT_EXCLUDE_PATTERNS;
124
+ const specToImplementations = /* @__PURE__ */ new Map();
125
+ for (const pattern of includePatterns) {
126
+ const files = await fs.glob({
127
+ pattern,
128
+ ignore: excludePatterns
129
+ });
130
+ for (const filePath of files) try {
131
+ const references = extractSpecReferences(await fs.readFile(filePath), filePath);
132
+ for (const ref of references) {
133
+ const existing = specToImplementations.get(ref.specKey) ?? [];
134
+ existing.push(ref);
135
+ specToImplementations.set(ref.specKey, existing);
136
+ }
137
+ } catch {}
138
+ }
139
+ return specToImplementations;
140
+ }
141
+
142
+ //#endregion
143
+ export { discoverAllImplementations, discoverImplementationsForSpec, extractSpecReferences, inferImplementationType };
144
+ //# sourceMappingURL=discovery.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"discovery.js","names":["IMPLEMENTATION_TYPE_PATTERNS: Record<string, ImplementationType>","matches: SpecReferenceMatch[]","match: RegExpExecArray | null","allMatches: SpecReferenceMatch[]"],"sources":["../../../src/services/implementation/discovery.ts"],"sourcesContent":["/**\n * Implementation auto-discovery service.\n *\n * Scans workspace files to find implementations that reference specs.\n * Uses regex-based pattern matching for performance (avoids full AST parsing).\n */\n\nimport type { FsAdapter } from '../../ports/fs';\nimport type { ImplementationType } from '@contractspec/lib.contracts';\nimport type { DiscoveryOptions, SpecReferenceMatch } from './types';\n\n/**\n * Patterns for detecting spec references in source code.\n */\nconst SPEC_REFERENCE_PATTERNS = {\n // import { SpecName } from './path'\n namedImport:\n /import\\s*\\{[^}]*\\b(\\w+(?:Spec|Contract|Command|Query))\\b[^}]*\\}\\s*from/g,\n // import SpecName from './path'\n defaultImport: /import\\s+(\\w+(?:Spec|Contract|Command|Query))\\s+from/g,\n // ContractHandler<typeof SpecName>\n contractHandler: /ContractHandler\\s*<\\s*typeof\\s+(\\w+)\\s*>/g,\n // typeof SpecName (generic usage)\n typeofSpec: /typeof\\s+(\\w+(?:Spec|Contract|Command|Query))\\b/g,\n // spec: SpecKey or spec = SpecKey\n specAssignment:\n /(?:spec|contract)\\s*[:=]\\s*(\\w+(?:Spec|Contract|Command|Query))\\b/gi,\n};\n\n/**\n * File patterns that indicate implementation types.\n */\nconst IMPLEMENTATION_TYPE_PATTERNS: Record<string, ImplementationType> = {\n '.handler.ts': 'handler',\n '.handler.tsx': 'handler',\n '.service.ts': 'service',\n '.service.tsx': 'service',\n '.test.ts': 'test',\n '.test.tsx': 'test',\n '.spec.ts': 'test',\n '.spec.tsx': 'test',\n '.component.tsx': 'component',\n '.tsx': 'component', // Default for TSX files\n '.form.tsx': 'form',\n '.hook.ts': 'hook',\n '.hook.tsx': 'hook',\n};\n\n/**\n * Infer implementation type from file path.\n */\nexport function inferImplementationType(filePath: string): ImplementationType {\n const lowerPath = filePath.toLowerCase();\n\n // Check specific patterns first (order matters)\n for (const [pattern, type] of Object.entries(IMPLEMENTATION_TYPE_PATTERNS)) {\n if (lowerPath.endsWith(pattern)) {\n return type;\n }\n }\n\n // Check directory patterns\n if (lowerPath.includes('/handlers/')) return 'handler';\n if (lowerPath.includes('/services/')) return 'service';\n if (lowerPath.includes('/components/')) return 'component';\n if (lowerPath.includes('/forms/')) return 'form';\n if (lowerPath.includes('/hooks/')) return 'hook';\n if (lowerPath.includes('/__tests__/')) return 'test';\n\n return 'other';\n}\n\n/**\n * Extract spec references from source code.\n */\nexport function extractSpecReferences(\n code: string,\n filePath: string\n): SpecReferenceMatch[] {\n const matches: SpecReferenceMatch[] = [];\n const seenSpecs = new Set<string>();\n\n // Helper to add unique matches\n const addMatch = (\n specKey: string,\n referenceType: SpecReferenceMatch['referenceType'],\n lineNumber?: number\n ) => {\n const key = `${specKey}:${referenceType}`;\n if (seenSpecs.has(key)) return;\n seenSpecs.add(key);\n\n matches.push({\n filePath,\n specKey,\n referenceType,\n lineNumber,\n inferredType: inferImplementationType(filePath),\n });\n };\n\n // Find line number for a match position\n const getLineNumber = (position: number): number => {\n const lines = code.substring(0, position).split('\\n');\n return lines.length;\n };\n\n // Check ContractHandler pattern (most specific)\n let match: RegExpExecArray | null;\n const handlerPattern = new RegExp(SPEC_REFERENCE_PATTERNS.contractHandler);\n while ((match = handlerPattern.exec(code)) !== null) {\n if (match[1]) {\n addMatch(match[1], 'handler', getLineNumber(match.index));\n }\n }\n\n // Check typeof pattern\n const typeofPattern = new RegExp(SPEC_REFERENCE_PATTERNS.typeofSpec);\n while ((match = typeofPattern.exec(code)) !== null) {\n if (match[1]) {\n addMatch(match[1], 'typeof', getLineNumber(match.index));\n }\n }\n\n // Check named imports\n const namedPattern = new RegExp(SPEC_REFERENCE_PATTERNS.namedImport);\n while ((match = namedPattern.exec(code)) !== null) {\n // Extract all spec names from the import statement\n const importBlock = match[0];\n const specNames = importBlock.match(\n /\\b(\\w+(?:Spec|Contract|Command|Query))\\b/g\n );\n if (specNames) {\n for (const name of specNames) {\n addMatch(name, 'import', getLineNumber(match.index));\n }\n }\n }\n\n // Check default imports\n const defaultPattern = new RegExp(SPEC_REFERENCE_PATTERNS.defaultImport);\n while ((match = defaultPattern.exec(code)) !== null) {\n if (match[1]) {\n addMatch(match[1], 'import', getLineNumber(match.index));\n }\n }\n\n // Check spec assignments\n const assignPattern = new RegExp(SPEC_REFERENCE_PATTERNS.specAssignment);\n while ((match = assignPattern.exec(code)) !== null) {\n if (match[1]) {\n addMatch(match[1], 'unknown', getLineNumber(match.index));\n }\n }\n\n return matches;\n}\n\n/**\n * Default glob patterns for implementation files.\n */\nconst DEFAULT_INCLUDE_PATTERNS = ['**/*.ts', '**/*.tsx'];\n\nconst DEFAULT_EXCLUDE_PATTERNS = [\n '**/node_modules/**',\n '**/dist/**',\n '**/.git/**',\n '**/*.d.ts',\n '**/*.operation.ts', // Skip spec files themselves\n '**/*.spec.ts', // Skip test spec files\n '**/*.feature.ts', // Skip feature files\n '**/*.event.ts', // Skip event spec files\n '**/*.presentation.ts', // Skip presentation files\n];\n\n/**\n * Discover implementations that reference a specific spec.\n */\nexport async function discoverImplementationsForSpec(\n specKey: string,\n adapters: { fs: FsAdapter },\n options: DiscoveryOptions = {}\n): Promise<SpecReferenceMatch[]> {\n const { fs } = adapters;\n const includePatterns = options.includePatterns ?? DEFAULT_INCLUDE_PATTERNS;\n const excludePatterns = options.excludePatterns ?? DEFAULT_EXCLUDE_PATTERNS;\n\n const allMatches: SpecReferenceMatch[] = [];\n\n // Scan each include pattern\n for (const pattern of includePatterns) {\n const files = await fs.glob({ pattern, ignore: excludePatterns });\n\n for (const filePath of files) {\n try {\n const content = await fs.readFile(filePath);\n const references = extractSpecReferences(content, filePath);\n\n // Filter to only references for the target spec\n const matchingRefs = references.filter(\n (ref) => ref.specKey === specKey\n );\n allMatches.push(...matchingRefs);\n } catch {\n // Skip files that can't be read\n }\n }\n }\n\n return allMatches;\n}\n\n/**\n * Discover all spec references in the workspace.\n * Returns a map of spec key to implementation references.\n */\nexport async function discoverAllImplementations(\n adapters: { fs: FsAdapter },\n options: DiscoveryOptions = {}\n): Promise<Map<string, SpecReferenceMatch[]>> {\n const { fs } = adapters;\n const includePatterns = options.includePatterns ?? DEFAULT_INCLUDE_PATTERNS;\n const excludePatterns = options.excludePatterns ?? DEFAULT_EXCLUDE_PATTERNS;\n\n const specToImplementations = new Map<string, SpecReferenceMatch[]>();\n\n // Scan each include pattern\n for (const pattern of includePatterns) {\n const files = await fs.glob({ pattern, ignore: excludePatterns });\n\n for (const filePath of files) {\n try {\n const content = await fs.readFile(filePath);\n const references = extractSpecReferences(content, filePath);\n\n // Group by spec key\n for (const ref of references) {\n const existing = specToImplementations.get(ref.specKey) ?? [];\n existing.push(ref);\n specToImplementations.set(ref.specKey, existing);\n }\n } catch {\n // Skip files that can't be read\n }\n }\n }\n\n return specToImplementations;\n}\n"],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { DiscoveryOptions, ImplementationSource, ImplementationStatus, ResolvedImplementation, SpecImplementationResult, SpecReferenceMatch } from "./types.js";
2
+ import { discoverAllImplementations, discoverImplementationsForSpec, extractSpecReferences, inferImplementationType } from "./discovery.js";
3
+ import { ResolverOptions, getImplementationSummary, resolveAllImplementations, resolveImplementations } from "./resolver.js";
@@ -0,0 +1,2 @@
1
+ import { discoverAllImplementations, discoverImplementationsForSpec, extractSpecReferences, inferImplementationType } from "./discovery.js";
2
+ import { getImplementationSummary, resolveAllImplementations, resolveImplementations } from "./resolver.js";
@@ -0,0 +1,44 @@
1
+ import { FsAdapter } from "../../ports/fs.js";
2
+ import { DiscoveryOptions, SpecImplementationResult } from "./types.js";
3
+ import { WorkspaceConfig } from "@contractspec/module.workspace";
4
+
5
+ //#region src/services/implementation/resolver.d.ts
6
+
7
+ /**
8
+ * Options for resolving implementations.
9
+ */
10
+ interface ResolverOptions extends DiscoveryOptions {
11
+ /** Include explicit implementations from spec */
12
+ includeExplicit?: boolean;
13
+ /** Include auto-discovered implementations */
14
+ includeDiscovered?: boolean;
15
+ /** Include convention-based implementations */
16
+ includeConvention?: boolean;
17
+ /** Output directory for convention-based paths */
18
+ outputDir?: string;
19
+ }
20
+ /**
21
+ * Resolve all implementations for a spec file.
22
+ */
23
+ declare function resolveImplementations(specFile: string, adapters: {
24
+ fs: FsAdapter;
25
+ }, config: WorkspaceConfig, options?: ResolverOptions): Promise<SpecImplementationResult>;
26
+ /**
27
+ * Resolve implementations for multiple spec files.
28
+ */
29
+ declare function resolveAllImplementations(specFiles: string[], adapters: {
30
+ fs: FsAdapter;
31
+ }, config: WorkspaceConfig, options?: ResolverOptions): Promise<SpecImplementationResult[]>;
32
+ /**
33
+ * Get implementation summary statistics.
34
+ */
35
+ declare function getImplementationSummary(results: SpecImplementationResult[]): {
36
+ total: number;
37
+ implemented: number;
38
+ partial: number;
39
+ missing: number;
40
+ coverage: number;
41
+ };
42
+ //#endregion
43
+ export { ResolverOptions, getImplementationSummary, resolveAllImplementations, resolveImplementations };
44
+ //# sourceMappingURL=resolver.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolver.d.ts","names":[],"sources":["../../../src/services/implementation/resolver.ts"],"sourcesContent":[],"mappings":";;;;;;;;AAiVA;AAEkB,UAvTD,eAAA,SAAwB,gBAuTvB,CAAA;EACR;EACC,eAAA,CAAA,EAAA,OAAA;EACA;EAAR,iBAAA,CAAA,EAAA,OAAA;EAAO;EA2BM,iBAAA,CAAA,EAAA,OAAwB;;;;;;;iBAzNlB,sBAAA;MAEJ;WACR,2BACC,kBACR,QAAQ;;;;iBAoLW,yBAAA;MAEJ;WACR,2BACC,kBACR,QAAQ;;;;iBA2BK,wBAAA,UAAkC"}
@@ -0,0 +1,224 @@
1
+ import { discoverImplementationsForSpec } from "./discovery.js";
2
+ import { scanSpecSource } from "@contractspec/module.workspace";
3
+ import { createHash } from "crypto";
4
+
5
+ //#region src/services/implementation/resolver.ts
6
+ /**
7
+ * Implementation resolver service.
8
+ *
9
+ * Resolves all implementations for a spec by merging:
10
+ * 1. Explicit mappings from spec.implementations
11
+ * 2. Auto-discovered references from workspace scanning
12
+ * 3. Convention-based paths (naming conventions)
13
+ */
14
+ const DEFAULT_OPTIONS = {
15
+ includeExplicit: true,
16
+ includeDiscovered: true,
17
+ includeConvention: true,
18
+ computeHashes: true
19
+ };
20
+ /**
21
+ * Convert string to kebab-case.
22
+ */
23
+ function toKebabCase(value) {
24
+ return value.replace(/\./g, "-").replace(/([a-z])([A-Z])/g, "$1-$2").toLowerCase();
25
+ }
26
+ /**
27
+ * Compute SHA256 hash of content.
28
+ */
29
+ function computeHash(content) {
30
+ return createHash("sha256").update(content).digest("hex");
31
+ }
32
+ /**
33
+ * Get convention-based implementation paths for a spec.
34
+ */
35
+ function getConventionPaths(specType, specKey, outputDir) {
36
+ const kebab = toKebabCase(specKey);
37
+ const paths = [];
38
+ if (specType === "operation") {
39
+ paths.push({
40
+ path: `${outputDir}/handlers/${kebab}.handler.ts`,
41
+ type: "handler"
42
+ });
43
+ paths.push({
44
+ path: `${outputDir}/handlers/${kebab}.handler.test.ts`,
45
+ type: "test"
46
+ });
47
+ }
48
+ if (specType === "presentation") {
49
+ paths.push({
50
+ path: `${outputDir}/components/${kebab}.tsx`,
51
+ type: "component"
52
+ });
53
+ paths.push({
54
+ path: `${outputDir}/components/${kebab}.test.tsx`,
55
+ type: "test"
56
+ });
57
+ }
58
+ if (specType === "form") {
59
+ paths.push({
60
+ path: `${outputDir}/forms/${kebab}.form.tsx`,
61
+ type: "form"
62
+ });
63
+ paths.push({
64
+ path: `${outputDir}/forms/${kebab}.form.test.tsx`,
65
+ type: "test"
66
+ });
67
+ }
68
+ if (specType === "event") {
69
+ paths.push({
70
+ path: `${outputDir}/handlers/${kebab}.handler.ts`,
71
+ type: "handler"
72
+ });
73
+ paths.push({
74
+ path: `${outputDir}/handlers/${kebab}.handler.test.ts`,
75
+ type: "test"
76
+ });
77
+ }
78
+ return paths;
79
+ }
80
+ /**
81
+ * Determine overall implementation status.
82
+ */
83
+ function determineStatus(implementations) {
84
+ if (implementations.length === 0) return "missing";
85
+ const existingImpls = implementations.filter((i) => i.exists);
86
+ implementations.filter((i) => i.type !== "test");
87
+ if (existingImpls.filter((i) => i.type !== "test").length === 0) return "missing";
88
+ if (implementations.every((i) => i.exists)) return "implemented";
89
+ return "partial";
90
+ }
91
+ /**
92
+ * Resolve all implementations for a spec file.
93
+ */
94
+ async function resolveImplementations(specFile, adapters, config, options = {}) {
95
+ const opts = {
96
+ ...DEFAULT_OPTIONS,
97
+ ...options
98
+ };
99
+ const { fs } = adapters;
100
+ if (!await fs.exists(specFile)) throw new Error(`Spec file not found: ${specFile}`);
101
+ const specContent = await fs.readFile(specFile);
102
+ const specHash = opts.computeHashes ? computeHash(specContent) : void 0;
103
+ const scan = scanSpecSource(specContent, specFile);
104
+ const specKey = scan.key ?? fs.basename(specFile).replace(/\.[jt]s$/, "");
105
+ const specVersion = scan.version ?? 1;
106
+ const specType = scan.specType ?? "operation";
107
+ const implementations = [];
108
+ const seenPaths = /* @__PURE__ */ new Set();
109
+ const addImpl = async (path, type, source, description) => {
110
+ if (seenPaths.has(path)) return;
111
+ seenPaths.add(path);
112
+ const exists = await fs.exists(path);
113
+ let contentHash;
114
+ if (exists && opts.computeHashes) try {
115
+ contentHash = computeHash(await fs.readFile(path));
116
+ } catch {}
117
+ implementations.push({
118
+ path,
119
+ type,
120
+ source,
121
+ exists,
122
+ contentHash,
123
+ description
124
+ });
125
+ };
126
+ if (opts.includeExplicit) {
127
+ const explicitImpls = parseExplicitImplementations(specContent);
128
+ for (const impl of explicitImpls) await addImpl(impl.path, impl.type, "explicit", impl.description);
129
+ }
130
+ if (opts.includeDiscovered) {
131
+ const discovered = await discoverImplementationsForSpec(specKey, adapters, opts);
132
+ const specKeyVariants = getSpecKeyVariants(specKey);
133
+ for (const variant of specKeyVariants) {
134
+ const variantDiscovered = await discoverImplementationsForSpec(variant, adapters, opts);
135
+ discovered.push(...variantDiscovered);
136
+ }
137
+ for (const ref of discovered) {
138
+ if (ref.filePath === specFile) continue;
139
+ await addImpl(ref.filePath, ref.inferredType, "discovered");
140
+ }
141
+ }
142
+ if (opts.includeConvention) {
143
+ const conventionPaths = getConventionPaths(specType, specKey, opts.outputDir ?? config.outputDir ?? "./src");
144
+ for (const { path, type } of conventionPaths) await addImpl(path, type, "convention");
145
+ }
146
+ return {
147
+ specKey,
148
+ specVersion,
149
+ specPath: specFile,
150
+ specType,
151
+ implementations,
152
+ status: determineStatus(implementations),
153
+ specHash
154
+ };
155
+ }
156
+ /**
157
+ * Parse explicit implementations from spec source code.
158
+ * Looks for: implementations: [{ path: '...', type: '...' }]
159
+ */
160
+ function parseExplicitImplementations(code) {
161
+ const implementations = [];
162
+ const implMatch = code.match(/implementations\s*:\s*\[([\s\S]*?)\]/);
163
+ if (!implMatch) return implementations;
164
+ const implBlock = implMatch[1];
165
+ if (!implBlock) return implementations;
166
+ const objRegex = /\{\s*path\s*:\s*['"`]([^'"`]+)['"`]\s*,\s*type\s*:\s*['"`]([^'"`]+)['"`](?:\s*,\s*description\s*:\s*['"`]([^'"`]+)['"`])?\s*\}/g;
167
+ let match;
168
+ while ((match = objRegex.exec(implBlock)) !== null) if (match[1] && match[2]) implementations.push({
169
+ path: match[1],
170
+ type: match[2],
171
+ description: match[3]
172
+ });
173
+ return implementations;
174
+ }
175
+ /**
176
+ * Get common variants of a spec key for discovery.
177
+ */
178
+ function getSpecKeyVariants(specKey) {
179
+ const variants = [];
180
+ const base = specKey.replace(/Spec$/, "").replace(/Contract$/, "").replace(/Command$/, "").replace(/Query$/, "");
181
+ if (base !== specKey) {
182
+ variants.push(base);
183
+ variants.push(`${base}Spec`);
184
+ variants.push(`${base}Contract`);
185
+ }
186
+ const parts = specKey.split(".");
187
+ if (parts.length > 1) {
188
+ const pascalName = parts.map((p) => p.charAt(0).toUpperCase() + p.slice(1)).join("");
189
+ variants.push(pascalName);
190
+ }
191
+ return variants;
192
+ }
193
+ /**
194
+ * Resolve implementations for multiple spec files.
195
+ */
196
+ async function resolveAllImplementations(specFiles, adapters, config, options = {}) {
197
+ const results = [];
198
+ for (const specFile of specFiles) try {
199
+ const result = await resolveImplementations(specFile, adapters, config, options);
200
+ results.push(result);
201
+ } catch (error) {
202
+ console.error(`Failed to resolve implementations for ${specFile}:`, error);
203
+ }
204
+ return results;
205
+ }
206
+ /**
207
+ * Get implementation summary statistics.
208
+ */
209
+ function getImplementationSummary(results) {
210
+ const implemented = results.filter((r) => r.status === "implemented").length;
211
+ const partial = results.filter((r) => r.status === "partial").length;
212
+ const missing = results.filter((r) => r.status === "missing").length;
213
+ return {
214
+ total: results.length,
215
+ implemented,
216
+ partial,
217
+ missing,
218
+ coverage: results.length > 0 ? Math.round(implemented / results.length * 100) : 100
219
+ };
220
+ }
221
+
222
+ //#endregion
223
+ export { getImplementationSummary, resolveAllImplementations, resolveImplementations };
224
+ //# sourceMappingURL=resolver.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resolver.js","names":["DEFAULT_OPTIONS: ResolverOptions","paths: { path: string; type: ImplementationType }[]","implementations: ResolvedImplementation[]","contentHash: string | undefined","implementations: ImplementationRef[]","match: RegExpExecArray | null","variants: string[]","results: SpecImplementationResult[]"],"sources":["../../../src/services/implementation/resolver.ts"],"sourcesContent":["/**\n * Implementation resolver service.\n *\n * Resolves all implementations for a spec by merging:\n * 1. Explicit mappings from spec.implementations\n * 2. Auto-discovered references from workspace scanning\n * 3. Convention-based paths (naming conventions)\n */\n\nimport { createHash } from 'crypto';\nimport type { WorkspaceConfig } from '@contractspec/module.workspace';\nimport { scanSpecSource } from '@contractspec/module.workspace';\nimport type {\n ImplementationRef,\n ImplementationType,\n} from '@contractspec/lib.contracts';\nimport type { FsAdapter } from '../../ports/fs';\nimport type {\n DiscoveryOptions,\n ImplementationStatus,\n ResolvedImplementation,\n SpecImplementationResult,\n} from './types';\nimport { discoverImplementationsForSpec } from './discovery';\n\n/**\n * Options for resolving implementations.\n */\nexport interface ResolverOptions extends DiscoveryOptions {\n /** Include explicit implementations from spec */\n includeExplicit?: boolean;\n /** Include auto-discovered implementations */\n includeDiscovered?: boolean;\n /** Include convention-based implementations */\n includeConvention?: boolean;\n /** Output directory for convention-based paths */\n outputDir?: string;\n}\n\nconst DEFAULT_OPTIONS: ResolverOptions = {\n includeExplicit: true,\n includeDiscovered: true,\n includeConvention: true,\n computeHashes: true,\n};\n\n/**\n * Convert string to kebab-case.\n */\nfunction toKebabCase(value: string): string {\n return value\n .replace(/\\./g, '-')\n .replace(/([a-z])([A-Z])/g, '$1-$2')\n .toLowerCase();\n}\n\n/**\n * Compute SHA256 hash of content.\n */\nfunction computeHash(content: string): string {\n return createHash('sha256').update(content).digest('hex');\n}\n\n/**\n * Get convention-based implementation paths for a spec.\n */\nfunction getConventionPaths(\n specType: string,\n specKey: string,\n outputDir: string\n): { path: string; type: ImplementationType }[] {\n const kebab = toKebabCase(specKey);\n const paths: { path: string; type: ImplementationType }[] = [];\n\n if (specType === 'operation') {\n paths.push({\n path: `${outputDir}/handlers/${kebab}.handler.ts`,\n type: 'handler',\n });\n paths.push({\n path: `${outputDir}/handlers/${kebab}.handler.test.ts`,\n type: 'test',\n });\n }\n\n if (specType === 'presentation') {\n paths.push({\n path: `${outputDir}/components/${kebab}.tsx`,\n type: 'component',\n });\n paths.push({\n path: `${outputDir}/components/${kebab}.test.tsx`,\n type: 'test',\n });\n }\n\n if (specType === 'form') {\n paths.push({\n path: `${outputDir}/forms/${kebab}.form.tsx`,\n type: 'form',\n });\n paths.push({\n path: `${outputDir}/forms/${kebab}.form.test.tsx`,\n type: 'test',\n });\n }\n\n if (specType === 'event') {\n paths.push({\n path: `${outputDir}/handlers/${kebab}.handler.ts`,\n type: 'handler',\n });\n paths.push({\n path: `${outputDir}/handlers/${kebab}.handler.test.ts`,\n type: 'test',\n });\n }\n\n return paths;\n}\n\n/**\n * Determine overall implementation status.\n */\nfunction determineStatus(\n implementations: ResolvedImplementation[]\n): ImplementationStatus {\n if (implementations.length === 0) {\n return 'missing';\n }\n\n const existingImpls = implementations.filter((i) => i.exists);\n const _nonTestImpls = implementations.filter((i) => i.type !== 'test');\n const existingNonTestImpls = existingImpls.filter((i) => i.type !== 'test');\n\n // If no non-test implementations exist, it's missing\n if (existingNonTestImpls.length === 0) {\n return 'missing';\n }\n\n // If all expected implementations exist, it's fully implemented\n if (implementations.every((i) => i.exists)) {\n return 'implemented';\n }\n\n // Some exist, some don't\n return 'partial';\n}\n\n/**\n * Resolve all implementations for a spec file.\n */\nexport async function resolveImplementations(\n specFile: string,\n adapters: { fs: FsAdapter },\n config: WorkspaceConfig,\n options: ResolverOptions = {}\n): Promise<SpecImplementationResult> {\n const opts = { ...DEFAULT_OPTIONS, ...options };\n const { fs } = adapters;\n\n // Read and parse spec file\n const specExists = await fs.exists(specFile);\n if (!specExists) {\n throw new Error(`Spec file not found: ${specFile}`);\n }\n\n const specContent = await fs.readFile(specFile);\n const specHash = opts.computeHashes ? computeHash(specContent) : undefined;\n const scan = scanSpecSource(specContent, specFile);\n\n const specKey = scan.key ?? fs.basename(specFile).replace(/\\.[jt]s$/, '');\n const specVersion = scan.version ?? 1;\n const specType = scan.specType ?? 'operation';\n\n const implementations: ResolvedImplementation[] = [];\n const seenPaths = new Set<string>();\n\n // Helper to add unique implementations\n const addImpl = async (\n path: string,\n type: ImplementationType,\n source: ResolvedImplementation['source'],\n description?: string\n ) => {\n if (seenPaths.has(path)) return;\n seenPaths.add(path);\n\n const exists = await fs.exists(path);\n let contentHash: string | undefined;\n\n if (exists && opts.computeHashes) {\n try {\n const content = await fs.readFile(path);\n contentHash = computeHash(content);\n } catch {\n // Ignore hash computation errors\n }\n }\n\n implementations.push({\n path,\n type,\n source,\n exists,\n contentHash,\n description,\n });\n };\n\n // 1. Add explicit implementations from spec\n if (opts.includeExplicit) {\n // Parse explicit implementations from spec source\n const explicitImpls = parseExplicitImplementations(specContent);\n for (const impl of explicitImpls) {\n await addImpl(impl.path, impl.type, 'explicit', impl.description);\n }\n }\n\n // 2. Add auto-discovered implementations\n if (opts.includeDiscovered) {\n const discovered = await discoverImplementationsForSpec(\n specKey,\n adapters,\n opts\n );\n\n // Also search for spec key variants\n const specKeyVariants = getSpecKeyVariants(specKey);\n for (const variant of specKeyVariants) {\n const variantDiscovered = await discoverImplementationsForSpec(\n variant,\n adapters,\n opts\n );\n discovered.push(...variantDiscovered);\n }\n\n for (const ref of discovered) {\n // Skip the spec file itself\n if (ref.filePath === specFile) continue;\n\n await addImpl(ref.filePath, ref.inferredType, 'discovered');\n }\n }\n\n // 3. Add convention-based implementations\n if (opts.includeConvention) {\n const outputDir = opts.outputDir ?? config.outputDir ?? './src';\n const conventionPaths = getConventionPaths(specType, specKey, outputDir);\n\n for (const { path, type } of conventionPaths) {\n await addImpl(path, type, 'convention');\n }\n }\n\n // Determine overall status\n const status = determineStatus(implementations);\n\n return {\n specKey,\n specVersion,\n specPath: specFile,\n specType,\n implementations,\n status,\n specHash,\n };\n}\n\n/**\n * Parse explicit implementations from spec source code.\n * Looks for: implementations: [{ path: '...', type: '...' }]\n */\nfunction parseExplicitImplementations(code: string): ImplementationRef[] {\n const implementations: ImplementationRef[] = [];\n\n // Simple regex to find implementations array\n const implMatch = code.match(/implementations\\s*:\\s*\\[([\\s\\S]*?)\\]/);\n\n if (!implMatch) return implementations;\n\n const implBlock = implMatch[1];\n if (!implBlock) return implementations;\n\n // Find each { path: '...', type: '...' } object\n const objRegex =\n /\\{\\s*path\\s*:\\s*['\"`]([^'\"`]+)['\"`]\\s*,\\s*type\\s*:\\s*['\"`]([^'\"`]+)['\"`](?:\\s*,\\s*description\\s*:\\s*['\"`]([^'\"`]+)['\"`])?\\s*\\}/g;\n\n let match: RegExpExecArray | null;\n while ((match = objRegex.exec(implBlock)) !== null) {\n if (match[1] && match[2]) {\n implementations.push({\n path: match[1],\n type: match[2] as ImplementationType,\n description: match[3],\n });\n }\n }\n\n return implementations;\n}\n\n/**\n * Get common variants of a spec key for discovery.\n */\nfunction getSpecKeyVariants(specKey: string): string[] {\n const variants: string[] = [];\n\n // Remove common suffixes\n const base = specKey\n .replace(/Spec$/, '')\n .replace(/Contract$/, '')\n .replace(/Command$/, '')\n .replace(/Query$/, '');\n\n if (base !== specKey) {\n variants.push(base);\n variants.push(`${base}Spec`);\n variants.push(`${base}Contract`);\n }\n\n // Add PascalCase variant\n const parts = specKey.split('.');\n if (parts.length > 1) {\n const pascalName = parts\n .map((p) => p.charAt(0).toUpperCase() + p.slice(1))\n .join('');\n variants.push(pascalName);\n }\n\n return variants;\n}\n\n/**\n * Resolve implementations for multiple spec files.\n */\nexport async function resolveAllImplementations(\n specFiles: string[],\n adapters: { fs: FsAdapter },\n config: WorkspaceConfig,\n options: ResolverOptions = {}\n): Promise<SpecImplementationResult[]> {\n const results: SpecImplementationResult[] = [];\n\n for (const specFile of specFiles) {\n try {\n const result = await resolveImplementations(\n specFile,\n adapters,\n config,\n options\n );\n results.push(result);\n } catch (error) {\n // Log error but continue with other specs\n console.error(\n `Failed to resolve implementations for ${specFile}:`,\n error\n );\n }\n }\n\n return results;\n}\n\n/**\n * Get implementation summary statistics.\n */\nexport function getImplementationSummary(results: SpecImplementationResult[]): {\n total: number;\n implemented: number;\n partial: number;\n missing: number;\n coverage: number;\n} {\n const implemented = results.filter((r) => r.status === 'implemented').length;\n const partial = results.filter((r) => r.status === 'partial').length;\n const missing = results.filter((r) => r.status === 'missing').length;\n\n return {\n total: results.length,\n implemented,\n partial,\n missing,\n coverage:\n results.length > 0\n ? Math.round((implemented / results.length) * 100)\n : 100,\n };\n}\n"],"mappings":"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"}