@aicqtools/guardrail 1.0.0-alpha.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (269) hide show
  1. package/LICENSE +21 -0
  2. package/dist/docs/build.d.ts +12 -0
  3. package/dist/docs/build.d.ts.map +1 -0
  4. package/dist/docs/build.js +23 -0
  5. package/dist/docs/build.js.map +1 -0
  6. package/dist/docs/index.d.ts +4 -0
  7. package/dist/docs/index.d.ts.map +1 -0
  8. package/dist/docs/index.js +3 -0
  9. package/dist/docs/index.js.map +1 -0
  10. package/dist/docs/render-rule-md.d.ts +4 -0
  11. package/dist/docs/render-rule-md.d.ts.map +1 -0
  12. package/dist/docs/render-rule-md.js +117 -0
  13. package/dist/docs/render-rule-md.js.map +1 -0
  14. package/dist/index.d.ts +12 -0
  15. package/dist/index.d.ts.map +1 -0
  16. package/dist/index.js +7 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/matcher/index.d.ts +4 -0
  19. package/dist/matcher/index.d.ts.map +1 -0
  20. package/dist/matcher/index.js +3 -0
  21. package/dist/matcher/index.js.map +1 -0
  22. package/dist/matcher/traverse.d.ts +3 -0
  23. package/dist/matcher/traverse.d.ts.map +1 -0
  24. package/dist/matcher/traverse.js +9 -0
  25. package/dist/matcher/traverse.js.map +1 -0
  26. package/dist/matcher/yaml-rule.d.ts +31 -0
  27. package/dist/matcher/yaml-rule.d.ts.map +1 -0
  28. package/dist/matcher/yaml-rule.js +30 -0
  29. package/dist/matcher/yaml-rule.js.map +1 -0
  30. package/dist/mcp/handlers.d.ts +36 -0
  31. package/dist/mcp/handlers.d.ts.map +1 -0
  32. package/dist/mcp/handlers.js +32 -0
  33. package/dist/mcp/handlers.js.map +1 -0
  34. package/dist/mcp/index.d.ts +6 -0
  35. package/dist/mcp/index.d.ts.map +1 -0
  36. package/dist/mcp/index.js +4 -0
  37. package/dist/mcp/index.js.map +1 -0
  38. package/dist/mcp/server.d.ts +9 -0
  39. package/dist/mcp/server.d.ts.map +1 -0
  40. package/dist/mcp/server.js +53 -0
  41. package/dist/mcp/server.js.map +1 -0
  42. package/dist/mcp/stdio.d.ts +3 -0
  43. package/dist/mcp/stdio.d.ts.map +1 -0
  44. package/dist/mcp/stdio.js +8 -0
  45. package/dist/mcp/stdio.js.map +1 -0
  46. package/dist/rules-default/ai-explainability-metadata.d.ts +3 -0
  47. package/dist/rules-default/ai-explainability-metadata.d.ts.map +1 -0
  48. package/dist/rules-default/ai-explainability-metadata.js +41 -0
  49. package/dist/rules-default/ai-explainability-metadata.js.map +1 -0
  50. package/dist/rules-default/api-response-shape.d.ts +3 -0
  51. package/dist/rules-default/api-response-shape.d.ts.map +1 -0
  52. package/dist/rules-default/api-response-shape.js +52 -0
  53. package/dist/rules-default/api-response-shape.js.map +1 -0
  54. package/dist/rules-default/async-await-consistency.d.ts +3 -0
  55. package/dist/rules-default/async-await-consistency.d.ts.map +1 -0
  56. package/dist/rules-default/async-await-consistency.js +38 -0
  57. package/dist/rules-default/async-await-consistency.js.map +1 -0
  58. package/dist/rules-default/audit-log-ai-decision.d.ts +3 -0
  59. package/dist/rules-default/audit-log-ai-decision.d.ts.map +1 -0
  60. package/dist/rules-default/audit-log-ai-decision.js +33 -0
  61. package/dist/rules-default/audit-log-ai-decision.js.map +1 -0
  62. package/dist/rules-default/camelcase-migration-column.d.ts +3 -0
  63. package/dist/rules-default/camelcase-migration-column.d.ts.map +1 -0
  64. package/dist/rules-default/camelcase-migration-column.js +56 -0
  65. package/dist/rules-default/camelcase-migration-column.js.map +1 -0
  66. package/dist/rules-default/controller-needs-async-wrapper.d.ts +3 -0
  67. package/dist/rules-default/controller-needs-async-wrapper.d.ts.map +1 -0
  68. package/dist/rules-default/controller-needs-async-wrapper.js +56 -0
  69. package/dist/rules-default/controller-needs-async-wrapper.js.map +1 -0
  70. package/dist/rules-default/enforce-utf8-encoding.d.ts +10 -0
  71. package/dist/rules-default/enforce-utf8-encoding.d.ts.map +1 -0
  72. package/dist/rules-default/enforce-utf8-encoding.js +28 -0
  73. package/dist/rules-default/enforce-utf8-encoding.js.map +1 -0
  74. package/dist/rules-default/explicit-kst-timezone.d.ts +3 -0
  75. package/dist/rules-default/explicit-kst-timezone.d.ts.map +1 -0
  76. package/dist/rules-default/explicit-kst-timezone.js +49 -0
  77. package/dist/rules-default/explicit-kst-timezone.js.map +1 -0
  78. package/dist/rules-default/fk-needs-on-delete.d.ts +3 -0
  79. package/dist/rules-default/fk-needs-on-delete.d.ts.map +1 -0
  80. package/dist/rules-default/fk-needs-on-delete.js +54 -0
  81. package/dist/rules-default/fk-needs-on-delete.js.map +1 -0
  82. package/dist/rules-default/human-oversight-checkpoint.d.ts +3 -0
  83. package/dist/rules-default/human-oversight-checkpoint.d.ts.map +1 -0
  84. package/dist/rules-default/human-oversight-checkpoint.js +37 -0
  85. package/dist/rules-default/human-oversight-checkpoint.js.map +1 -0
  86. package/dist/rules-default/index.d.ts +6 -0
  87. package/dist/rules-default/index.d.ts.map +1 -0
  88. package/dist/rules-default/index.js +138 -0
  89. package/dist/rules-default/index.js.map +1 -0
  90. package/dist/rules-default/korean-comment-encoding.d.ts +3 -0
  91. package/dist/rules-default/korean-comment-encoding.d.ts.map +1 -0
  92. package/dist/rules-default/korean-comment-encoding.js +28 -0
  93. package/dist/rules-default/korean-comment-encoding.js.map +1 -0
  94. package/dist/rules-default/mask-card-number.d.ts +3 -0
  95. package/dist/rules-default/mask-card-number.d.ts.map +1 -0
  96. package/dist/rules-default/mask-card-number.js +45 -0
  97. package/dist/rules-default/mask-card-number.js.map +1 -0
  98. package/dist/rules-default/mask-pii-in-ai-prompt.d.ts +3 -0
  99. package/dist/rules-default/mask-pii-in-ai-prompt.d.ts.map +1 -0
  100. package/dist/rules-default/mask-pii-in-ai-prompt.js +41 -0
  101. package/dist/rules-default/mask-pii-in-ai-prompt.js.map +1 -0
  102. package/dist/rules-default/naver-kakao-oauth-webview.d.ts +3 -0
  103. package/dist/rules-default/naver-kakao-oauth-webview.d.ts.map +1 -0
  104. package/dist/rules-default/naver-kakao-oauth-webview.js +40 -0
  105. package/dist/rules-default/naver-kakao-oauth-webview.js.map +1 -0
  106. package/dist/rules-default/no-bare-except.d.ts +7 -0
  107. package/dist/rules-default/no-bare-except.d.ts.map +1 -0
  108. package/dist/rules-default/no-bare-except.js +23 -0
  109. package/dist/rules-default/no-bare-except.js.map +1 -0
  110. package/dist/rules-default/no-bare-throw.d.ts +7 -0
  111. package/dist/rules-default/no-bare-throw.d.ts.map +1 -0
  112. package/dist/rules-default/no-bare-throw.js +32 -0
  113. package/dist/rules-default/no-bare-throw.js.map +1 -0
  114. package/dist/rules-default/no-boolean-trap.d.ts +8 -0
  115. package/dist/rules-default/no-boolean-trap.d.ts.map +1 -0
  116. package/dist/rules-default/no-boolean-trap.js +33 -0
  117. package/dist/rules-default/no-boolean-trap.js.map +1 -0
  118. package/dist/rules-default/no-console-log.d.ts +3 -0
  119. package/dist/rules-default/no-console-log.d.ts.map +1 -0
  120. package/dist/rules-default/no-console-log.js +19 -0
  121. package/dist/rules-default/no-console-log.js.map +1 -0
  122. package/dist/rules-default/no-cvv-logging.d.ts +3 -0
  123. package/dist/rules-default/no-cvv-logging.d.ts.map +1 -0
  124. package/dist/rules-default/no-cvv-logging.js +36 -0
  125. package/dist/rules-default/no-cvv-logging.js.map +1 -0
  126. package/dist/rules-default/no-default-export-from-libs.d.ts +7 -0
  127. package/dist/rules-default/no-default-export-from-libs.d.ts.map +1 -0
  128. package/dist/rules-default/no-default-export-from-libs.js +24 -0
  129. package/dist/rules-default/no-default-export-from-libs.js.map +1 -0
  130. package/dist/rules-default/no-direct-anthropic.yaml +9 -0
  131. package/dist/rules-default/no-direct-openai.yaml +9 -0
  132. package/dist/rules-default/no-empty-catch.d.ts +7 -0
  133. package/dist/rules-default/no-empty-catch.d.ts.map +1 -0
  134. package/dist/rules-default/no-empty-catch.js +30 -0
  135. package/dist/rules-default/no-empty-catch.js.map +1 -0
  136. package/dist/rules-default/no-fstring-sql.d.ts +3 -0
  137. package/dist/rules-default/no-fstring-sql.d.ts.map +1 -0
  138. package/dist/rules-default/no-fstring-sql.js +24 -0
  139. package/dist/rules-default/no-fstring-sql.js.map +1 -0
  140. package/dist/rules-default/no-id-overwrite.d.ts +7 -0
  141. package/dist/rules-default/no-id-overwrite.d.ts.map +1 -0
  142. package/dist/rules-default/no-id-overwrite.js +26 -0
  143. package/dist/rules-default/no-id-overwrite.js.map +1 -0
  144. package/dist/rules-default/no-inline-date.yaml +9 -0
  145. package/dist/rules-default/no-inline-math-round.yaml +12 -0
  146. package/dist/rules-default/no-jsonb-circular.d.ts +3 -0
  147. package/dist/rules-default/no-jsonb-circular.d.ts.map +1 -0
  148. package/dist/rules-default/no-jsonb-circular.js +33 -0
  149. package/dist/rules-default/no-jsonb-circular.js.map +1 -0
  150. package/dist/rules-default/no-magic-number.d.ts +3 -0
  151. package/dist/rules-default/no-magic-number.d.ts.map +1 -0
  152. package/dist/rules-default/no-magic-number.js +42 -0
  153. package/dist/rules-default/no-magic-number.js.map +1 -0
  154. package/dist/rules-default/no-mutable-default-arg.d.ts +3 -0
  155. package/dist/rules-default/no-mutable-default-arg.d.ts.map +1 -0
  156. package/dist/rules-default/no-mutable-default-arg.js +30 -0
  157. package/dist/rules-default/no-mutable-default-arg.js.map +1 -0
  158. package/dist/rules-default/no-pickle.yaml +12 -0
  159. package/dist/rules-default/no-plain-card-number.d.ts +3 -0
  160. package/dist/rules-default/no-plain-card-number.d.ts.map +1 -0
  161. package/dist/rules-default/no-plain-card-number.js +51 -0
  162. package/dist/rules-default/no-plain-card-number.js.map +1 -0
  163. package/dist/rules-default/no-print-in-prod.yaml +9 -0
  164. package/dist/rules-default/no-process-env-leak.d.ts +7 -0
  165. package/dist/rules-default/no-process-env-leak.d.ts.map +1 -0
  166. package/dist/rules-default/no-process-env-leak.js +34 -0
  167. package/dist/rules-default/no-process-env-leak.js.map +1 -0
  168. package/dist/rules-default/no-shell-true.d.ts +3 -0
  169. package/dist/rules-default/no-shell-true.d.ts.map +1 -0
  170. package/dist/rules-default/no-shell-true.js +42 -0
  171. package/dist/rules-default/no-shell-true.js.map +1 -0
  172. package/dist/rules-default/prefer-const-array.d.ts +9 -0
  173. package/dist/rules-default/prefer-const-array.d.ts.map +1 -0
  174. package/dist/rules-default/prefer-const-array.js +34 -0
  175. package/dist/rules-default/prefer-const-array.js.map +1 -0
  176. package/dist/rules-default/prefer-named-imports.d.ts +8 -0
  177. package/dist/rules-default/prefer-named-imports.d.ts.map +1 -0
  178. package/dist/rules-default/prefer-named-imports.js +25 -0
  179. package/dist/rules-default/prefer-named-imports.js.map +1 -0
  180. package/dist/rules-default/preserve-transaction-log.d.ts +3 -0
  181. package/dist/rules-default/preserve-transaction-log.d.ts.map +1 -0
  182. package/dist/rules-default/preserve-transaction-log.js +33 -0
  183. package/dist/rules-default/preserve-transaction-log.js.map +1 -0
  184. package/dist/rules-default/pytest-fixture-naming.d.ts +3 -0
  185. package/dist/rules-default/pytest-fixture-naming.d.ts.map +1 -0
  186. package/dist/rules-default/pytest-fixture-naming.js +36 -0
  187. package/dist/rules-default/pytest-fixture-naming.js.map +1 -0
  188. package/dist/rules-default/requests-needs-timeout.d.ts +3 -0
  189. package/dist/rules-default/requests-needs-timeout.d.ts.map +1 -0
  190. package/dist/rules-default/requests-needs-timeout.js +43 -0
  191. package/dist/rules-default/requests-needs-timeout.js.map +1 -0
  192. package/dist/rules-default/require-idempotency-key.d.ts +3 -0
  193. package/dist/rules-default/require-idempotency-key.d.ts.map +1 -0
  194. package/dist/rules-default/require-idempotency-key.js +47 -0
  195. package/dist/rules-default/require-idempotency-key.js.map +1 -0
  196. package/dist/rules-default/require-tls-1-2-plus.d.ts +3 -0
  197. package/dist/rules-default/require-tls-1-2-plus.d.ts.map +1 -0
  198. package/dist/rules-default/require-tls-1-2-plus.js +30 -0
  199. package/dist/rules-default/require-tls-1-2-plus.js.map +1 -0
  200. package/dist/rules-default/rfc5987-korean-filename.d.ts +3 -0
  201. package/dist/rules-default/rfc5987-korean-filename.d.ts.map +1 -0
  202. package/dist/rules-default/rfc5987-korean-filename.js +32 -0
  203. package/dist/rules-default/rfc5987-korean-filename.js.map +1 -0
  204. package/dist/rules-default/route-needs-auth.d.ts +3 -0
  205. package/dist/rules-default/route-needs-auth.d.ts.map +1 -0
  206. package/dist/rules-default/route-needs-auth.js +48 -0
  207. package/dist/rules-default/route-needs-auth.js.map +1 -0
  208. package/dist/rules-default/route-needs-rate-limit.d.ts +3 -0
  209. package/dist/rules-default/route-needs-rate-limit.d.ts.map +1 -0
  210. package/dist/rules-default/route-needs-rate-limit.js +47 -0
  211. package/dist/rules-default/route-needs-rate-limit.js.map +1 -0
  212. package/dist/rules-default/separate-refund-permission.d.ts +3 -0
  213. package/dist/rules-default/separate-refund-permission.d.ts.map +1 -0
  214. package/dist/rules-default/separate-refund-permission.js +33 -0
  215. package/dist/rules-default/separate-refund-permission.js.map +1 -0
  216. package/dist/rules-default/track-ai-model-version.d.ts +3 -0
  217. package/dist/rules-default/track-ai-model-version.d.ts.map +1 -0
  218. package/dist/rules-default/track-ai-model-version.js +37 -0
  219. package/dist/rules-default/track-ai-model-version.js.map +1 -0
  220. package/dist/rules-default/type-hint-required-public.d.ts +10 -0
  221. package/dist/rules-default/type-hint-required-public.d.ts.map +1 -0
  222. package/dist/rules-default/type-hint-required-public.js +29 -0
  223. package/dist/rules-default/type-hint-required-public.js.map +1 -0
  224. package/dist/rules-default/verify-pg-response.d.ts +3 -0
  225. package/dist/rules-default/verify-pg-response.d.ts.map +1 -0
  226. package/dist/rules-default/verify-pg-response.js +37 -0
  227. package/dist/rules-default/verify-pg-response.js.map +1 -0
  228. package/dist/rules-default/won-format-thousands.d.ts +3 -0
  229. package/dist/rules-default/won-format-thousands.d.ts.map +1 -0
  230. package/dist/rules-default/won-format-thousands.js +24 -0
  231. package/dist/rules-default/won-format-thousands.js.map +1 -0
  232. package/dist/runner/context.d.ts +12 -0
  233. package/dist/runner/context.d.ts.map +1 -0
  234. package/dist/runner/context.js +34 -0
  235. package/dist/runner/context.js.map +1 -0
  236. package/dist/runner/index.d.ts +7 -0
  237. package/dist/runner/index.d.ts.map +1 -0
  238. package/dist/runner/index.js +5 -0
  239. package/dist/runner/index.js.map +1 -0
  240. package/dist/runner/ruleset-signature.d.ts +4 -0
  241. package/dist/runner/ruleset-signature.d.ts.map +1 -0
  242. package/dist/runner/ruleset-signature.js +11 -0
  243. package/dist/runner/ruleset-signature.js.map +1 -0
  244. package/dist/runner/run-file.d.ts +10 -0
  245. package/dist/runner/run-file.d.ts.map +1 -0
  246. package/dist/runner/run-file.js +20 -0
  247. package/dist/runner/run-file.js.map +1 -0
  248. package/dist/runner/run-project.d.ts +12 -0
  249. package/dist/runner/run-project.d.ts.map +1 -0
  250. package/dist/runner/run-project.js +55 -0
  251. package/dist/runner/run-project.js.map +1 -0
  252. package/dist/runner/run-rule.d.ts +5 -0
  253. package/dist/runner/run-rule.d.ts.map +1 -0
  254. package/dist/runner/run-rule.js +37 -0
  255. package/dist/runner/run-rule.js.map +1 -0
  256. package/dist/sync/index.d.ts +5 -0
  257. package/dist/sync/index.d.ts.map +1 -0
  258. package/dist/sync/index.js +3 -0
  259. package/dist/sync/index.js.map +1 -0
  260. package/dist/sync/render.d.ts +9 -0
  261. package/dist/sync/render.d.ts.map +1 -0
  262. package/dist/sync/render.js +70 -0
  263. package/dist/sync/render.js.map +1 -0
  264. package/dist/sync/sync-files.d.ts +12 -0
  265. package/dist/sync/sync-files.d.ts.map +1 -0
  266. package/dist/sync/sync-files.js +29 -0
  267. package/dist/sync/sync-files.js.map +1 -0
  268. package/package.json +62 -0
  269. package/scripts/copy-yaml-rules.mjs +19 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-pg-response.d.ts","sourceRoot":"","sources":["../../src/rules-default/verify-pg-response.ts"],"names":[],"mappings":";AA0BA,wBAYG"}
