@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.
- package/LICENSE +21 -0
- package/dist/docs/build.d.ts +12 -0
- package/dist/docs/build.d.ts.map +1 -0
- package/dist/docs/build.js +23 -0
- package/dist/docs/build.js.map +1 -0
- package/dist/docs/index.d.ts +4 -0
- package/dist/docs/index.d.ts.map +1 -0
- package/dist/docs/index.js +3 -0
- package/dist/docs/index.js.map +1 -0
- package/dist/docs/render-rule-md.d.ts +4 -0
- package/dist/docs/render-rule-md.d.ts.map +1 -0
- package/dist/docs/render-rule-md.js +117 -0
- package/dist/docs/render-rule-md.js.map +1 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +7 -0
- package/dist/index.js.map +1 -0
- package/dist/matcher/index.d.ts +4 -0
- package/dist/matcher/index.d.ts.map +1 -0
- package/dist/matcher/index.js +3 -0
- package/dist/matcher/index.js.map +1 -0
- package/dist/matcher/traverse.d.ts +3 -0
- package/dist/matcher/traverse.d.ts.map +1 -0
- package/dist/matcher/traverse.js +9 -0
- package/dist/matcher/traverse.js.map +1 -0
- package/dist/matcher/yaml-rule.d.ts +31 -0
- package/dist/matcher/yaml-rule.d.ts.map +1 -0
- package/dist/matcher/yaml-rule.js +30 -0
- package/dist/matcher/yaml-rule.js.map +1 -0
- package/dist/mcp/handlers.d.ts +36 -0
- package/dist/mcp/handlers.d.ts.map +1 -0
- package/dist/mcp/handlers.js +32 -0
- package/dist/mcp/handlers.js.map +1 -0
- package/dist/mcp/index.d.ts +6 -0
- package/dist/mcp/index.d.ts.map +1 -0
- package/dist/mcp/index.js +4 -0
- package/dist/mcp/index.js.map +1 -0
- package/dist/mcp/server.d.ts +9 -0
- package/dist/mcp/server.d.ts.map +1 -0
- package/dist/mcp/server.js +53 -0
- package/dist/mcp/server.js.map +1 -0
- package/dist/mcp/stdio.d.ts +3 -0
- package/dist/mcp/stdio.d.ts.map +1 -0
- package/dist/mcp/stdio.js +8 -0
- package/dist/mcp/stdio.js.map +1 -0
- package/dist/rules-default/ai-explainability-metadata.d.ts +3 -0
- package/dist/rules-default/ai-explainability-metadata.d.ts.map +1 -0
- package/dist/rules-default/ai-explainability-metadata.js +41 -0
- package/dist/rules-default/ai-explainability-metadata.js.map +1 -0
- package/dist/rules-default/api-response-shape.d.ts +3 -0
- package/dist/rules-default/api-response-shape.d.ts.map +1 -0
- package/dist/rules-default/api-response-shape.js +52 -0
- package/dist/rules-default/api-response-shape.js.map +1 -0
- package/dist/rules-default/async-await-consistency.d.ts +3 -0
- package/dist/rules-default/async-await-consistency.d.ts.map +1 -0
- package/dist/rules-default/async-await-consistency.js +38 -0
- package/dist/rules-default/async-await-consistency.js.map +1 -0
- package/dist/rules-default/audit-log-ai-decision.d.ts +3 -0
- package/dist/rules-default/audit-log-ai-decision.d.ts.map +1 -0
- package/dist/rules-default/audit-log-ai-decision.js +33 -0
- package/dist/rules-default/audit-log-ai-decision.js.map +1 -0
- package/dist/rules-default/camelcase-migration-column.d.ts +3 -0
- package/dist/rules-default/camelcase-migration-column.d.ts.map +1 -0
- package/dist/rules-default/camelcase-migration-column.js +56 -0
- package/dist/rules-default/camelcase-migration-column.js.map +1 -0
- package/dist/rules-default/controller-needs-async-wrapper.d.ts +3 -0
- package/dist/rules-default/controller-needs-async-wrapper.d.ts.map +1 -0
- package/dist/rules-default/controller-needs-async-wrapper.js +56 -0
- package/dist/rules-default/controller-needs-async-wrapper.js.map +1 -0
- package/dist/rules-default/enforce-utf8-encoding.d.ts +10 -0
- package/dist/rules-default/enforce-utf8-encoding.d.ts.map +1 -0
- package/dist/rules-default/enforce-utf8-encoding.js +28 -0
- package/dist/rules-default/enforce-utf8-encoding.js.map +1 -0
- package/dist/rules-default/explicit-kst-timezone.d.ts +3 -0
- package/dist/rules-default/explicit-kst-timezone.d.ts.map +1 -0
- package/dist/rules-default/explicit-kst-timezone.js +49 -0
- package/dist/rules-default/explicit-kst-timezone.js.map +1 -0
- package/dist/rules-default/fk-needs-on-delete.d.ts +3 -0
- package/dist/rules-default/fk-needs-on-delete.d.ts.map +1 -0
- package/dist/rules-default/fk-needs-on-delete.js +54 -0
- package/dist/rules-default/fk-needs-on-delete.js.map +1 -0
- package/dist/rules-default/human-oversight-checkpoint.d.ts +3 -0
- package/dist/rules-default/human-oversight-checkpoint.d.ts.map +1 -0
- package/dist/rules-default/human-oversight-checkpoint.js +37 -0
- package/dist/rules-default/human-oversight-checkpoint.js.map +1 -0
- package/dist/rules-default/index.d.ts +6 -0
- package/dist/rules-default/index.d.ts.map +1 -0
- package/dist/rules-default/index.js +138 -0
- package/dist/rules-default/index.js.map +1 -0
- package/dist/rules-default/korean-comment-encoding.d.ts +3 -0
- package/dist/rules-default/korean-comment-encoding.d.ts.map +1 -0
- package/dist/rules-default/korean-comment-encoding.js +28 -0
- package/dist/rules-default/korean-comment-encoding.js.map +1 -0
- package/dist/rules-default/mask-card-number.d.ts +3 -0
- package/dist/rules-default/mask-card-number.d.ts.map +1 -0
- package/dist/rules-default/mask-card-number.js +45 -0
- package/dist/rules-default/mask-card-number.js.map +1 -0
- package/dist/rules-default/mask-pii-in-ai-prompt.d.ts +3 -0
- package/dist/rules-default/mask-pii-in-ai-prompt.d.ts.map +1 -0
- package/dist/rules-default/mask-pii-in-ai-prompt.js +41 -0
- package/dist/rules-default/mask-pii-in-ai-prompt.js.map +1 -0
- package/dist/rules-default/naver-kakao-oauth-webview.d.ts +3 -0
- package/dist/rules-default/naver-kakao-oauth-webview.d.ts.map +1 -0
- package/dist/rules-default/naver-kakao-oauth-webview.js +40 -0
- package/dist/rules-default/naver-kakao-oauth-webview.js.map +1 -0
- package/dist/rules-default/no-bare-except.d.ts +7 -0
- package/dist/rules-default/no-bare-except.d.ts.map +1 -0
- package/dist/rules-default/no-bare-except.js +23 -0
- package/dist/rules-default/no-bare-except.js.map +1 -0
- package/dist/rules-default/no-bare-throw.d.ts +7 -0
- package/dist/rules-default/no-bare-throw.d.ts.map +1 -0
- package/dist/rules-default/no-bare-throw.js +32 -0
- package/dist/rules-default/no-bare-throw.js.map +1 -0
- package/dist/rules-default/no-boolean-trap.d.ts +8 -0
- package/dist/rules-default/no-boolean-trap.d.ts.map +1 -0
- package/dist/rules-default/no-boolean-trap.js +33 -0
- package/dist/rules-default/no-boolean-trap.js.map +1 -0
- package/dist/rules-default/no-console-log.d.ts +3 -0
- package/dist/rules-default/no-console-log.d.ts.map +1 -0
- package/dist/rules-default/no-console-log.js +19 -0
- package/dist/rules-default/no-console-log.js.map +1 -0
- package/dist/rules-default/no-cvv-logging.d.ts +3 -0
- package/dist/rules-default/no-cvv-logging.d.ts.map +1 -0
- package/dist/rules-default/no-cvv-logging.js +36 -0
- package/dist/rules-default/no-cvv-logging.js.map +1 -0
- package/dist/rules-default/no-default-export-from-libs.d.ts +7 -0
- package/dist/rules-default/no-default-export-from-libs.d.ts.map +1 -0
- package/dist/rules-default/no-default-export-from-libs.js +24 -0
- package/dist/rules-default/no-default-export-from-libs.js.map +1 -0
- package/dist/rules-default/no-direct-anthropic.yaml +9 -0
- package/dist/rules-default/no-direct-openai.yaml +9 -0
- package/dist/rules-default/no-empty-catch.d.ts +7 -0
- package/dist/rules-default/no-empty-catch.d.ts.map +1 -0
- package/dist/rules-default/no-empty-catch.js +30 -0
- package/dist/rules-default/no-empty-catch.js.map +1 -0
- package/dist/rules-default/no-fstring-sql.d.ts +3 -0
- package/dist/rules-default/no-fstring-sql.d.ts.map +1 -0
- package/dist/rules-default/no-fstring-sql.js +24 -0
- package/dist/rules-default/no-fstring-sql.js.map +1 -0
- package/dist/rules-default/no-id-overwrite.d.ts +7 -0
- package/dist/rules-default/no-id-overwrite.d.ts.map +1 -0
- package/dist/rules-default/no-id-overwrite.js +26 -0
- package/dist/rules-default/no-id-overwrite.js.map +1 -0
- package/dist/rules-default/no-inline-date.yaml +9 -0
- package/dist/rules-default/no-inline-math-round.yaml +12 -0
- package/dist/rules-default/no-jsonb-circular.d.ts +3 -0
- package/dist/rules-default/no-jsonb-circular.d.ts.map +1 -0
- package/dist/rules-default/no-jsonb-circular.js +33 -0
- package/dist/rules-default/no-jsonb-circular.js.map +1 -0
- package/dist/rules-default/no-magic-number.d.ts +3 -0
- package/dist/rules-default/no-magic-number.d.ts.map +1 -0
- package/dist/rules-default/no-magic-number.js +42 -0
- package/dist/rules-default/no-magic-number.js.map +1 -0
- package/dist/rules-default/no-mutable-default-arg.d.ts +3 -0
- package/dist/rules-default/no-mutable-default-arg.d.ts.map +1 -0
- package/dist/rules-default/no-mutable-default-arg.js +30 -0
- package/dist/rules-default/no-mutable-default-arg.js.map +1 -0
- package/dist/rules-default/no-pickle.yaml +12 -0
- package/dist/rules-default/no-plain-card-number.d.ts +3 -0
- package/dist/rules-default/no-plain-card-number.d.ts.map +1 -0
- package/dist/rules-default/no-plain-card-number.js +51 -0
- package/dist/rules-default/no-plain-card-number.js.map +1 -0
- package/dist/rules-default/no-print-in-prod.yaml +9 -0
- package/dist/rules-default/no-process-env-leak.d.ts +7 -0
- package/dist/rules-default/no-process-env-leak.d.ts.map +1 -0
- package/dist/rules-default/no-process-env-leak.js +34 -0
- package/dist/rules-default/no-process-env-leak.js.map +1 -0
- package/dist/rules-default/no-shell-true.d.ts +3 -0
- package/dist/rules-default/no-shell-true.d.ts.map +1 -0
- package/dist/rules-default/no-shell-true.js +42 -0
- package/dist/rules-default/no-shell-true.js.map +1 -0
- package/dist/rules-default/prefer-const-array.d.ts +9 -0
- package/dist/rules-default/prefer-const-array.d.ts.map +1 -0
- package/dist/rules-default/prefer-const-array.js +34 -0
- package/dist/rules-default/prefer-const-array.js.map +1 -0
- package/dist/rules-default/prefer-named-imports.d.ts +8 -0
- package/dist/rules-default/prefer-named-imports.d.ts.map +1 -0
- package/dist/rules-default/prefer-named-imports.js +25 -0
- package/dist/rules-default/prefer-named-imports.js.map +1 -0
- package/dist/rules-default/preserve-transaction-log.d.ts +3 -0
- package/dist/rules-default/preserve-transaction-log.d.ts.map +1 -0
- package/dist/rules-default/preserve-transaction-log.js +33 -0
- package/dist/rules-default/preserve-transaction-log.js.map +1 -0
- package/dist/rules-default/pytest-fixture-naming.d.ts +3 -0
- package/dist/rules-default/pytest-fixture-naming.d.ts.map +1 -0
- package/dist/rules-default/pytest-fixture-naming.js +36 -0
- package/dist/rules-default/pytest-fixture-naming.js.map +1 -0
- package/dist/rules-default/requests-needs-timeout.d.ts +3 -0
- package/dist/rules-default/requests-needs-timeout.d.ts.map +1 -0
- package/dist/rules-default/requests-needs-timeout.js +43 -0
- package/dist/rules-default/requests-needs-timeout.js.map +1 -0
- package/dist/rules-default/require-idempotency-key.d.ts +3 -0
- package/dist/rules-default/require-idempotency-key.d.ts.map +1 -0
- package/dist/rules-default/require-idempotency-key.js +47 -0
- package/dist/rules-default/require-idempotency-key.js.map +1 -0
- package/dist/rules-default/require-tls-1-2-plus.d.ts +3 -0
- package/dist/rules-default/require-tls-1-2-plus.d.ts.map +1 -0
- package/dist/rules-default/require-tls-1-2-plus.js +30 -0
- package/dist/rules-default/require-tls-1-2-plus.js.map +1 -0
- package/dist/rules-default/rfc5987-korean-filename.d.ts +3 -0
- package/dist/rules-default/rfc5987-korean-filename.d.ts.map +1 -0
- package/dist/rules-default/rfc5987-korean-filename.js +32 -0
- package/dist/rules-default/rfc5987-korean-filename.js.map +1 -0
- package/dist/rules-default/route-needs-auth.d.ts +3 -0
- package/dist/rules-default/route-needs-auth.d.ts.map +1 -0
- package/dist/rules-default/route-needs-auth.js +48 -0
- package/dist/rules-default/route-needs-auth.js.map +1 -0
- package/dist/rules-default/route-needs-rate-limit.d.ts +3 -0
- package/dist/rules-default/route-needs-rate-limit.d.ts.map +1 -0
- package/dist/rules-default/route-needs-rate-limit.js +47 -0
- package/dist/rules-default/route-needs-rate-limit.js.map +1 -0
- package/dist/rules-default/separate-refund-permission.d.ts +3 -0
- package/dist/rules-default/separate-refund-permission.d.ts.map +1 -0
- package/dist/rules-default/separate-refund-permission.js +33 -0
- package/dist/rules-default/separate-refund-permission.js.map +1 -0
- package/dist/rules-default/track-ai-model-version.d.ts +3 -0
- package/dist/rules-default/track-ai-model-version.d.ts.map +1 -0
- package/dist/rules-default/track-ai-model-version.js +37 -0
- package/dist/rules-default/track-ai-model-version.js.map +1 -0
- package/dist/rules-default/type-hint-required-public.d.ts +10 -0
- package/dist/rules-default/type-hint-required-public.d.ts.map +1 -0
- package/dist/rules-default/type-hint-required-public.js +29 -0
- package/dist/rules-default/type-hint-required-public.js.map +1 -0
- package/dist/rules-default/verify-pg-response.d.ts +3 -0
- package/dist/rules-default/verify-pg-response.d.ts.map +1 -0
- package/dist/rules-default/verify-pg-response.js +37 -0
- package/dist/rules-default/verify-pg-response.js.map +1 -0
- package/dist/rules-default/won-format-thousands.d.ts +3 -0
- package/dist/rules-default/won-format-thousands.d.ts.map +1 -0
- package/dist/rules-default/won-format-thousands.js +24 -0
- package/dist/rules-default/won-format-thousands.js.map +1 -0
- package/dist/runner/context.d.ts +12 -0
- package/dist/runner/context.d.ts.map +1 -0
- package/dist/runner/context.js +34 -0
- package/dist/runner/context.js.map +1 -0
- package/dist/runner/index.d.ts +7 -0
- package/dist/runner/index.d.ts.map +1 -0
- package/dist/runner/index.js +5 -0
- package/dist/runner/index.js.map +1 -0
- package/dist/runner/ruleset-signature.d.ts +4 -0
- package/dist/runner/ruleset-signature.d.ts.map +1 -0
- package/dist/runner/ruleset-signature.js +11 -0
- package/dist/runner/ruleset-signature.js.map +1 -0
- package/dist/runner/run-file.d.ts +10 -0
- package/dist/runner/run-file.d.ts.map +1 -0
- package/dist/runner/run-file.js +20 -0
- package/dist/runner/run-file.js.map +1 -0
- package/dist/runner/run-project.d.ts +12 -0
- package/dist/runner/run-project.d.ts.map +1 -0
- package/dist/runner/run-project.js +55 -0
- package/dist/runner/run-project.js.map +1 -0
- package/dist/runner/run-rule.d.ts +5 -0
- package/dist/runner/run-rule.d.ts.map +1 -0
- package/dist/runner/run-rule.js +37 -0
- package/dist/runner/run-rule.js.map +1 -0
- package/dist/sync/index.d.ts +5 -0
- package/dist/sync/index.d.ts.map +1 -0
- package/dist/sync/index.js +3 -0
- package/dist/sync/index.js.map +1 -0
- package/dist/sync/render.d.ts +9 -0
- package/dist/sync/render.d.ts.map +1 -0
- package/dist/sync/render.js +70 -0
- package/dist/sync/render.js.map +1 -0
- package/dist/sync/sync-files.d.ts +12 -0
- package/dist/sync/sync-files.d.ts.map +1 -0
- package/dist/sync/sync-files.js +29 -0
- package/dist/sync/sync-files.js.map +1 -0
- package/package.json +62 -0
- package/scripts/copy-yaml-rules.mjs +19 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ai-explainability-metadata.js","sourceRoot":"","sources":["../../src/rules-default/ai-explainability-metadata.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAGjD;;;;;;;;;GASG;AACH,MAAM,MAAM,GAAG,qEAAqE,CAAC;AACrF,MAAM,YAAY,GAAG,mEAAmE,CAAC;AAEzF,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,4BAA4B;IAChC,QAAQ,EAAE,CAAC,YAAY,EAAE,KAAK,CAAC;IAC/B,QAAQ,EAAE,MAAM;IAChB,OAAO,EAAE,uHAAuH;IAChI,SAAS,EAAE,wEAAwE;IACnF,IAAI,EAAE,2FAA2F;IACjG,QAAQ,EAAE;QACR,eAAe,CAAC,IAAuB,EAAE,GAAgB;YACvD,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM;gBAAE,OAAO;YACpB,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAClC,IAAI,CAAC,6BAA6B,CAAC,IAAI,CAAC,MAAM,CAAC;gBAAE,OAAO;YACxD,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YACjD,IAAI,CAAC,IAAI;gBAAE,OAAO;YAClB,MAAM,QAAQ,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAClC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC;gBAAE,OAAO;YACnC,IAAI,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC;gBAAE,OAAO;YACxC,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QACvB,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api-response-shape.d.ts","sourceRoot":"","sources":["../../src/rules-default/api-response-shape.ts"],"names":[],"mappings":";AA0BA,wBAmBG"}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
function isResJsonCall(callee, textOf) {
|
|
3
|
+
if (callee.type !== 'member_expression')
|
|
4
|
+
return false;
|
|
5
|
+
const obj = callee.childForFieldName('object');
|
|
6
|
+
const prop = callee.childForFieldName('property');
|
|
7
|
+
if (!obj || !prop)
|
|
8
|
+
return false;
|
|
9
|
+
return textOf(obj) === 'res' && textOf(prop) === 'json';
|
|
10
|
+
}
|
|
11
|
+
function objectHasKey(obj, key, textOf) {
|
|
12
|
+
for (let i = 0; i < obj.namedChildCount; i++) {
|
|
13
|
+
const pair = obj.namedChild(i);
|
|
14
|
+
if (!pair)
|
|
15
|
+
continue;
|
|
16
|
+
if (pair.type === 'shorthand_property_identifier') {
|
|
17
|
+
if (textOf(pair) === key)
|
|
18
|
+
return true;
|
|
19
|
+
continue;
|
|
20
|
+
}
|
|
21
|
+
if (pair.type !== 'pair')
|
|
22
|
+
continue;
|
|
23
|
+
const k = pair.childForFieldName('key');
|
|
24
|
+
if (k && textOf(k).replace(/['"]/g, '') === key)
|
|
25
|
+
return true;
|
|
26
|
+
}
|
|
27
|
+
return false;
|
|
28
|
+
}
|
|
29
|
+
export default defineRule({
|
|
30
|
+
id: 'api-response-shape',
|
|
31
|
+
language: ['typescript', 'javascript', 'tsx'],
|
|
32
|
+
severity: 'warning',
|
|
33
|
+
message: 'API responses must use the standard shape: { success, data, message }.',
|
|
34
|
+
messageKo: 'API 응답은 { success, data, message } 표준 형태여야 합니다.',
|
|
35
|
+
visitors: {
|
|
36
|
+
call_expression(node, ctx) {
|
|
37
|
+
const callee = node.childForFieldName('function');
|
|
38
|
+
if (!callee || !isResJsonCall(callee, ctx.textOf))
|
|
39
|
+
return;
|
|
40
|
+
const args = node.childForFieldName('arguments');
|
|
41
|
+
if (!args || args.namedChildCount === 0)
|
|
42
|
+
return;
|
|
43
|
+
const first = args.namedChild(0);
|
|
44
|
+
if (!first || first.type !== 'object')
|
|
45
|
+
return;
|
|
46
|
+
if (!objectHasKey(first, 'success', ctx.textOf)) {
|
|
47
|
+
ctx.report({ node: first });
|
|
48
|
+
}
|
|
49
|
+
},
|
|
50
|
+
},
|
|
51
|
+
});
|
|
52
|
+
//# sourceMappingURL=api-response-shape.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api-response-shape.js","sourceRoot":"","sources":["../../src/rules-default/api-response-shape.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,SAAS,aAAa,CAAC,MAAyB,EAAE,MAAwC;IACxF,IAAI,MAAM,CAAC,IAAI,KAAK,mBAAmB;QAAE,OAAO,KAAK,CAAC;IACtD,MAAM,GAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;IAC/C,MAAM,IAAI,GAAG,MAAM,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;IAClD,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI;QAAE,OAAO,KAAK,CAAC;IAChC,OAAO,MAAM,CAAC,GAAG,CAAC,KAAK,KAAK,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,MAAM,CAAC;AAC1D,CAAC;AAED,SAAS,YAAY,CAAC,GAAsB,EAAE,GAAW,EAAE,MAAwC;IACjG,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;QAC7C,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,CAAC,IAAI;YAAE,SAAS;QACpB,IAAI,IAAI,CAAC,IAAI,KAAK,+BAA+B,EAAE,CAAC;YAClD,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG;gBAAE,OAAO,IAAI,CAAC;YACtC,SAAS;QACX,CAAC;QACD,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM;YAAE,SAAS;QACnC,MAAM,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QACxC,IAAI,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,KAAK,GAAG;YAAE,OAAO,IAAI,CAAC;IAC/D,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,oBAAoB;IACxB,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,KAAK,CAAC;IAC7C,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,wEAAwE;IACjF,SAAS,EAAE,iDAAiD;IAC5D,QAAQ,EAAE;QACR,eAAe,CAAC,IAAI,EAAE,GAAG;YACvB,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC;gBAAE,OAAO;YAC1D,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YACjD,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,eAAe,KAAK,CAAC;gBAAE,OAAO;YAChD,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YACjC,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ;gBAAE,OAAO;YAC9C,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC;gBAChD,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;YAC9B,CAAC;QACH,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"async-await-consistency.d.ts","sourceRoot":"","sources":["../../src/rules-default/async-await-consistency.ts"],"names":[],"mappings":";AAmBA,wBAeG"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
function bodyHasAwait(body) {
|
|
3
|
+
function walk(n) {
|
|
4
|
+
if (n.type === 'await')
|
|
5
|
+
return true;
|
|
6
|
+
// Don't descend into nested function/class definitions
|
|
7
|
+
if (n.type === 'function_definition' || n.type === 'lambda' || n.type === 'class_definition') {
|
|
8
|
+
return false;
|
|
9
|
+
}
|
|
10
|
+
for (let i = 0; i < n.namedChildCount; i++) {
|
|
11
|
+
const child = n.namedChild(i);
|
|
12
|
+
if (child && walk(child))
|
|
13
|
+
return true;
|
|
14
|
+
}
|
|
15
|
+
return false;
|
|
16
|
+
}
|
|
17
|
+
return walk(body);
|
|
18
|
+
}
|
|
19
|
+
export default defineRule({
|
|
20
|
+
id: 'async-await-consistency',
|
|
21
|
+
language: 'python',
|
|
22
|
+
severity: 'warning',
|
|
23
|
+
message: '`async def` function has no `await` — likely should be a regular `def`.',
|
|
24
|
+
messageKo: '`async def` 함수에 `await`가 없습니다 — 일반 `def`로 정의하세요.',
|
|
25
|
+
visitors: {
|
|
26
|
+
function_definition(node, ctx) {
|
|
27
|
+
const text = ctx.textOf(node);
|
|
28
|
+
if (!text.startsWith('async '))
|
|
29
|
+
return;
|
|
30
|
+
const body = node.childForFieldName('body');
|
|
31
|
+
if (!body)
|
|
32
|
+
return;
|
|
33
|
+
if (!bodyHasAwait(body))
|
|
34
|
+
ctx.report({ node });
|
|
35
|
+
},
|
|
36
|
+
},
|
|
37
|
+
});
|
|
38
|
+
//# sourceMappingURL=async-await-consistency.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"async-await-consistency.js","sourceRoot":"","sources":["../../src/rules-default/async-await-consistency.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,SAAS,YAAY,CAAC,IAAuB;IAC3C,SAAS,IAAI,CAAC,CAAoB;QAChC,IAAI,CAAC,CAAC,IAAI,KAAK,OAAO;YAAE,OAAO,IAAI,CAAC;QACpC,uDAAuD;QACvD,IAAI,CAAC,CAAC,IAAI,KAAK,qBAAqB,IAAI,CAAC,CAAC,IAAI,KAAK,QAAQ,IAAI,CAAC,CAAC,IAAI,KAAK,kBAAkB,EAAE,CAAC;YAC7F,OAAO,KAAK,CAAC;QACf,CAAC;QACD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;YAC3C,MAAM,KAAK,GAAG,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;YAC9B,IAAI,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC;gBAAE,OAAO,IAAI,CAAC;QACxC,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IACD,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC;AACpB,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,yBAAyB;IAC7B,QAAQ,EAAE,QAAQ;IAClB,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,yEAAyE;IAClF,SAAS,EAAE,kDAAkD;IAC7D,QAAQ,EAAE;QACR,mBAAmB,CAAC,IAAI,EAAE,GAAG;YAC3B,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAC9B,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC;gBAAE,OAAO;YACvC,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;YAC5C,IAAI,CAAC,IAAI;gBAAE,OAAO;YAClB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QAChD,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-log-ai-decision.d.ts","sourceRoot":"","sources":["../../src/rules-default/audit-log-ai-decision.ts"],"names":[],"mappings":";AAsBA,wBAYG"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
/**
|
|
3
|
+
* FSC AI guideline: every AI inference decision must produce an audit log entry.
|
|
4
|
+
* Heuristic: if a function body invokes an AI SDK (openai/anthropic/aiClient) but
|
|
5
|
+
* has no logger / audit / console.log call, flag it.
|
|
6
|
+
*
|
|
7
|
+
* Limitation: false positives when logging is delegated to a wrapper that doesn't
|
|
8
|
+
* match the regex. Flip severity to warning if your codebase wraps logging extensively.
|
|
9
|
+
*/
|
|
10
|
+
const AI_CALL_PATTERN = /\b(openai|anthropic|aiClient)\.\w+/;
|
|
11
|
+
const LOG_PATTERN = /\b(logger|auditLog|audit|trace|winston|pino)\.\w+|\bconsole\.(log|info|warn|error)/;
|
|
12
|
+
function check(node, ctx) {
|
|
13
|
+
const text = ctx.textOf(node);
|
|
14
|
+
if (!AI_CALL_PATTERN.test(text))
|
|
15
|
+
return;
|
|
16
|
+
if (LOG_PATTERN.test(text))
|
|
17
|
+
return;
|
|
18
|
+
ctx.report({ node });
|
|
19
|
+
}
|
|
20
|
+
export default defineRule({
|
|
21
|
+
id: 'audit-log-ai-decision',
|
|
22
|
+
language: ['typescript', 'tsx'],
|
|
23
|
+
severity: 'error',
|
|
24
|
+
message: 'AI inference call must be paired with an audit log entry (FSC AI guideline — auditability).',
|
|
25
|
+
messageKo: 'AI 추론 호출은 감사 로그가 필수입니다 (금감원 AI 가이드라인 — 감사 추적성).',
|
|
26
|
+
docs: 'https://github.com/aicqtools/aicqtools/blob/main/docs/rules/audit-log-ai-decision.md',
|
|
27
|
+
visitors: {
|
|
28
|
+
function_declaration: check,
|
|
29
|
+
arrow_function: check,
|
|
30
|
+
method_definition: check,
|
|
31
|
+
},
|
|
32
|
+
});
|
|
33
|
+
//# sourceMappingURL=audit-log-ai-decision.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-log-ai-decision.js","sourceRoot":"","sources":["../../src/rules-default/audit-log-ai-decision.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAGjD;;;;;;;GAOG;AACH,MAAM,eAAe,GAAG,oCAAoC,CAAC;AAC7D,MAAM,WAAW,GAAG,oFAAoF,CAAC;AAEzG,SAAS,KAAK,CAAC,IAAuB,EAAE,GAAgB;IACtD,MAAM,IAAI,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IAC9B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IACxC,IAAI,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IACnC,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;AACvB,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,uBAAuB;IAC3B,QAAQ,EAAE,CAAC,YAAY,EAAE,KAAK,CAAC;IAC/B,QAAQ,EAAE,OAAO;IACjB,OAAO,EAAE,6FAA6F;IACtG,SAAS,EAAE,iDAAiD;IAC5D,IAAI,EAAE,sFAAsF;IAC5F,QAAQ,EAAE;QACR,oBAAoB,EAAE,KAAK;QAC3B,cAAc,EAAE,KAAK;QACrB,iBAAiB,EAAE,KAAK;KACzB;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"camelcase-migration-column.d.ts","sourceRoot":"","sources":["../../src/rules-default/camelcase-migration-column.ts"],"names":[],"mappings":";AAiBA,wBAgCG"}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
const SEQUELIZE_MIGRATION_FNS = new Set(['createTable', 'addColumn', 'changeColumn']);
|
|
3
|
+
const SNAKE_CASE_PATTERN = /^[a-z][a-z0-9]*(_[a-z0-9]+)+$/;
|
|
4
|
+
function findColumnDefObject(callNode) {
|
|
5
|
+
const args = callNode.childForFieldName('arguments');
|
|
6
|
+
if (!args)
|
|
7
|
+
return null;
|
|
8
|
+
// Sequelize patterns: createTable('users', { columnDef }), addColumn('users', 'colName', {def}), etc.
|
|
9
|
+
for (let i = 0; i < args.namedChildCount; i++) {
|
|
10
|
+
const arg = args.namedChild(i);
|
|
11
|
+
if (arg && arg.type === 'object')
|
|
12
|
+
return arg;
|
|
13
|
+
}
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
export default defineRule({
|
|
17
|
+
id: 'camelcase-migration-column',
|
|
18
|
+
language: ['typescript', 'javascript'],
|
|
19
|
+
severity: 'warning',
|
|
20
|
+
message: 'Sequelize migration column should use camelCase to match model attribute (avoid `underscored: true` divergence).',
|
|
21
|
+
messageKo: 'Sequelize 마이그레이션 컬럼명은 모델과 일치하도록 camelCase 사용 (snake_case는 underscored 옵션 충돌 위험).',
|
|
22
|
+
docs: 'https://github.com/aicqtools/aicqtools/blob/main/docs/rules/camelcase-migration-column.md',
|
|
23
|
+
visitors: {
|
|
24
|
+
call_expression(node, ctx) {
|
|
25
|
+
const fn = node.childForFieldName('function');
|
|
26
|
+
if (!fn || fn.type !== 'member_expression')
|
|
27
|
+
return;
|
|
28
|
+
const prop = fn.childForFieldName('property');
|
|
29
|
+
const obj = fn.childForFieldName('object');
|
|
30
|
+
if (!prop || !obj)
|
|
31
|
+
return;
|
|
32
|
+
// Only inside `queryInterface.createTable(...)` etc.
|
|
33
|
+
if (ctx.textOf(obj) !== 'queryInterface')
|
|
34
|
+
return;
|
|
35
|
+
if (!SEQUELIZE_MIGRATION_FNS.has(ctx.textOf(prop)))
|
|
36
|
+
return;
|
|
37
|
+
const colObj = findColumnDefObject(node);
|
|
38
|
+
if (!colObj)
|
|
39
|
+
return;
|
|
40
|
+
// Check each top-level key for snake_case
|
|
41
|
+
for (let i = 0; i < colObj.namedChildCount; i++) {
|
|
42
|
+
const pair = colObj.namedChild(i);
|
|
43
|
+
if (!pair || pair.type !== 'pair')
|
|
44
|
+
continue;
|
|
45
|
+
const key = pair.childForFieldName('key');
|
|
46
|
+
if (!key)
|
|
47
|
+
continue;
|
|
48
|
+
const keyText = ctx.textOf(key).replace(/['"]/g, '');
|
|
49
|
+
if (SNAKE_CASE_PATTERN.test(keyText)) {
|
|
50
|
+
ctx.report({ node: key });
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
},
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
//# sourceMappingURL=camelcase-migration-column.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"camelcase-migration-column.js","sourceRoot":"","sources":["../../src/rules-default/camelcase-migration-column.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,MAAM,uBAAuB,GAAG,IAAI,GAAG,CAAC,CAAC,aAAa,EAAE,WAAW,EAAE,cAAc,CAAC,CAAC,CAAC;AACtF,MAAM,kBAAkB,GAAG,+BAA+B,CAAC;AAE3D,SAAS,mBAAmB,CAAC,QAA2B;IACtD,MAAM,IAAI,GAAG,QAAQ,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;IACrD,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IACvB,sGAAsG;IACtG,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9C,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,GAAG,IAAI,GAAG,CAAC,IAAI,KAAK,QAAQ;YAAE,OAAO,GAAG,CAAC;IAC/C,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,4BAA4B;IAChC,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC;IACtC,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,kHAAkH;IAC3H,SAAS,EAAE,kFAAkF;IAC7F,IAAI,EAAE,2FAA2F;IACjG,QAAQ,EAAE;QACR,eAAe,CAAC,IAAI,EAAE,GAAG;YACvB,MAAM,EAAE,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAC9C,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC,IAAI,KAAK,mBAAmB;gBAAE,OAAO;YACnD,MAAM,IAAI,GAAG,EAAE,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAC9C,MAAM,GAAG,GAAG,EAAE,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;YAC3C,IAAI,CAAC,IAAI,IAAI,CAAC,GAAG;gBAAE,OAAO;YAC1B,qDAAqD;YACrD,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,gBAAgB;gBAAE,OAAO;YACjD,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAAE,OAAO;YAC3D,MAAM,MAAM,GAAG,mBAAmB,CAAC,IAAI,CAAC,CAAC;YACzC,IAAI,CAAC,MAAM;gBAAE,OAAO;YACpB,0CAA0C;YAC1C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;gBAChD,MAAM,IAAI,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;gBAClC,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM;oBAAE,SAAS;gBAC5C,MAAM,GAAG,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;gBAC1C,IAAI,CAAC,GAAG;oBAAE,SAAS;gBACnB,MAAM,OAAO,GAAG,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;gBACrD,IAAI,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;oBACrC,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC,CAAC;gBAC5B,CAAC;YACH,CAAC;QACH,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"controller-needs-async-wrapper.d.ts","sourceRoot":"","sources":["../../src/rules-default/controller-needs-async-wrapper.ts"],"names":[],"mappings":";AA+BA,wBAmBG"}
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
const ROUTER_METHODS = new Set(['get', 'post', 'put', 'patch', 'delete']);
|
|
3
|
+
function isRouterCall(callee, textOf) {
|
|
4
|
+
if (callee.type !== 'member_expression')
|
|
5
|
+
return false;
|
|
6
|
+
const obj = callee.childForFieldName('object');
|
|
7
|
+
const prop = callee.childForFieldName('property');
|
|
8
|
+
if (!obj || !prop)
|
|
9
|
+
return false;
|
|
10
|
+
const objText = textOf(obj);
|
|
11
|
+
if (objText !== 'router' && objText !== 'app')
|
|
12
|
+
return false;
|
|
13
|
+
return ROUTER_METHODS.has(textOf(prop));
|
|
14
|
+
}
|
|
15
|
+
function isWrappedHandler(arg, textOf) {
|
|
16
|
+
if (arg.type !== 'call_expression')
|
|
17
|
+
return false;
|
|
18
|
+
const fn = arg.childForFieldName('function');
|
|
19
|
+
if (!fn)
|
|
20
|
+
return false;
|
|
21
|
+
return /asyncWrapper|asyncHandler|catchAsync/i.test(textOf(fn));
|
|
22
|
+
}
|
|
23
|
+
function isAsyncFunction(arg) {
|
|
24
|
+
if (arg.type !== 'arrow_function' && arg.type !== 'function_expression')
|
|
25
|
+
return false;
|
|
26
|
+
for (let i = 0; i < arg.childCount; i++) {
|
|
27
|
+
const child = arg.child(i);
|
|
28
|
+
if (child && child.type === 'async')
|
|
29
|
+
return true;
|
|
30
|
+
}
|
|
31
|
+
return arg.text.startsWith('async ');
|
|
32
|
+
}
|
|
33
|
+
export default defineRule({
|
|
34
|
+
id: 'controller-needs-async-wrapper',
|
|
35
|
+
language: ['typescript', 'javascript', 'tsx'],
|
|
36
|
+
severity: 'error',
|
|
37
|
+
message: 'Async route handlers must be wrapped with asyncWrapper / asyncHandler.',
|
|
38
|
+
messageKo: '비동기 라우트 핸들러는 asyncWrapper / asyncHandler로 감싸야 합니다.',
|
|
39
|
+
visitors: {
|
|
40
|
+
call_expression(node, ctx) {
|
|
41
|
+
const callee = node.childForFieldName('function');
|
|
42
|
+
if (!callee || !isRouterCall(callee, ctx.textOf))
|
|
43
|
+
return;
|
|
44
|
+
const args = node.childForFieldName('arguments');
|
|
45
|
+
if (!args || args.namedChildCount < 2)
|
|
46
|
+
return;
|
|
47
|
+
const last = args.namedChild(args.namedChildCount - 1);
|
|
48
|
+
if (!last)
|
|
49
|
+
return;
|
|
50
|
+
if (isAsyncFunction(last) && !isWrappedHandler(last, ctx.textOf)) {
|
|
51
|
+
ctx.report({ node: last });
|
|
52
|
+
}
|
|
53
|
+
},
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
//# sourceMappingURL=controller-needs-async-wrapper.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"controller-needs-async-wrapper.js","sourceRoot":"","sources":["../../src/rules-default/controller-needs-async-wrapper.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,MAAM,cAAc,GAAG,IAAI,GAAG,CAAC,CAAC,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC;AAE1E,SAAS,YAAY,CAAC,MAAyB,EAAE,MAAwC;IACvF,IAAI,MAAM,CAAC,IAAI,KAAK,mBAAmB;QAAE,OAAO,KAAK,CAAC;IACtD,MAAM,GAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;IAC/C,MAAM,IAAI,GAAG,MAAM,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;IAClD,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI;QAAE,OAAO,KAAK,CAAC;IAChC,MAAM,OAAO,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;IAC5B,IAAI,OAAO,KAAK,QAAQ,IAAI,OAAO,KAAK,KAAK;QAAE,OAAO,KAAK,CAAC;IAC5D,OAAO,cAAc,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;AAC1C,CAAC;AAED,SAAS,gBAAgB,CAAC,GAAsB,EAAE,MAAwC;IACxF,IAAI,GAAG,CAAC,IAAI,KAAK,iBAAiB;QAAE,OAAO,KAAK,CAAC;IACjD,MAAM,EAAE,GAAG,GAAG,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;IAC7C,IAAI,CAAC,EAAE;QAAE,OAAO,KAAK,CAAC;IACtB,OAAO,uCAAuC,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;AAClE,CAAC;AAED,SAAS,eAAe,CAAC,GAAsB;IAC7C,IAAI,GAAG,CAAC,IAAI,KAAK,gBAAgB,IAAI,GAAG,CAAC,IAAI,KAAK,qBAAqB;QAAE,OAAO,KAAK,CAAC;IACtF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;QACxC,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAC3B,IAAI,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,OAAO;YAAE,OAAO,IAAI,CAAC;IACnD,CAAC;IACD,OAAO,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;AACvC,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,gCAAgC;IACpC,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,KAAK,CAAC;IAC7C,QAAQ,EAAE,OAAO;IACjB,OAAO,EAAE,wEAAwE;IACjF,SAAS,EAAE,oDAAoD;IAC/D,QAAQ,EAAE;QACR,eAAe,CAAC,IAAI,EAAE,GAAG;YACvB,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC;gBAAE,OAAO;YACzD,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YACjD,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,eAAe,GAAG,CAAC;gBAAE,OAAO;YAC9C,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,eAAe,GAAG,CAAC,CAAC,CAAC;YACvD,IAAI,CAAC,IAAI;gBAAE,OAAO;YAClB,IAAI,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC;gBACjE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;YAC7B,CAAC;QACH,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Detect EUC-KR / CP949 / non-UTF-8 file encoding by checking the first few bytes of the source.
|
|
3
|
+
* UTF-8 BOM is acceptable (some Windows tools add it).
|
|
4
|
+
*
|
|
5
|
+
* Heuristic limitation: tree-sitter has already decoded the source as UTF-8, so we look for
|
|
6
|
+
* `replacement character (U+FFFD)` patterns which appear when invalid UTF-8 was decoded loosely.
|
|
7
|
+
*/
|
|
8
|
+
declare const _default: import("@aicqtools/rule-sdk").FunctionRule;
|
|
9
|
+
export default _default;
|
|
10
|
+
//# sourceMappingURL=enforce-utf8-encoding.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"enforce-utf8-encoding.d.ts","sourceRoot":"","sources":["../../src/rules-default/enforce-utf8-encoding.ts"],"names":[],"mappings":"AAEA;;;;;;GAMG;;AACH,wBAgBG"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
/**
|
|
3
|
+
* Detect EUC-KR / CP949 / non-UTF-8 file encoding by checking the first few bytes of the source.
|
|
4
|
+
* UTF-8 BOM is acceptable (some Windows tools add it).
|
|
5
|
+
*
|
|
6
|
+
* Heuristic limitation: tree-sitter has already decoded the source as UTF-8, so we look for
|
|
7
|
+
* `replacement character (U+FFFD)` patterns which appear when invalid UTF-8 was decoded loosely.
|
|
8
|
+
*/
|
|
9
|
+
export default defineRule({
|
|
10
|
+
id: 'enforce-utf8-encoding',
|
|
11
|
+
language: ['typescript', 'javascript', 'tsx', 'python'],
|
|
12
|
+
severity: 'error',
|
|
13
|
+
message: 'File appears to use non-UTF-8 encoding (replacement chars detected) — re-save as UTF-8.',
|
|
14
|
+
messageKo: '파일이 UTF-8 외 인코딩으로 보입니다 (U+FFFD 검출) — UTF-8로 재저장하세요.',
|
|
15
|
+
visitors: {
|
|
16
|
+
program(node, ctx) {
|
|
17
|
+
// Heuristic: if source contains the Unicode replacement char, decoding likely failed.
|
|
18
|
+
if (ctx.source.includes('�'))
|
|
19
|
+
ctx.report({ node });
|
|
20
|
+
},
|
|
21
|
+
module(node, ctx) {
|
|
22
|
+
// Python's root node is "module", TS's is "program"
|
|
23
|
+
if (ctx.source.includes('�'))
|
|
24
|
+
ctx.report({ node });
|
|
25
|
+
},
|
|
26
|
+
},
|
|
27
|
+
});
|
|
28
|
+
//# sourceMappingURL=enforce-utf8-encoding.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"enforce-utf8-encoding.js","sourceRoot":"","sources":["../../src/rules-default/enforce-utf8-encoding.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD;;;;;;GAMG;AACH,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,uBAAuB;IAC3B,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,KAAK,EAAE,QAAQ,CAAC;IACvD,QAAQ,EAAE,OAAO;IACjB,OAAO,EAAE,yFAAyF;IAClG,SAAS,EAAE,qDAAqD;IAChE,QAAQ,EAAE;QACR,OAAO,CAAC,IAAI,EAAE,GAAG;YACf,sFAAsF;YACtF,IAAI,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QACrD,CAAC;QACD,MAAM,CAAC,IAAI,EAAE,GAAG;YACd,oDAAoD;YACpD,IAAI,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QACrD,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"explicit-kst-timezone.d.ts","sourceRoot":"","sources":["../../src/rules-default/explicit-kst-timezone.ts"],"names":[],"mappings":";AA0BA,wBAkBG"}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
/**
|
|
3
|
+
* `toLocaleString()` / `toLocaleDateString()` without explicit timezone option
|
|
4
|
+
* defaults to the runtime's local zone — non-deterministic in CI / serverless.
|
|
5
|
+
* Korean projects should pass `{ timeZone: 'Asia/Seoul' }`.
|
|
6
|
+
*
|
|
7
|
+
* Known limitation: doesn't catch indirect calls (e.g., via a wrapper function).
|
|
8
|
+
*/
|
|
9
|
+
const LOCALE_METHODS = new Set([
|
|
10
|
+
'toLocaleString',
|
|
11
|
+
'toLocaleDateString',
|
|
12
|
+
'toLocaleTimeString',
|
|
13
|
+
]);
|
|
14
|
+
function hasTimezoneOption(args, textOf) {
|
|
15
|
+
if (!args)
|
|
16
|
+
return false;
|
|
17
|
+
for (let i = 0; i < args.namedChildCount; i++) {
|
|
18
|
+
const arg = args.namedChild(i);
|
|
19
|
+
if (!arg || arg.type !== 'object')
|
|
20
|
+
continue;
|
|
21
|
+
if (/timeZone|timezone/i.test(textOf(arg)))
|
|
22
|
+
return true;
|
|
23
|
+
}
|
|
24
|
+
return false;
|
|
25
|
+
}
|
|
26
|
+
export default defineRule({
|
|
27
|
+
id: 'explicit-kst-timezone',
|
|
28
|
+
language: ['typescript', 'javascript', 'tsx'],
|
|
29
|
+
severity: 'warning',
|
|
30
|
+
message: 'Locale-aware date method without `timeZone` option is non-deterministic — pass `Asia/Seoul` explicitly.',
|
|
31
|
+
messageKo: '`timeZone` 옵션 없는 locale 날짜 메서드는 환경별 결과 차이 — `Asia/Seoul` 명시 권장.',
|
|
32
|
+
docs: 'https://github.com/aicqtools/aicqtools/blob/main/docs/rules/explicit-kst-timezone.md',
|
|
33
|
+
visitors: {
|
|
34
|
+
call_expression(node, ctx) {
|
|
35
|
+
const fn = node.childForFieldName('function');
|
|
36
|
+
if (!fn || fn.type !== 'member_expression')
|
|
37
|
+
return;
|
|
38
|
+
const prop = fn.childForFieldName('property');
|
|
39
|
+
if (!prop)
|
|
40
|
+
return;
|
|
41
|
+
if (!LOCALE_METHODS.has(ctx.textOf(prop)))
|
|
42
|
+
return;
|
|
43
|
+
const args = node.childForFieldName('arguments');
|
|
44
|
+
if (!hasTimezoneOption(args, ctx.textOf))
|
|
45
|
+
ctx.report({ node });
|
|
46
|
+
},
|
|
47
|
+
},
|
|
48
|
+
});
|
|
49
|
+
//# sourceMappingURL=explicit-kst-timezone.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"explicit-kst-timezone.js","sourceRoot":"","sources":["../../src/rules-default/explicit-kst-timezone.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD;;;;;;GAMG;AACH,MAAM,cAAc,GAAG,IAAI,GAAG,CAAC;IAC7B,gBAAgB;IAChB,oBAAoB;IACpB,oBAAoB;CACrB,CAAC,CAAC;AAEH,SAAS,iBAAiB,CAAC,IAA8B,EAAE,MAAwC;IACjG,IAAI,CAAC,IAAI;QAAE,OAAO,KAAK,CAAC;IACxB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9C,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,CAAC,GAAG,IAAI,GAAG,CAAC,IAAI,KAAK,QAAQ;YAAE,SAAS;QAC5C,IAAI,oBAAoB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YAAE,OAAO,IAAI,CAAC;IAC1D,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,uBAAuB;IAC3B,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,EAAE,KAAK,CAAC;IAC7C,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,yGAAyG;IAClH,SAAS,EAAE,iEAAiE;IAC5E,IAAI,EAAE,sFAAsF;IAC5F,QAAQ,EAAE;QACR,eAAe,CAAC,IAAI,EAAE,GAAG;YACvB,MAAM,EAAE,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAC9C,IAAI,CAAC,EAAE,IAAI,EAAE,CAAC,IAAI,KAAK,mBAAmB;gBAAE,OAAO;YACnD,MAAM,IAAI,GAAG,EAAE,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAC9C,IAAI,CAAC,IAAI;gBAAE,OAAO;YAClB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAAE,OAAO;YAClD,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YACjD,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QACjE,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fk-needs-on-delete.d.ts","sourceRoot":"","sources":["../../src/rules-default/fk-needs-on-delete.ts"],"names":[],"mappings":";AA2BA,wBAsBG"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
const FK_METHODS = new Set(['belongsTo', 'hasMany', 'hasOne', 'belongsToMany']);
|
|
3
|
+
function findObjectArg(args) {
|
|
4
|
+
for (let i = 0; i < args.namedChildCount; i++) {
|
|
5
|
+
const arg = args.namedChild(i);
|
|
6
|
+
if (arg && arg.type === 'object')
|
|
7
|
+
return arg;
|
|
8
|
+
}
|
|
9
|
+
return null;
|
|
10
|
+
}
|
|
11
|
+
function objectHasKey(obj, key, textOf) {
|
|
12
|
+
for (let i = 0; i < obj.namedChildCount; i++) {
|
|
13
|
+
const pair = obj.namedChild(i);
|
|
14
|
+
if (!pair || (pair.type !== 'pair' && pair.type !== 'shorthand_property_identifier'))
|
|
15
|
+
continue;
|
|
16
|
+
if (pair.type === 'shorthand_property_identifier') {
|
|
17
|
+
if (textOf(pair) === key)
|
|
18
|
+
return true;
|
|
19
|
+
continue;
|
|
20
|
+
}
|
|
21
|
+
const k = pair.childForFieldName('key');
|
|
22
|
+
if (k && textOf(k).replace(/['"]/g, '') === key)
|
|
23
|
+
return true;
|
|
24
|
+
}
|
|
25
|
+
return false;
|
|
26
|
+
}
|
|
27
|
+
export default defineRule({
|
|
28
|
+
id: 'fk-needs-on-delete',
|
|
29
|
+
language: ['typescript', 'javascript'],
|
|
30
|
+
severity: 'error',
|
|
31
|
+
message: 'Foreign-key association is missing an explicit `onDelete` policy.',
|
|
32
|
+
messageKo: 'FK 관계에 명시적 `onDelete` 정책이 빠졌습니다.',
|
|
33
|
+
visitors: {
|
|
34
|
+
call_expression(node, ctx) {
|
|
35
|
+
const callee = node.childForFieldName('function');
|
|
36
|
+
if (!callee || callee.type !== 'member_expression')
|
|
37
|
+
return;
|
|
38
|
+
const prop = callee.childForFieldName('property');
|
|
39
|
+
if (!prop || !FK_METHODS.has(ctx.textOf(prop)))
|
|
40
|
+
return;
|
|
41
|
+
const args = node.childForFieldName('arguments');
|
|
42
|
+
if (!args)
|
|
43
|
+
return;
|
|
44
|
+
const opts = findObjectArg(args);
|
|
45
|
+
if (!opts) {
|
|
46
|
+
ctx.report({ node });
|
|
47
|
+
return;
|
|
48
|
+
}
|
|
49
|
+
if (!objectHasKey(opts, 'onDelete', ctx.textOf))
|
|
50
|
+
ctx.report({ node });
|
|
51
|
+
},
|
|
52
|
+
},
|
|
53
|
+
});
|
|
54
|
+
//# sourceMappingURL=fk-needs-on-delete.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fk-needs-on-delete.js","sourceRoot":"","sources":["../../src/rules-default/fk-needs-on-delete.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,CAAC,WAAW,EAAE,SAAS,EAAE,QAAQ,EAAE,eAAe,CAAC,CAAC,CAAC;AAEhF,SAAS,aAAa,CAAC,IAAuB;IAC5C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9C,MAAM,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,GAAG,IAAI,GAAG,CAAC,IAAI,KAAK,QAAQ;YAAE,OAAO,GAAG,CAAC;IAC/C,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAS,YAAY,CAAC,GAAsB,EAAE,GAAW,EAAE,MAAwC;IACjG,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,eAAe,EAAE,CAAC,EAAE,EAAE,CAAC;QAC7C,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAC/B,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,KAAK,MAAM,IAAI,IAAI,CAAC,IAAI,KAAK,+BAA+B,CAAC;YAAE,SAAS;QAC/F,IAAI,IAAI,CAAC,IAAI,KAAK,+BAA+B,EAAE,CAAC;YAClD,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,GAAG;gBAAE,OAAO,IAAI,CAAC;YACtC,SAAS;QACX,CAAC;QACD,MAAM,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QACxC,IAAI,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,KAAK,GAAG;YAAE,OAAO,IAAI,CAAC;IAC/D,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,oBAAoB;IACxB,QAAQ,EAAE,CAAC,YAAY,EAAE,YAAY,CAAC;IACtC,QAAQ,EAAE,OAAO;IACjB,OAAO,EAAE,mEAAmE;IAC5E,SAAS,EAAE,kCAAkC;IAC7C,QAAQ,EAAE;QACR,eAAe,CAAC,IAAI,EAAE,GAAG;YACvB,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAClD,IAAI,CAAC,MAAM,IAAI,MAAM,CAAC,IAAI,KAAK,mBAAmB;gBAAE,OAAO;YAC3D,MAAM,IAAI,GAAG,MAAM,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;YAClD,IAAI,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBAAE,OAAO;YACvD,MAAM,IAAI,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;YACjD,IAAI,CAAC,IAAI;gBAAE,OAAO;YAClB,MAAM,IAAI,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;YACjC,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;gBACrB,OAAO;YACT,CAAC;YACD,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC;gBAAE,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QACxE,CAAC;KACF;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"human-oversight-checkpoint.d.ts","sourceRoot":"","sources":["../../src/rules-default/human-oversight-checkpoint.ts"],"names":[],"mappings":";AAyBA,wBAYG"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { defineRule } from '@aicqtools/rule-sdk';
|
|
2
|
+
/**
|
|
3
|
+
* FSC AI guideline: AI-driven decisions writing to authoritative storage must
|
|
4
|
+
* pass through a human oversight checkpoint. Detect functions that invoke an
|
|
5
|
+
* AI SDK and immediately persist the result via DB writes (insert/update/
|
|
6
|
+
* create) without an explicit review marker (review/approve/oversight).
|
|
7
|
+
*
|
|
8
|
+
* Limitation: heuristic — best-effort detection. False positives in pipelines
|
|
9
|
+
* that intentionally store the raw model output for downstream review.
|
|
10
|
+
*/
|
|
11
|
+
const AI_CALL = /\b(openai|anthropic|aiClient)\.\w+/;
|
|
12
|
+
const DB_WRITE = /\b(prisma|knex|db|sequelize|mongoose)\.\w+\.(insert|update|create|save|upsert|insertMany)|\.\$create\(|\.\$update\(/;
|
|
13
|
+
const REVIEW_MARKER = /\b(review|approve|oversight|humanCheck|approveByUser|requiresApproval)\b/i;
|
|
14
|
+
function check(node, ctx) {
|
|
15
|
+
const text = ctx.textOf(node);
|
|
16
|
+
if (!AI_CALL.test(text))
|
|
17
|
+
return;
|
|
18
|
+
if (!DB_WRITE.test(text))
|
|
19
|
+
return;
|
|
20
|
+
if (REVIEW_MARKER.test(text))
|
|
21
|
+
return;
|
|
22
|
+
ctx.report({ node });
|
|
23
|
+
}
|
|
24
|
+
export default defineRule({
|
|
25
|
+
id: 'human-oversight-checkpoint',
|
|
26
|
+
language: ['typescript', 'tsx'],
|
|
27
|
+
severity: 'warning',
|
|
28
|
+
message: 'AI inference result is persisted without a human oversight marker (FSC AI guideline — human-in-the-loop checkpoint).',
|
|
29
|
+
messageKo: 'AI 추론 결과가 인간 검토 표시 없이 저장됩니다 (금감원 AI 가이드라인 — 인간 개입 포인트).',
|
|
30
|
+
docs: 'https://github.com/aicqtools/aicqtools/blob/main/docs/rules/human-oversight-checkpoint.md',
|
|
31
|
+
visitors: {
|
|
32
|
+
function_declaration: check,
|
|
33
|
+
arrow_function: check,
|
|
34
|
+
method_definition: check,
|
|
35
|
+
},
|
|
36
|
+
});
|
|
37
|
+
//# sourceMappingURL=human-oversight-checkpoint.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"human-oversight-checkpoint.js","sourceRoot":"","sources":["../../src/rules-default/human-oversight-checkpoint.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAGjD;;;;;;;;GAQG;AACH,MAAM,OAAO,GAAG,oCAAoC,CAAC;AACrD,MAAM,QAAQ,GAAG,qHAAqH,CAAC;AACvI,MAAM,aAAa,GAAG,2EAA2E,CAAC;AAElG,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,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IACjC,IAAI,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC;QAAE,OAAO;IACrC,GAAG,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;AACvB,CAAC;AAED,eAAe,UAAU,CAAC;IACxB,EAAE,EAAE,4BAA4B;IAChC,QAAQ,EAAE,CAAC,YAAY,EAAE,KAAK,CAAC;IAC/B,QAAQ,EAAE,SAAS;IACnB,OAAO,EAAE,sHAAsH;IAC/H,SAAS,EAAE,yDAAyD;IACpE,IAAI,EAAE,2FAA2F;IACjG,QAAQ,EAAE;QACR,oBAAoB,EAAE,KAAK;QAC3B,cAAc,EAAE,KAAK;QACrB,iBAAiB,EAAE,KAAK;KACzB;CACF,CAAC,CAAC"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { Rule } from '@aicqtools/rule-sdk';
|
|
2
|
+
export declare const builtinFunctionRules: readonly Rule[];
|
|
3
|
+
export declare function loadBuiltinYamlRules(): Promise<Rule[]>;
|
|
4
|
+
export declare function loadAllBuiltinRules(): Promise<Rule[]>;
|
|
5
|
+
export declare function loadFunctionRulesFromDir(dir: string): Promise<Rule[]>;
|
|
6
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/rules-default/index.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,qBAAqB,CAAC;AA0DhD,eAAO,MAAM,oBAAoB,EAAE,SAAS,IAAI,EAmD/C,CAAC;AAEF,wBAAsB,oBAAoB,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC,CAW5D;AAED,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC,CAG3D;AAED,wBAAsB,wBAAwB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC,CAY3E"}
|