@ontrails/warden 1.0.0-beta.13 → 1.0.0-beta.15

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 (474) hide show
  1. package/.turbo/turbo-lint.log +1 -1
  2. package/CHANGELOG.md +30 -0
  3. package/README.md +31 -20
  4. package/dist/cli.d.ts +19 -2
  5. package/dist/cli.d.ts.map +1 -1
  6. package/dist/cli.js +261 -64
  7. package/dist/cli.js.map +1 -1
  8. package/dist/draft.d.ts +5 -0
  9. package/dist/draft.d.ts.map +1 -0
  10. package/dist/draft.js +16 -0
  11. package/dist/draft.js.map +1 -0
  12. package/dist/drift.d.ts +10 -7
  13. package/dist/drift.d.ts.map +1 -1
  14. package/dist/drift.js +50 -16
  15. package/dist/drift.js.map +1 -1
  16. package/dist/formatters.d.ts +2 -1
  17. package/dist/formatters.d.ts.map +1 -1
  18. package/dist/formatters.js +15 -4
  19. package/dist/formatters.js.map +1 -1
  20. package/dist/index.d.ts +9 -17
  21. package/dist/index.d.ts.map +1 -1
  22. package/dist/index.js +10 -17
  23. package/dist/index.js.map +1 -1
  24. package/dist/rules/ast.d.ts +412 -7
  25. package/dist/rules/ast.d.ts.map +1 -1
  26. package/dist/rules/ast.js +1847 -102
  27. package/dist/rules/ast.js.map +1 -1
  28. package/dist/rules/circular-refs.d.ts +6 -0
  29. package/dist/rules/circular-refs.d.ts.map +1 -0
  30. package/dist/rules/circular-refs.js +83 -0
  31. package/dist/rules/circular-refs.js.map +1 -0
  32. package/dist/rules/context-no-surface-types.d.ts.map +1 -1
  33. package/dist/rules/context-no-surface-types.js +59 -3
  34. package/dist/rules/context-no-surface-types.js.map +1 -1
  35. package/dist/rules/contour-exists.d.ts +7 -0
  36. package/dist/rules/contour-exists.d.ts.map +1 -0
  37. package/dist/rules/contour-exists.js +113 -0
  38. package/dist/rules/contour-exists.js.map +1 -0
  39. package/dist/rules/contour-ids.d.ts +10 -0
  40. package/dist/rules/contour-ids.d.ts.map +1 -0
  41. package/dist/rules/contour-ids.js +12 -0
  42. package/dist/rules/contour-ids.js.map +1 -0
  43. package/dist/rules/cross-declarations.d.ts.map +1 -1
  44. package/dist/rules/cross-declarations.js +171 -57
  45. package/dist/rules/cross-declarations.js.map +1 -1
  46. package/dist/rules/dead-internal-trail.d.ts +3 -0
  47. package/dist/rules/dead-internal-trail.d.ts.map +1 -0
  48. package/dist/rules/dead-internal-trail.js +80 -0
  49. package/dist/rules/dead-internal-trail.js.map +1 -0
  50. package/dist/rules/draft-file-marking.d.ts +6 -0
  51. package/dist/rules/draft-file-marking.d.ts.map +1 -0
  52. package/dist/rules/draft-file-marking.js +87 -0
  53. package/dist/rules/draft-file-marking.js.map +1 -0
  54. package/dist/rules/draft-visible-debt.d.ts +12 -0
  55. package/dist/rules/draft-visible-debt.d.ts.map +1 -0
  56. package/dist/rules/draft-visible-debt.js +50 -0
  57. package/dist/rules/draft-visible-debt.js.map +1 -0
  58. package/dist/rules/error-mapping-completeness.d.ts +13 -0
  59. package/dist/rules/error-mapping-completeness.d.ts.map +1 -0
  60. package/dist/rules/error-mapping-completeness.js +160 -0
  61. package/dist/rules/error-mapping-completeness.js.map +1 -0
  62. package/dist/rules/example-valid.d.ts +6 -0
  63. package/dist/rules/example-valid.d.ts.map +1 -0
  64. package/dist/rules/example-valid.js +203 -0
  65. package/dist/rules/example-valid.js.map +1 -0
  66. package/dist/rules/fires-declarations.d.ts +16 -0
  67. package/dist/rules/fires-declarations.d.ts.map +1 -0
  68. package/dist/rules/fires-declarations.js +444 -0
  69. package/dist/rules/fires-declarations.js.map +1 -0
  70. package/dist/rules/implementation-returns-result.d.ts +9 -0
  71. package/dist/rules/implementation-returns-result.d.ts.map +1 -1
  72. package/dist/rules/implementation-returns-result.js +638 -76
  73. package/dist/rules/implementation-returns-result.js.map +1 -1
  74. package/dist/rules/incomplete-accessor-for-standard-op.d.ts +30 -0
  75. package/dist/rules/incomplete-accessor-for-standard-op.d.ts.map +1 -0
  76. package/dist/rules/incomplete-accessor-for-standard-op.js +226 -0
  77. package/dist/rules/incomplete-accessor-for-standard-op.js.map +1 -0
  78. package/dist/rules/incomplete-crud.d.ts +21 -0
  79. package/dist/rules/incomplete-crud.d.ts.map +1 -0
  80. package/dist/rules/incomplete-crud.js +368 -0
  81. package/dist/rules/incomplete-crud.js.map +1 -0
  82. package/dist/rules/index.d.ts +40 -7
  83. package/dist/rules/index.d.ts.map +1 -1
  84. package/dist/rules/index.js +91 -15
  85. package/dist/rules/index.js.map +1 -1
  86. package/dist/rules/intent-propagation.d.ts +3 -0
  87. package/dist/rules/intent-propagation.d.ts.map +1 -0
  88. package/dist/rules/intent-propagation.js +57 -0
  89. package/dist/rules/intent-propagation.js.map +1 -0
  90. package/dist/rules/missing-reconcile.d.ts +3 -0
  91. package/dist/rules/missing-reconcile.d.ts.map +1 -0
  92. package/dist/rules/missing-reconcile.js +44 -0
  93. package/dist/rules/missing-reconcile.js.map +1 -0
  94. package/dist/rules/missing-visibility.d.ts +3 -0
  95. package/dist/rules/missing-visibility.d.ts.map +1 -0
  96. package/dist/rules/missing-visibility.js +63 -0
  97. package/dist/rules/missing-visibility.js.map +1 -0
  98. package/dist/rules/no-direct-impl-in-route.d.ts.map +1 -1
  99. package/dist/rules/no-direct-impl-in-route.js +0 -3
  100. package/dist/rules/no-direct-impl-in-route.js.map +1 -1
  101. package/dist/rules/no-direct-implementation-call.js +1 -1
  102. package/dist/rules/no-direct-implementation-call.js.map +1 -1
  103. package/dist/rules/no-sync-result-assumption.d.ts.map +1 -1
  104. package/dist/rules/no-sync-result-assumption.js +870 -61
  105. package/dist/rules/no-sync-result-assumption.js.map +1 -1
  106. package/dist/rules/no-throw-in-detour-recover.d.ts +3 -0
  107. package/dist/rules/no-throw-in-detour-recover.d.ts.map +1 -0
  108. package/dist/rules/no-throw-in-detour-recover.js +147 -0
  109. package/dist/rules/no-throw-in-detour-recover.js.map +1 -0
  110. package/dist/rules/no-throw-in-detour-target.d.ts +4 -1
  111. package/dist/rules/no-throw-in-detour-target.d.ts.map +1 -1
  112. package/dist/rules/no-throw-in-detour-target.js +6 -3
  113. package/dist/rules/no-throw-in-detour-target.js.map +1 -1
  114. package/dist/rules/no-throw-in-implementation.d.ts +4 -2
  115. package/dist/rules/no-throw-in-implementation.d.ts.map +1 -1
  116. package/dist/rules/no-throw-in-implementation.js +6 -4
  117. package/dist/rules/no-throw-in-implementation.js.map +1 -1
  118. package/dist/rules/on-references-exist.d.ts +14 -0
  119. package/dist/rules/on-references-exist.d.ts.map +1 -0
  120. package/dist/rules/on-references-exist.js +109 -0
  121. package/dist/rules/on-references-exist.js.map +1 -0
  122. package/dist/rules/orphaned-signal.d.ts +3 -0
  123. package/dist/rules/orphaned-signal.d.ts.map +1 -0
  124. package/dist/rules/orphaned-signal.js +67 -0
  125. package/dist/rules/orphaned-signal.js.map +1 -0
  126. package/dist/rules/permit-governance.d.ts +3 -0
  127. package/dist/rules/permit-governance.d.ts.map +1 -0
  128. package/dist/rules/permit-governance.js +15 -0
  129. package/dist/rules/permit-governance.js.map +1 -0
  130. package/dist/rules/reference-exists.d.ts +6 -0
  131. package/dist/rules/reference-exists.d.ts.map +1 -0
  132. package/dist/rules/reference-exists.js +47 -0
  133. package/dist/rules/reference-exists.js.map +1 -0
  134. package/dist/rules/registry-names.d.ts +8 -0
  135. package/dist/rules/registry-names.d.ts.map +1 -0
  136. package/dist/rules/registry-names.js +83 -0
  137. package/dist/rules/registry-names.js.map +1 -0
  138. package/dist/rules/resource-declarations.d.ts +14 -0
  139. package/dist/rules/resource-declarations.d.ts.map +1 -0
  140. package/dist/rules/resource-declarations.js +413 -0
  141. package/dist/rules/resource-declarations.js.map +1 -0
  142. package/dist/rules/resource-exists.d.ts +6 -0
  143. package/dist/rules/resource-exists.d.ts.map +1 -0
  144. package/dist/rules/resource-exists.js +90 -0
  145. package/dist/rules/resource-exists.js.map +1 -0
  146. package/dist/rules/resource-id-grammar.d.ts +3 -0
  147. package/dist/rules/resource-id-grammar.d.ts.map +1 -0
  148. package/dist/rules/resource-id-grammar.js +39 -0
  149. package/dist/rules/resource-id-grammar.js.map +1 -0
  150. package/dist/rules/specs.d.ts.map +1 -1
  151. package/dist/rules/specs.js +5 -1
  152. package/dist/rules/specs.js.map +1 -1
  153. package/dist/rules/types.d.ts +53 -4
  154. package/dist/rules/types.d.ts.map +1 -1
  155. package/dist/rules/unreachable-detour-shadowing.d.ts +3 -0
  156. package/dist/rules/unreachable-detour-shadowing.d.ts.map +1 -0
  157. package/dist/rules/unreachable-detour-shadowing.js +202 -0
  158. package/dist/rules/unreachable-detour-shadowing.js.map +1 -0
  159. package/dist/rules/valid-describe-refs.d.ts.map +1 -1
  160. package/dist/rules/valid-describe-refs.js +132 -16
  161. package/dist/rules/valid-describe-refs.js.map +1 -1
  162. package/dist/rules/valid-detour-contract.d.ts +3 -0
  163. package/dist/rules/valid-detour-contract.d.ts.map +1 -0
  164. package/dist/rules/valid-detour-contract.js +47 -0
  165. package/dist/rules/valid-detour-contract.js.map +1 -0
  166. package/dist/rules/valid-detour-refs.d.ts.map +1 -1
  167. package/dist/rules/valid-detour-refs.js +73 -82
  168. package/dist/rules/valid-detour-refs.js.map +1 -1
  169. package/dist/rules/warden-export-symmetry.d.ts +7 -0
  170. package/dist/rules/warden-export-symmetry.d.ts.map +1 -0
  171. package/dist/rules/warden-export-symmetry.js +352 -0
  172. package/dist/rules/warden-export-symmetry.js.map +1 -0
  173. package/dist/rules/warden-rules-use-ast.d.ts +17 -0
  174. package/dist/rules/warden-rules-use-ast.d.ts.map +1 -0
  175. package/dist/rules/warden-rules-use-ast.js +778 -0
  176. package/dist/rules/warden-rules-use-ast.js.map +1 -0
  177. package/dist/trails/circular-refs.trail.d.ts +24 -0
  178. package/dist/trails/circular-refs.trail.d.ts.map +1 -0
  179. package/dist/trails/circular-refs.trail.js +29 -0
  180. package/dist/trails/circular-refs.trail.js.map +1 -0
  181. package/dist/trails/context-no-surface-types.trail.d.ts +2 -2
  182. package/dist/trails/context-no-surface-types.trail.d.ts.map +1 -1
  183. package/dist/trails/context-no-trailhead-types.trail.d.ts +2 -2
  184. package/dist/trails/context-no-trailhead-types.trail.d.ts.map +1 -1
  185. package/dist/trails/contour-exists.trail.d.ts +24 -0
  186. package/dist/trails/contour-exists.trail.d.ts.map +1 -0
  187. package/dist/trails/contour-exists.trail.js +21 -0
  188. package/dist/trails/contour-exists.trail.js.map +1 -0
  189. package/dist/trails/cross-declarations.trail.d.ts +2 -2
  190. package/dist/trails/cross-declarations.trail.d.ts.map +1 -1
  191. package/dist/trails/dead-internal-trail.trail.d.ts +24 -0
  192. package/dist/trails/dead-internal-trail.trail.d.ts.map +1 -0
  193. package/dist/trails/dead-internal-trail.trail.js +26 -0
  194. package/dist/trails/dead-internal-trail.trail.js.map +1 -0
  195. package/dist/trails/{provision-declarations.trail.d.ts → draft-file-marking.trail.d.ts} +3 -3
  196. package/dist/trails/draft-file-marking.trail.d.ts.map +1 -0
  197. package/dist/trails/draft-file-marking.trail.js +16 -0
  198. package/dist/trails/draft-file-marking.trail.js.map +1 -0
  199. package/dist/trails/draft-visible-debt.trail.d.ts +13 -0
  200. package/dist/trails/draft-visible-debt.trail.d.ts.map +1 -0
  201. package/dist/trails/draft-visible-debt.trail.js +16 -0
  202. package/dist/trails/draft-visible-debt.trail.js.map +1 -0
  203. package/dist/trails/error-mapping-completeness.trail.d.ts +13 -0
  204. package/dist/trails/error-mapping-completeness.trail.d.ts.map +1 -0
  205. package/dist/trails/error-mapping-completeness.trail.js +29 -0
  206. package/dist/trails/error-mapping-completeness.trail.js.map +1 -0
  207. package/dist/trails/{follow-declarations.trail.d.ts → example-valid.trail.d.ts} +3 -3
  208. package/dist/trails/example-valid.trail.d.ts.map +1 -0
  209. package/dist/trails/example-valid.trail.js +25 -0
  210. package/dist/trails/example-valid.trail.js.map +1 -0
  211. package/dist/trails/fires-declarations.trail.d.ts +13 -0
  212. package/dist/trails/fires-declarations.trail.d.ts.map +1 -0
  213. package/dist/trails/fires-declarations.trail.js +22 -0
  214. package/dist/trails/fires-declarations.trail.js.map +1 -0
  215. package/dist/trails/implementation-returns-result.trail.d.ts +2 -2
  216. package/dist/trails/implementation-returns-result.trail.d.ts.map +1 -1
  217. package/dist/trails/incomplete-accessor-for-standard-op.trail.d.ts +12 -0
  218. package/dist/trails/incomplete-accessor-for-standard-op.trail.d.ts.map +1 -0
  219. package/dist/trails/incomplete-accessor-for-standard-op.trail.js +60 -0
  220. package/dist/trails/incomplete-accessor-for-standard-op.trail.js.map +1 -0
  221. package/dist/trails/incomplete-crud.trail.d.ts +24 -0
  222. package/dist/trails/incomplete-crud.trail.d.ts.map +1 -0
  223. package/dist/trails/incomplete-crud.trail.js +39 -0
  224. package/dist/trails/incomplete-crud.trail.js.map +1 -0
  225. package/dist/trails/index.d.ts +29 -7
  226. package/dist/trails/index.d.ts.map +1 -1
  227. package/dist/trails/index.js +28 -6
  228. package/dist/trails/index.js.map +1 -1
  229. package/dist/trails/intent-propagation.trail.d.ts +24 -0
  230. package/dist/trails/intent-propagation.trail.d.ts.map +1 -0
  231. package/dist/trails/intent-propagation.trail.js +30 -0
  232. package/dist/trails/intent-propagation.trail.js.map +1 -0
  233. package/dist/trails/missing-reconcile.trail.d.ts +24 -0
  234. package/dist/trails/missing-reconcile.trail.d.ts.map +1 -0
  235. package/dist/trails/missing-reconcile.trail.js +33 -0
  236. package/dist/trails/missing-reconcile.trail.js.map +1 -0
  237. package/dist/trails/missing-visibility.trail.d.ts +24 -0
  238. package/dist/trails/missing-visibility.trail.d.ts.map +1 -0
  239. package/dist/trails/missing-visibility.trail.js +22 -0
  240. package/dist/trails/missing-visibility.trail.js.map +1 -0
  241. package/dist/trails/no-direct-impl-in-route.trail.d.ts +2 -2
  242. package/dist/trails/no-direct-impl-in-route.trail.d.ts.map +1 -1
  243. package/dist/trails/no-direct-implementation-call.trail.d.ts +2 -2
  244. package/dist/trails/no-direct-implementation-call.trail.d.ts.map +1 -1
  245. package/dist/trails/no-sync-result-assumption.trail.d.ts +2 -2
  246. package/dist/trails/no-sync-result-assumption.trail.d.ts.map +1 -1
  247. package/dist/trails/no-throw-in-detour-recover.trail.d.ts +13 -0
  248. package/dist/trails/no-throw-in-detour-recover.trail.d.ts.map +1 -0
  249. package/dist/trails/no-throw-in-detour-recover.trail.js +24 -0
  250. package/dist/trails/no-throw-in-detour-recover.trail.js.map +1 -0
  251. package/dist/trails/no-throw-in-detour-target.trail.d.ts +13 -3
  252. package/dist/trails/no-throw-in-detour-target.trail.d.ts.map +1 -1
  253. package/dist/trails/no-throw-in-implementation.trail.d.ts +2 -2
  254. package/dist/trails/no-throw-in-implementation.trail.d.ts.map +1 -1
  255. package/dist/trails/on-references-exist.trail.d.ts +24 -0
  256. package/dist/trails/on-references-exist.trail.d.ts.map +1 -0
  257. package/dist/trails/on-references-exist.trail.js +21 -0
  258. package/dist/trails/on-references-exist.trail.js.map +1 -0
  259. package/dist/trails/orphaned-signal.trail.d.ts +24 -0
  260. package/dist/trails/orphaned-signal.trail.d.ts.map +1 -0
  261. package/dist/trails/orphaned-signal.trail.js +36 -0
  262. package/dist/trails/orphaned-signal.trail.js.map +1 -0
  263. package/dist/trails/permit-governance.trail.d.ts +12 -0
  264. package/dist/trails/permit-governance.trail.d.ts.map +1 -0
  265. package/dist/trails/permit-governance.trail.js +47 -0
  266. package/dist/trails/permit-governance.trail.js.map +1 -0
  267. package/dist/trails/prefer-schema-inference.trail.d.ts +2 -2
  268. package/dist/trails/prefer-schema-inference.trail.d.ts.map +1 -1
  269. package/dist/trails/reference-exists.trail.d.ts +24 -0
  270. package/dist/trails/reference-exists.trail.d.ts.map +1 -0
  271. package/dist/trails/reference-exists.trail.js +25 -0
  272. package/dist/trails/reference-exists.trail.js.map +1 -0
  273. package/dist/trails/resource-declarations.trail.d.ts +13 -0
  274. package/dist/trails/resource-declarations.trail.d.ts.map +1 -0
  275. package/dist/trails/{provision-declarations.trail.js → resource-declarations.trail.js} +7 -7
  276. package/dist/trails/resource-declarations.trail.js.map +1 -0
  277. package/dist/trails/resource-exists.trail.d.ts +24 -0
  278. package/dist/trails/resource-exists.trail.d.ts.map +1 -0
  279. package/dist/trails/{provision-exists.trail.js → resource-exists.trail.js} +8 -8
  280. package/dist/trails/resource-exists.trail.js.map +1 -0
  281. package/dist/trails/resource-id-grammar.trail.d.ts +13 -0
  282. package/dist/trails/resource-id-grammar.trail.d.ts.map +1 -0
  283. package/dist/trails/resource-id-grammar.trail.js +38 -0
  284. package/dist/trails/resource-id-grammar.trail.js.map +1 -0
  285. package/dist/trails/run.d.ts +25 -9
  286. package/dist/trails/run.d.ts.map +1 -1
  287. package/dist/trails/run.js +63 -19
  288. package/dist/trails/run.js.map +1 -1
  289. package/dist/trails/schema.d.ts +28 -3
  290. package/dist/trails/schema.d.ts.map +1 -1
  291. package/dist/trails/schema.js +57 -4
  292. package/dist/trails/schema.js.map +1 -1
  293. package/dist/trails/unreachable-detour-shadowing.trail.d.ts +13 -0
  294. package/dist/trails/unreachable-detour-shadowing.trail.d.ts.map +1 -0
  295. package/dist/trails/unreachable-detour-shadowing.trail.js +44 -0
  296. package/dist/trails/unreachable-detour-shadowing.trail.js.map +1 -0
  297. package/dist/trails/valid-describe-refs.trail.d.ts +12 -3
  298. package/dist/trails/valid-describe-refs.trail.d.ts.map +1 -1
  299. package/dist/trails/valid-detour-contract.trail.d.ts +12 -0
  300. package/dist/trails/valid-detour-contract.trail.d.ts.map +1 -0
  301. package/dist/trails/valid-detour-contract.trail.js +66 -0
  302. package/dist/trails/valid-detour-contract.trail.js.map +1 -0
  303. package/dist/trails/valid-detour-refs.trail.d.ts +13 -3
  304. package/dist/trails/valid-detour-refs.trail.d.ts.map +1 -1
  305. package/dist/trails/warden-export-symmetry.trail.d.ts +13 -0
  306. package/dist/trails/warden-export-symmetry.trail.d.ts.map +1 -0
  307. package/dist/trails/warden-export-symmetry.trail.js +16 -0
  308. package/dist/trails/warden-export-symmetry.trail.js.map +1 -0
  309. package/dist/trails/warden-rules-use-ast.trail.d.ts +13 -0
  310. package/dist/trails/warden-rules-use-ast.trail.d.ts.map +1 -0
  311. package/dist/trails/warden-rules-use-ast.trail.js +41 -0
  312. package/dist/trails/warden-rules-use-ast.trail.js.map +1 -0
  313. package/dist/trails/wrap-rule.d.ts +16 -2
  314. package/dist/trails/wrap-rule.d.ts.map +1 -1
  315. package/dist/trails/wrap-rule.js +71 -11
  316. package/dist/trails/wrap-rule.js.map +1 -1
  317. package/package.json +7 -4
  318. package/src/__tests__/ast.test.ts +613 -0
  319. package/src/__tests__/circular-refs.test.ts +121 -0
  320. package/src/__tests__/cli.test.ts +360 -32
  321. package/src/__tests__/contour-exists.test.ts +203 -0
  322. package/src/__tests__/cross-declarations.test.ts +245 -0
  323. package/src/__tests__/dead-internal-trail.test.ts +81 -0
  324. package/src/__tests__/draft-rules-context.test.ts +150 -0
  325. package/src/__tests__/drift.test.ts +75 -5
  326. package/src/__tests__/error-mapping-completeness.test.ts +56 -0
  327. package/src/__tests__/example-valid.test.ts +101 -0
  328. package/src/__tests__/fires-declarations-param-destructure.test.ts +54 -0
  329. package/src/__tests__/fires-declarations.test.ts +652 -0
  330. package/src/__tests__/formatters.test.ts +2 -2
  331. package/src/__tests__/implementation-returns-result.test.ts +1016 -2
  332. package/src/__tests__/incomplete-accessor-for-standard-op.test.ts +337 -0
  333. package/src/__tests__/incomplete-crud.test.ts +498 -0
  334. package/src/__tests__/intent-propagation.test.ts +116 -0
  335. package/src/__tests__/missing-reconcile.test.ts +154 -0
  336. package/src/__tests__/missing-visibility.test.ts +108 -0
  337. package/src/__tests__/no-sync-result-assumption.test.ts +870 -39
  338. package/src/__tests__/no-throw-in-detour-recover.test.ts +93 -0
  339. package/src/__tests__/no-throw-in-implementation.test.ts +88 -0
  340. package/src/__tests__/on-references-exist.test.ts +151 -0
  341. package/src/__tests__/orphaned-signal.test.ts +137 -0
  342. package/src/__tests__/permit-governance.test.ts +66 -0
  343. package/src/__tests__/reference-exists.test.ts +281 -0
  344. package/src/__tests__/resource-declarations.test.ts +448 -0
  345. package/src/__tests__/resource-exists.test.ts +122 -0
  346. package/src/__tests__/resource-id-grammar.test.ts +50 -0
  347. package/src/__tests__/rules.test.ts +17 -77
  348. package/src/__tests__/topo-aware-rule.test.ts +257 -0
  349. package/src/__tests__/trails.test.ts +2 -2
  350. package/src/__tests__/unreachable-detour-shadowing.test.ts +128 -0
  351. package/src/__tests__/valid-describe-refs.test.ts +183 -0
  352. package/src/__tests__/valid-detour-contract.test.ts +86 -0
  353. package/src/__tests__/warden-export-symmetry.test.ts +251 -0
  354. package/src/__tests__/warden-rules-use-ast.test.ts +468 -0
  355. package/src/__tests__/wrap-rule.test.ts +3 -3
  356. package/src/cli.ts +458 -91
  357. package/src/draft.ts +22 -0
  358. package/src/drift.ts +63 -21
  359. package/src/formatters.ts +15 -4
  360. package/src/index.ts +62 -23
  361. package/src/rules/ast.ts +2715 -119
  362. package/src/rules/circular-refs.ts +154 -0
  363. package/src/rules/{context-no-trailhead-types.ts → context-no-surface-types.ts} +72 -12
  364. package/src/rules/contour-exists.ts +251 -0
  365. package/src/rules/contour-ids.ts +15 -0
  366. package/src/rules/cross-declarations.ts +277 -69
  367. package/src/rules/dead-internal-trail.ts +141 -0
  368. package/src/rules/draft-file-marking.ts +160 -0
  369. package/src/rules/draft-visible-debt.ts +87 -0
  370. package/src/rules/error-mapping-completeness.ts +273 -0
  371. package/src/rules/example-valid.ts +401 -0
  372. package/src/rules/fires-declarations.ts +609 -0
  373. package/src/rules/implementation-returns-result.ts +1042 -122
  374. package/src/rules/incomplete-accessor-for-standard-op.ts +315 -0
  375. package/src/rules/incomplete-crud.ts +579 -0
  376. package/src/rules/index.ts +95 -16
  377. package/src/rules/intent-propagation.ts +142 -0
  378. package/src/rules/missing-reconcile.ts +98 -0
  379. package/src/rules/missing-visibility.ts +110 -0
  380. package/src/rules/no-direct-impl-in-route.ts +0 -4
  381. package/src/rules/no-direct-implementation-call.ts +1 -1
  382. package/src/rules/no-sync-result-assumption.ts +1134 -96
  383. package/src/rules/no-throw-in-detour-recover.ts +225 -0
  384. package/src/rules/no-throw-in-implementation.ts +6 -4
  385. package/src/rules/on-references-exist.ts +194 -0
  386. package/src/rules/orphaned-signal.ts +150 -0
  387. package/src/rules/permit-governance.ts +25 -0
  388. package/src/rules/reference-exists.ts +98 -0
  389. package/src/rules/registry-names.ts +83 -0
  390. package/src/rules/{provision-declarations.ts → resource-declarations.ts} +208 -138
  391. package/src/rules/{provision-exists.ts → resource-exists.ts} +48 -51
  392. package/src/rules/resource-id-grammar.ts +65 -0
  393. package/src/rules/specs.ts +5 -1
  394. package/src/rules/types.ts +57 -4
  395. package/src/rules/unreachable-detour-shadowing.ts +375 -0
  396. package/src/rules/valid-describe-refs.ts +160 -32
  397. package/src/rules/valid-detour-contract.ts +78 -0
  398. package/src/rules/warden-export-symmetry.ts +533 -0
  399. package/src/rules/warden-rules-use-ast.ts +996 -0
  400. package/src/trails/circular-refs.trail.ts +29 -0
  401. package/src/trails/{context-no-trailhead-types.trail.ts → context-no-surface-types.trail.ts} +4 -4
  402. package/src/trails/contour-exists.trail.ts +21 -0
  403. package/src/trails/dead-internal-trail.trail.ts +26 -0
  404. package/src/trails/draft-file-marking.trail.ts +16 -0
  405. package/src/trails/draft-visible-debt.trail.ts +16 -0
  406. package/src/trails/error-mapping-completeness.trail.ts +29 -0
  407. package/src/trails/example-valid.trail.ts +25 -0
  408. package/src/trails/fires-declarations.trail.ts +22 -0
  409. package/src/trails/incomplete-accessor-for-standard-op.trail.ts +76 -0
  410. package/src/trails/incomplete-crud.trail.ts +39 -0
  411. package/src/trails/index.ts +40 -7
  412. package/src/trails/intent-propagation.trail.ts +30 -0
  413. package/src/trails/missing-reconcile.trail.ts +33 -0
  414. package/src/trails/missing-visibility.trail.ts +22 -0
  415. package/src/trails/no-throw-in-detour-recover.trail.ts +24 -0
  416. package/src/trails/on-references-exist.trail.ts +21 -0
  417. package/src/trails/orphaned-signal.trail.ts +36 -0
  418. package/src/trails/permit-governance.trail.ts +51 -0
  419. package/src/trails/reference-exists.trail.ts +25 -0
  420. package/src/trails/{provision-declarations.trail.ts → resource-declarations.trail.ts} +6 -6
  421. package/src/trails/{provision-exists.trail.ts → resource-exists.trail.ts} +7 -7
  422. package/src/trails/resource-id-grammar.trail.ts +39 -0
  423. package/src/trails/run.ts +121 -24
  424. package/src/trails/schema.ts +66 -4
  425. package/src/trails/unreachable-detour-shadowing.trail.ts +45 -0
  426. package/src/trails/valid-detour-contract.trail.ts +71 -0
  427. package/src/trails/warden-export-symmetry.trail.ts +16 -0
  428. package/src/trails/warden-rules-use-ast.trail.ts +45 -0
  429. package/src/trails/wrap-rule.ts +104 -12
  430. package/tsconfig.tests.json +10 -0
  431. package/tsconfig.tsbuildinfo +1 -1
  432. package/dist/rules/follow-declarations.d.ts +0 -13
  433. package/dist/rules/follow-declarations.d.ts.map +0 -1
  434. package/dist/rules/follow-declarations.js +0 -264
  435. package/dist/rules/follow-declarations.js.map +0 -1
  436. package/dist/rules/provision-declarations.d.ts +0 -14
  437. package/dist/rules/provision-declarations.d.ts.map +0 -1
  438. package/dist/rules/provision-declarations.js +0 -344
  439. package/dist/rules/provision-declarations.js.map +0 -1
  440. package/dist/rules/provision-exists.d.ts +0 -6
  441. package/dist/rules/provision-exists.d.ts.map +0 -1
  442. package/dist/rules/provision-exists.js +0 -89
  443. package/dist/rules/provision-exists.js.map +0 -1
  444. package/dist/rules/service-declarations.d.ts +0 -16
  445. package/dist/rules/service-declarations.d.ts.map +0 -1
  446. package/dist/rules/service-declarations.js +0 -346
  447. package/dist/rules/service-declarations.js.map +0 -1
  448. package/dist/rules/service-exists.d.ts +0 -8
  449. package/dist/rules/service-exists.d.ts.map +0 -1
  450. package/dist/rules/service-exists.js +0 -91
  451. package/dist/rules/service-exists.js.map +0 -1
  452. package/dist/trails/follow-declarations.trail.d.ts.map +0 -1
  453. package/dist/trails/follow-declarations.trail.js +0 -22
  454. package/dist/trails/follow-declarations.trail.js.map +0 -1
  455. package/dist/trails/provision-declarations.trail.d.ts.map +0 -1
  456. package/dist/trails/provision-declarations.trail.js.map +0 -1
  457. package/dist/trails/provision-exists.trail.d.ts +0 -15
  458. package/dist/trails/provision-exists.trail.d.ts.map +0 -1
  459. package/dist/trails/provision-exists.trail.js.map +0 -1
  460. package/dist/trails/service-declarations.trail.d.ts +0 -26
  461. package/dist/trails/service-declarations.trail.d.ts.map +0 -1
  462. package/dist/trails/service-declarations.trail.js +0 -27
  463. package/dist/trails/service-declarations.trail.js.map +0 -1
  464. package/dist/trails/service-exists.trail.d.ts +0 -32
  465. package/dist/trails/service-exists.trail.d.ts.map +0 -1
  466. package/dist/trails/service-exists.trail.js +0 -29
  467. package/dist/trails/service-exists.trail.js.map +0 -1
  468. package/src/__tests__/no-throw-in-detour-target.test.ts +0 -78
  469. package/src/__tests__/provision-declarations.test.ts +0 -318
  470. package/src/__tests__/provision-exists.test.ts +0 -122
  471. package/src/rules/no-throw-in-detour-target.ts +0 -150
  472. package/src/rules/valid-detour-refs.ts +0 -187
  473. package/src/trails/no-throw-in-detour-target.trail.ts +0 -20
  474. package/src/trails/valid-detour-refs.trail.ts +0 -24
