@trynullsec/s1-zk 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LIMITATIONS.md +17 -0
- package/README.md +149 -0
- package/ROADMAP.md +21 -0
- package/RULES.md +229 -0
- package/benchmarks/historical/orchard-inspired/partial-ec-mul-halo2.rs +53 -0
- package/benchmarks/historical/orchard-inspired/safe-ec-mul-halo2.rs +69 -0
- package/dist/ai/prompt-builder.d.ts +2 -0
- package/dist/ai/prompt-builder.js +33 -0
- package/dist/ai/prompt-builder.js.map +1 -0
- package/dist/ai/reasoning-interface.d.ts +15 -0
- package/dist/ai/reasoning-interface.js +2 -0
- package/dist/ai/reasoning-interface.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +71 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +4 -0
- package/dist/config.js +41 -0
- package/dist/config.js.map +1 -0
- package/dist/core/audit-engine.d.ts +3 -0
- package/dist/core/audit-engine.js +31 -0
- package/dist/core/audit-engine.js.map +1 -0
- package/dist/core/file-loader.d.ts +6 -0
- package/dist/core/file-loader.js +28 -0
- package/dist/core/file-loader.js.map +1 -0
- package/dist/core/issue-builder.d.ts +16 -0
- package/dist/core/issue-builder.js +27 -0
- package/dist/core/issue-builder.js.map +1 -0
- package/dist/core/range-check-classifier.d.ts +16 -0
- package/dist/core/range-check-classifier.js +194 -0
- package/dist/core/range-check-classifier.js.map +1 -0
- package/dist/core/rule-engine.d.ts +10 -0
- package/dist/core/rule-engine.js +41 -0
- package/dist/core/rule-engine.js.map +1 -0
- package/dist/core/severity.d.ts +6 -0
- package/dist/core/severity.js +21 -0
- package/dist/core/severity.js.map +1 -0
- package/dist/core/source-map.d.ts +3 -0
- package/dist/core/source-map.js +16 -0
- package/dist/core/source-map.js.map +1 -0
- package/dist/frontends/circom/circom-comments.d.ts +8 -0
- package/dist/frontends/circom/circom-comments.js +52 -0
- package/dist/frontends/circom/circom-comments.js.map +1 -0
- package/dist/frontends/circom/circom-ir-builder.d.ts +2 -0
- package/dist/frontends/circom/circom-ir-builder.js +12 -0
- package/dist/frontends/circom/circom-ir-builder.js.map +1 -0
- package/dist/frontends/circom/circom-parser.d.ts +2 -0
- package/dist/frontends/circom/circom-parser.js +218 -0
- package/dist/frontends/circom/circom-parser.js.map +1 -0
- package/dist/frontends/circom/circom-tokenizer.d.ts +7 -0
- package/dist/frontends/circom/circom-tokenizer.js +79 -0
- package/dist/frontends/circom/circom-tokenizer.js.map +1 -0
- package/dist/frontends/circom/circom-utils.d.ts +9 -0
- package/dist/frontends/circom/circom-utils.js +88 -0
- package/dist/frontends/circom/circom-utils.js.map +1 -0
- package/dist/frontends/gnark/gnark-adapter.d.ts +5 -0
- package/dist/frontends/gnark/gnark-adapter.js +7 -0
- package/dist/frontends/gnark/gnark-adapter.js.map +1 -0
- package/dist/frontends/halo2/halo2-adapter.d.ts +8 -0
- package/dist/frontends/halo2/halo2-adapter.js +12 -0
- package/dist/frontends/halo2/halo2-adapter.js.map +1 -0
- package/dist/frontends/halo2/halo2-constraint-extractor.d.ts +29 -0
- package/dist/frontends/halo2/halo2-constraint-extractor.js +38 -0
- package/dist/frontends/halo2/halo2-constraint-extractor.js.map +1 -0
- package/dist/frontends/halo2/halo2-constraint-graph.d.ts +16 -0
- package/dist/frontends/halo2/halo2-constraint-graph.js +96 -0
- package/dist/frontends/halo2/halo2-constraint-graph.js.map +1 -0
- package/dist/frontends/halo2/halo2-dataflow.d.ts +11 -0
- package/dist/frontends/halo2/halo2-dataflow.js +19 -0
- package/dist/frontends/halo2/halo2-dataflow.js.map +1 -0
- package/dist/frontends/halo2/halo2-expression-parser.d.ts +14 -0
- package/dist/frontends/halo2/halo2-expression-parser.js +62 -0
- package/dist/frontends/halo2/halo2-expression-parser.js.map +1 -0
- package/dist/frontends/halo2/halo2-ir-builder.d.ts +2 -0
- package/dist/frontends/halo2/halo2-ir-builder.js +17 -0
- package/dist/frontends/halo2/halo2-ir-builder.js.map +1 -0
- package/dist/frontends/halo2/halo2-parser.d.ts +2 -0
- package/dist/frontends/halo2/halo2-parser.js +316 -0
- package/dist/frontends/halo2/halo2-parser.js.map +1 -0
- package/dist/frontends/halo2/halo2-patterns.d.ts +6 -0
- package/dist/frontends/halo2/halo2-patterns.js +54 -0
- package/dist/frontends/halo2/halo2-patterns.js.map +1 -0
- package/dist/frontends/halo2/halo2-types.d.ts +100 -0
- package/dist/frontends/halo2/halo2-types.js +2 -0
- package/dist/frontends/halo2/halo2-types.js.map +1 -0
- package/dist/frontends/noir/noir-adapter.d.ts +5 -0
- package/dist/frontends/noir/noir-adapter.js +7 -0
- package/dist/frontends/noir/noir-adapter.js.map +1 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.js +4 -0
- package/dist/index.js.map +1 -0
- package/dist/ir/circuit-ir.d.ts +1 -0
- package/dist/ir/circuit-ir.js +2 -0
- package/dist/ir/circuit-ir.js.map +1 -0
- package/dist/ir/component-table.d.ts +6 -0
- package/dist/ir/component-table.js +11 -0
- package/dist/ir/component-table.js.map +1 -0
- package/dist/ir/constraint-graph.d.ts +26 -0
- package/dist/ir/constraint-graph.js +130 -0
- package/dist/ir/constraint-graph.js.map +1 -0
- package/dist/ir/dataflow.d.ts +3 -0
- package/dist/ir/dataflow.js +10 -0
- package/dist/ir/dataflow.js.map +1 -0
- package/dist/ir/reference-index.d.ts +6 -0
- package/dist/ir/reference-index.js +18 -0
- package/dist/ir/reference-index.js.map +1 -0
- package/dist/ir/signal-table.d.ts +6 -0
- package/dist/ir/signal-table.js +16 -0
- package/dist/ir/signal-table.js.map +1 -0
- package/dist/report/json.d.ts +2 -0
- package/dist/report/json.js +4 -0
- package/dist/report/json.js.map +1 -0
- package/dist/report/markdown.d.ts +2 -0
- package/dist/report/markdown.js +45 -0
- package/dist/report/markdown.js.map +1 -0
- package/dist/report/sarif.d.ts +2 -0
- package/dist/report/sarif.js +52 -0
- package/dist/report/sarif.js.map +1 -0
- package/dist/report/summary.d.ts +2 -0
- package/dist/report/summary.js +7 -0
- package/dist/report/summary.js.map +1 -0
- package/dist/report/terminal.d.ts +2 -0
- package/dist/report/terminal.js +50 -0
- package/dist/report/terminal.js.map +1 -0
- package/dist/rules/NS-ZK-001-dangerous-hint-assignment.d.ts +2 -0
- package/dist/rules/NS-ZK-001-dangerous-hint-assignment.js +27 -0
- package/dist/rules/NS-ZK-001-dangerous-hint-assignment.js.map +1 -0
- package/dist/rules/NS-ZK-002-assigned-but-unconstrained.d.ts +2 -0
- package/dist/rules/NS-ZK-002-assigned-but-unconstrained.js +22 -0
- package/dist/rules/NS-ZK-002-assigned-but-unconstrained.js.map +1 -0
- package/dist/rules/NS-ZK-003-unbound-public-input.d.ts +2 -0
- package/dist/rules/NS-ZK-003-unbound-public-input.js +22 -0
- package/dist/rules/NS-ZK-003-unbound-public-input.js.map +1 -0
- package/dist/rules/NS-ZK-004-unconstrained-output.d.ts +2 -0
- package/dist/rules/NS-ZK-004-unconstrained-output.js +27 -0
- package/dist/rules/NS-ZK-004-unconstrained-output.js.map +1 -0
- package/dist/rules/NS-ZK-005-missing-booleanity.d.ts +2 -0
- package/dist/rules/NS-ZK-005-missing-booleanity.js +26 -0
- package/dist/rules/NS-ZK-005-missing-booleanity.js.map +1 -0
- package/dist/rules/NS-ZK-006-missing-range-check.d.ts +2 -0
- package/dist/rules/NS-ZK-006-missing-range-check.js +32 -0
- package/dist/rules/NS-ZK-006-missing-range-check.js.map +1 -0
- package/dist/rules/NS-ZK-007-unsafe-assertion.d.ts +2 -0
- package/dist/rules/NS-ZK-007-unsafe-assertion.js +24 -0
- package/dist/rules/NS-ZK-007-unsafe-assertion.js.map +1 -0
- package/dist/rules/NS-ZK-008-unsafe-division-or-inverse.d.ts +2 -0
- package/dist/rules/NS-ZK-008-unsafe-division-or-inverse.js +39 -0
- package/dist/rules/NS-ZK-008-unsafe-division-or-inverse.js.map +1 -0
- package/dist/rules/NS-ZK-009-unconstrained-component-output.d.ts +2 -0
- package/dist/rules/NS-ZK-009-unconstrained-component-output.js +25 -0
- package/dist/rules/NS-ZK-009-unconstrained-component-output.js.map +1 -0
- package/dist/rules/NS-ZK-010-alias-overflow-risk.d.ts +2 -0
- package/dist/rules/NS-ZK-010-alias-overflow-risk.js +48 -0
- package/dist/rules/NS-ZK-010-alias-overflow-risk.js.map +1 -0
- package/dist/rules/NS-ZK-011-unused-signal.d.ts +2 -0
- package/dist/rules/NS-ZK-011-unused-signal.js +26 -0
- package/dist/rules/NS-ZK-011-unused-signal.js.map +1 -0
- package/dist/rules/NS-ZK-012-suspicious-selector.d.ts +2 -0
- package/dist/rules/NS-ZK-012-suspicious-selector.js +33 -0
- package/dist/rules/NS-ZK-012-suspicious-selector.js.map +1 -0
- package/dist/rules/halo2/NS-H2-001-assigned-advice-not-constrained.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-001-assigned-advice-not-constrained.js +40 -0
- package/dist/rules/halo2/NS-H2-001-assigned-advice-not-constrained.js.map +1 -0
- package/dist/rules/halo2/NS-H2-002-instance-not-bound.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-002-instance-not-bound.js +32 -0
- package/dist/rules/halo2/NS-H2-002-instance-not-bound.js.map +1 -0
- package/dist/rules/halo2/NS-H2-003-selector-risk.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-003-selector-risk.js +37 -0
- package/dist/rules/halo2/NS-H2-003-selector-risk.js.map +1 -0
- package/dist/rules/halo2/NS-H2-004-unsafe-inverse.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-004-unsafe-inverse.js +43 -0
- package/dist/rules/halo2/NS-H2-004-unsafe-inverse.js.map +1 -0
- package/dist/rules/halo2/NS-H2-005-partial-ec-operation.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-005-partial-ec-operation.js +43 -0
- package/dist/rules/halo2/NS-H2-005-partial-ec-operation.js.map +1 -0
- package/dist/rules/halo2/NS-H2-006-missing-enable-equality.d.ts +2 -0
- package/dist/rules/halo2/NS-H2-006-missing-enable-equality.js +31 -0
- package/dist/rules/halo2/NS-H2-006-missing-enable-equality.js.map +1 -0
- package/dist/rules/halo2/halo2-rule-utils.d.ts +8 -0
- package/dist/rules/halo2/halo2-rule-utils.js +24 -0
- package/dist/rules/halo2/halo2-rule-utils.js.map +1 -0
- package/dist/rules/index.d.ts +2 -0
- package/dist/rules/index.js +39 -0
- package/dist/rules/index.js.map +1 -0
- package/dist/scanner.d.ts +8 -0
- package/dist/scanner.js +52 -0
- package/dist/scanner.js.map +1 -0
- package/dist/types.d.ts +171 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/examples/halo2/safe/bound-instance.rs +23 -0
- package/examples/halo2/safe/constrained-advice.rs +27 -0
- package/examples/halo2/safe/safe-inverse.rs +30 -0
- package/examples/halo2/safe/safe-selector.rs +27 -0
- package/examples/halo2/vulnerable/missing-selector-booleanity.rs +20 -0
- package/examples/halo2/vulnerable/partial-ec-mul.rs +23 -0
- package/examples/halo2/vulnerable/unbound-instance.rs +20 -0
- package/examples/halo2/vulnerable/unconstrained-advice.rs +24 -0
- package/examples/halo2/vulnerable/unsafe-inverse.rs +20 -0
- package/examples/safe/safe-boolean.circom +15 -0
- package/examples/safe/safe-component-output.circom +18 -0
- package/examples/safe/safe-division.circom +14 -0
- package/examples/safe/safe-public-input.circom +15 -0
- package/examples/safe/safe-range-check.circom +16 -0
- package/examples/safe/safe-transfer.circom +19 -0
- package/examples/vulnerable/alias-overflow-risk.circom +12 -0
- package/examples/vulnerable/merkle-selector-bug.circom +12 -0
- package/examples/vulnerable/missing-boolean.circom +11 -0
- package/examples/vulnerable/missing-range-check.circom +11 -0
- package/examples/vulnerable/public-input-not-bound.circom +12 -0
- package/examples/vulnerable/unconstrained-component-output.circom +15 -0
- package/examples/vulnerable/unconstrained-transfer.circom +17 -0
- package/examples/vulnerable/unsafe-assertion.circom +12 -0
- package/examples/vulnerable/unsafe-division.circom +11 -0
- package/package.json +46 -0
package/dist/cli.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":"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"}
|
package/dist/config.d.ts
ADDED
package/dist/config.js
ADDED
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync } from "node:fs";
|
|
2
|
+
import { resolve } from "node:path";
|
|
3
|
+
import { normalizeSeverity } from "./core/severity.js";
|
|
4
|
+
export const defaultConfig = {
|
|
5
|
+
rules: {},
|
|
6
|
+
ignore: ["node_modules", "dist", "circomlib"],
|
|
7
|
+
failOn: "CRITICAL",
|
|
8
|
+
format: "terminal",
|
|
9
|
+
field: "BN254"
|
|
10
|
+
};
|
|
11
|
+
function isFormat(value) {
|
|
12
|
+
return value === "terminal" || value === "json" || value === "markdown" || value === "sarif";
|
|
13
|
+
}
|
|
14
|
+
export function loadConfig(cwd = process.cwd(), configPath) {
|
|
15
|
+
const path = configPath ? resolve(cwd, configPath) : resolve(cwd, ".nullsec-zk.json");
|
|
16
|
+
if (!existsSync(path))
|
|
17
|
+
return { ...defaultConfig, rules: { ...defaultConfig.rules }, ignore: [...defaultConfig.ignore] };
|
|
18
|
+
const parsed = JSON.parse(readFileSync(path, "utf8"));
|
|
19
|
+
return {
|
|
20
|
+
rules: parsed.rules ?? {},
|
|
21
|
+
ignore: Array.isArray(parsed.ignore) ? parsed.ignore : [...defaultConfig.ignore],
|
|
22
|
+
failOn: normalizeSeverity(String(parsed.failOn ?? defaultConfig.failOn), defaultConfig.failOn),
|
|
23
|
+
format: isFormat(parsed.format) ? parsed.format : defaultConfig.format,
|
|
24
|
+
field: parsed.field ?? defaultConfig.field
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
export function writeDefaultConfig(cwd = process.cwd()) {
|
|
28
|
+
const path = resolve(cwd, ".nullsec-zk.json");
|
|
29
|
+
const payload = {
|
|
30
|
+
rules: {
|
|
31
|
+
"NS-ZK-006": "medium"
|
|
32
|
+
},
|
|
33
|
+
ignore: ["node_modules", "circomlib"],
|
|
34
|
+
failOn: "CRITICAL",
|
|
35
|
+
format: "terminal",
|
|
36
|
+
field: "BN254"
|
|
37
|
+
};
|
|
38
|
+
writeFileSync(path, `${JSON.stringify(payload, null, 2)}\n`);
|
|
39
|
+
return path;
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AAEvD,MAAM,CAAC,MAAM,aAAa,GAAkB;IAC1C,KAAK,EAAE,EAAE;IACT,MAAM,EAAE,CAAC,cAAc,EAAE,MAAM,EAAE,WAAW,CAAC;IAC7C,MAAM,EAAE,UAAU;IAClB,MAAM,EAAE,UAAU;IAClB,KAAK,EAAE,OAAO;CACf,CAAC;AAEF,SAAS,QAAQ,CAAC,KAAc;IAC9B,OAAO,KAAK,KAAK,UAAU,IAAI,KAAK,KAAK,MAAM,IAAI,KAAK,KAAK,UAAU,IAAI,KAAK,KAAK,OAAO,CAAC;AAC/F,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,EAAE,UAAmB;IACjE,MAAM,IAAI,GAAG,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,kBAAkB,CAAC,CAAC;IACtF,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;QAAE,OAAO,EAAE,GAAG,aAAa,EAAE,KAAK,EAAE,EAAE,GAAG,aAAa,CAAC,KAAK,EAAE,EAAE,MAAM,EAAE,CAAC,GAAG,aAAa,CAAC,MAAM,CAAC,EAAE,CAAC;IAEzH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,CAA2B,CAAC;IAChF,OAAO;QACL,KAAK,EAAE,MAAM,CAAC,KAAK,IAAI,EAAE;QACzB,MAAM,EAAE,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,aAAa,CAAC,MAAM,CAAC;QAChF,MAAM,EAAE,iBAAiB,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,IAAI,aAAa,CAAC,MAAM,CAAC,EAAE,aAAa,CAAC,MAAM,CAAC;QAC9F,MAAM,EAAE,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM;QACtE,KAAK,EAAE,MAAM,CAAC,KAAK,IAAI,aAAa,CAAC,KAAK;KAC3C,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE;IACpD,MAAM,IAAI,GAAG,OAAO,CAAC,GAAG,EAAE,kBAAkB,CAAC,CAAC;IAC9C,MAAM,OAAO,GAAG;QACd,KAAK,EAAE;YACL,WAAW,EAAE,QAAQ;SACtB;QACD,MAAM,EAAE,CAAC,cAAc,EAAE,WAAW,CAAC;QACrC,MAAM,EAAE,UAA6B;QACrC,MAAM,EAAE,UAAiC;QACzC,KAAK,EAAE,OAAO;KACf,CAAC;IACF,aAAa,CAAC,IAAI,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC;IAC7D,OAAO,IAAI,CAAC;AACd,CAAC"}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
import type { AuditResult, NullsecConfig, ParsedCircuitFile } from "../types.js";
|
|
2
|
+
import type { Halo2CircuitFile } from "../frontends/halo2/halo2-types.js";
|
|
3
|
+
export declare function auditParsedFiles(target: string, parsedFiles: ParsedCircuitFile[], config: NullsecConfig, halo2Files?: Halo2CircuitFile[]): AuditResult;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { buildCircuitIR } from "../frontends/circom/circom-ir-builder.js";
|
|
2
|
+
import { buildHalo2IR } from "../frontends/halo2/halo2-ir-builder.js";
|
|
3
|
+
import { ConstraintGraph } from "../ir/constraint-graph.js";
|
|
4
|
+
import { allRules } from "../rules/index.js";
|
|
5
|
+
import { summarizeIssues } from "../report/summary.js";
|
|
6
|
+
import { RuleEngine } from "./rule-engine.js";
|
|
7
|
+
function frontendName(circomCount, halo2Count) {
|
|
8
|
+
if (circomCount > 0 && halo2Count > 0)
|
|
9
|
+
return "Mixed";
|
|
10
|
+
if (halo2Count > 0)
|
|
11
|
+
return "Halo2";
|
|
12
|
+
return "Circom";
|
|
13
|
+
}
|
|
14
|
+
export function auditParsedFiles(target, parsedFiles, config, halo2Files = []) {
|
|
15
|
+
const ir = buildCircuitIR(parsedFiles);
|
|
16
|
+
const halo2 = buildHalo2IR(halo2Files);
|
|
17
|
+
const graph = new ConstraintGraph(ir);
|
|
18
|
+
const engine = new RuleEngine(allRules);
|
|
19
|
+
const { issues, rulesExecuted } = engine.run({ target, ir, graph, config, halo2 });
|
|
20
|
+
return {
|
|
21
|
+
tool: { name: "Nullsec S1-ZK", version: "1.0.0" },
|
|
22
|
+
target,
|
|
23
|
+
frontend: frontendName(parsedFiles.length, halo2Files.length),
|
|
24
|
+
filesScanned: parsedFiles.length + halo2Files.length,
|
|
25
|
+
rulesExecuted,
|
|
26
|
+
summary: summarizeIssues(issues),
|
|
27
|
+
issues,
|
|
28
|
+
parserWarnings: [...ir.parserWarnings, ...halo2.parserWarnings]
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=audit-engine.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"audit-engine.js","sourceRoot":"","sources":["../../src/core/audit-engine.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,MAAM,0CAA0C,CAAC;AAC1E,OAAO,EAAE,YAAY,EAAE,MAAM,wCAAwC,CAAC;AAEtE,OAAO,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AAC5D,OAAO,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAC7C,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,EAAE,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAE9C,SAAS,YAAY,CAAC,WAAmB,EAAE,UAAkB;IAC3D,IAAI,WAAW,GAAG,CAAC,IAAI,UAAU,GAAG,CAAC;QAAE,OAAO,OAAO,CAAC;IACtD,IAAI,UAAU,GAAG,CAAC;QAAE,OAAO,OAAO,CAAC;IACnC,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,MAAc,EAAE,WAAgC,EAAE,MAAqB,EAAE,aAAiC,EAAE;IAC3I,MAAM,EAAE,GAAG,cAAc,CAAC,WAAW,CAAC,CAAC;IACvC,MAAM,KAAK,GAAG,YAAY,CAAC,UAAU,CAAC,CAAC;IACvC,MAAM,KAAK,GAAG,IAAI,eAAe,CAAC,EAAE,CAAC,CAAC;IACtC,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,QAAQ,CAAC,CAAC;IACxC,MAAM,EAAE,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,CAAC,GAAG,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC;IACnF,OAAO;QACL,IAAI,EAAE,EAAE,IAAI,EAAE,eAAe,EAAE,OAAO,EAAE,OAAO,EAAE;QACjD,MAAM;QACN,QAAQ,EAAE,YAAY,CAAC,WAAW,CAAC,MAAM,EAAE,UAAU,CAAC,MAAM,CAAC;QAC7D,YAAY,EAAE,WAAW,CAAC,MAAM,GAAG,UAAU,CAAC,MAAM;QACpD,aAAa;QACb,OAAO,EAAE,eAAe,CAAC,MAAM,CAAC;QAChC,MAAM;QACN,cAAc,EAAE,CAAC,GAAG,EAAE,CAAC,cAAc,EAAE,GAAG,KAAK,CAAC,cAAc,CAAC;KAChE,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export interface LoadedFile {
|
|
2
|
+
filePath: string;
|
|
3
|
+
rawSource: string;
|
|
4
|
+
}
|
|
5
|
+
export declare function loadCircomFiles(target: string, ignore?: string[]): Promise<LoadedFile[]>;
|
|
6
|
+
export declare function loadRustFiles(target: string, ignore?: string[]): Promise<LoadedFile[]>;
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { readFile } from "node:fs/promises";
|
|
2
|
+
import { statSync } from "node:fs";
|
|
3
|
+
import { resolve } from "node:path";
|
|
4
|
+
import fg from "fast-glob";
|
|
5
|
+
async function loadFilesByExtension(target, extensions, ignore = []) {
|
|
6
|
+
const absoluteTarget = resolve(process.cwd(), target);
|
|
7
|
+
const stat = statSync(absoluteTarget);
|
|
8
|
+
const patterns = stat.isDirectory() ? extensions.map((extension) => `**/*${extension}`) : [absoluteTarget];
|
|
9
|
+
const cwd = stat.isDirectory() ? absoluteTarget : process.cwd();
|
|
10
|
+
const entries = await fg(patterns, {
|
|
11
|
+
cwd,
|
|
12
|
+
absolute: true,
|
|
13
|
+
onlyFiles: true,
|
|
14
|
+
ignore: ignore.map((entry) => `**/${entry}/**`)
|
|
15
|
+
});
|
|
16
|
+
const unique = [...new Set(entries)].filter((file) => extensions.some((extension) => file.endsWith(extension)));
|
|
17
|
+
return Promise.all(unique.sort().map(async (filePath) => ({
|
|
18
|
+
filePath,
|
|
19
|
+
rawSource: await readFile(filePath, "utf8")
|
|
20
|
+
})));
|
|
21
|
+
}
|
|
22
|
+
export async function loadCircomFiles(target, ignore = []) {
|
|
23
|
+
return loadFilesByExtension(target, [".circom"], ignore);
|
|
24
|
+
}
|
|
25
|
+
export async function loadRustFiles(target, ignore = []) {
|
|
26
|
+
return loadFilesByExtension(target, [".rs"], ignore);
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=file-loader.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"file-loader.js","sourceRoot":"","sources":["../../src/core/file-loader.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAC5C,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AACnC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,MAAM,WAAW,CAAC;AAO3B,KAAK,UAAU,oBAAoB,CAAC,MAAc,EAAE,UAAoB,EAAE,SAAmB,EAAE;IAC7F,MAAM,cAAc,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,MAAM,CAAC,CAAC;IACtD,MAAM,IAAI,GAAG,QAAQ,CAAC,cAAc,CAAC,CAAC;IACtC,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,OAAO,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC;IAC3G,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;IAChE,MAAM,OAAO,GAAG,MAAM,EAAE,CAAC,QAAQ,EAAE;QACjC,GAAG;QACH,QAAQ,EAAE,IAAI;QACd,SAAS,EAAE,IAAI;QACf,MAAM,EAAE,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,MAAM,KAAK,KAAK,CAAC;KAChD,CAAC,CAAC;IAEH,MAAM,MAAM,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;IAChH,OAAO,OAAO,CAAC,GAAG,CAChB,MAAM,CAAC,IAAI,EAAE,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC;QACrC,QAAQ;QACR,SAAS,EAAE,MAAM,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;KAC5C,CAAC,CAAC,CACJ,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,MAAc,EAAE,SAAmB,EAAE;IACzE,OAAO,oBAAoB,CAAC,MAAM,EAAE,CAAC,SAAS,CAAC,EAAE,MAAM,CAAC,CAAC;AAC3D,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,aAAa,CAAC,MAAc,EAAE,SAAmB,EAAE;IACvE,OAAO,oBAAoB,CAAC,MAAM,EAAE,CAAC,KAAK,CAAC,EAAE,MAAM,CAAC,CAAC;AACvD,CAAC"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type { Confidence, Issue, Severity, SourceLocation } from "../types.js";
|
|
2
|
+
export declare function resetIssueCounter(): void;
|
|
3
|
+
export declare function buildIssue(input: {
|
|
4
|
+
ruleId: string;
|
|
5
|
+
title: string;
|
|
6
|
+
severity: Severity;
|
|
7
|
+
confidence?: Confidence;
|
|
8
|
+
location: SourceLocation;
|
|
9
|
+
signalName?: string;
|
|
10
|
+
explanation: string;
|
|
11
|
+
impact: string;
|
|
12
|
+
suggestedFix: string;
|
|
13
|
+
references?: string[];
|
|
14
|
+
tags?: string[];
|
|
15
|
+
metadata?: Record<string, unknown>;
|
|
16
|
+
}): Issue;
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
let issueCounter = 0;
|
|
2
|
+
export function resetIssueCounter() {
|
|
3
|
+
issueCounter = 0;
|
|
4
|
+
}
|
|
5
|
+
export function buildIssue(input) {
|
|
6
|
+
issueCounter += 1;
|
|
7
|
+
return {
|
|
8
|
+
id: `${input.ruleId}-${String(issueCounter).padStart(4, "0")}`,
|
|
9
|
+
ruleId: input.ruleId,
|
|
10
|
+
title: input.title,
|
|
11
|
+
severity: input.severity,
|
|
12
|
+
confidence: input.confidence ?? "HIGH",
|
|
13
|
+
file: input.location.file,
|
|
14
|
+
line: input.location.line,
|
|
15
|
+
column: input.location.column,
|
|
16
|
+
snippet: input.location.snippet,
|
|
17
|
+
templateName: input.location.templateName,
|
|
18
|
+
signalName: input.signalName,
|
|
19
|
+
explanation: input.explanation,
|
|
20
|
+
impact: input.impact,
|
|
21
|
+
suggestedFix: input.suggestedFix,
|
|
22
|
+
references: input.references,
|
|
23
|
+
tags: input.tags ?? [],
|
|
24
|
+
metadata: input.metadata
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=issue-builder.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"issue-builder.js","sourceRoot":"","sources":["../../src/core/issue-builder.ts"],"names":[],"mappings":"AAEA,IAAI,YAAY,GAAG,CAAC,CAAC;AAErB,MAAM,UAAU,iBAAiB;IAC/B,YAAY,GAAG,CAAC,CAAC;AACnB,CAAC;AAED,MAAM,UAAU,UAAU,CAAC,KAa1B;IACC,YAAY,IAAI,CAAC,CAAC;IAClB,OAAO;QACL,EAAE,EAAE,GAAG,KAAK,CAAC,MAAM,IAAI,MAAM,CAAC,YAAY,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE;QAC9D,MAAM,EAAE,KAAK,CAAC,MAAM;QACpB,KAAK,EAAE,KAAK,CAAC,KAAK;QAClB,QAAQ,EAAE,KAAK,CAAC,QAAQ;QACxB,UAAU,EAAE,KAAK,CAAC,UAAU,IAAI,MAAM;QACtC,IAAI,EAAE,KAAK,CAAC,QAAQ,CAAC,IAAI;QACzB,IAAI,EAAE,KAAK,CAAC,QAAQ,CAAC,IAAI;QACzB,MAAM,EAAE,KAAK,CAAC,QAAQ,CAAC,MAAM;QAC7B,OAAO,EAAE,KAAK,CAAC,QAAQ,CAAC,OAAO;QAC/B,YAAY,EAAE,KAAK,CAAC,QAAQ,CAAC,YAAY;QACzC,UAAU,EAAE,KAAK,CAAC,UAAU;QAC5B,WAAW,EAAE,KAAK,CAAC,WAAW;QAC9B,MAAM,EAAE,KAAK,CAAC,MAAM;QACpB,YAAY,EAAE,KAAK,CAAC,YAAY;QAChC,UAAU,EAAE,KAAK,CAAC,UAAU;QAC5B,IAAI,EAAE,KAAK,CAAC,IAAI,IAAI,EAAE;QACtB,QAAQ,EAAE,KAAK,CAAC,QAAQ;KACzB,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import type { AuditContext, Confidence, Severity, SignalDeclaration, SignalKind } from "../types.js";
|
|
2
|
+
export type RangeCheckCategory = "amount" | "balance" | "fee" | "nonce" | "index" | "timestamp" | "quantity" | "count" | "size" | "limb" | "hash" | "commitment" | "root" | "nullifier" | "selector" | "derived" | "unknown";
|
|
3
|
+
export interface RangeCheckAssessment {
|
|
4
|
+
shouldReport: boolean;
|
|
5
|
+
category: RangeCheckCategory;
|
|
6
|
+
confidence: Confidence;
|
|
7
|
+
severity: Severity;
|
|
8
|
+
explanation: string;
|
|
9
|
+
}
|
|
10
|
+
export declare function classifyRangeCheckSignal(name: string, kind: SignalKind): RangeCheckCategory;
|
|
11
|
+
export declare function isHashLikeCategory(category: RangeCheckCategory): boolean;
|
|
12
|
+
export declare function isPrimaryRangeCategory(category: RangeCheckCategory): boolean;
|
|
13
|
+
export declare function isHashComponentOutput(signal: SignalDeclaration, context: AuditContext): boolean;
|
|
14
|
+
export declare function isUsedAsBoundedInteger(signal: SignalDeclaration, context: AuditContext): boolean;
|
|
15
|
+
export declare function operandsHaveRangeCheck(signal: SignalDeclaration, context: AuditContext): boolean;
|
|
16
|
+
export declare function assessRangeCheckRisk(signal: SignalDeclaration, context: AuditContext): RangeCheckAssessment;
|
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
import { baseSignalName } from "../frontends/circom/circom-utils.js";
|
|
2
|
+
const PRIMARY_RANGE_CATEGORIES = new Set([
|
|
3
|
+
"amount",
|
|
4
|
+
"balance",
|
|
5
|
+
"fee",
|
|
6
|
+
"nonce",
|
|
7
|
+
"index",
|
|
8
|
+
"timestamp",
|
|
9
|
+
"quantity",
|
|
10
|
+
"count",
|
|
11
|
+
"size",
|
|
12
|
+
"limb"
|
|
13
|
+
]);
|
|
14
|
+
const HASH_COMPONENT_PATTERN = /Poseidon|Pedersen|Rescue|MiMC|MimcSponge|Sha256|Blake/i;
|
|
15
|
+
function sameTemplate(a, b) {
|
|
16
|
+
if (!a || !b)
|
|
17
|
+
return true;
|
|
18
|
+
return a === b;
|
|
19
|
+
}
|
|
20
|
+
export function classifyRangeCheckSignal(name, kind) {
|
|
21
|
+
const lower = name.toLowerCase();
|
|
22
|
+
if (/nullifier/.test(lower))
|
|
23
|
+
return "nullifier";
|
|
24
|
+
if (/hash/.test(lower))
|
|
25
|
+
return "hash";
|
|
26
|
+
if (/commitment/.test(lower))
|
|
27
|
+
return "commitment";
|
|
28
|
+
if (/^root$|merkleroot|merkle_root|computedroot|state_root|treeroot/.test(lower) || (lower.includes("root") && kind === "input")) {
|
|
29
|
+
return "root";
|
|
30
|
+
}
|
|
31
|
+
if (kind === "output" && /^remaining|^allowed|^total|^new|^updated|^adjusted|^resulting|^computed/.test(lower)) {
|
|
32
|
+
return "derived";
|
|
33
|
+
}
|
|
34
|
+
if (/selector|select/.test(lower))
|
|
35
|
+
return "selector";
|
|
36
|
+
if (lower === "amount" || /(^|_)amount($|_)/.test(lower))
|
|
37
|
+
return "amount";
|
|
38
|
+
if (lower === "balance" || /(^|_)balance($|_)/.test(lower))
|
|
39
|
+
return "balance";
|
|
40
|
+
if (/\bfee\b|^fee|_fee$/.test(lower))
|
|
41
|
+
return "fee";
|
|
42
|
+
if (/\bnonce\b/.test(lower))
|
|
43
|
+
return "nonce";
|
|
44
|
+
if (/\bindex\b|\bidx\b|_index$|_idx$/.test(lower))
|
|
45
|
+
return "index";
|
|
46
|
+
if (/timestamp|time_stamp/.test(lower))
|
|
47
|
+
return "timestamp";
|
|
48
|
+
if (/\bquantity\b|\bqty\b/.test(lower))
|
|
49
|
+
return "quantity";
|
|
50
|
+
if (/\bcount\b/.test(lower))
|
|
51
|
+
return "count";
|
|
52
|
+
if (/\bsize\b|\blength\b/.test(lower))
|
|
53
|
+
return "size";
|
|
54
|
+
if (/\blimb\b|\blimbs\b/.test(lower))
|
|
55
|
+
return "limb";
|
|
56
|
+
return "unknown";
|
|
57
|
+
}
|
|
58
|
+
export function isHashLikeCategory(category) {
|
|
59
|
+
return category === "hash" || category === "commitment" || category === "root" || category === "nullifier";
|
|
60
|
+
}
|
|
61
|
+
export function isPrimaryRangeCategory(category) {
|
|
62
|
+
return PRIMARY_RANGE_CATEGORIES.has(category);
|
|
63
|
+
}
|
|
64
|
+
function namesEquivalent(reference, signalName) {
|
|
65
|
+
return baseSignalName(reference) === baseSignalName(signalName);
|
|
66
|
+
}
|
|
67
|
+
export function isHashComponentOutput(signal, context) {
|
|
68
|
+
return context.graph.assignmentsForSignal(signal.name).some((assignment) => {
|
|
69
|
+
const match = assignment.rhs.match(/^([A-Za-z_][A-Za-z0-9_]*)\.out\b/);
|
|
70
|
+
if (!match)
|
|
71
|
+
return false;
|
|
72
|
+
const component = context.ir.components.find((candidate) => candidate.name === match[1]);
|
|
73
|
+
return Boolean(component && HASH_COMPONENT_PATTERN.test(component.templateType ?? ""));
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
export function isUsedAsBoundedInteger(signal, context) {
|
|
77
|
+
const references = context.graph.signalReferences(signal.name).filter((ref) => sameTemplate(ref.templateName, signal.templateName));
|
|
78
|
+
return references.some((ref) => {
|
|
79
|
+
const snippet = (ref.snippet ?? "").replace(/\s+/g, " ");
|
|
80
|
+
if (/LessThan|LessEqThan|GreaterThan|GreaterEqThan|Num2Bits|RangeCheck|rangeCheck|AliasCheck|Decompose/.test(snippet)) {
|
|
81
|
+
return true;
|
|
82
|
+
}
|
|
83
|
+
const mentionsSignal = snippet.includes(signal.baseName);
|
|
84
|
+
if (!mentionsSignal)
|
|
85
|
+
return false;
|
|
86
|
+
if (/[+\-*/]/.test(snippet) && !/===/.test(snippet))
|
|
87
|
+
return true;
|
|
88
|
+
if (/===/.test(snippet) && /[+\-*/]/.test(snippet))
|
|
89
|
+
return true;
|
|
90
|
+
return false;
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
export function operandsHaveRangeCheck(signal, context) {
|
|
94
|
+
const assignments = context.graph
|
|
95
|
+
.assignmentsForSignal(signal.name)
|
|
96
|
+
.filter((assignment) => sameTemplate(assignment.templateName, signal.templateName));
|
|
97
|
+
return assignments.some((assignment) => {
|
|
98
|
+
const operands = assignment.referencedSignals.filter((ref) => !namesEquivalent(ref, signal.name));
|
|
99
|
+
const primaryOperands = operands.filter((operand) => isPrimaryRangeCategory(classifyRangeCheckSignal(baseSignalName(operand), "input")));
|
|
100
|
+
if (primaryOperands.length === 0)
|
|
101
|
+
return true;
|
|
102
|
+
return primaryOperands.every((operand) => context.graph.hasRangeCheck(operand, signal.templateName));
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
function isArithmeticOutput(signal, context) {
|
|
106
|
+
if (signal.kind !== "output")
|
|
107
|
+
return false;
|
|
108
|
+
return context.graph
|
|
109
|
+
.assignmentsForSignal(signal.name)
|
|
110
|
+
.filter((assignment) => sameTemplate(assignment.templateName, signal.templateName))
|
|
111
|
+
.some((assignment) => /[+\-*/]/.test(assignment.rhs));
|
|
112
|
+
}
|
|
113
|
+
export function assessRangeCheckRisk(signal, context) {
|
|
114
|
+
const category = classifyRangeCheckSignal(signal.baseName, signal.kind);
|
|
115
|
+
const templateName = signal.templateName;
|
|
116
|
+
if (context.graph.hasRangeCheck(signal.name, templateName)) {
|
|
117
|
+
return {
|
|
118
|
+
shouldReport: false,
|
|
119
|
+
category,
|
|
120
|
+
confidence: "HIGH",
|
|
121
|
+
severity: "INFO",
|
|
122
|
+
explanation: "Signal already has an obvious range-check pattern."
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
if (context.graph.signalReferences(signal.name).filter((ref) => sameTemplate(ref.templateName, templateName)).length === 0) {
|
|
126
|
+
return {
|
|
127
|
+
shouldReport: false,
|
|
128
|
+
category,
|
|
129
|
+
confidence: "LOW",
|
|
130
|
+
severity: "INFO",
|
|
131
|
+
explanation: "Signal is not referenced in parsed constraints or assignments."
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
const hashOutput = isHashComponentOutput(signal, context);
|
|
135
|
+
const boundedIntegerUse = isUsedAsBoundedInteger(signal, context);
|
|
136
|
+
const derivedOutput = category === "derived" || isArithmeticOutput(signal, context);
|
|
137
|
+
const operandsChecked = operandsHaveRangeCheck(signal, context);
|
|
138
|
+
if (hashOutput || isHashLikeCategory(category)) {
|
|
139
|
+
if (!boundedIntegerUse) {
|
|
140
|
+
return {
|
|
141
|
+
shouldReport: false,
|
|
142
|
+
category,
|
|
143
|
+
confidence: "LOW",
|
|
144
|
+
severity: "INFO",
|
|
145
|
+
explanation: `The ${category} signal \`${signal.name}\` is used for binding or hashing, not as a bounded integer.`
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
return {
|
|
149
|
+
shouldReport: true,
|
|
150
|
+
category,
|
|
151
|
+
confidence: "LOW",
|
|
152
|
+
severity: "LOW",
|
|
153
|
+
explanation: `The ${category} signal \`${signal.name}\` is used in arithmetic without an obvious range-check pattern. Confidence is low because hash-like values are usually field elements, not bounded integers.`
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
if (category === "derived" || derivedOutput) {
|
|
157
|
+
if (operandsChecked) {
|
|
158
|
+
return {
|
|
159
|
+
shouldReport: false,
|
|
160
|
+
category: "derived",
|
|
161
|
+
confidence: "LOW",
|
|
162
|
+
severity: "INFO",
|
|
163
|
+
explanation: `The derived output \`${signal.name}\` is computed from operands that already have range checks.`
|
|
164
|
+
};
|
|
165
|
+
}
|
|
166
|
+
if (signal.kind === "output") {
|
|
167
|
+
return {
|
|
168
|
+
shouldReport: true,
|
|
169
|
+
category: "derived",
|
|
170
|
+
confidence: "LOW",
|
|
171
|
+
severity: "MEDIUM",
|
|
172
|
+
explanation: `The derived output \`${signal.name}\` is produced by arithmetic without an obvious range check on its operands or result.`
|
|
173
|
+
};
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
if (isPrimaryRangeCategory(category)) {
|
|
177
|
+
const inputTarget = signal.kind === "input";
|
|
178
|
+
return {
|
|
179
|
+
shouldReport: true,
|
|
180
|
+
category,
|
|
181
|
+
confidence: inputTarget ? "HIGH" : "MEDIUM",
|
|
182
|
+
severity: inputTarget ? "HIGH" : "MEDIUM",
|
|
183
|
+
explanation: `The ${category} signal \`${signal.name}\` is used without an obvious Num2Bits, RangeCheck, comparator, decomposition, or AliasCheck pattern.`
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
return {
|
|
187
|
+
shouldReport: false,
|
|
188
|
+
category,
|
|
189
|
+
confidence: "LOW",
|
|
190
|
+
severity: "INFO",
|
|
191
|
+
explanation: `The signal \`${signal.name}\` does not match a primary bounded-integer range-check target.`
|
|
192
|
+
};
|
|
193
|
+
}
|
|
194
|
+
//# sourceMappingURL=range-check-classifier.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"range-check-classifier.js","sourceRoot":"","sources":["../../src/core/range-check-classifier.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,cAAc,EAAE,MAAM,qCAAqC,CAAC;AAqBrE,MAAM,wBAAwB,GAAG,IAAI,GAAG,CAAqB;IAC3D,QAAQ;IACR,SAAS;IACT,KAAK;IACL,OAAO;IACP,OAAO;IACP,WAAW;IACX,UAAU;IACV,OAAO;IACP,MAAM;IACN,MAAM;CACP,CAAC,CAAC;AAEH,MAAM,sBAAsB,GAAG,wDAAwD,CAAC;AAUxF,SAAS,YAAY,CAAC,CAAU,EAAE,CAAU;IAC1C,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC;QAAE,OAAO,IAAI,CAAC;IAC1B,OAAO,CAAC,KAAK,CAAC,CAAC;AACjB,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,IAAY,EAAE,IAAgB;IACrE,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;IAEjC,IAAI,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,WAAW,CAAC;IAChD,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,MAAM,CAAC;IACtC,IAAI,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,YAAY,CAAC;IAClD,IAAI,gEAAgE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,IAAI,KAAK,OAAO,CAAC,EAAE,CAAC;QACjI,OAAO,MAAM,CAAC;IAChB,CAAC;IACD,IAAI,IAAI,KAAK,QAAQ,IAAI,yEAAyE,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;QAC/G,OAAO,SAAS,CAAC;IACnB,CAAC;IACD,IAAI,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,UAAU,CAAC;IAErD,IAAI,KAAK,KAAK,QAAQ,IAAI,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,QAAQ,CAAC;IAC1E,IAAI,KAAK,KAAK,SAAS,IAAI,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,SAAS,CAAC;IAC7E,IAAI,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC;IACnD,IAAI,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,OAAO,CAAC;IAC5C,IAAI,iCAAiC,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,OAAO,CAAC;IAClE,IAAI,sBAAsB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,WAAW,CAAC;IAC3D,IAAI,sBAAsB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,UAAU,CAAC;IAC1D,IAAI,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,OAAO,CAAC;IAC5C,IAAI,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,MAAM,CAAC;IACrD,IAAI,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC;QAAE,OAAO,MAAM,CAAC;IAEpD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,QAA4B;IAC7D,OAAO,QAAQ,KAAK,MAAM,IAAI,QAAQ,KAAK,YAAY,IAAI,QAAQ,KAAK,MAAM,IAAI,QAAQ,KAAK,WAAW,CAAC;AAC7G,CAAC;AAED,MAAM,UAAU,sBAAsB,CAAC,QAA4B;IACjE,OAAO,wBAAwB,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;AAChD,CAAC;AAED,SAAS,eAAe,CAAC,SAAiB,EAAE,UAAkB;IAC5D,OAAO,cAAc,CAAC,SAAS,CAAC,KAAK,cAAc,CAAC,UAAU,CAAC,CAAC;AAClE,CAAC;AAED,MAAM,UAAU,qBAAqB,CAAC,MAAyB,EAAE,OAAqB;IACpF,OAAO,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,UAAU,EAAE,EAAE;QACzE,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;QACvE,IAAI,CAAC,KAAK;YAAE,OAAO,KAAK,CAAC;QACzB,MAAM,SAAS,GAAG,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACzF,OAAO,OAAO,CAAC,SAAS,IAAI,sBAAsB,CAAC,IAAI,CAAC,SAAS,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC,CAAC;IACzF,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,sBAAsB,CAAC,MAAyB,EAAE,OAAqB;IACrF,MAAM,UAAU,GAAG,OAAO,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,YAAY,CAAC,GAAG,CAAC,YAAY,EAAE,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IAEpI,OAAO,UAAU,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;QAC7B,MAAM,OAAO,GAAG,CAAC,GAAG,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QACzD,IAAI,mGAAmG,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YACtH,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,cAAc,GAAG,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QACzD,IAAI,CAAC,cAAc;YAAE,OAAO,KAAK,CAAC;QAElC,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC;YAAE,OAAO,IAAI,CAAC;QACjE,IAAI,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC;YAAE,OAAO,IAAI,CAAC;QAEhE,OAAO,KAAK,CAAC;IACf,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,UAAU,sBAAsB,CAAC,MAAyB,EAAE,OAAqB;IACrF,MAAM,WAAW,GAAG,OAAO,CAAC,KAAK;SAC9B,oBAAoB,CAAC,MAAM,CAAC,IAAI,CAAC;SACjC,MAAM,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,YAAY,CAAC,UAAU,CAAC,YAAY,EAAE,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IAEtF,OAAO,WAAW,CAAC,IAAI,CAAC,CAAC,UAAU,EAAE,EAAE;QACrC,MAAM,QAAQ,GAAG,UAAU,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,eAAe,CAAC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;QAClG,MAAM,eAAe,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,sBAAsB,CAAC,wBAAwB,CAAC,cAAc,CAAC,OAAO,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC;QAEzI,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QAC9C,OAAO,eAAe,CAAC,KAAK,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IACvG,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CAAC,MAAyB,EAAE,OAAqB;IAC1E,IAAI,MAAM,CAAC,IAAI,KAAK,QAAQ;QAAE,OAAO,KAAK,CAAC;IAC3C,OAAO,OAAO,CAAC,KAAK;SACjB,oBAAoB,CAAC,MAAM,CAAC,IAAI,CAAC;SACjC,MAAM,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,YAAY,CAAC,UAAU,CAAC,YAAY,EAAE,MAAM,CAAC,YAAY,CAAC,CAAC;SAClF,IAAI,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC;AAC1D,CAAC;AAED,MAAM,UAAU,oBAAoB,CAAC,MAAyB,EAAE,OAAqB;IACnF,MAAM,QAAQ,GAAG,wBAAwB,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;IACxE,MAAM,YAAY,GAAG,MAAM,CAAC,YAAY,CAAC;IAEzC,IAAI,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,EAAE,YAAY,CAAC,EAAE,CAAC;QAC3D,OAAO;YACL,YAAY,EAAE,KAAK;YACnB,QAAQ;YACR,UAAU,EAAE,MAAM;YAClB,QAAQ,EAAE,MAAM;YAChB,WAAW,EAAE,oDAAoD;SAClE,CAAC;IACJ,CAAC;IAED,IAAI,OAAO,CAAC,KAAK,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,YAAY,CAAC,GAAG,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC3H,OAAO;YACL,YAAY,EAAE,KAAK;YACnB,QAAQ;YACR,UAAU,EAAE,KAAK;YACjB,QAAQ,EAAE,MAAM;YAChB,WAAW,EAAE,gEAAgE;SAC9E,CAAC;IACJ,CAAC;IAED,MAAM,UAAU,GAAG,qBAAqB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC1D,MAAM,iBAAiB,GAAG,sBAAsB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAClE,MAAM,aAAa,GAAG,QAAQ,KAAK,SAAS,IAAI,kBAAkB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACpF,MAAM,eAAe,GAAG,sBAAsB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAEhE,IAAI,UAAU,IAAI,kBAAkB,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC/C,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACvB,OAAO;gBACL,YAAY,EAAE,KAAK;gBACnB,QAAQ;gBACR,UAAU,EAAE,KAAK;gBACjB,QAAQ,EAAE,MAAM;gBAChB,WAAW,EAAE,OAAO,QAAQ,aAAa,MAAM,CAAC,IAAI,8DAA8D;aACnH,CAAC;QACJ,CAAC;QAED,OAAO;YACL,YAAY,EAAE,IAAI;YAClB,QAAQ;YACR,UAAU,EAAE,KAAK;YACjB,QAAQ,EAAE,KAAK;YACf,WAAW,EAAE,OAAO,QAAQ,aAAa,MAAM,CAAC,IAAI,+JAA+J;SACpN,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,KAAK,SAAS,IAAI,aAAa,EAAE,CAAC;QAC5C,IAAI,eAAe,EAAE,CAAC;YACpB,OAAO;gBACL,YAAY,EAAE,KAAK;gBACnB,QAAQ,EAAE,SAAS;gBACnB,UAAU,EAAE,KAAK;gBACjB,QAAQ,EAAE,MAAM;gBAChB,WAAW,EAAE,wBAAwB,MAAM,CAAC,IAAI,8DAA8D;aAC/G,CAAC;QACJ,CAAC;QAED,IAAI,MAAM,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,OAAO;gBACL,YAAY,EAAE,IAAI;gBAClB,QAAQ,EAAE,SAAS;gBACnB,UAAU,EAAE,KAAK;gBACjB,QAAQ,EAAE,QAAQ;gBAClB,WAAW,EAAE,wBAAwB,MAAM,CAAC,IAAI,wFAAwF;aACzI,CAAC;QACJ,CAAC;IACH,CAAC;IAED,IAAI,sBAAsB,CAAC,QAAQ,CAAC,EAAE,CAAC;QACrC,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,KAAK,OAAO,CAAC;QAC5C,OAAO;YACL,YAAY,EAAE,IAAI;YAClB,QAAQ;YACR,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ;YAC3C,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ;YACzC,WAAW,EAAE,OAAO,QAAQ,aAAa,MAAM,CAAC,IAAI,uGAAuG;SAC5J,CAAC;IACJ,CAAC;IAED,OAAO;QACL,YAAY,EAAE,KAAK;QACnB,QAAQ;QACR,UAAU,EAAE,KAAK;QACjB,QAAQ,EAAE,MAAM;QAChB,WAAW,EAAE,gBAAgB,MAAM,CAAC,IAAI,iEAAiE;KAC1G,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import type { AuditContext, Issue, Rule } from "../types.js";
|
|
2
|
+
export declare class RuleEngine {
|
|
3
|
+
private readonly rules;
|
|
4
|
+
constructor(rules: Rule[]);
|
|
5
|
+
run(context: AuditContext): {
|
|
6
|
+
issues: Issue[];
|
|
7
|
+
rulesExecuted: number;
|
|
8
|
+
};
|
|
9
|
+
private sortAndDeduplicate;
|
|
10
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { compareSeverity, normalizeSeverity } from "./severity.js";
|
|
2
|
+
import { resetIssueCounter } from "./issue-builder.js";
|
|
3
|
+
function configuredSeverity(ruleId, configValue, fallback) {
|
|
4
|
+
return normalizeSeverity(configValue, fallback);
|
|
5
|
+
}
|
|
6
|
+
export class RuleEngine {
|
|
7
|
+
rules;
|
|
8
|
+
constructor(rules) {
|
|
9
|
+
this.rules = rules;
|
|
10
|
+
}
|
|
11
|
+
run(context) {
|
|
12
|
+
resetIssueCounter();
|
|
13
|
+
const issues = [];
|
|
14
|
+
let rulesExecuted = 0;
|
|
15
|
+
for (const rule of this.rules) {
|
|
16
|
+
const setting = context.config.rules[rule.id] ?? context.config.rules[rule.id.replace(/-.*/, "")];
|
|
17
|
+
if (String(setting).toLowerCase() === "off")
|
|
18
|
+
continue;
|
|
19
|
+
rulesExecuted += 1;
|
|
20
|
+
const ruleIssues = rule.analyze(context).map((issue) => ({
|
|
21
|
+
...issue,
|
|
22
|
+
severity: configuredSeverity(rule.id, typeof setting === "string" ? setting : undefined, issue.severity)
|
|
23
|
+
}));
|
|
24
|
+
issues.push(...ruleIssues);
|
|
25
|
+
}
|
|
26
|
+
return { issues: this.sortAndDeduplicate(issues), rulesExecuted };
|
|
27
|
+
}
|
|
28
|
+
sortAndDeduplicate(issues) {
|
|
29
|
+
const seen = new Set();
|
|
30
|
+
const deduped = [];
|
|
31
|
+
for (const issue of issues) {
|
|
32
|
+
const key = `${issue.ruleId}:${issue.file}:${issue.line}:${issue.signalName ?? ""}:${issue.snippet ?? ""}`;
|
|
33
|
+
if (seen.has(key))
|
|
34
|
+
continue;
|
|
35
|
+
seen.add(key);
|
|
36
|
+
deduped.push(issue);
|
|
37
|
+
}
|
|
38
|
+
return deduped.sort((a, b) => compareSeverity(a.severity, b.severity) || a.file.localeCompare(b.file) || a.line - b.line);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
//# sourceMappingURL=rule-engine.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"rule-engine.js","sourceRoot":"","sources":["../../src/core/rule-engine.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,eAAe,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AACnE,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AAEvD,SAAS,kBAAkB,CAAC,MAAc,EAAE,WAA+B,EAAE,QAAkB;IAC7F,OAAO,iBAAiB,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;AAClD,CAAC;AAED,MAAM,OAAO,UAAU;IACQ;IAA7B,YAA6B,KAAa;QAAb,UAAK,GAAL,KAAK,CAAQ;IAAG,CAAC;IAE9C,GAAG,CAAC,OAAqB;QACvB,iBAAiB,EAAE,CAAC;QACpB,MAAM,MAAM,GAAY,EAAE,CAAC;QAC3B,IAAI,aAAa,GAAG,CAAC,CAAC;QAEtB,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YAC9B,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,CAAC;YAClG,IAAI,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,KAAK,KAAK;gBAAE,SAAS;YACtD,aAAa,IAAI,CAAC,CAAC;YACnB,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBACvD,GAAG,KAAK;gBACR,QAAQ,EAAE,kBAAkB,CAAC,IAAI,CAAC,EAAE,EAAE,OAAO,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,KAAK,CAAC,QAAQ,CAAC;aACzG,CAAC,CAAC,CAAC;YACJ,MAAM,CAAC,IAAI,CAAC,GAAG,UAAU,CAAC,CAAC;QAC7B,CAAC;QAED,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,EAAE,aAAa,EAAE,CAAC;IACpE,CAAC;IAEO,kBAAkB,CAAC,MAAe;QACxC,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;QAC/B,MAAM,OAAO,GAAY,EAAE,CAAC;QAC5B,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,MAAM,GAAG,GAAG,GAAG,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,IAAI,IAAI,KAAK,CAAC,IAAI,IAAI,KAAK,CAAC,UAAU,IAAI,EAAE,IAAI,KAAK,CAAC,OAAO,IAAI,EAAE,EAAE,CAAC;YAC3G,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;gBAAE,SAAS;YAC5B,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YACd,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtB,CAAC;QACD,OAAO,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,eAAe,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC;IAC5H,CAAC;CACF"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { Severity } from "../types.js";
|
|
2
|
+
export declare const severityOrder: Record<Severity, number>;
|
|
3
|
+
export declare const severities: Severity[];
|
|
4
|
+
export declare function normalizeSeverity(value: string | undefined, fallback: Severity): Severity;
|
|
5
|
+
export declare function isAtOrAbove(severity: Severity, threshold: Severity): boolean;
|
|
6
|
+
export declare function compareSeverity(a: Severity, b: Severity): number;
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
export const severityOrder = {
|
|
2
|
+
CRITICAL: 5,
|
|
3
|
+
HIGH: 4,
|
|
4
|
+
MEDIUM: 3,
|
|
5
|
+
LOW: 2,
|
|
6
|
+
INFO: 1
|
|
7
|
+
};
|
|
8
|
+
export const severities = ["CRITICAL", "HIGH", "MEDIUM", "LOW", "INFO"];
|
|
9
|
+
export function normalizeSeverity(value, fallback) {
|
|
10
|
+
if (!value)
|
|
11
|
+
return fallback;
|
|
12
|
+
const upper = value.toUpperCase();
|
|
13
|
+
return severities.includes(upper) ? upper : fallback;
|
|
14
|
+
}
|
|
15
|
+
export function isAtOrAbove(severity, threshold) {
|
|
16
|
+
return severityOrder[severity] >= severityOrder[threshold];
|
|
17
|
+
}
|
|
18
|
+
export function compareSeverity(a, b) {
|
|
19
|
+
return severityOrder[b] - severityOrder[a];
|
|
20
|
+
}
|
|
21
|
+
//# sourceMappingURL=severity.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"severity.js","sourceRoot":"","sources":["../../src/core/severity.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,MAAM,aAAa,GAA6B;IACrD,QAAQ,EAAE,CAAC;IACX,IAAI,EAAE,CAAC;IACP,MAAM,EAAE,CAAC;IACT,GAAG,EAAE,CAAC;IACN,IAAI,EAAE,CAAC;CACR,CAAC;AAEF,MAAM,CAAC,MAAM,UAAU,GAAe,CAAC,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;AAEpF,MAAM,UAAU,iBAAiB,CAAC,KAAyB,EAAE,QAAkB;IAC7E,IAAI,CAAC,KAAK;QAAE,OAAO,QAAQ,CAAC;IAC5B,MAAM,KAAK,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC;IAClC,OAAO,UAAU,CAAC,QAAQ,CAAC,KAAiB,CAAC,CAAC,CAAC,CAAE,KAAkB,CAAC,CAAC,CAAC,QAAQ,CAAC;AACjF,CAAC;AAED,MAAM,UAAU,WAAW,CAAC,QAAkB,EAAE,SAAmB;IACjE,OAAO,aAAa,CAAC,QAAQ,CAAC,IAAI,aAAa,CAAC,SAAS,CAAC,CAAC;AAC7D,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,CAAW,EAAE,CAAW;IACtD,OAAO,aAAa,CAAC,CAAC,CAAC,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC;AAC7C,CAAC"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export function getLine(source, line) {
|
|
2
|
+
return source.split(/\r?\n/)[line - 1]?.trimEnd() ?? "";
|
|
3
|
+
}
|
|
4
|
+
export function columnOf(lineText, needle) {
|
|
5
|
+
const index = lineText.indexOf(needle);
|
|
6
|
+
return index >= 0 ? index + 1 : 1;
|
|
7
|
+
}
|
|
8
|
+
export function lineStartOffsets(source) {
|
|
9
|
+
const starts = [0];
|
|
10
|
+
for (let i = 0; i < source.length; i += 1) {
|
|
11
|
+
if (source[i] === "\n")
|
|
12
|
+
starts.push(i + 1);
|
|
13
|
+
}
|
|
14
|
+
return starts;
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=source-map.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"source-map.js","sourceRoot":"","sources":["../../src/core/source-map.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,OAAO,CAAC,MAAc,EAAE,IAAY;IAClD,OAAO,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;AAC1D,CAAC;AAED,MAAM,UAAU,QAAQ,CAAC,QAAgB,EAAE,MAAc;IACvD,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IACvC,OAAO,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AACpC,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,MAAc;IAC7C,MAAM,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC;IACnB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;QAC1C,IAAI,MAAM,CAAC,CAAC,CAAC,KAAK,IAAI;YAAE,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IAC7C,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC"}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
export function stripCircomComments(source) {
|
|
2
|
+
let output = "";
|
|
3
|
+
const comments = [];
|
|
4
|
+
let line = 1;
|
|
5
|
+
let i = 0;
|
|
6
|
+
while (i < source.length) {
|
|
7
|
+
const ch = source[i];
|
|
8
|
+
const next = source[i + 1];
|
|
9
|
+
if (ch === "/" && next === "/") {
|
|
10
|
+
const startLine = line;
|
|
11
|
+
let text = "";
|
|
12
|
+
while (i < source.length && source[i] !== "\n") {
|
|
13
|
+
text += source[i];
|
|
14
|
+
output += " ";
|
|
15
|
+
i += 1;
|
|
16
|
+
}
|
|
17
|
+
comments.push({ line: startLine, text });
|
|
18
|
+
continue;
|
|
19
|
+
}
|
|
20
|
+
if (ch === "/" && next === "*") {
|
|
21
|
+
const startLine = line;
|
|
22
|
+
let text = "";
|
|
23
|
+
while (i < source.length) {
|
|
24
|
+
const current = source[i];
|
|
25
|
+
const following = source[i + 1];
|
|
26
|
+
text += current;
|
|
27
|
+
if (current === "\n") {
|
|
28
|
+
output += "\n";
|
|
29
|
+
line += 1;
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
output += " ";
|
|
33
|
+
}
|
|
34
|
+
i += 1;
|
|
35
|
+
if (current === "*" && following === "/") {
|
|
36
|
+
text += following;
|
|
37
|
+
output += " ";
|
|
38
|
+
i += 1;
|
|
39
|
+
break;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
comments.push({ line: startLine, text });
|
|
43
|
+
continue;
|
|
44
|
+
}
|
|
45
|
+
output += ch;
|
|
46
|
+
if (ch === "\n")
|
|
47
|
+
line += 1;
|
|
48
|
+
i += 1;
|
|
49
|
+
}
|
|
50
|
+
return { source: output, comments };
|
|
51
|
+
}
|
|
52
|
+
//# sourceMappingURL=circom-comments.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"circom-comments.js","sourceRoot":"","sources":["../../../src/frontends/circom/circom-comments.ts"],"names":[],"mappings":"AAKA,MAAM,UAAU,mBAAmB,CAAC,MAAc;IAChD,IAAI,MAAM,GAAG,EAAE,CAAC;IAChB,MAAM,QAAQ,GAA0C,EAAE,CAAC;IAC3D,IAAI,IAAI,GAAG,CAAC,CAAC;IACb,IAAI,CAAC,GAAG,CAAC,CAAC;IAEV,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;QACzB,MAAM,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACrB,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QAE3B,IAAI,EAAE,KAAK,GAAG,IAAI,IAAI,KAAK,GAAG,EAAE,CAAC;YAC/B,MAAM,SAAS,GAAG,IAAI,CAAC;YACvB,IAAI,IAAI,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;gBAC/C,IAAI,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC;gBAClB,MAAM,IAAI,GAAG,CAAC;gBACd,CAAC,IAAI,CAAC,CAAC;YACT,CAAC;YACD,QAAQ,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YACzC,SAAS;QACX,CAAC;QAED,IAAI,EAAE,KAAK,GAAG,IAAI,IAAI,KAAK,GAAG,EAAE,CAAC;YAC/B,MAAM,SAAS,GAAG,IAAI,CAAC;YACvB,IAAI,IAAI,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;gBACzB,MAAM,OAAO,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;gBAC1B,MAAM,SAAS,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;gBAChC,IAAI,IAAI,OAAO,CAAC;gBAChB,IAAI,OAAO,KAAK,IAAI,EAAE,CAAC;oBACrB,MAAM,IAAI,IAAI,CAAC;oBACf,IAAI,IAAI,CAAC,CAAC;gBACZ,CAAC;qBAAM,CAAC;oBACN,MAAM,IAAI,GAAG,CAAC;gBAChB,CAAC;gBACD,CAAC,IAAI,CAAC,CAAC;gBACP,IAAI,OAAO,KAAK,GAAG,IAAI,SAAS,KAAK,GAAG,EAAE,CAAC;oBACzC,IAAI,IAAI,SAAS,CAAC;oBAClB,MAAM,IAAI,GAAG,CAAC;oBACd,CAAC,IAAI,CAAC,CAAC;oBACP,MAAM;gBACR,CAAC;YACH,CAAC;YACD,QAAQ,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YACzC,SAAS;QACX,CAAC;QAED,MAAM,IAAI,EAAE,CAAC;QACb,IAAI,EAAE,KAAK,IAAI;YAAE,IAAI,IAAI,CAAC,CAAC;QAC3B,CAAC,IAAI,CAAC,CAAC;IACT,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;AACtC,CAAC"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export function buildCircuitIR(files) {
|
|
2
|
+
return {
|
|
3
|
+
files,
|
|
4
|
+
signals: files.flatMap((file) => file.signals),
|
|
5
|
+
components: files.flatMap((file) => file.components),
|
|
6
|
+
assignments: files.flatMap((file) => file.assignments),
|
|
7
|
+
constraints: files.flatMap((file) => file.constraints),
|
|
8
|
+
assertions: files.flatMap((file) => file.assertions),
|
|
9
|
+
parserWarnings: files.flatMap((file) => file.parserWarnings)
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=circom-ir-builder.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"circom-ir-builder.js","sourceRoot":"","sources":["../../../src/frontends/circom/circom-ir-builder.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,cAAc,CAAC,KAA0B;IACvD,OAAO;QACL,KAAK;QACL,OAAO,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC;QAC9C,UAAU,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC;QACpD,WAAW,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC;QACtD,WAAW,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC;QACtD,UAAU,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC;QACpD,cAAc,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC;KAC7D,CAAC;AACJ,CAAC"}
|