@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.
- package/.turbo/turbo-lint.log +1 -1
- package/CHANGELOG.md +30 -0
- package/README.md +31 -20
- package/dist/cli.d.ts +19 -2
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +261 -64
- package/dist/cli.js.map +1 -1
- package/dist/draft.d.ts +5 -0
- package/dist/draft.d.ts.map +1 -0
- package/dist/draft.js +16 -0
- package/dist/draft.js.map +1 -0
- package/dist/drift.d.ts +10 -7
- package/dist/drift.d.ts.map +1 -1
- package/dist/drift.js +50 -16
- package/dist/drift.js.map +1 -1
- package/dist/formatters.d.ts +2 -1
- package/dist/formatters.d.ts.map +1 -1
- package/dist/formatters.js +15 -4
- package/dist/formatters.js.map +1 -1
- package/dist/index.d.ts +9 -17
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +10 -17
- package/dist/index.js.map +1 -1
- package/dist/rules/ast.d.ts +412 -7
- package/dist/rules/ast.d.ts.map +1 -1
- package/dist/rules/ast.js +1847 -102
- package/dist/rules/ast.js.map +1 -1
- package/dist/rules/circular-refs.d.ts +6 -0
- package/dist/rules/circular-refs.d.ts.map +1 -0
- package/dist/rules/circular-refs.js +83 -0
- package/dist/rules/circular-refs.js.map +1 -0
- package/dist/rules/context-no-surface-types.d.ts.map +1 -1
- package/dist/rules/context-no-surface-types.js +59 -3
- package/dist/rules/context-no-surface-types.js.map +1 -1
- package/dist/rules/contour-exists.d.ts +7 -0
- package/dist/rules/contour-exists.d.ts.map +1 -0
- package/dist/rules/contour-exists.js +113 -0
- package/dist/rules/contour-exists.js.map +1 -0
- package/dist/rules/contour-ids.d.ts +10 -0
- package/dist/rules/contour-ids.d.ts.map +1 -0
- package/dist/rules/contour-ids.js +12 -0
- package/dist/rules/contour-ids.js.map +1 -0
- package/dist/rules/cross-declarations.d.ts.map +1 -1
- package/dist/rules/cross-declarations.js +171 -57
- package/dist/rules/cross-declarations.js.map +1 -1
- package/dist/rules/dead-internal-trail.d.ts +3 -0
- package/dist/rules/dead-internal-trail.d.ts.map +1 -0
- package/dist/rules/dead-internal-trail.js +80 -0
- package/dist/rules/dead-internal-trail.js.map +1 -0
- package/dist/rules/draft-file-marking.d.ts +6 -0
- package/dist/rules/draft-file-marking.d.ts.map +1 -0
- package/dist/rules/draft-file-marking.js +87 -0
- package/dist/rules/draft-file-marking.js.map +1 -0
- package/dist/rules/draft-visible-debt.d.ts +12 -0
- package/dist/rules/draft-visible-debt.d.ts.map +1 -0
- package/dist/rules/draft-visible-debt.js +50 -0
- package/dist/rules/draft-visible-debt.js.map +1 -0
- package/dist/rules/error-mapping-completeness.d.ts +13 -0
- package/dist/rules/error-mapping-completeness.d.ts.map +1 -0
- package/dist/rules/error-mapping-completeness.js +160 -0
- package/dist/rules/error-mapping-completeness.js.map +1 -0
- package/dist/rules/example-valid.d.ts +6 -0
- package/dist/rules/example-valid.d.ts.map +1 -0
- package/dist/rules/example-valid.js +203 -0
- package/dist/rules/example-valid.js.map +1 -0
- package/dist/rules/fires-declarations.d.ts +16 -0
- package/dist/rules/fires-declarations.d.ts.map +1 -0
- package/dist/rules/fires-declarations.js +444 -0
- package/dist/rules/fires-declarations.js.map +1 -0
- package/dist/rules/implementation-returns-result.d.ts +9 -0
- package/dist/rules/implementation-returns-result.d.ts.map +1 -1
- package/dist/rules/implementation-returns-result.js +638 -76
- package/dist/rules/implementation-returns-result.js.map +1 -1
- package/dist/rules/incomplete-accessor-for-standard-op.d.ts +30 -0
- package/dist/rules/incomplete-accessor-for-standard-op.d.ts.map +1 -0
- package/dist/rules/incomplete-accessor-for-standard-op.js +226 -0
- package/dist/rules/incomplete-accessor-for-standard-op.js.map +1 -0
- package/dist/rules/incomplete-crud.d.ts +21 -0
- package/dist/rules/incomplete-crud.d.ts.map +1 -0
- package/dist/rules/incomplete-crud.js +368 -0
- package/dist/rules/incomplete-crud.js.map +1 -0
- package/dist/rules/index.d.ts +40 -7
- package/dist/rules/index.d.ts.map +1 -1
- package/dist/rules/index.js +91 -15
- package/dist/rules/index.js.map +1 -1
- package/dist/rules/intent-propagation.d.ts +3 -0
- package/dist/rules/intent-propagation.d.ts.map +1 -0
- package/dist/rules/intent-propagation.js +57 -0
- package/dist/rules/intent-propagation.js.map +1 -0
- package/dist/rules/missing-reconcile.d.ts +3 -0
- package/dist/rules/missing-reconcile.d.ts.map +1 -0
- package/dist/rules/missing-reconcile.js +44 -0
- package/dist/rules/missing-reconcile.js.map +1 -0
- package/dist/rules/missing-visibility.d.ts +3 -0
- package/dist/rules/missing-visibility.d.ts.map +1 -0
- package/dist/rules/missing-visibility.js +63 -0
- package/dist/rules/missing-visibility.js.map +1 -0
- package/dist/rules/no-direct-impl-in-route.d.ts.map +1 -1
- package/dist/rules/no-direct-impl-in-route.js +0 -3
- package/dist/rules/no-direct-impl-in-route.js.map +1 -1
- package/dist/rules/no-direct-implementation-call.js +1 -1
- package/dist/rules/no-direct-implementation-call.js.map +1 -1
- package/dist/rules/no-sync-result-assumption.d.ts.map +1 -1
- package/dist/rules/no-sync-result-assumption.js +870 -61
- package/dist/rules/no-sync-result-assumption.js.map +1 -1
- package/dist/rules/no-throw-in-detour-recover.d.ts +3 -0
- package/dist/rules/no-throw-in-detour-recover.d.ts.map +1 -0
- package/dist/rules/no-throw-in-detour-recover.js +147 -0
- package/dist/rules/no-throw-in-detour-recover.js.map +1 -0
- package/dist/rules/no-throw-in-detour-target.d.ts +4 -1
- package/dist/rules/no-throw-in-detour-target.d.ts.map +1 -1
- package/dist/rules/no-throw-in-detour-target.js +6 -3
- package/dist/rules/no-throw-in-detour-target.js.map +1 -1
- package/dist/rules/no-throw-in-implementation.d.ts +4 -2
- package/dist/rules/no-throw-in-implementation.d.ts.map +1 -1
- package/dist/rules/no-throw-in-implementation.js +6 -4
- package/dist/rules/no-throw-in-implementation.js.map +1 -1
- package/dist/rules/on-references-exist.d.ts +14 -0
- package/dist/rules/on-references-exist.d.ts.map +1 -0
- package/dist/rules/on-references-exist.js +109 -0
- package/dist/rules/on-references-exist.js.map +1 -0
- package/dist/rules/orphaned-signal.d.ts +3 -0
- package/dist/rules/orphaned-signal.d.ts.map +1 -0
- package/dist/rules/orphaned-signal.js +67 -0
- package/dist/rules/orphaned-signal.js.map +1 -0
- package/dist/rules/permit-governance.d.ts +3 -0
- package/dist/rules/permit-governance.d.ts.map +1 -0
- package/dist/rules/permit-governance.js +15 -0
- package/dist/rules/permit-governance.js.map +1 -0
- package/dist/rules/reference-exists.d.ts +6 -0
- package/dist/rules/reference-exists.d.ts.map +1 -0
- package/dist/rules/reference-exists.js +47 -0
- package/dist/rules/reference-exists.js.map +1 -0
- package/dist/rules/registry-names.d.ts +8 -0
- package/dist/rules/registry-names.d.ts.map +1 -0
- package/dist/rules/registry-names.js +83 -0
- package/dist/rules/registry-names.js.map +1 -0
- package/dist/rules/resource-declarations.d.ts +14 -0
- package/dist/rules/resource-declarations.d.ts.map +1 -0
- package/dist/rules/resource-declarations.js +413 -0
- package/dist/rules/resource-declarations.js.map +1 -0
- package/dist/rules/resource-exists.d.ts +6 -0
- package/dist/rules/resource-exists.d.ts.map +1 -0
- package/dist/rules/resource-exists.js +90 -0
- package/dist/rules/resource-exists.js.map +1 -0
- package/dist/rules/resource-id-grammar.d.ts +3 -0
- package/dist/rules/resource-id-grammar.d.ts.map +1 -0
- package/dist/rules/resource-id-grammar.js +39 -0
- package/dist/rules/resource-id-grammar.js.map +1 -0
- package/dist/rules/specs.d.ts.map +1 -1
- package/dist/rules/specs.js +5 -1
- package/dist/rules/specs.js.map +1 -1
- package/dist/rules/types.d.ts +53 -4
- package/dist/rules/types.d.ts.map +1 -1
- package/dist/rules/unreachable-detour-shadowing.d.ts +3 -0
- package/dist/rules/unreachable-detour-shadowing.d.ts.map +1 -0
- package/dist/rules/unreachable-detour-shadowing.js +202 -0
- package/dist/rules/unreachable-detour-shadowing.js.map +1 -0
- package/dist/rules/valid-describe-refs.d.ts.map +1 -1
- package/dist/rules/valid-describe-refs.js +132 -16
- package/dist/rules/valid-describe-refs.js.map +1 -1
- package/dist/rules/valid-detour-contract.d.ts +3 -0
- package/dist/rules/valid-detour-contract.d.ts.map +1 -0
- package/dist/rules/valid-detour-contract.js +47 -0
- package/dist/rules/valid-detour-contract.js.map +1 -0
- package/dist/rules/valid-detour-refs.d.ts.map +1 -1
- package/dist/rules/valid-detour-refs.js +73 -82
- package/dist/rules/valid-detour-refs.js.map +1 -1
- package/dist/rules/warden-export-symmetry.d.ts +7 -0
- package/dist/rules/warden-export-symmetry.d.ts.map +1 -0
- package/dist/rules/warden-export-symmetry.js +352 -0
- package/dist/rules/warden-export-symmetry.js.map +1 -0
- package/dist/rules/warden-rules-use-ast.d.ts +17 -0
- package/dist/rules/warden-rules-use-ast.d.ts.map +1 -0
- package/dist/rules/warden-rules-use-ast.js +778 -0
- package/dist/rules/warden-rules-use-ast.js.map +1 -0
- package/dist/trails/circular-refs.trail.d.ts +24 -0
- package/dist/trails/circular-refs.trail.d.ts.map +1 -0
- package/dist/trails/circular-refs.trail.js +29 -0
- package/dist/trails/circular-refs.trail.js.map +1 -0
- package/dist/trails/context-no-surface-types.trail.d.ts +2 -2
- package/dist/trails/context-no-surface-types.trail.d.ts.map +1 -1
- package/dist/trails/context-no-trailhead-types.trail.d.ts +2 -2
- package/dist/trails/context-no-trailhead-types.trail.d.ts.map +1 -1
- package/dist/trails/contour-exists.trail.d.ts +24 -0
- package/dist/trails/contour-exists.trail.d.ts.map +1 -0
- package/dist/trails/contour-exists.trail.js +21 -0
- package/dist/trails/contour-exists.trail.js.map +1 -0
- package/dist/trails/cross-declarations.trail.d.ts +2 -2
- package/dist/trails/cross-declarations.trail.d.ts.map +1 -1
- package/dist/trails/dead-internal-trail.trail.d.ts +24 -0
- package/dist/trails/dead-internal-trail.trail.d.ts.map +1 -0
- package/dist/trails/dead-internal-trail.trail.js +26 -0
- package/dist/trails/dead-internal-trail.trail.js.map +1 -0
- package/dist/trails/{provision-declarations.trail.d.ts → draft-file-marking.trail.d.ts} +3 -3
- package/dist/trails/draft-file-marking.trail.d.ts.map +1 -0
- package/dist/trails/draft-file-marking.trail.js +16 -0
- package/dist/trails/draft-file-marking.trail.js.map +1 -0
- package/dist/trails/draft-visible-debt.trail.d.ts +13 -0
- package/dist/trails/draft-visible-debt.trail.d.ts.map +1 -0
- package/dist/trails/draft-visible-debt.trail.js +16 -0
- package/dist/trails/draft-visible-debt.trail.js.map +1 -0
- package/dist/trails/error-mapping-completeness.trail.d.ts +13 -0
- package/dist/trails/error-mapping-completeness.trail.d.ts.map +1 -0
- package/dist/trails/error-mapping-completeness.trail.js +29 -0
- package/dist/trails/error-mapping-completeness.trail.js.map +1 -0
- package/dist/trails/{follow-declarations.trail.d.ts → example-valid.trail.d.ts} +3 -3
- package/dist/trails/example-valid.trail.d.ts.map +1 -0
- package/dist/trails/example-valid.trail.js +25 -0
- package/dist/trails/example-valid.trail.js.map +1 -0
- package/dist/trails/fires-declarations.trail.d.ts +13 -0
- package/dist/trails/fires-declarations.trail.d.ts.map +1 -0
- package/dist/trails/fires-declarations.trail.js +22 -0
- package/dist/trails/fires-declarations.trail.js.map +1 -0
- package/dist/trails/implementation-returns-result.trail.d.ts +2 -2
- package/dist/trails/implementation-returns-result.trail.d.ts.map +1 -1
- package/dist/trails/incomplete-accessor-for-standard-op.trail.d.ts +12 -0
- package/dist/trails/incomplete-accessor-for-standard-op.trail.d.ts.map +1 -0
- package/dist/trails/incomplete-accessor-for-standard-op.trail.js +60 -0
- package/dist/trails/incomplete-accessor-for-standard-op.trail.js.map +1 -0
- package/dist/trails/incomplete-crud.trail.d.ts +24 -0
- package/dist/trails/incomplete-crud.trail.d.ts.map +1 -0
- package/dist/trails/incomplete-crud.trail.js +39 -0
- package/dist/trails/incomplete-crud.trail.js.map +1 -0
- package/dist/trails/index.d.ts +29 -7
- package/dist/trails/index.d.ts.map +1 -1
- package/dist/trails/index.js +28 -6
- package/dist/trails/index.js.map +1 -1
- package/dist/trails/intent-propagation.trail.d.ts +24 -0
- package/dist/trails/intent-propagation.trail.d.ts.map +1 -0
- package/dist/trails/intent-propagation.trail.js +30 -0
- package/dist/trails/intent-propagation.trail.js.map +1 -0
- package/dist/trails/missing-reconcile.trail.d.ts +24 -0
- package/dist/trails/missing-reconcile.trail.d.ts.map +1 -0
- package/dist/trails/missing-reconcile.trail.js +33 -0
- package/dist/trails/missing-reconcile.trail.js.map +1 -0
- package/dist/trails/missing-visibility.trail.d.ts +24 -0
- package/dist/trails/missing-visibility.trail.d.ts.map +1 -0
- package/dist/trails/missing-visibility.trail.js +22 -0
- package/dist/trails/missing-visibility.trail.js.map +1 -0
- package/dist/trails/no-direct-impl-in-route.trail.d.ts +2 -2
- package/dist/trails/no-direct-impl-in-route.trail.d.ts.map +1 -1
- package/dist/trails/no-direct-implementation-call.trail.d.ts +2 -2
- package/dist/trails/no-direct-implementation-call.trail.d.ts.map +1 -1
- package/dist/trails/no-sync-result-assumption.trail.d.ts +2 -2
- package/dist/trails/no-sync-result-assumption.trail.d.ts.map +1 -1
- package/dist/trails/no-throw-in-detour-recover.trail.d.ts +13 -0
- package/dist/trails/no-throw-in-detour-recover.trail.d.ts.map +1 -0
- package/dist/trails/no-throw-in-detour-recover.trail.js +24 -0
- package/dist/trails/no-throw-in-detour-recover.trail.js.map +1 -0
- package/dist/trails/no-throw-in-detour-target.trail.d.ts +13 -3
- package/dist/trails/no-throw-in-detour-target.trail.d.ts.map +1 -1
- package/dist/trails/no-throw-in-implementation.trail.d.ts +2 -2
- package/dist/trails/no-throw-in-implementation.trail.d.ts.map +1 -1
- package/dist/trails/on-references-exist.trail.d.ts +24 -0
- package/dist/trails/on-references-exist.trail.d.ts.map +1 -0
- package/dist/trails/on-references-exist.trail.js +21 -0
- package/dist/trails/on-references-exist.trail.js.map +1 -0
- package/dist/trails/orphaned-signal.trail.d.ts +24 -0
- package/dist/trails/orphaned-signal.trail.d.ts.map +1 -0
- package/dist/trails/orphaned-signal.trail.js +36 -0
- package/dist/trails/orphaned-signal.trail.js.map +1 -0
- package/dist/trails/permit-governance.trail.d.ts +12 -0
- package/dist/trails/permit-governance.trail.d.ts.map +1 -0
- package/dist/trails/permit-governance.trail.js +47 -0
- package/dist/trails/permit-governance.trail.js.map +1 -0
- package/dist/trails/prefer-schema-inference.trail.d.ts +2 -2
- package/dist/trails/prefer-schema-inference.trail.d.ts.map +1 -1
- package/dist/trails/reference-exists.trail.d.ts +24 -0
- package/dist/trails/reference-exists.trail.d.ts.map +1 -0
- package/dist/trails/reference-exists.trail.js +25 -0
- package/dist/trails/reference-exists.trail.js.map +1 -0
- package/dist/trails/resource-declarations.trail.d.ts +13 -0
- package/dist/trails/resource-declarations.trail.d.ts.map +1 -0
- package/dist/trails/{provision-declarations.trail.js → resource-declarations.trail.js} +7 -7
- package/dist/trails/resource-declarations.trail.js.map +1 -0
- package/dist/trails/resource-exists.trail.d.ts +24 -0
- package/dist/trails/resource-exists.trail.d.ts.map +1 -0
- package/dist/trails/{provision-exists.trail.js → resource-exists.trail.js} +8 -8
- package/dist/trails/resource-exists.trail.js.map +1 -0
- package/dist/trails/resource-id-grammar.trail.d.ts +13 -0
- package/dist/trails/resource-id-grammar.trail.d.ts.map +1 -0
- package/dist/trails/resource-id-grammar.trail.js +38 -0
- package/dist/trails/resource-id-grammar.trail.js.map +1 -0
- package/dist/trails/run.d.ts +25 -9
- package/dist/trails/run.d.ts.map +1 -1
- package/dist/trails/run.js +63 -19
- package/dist/trails/run.js.map +1 -1
- package/dist/trails/schema.d.ts +28 -3
- package/dist/trails/schema.d.ts.map +1 -1
- package/dist/trails/schema.js +57 -4
- package/dist/trails/schema.js.map +1 -1
- package/dist/trails/unreachable-detour-shadowing.trail.d.ts +13 -0
- package/dist/trails/unreachable-detour-shadowing.trail.d.ts.map +1 -0
- package/dist/trails/unreachable-detour-shadowing.trail.js +44 -0
- package/dist/trails/unreachable-detour-shadowing.trail.js.map +1 -0
- package/dist/trails/valid-describe-refs.trail.d.ts +12 -3
- package/dist/trails/valid-describe-refs.trail.d.ts.map +1 -1
- package/dist/trails/valid-detour-contract.trail.d.ts +12 -0
- package/dist/trails/valid-detour-contract.trail.d.ts.map +1 -0
- package/dist/trails/valid-detour-contract.trail.js +66 -0
- package/dist/trails/valid-detour-contract.trail.js.map +1 -0
- package/dist/trails/valid-detour-refs.trail.d.ts +13 -3
- package/dist/trails/valid-detour-refs.trail.d.ts.map +1 -1
- package/dist/trails/warden-export-symmetry.trail.d.ts +13 -0
- package/dist/trails/warden-export-symmetry.trail.d.ts.map +1 -0
- package/dist/trails/warden-export-symmetry.trail.js +16 -0
- package/dist/trails/warden-export-symmetry.trail.js.map +1 -0
- package/dist/trails/warden-rules-use-ast.trail.d.ts +13 -0
- package/dist/trails/warden-rules-use-ast.trail.d.ts.map +1 -0
- package/dist/trails/warden-rules-use-ast.trail.js +41 -0
- package/dist/trails/warden-rules-use-ast.trail.js.map +1 -0
- package/dist/trails/wrap-rule.d.ts +16 -2
- package/dist/trails/wrap-rule.d.ts.map +1 -1
- package/dist/trails/wrap-rule.js +71 -11
- package/dist/trails/wrap-rule.js.map +1 -1
- package/package.json +7 -4
- package/src/__tests__/ast.test.ts +613 -0
- package/src/__tests__/circular-refs.test.ts +121 -0
- package/src/__tests__/cli.test.ts +360 -32
- package/src/__tests__/contour-exists.test.ts +203 -0
- package/src/__tests__/cross-declarations.test.ts +245 -0
- package/src/__tests__/dead-internal-trail.test.ts +81 -0
- package/src/__tests__/draft-rules-context.test.ts +150 -0
- package/src/__tests__/drift.test.ts +75 -5
- package/src/__tests__/error-mapping-completeness.test.ts +56 -0
- package/src/__tests__/example-valid.test.ts +101 -0
- package/src/__tests__/fires-declarations-param-destructure.test.ts +54 -0
- package/src/__tests__/fires-declarations.test.ts +652 -0
- package/src/__tests__/formatters.test.ts +2 -2
- package/src/__tests__/implementation-returns-result.test.ts +1016 -2
- package/src/__tests__/incomplete-accessor-for-standard-op.test.ts +337 -0
- package/src/__tests__/incomplete-crud.test.ts +498 -0
- package/src/__tests__/intent-propagation.test.ts +116 -0
- package/src/__tests__/missing-reconcile.test.ts +154 -0
- package/src/__tests__/missing-visibility.test.ts +108 -0
- package/src/__tests__/no-sync-result-assumption.test.ts +870 -39
- package/src/__tests__/no-throw-in-detour-recover.test.ts +93 -0
- package/src/__tests__/no-throw-in-implementation.test.ts +88 -0
- package/src/__tests__/on-references-exist.test.ts +151 -0
- package/src/__tests__/orphaned-signal.test.ts +137 -0
- package/src/__tests__/permit-governance.test.ts +66 -0
- package/src/__tests__/reference-exists.test.ts +281 -0
- package/src/__tests__/resource-declarations.test.ts +448 -0
- package/src/__tests__/resource-exists.test.ts +122 -0
- package/src/__tests__/resource-id-grammar.test.ts +50 -0
- package/src/__tests__/rules.test.ts +17 -77
- package/src/__tests__/topo-aware-rule.test.ts +257 -0
- package/src/__tests__/trails.test.ts +2 -2
- package/src/__tests__/unreachable-detour-shadowing.test.ts +128 -0
- package/src/__tests__/valid-describe-refs.test.ts +183 -0
- package/src/__tests__/valid-detour-contract.test.ts +86 -0
- package/src/__tests__/warden-export-symmetry.test.ts +251 -0
- package/src/__tests__/warden-rules-use-ast.test.ts +468 -0
- package/src/__tests__/wrap-rule.test.ts +3 -3
- package/src/cli.ts +458 -91
- package/src/draft.ts +22 -0
- package/src/drift.ts +63 -21
- package/src/formatters.ts +15 -4
- package/src/index.ts +62 -23
- package/src/rules/ast.ts +2715 -119
- package/src/rules/circular-refs.ts +154 -0
- package/src/rules/{context-no-trailhead-types.ts → context-no-surface-types.ts} +72 -12
- package/src/rules/contour-exists.ts +251 -0
- package/src/rules/contour-ids.ts +15 -0
- package/src/rules/cross-declarations.ts +277 -69
- package/src/rules/dead-internal-trail.ts +141 -0
- package/src/rules/draft-file-marking.ts +160 -0
- package/src/rules/draft-visible-debt.ts +87 -0
- package/src/rules/error-mapping-completeness.ts +273 -0
- package/src/rules/example-valid.ts +401 -0
- package/src/rules/fires-declarations.ts +609 -0
- package/src/rules/implementation-returns-result.ts +1042 -122
- package/src/rules/incomplete-accessor-for-standard-op.ts +315 -0
- package/src/rules/incomplete-crud.ts +579 -0
- package/src/rules/index.ts +95 -16
- package/src/rules/intent-propagation.ts +142 -0
- package/src/rules/missing-reconcile.ts +98 -0
- package/src/rules/missing-visibility.ts +110 -0
- package/src/rules/no-direct-impl-in-route.ts +0 -4
- package/src/rules/no-direct-implementation-call.ts +1 -1
- package/src/rules/no-sync-result-assumption.ts +1134 -96
- package/src/rules/no-throw-in-detour-recover.ts +225 -0
- package/src/rules/no-throw-in-implementation.ts +6 -4
- package/src/rules/on-references-exist.ts +194 -0
- package/src/rules/orphaned-signal.ts +150 -0
- package/src/rules/permit-governance.ts +25 -0
- package/src/rules/reference-exists.ts +98 -0
- package/src/rules/registry-names.ts +83 -0
- package/src/rules/{provision-declarations.ts → resource-declarations.ts} +208 -138
- package/src/rules/{provision-exists.ts → resource-exists.ts} +48 -51
- package/src/rules/resource-id-grammar.ts +65 -0
- package/src/rules/specs.ts +5 -1
- package/src/rules/types.ts +57 -4
- package/src/rules/unreachable-detour-shadowing.ts +375 -0
- package/src/rules/valid-describe-refs.ts +160 -32
- package/src/rules/valid-detour-contract.ts +78 -0
- package/src/rules/warden-export-symmetry.ts +533 -0
- package/src/rules/warden-rules-use-ast.ts +996 -0
- package/src/trails/circular-refs.trail.ts +29 -0
- package/src/trails/{context-no-trailhead-types.trail.ts → context-no-surface-types.trail.ts} +4 -4
- package/src/trails/contour-exists.trail.ts +21 -0
- package/src/trails/dead-internal-trail.trail.ts +26 -0
- package/src/trails/draft-file-marking.trail.ts +16 -0
- package/src/trails/draft-visible-debt.trail.ts +16 -0
- package/src/trails/error-mapping-completeness.trail.ts +29 -0
- package/src/trails/example-valid.trail.ts +25 -0
- package/src/trails/fires-declarations.trail.ts +22 -0
- package/src/trails/incomplete-accessor-for-standard-op.trail.ts +76 -0
- package/src/trails/incomplete-crud.trail.ts +39 -0
- package/src/trails/index.ts +40 -7
- package/src/trails/intent-propagation.trail.ts +30 -0
- package/src/trails/missing-reconcile.trail.ts +33 -0
- package/src/trails/missing-visibility.trail.ts +22 -0
- package/src/trails/no-throw-in-detour-recover.trail.ts +24 -0
- package/src/trails/on-references-exist.trail.ts +21 -0
- package/src/trails/orphaned-signal.trail.ts +36 -0
- package/src/trails/permit-governance.trail.ts +51 -0
- package/src/trails/reference-exists.trail.ts +25 -0
- package/src/trails/{provision-declarations.trail.ts → resource-declarations.trail.ts} +6 -6
- package/src/trails/{provision-exists.trail.ts → resource-exists.trail.ts} +7 -7
- package/src/trails/resource-id-grammar.trail.ts +39 -0
- package/src/trails/run.ts +121 -24
- package/src/trails/schema.ts +66 -4
- package/src/trails/unreachable-detour-shadowing.trail.ts +45 -0
- package/src/trails/valid-detour-contract.trail.ts +71 -0
- package/src/trails/warden-export-symmetry.trail.ts +16 -0
- package/src/trails/warden-rules-use-ast.trail.ts +45 -0
- package/src/trails/wrap-rule.ts +104 -12
- package/tsconfig.tests.json +10 -0
- package/tsconfig.tsbuildinfo +1 -1
- package/dist/rules/follow-declarations.d.ts +0 -13
- package/dist/rules/follow-declarations.d.ts.map +0 -1
- package/dist/rules/follow-declarations.js +0 -264
- package/dist/rules/follow-declarations.js.map +0 -1
- package/dist/rules/provision-declarations.d.ts +0 -14
- package/dist/rules/provision-declarations.d.ts.map +0 -1
- package/dist/rules/provision-declarations.js +0 -344
- package/dist/rules/provision-declarations.js.map +0 -1
- package/dist/rules/provision-exists.d.ts +0 -6
- package/dist/rules/provision-exists.d.ts.map +0 -1
- package/dist/rules/provision-exists.js +0 -89
- package/dist/rules/provision-exists.js.map +0 -1
- package/dist/rules/service-declarations.d.ts +0 -16
- package/dist/rules/service-declarations.d.ts.map +0 -1
- package/dist/rules/service-declarations.js +0 -346
- package/dist/rules/service-declarations.js.map +0 -1
- package/dist/rules/service-exists.d.ts +0 -8
- package/dist/rules/service-exists.d.ts.map +0 -1
- package/dist/rules/service-exists.js +0 -91
- package/dist/rules/service-exists.js.map +0 -1
- package/dist/trails/follow-declarations.trail.d.ts.map +0 -1
- package/dist/trails/follow-declarations.trail.js +0 -22
- package/dist/trails/follow-declarations.trail.js.map +0 -1
- package/dist/trails/provision-declarations.trail.d.ts.map +0 -1
- package/dist/trails/provision-declarations.trail.js.map +0 -1
- package/dist/trails/provision-exists.trail.d.ts +0 -15
- package/dist/trails/provision-exists.trail.d.ts.map +0 -1
- package/dist/trails/provision-exists.trail.js.map +0 -1
- package/dist/trails/service-declarations.trail.d.ts +0 -26
- package/dist/trails/service-declarations.trail.d.ts.map +0 -1
- package/dist/trails/service-declarations.trail.js +0 -27
- package/dist/trails/service-declarations.trail.js.map +0 -1
- package/dist/trails/service-exists.trail.d.ts +0 -32
- package/dist/trails/service-exists.trail.d.ts.map +0 -1
- package/dist/trails/service-exists.trail.js +0 -29
- package/dist/trails/service-exists.trail.js.map +0 -1
- package/src/__tests__/no-throw-in-detour-target.test.ts +0 -78
- package/src/__tests__/provision-declarations.test.ts +0 -318
- package/src/__tests__/provision-exists.test.ts +0 -122
- package/src/rules/no-throw-in-detour-target.ts +0 -150
- package/src/rules/valid-detour-refs.ts +0 -187
- package/src/trails/no-throw-in-detour-target.trail.ts +0 -20
- package/src/trails/valid-detour-refs.trail.ts +0 -24
|
@@ -1,27 +1,27 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { resourceExists } from '../rules/resource-exists.js';
|
|
2
2
|
import { wrapRule } from './wrap-rule.js';
|
|
3
3
|
|
|
4
|
-
export const
|
|
4
|
+
export const resourceExistsTrail = wrapRule({
|
|
5
5
|
examples: [
|
|
6
6
|
{
|
|
7
7
|
expected: { diagnostics: [] },
|
|
8
8
|
input: {
|
|
9
9
|
filePath: 'clean.ts',
|
|
10
|
-
|
|
10
|
+
knownResourceIds: ['db.main'],
|
|
11
11
|
knownTrailIds: ['entity.show'],
|
|
12
|
-
sourceCode: `const db =
|
|
12
|
+
sourceCode: `const db = resource("db.main", {
|
|
13
13
|
create: () => Result.ok({ source: "factory" }),
|
|
14
14
|
});
|
|
15
15
|
|
|
16
16
|
trail("entity.show", {
|
|
17
|
-
|
|
17
|
+
resources: [db],
|
|
18
18
|
blaze: async (_input, ctx) => {
|
|
19
19
|
return Result.ok(db.from(ctx));
|
|
20
20
|
}
|
|
21
21
|
})`,
|
|
22
22
|
},
|
|
23
|
-
name: 'Declared
|
|
23
|
+
name: 'Declared resources resolve to known project resources',
|
|
24
24
|
},
|
|
25
25
|
],
|
|
26
|
-
rule:
|
|
26
|
+
rule: resourceExists,
|
|
27
27
|
});
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { resourceIdGrammar } from '../rules/resource-id-grammar.js';
|
|
2
|
+
import { wrapRule } from './wrap-rule.js';
|
|
3
|
+
|
|
4
|
+
export const resourceIdGrammarTrail = wrapRule({
|
|
5
|
+
examples: [
|
|
6
|
+
{
|
|
7
|
+
expected: { diagnostics: [] },
|
|
8
|
+
input: {
|
|
9
|
+
filePath: 'clean.ts',
|
|
10
|
+
sourceCode: `const db = resource("db.main", {
|
|
11
|
+
create: () => Result.ok({}),
|
|
12
|
+
});`,
|
|
13
|
+
},
|
|
14
|
+
name: 'Resource ids stay free of the scope separator',
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
expected: {
|
|
18
|
+
diagnostics: [
|
|
19
|
+
{
|
|
20
|
+
filePath: 'invalid.ts',
|
|
21
|
+
line: 1,
|
|
22
|
+
message:
|
|
23
|
+
'Resource "billing:primary" is invalid because resource ids may not contain ":".',
|
|
24
|
+
rule: 'resource-id-grammar',
|
|
25
|
+
severity: 'error',
|
|
26
|
+
},
|
|
27
|
+
],
|
|
28
|
+
},
|
|
29
|
+
input: {
|
|
30
|
+
filePath: 'invalid.ts',
|
|
31
|
+
sourceCode: `const db = resource("billing:primary", {
|
|
32
|
+
create: () => Result.ok({}),
|
|
33
|
+
});`,
|
|
34
|
+
},
|
|
35
|
+
name: 'Colon-separated resource ids are rejected',
|
|
36
|
+
},
|
|
37
|
+
],
|
|
38
|
+
rule: resourceIdGrammar,
|
|
39
|
+
});
|
package/src/trails/run.ts
CHANGED
|
@@ -1,51 +1,148 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Run
|
|
2
|
+
* Run file-scoped warden rule trails against a single source file.
|
|
3
3
|
*
|
|
4
|
-
* Returns a flat array of diagnostics from every rule.
|
|
4
|
+
* Returns a flat array of diagnostics from every source-aware rule. Built-in
|
|
5
|
+
* topo-aware rules are dispatched separately via `runTopoAwareWardenTrails()`
|
|
6
|
+
* so callers that loop files do not duplicate graph-level findings.
|
|
5
7
|
*/
|
|
6
8
|
|
|
9
|
+
import type { Topo } from '@ontrails/core';
|
|
7
10
|
import { run } from '@ontrails/core';
|
|
8
11
|
|
|
12
|
+
import { wardenTopoRules } from '../rules/index.js';
|
|
9
13
|
import type { WardenDiagnostic } from '../rules/types.js';
|
|
10
14
|
import type { RuleOutput } from './schema.js';
|
|
11
15
|
import { wardenTopo } from './topo.js';
|
|
12
16
|
|
|
13
17
|
/**
|
|
14
|
-
* Run all warden rule trails for a given file and collect diagnostics.
|
|
18
|
+
* Run all file-scoped warden rule trails for a given file and collect diagnostics.
|
|
15
19
|
*
|
|
16
20
|
* Each rule trail runs independently. Errors from individual trails are
|
|
17
21
|
* silently skipped so that one broken rule does not block the rest.
|
|
18
22
|
*/
|
|
19
|
-
|
|
23
|
+
const appendDiagnostics = (
|
|
24
|
+
target: WardenDiagnostic[],
|
|
25
|
+
diagnostics: readonly WardenDiagnostic[]
|
|
26
|
+
): void => {
|
|
27
|
+
for (const diagnostic of diagnostics) {
|
|
28
|
+
target.push(diagnostic);
|
|
29
|
+
}
|
|
30
|
+
};
|
|
31
|
+
|
|
32
|
+
type TrailIntentMap = Readonly<Record<string, 'destroy' | 'read' | 'write'>>;
|
|
33
|
+
|
|
34
|
+
interface ProjectRuleOptions {
|
|
35
|
+
readonly contourReferencesByName?: Readonly<
|
|
36
|
+
Record<string, readonly string[]>
|
|
37
|
+
>;
|
|
38
|
+
readonly crossTargetTrailIds?: readonly string[];
|
|
39
|
+
readonly crudTableIds?: readonly string[];
|
|
40
|
+
readonly crudCoverageByEntity?: Readonly<Record<string, readonly string[]>>;
|
|
41
|
+
readonly knownContourIds?: readonly string[];
|
|
42
|
+
readonly knownResourceIds?: readonly string[];
|
|
43
|
+
readonly knownSignalIds?: readonly string[];
|
|
44
|
+
readonly knownTrailIds?: readonly string[];
|
|
45
|
+
readonly onTargetSignalIds?: readonly string[];
|
|
46
|
+
readonly reconcileTableIds?: readonly string[];
|
|
47
|
+
readonly trailIntentsById?: TrailIntentMap;
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
const PROJECT_OPTION_KEYS = [
|
|
51
|
+
'contourReferencesByName',
|
|
52
|
+
'crossTargetTrailIds',
|
|
53
|
+
'crudTableIds',
|
|
54
|
+
'crudCoverageByEntity',
|
|
55
|
+
'knownContourIds',
|
|
56
|
+
'knownResourceIds',
|
|
57
|
+
'knownSignalIds',
|
|
58
|
+
'knownTrailIds',
|
|
59
|
+
'onTargetSignalIds',
|
|
60
|
+
'reconcileTableIds',
|
|
61
|
+
'trailIntentsById',
|
|
62
|
+
] as const satisfies readonly (keyof ProjectRuleOptions)[];
|
|
63
|
+
|
|
64
|
+
const hasProjectOptions = (options?: ProjectRuleOptions): boolean =>
|
|
65
|
+
Boolean(
|
|
66
|
+
options && PROJECT_OPTION_KEYS.some((key) => options[key] !== undefined)
|
|
67
|
+
);
|
|
68
|
+
|
|
69
|
+
const collectProjectOptions = (
|
|
70
|
+
options?: ProjectRuleOptions
|
|
71
|
+
): ProjectRuleOptions => {
|
|
72
|
+
if (!options) {
|
|
73
|
+
return {};
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
return Object.fromEntries(
|
|
77
|
+
PROJECT_OPTION_KEYS.flatMap((key) => {
|
|
78
|
+
const value = options[key];
|
|
79
|
+
return value === undefined ? [] : [[key, value] as const];
|
|
80
|
+
})
|
|
81
|
+
) as ProjectRuleOptions;
|
|
82
|
+
};
|
|
83
|
+
|
|
84
|
+
const buildRuleInput = (
|
|
20
85
|
filePath: string,
|
|
21
86
|
sourceCode: string,
|
|
22
|
-
options?:
|
|
23
|
-
|
|
24
|
-
|
|
87
|
+
options?: ProjectRuleOptions
|
|
88
|
+
): {
|
|
89
|
+
readonly filePath: string;
|
|
90
|
+
readonly sourceCode: string;
|
|
91
|
+
} & ProjectRuleOptions => {
|
|
92
|
+
const base = { filePath, sourceCode };
|
|
93
|
+
if (!hasProjectOptions(options)) {
|
|
94
|
+
return base;
|
|
25
95
|
}
|
|
96
|
+
|
|
97
|
+
return { ...base, ...collectProjectOptions(options) };
|
|
98
|
+
};
|
|
99
|
+
|
|
100
|
+
const topoAwareTrailIds = new Set(
|
|
101
|
+
[...wardenTopoRules.keys()].map((ruleName) => `warden.rule.${ruleName}`)
|
|
102
|
+
);
|
|
103
|
+
|
|
104
|
+
export const runWardenTrails = async (
|
|
105
|
+
filePath: string,
|
|
106
|
+
sourceCode: string,
|
|
107
|
+
options?: ProjectRuleOptions
|
|
26
108
|
): Promise<readonly WardenDiagnostic[]> => {
|
|
27
109
|
const allDiagnostics: WardenDiagnostic[] = [];
|
|
110
|
+
const input = buildRuleInput(filePath, sourceCode, options);
|
|
28
111
|
|
|
29
112
|
for (const id of wardenTopo.ids()) {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
filePath,
|
|
34
|
-
...(options?.knownProvisionIds
|
|
35
|
-
? { knownProvisionIds: options.knownProvisionIds }
|
|
36
|
-
: {}),
|
|
37
|
-
...(options?.knownTrailIds
|
|
38
|
-
? { knownTrailIds: options.knownTrailIds }
|
|
39
|
-
: {}),
|
|
40
|
-
sourceCode,
|
|
41
|
-
}
|
|
42
|
-
: { filePath, sourceCode };
|
|
113
|
+
if (topoAwareTrailIds.has(id)) {
|
|
114
|
+
continue;
|
|
115
|
+
}
|
|
43
116
|
const result = await run(wardenTopo, id, input);
|
|
44
117
|
if (result.isOk()) {
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
118
|
+
appendDiagnostics(
|
|
119
|
+
allDiagnostics,
|
|
120
|
+
(result.value as RuleOutput).diagnostics
|
|
121
|
+
);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
return allDiagnostics;
|
|
126
|
+
};
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* Run the built-in topo-aware warden rule trails once against a resolved topo.
|
|
130
|
+
*
|
|
131
|
+
* Unlike `runWardenTrails()`, which is file-scoped, topo-aware rules inspect
|
|
132
|
+
* the compiled graph and should only be dispatched once per topo.
|
|
133
|
+
*/
|
|
134
|
+
export const runTopoAwareWardenTrails = async (
|
|
135
|
+
topo: Topo
|
|
136
|
+
): Promise<readonly WardenDiagnostic[]> => {
|
|
137
|
+
const allDiagnostics: WardenDiagnostic[] = [];
|
|
138
|
+
|
|
139
|
+
for (const id of topoAwareTrailIds) {
|
|
140
|
+
const result = await run(wardenTopo, id, { topo });
|
|
141
|
+
if (result.isOk()) {
|
|
142
|
+
appendDiagnostics(
|
|
143
|
+
allDiagnostics,
|
|
144
|
+
(result.value as RuleOutput).diagnostics
|
|
145
|
+
);
|
|
49
146
|
}
|
|
50
147
|
}
|
|
51
148
|
|
package/src/trails/schema.ts
CHANGED
|
@@ -5,6 +5,7 @@
|
|
|
5
5
|
* (array of diagnostics) shape.
|
|
6
6
|
*/
|
|
7
7
|
|
|
8
|
+
import type { Topo } from '@ontrails/core';
|
|
8
9
|
import { z } from 'zod';
|
|
9
10
|
|
|
10
11
|
/** A single diagnostic emitted by a warden rule trail. */
|
|
@@ -26,18 +27,78 @@ export const ruleInput = z.object({
|
|
|
26
27
|
* Extended input for project-aware warden rule trails.
|
|
27
28
|
*
|
|
28
29
|
* Adds `knownTrailIds` so the caller can supply cross-file context and avoid
|
|
29
|
-
* false positives for
|
|
30
|
-
* files.
|
|
30
|
+
* false positives for `@see` references or cross-file contour relationships.
|
|
31
31
|
*/
|
|
32
32
|
export const projectAwareRuleInput = ruleInput.extend({
|
|
33
|
-
|
|
33
|
+
contourReferencesByName: z
|
|
34
|
+
.record(z.string(), z.array(z.string()))
|
|
35
|
+
.optional()
|
|
36
|
+
.describe('Declared contour references keyed by source contour name'),
|
|
37
|
+
crossTargetTrailIds: z
|
|
38
|
+
.array(z.string())
|
|
39
|
+
.optional()
|
|
40
|
+
.describe('Trail IDs referenced by crosses arrays across the project'),
|
|
41
|
+
crudCoverageByEntity: z
|
|
42
|
+
.record(z.string(), z.array(z.string()))
|
|
43
|
+
.optional()
|
|
44
|
+
.describe(
|
|
45
|
+
'CRUD operation coverage per entity aggregated across the project'
|
|
46
|
+
),
|
|
47
|
+
crudTableIds: z
|
|
48
|
+
.array(z.string())
|
|
49
|
+
.optional()
|
|
50
|
+
.describe('Store table IDs used with CRUD factories across the project'),
|
|
51
|
+
knownContourIds: z
|
|
52
|
+
.array(z.string())
|
|
53
|
+
.optional()
|
|
54
|
+
.describe('Contour names known across the project'),
|
|
55
|
+
knownResourceIds: z
|
|
56
|
+
.array(z.string())
|
|
57
|
+
.optional()
|
|
58
|
+
.describe('Resource IDs known across the project'),
|
|
59
|
+
knownSignalIds: z
|
|
34
60
|
.array(z.string())
|
|
35
61
|
.optional()
|
|
36
|
-
.describe('
|
|
62
|
+
.describe('Signal IDs known across the project'),
|
|
37
63
|
knownTrailIds: z
|
|
38
64
|
.array(z.string())
|
|
39
65
|
.optional()
|
|
40
66
|
.describe('Trail IDs known across the project'),
|
|
67
|
+
onTargetSignalIds: z
|
|
68
|
+
.array(z.string())
|
|
69
|
+
.optional()
|
|
70
|
+
.describe('Signal IDs referenced by trail on arrays across the project'),
|
|
71
|
+
reconcileTableIds: z
|
|
72
|
+
.array(z.string())
|
|
73
|
+
.optional()
|
|
74
|
+
.describe('Store table IDs used with reconcile trails across the project'),
|
|
75
|
+
trailIntentsById: z
|
|
76
|
+
.record(z.string(), z.enum(['read', 'write', 'destroy']))
|
|
77
|
+
.optional()
|
|
78
|
+
.describe('Normalized trail intents keyed by trail ID'),
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Input for topo-aware warden rule trails.
|
|
83
|
+
*
|
|
84
|
+
* The `Topo` graph is not a serializable value, so the schema accepts it
|
|
85
|
+
* as an opaque `z.custom`. Topo-aware rules are invoked from the warden
|
|
86
|
+
* runtime with a live, resolved topo reference — they are not expected
|
|
87
|
+
* to be called across a network boundary.
|
|
88
|
+
*/
|
|
89
|
+
export const topoAwareRuleInput = z.object({
|
|
90
|
+
topo: z
|
|
91
|
+
.custom<Topo>(
|
|
92
|
+
(value) =>
|
|
93
|
+
typeof value === 'object' &&
|
|
94
|
+
value !== null &&
|
|
95
|
+
'trails' in value &&
|
|
96
|
+
'resources' in value &&
|
|
97
|
+
'contours' in value &&
|
|
98
|
+
'signals' in value,
|
|
99
|
+
{ message: 'Expected a resolved Topo instance' }
|
|
100
|
+
)
|
|
101
|
+
.describe('Resolved topo graph under inspection'),
|
|
41
102
|
});
|
|
42
103
|
|
|
43
104
|
/** Output returned by every warden rule trail. */
|
|
@@ -47,4 +108,5 @@ export const ruleOutput = z.object({
|
|
|
47
108
|
|
|
48
109
|
export type RuleInput = z.infer<typeof ruleInput>;
|
|
49
110
|
export type ProjectAwareRuleInput = z.infer<typeof projectAwareRuleInput>;
|
|
111
|
+
export type TopoAwareRuleInput = z.infer<typeof topoAwareRuleInput>;
|
|
50
112
|
export type RuleOutput = z.infer<typeof ruleOutput>;
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { unreachableDetourShadowing } from '../rules/unreachable-detour-shadowing.js';
|
|
2
|
+
import { wrapRule } from './wrap-rule.js';
|
|
3
|
+
|
|
4
|
+
export const unreachableDetourShadowingTrail = wrapRule({
|
|
5
|
+
examples: [
|
|
6
|
+
{
|
|
7
|
+
expected: { diagnostics: [] },
|
|
8
|
+
input: {
|
|
9
|
+
filePath: 'clean.ts',
|
|
10
|
+
sourceCode: `trail("entity.save", {
|
|
11
|
+
detours: [
|
|
12
|
+
{ on: ConflictError, recover: async () => Result.ok({ winner: "specific" }) },
|
|
13
|
+
{ on: TrailsError, recover: async () => Result.ok({ winner: "broad" }) },
|
|
14
|
+
],
|
|
15
|
+
});`,
|
|
16
|
+
},
|
|
17
|
+
name: 'Specific detours can precede broader ones',
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
expected: {
|
|
21
|
+
diagnostics: [
|
|
22
|
+
{
|
|
23
|
+
filePath: 'shadowed.ts',
|
|
24
|
+
line: 4,
|
|
25
|
+
message:
|
|
26
|
+
'Trail "entity.save" declares detour on "ConflictError" after earlier detour on "TrailsError". Because "TrailsError" matches "ConflictError" first, the later detour is unreachable.',
|
|
27
|
+
rule: 'unreachable-detour-shadowing',
|
|
28
|
+
severity: 'error',
|
|
29
|
+
},
|
|
30
|
+
],
|
|
31
|
+
},
|
|
32
|
+
input: {
|
|
33
|
+
filePath: 'shadowed.ts',
|
|
34
|
+
sourceCode: `trail("entity.save", {
|
|
35
|
+
detours: [
|
|
36
|
+
{ on: TrailsError, recover: async () => Result.ok({ winner: "broad" }) },
|
|
37
|
+
{ on: ConflictError, recover: async () => Result.ok({ winner: "specific" }) },
|
|
38
|
+
],
|
|
39
|
+
});`,
|
|
40
|
+
},
|
|
41
|
+
name: 'Broader detours declared first shadow later specific ones',
|
|
42
|
+
},
|
|
43
|
+
],
|
|
44
|
+
rule: unreachableDetourShadowing,
|
|
45
|
+
});
|
|
@@ -0,0 +1,71 @@
|
|
|
1
|
+
import { ConflictError, Result, topo, trail } from '@ontrails/core';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
|
|
4
|
+
import { validDetourContract } from '../rules/valid-detour-contract.js';
|
|
5
|
+
import { wrapTopoRule } from './wrap-rule.js';
|
|
6
|
+
|
|
7
|
+
const validTrail = trail('entity.save', {
|
|
8
|
+
blaze: () => Result.ok({ ok: true }),
|
|
9
|
+
detours: [
|
|
10
|
+
{
|
|
11
|
+
on: ConflictError,
|
|
12
|
+
recover: async () => {
|
|
13
|
+
const result = await Promise.resolve(Result.ok({ ok: true }));
|
|
14
|
+
return result;
|
|
15
|
+
},
|
|
16
|
+
},
|
|
17
|
+
],
|
|
18
|
+
input: z.object({}),
|
|
19
|
+
output: z.object({ ok: z.boolean() }),
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
const invalidContractTrail = {
|
|
23
|
+
...validTrail,
|
|
24
|
+
detours: [
|
|
25
|
+
{
|
|
26
|
+
on: 'ConflictError',
|
|
27
|
+
recover: 'not a function',
|
|
28
|
+
},
|
|
29
|
+
],
|
|
30
|
+
} as unknown as typeof validTrail;
|
|
31
|
+
|
|
32
|
+
export const validDetourContractTrail = wrapTopoRule({
|
|
33
|
+
examples: [
|
|
34
|
+
{
|
|
35
|
+
expected: { diagnostics: [] },
|
|
36
|
+
input: {
|
|
37
|
+
topo: topo('trl-380-valid-detour-contract', { validTrail }),
|
|
38
|
+
},
|
|
39
|
+
name: 'Detours with an error constructor and recover function stay clean',
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
expected: {
|
|
43
|
+
diagnostics: [
|
|
44
|
+
{
|
|
45
|
+
filePath: '<topo>',
|
|
46
|
+
line: 1,
|
|
47
|
+
message:
|
|
48
|
+
'Trail "entity.save" detour[0] must declare an error constructor in on:. Received ConflictError.',
|
|
49
|
+
rule: 'valid-detour-contract',
|
|
50
|
+
severity: 'error',
|
|
51
|
+
},
|
|
52
|
+
{
|
|
53
|
+
filePath: '<topo>',
|
|
54
|
+
line: 1,
|
|
55
|
+
message:
|
|
56
|
+
'Trail "entity.save" detour[0] must declare a callable recover function.',
|
|
57
|
+
rule: 'valid-detour-contract',
|
|
58
|
+
severity: 'error',
|
|
59
|
+
},
|
|
60
|
+
],
|
|
61
|
+
},
|
|
62
|
+
input: {
|
|
63
|
+
topo: topo('trl-380-invalid-detour-contract', {
|
|
64
|
+
invalidContractTrail,
|
|
65
|
+
} as Record<string, unknown>),
|
|
66
|
+
},
|
|
67
|
+
name: 'Malformed detour contracts emit diagnostics',
|
|
68
|
+
},
|
|
69
|
+
],
|
|
70
|
+
rule: validDetourContract,
|
|
71
|
+
});
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { wardenExportSymmetry } from '../rules/warden-export-symmetry.js';
|
|
2
|
+
import { wrapRule } from './wrap-rule.js';
|
|
3
|
+
|
|
4
|
+
export const wardenExportSymmetryTrail = wrapRule({
|
|
5
|
+
examples: [
|
|
6
|
+
{
|
|
7
|
+
expected: { diagnostics: [] },
|
|
8
|
+
input: {
|
|
9
|
+
filePath: 'packages/other-pkg/src/index.ts',
|
|
10
|
+
sourceCode: `export { somethingElse } from './other.js';\n`,
|
|
11
|
+
},
|
|
12
|
+
name: 'Ignores files outside the warden barrel',
|
|
13
|
+
},
|
|
14
|
+
],
|
|
15
|
+
rule: wardenExportSymmetry,
|
|
16
|
+
});
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { fileURLToPath } from 'node:url';
|
|
2
|
+
import { wardenRulesUseAst } from '../rules/warden-rules-use-ast.js';
|
|
3
|
+
import { wrapRule } from './wrap-rule.js';
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Resolve a filePath inside this package's `src/rules/` directory so the
|
|
7
|
+
* positive example fires the path-anchored scope check. Anchoring via
|
|
8
|
+
* `import.meta.url` keeps the example robust to the cwd under which tests run.
|
|
9
|
+
*/
|
|
10
|
+
const fakeRulePath = fileURLToPath(
|
|
11
|
+
new URL('../rules/fake-rule.ts', import.meta.url)
|
|
12
|
+
);
|
|
13
|
+
|
|
14
|
+
export const wardenRulesUseAstTrail = wrapRule({
|
|
15
|
+
examples: [
|
|
16
|
+
{
|
|
17
|
+
expected: { diagnostics: [] },
|
|
18
|
+
input: {
|
|
19
|
+
filePath: 'packages/other-pkg/src/index.ts',
|
|
20
|
+
sourceCode: `const lines = sourceCode.split('\\n');\n`,
|
|
21
|
+
},
|
|
22
|
+
name: 'Ignores files outside the warden rules directory',
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
expected: {
|
|
26
|
+
diagnostics: [
|
|
27
|
+
{
|
|
28
|
+
filePath: fakeRulePath,
|
|
29
|
+
line: 1,
|
|
30
|
+
message:
|
|
31
|
+
'warden-rules-use-ast: sourceCode.split(...) treats source text as a string. Warden rules must inspect the AST via packages/warden/src/rules/ast.ts helpers, not regex-scan raw source text. Use findStringLiterals, findTrailDefinitions, findConfigProperty, or a similar AST walker. Raw-text scanning produces false positives on string literals, template payloads, and docstrings — see TRL-335, ADR-0036.',
|
|
32
|
+
rule: 'warden-rules-use-ast',
|
|
33
|
+
severity: 'error',
|
|
34
|
+
},
|
|
35
|
+
],
|
|
36
|
+
},
|
|
37
|
+
input: {
|
|
38
|
+
filePath: fakeRulePath,
|
|
39
|
+
sourceCode: `export const r = { check(sourceCode: string) { return sourceCode.split('\\n'); } };\n`,
|
|
40
|
+
},
|
|
41
|
+
name: 'Flags sourceCode.split(...) in a rule file',
|
|
42
|
+
},
|
|
43
|
+
],
|
|
44
|
+
rule: wardenRulesUseAst,
|
|
45
|
+
});
|
package/src/trails/wrap-rule.ts
CHANGED
|
@@ -4,16 +4,27 @@
|
|
|
4
4
|
* Keeps each rule trail file minimal — just the import + examples.
|
|
5
5
|
*/
|
|
6
6
|
|
|
7
|
-
import { trail, Result } from '@ontrails/core';
|
|
7
|
+
import { InternalError, trail, Result } from '@ontrails/core';
|
|
8
8
|
import type { Trail } from '@ontrails/core';
|
|
9
9
|
|
|
10
10
|
import type {
|
|
11
11
|
ProjectAwareWardenRule,
|
|
12
12
|
ProjectContext,
|
|
13
|
+
TopoAwareWardenRule,
|
|
13
14
|
WardenRule,
|
|
14
15
|
} from '../rules/types.js';
|
|
15
|
-
import {
|
|
16
|
-
|
|
16
|
+
import {
|
|
17
|
+
projectAwareRuleInput,
|
|
18
|
+
ruleInput,
|
|
19
|
+
ruleOutput,
|
|
20
|
+
topoAwareRuleInput,
|
|
21
|
+
} from './schema.js';
|
|
22
|
+
import type {
|
|
23
|
+
ProjectAwareRuleInput,
|
|
24
|
+
RuleInput,
|
|
25
|
+
RuleOutput,
|
|
26
|
+
TopoAwareRuleInput,
|
|
27
|
+
} from './schema.js';
|
|
17
28
|
|
|
18
29
|
interface WrapRuleOptions {
|
|
19
30
|
/** The existing warden rule to wrap. */
|
|
@@ -29,6 +40,53 @@ interface WrapProjectAwareRuleOptions {
|
|
|
29
40
|
readonly examples: Trail<ProjectAwareRuleInput, RuleOutput>['examples'];
|
|
30
41
|
}
|
|
31
42
|
|
|
43
|
+
const buildProjectContext = (input: ProjectAwareRuleInput): ProjectContext => ({
|
|
44
|
+
...(input.contourReferencesByName
|
|
45
|
+
? {
|
|
46
|
+
contourReferencesByName: new Map(
|
|
47
|
+
Object.entries(input.contourReferencesByName)
|
|
48
|
+
),
|
|
49
|
+
}
|
|
50
|
+
: {}),
|
|
51
|
+
...(input.crudTableIds ? { crudTableIds: new Set(input.crudTableIds) } : {}),
|
|
52
|
+
...(input.crudCoverageByEntity
|
|
53
|
+
? {
|
|
54
|
+
crudCoverageByEntity: new Map(
|
|
55
|
+
Object.entries(input.crudCoverageByEntity).map(
|
|
56
|
+
([entityId, operations]) => [
|
|
57
|
+
entityId,
|
|
58
|
+
new Set(operations) as ReadonlySet<string>,
|
|
59
|
+
]
|
|
60
|
+
)
|
|
61
|
+
),
|
|
62
|
+
}
|
|
63
|
+
: {}),
|
|
64
|
+
...(input.knownContourIds
|
|
65
|
+
? { knownContourIds: new Set(input.knownContourIds) }
|
|
66
|
+
: {}),
|
|
67
|
+
knownTrailIds: input.knownTrailIds
|
|
68
|
+
? new Set(input.knownTrailIds)
|
|
69
|
+
: new Set<string>(),
|
|
70
|
+
...(input.crossTargetTrailIds
|
|
71
|
+
? { crossTargetTrailIds: new Set(input.crossTargetTrailIds) }
|
|
72
|
+
: {}),
|
|
73
|
+
...(input.knownResourceIds
|
|
74
|
+
? { knownResourceIds: new Set(input.knownResourceIds) }
|
|
75
|
+
: {}),
|
|
76
|
+
...(input.knownSignalIds
|
|
77
|
+
? { knownSignalIds: new Set(input.knownSignalIds) }
|
|
78
|
+
: {}),
|
|
79
|
+
...(input.onTargetSignalIds
|
|
80
|
+
? { onTargetSignalIds: new Set(input.onTargetSignalIds) }
|
|
81
|
+
: {}),
|
|
82
|
+
...(input.reconcileTableIds
|
|
83
|
+
? { reconcileTableIds: new Set(input.reconcileTableIds) }
|
|
84
|
+
: {}),
|
|
85
|
+
...(input.trailIntentsById
|
|
86
|
+
? { trailIntentsById: new Map(Object.entries(input.trailIntentsById)) }
|
|
87
|
+
: {}),
|
|
88
|
+
});
|
|
89
|
+
|
|
32
90
|
/**
|
|
33
91
|
* Wrap an existing `WardenRule` as a trail with typed input/output.
|
|
34
92
|
*
|
|
@@ -50,18 +108,10 @@ export function wrapRule(
|
|
|
50
108
|
const projectAwareRule = rule as ProjectAwareWardenRule;
|
|
51
109
|
return trail(`warden.rule.${rule.name}`, {
|
|
52
110
|
blaze: (input: ProjectAwareRuleInput) => {
|
|
53
|
-
const context = {
|
|
54
|
-
knownProvisionIds: input.knownProvisionIds
|
|
55
|
-
? new Set(input.knownProvisionIds)
|
|
56
|
-
: undefined,
|
|
57
|
-
knownTrailIds: input.knownTrailIds
|
|
58
|
-
? new Set(input.knownTrailIds)
|
|
59
|
-
: new Set<string>(),
|
|
60
|
-
} as ProjectContext;
|
|
61
111
|
const diagnostics = projectAwareRule.checkWithContext(
|
|
62
112
|
input.sourceCode,
|
|
63
113
|
input.filePath,
|
|
64
|
-
|
|
114
|
+
buildProjectContext(input)
|
|
65
115
|
);
|
|
66
116
|
return Result.ok({ diagnostics: [...diagnostics] });
|
|
67
117
|
},
|
|
@@ -90,3 +140,45 @@ export function wrapRule(
|
|
|
90
140
|
output: ruleOutput,
|
|
91
141
|
});
|
|
92
142
|
}
|
|
143
|
+
|
|
144
|
+
interface WrapTopoRuleOptions {
|
|
145
|
+
/** The existing topo-aware warden rule to wrap. */
|
|
146
|
+
readonly rule: TopoAwareWardenRule;
|
|
147
|
+
/** Trail examples for testing and documentation. */
|
|
148
|
+
readonly examples: Trail<TopoAwareRuleInput, RuleOutput>['examples'];
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Wrap an existing `TopoAwareWardenRule` as a trail.
|
|
153
|
+
*
|
|
154
|
+
* Mirrors `wrapRule` for the per-topo dispatch path. Topo-aware rules run
|
|
155
|
+
* once per topo against the compiled runtime graph rather than per file,
|
|
156
|
+
* so the trail accepts the live `Topo` as input.
|
|
157
|
+
*/
|
|
158
|
+
export const wrapTopoRule = (
|
|
159
|
+
options: WrapTopoRuleOptions
|
|
160
|
+
): Trail<TopoAwareRuleInput, RuleOutput> => {
|
|
161
|
+
const { rule, examples } = options;
|
|
162
|
+
return trail(`warden.rule.${rule.name}`, {
|
|
163
|
+
blaze: async (input: TopoAwareRuleInput) => {
|
|
164
|
+
try {
|
|
165
|
+
const diagnostics = await rule.checkTopo(input.topo);
|
|
166
|
+
return Result.ok({ diagnostics: [...diagnostics] });
|
|
167
|
+
} catch (error) {
|
|
168
|
+
const cause = error instanceof Error ? error : new Error(String(error));
|
|
169
|
+
return Result.err(
|
|
170
|
+
new InternalError(
|
|
171
|
+
`Topo-aware rule "${rule.name}" threw while inspecting topo: ${cause.message}`,
|
|
172
|
+
{ cause }
|
|
173
|
+
)
|
|
174
|
+
);
|
|
175
|
+
}
|
|
176
|
+
},
|
|
177
|
+
description: rule.description,
|
|
178
|
+
examples,
|
|
179
|
+
input: topoAwareRuleInput,
|
|
180
|
+
intent: 'read',
|
|
181
|
+
meta: { category: 'governance', severity: rule.severity },
|
|
182
|
+
output: ruleOutput,
|
|
183
|
+
});
|
|
184
|
+
};
|