@@ -0,0 +1,609 @@
1
+ /**
2
+ * Validates that `ctx.fire()` calls match the declared `fires` array.
3
+ *
4
+ * Statically analyzes trail `blaze` functions to find `ctx.fire('signalId', ...)`
5
+ * calls and compares them against the `fires: [...]` declaration in the trail
6
+ * config. Reports errors for undeclared fires and warnings for unused ones.
7
+ *
8
+ * Mirrors `cross-declarations` structurally — same extraction, same reporting
9
+ * shape, same const-identifier resolution, same context-parameter handling.
10
+ */
11
+
12
+ import {
13
+ extractFirstStringArg,
14
+ extractStringLiteral,
15
+ findConfigProperty,
16
+ findBlazeBodies,
17
+ findTrailDefinitions,
18
+ identifierName,
19
+ offsetToLine,
20
+ parse,
21
+ deriveConstString,
22
+ walkScope,
23
+ } from './ast.js';
24
+ import type { AstNode } from './ast.js';
25
+ import { isTestFile } from './scan.js';
26
+ import type { WardenDiagnostic, WardenRule } from './types.js';
27
+
28
+ // ---------------------------------------------------------------------------
29
+ // Const identifier resolution
30
+ // ---------------------------------------------------------------------------
31
+
32
+ /** Try to resolve an Identifier element to a string via const declaration. */
33
+ const resolveIdentifierElement = (
34
+ el: AstNode,
35
+ sourceCode: string
36
+ ): string | null => {
37
+ const name = identifierName(el);
38
+ if (!name) {
39
+ return null;
40
+ }
41
+ return deriveConstString(name, sourceCode);
42
+ };
43
+
44
+ /**
45
+ * Resolve an array element to a static signal ID when possible.
46
+ *
47
+ * Returns null for entries the rule can't statically resolve — callers should
48
+ * treat "unresolved" as "trust the runtime" rather than a missing declaration.
49
+ * In particular, object-form references (e.g. `fires: [orderPlaced]` where
50
+ * `orderPlaced` is a `Signal` imported from elsewhere) resolve via runtime
51
+ * normalization in `trail()`, not at lint time.
52
+ */
53
+ const resolveFireElementId = (
54
+ element: AstNode,
55
+ sourceCode: string
56
+ ): string | null => {
57
+ const literalValue = extractStringLiteral(element);
58
+ if (literalValue !== null) {
59
+ return literalValue;
60
+ }
61
+
62
+ if (element.type === 'Identifier') {
63
+ return resolveIdentifierElement(element, sourceCode);
64
+ }
65
+
66
+ return null;
67
+ };
68
+
69
+ // ---------------------------------------------------------------------------
70
+ // Declared fires extraction
71
+ // ---------------------------------------------------------------------------
72
+
73
+ /** Extract the ArrayExpression elements from a config's `fires` property. */
74
+ const getFiresElements = (config: AstNode): readonly AstNode[] | null => {
75
+ const firesProp = findConfigProperty(config, 'fires');
76
+ if (!firesProp) {
77
+ return null;
78
+ }
79
+
80
+ const arrayNode = firesProp.value;
81
+ if (!arrayNode || (arrayNode as AstNode).type !== 'ArrayExpression') {
82
+ return null;
83
+ }
84
+
85
+ const elements = (arrayNode as AstNode)['elements'] as
86
+ | readonly AstNode[]
87
+ | undefined;
88
+ return elements ?? null;
89
+ };
90
+
91
+ interface DeclaredFires {
92
+ /** Statically resolved signal ids from string literals / const identifiers. */
93
+ readonly ids: ReadonlySet<string>;
94
+ /** True if any element could not be statically resolved (e.g. Signal value). */
95
+ readonly hasUnresolved: boolean;
96
+ }
97
+
98
+ /**
99
+ * Extract declared fires from a `fires: [...]` array.
100
+ *
101
+ * Object-form entries (`fires: [someSignal]`) cannot be resolved at lint time;
102
+ * they're normalized at runtime by `trail()`. When any entry is unresolved,
103
+ * the rule reports `hasUnresolved: true`, and callers should suppress the
104
+ * "undeclared" diagnostic since the declared set is incomplete from our view.
105
+ */
106
+ const resolveDeclaredFiresElements = (
107
+ elements: readonly AstNode[],
108
+ sourceCode: string
109
+ ): DeclaredFires => {
110
+ const ids = new Set<string>();
111
+ let hasUnresolved = false;
112
+ for (const element of elements) {
113
+ const resolved = resolveFireElementId(element, sourceCode);
114
+ if (resolved) {
115
+ ids.add(resolved);
116
+ } else {
117
+ hasUnresolved = true;
118
+ }
119
+ }
120
+ return { hasUnresolved, ids };
121
+ };
122
+
123
+ const extractDeclaredFires = (
124
+ config: AstNode,
125
+ sourceCode: string
126
+ ): DeclaredFires => {
127
+ const elements = getFiresElements(config);
128
+ return elements
129
+ ? resolveDeclaredFiresElements(elements, sourceCode)
130
+ : { hasUnresolved: false, ids: new Set() };
131
+ };
132
+
133
+ // ---------------------------------------------------------------------------
134
+ // Called fires extraction — member expression helpers
135
+ // ---------------------------------------------------------------------------
136
+
137
+ const MEMBER_TYPES = new Set(['StaticMemberExpression', 'MemberExpression']);
138
+
139
+ /** Extract object and property Identifier names from a MemberExpression. */
140
+ const extractMemberPair = (
141
+ callee: AstNode
142
+ ): { objName: string; propName: string } | null => {
143
+ if (!MEMBER_TYPES.has(callee.type)) {
144
+ return null;
145
+ }
146
+
147
+ const objName = identifierName(
148
+ (callee as unknown as { object?: AstNode }).object
149
+ );
150
+ const propName = identifierName(
151
+ (callee as unknown as { property?: AstNode }).property
152
+ );
153
+
154
+ return objName && propName ? { objName, propName } : null;
155
+ };
156
+
157
+ /**
158
+ * Extract the second parameter node from a blaze function node.
159
+ *
160
+ * Handles `(input, ctx) => ...`, `async (input, context) => ...`,
161
+ * `function(input, ctx) { ... }`, and parameter-level destructuring
162
+ * like `(input, { fire }) => ...`.
163
+ */
164
+ const extractContextParamNode = (blazeBody: AstNode): AstNode | null => {
165
+ const params = blazeBody['params'] as readonly AstNode[] | undefined;
166
+ if (!params || params.length < 2) {
167
+ return null;
168
+ }
169
+ return params[1] ?? null;
170
+ };
171
+
172
+ /** Extract the local name bound to `fire` inside an ObjectPattern Property. */
173
+ const extractFireLocalName = (prop: AstNode): string | null => {
174
+ if (prop.type !== 'Property') {
175
+ return null;
176
+ }
177
+ const { key } = prop as unknown as { key?: AstNode };
178
+ const { value } = prop as unknown as { value?: AstNode };
179
+ const keyName = identifierName(key);
180
+ if (keyName !== 'fire') {
181
+ return null;
182
+ }
183
+ // `{ fire }` → key and value are the same Identifier (shorthand).
184
+ // `{ fire: emit }` → value is a distinct Identifier.
185
+ return identifierName(value) ?? keyName;
186
+ };
187
+
188
+ /** Collect `fire` local names from an ObjectPattern's properties into `names`. */
189
+ const collectFireNamesFromPattern = (
190
+ pattern: AstNode,
191
+ names: Set<string>
192
+ ): void => {
193
+ const { properties } = pattern as unknown as {
194
+ properties?: readonly AstNode[];
195
+ };
196
+ if (!properties) {
197
+ return;
198
+ }
199
+ for (const prop of properties) {
200
+ const localName = extractFireLocalName(prop);
201
+ if (localName) {
202
+ names.add(localName);
203
+ }
204
+ }
205
+ };
206
+
207
+ /**
208
+ * Extract the second parameter name from a blaze function node.
209
+ *
210
+ * Returns null when the parameter is not a plain Identifier (e.g. when the
211
+ * author destructures `{ fire }` in the parameter list). Parameter-level
212
+ * destructuring is handled separately by `collectParamFireNames`.
213
+ *
214
+ * Also handles defaulted parameters like `(input, ctx = fallback) => ...`
215
+ * (AssignmentPattern whose `.left` is the Identifier). Without this, valid
216
+ * signatures would silently drop out of ctx-access analysis.
217
+ */
218
+ const extractContextParamName = (blazeBody: AstNode): string | null => {
219
+ const param = extractContextParamNode(blazeBody);
220
+ if (!param) {
221
+ return null;
222
+ }
223
+ if (param.type === 'AssignmentPattern') {
224
+ const { left } = param as unknown as { left?: AstNode };
225
+ return identifierName(left);
226
+ }
227
+ return identifierName(param);
228
+ };
229
+
230
+ /**
231
+ * Collect `fire` local names bound via parameter-level destructuring.
232
+ *
233
+ * Recognizes `(input, { fire }) => ...` and `(input, { fire: emit }) => ...`.
234
+ * When the blaze author destructures in the parameter list, there is no
235
+ * enclosing `ctx` identifier to track — we seed the fire local set directly
236
+ * from the ObjectPattern in `params[1]`.
237
+ */
238
+ const collectParamFireNames = (body: AstNode): ReadonlySet<string> => {
239
+ const param = extractContextParamNode(body);
240
+ if (!param || param.type !== 'ObjectPattern') {
241
+ return new Set();
242
+ }
243
+ const names = new Set<string>();
244
+ collectFireNamesFromPattern(param, names);
245
+ return names;
246
+ };
247
+
248
+ /** Check if a callee is a member-style fire call: <ctxName>.fire(...). */
249
+ const isMemberFireCall = (
250
+ callee: AstNode,
251
+ ctxNames: ReadonlySet<string>
252
+ ): boolean => {
253
+ const pair = extractMemberPair(callee);
254
+ return !!pair && ctxNames.has(pair.objName) && pair.propName === 'fire';
255
+ };
256
+
257
+ /**
258
+ * Check if a node is a `<ctxName>.fire(...)` call and return the string signal ID.
259
+ *
260
+ * Also matches bare `<fireLocalName>(...)` calls, but only when the local name
261
+ * was verifiably destructured from the trail context (e.g. `const { fire } = ctx`
262
+ * or `const { fire: emit } = ctx`). Unrelated local `fire()` helpers are
263
+ * ignored — see `collectDestructuredFireNames`.
264
+ */
265
+ const isTrackedFireCallee = (
266
+ callee: AstNode,
267
+ ctxNames: ReadonlySet<string>,
268
+ fireLocalNames: ReadonlySet<string>
269
+ ): boolean => {
270
+ if (isMemberFireCall(callee, ctxNames)) {
271
+ return true;
272
+ }
273
+ const calleeName = identifierName(callee);
274
+ return !!calleeName && fireLocalNames.has(calleeName);
275
+ };
276
+
277
+ const extractFireCallId = (
278
+ node: AstNode,
279
+ ctxNames: ReadonlySet<string>,
280
+ fireLocalNames: ReadonlySet<string>
281
+ ): string | null => {
282
+ if (node.type !== 'CallExpression') {
283
+ return null;
284
+ }
285
+ const callee = node['callee'] as AstNode | undefined;
286
+ if (!callee) {
287
+ return null;
288
+ }
289
+ return isTrackedFireCallee(callee, ctxNames, fireLocalNames)
290
+ ? extractFirstStringArg(node)
291
+ : null;
292
+ };
293
+
294
+ /**
295
+ * Walk a blaze body and collect local names bound to `ctx.fire` via destructure.
296
+ *
297
+ * Recognizes:
298
+ * - `const { fire } = ctx;` → adds `fire`
299
+ * - `const { fire: emit } = context;` → adds `emit`
300
+ *
301
+ * Only destructures whose init is one of the tracked ctx parameter names are
302
+ * accepted. This prevents unrelated local `fire` helpers from being treated as
303
+ * calls into the trail context.
304
+ */
305
+ /** Check if a VariableDeclarator destructures from a known ctx identifier. */
306
+ const getCtxDestructurePattern = (
307
+ node: AstNode,
308
+ ctxNames: ReadonlySet<string>
309
+ ): AstNode | null => {
310
+ if (node.type !== 'VariableDeclarator') {
311
+ return null;
312
+ }
313
+ const { id, init } = node as unknown as {
314
+ readonly id?: AstNode;
315
+ readonly init?: AstNode;
316
+ };
317
+ if (!id || id.type !== 'ObjectPattern' || !init) {
318
+ return null;
319
+ }
320
+ const initName = identifierName(init);
321
+ if (!initName || !ctxNames.has(initName)) {
322
+ return null;
323
+ }
324
+ return id;
325
+ };
326
+
327
+ /**
328
+ * Collect `fire` local names destructured from ctx at the TOP LEVEL of the
329
+ * blaze body. Destructures inside nested functions are intentionally ignored
330
+ * to avoid leaking nested-scope bindings into the outer blaze scope — a
331
+ * `const { fire } = ctx` inside a nested helper should not cause an outer
332
+ * bare `fire('x')` to be treated as a ctx-bound call.
333
+ *
334
+ * Tradeoff: nested-scope destructures lose tracking entirely. Calls inside
335
+ * nested functions that rely on their own destructure will not be analyzed.
336
+ * This is a conservative precision loss; a full scope walker is a follow-up.
337
+ *
338
+ * Tradeoff: only `const` destructures are tracked. `let` and `var` bindings
339
+ * allow reassignment (`let { fire } = ctx; fire = other; fire('x')`) which
340
+ * this flow-insensitive walker cannot follow. Skipping them trades a small
341
+ * amount of precision — `let { fire } = ctx` is rare — for eliminating a
342
+ * class of false positives. The runtime + signal-id cross-check still
343
+ * validate real undeclared fires.
344
+ */
345
+ /** Get the top-level statements of a blaze function's BlockStatement body. */
346
+ const getTopLevelStatements = (body: AstNode): readonly AstNode[] => {
347
+ const blockBody = (body as unknown as { body?: AstNode }).body;
348
+ if (!blockBody || blockBody.type !== 'BlockStatement') {
349
+ return [];
350
+ }
351
+ return (blockBody as unknown as { body?: readonly AstNode[] }).body ?? [];
352
+ };
353
+
354
+ /** Collect fire-local names from a single top-level VariableDeclaration. */
355
+ const collectFireNamesFromDeclaration = (
356
+ stmt: AstNode,
357
+ ctxNames: ReadonlySet<string>,
358
+ names: Set<string>
359
+ ): void => {
360
+ if (stmt.type !== 'VariableDeclaration') {
361
+ return;
362
+ }
363
+ // Only track `const` destructures. `let` and `var` allow reassignment that
364
+ // a single-pass walker cannot track, so `let { fire } = ctx; fire = other;
365
+ // fire('x')` would otherwise be a false positive. Skipping non-const is a
366
+ // small precision loss (see TSDoc on `collectDestructuredFireNames`) in
367
+ // exchange for eliminating that class of false positives.
368
+ const { kind } = stmt as unknown as { kind?: string };
369
+ if (kind !== 'const') {
370
+ return;
371
+ }
372
+ const declarations =
373
+ (stmt as unknown as { declarations?: readonly AstNode[] }).declarations ??
374
+ [];
375
+ for (const decl of declarations) {
376
+ const pattern = getCtxDestructurePattern(decl, ctxNames);
377
+ if (pattern) {
378
+ collectFireNamesFromPattern(pattern, names);
379
+ }
380
+ }
381
+ };
382
+
383
+ const collectDestructuredFireNames = (
384
+ body: AstNode,
385
+ ctxNames: ReadonlySet<string>
386
+ ): ReadonlySet<string> => {
387
+ const names = new Set<string>();
388
+ for (const stmt of getTopLevelStatements(body)) {
389
+ collectFireNamesFromDeclaration(stmt, ctxNames, names);
390
+ }
391
+ return names;
392
+ };
393
+
394
+ /**
395
+ * Build the set of context parameter names to match against.
396
+ *
397
+ * Returns ONLY the actual second-parameter name from the blaze signature.
398
+ * No seeded defaults: if the blaze has no second parameter, the returned set
399
+ * is empty and no `ctx.fire(...)` / `context.fire(...)` calls are tracked
400
+ * for that blaze. An unrelated closure-scoped `ctx` identifier is not the
401
+ * trail context and must not be treated as one.
402
+ */
403
+ const buildCtxNames = (body: AstNode): ReadonlySet<string> => {
404
+ const ctxNames = new Set<string>();
405
+ const paramName = extractContextParamName(body);
406
+ if (paramName) {
407
+ ctxNames.add(paramName);
408
+ }
409
+ return ctxNames;
410
+ };
411
+
412
+ /**
413
+ * Walk blaze bodies and collect all statically resolvable ctx.fire() signal IDs.
414
+ *
415
+ * Traversal uses `walkScope`, which stops at nested function boundaries
416
+ * (FunctionDeclaration, FunctionExpression, ArrowFunctionExpression). This
417
+ * mirrors the top-level-only behavior of `collectDestructuredFireNames` and
418
+ * avoids false positives when a nested function parameter shadows `ctx` or a
419
+ * destructured `fire` local:
420
+ *
421
+ * ```ts
422
+ * blaze: async (_, ctx) => {
423
+ * const { fire } = ctx;
424
+ * function nested(fire) { fire('shadow'); } // ignored — shadowed
425
+ * function other(ctx) { ctx.fire('x'); } // ignored — shadowed
426
+ * return Result.ok({});
427
+ * }
428
+ * ```
429
+ *
430
+ * Tradeoff: legitimate helper-scoped fire calls are not statically analyzed
431
+ * today. This includes both direct `ctx.fire(...)` inside a nested helper and
432
+ * helper-local destructures like `const { fire } = ctx` inside that helper.
433
+ * The runtime + signal-id cross-check still validate them; the warden just
434
+ * can't prove them at lint time. A fuller helper-aware scope walker remains
435
+ * follow-up work if this precision loss becomes meaningful in practice.
436
+ */
437
+ const extractCalledFires = (config: AstNode): ReadonlySet<string> => {
438
+ const ids = new Set<string>();
439
+
440
+ for (const body of findBlazeBodies(config)) {
441
+ const ctxNames = buildCtxNames(body);
442
+ const bodyFireNames = collectDestructuredFireNames(body, ctxNames);
443
+ const paramFireNames = collectParamFireNames(body);
444
+ const fireLocalNames = new Set<string>([
445
+ ...bodyFireNames,
446
+ ...paramFireNames,
447
+ ]);
448
+
449
+ walkScope(body, (node) => {
450
+ const id = extractFireCallId(node, ctxNames, fireLocalNames);
451
+ if (id) {
452
+ ids.add(id);
453
+ }
454
+ });
455
+ }
456
+
457
+ return ids;
458
+ };
459
+
460
+ // ---------------------------------------------------------------------------
461
+ // Diagnostic builders
462
+ // ---------------------------------------------------------------------------
463
+
464
+ const buildUndeclaredDiagnostic = (
465
+ trailId: string,
466
+ signalId: string,
467
+ filePath: string,
468
+ line: number,
469
+ softened = false
470
+ ): WardenDiagnostic => ({
471
+ filePath,
472
+ line,
473
+ message: softened
474
+ ? `Trail "${trailId}": ctx.fire('${signalId}') called but '${signalId}' is not declared in fires (may be declared via object-form fires entries)`
475
+ : `Trail "${trailId}": ctx.fire('${signalId}') called but '${signalId}' is not declared in fires`,
476
+ rule: 'fires-declarations',
477
+ severity: softened ? 'warn' : 'error',
478
+ });
479
+
480
+ const buildUnusedDiagnostic = (
481
+ trailId: string,
482
+ signalId: string,
483
+ filePath: string,
484
+ line: number
485
+ ): WardenDiagnostic => ({
486
+ filePath,
487
+ line,
488
+ message: `Trail "${trailId}": '${signalId}' declared in fires but ctx.fire('${signalId}') never called`,
489
+ rule: 'fires-declarations',
490
+ severity: 'warn',
491
+ });
492
+
493
+ // ---------------------------------------------------------------------------
494
+ // Comparison
495
+ // ---------------------------------------------------------------------------
496
+
497
+ /** Emit error for each called ID not present in declared set. */
498
+ const reportUndeclared = (
499
+ called: ReadonlySet<string>,
500
+ declared: ReadonlySet<string>,
501
+ ctx: {
502
+ trailId: string;
503
+ filePath: string;
504
+ line: number;
505
+ softened?: boolean;
506
+ },
507
+ diagnostics: WardenDiagnostic[]
508
+ ): void => {
509
+ for (const id of called) {
510
+ if (!declared.has(id)) {
511
+ diagnostics.push(
512
+ buildUndeclaredDiagnostic(
513
+ ctx.trailId,
514
+ id,
515
+ ctx.filePath,
516
+ ctx.line,
517
+ ctx.softened
518
+ )
519
+ );
520
+ }
521
+ }
522
+ };
523
+
524
+ /**
525
+ * Emit warning for each declared ID not present in called set.
526
+ *
527
+ * Note: unlike `reportUndeclared`, this function does NOT soften its
528
+ * diagnostics when `hasUnresolved` is true. The asymmetry is intentional —
529
+ * softening only applies to the undeclared direction because unresolved
530
+ * Signal-value entries might cover an unknown set of called IDs. In the
531
+ * unused direction, a declared string-literal that is never called is
532
+ * genuinely unused regardless of whether other entries are unresolved.
533
+ */
534
+ const reportUnused = (
535
+ declared: ReadonlySet<string>,
536
+ called: ReadonlySet<string>,
537
+ ctx: { trailId: string; filePath: string; line: number },
538
+ diagnostics: WardenDiagnostic[]
539
+ ): void => {
540
+ for (const id of declared) {
541
+ if (!called.has(id)) {
542
+ diagnostics.push(
543
+ buildUnusedDiagnostic(ctx.trailId, id, ctx.filePath, ctx.line)
544
+ );
545
+ }
546
+ }
547
+ };
548
+
549
+ const checkTrailDefinition = (
550
+ def: { id: string; config: AstNode; start: number },
551
+ filePath: string,
552
+ sourceCode: string,
553
+ diagnostics: WardenDiagnostic[]
554
+ ): void => {
555
+ const declared = extractDeclaredFires(def.config, sourceCode);
556
+ const called = extractCalledFires(def.config);
557
+
558
+ if (declared.ids.size === 0 && !declared.hasUnresolved && called.size === 0) {
559
+ return;
560
+ }
561
+
562
+ const line = offsetToLine(sourceCode, def.start);
563
+ const ctx = { filePath, line, trailId: def.id };
564
+
565
+ // When the declared array contains object-form references we can't resolve,
566
+ // downgrade "undeclared" diagnostics from error to warn with a disclaimer
567
+ // instead of suppressing entirely. The developer still sees genuinely
568
+ // undeclared calls, but we can't statically prove the call isn't covered by
569
+ // a Signal-value entry the runtime will normalize.
570
+ reportUndeclared(
571
+ called,
572
+ declared.ids,
573
+ { ...ctx, softened: declared.hasUnresolved },
574
+ diagnostics
575
+ );
576
+ reportUnused(declared.ids, called, ctx, diagnostics);
577
+ };
578
+
579
+ // ---------------------------------------------------------------------------
580
+ // Rule
581
+ // ---------------------------------------------------------------------------
582
+
583
+ /**
584
+ * Validates that `ctx.fire()` calls align with declared `fires` arrays.
585
+ */
586
+ export const firesDeclarations: WardenRule = {
587
+ check(sourceCode: string, filePath: string): readonly WardenDiagnostic[] {
588
+ if (isTestFile(filePath)) {
589
+ return [];
590
+ }
591
+
592
+ const ast = parse(filePath, sourceCode);
593
+ if (!ast) {
594
+ return [];
595
+ }
596
+
597
+ const diagnostics: WardenDiagnostic[] = [];
598
+
599
+ for (const def of findTrailDefinitions(ast)) {
600
+ checkTrailDefinition(def, filePath, sourceCode, diagnostics);
601
+ }
602
+
603
+ return diagnostics;
604
+ },
605
+ description:
606
+ 'Ensure ctx.fire() calls match the declared fires array in trail definitions.',
607
+ name: 'fires-declarations',
608
+ severity: 'error',
609
+ };