@@ -0,0 +1,37 @@
1
+ import { defineRule } from '@aicqtools/rule-sdk';
2
+ /**
3
+ * PCI DSS / payment-domain best practice: external PG (payment gateway)
4
+ * responses must be verified (signature/hash/checksum) before being
5
+ * trusted for downstream state changes.
6
+ *
7
+ * Heuristic: function bodies that contain a PG-domain HTTP call
8
+ * (`/payments`, `/pay`, `pg.`) and proceed without invoking
9
+ * `signature` / `hash` / `checksum` / `verify`.
10
+ *
11
+ * Limitation: false negatives if verification is delegated to a wrapper
12
+ * function. Severity is `warning` to limit noise.
13
+ */
14
+ const PG_CALL = /\b(axios|fetch|got|http\.(get|post)|httpClient)\b.*\b(payments?|pg|inicis|toss|kakao|naver)\b/i;
15
+ const VERIFICATION = /\b(signature|hash|checksum|verify|hmac|hmacSha)\b/i;
16
+ function check(node, ctx) {
17
+ const text = ctx.textOf(node);
18
+ if (!PG_CALL.test(text))
19
+ return;
20
+ if (VERIFICATION.test(text))
21
+ return;
22
+ ctx.report({ node });
23
+ }
24
+ export default defineRule({
25
+ id: 'verify-pg-response',
26
+ language: ['typescript', 'tsx'],
27
+ severity: 'warning',
28
+ message: 'Payment gateway response is consumed without signature/hash verification (PCI DSS § 6.2.4 — input validation; payment-domain best practice).',
29
+ messageKo: '결제 게이트웨이 응답을 서명/해시 검증 없이 사용합니다 (PCI DSS § 6.2.4 — 입력 검증; 결제 도메인 권장 사항).',
30
+ docs: 'https://github.com/aicqtools/aicqtools/blob/main/docs/rules/verify-pg-response.md',
31
+ visitors: {
32
+ function_declaration: check,
33
+ arrow_function: check,
34
+ method_definition: check,
35
+ },
36
+ });
37
+ //# sourceMappingURL=verify-pg-response.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify-pg-response.js","sourceRoot":"","sources":["../../src/rules-default/verify-pg-response.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAGjD;;;;;;;;;;;GAWG;AACH,MAAM,OAAO,GAAG,gGAAgG,CAAC;AACjH,MAAM,YAAY,GAAG,oDAAoD,CAAC;AAE1E,SAAS,KAAK,CAAC,IAAuB,EAAE,GAAgB;IACtD,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IAC9B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IAChC,IAAI,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IACpC,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;AACvB,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,oBAAoB;IACxB,QAAQ,EAAE,CAAC,YAAY,EAAE,KAAK,CAAC;IAC/B,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,8IAA8I;IACvJ,SAAS,EAAE,yEAAyE;IACpF,IAAI,EAAE,mFAAmF;IACzF,QAAQ,EAAE;QACR,oBAAoB,EAAE,KAAK;QAC3B,cAAc,EAAE,KAAK;QACrB,iBAAiB,EAAE,KAAK;KACzB;CACF,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ declare const _default: import("@aicqtools/rule-sdk").FunctionRule;
2
+ export default _default;
3
+ //# sourceMappingURL=won-format-thousands.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"won-format-thousands.d.ts","sourceRoot":"","sources":["../../src/rules-default/won-format-thousands.ts"],"names":[],"mappings":";AAWA,wBAYG"}
@@ -0,0 +1,24 @@
1
+ import { defineRule } from '@aicqtools/rule-sdk';
2
+ /**
3
+ * Detect string literals like `"5000원"` or `"50000원"` (≥4 digits) without a thousands
4
+ * separator. UX best practice in Korea: always show ₩ amounts as `5,000원`.
5
+ *
6
+ * Known limitation: doesn't catch programmatically-formatted amounts (template literals
7
+ * with computed values are intentionally skipped to avoid false positives).
8
+ */
9
+ const WON_NUMBER_PATTERN = /["'`](\d{4,})원["'`]/;
10
+ export default defineRule({
11
+ id: 'won-format-thousands',
12
+ language: ['typescript', 'javascript', 'tsx'],
13
+ severity: 'info',
14
+ message: 'Won amount should use thousands separator: `5,000원` instead of `5000원`.',
15
+ messageKo: '원화 금액은 천단위 콤마 권장: `5000원` → `5,000원`.',
16
+ visitors: {
17
+ string(node, ctx) {
18
+ const text = ctx.textOf(node);
19
+ if (WON_NUMBER_PATTERN.test(text))
20
+ ctx.report({ node });
21
+ },
22
+ },
23
+ });
24
+ //# sourceMappingURL=won-format-thousands.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"won-format-thousands.js","sourceRoot":"","sources":["../../src/rules-default/won-format-thousands.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD;;;;;;GAMG;AACH,MAAM,kBAAkB,GAAG,qBAAqB,CAAC;AAEjD,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,sBAAsB;IAC1B,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,KAAK,CAAC;IAC7C,QAAQ,EAAE,MAAM;IAChB,OAAO,EAAE,yEAAyE;IAClF,SAAS,EAAE,uCAAuC;IAClD,QAAQ,EAAE;QACR,MAAM,CAAC,IAAI,EAAE,GAAG;YACd,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC9B,IAAI,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QAC1D,CAAC;KACF;CACF,CAAC,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type Parser from 'tree-sitter';
2
+ import type { Diagnostic, Language, Range } from '@aicqtools/core';
3
+ import type { RuleContext, RuleMeta } from '@aicqtools/rule-sdk';
4
+ export declare function rangeOfNode(node: Parser.SyntaxNode): Range;
5
+ export interface RunContext {
6
+ readonly filePath: string;
7
+ readonly source: string;
8
+ readonly language: Language;
9
+ readonly diagnostics: Diagnostic[];
10
+ }
11
+ export declare function makeRuleContext(run: RunContext, meta: RuleMeta): RuleContext;
12
+ //# sourceMappingURL=context.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/runner/context.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,MAAM,aAAa,CAAC;AACtC,OAAO,KAAK,EAAE,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAY,MAAM,iBAAiB,CAAC;AAC7E,OAAO,KAAK,EAAE,WAAW,EAAE,QAAQ,EAAc,MAAM,qBAAqB,CAAC;AAE7E,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,CAAC,UAAU,GAAG,KAAK,CAK1D;AAED,MAAM,WAAW,UAAU;IACzB,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;IAC1B,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC;IACxB,QAAQ,CAAC,QAAQ,EAAE,QAAQ,CAAC;IAC5B,QAAQ,CAAC,WAAW,EAAE,UAAU,EAAE,CAAC;CACpC;AAED,wBAAgB,eAAe,CAAC,GAAG,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,GAAG,WAAW,CA0B5E"}
@@ -0,0 +1,34 @@
1
+ export function rangeOfNode(node) {
2
+ return {
3
+ start: { line: node.startPosition.row + 1, column: node.startPosition.column + 1 },
4
+ end: { line: node.endPosition.row + 1, column: node.endPosition.column + 1 },
5
+ };
6
+ }
7
+ export function makeRuleContext(run, meta) {
8
+ return {
9
+ filePath: run.filePath,
10
+ source: run.source,
11
+ language: run.language,
12
+ textOf(node) {
13
+ return run.source.slice(node.startIndex, node.endIndex);
14
+ },
15
+ rangeOf(node) {
16
+ return rangeOfNode(node);
17
+ },
18
+ report(args) {
19
+ const diagnostic = {
20
+ ruleId: meta.id,
21
+ severity: meta.severity,
22
+ message: args.message ?? meta.message,
23
+ ...(args.messageKo ?? meta.messageKo
24
+ ? { messageKo: args.messageKo ?? meta.messageKo }
25
+ : {}),
26
+ file: run.filePath,
27
+ range: rangeOfNode(args.node),
28
+ ...(meta.docs ? { docs: meta.docs } : {}),
29
+ };
30
+ run.diagnostics.push(diagnostic);
31
+ },
32
+ };
33
+ }
34
+ //# sourceMappingURL=context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/runner/context.ts"],"names":[],"mappings":"AAIA,MAAM,UAAU,WAAW,CAAC,IAAuB;IACjD,OAAO;QACL,KAAK,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,aAAa,CAAC,GAAG,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE;QAClF,GAAG,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,GAAG,GAAG,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE;KAC7E,CAAC;AACJ,CAAC;AASD,MAAM,UAAU,eAAe,CAAC,GAAe,EAAE,IAAc;IAC7D,OAAO;QACL,QAAQ,EAAE,GAAG,CAAC,QAAQ;QACtB,MAAM,EAAE,GAAG,CAAC,MAAM;QAClB,QAAQ,EAAE,GAAG,CAAC,QAAQ;QACtB,MAAM,CAAC,IAAI;YACT,OAAO,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC1D,CAAC;QACD,OAAO,CAAC,IAAI;YACV,OAAO,WAAW,CAAC,IAAI,CAAC,CAAC;QAC3B,CAAC;QACD,MAAM,CAAC,IAAgB;YACrB,MAAM,UAAU,GAAe;gBAC7B,MAAM,EAAE,IAAI,CAAC,EAAE;gBACf,QAAQ,EAAE,IAAI,CAAC,QAAoB;gBACnC,OAAO,EAAE,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,OAAO;gBACrC,GAAG,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,SAAS;oBAClC,CAAC,CAAC,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,SAAS,EAAE;oBACjD,CAAC,CAAC,EAAE,CAAC;gBACP,IAAI,EAAE,GAAG,CAAC,QAAQ;gBAClB,KAAK,EAAE,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC;gBAC7B,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;aAC5B,CAAC;YAChB,GAAG,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACnC,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,7 @@
1
+ export { runFile, runFileWithSource } from './run-file.js';
2
+ export { runProject } from './run-project.js';
3
+ export { runRule } from './run-rule.js';
4
+ export { ruleSignature, rulesetSignature } from './ruleset-signature.js';
5
+ export type { RunFileResult } from './run-file.js';
6
+ export type { RunProjectOptions } from './run-project.js';
7
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/runner/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AACxC,OAAO,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AACzE,YAAY,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AACnD,YAAY,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC"}
@@ -0,0 +1,5 @@
1
+ export { runFile, runFileWithSource } from './run-file.js';
2
+ export { runProject } from './run-project.js';
3
+ export { runRule } from './run-rule.js';
4
+ export { ruleSignature, rulesetSignature } from './ruleset-signature.js';
5
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/runner/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAC9C,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AACxC,OAAO,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { Rule } from '@aicqtools/rule-sdk';
2
+ export declare function ruleSignature(rule: Rule): string;
3
+ export declare function rulesetSignature(rules: readonly Rule[]): string[];
4
+ //# sourceMappingURL=ruleset-signature.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ruleset-signature.d.ts","sourceRoot":"","sources":["../../src/runner/ruleset-signature.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AAEhD,wBAAgB,aAAa,CAAC,IAAI,EAAE,IAAI,GAAG,MAAM,CAMhD;AAED,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,GAAG,MAAM,EAAE,CAEjE"}
@@ -0,0 +1,11 @@
1
+ export function ruleSignature(rule) {
2
+ if (rule.kind === 'pattern') {
3
+ return `pattern:${rule.id}:${rule.severity}:${rule.query}`;
4
+ }
5
+ const visitorKeys = Object.keys(rule.visitors).sort().join(',');
6
+ return `function:${rule.id}:${rule.severity}:${visitorKeys}`;
7
+ }
8
+ export function rulesetSignature(rules) {
9
+ return rules.map(ruleSignature);
10
+ }
11
+ //# sourceMappingURL=ruleset-signature.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ruleset-signature.js","sourceRoot":"","sources":["../../src/runner/ruleset-signature.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,aAAa,CAAC,IAAU;IACtC,IAAI,IAAI,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;QAC5B,OAAO,WAAW,IAAI,CAAC,EAAE,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;IAC7D,CAAC;IACD,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAChE,OAAO,YAAY,IAAI,CAAC,EAAE,IAAI,IAAI,CAAC,QAAQ,IAAI,WAAW,EAAE,CAAC;AAC/D,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,KAAsB;IACrD,OAAO,KAAK,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;AAClC,CAAC"}
@@ -0,0 +1,10 @@
1
+ import type { Diagnostic, Language } from '@aicqtools/core';
2
+ import type { Rule } from '@aicqtools/rule-sdk';
3
+ export interface RunFileResult {
4
+ readonly filePath: string;
5
+ readonly language: Language | null;
6
+ readonly diagnostics: readonly Diagnostic[];
7
+ }
8
+ export declare function runFile(filePath: string, rules: readonly Rule[]): Promise<RunFileResult>;
9
+ export declare function runFileWithSource(filePath: string, source: string, language: Language, rules: readonly Rule[]): RunFileResult;
10
+ //# sourceMappingURL=run-file.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-file.d.ts","sourceRoot":"","sources":["../../src/runner/run-file.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAE5D,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AAGhD,MAAM,WAAW,aAAa;IAC5B,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;IAC1B,QAAQ,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI,CAAC;IACnC,QAAQ,CAAC,WAAW,EAAE,SAAS,UAAU,EAAE,CAAC;CAC7C;AAED,wBAAsB,OAAO,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,SAAS,IAAI,EAAE,GAAG,OAAO,CAAC,aAAa,CAAC,CAM9F;AAED,wBAAgB,iBAAiB,CAC/B,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,MAAM,EACd,QAAQ,EAAE,QAAQ,EAClB,KAAK,EAAE,SAAS,IAAI,EAAE,GACrB,aAAa,CAQf"}
@@ -0,0 +1,20 @@
1
+ import { readFile } from 'node:fs/promises';
2
+ import { detectLanguage, parseSource } from '@aicqtools/core';
3
+ import { runRule } from './run-rule.js';
4
+ export async function runFile(filePath, rules) {
5
+ const language = detectLanguage(filePath);
6
+ if (!language)
7
+ return { filePath, language: null, diagnostics: [] };
8
+ const source = await readFile(filePath, 'utf-8');
9
+ return runFileWithSource(filePath, source, language, rules);
10
+ }
11
+ export function runFileWithSource(filePath, source, language, rules) {
12
+ const tree = parseSource(language, source);
13
+ const diagnostics = [];
14
+ const run = { filePath, source, language, diagnostics };
15
+ for (const rule of rules) {
16
+ runRule(rule, run, tree);
17
+ }
18
+ return { filePath, language, diagnostics };
19
+ }
20
+ //# sourceMappingURL=run-file.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-file.js","sourceRoot":"","sources":["../../src/runner/run-file.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAE5C,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAE9D,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AAQxC,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,QAAgB,EAAE,KAAsB;IACpE,MAAM,QAAQ,GAAG,cAAc,CAAC,QAAQ,CAAC,CAAC;IAC1C,IAAI,CAAC,QAAQ;QAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,EAAE,CAAC;IAEpE,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IACjD,OAAO,iBAAiB,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC;AAC9D,CAAC;AAED,MAAM,UAAU,iBAAiB,CAC/B,QAAgB,EAChB,MAAc,EACd,QAAkB,EAClB,KAAsB;IAEtB,MAAM,IAAI,GAAG,WAAW,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC3C,MAAM,WAAW,GAAiB,EAAE,CAAC;IACrC,MAAM,GAAG,GAAG,EAAE,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;IACxD,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,OAAO,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;IAC3B,CAAC;IACD,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,WAAW,EAAE,CAAC;AAC7C,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { CheckResult } from '@aicqtools/core';
2
+ import { FileCache } from '@aicqtools/core';
3
+ import type { Rule } from '@aicqtools/rule-sdk';
4
+ export interface RunProjectOptions {
5
+ readonly cwd: string;
6
+ readonly include: readonly string[];
7
+ readonly exclude: readonly string[];
8
+ readonly rules: readonly Rule[];
9
+ readonly cache?: FileCache;
10
+ }
11
+ export declare function runProject(opts: RunProjectOptions): Promise<CheckResult>;
12
+ //# sourceMappingURL=run-project.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-project.d.ts","sourceRoot":"","sources":["../../src/runner/run-project.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,WAAW,EAAc,MAAM,iBAAiB,CAAC;AAC/D,OAAO,EAAE,SAAS,EAAwB,MAAM,iBAAiB,CAAC;AAClE,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AAIhD,MAAM,WAAW,iBAAiB;IAChC,QAAQ,CAAC,GAAG,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,OAAO,EAAE,SAAS,MAAM,EAAE,CAAC;IACpC,QAAQ,CAAC,OAAO,EAAE,SAAS,MAAM,EAAE,CAAC;IACpC,QAAQ,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,CAAC;IAChC,QAAQ,CAAC,KAAK,CAAC,EAAE,SAAS,CAAC;CAC5B;AAED,wBAAsB,UAAU,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAAC,WAAW,CAAC,CAqD9E"}
@@ -0,0 +1,55 @@
1
+ import fastGlob from 'fast-glob';
2
+ import { stat } from 'node:fs/promises';
3
+ import { FileCache, hashRulesetSignature } from '@aicqtools/core';
4
+ import { runFile } from './run-file.js';
5
+ import { rulesetSignature } from './ruleset-signature.js';
6
+ export async function runProject(opts) {
7
+ const start = Date.now();
8
+ const files = await fastGlob([...opts.include], {
9
+ cwd: opts.cwd,
10
+ ignore: [...opts.exclude],
11
+ absolute: true,
12
+ onlyFiles: true,
13
+ dot: false,
14
+ });
15
+ const cache = opts.cache;
16
+ const rulesHash = cache ? hashRulesetSignature(rulesetSignature(opts.rules)) : '';
17
+ const diagnostics = [];
18
+ for (const file of files) {
19
+ if (cache) {
20
+ try {
21
+ const st = await stat(file);
22
+ const cached = cache.get({
23
+ filePath: file,
24
+ mtime: Math.floor(st.mtimeMs),
25
+ size: st.size,
26
+ rulesHash,
27
+ });
28
+ if (cached) {
29
+ diagnostics.push(...cached);
30
+ continue;
31
+ }
32
+ const result = await runFile(file, opts.rules);
33
+ cache.set({
34
+ filePath: file,
35
+ mtime: Math.floor(st.mtimeMs),
36
+ size: st.size,
37
+ rulesHash,
38
+ }, result.diagnostics);
39
+ diagnostics.push(...result.diagnostics);
40
+ continue;
41
+ }
42
+ catch {
43
+ // fall through to non-cached path
44
+ }
45
+ }
46
+ const result = await runFile(file, opts.rules);
47
+ diagnostics.push(...result.diagnostics);
48
+ }
49
+ return {
50
+ diagnostics,
51
+ filesScanned: files.length,
52
+ durationMs: Date.now() - start,
53
+ };
54
+ }
55
+ //# sourceMappingURL=run-project.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-project.js","sourceRoot":"","sources":["../../src/runner/run-project.ts"],"names":[],"mappings":"AAAA,OAAO,QAAQ,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AAExC,OAAO,EAAE,SAAS,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AAElE,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,CAAC;AACxC,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAU1D,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,IAAuB;IACtD,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACzB,MAAM,KAAK,GAAG,MAAM,QAAQ,CAAC,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,EAAE;QAC9C,GAAG,EAAE,IAAI,CAAC,GAAG;QACb,MAAM,EAAE,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC;QACzB,QAAQ,EAAE,IAAI;QACd,SAAS,EAAE,IAAI;QACf,GAAG,EAAE,KAAK;KACX,CAAC,CAAC;IAEH,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC;IACzB,MAAM,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,oBAAoB,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAClF,MAAM,WAAW,GAAiB,EAAE,CAAC;IAErC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,IAAI,KAAK,EAAE,CAAC;YACV,IAAI,CAAC;gBACH,MAAM,EAAE,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC5B,MAAM,MAAM,GAAG,KAAK,CAAC,GAAG,CAAC;oBACvB,QAAQ,EAAE,IAAI;oBACd,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,CAAC;oBAC7B,IAAI,EAAE,EAAE,CAAC,IAAI;oBACb,SAAS;iBACV,CAAC,CAAC;gBACH,IAAI,MAAM,EAAE,CAAC;oBACX,WAAW,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,CAAC;oBAC5B,SAAS;gBACX,CAAC;gBACD,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC/C,KAAK,CAAC,GAAG,CACP;oBACE,QAAQ,EAAE,IAAI;oBACd,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,CAAC;oBAC7B,IAAI,EAAE,EAAE,CAAC,IAAI;oBACb,SAAS;iBACV,EACD,MAAM,CAAC,WAAW,CACnB,CAAC;gBACF,WAAW,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;gBACxC,SAAS;YACX,CAAC;YAAC,MAAM,CAAC;gBACP,kCAAkC;YACpC,CAAC;QACH,CAAC;QACD,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAC/C,WAAW,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;IAC1C,CAAC;IAED,OAAO;QACL,WAAW;QACX,YAAY,EAAE,KAAK,CAAC,MAAM;QAC1B,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK;KAC/B,CAAC;AACJ,CAAC"}
@@ -0,0 +1,5 @@
1
+ import Parser from 'tree-sitter';
2
+ import type { Rule } from '@aicqtools/rule-sdk';
3
+ import { type RunContext } from './context.js';
4
+ export declare function runRule(rule: Rule, run: RunContext, tree: Parser.Tree): void;
5
+ //# sourceMappingURL=run-rule.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-rule.d.ts","sourceRoot":"","sources":["../../src/runner/run-rule.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,aAAa,CAAC;AAEjC,OAAO,KAAK,EAAE,IAAI,EAA6B,MAAM,qBAAqB,CAAC;AAE3E,OAAO,EAAmB,KAAK,UAAU,EAAE,MAAM,cAAc,CAAC;AA4BhE,wBAAgB,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,GAAG,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,CAAC,IAAI,GAAG,IAAI,CAI5E"}
@@ -0,0 +1,37 @@
1
+ import Parser from 'tree-sitter';
2
+ import { traverse } from '../matcher/traverse.js';
3
+ import { makeRuleContext } from './context.js';
4
+ import { getParser } from '@aicqtools/core';
5
+ function ruleAppliesTo(rule, language) {
6
+ const langs = Array.isArray(rule.language) ? rule.language : [rule.language];
7
+ return langs.includes(language);
8
+ }
9
+ function runFunctionRule(rule, run, tree) {
10
+ const ctx = makeRuleContext(run, rule);
11
+ traverse(tree.rootNode, (node) => {
12
+ const visitor = rule.visitors[node.type];
13
+ if (visitor)
14
+ visitor(node, ctx);
15
+ });
16
+ }
17
+ function runPatternRule(rule, run, tree) {
18
+ const parser = getParser(run.language);
19
+ const lang = parser.getLanguage();
20
+ const query = new Parser.Query(lang, rule.query);
21
+ const matches = query.matches(tree.rootNode);
22
+ const ctx = makeRuleContext(run, rule);
23
+ for (const match of matches) {
24
+ const target = match.captures[0]?.node;
25
+ if (target)
26
+ ctx.report({ node: target });
27
+ }
28
+ }
29
+ export function runRule(rule, run, tree) {
30
+ if (!ruleAppliesTo(rule, run.language))
31
+ return;
32
+ if (rule.kind === 'function')
33
+ runFunctionRule(rule, run, tree);
34
+ else
35
+ runPatternRule(rule, run, tree);
36
+ }
37
+ //# sourceMappingURL=run-rule.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-rule.js","sourceRoot":"","sources":["../../src/runner/run-rule.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,aAAa,CAAC;AAGjC,OAAO,EAAE,QAAQ,EAAE,MAAM,wBAAwB,CAAC;AAClD,OAAO,EAAE,eAAe,EAAmB,MAAM,cAAc,CAAC;AAChE,OAAO,EAAE,SAAS,EAAE,MAAM,iBAAiB,CAAC;AAE5C,SAAS,aAAa,CAAC,IAAU,EAAE,QAAkB;IACnD,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC7E,OAAO,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;AAClC,CAAC;AAED,SAAS,eAAe,CAAC,IAAkB,EAAE,GAAe,EAAE,IAAiB;IAC7E,MAAM,GAAG,GAAG,eAAe,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IACvC,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,IAAI,EAAE,EAAE;QAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzC,IAAI,OAAO;YAAE,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,cAAc,CAAC,IAAiB,EAAE,GAAe,EAAE,IAAiB;IAC3E,MAAM,MAAM,GAAG,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACvC,MAAM,IAAI,GAAG,MAAM,CAAC,WAAW,EAAE,CAAC;IAClC,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,KAAK,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;IACjD,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC7C,MAAM,GAAG,GAAG,eAAe,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IACvC,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;QAC5B,MAAM,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC;QACvC,IAAI,MAAM;YAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;IAC3C,CAAC;AACH,CAAC;AAED,MAAM,UAAU,OAAO,CAAC,IAAU,EAAE,GAAe,EAAE,IAAiB;IACpE,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,QAAQ,CAAC;QAAE,OAAO;IAC/C,IAAI,IAAI,CAAC,IAAI,KAAK,UAAU;QAAE,eAAe,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;;QAC1D,cAAc,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;AACvC,CAAC"}
@@ -0,0 +1,5 @@
1
+ export { renderRules, injectIntoMarkdown, MARKER_START, MARKER_END, } from './render.js';
2
+ export type { RenderOptions } from './render.js';
3
+ export { syncAiRules } from './sync-files.js';
4
+ export type { SyncOptions, SyncTarget } from './sync-files.js';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/sync/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,WAAW,EACX,kBAAkB,EAClB,YAAY,EACZ,UAAU,GACX,MAAM,aAAa,CAAC;AACrB,YAAY,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,YAAY,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC"}
@@ -0,0 +1,3 @@
1
+ export { renderRules, injectIntoMarkdown, MARKER_START, MARKER_END, } from './render.js';
2
+ export { syncAiRules } from './sync-files.js';
3
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sync/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,WAAW,EACX,kBAAkB,EAClB,YAAY,EACZ,UAAU,GACX,MAAM,aAAa,CAAC;AAErB,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { Rule } from '@aicqtools/rule-sdk';
2
+ export interface RenderOptions {
3
+ readonly locale: 'ko' | 'en';
4
+ }
5
+ export declare function renderRules(rules: readonly Rule[], opts: RenderOptions): string;
6
+ export declare const MARKER_START = "<!-- aicq:rules:start -->";
7
+ export declare const MARKER_END = "<!-- aicq:rules:end -->";
8
+ export declare function injectIntoMarkdown(existing: string | null, body: string): string;
9
+ //# sourceMappingURL=render.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"render.d.ts","sourceRoot":"","sources":["../../src/sync/render.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AAEhD,MAAM,WAAW,aAAa;IAC5B,QAAQ,CAAC,MAAM,EAAE,IAAI,GAAG,IAAI,CAAC;CAC9B;AAoCD,wBAAgB,WAAW,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,EAAE,IAAI,EAAE,aAAa,GAAG,MAAM,CAmB/E;AAED,eAAO,MAAM,YAAY,8BAA8B,CAAC;AACxD,eAAO,MAAM,UAAU,4BAA4B,CAAC;AAEpD,wBAAgB,kBAAkB,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,CAUhF"}
@@ -0,0 +1,70 @@
1
+ function severityOrder(s) {
2
+ if (s === 'error')
3
+ return 0;
4
+ if (s === 'warning')
5
+ return 1;
6
+ return 2;
7
+ }
8
+ function sectionTitle(locale, s) {
9
+ if (locale === 'ko') {
10
+ if (s === 'error')
11
+ return '## 오류 (반드시 지킬 것)';
12
+ if (s === 'warning')
13
+ return '## 경고';
14
+ return '## 정보';
15
+ }
16
+ if (s === 'error')
17
+ return '## Errors (must not violate)';
18
+ if (s === 'warning')
19
+ return '## Warnings';
20
+ return '## Info';
21
+ }
22
+ function header(locale) {
23
+ if (locale === 'ko') {
24
+ return `# AI 코딩 가이드라인\n\n아래 규칙은 aicq가 커밋 단계에서 강제합니다. 자동 생성된 내용 — 수정은 \`aicq/rules/\` 에서.`;
25
+ }
26
+ return `# AI Coding Guidelines\n\nThe rules below are enforced by aicq at commit time. Generated automatically — edit \`aicq/rules/\` to change.`;
27
+ }
28
+ function getMessage(rule, locale) {
29
+ if (locale === 'ko' && rule.messageKo)
30
+ return rule.messageKo;
31
+ return rule.message;
32
+ }
33
+ function renderRule(rule, locale) {
34
+ const langs = Array.isArray(rule.language) ? rule.language : [rule.language];
35
+ return `### ${rule.id}\n${getMessage(rule, locale)}\n- Languages: ${langs.join(', ')}`;
36
+ }
37
+ export function renderRules(rules, opts) {
38
+ const sorted = [...rules].sort((a, b) => {
39
+ const sev = severityOrder(a.severity) - severityOrder(b.severity);
40
+ return sev !== 0 ? sev : a.id.localeCompare(b.id);
41
+ });
42
+ const lines = [header(opts.locale), ''];
43
+ let lastSeverity = null;
44
+ for (const rule of sorted) {
45
+ if (rule.severity !== lastSeverity) {
46
+ lines.push('');
47
+ lines.push(sectionTitle(opts.locale, rule.severity));
48
+ lines.push('');
49
+ lastSeverity = rule.severity;
50
+ }
51
+ lines.push(renderRule(rule, opts.locale));
52
+ lines.push('');
53
+ }
54
+ return lines.join('\n').trimEnd() + '\n';
55
+ }
56
+ export const MARKER_START = '<!-- aicq:rules:start -->';
57
+ export const MARKER_END = '<!-- aicq:rules:end -->';
58
+ export function injectIntoMarkdown(existing, body) {
59
+ const block = `${MARKER_START}\n${body}${MARKER_END}\n`;
60
+ if (existing === null || existing.length === 0)
61
+ return block;
62
+ const startIdx = existing.indexOf(MARKER_START);
63
+ const endIdx = existing.indexOf(MARKER_END);
64
+ if (startIdx === -1 || endIdx === -1 || endIdx < startIdx) {
65
+ const sep = existing.endsWith('\n') ? '\n' : '\n\n';
66
+ return `${existing}${sep}${block}`;
67
+ }
68
+ return `${existing.slice(0, startIdx)}${block}${existing.slice(endIdx + MARKER_END.length).replace(/^\n/, '')}`;
69
+ }
70
+ //# sourceMappingURL=render.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"render.js","sourceRoot":"","sources":["../../src/sync/render.ts"],"names":[],"mappings":"AAOA,SAAS,aAAa,CAAC,CAAW;IAChC,IAAI,CAAC,KAAK,OAAO;QAAE,OAAO,CAAC,CAAC;IAC5B,IAAI,CAAC,KAAK,SAAS;QAAE,OAAO,CAAC,CAAC;IAC9B,OAAO,CAAC,CAAC;AACX,CAAC;AAED,SAAS,YAAY,CAAC,MAAmB,EAAE,CAAW;IACpD,IAAI,MAAM,KAAK,IAAI,EAAE,CAAC;QACpB,IAAI,CAAC,KAAK,OAAO;YAAE,OAAO,kBAAkB,CAAC;QAC7C,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,OAAO,CAAC;QACpC,OAAO,OAAO,CAAC;IACjB,CAAC;IACD,IAAI,CAAC,KAAK,OAAO;QAAE,OAAO,8BAA8B,CAAC;IACzD,IAAI,CAAC,KAAK,SAAS;QAAE,OAAO,aAAa,CAAC;IAC1C,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,MAAM,CAAC,MAAmB;IACjC,IAAI,MAAM,KAAK,IAAI,EAAE,CAAC;QACpB,OAAO,kFAAkF,CAAC;IAC5F,CAAC;IACD,OAAO,0IAA0I,CAAC;AACpJ,CAAC;AAED,SAAS,UAAU,CAAC,IAAU,EAAE,MAAmB;IACjD,IAAI,MAAM,KAAK,IAAI,IAAI,IAAI,CAAC,SAAS;QAAE,OAAO,IAAI,CAAC,SAAS,CAAC;IAC7D,OAAO,IAAI,CAAC,OAAO,CAAC;AACtB,CAAC;AAED,SAAS,UAAU,CAAC,IAAU,EAAE,MAAmB;IACjD,MAAM,KAAK,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC7E,OAAO,OAAO,IAAI,CAAC,EAAE,KAAK,UAAU,CAAC,IAAI,EAAE,MAAM,CAAC,kBAAkB,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;AACzF,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,KAAsB,EAAE,IAAmB;IACrE,MAAM,MAAM,GAAG,CAAC,GAAG,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;QACtC,MAAM,GAAG,GAAG,aAAa,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,aAAa,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;QAClE,OAAO,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,MAAM,KAAK,GAAa,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;IAClD,IAAI,YAAY,GAAoB,IAAI,CAAC;IACzC,KAAK,MAAM,IAAI,IAAI,MAAM,EAAE,CAAC;QAC1B,IAAI,IAAI,CAAC,QAAQ,KAAK,YAAY,EAAE,CAAC;YACnC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;YACrD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC;QAC/B,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;QAC1C,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACjB,CAAC;IACD,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC;AAC3C,CAAC;AAED,MAAM,CAAC,MAAM,YAAY,GAAG,2BAA2B,CAAC;AACxD,MAAM,CAAC,MAAM,UAAU,GAAG,yBAAyB,CAAC;AAEpD,MAAM,UAAU,kBAAkB,CAAC,QAAuB,EAAE,IAAY;IACtE,MAAM,KAAK,GAAG,GAAG,YAAY,KAAK,IAAI,GAAG,UAAU,IAAI,CAAC;IACxD,IAAI,QAAQ,KAAK,IAAI,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC;IAC7D,MAAM,QAAQ,GAAG,QAAQ,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IAChD,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC5C,IAAI,QAAQ,KAAK,CAAC,CAAC,IAAI,MAAM,KAAK,CAAC,CAAC,IAAI,MAAM,GAAG,QAAQ,EAAE,CAAC;QAC1D,MAAM,GAAG,GAAG,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC;QACpD,OAAO,GAAG,QAAQ,GAAG,GAAG,GAAG,KAAK,EAAE,CAAC;IACrC,CAAC;IACD,OAAO,GAAG,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,QAAQ,CAAC,GAAG,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,EAAE,CAAC;AAClH,CAAC"}
@@ -0,0 +1,12 @@
1
+ import type { Rule } from '@aicqtools/rule-sdk';
2
+ export interface SyncTarget {
3
+ readonly path: string;
4
+ readonly markdown: boolean;
5
+ }
6
+ export interface SyncOptions {
7
+ readonly cwd: string;
8
+ readonly locale: 'ko' | 'en';
9
+ readonly targets?: readonly SyncTarget[];
10
+ }
11
+ export declare function syncAiRules(rules: readonly Rule[], opts: SyncOptions): Promise<string[]>;
12
+ //# sourceMappingURL=sync-files.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync-files.d.ts","sourceRoot":"","sources":["../../src/sync/sync-files.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AAGhD,MAAM,WAAW,UAAU;IACzB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC;CAC5B;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,CAAC,GAAG,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,MAAM,EAAE,IAAI,GAAG,IAAI,CAAC;IAC7B,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,UAAU,EAAE,CAAC;CAC1C;AAOD,wBAAsB,WAAW,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,EAAE,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CAmB9F"}
@@ -0,0 +1,29 @@
1
+ import { readFile, writeFile } from 'node:fs/promises';
2
+ import { existsSync } from 'node:fs';
3
+ import { resolve } from 'node:path';
4
+ import { injectIntoMarkdown, renderRules } from './render.js';
5
+ const DEFAULT_TARGETS = [
6
+ { path: '.cursorrules', markdown: false },
7
+ { path: 'CLAUDE.md', markdown: true },
8
+ ];
9
+ export async function syncAiRules(rules, opts) {
10
+ const cwd = resolve(opts.cwd);
11
+ const body = renderRules(rules, { locale: opts.locale });
12
+ const targets = opts.targets ?? DEFAULT_TARGETS;
13
+ const written = [];
14
+ for (const t of targets) {
15
+ const fullPath = resolve(cwd, t.path);
16
+ let next;
17
+ if (t.markdown) {
18
+ const existing = existsSync(fullPath) ? await readFile(fullPath, 'utf-8') : null;
19
+ next = injectIntoMarkdown(existing, body);
20
+ }
21
+ else {
22
+ next = body;
23
+ }
24
+ await writeFile(fullPath, next, 'utf-8');
25
+ written.push(fullPath);
26
+ }
27
+ return written;
28
+ }
29
+ //# sourceMappingURL=sync-files.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync-files.js","sourceRoot":"","sources":["../../src/sync/sync-files.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AACvD,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,OAAO,EAAE,kBAAkB,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAa9D,MAAM,eAAe,GAA0B;IAC7C,EAAE,IAAI,EAAE,cAAc,EAAE,QAAQ,EAAE,KAAK,EAAE;IACzC,EAAE,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,IAAI,EAAE;CACtC,CAAC;AAEF,MAAM,CAAC,KAAK,UAAU,WAAW,CAAC,KAAsB,EAAE,IAAiB;IACzE,MAAM,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC9B,MAAM,IAAI,GAAG,WAAW,CAAC,KAAK,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;IACzD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,eAAe,CAAC;IAChD,MAAM,OAAO,GAAa,EAAE,CAAC;IAE7B,KAAK,MAAM,CAAC,IAAI,OAAO,EAAE,CAAC;QACxB,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,IAAY,CAAC;QACjB,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC;YACf,MAAM,QAAQ,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,MAAM,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YACjF,IAAI,GAAG,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;QAC5C,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,IAAI,CAAC;QACd,CAAC;QACD,MAAM,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;QACzC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACzB,CAAC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC"}
package/package.json ADDED
@@ -0,0 +1,62 @@
1
+ {
2
+ "name": "@aicqtools/guardrail",
3
+ "version": "1.0.0-alpha.2",
4
+ "type": "module",
5
+ "description": "Deterministic AI code guardrail engine — 37 built-in rules, hybrid YAML/function DSL, MCP server, .cursorrules sync, sqlite cache.",
6
+ "keywords": [
7
+ "aicq",
8
+ "aicqtools",
9
+ "guardrail",
10
+ "ai-code-quality",
11
+ "tree-sitter",
12
+ "linter",
13
+ "mcp",
14
+ "code-review"
15
+ ],
16
+ "homepage": "https://github.com/aicqtools/aicqtools/tree/main/modules/guardrail",
17
+ "repository": {
18
+ "type": "git",
19
+ "url": "https://github.com/aicqtools/aicqtools.git",
20
+ "directory": "modules/guardrail"
21
+ },
22
+ "bugs": "https://github.com/aicqtools/aicqtools/issues",
23
+ "author": "Eom Sik <neuralflux3@gmail.com>",
24
+ "license": "MIT",
25
+ "main": "./dist/index.js",
26
+ "types": "./dist/index.d.ts",
27
+ "exports": {
28
+ ".": "./dist/index.js",
29
+ "./runner": "./dist/runner/index.js",
30
+ "./matcher": "./dist/matcher/index.js",
31
+ "./mcp": "./dist/mcp/index.js",
32
+ "./sync": "./dist/sync/index.js"
33
+ },
34
+ "files": [
35
+ "dist/",
36
+ "scripts/",
37
+ "README.md"
38
+ ],
39
+ "publishConfig": {
40
+ "access": "public"
41
+ },
42
+ "dependencies": {
43
+ "@modelcontextprotocol/sdk": "^1.29.0",
44
+ "fast-glob": "^3.3.2",
45
+ "tree-sitter": "~0.21.1",
46
+ "yaml": "^2.6.1",
47
+ "zod": "^3.23.8",
48
+ "@aicqtools/core": "1.0.0-alpha.2",
49
+ "@aicqtools/rule-sdk": "1.0.0-alpha.2"
50
+ },
51
+ "devDependencies": {
52
+ "@types/node": "^22.10.0",
53
+ "typescript": "^5.7.2",
54
+ "vitest": "^2.1.8"
55
+ },
56
+ "scripts": {
57
+ "build": "tsc && node scripts/copy-yaml-rules.mjs",
58
+ "test": "vitest run --passWithNoTests",
59
+ "typecheck": "tsc --noEmit",
60
+ "clean": "rm -rf dist .turbo"
61
+ }
62
+ }
@@ -0,0 +1,19 @@
1
+ import { readdir, mkdir, copyFile } from 'node:fs/promises';
2
+ import { dirname, extname, join, resolve } from 'node:path';
3
+ import { fileURLToPath } from 'node:url';
4
+
5
+ const here = dirname(fileURLToPath(import.meta.url));
6
+ const src = resolve(here, '../src/rules-default');
7
+ const dest = resolve(here, '../dist/rules-default');
8
+
9
+ await mkdir(dest, { recursive: true });
10
+ const entries = await readdir(src);
11
+ let copied = 0;
12
+ for (const entry of entries) {
13
+ const ext = extname(entry);
14
+ if (ext === '.yaml' || ext === '.yml') {
15
+ await copyFile(join(src, entry), join(dest, entry));
16
+ copied += 1;
17
+ }
18
+ }
19
+ process.stdout.write(`copy-yaml-rules: ${copied} file(s)\n`);