cclaw-cli 7.7.1 → 8.1.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/README.md +210 -134
- package/dist/artifact-frontmatter.d.ts +51 -0
- package/dist/artifact-frontmatter.js +131 -0
- package/dist/artifact-paths.d.ts +7 -27
- package/dist/artifact-paths.js +20 -249
- package/dist/cancel.d.ts +16 -0
- package/dist/cancel.js +66 -0
- package/dist/cli.d.ts +2 -27
- package/dist/cli.js +90 -508
- package/dist/compound.d.ts +26 -0
- package/dist/compound.js +96 -0
- package/dist/config.d.ts +14 -51
- package/dist/config.js +23 -359
- package/dist/constants.d.ts +11 -18
- package/dist/constants.js +19 -106
- package/dist/content/antipatterns.d.ts +1 -0
- package/dist/content/antipatterns.js +109 -0
- package/dist/content/artifact-templates.d.ts +10 -0
- package/dist/content/artifact-templates.js +550 -0
- package/dist/content/cancel-command.d.ts +2 -2
- package/dist/content/cancel-command.js +25 -17
- package/dist/content/core-agents.d.ts +9 -233
- package/dist/content/core-agents.js +39 -768
- package/dist/content/decision-protocol.d.ts +1 -12
- package/dist/content/decision-protocol.js +27 -20
- package/dist/content/examples.d.ts +8 -42
- package/dist/content/examples.js +293 -425
- package/dist/content/idea-command.d.ts +2 -0
- package/dist/content/idea-command.js +38 -0
- package/dist/content/iron-laws.d.ts +4 -138
- package/dist/content/iron-laws.js +18 -197
- package/dist/content/meta-skill.d.ts +1 -3
- package/dist/content/meta-skill.js +57 -134
- package/dist/content/node-hooks.d.ts +12 -8
- package/dist/content/node-hooks.js +188 -838
- package/dist/content/recovery.d.ts +8 -0
- package/dist/content/recovery.js +179 -0
- package/dist/content/reference-patterns.d.ts +4 -13
- package/dist/content/reference-patterns.js +260 -389
- package/dist/content/research-playbooks.d.ts +8 -8
- package/dist/content/research-playbooks.js +108 -121
- package/dist/content/review-loop.d.ts +6 -192
- package/dist/content/review-loop.js +29 -731
- package/dist/content/skills.d.ts +8 -38
- package/dist/content/skills.js +681 -732
- package/dist/content/specialist-prompts/architect.d.ts +1 -0
- package/dist/content/specialist-prompts/architect.js +225 -0
- package/dist/content/specialist-prompts/brainstormer.d.ts +1 -0
- package/dist/content/specialist-prompts/brainstormer.js +168 -0
- package/dist/content/specialist-prompts/index.d.ts +2 -0
- package/dist/content/specialist-prompts/index.js +14 -0
- package/dist/content/specialist-prompts/planner.d.ts +1 -0
- package/dist/content/specialist-prompts/planner.js +182 -0
- package/dist/content/specialist-prompts/reviewer.d.ts +1 -0
- package/dist/content/specialist-prompts/reviewer.js +193 -0
- package/dist/content/specialist-prompts/security-reviewer.d.ts +1 -0
- package/dist/content/specialist-prompts/security-reviewer.js +133 -0
- package/dist/content/specialist-prompts/slice-builder.d.ts +1 -0
- package/dist/content/specialist-prompts/slice-builder.js +232 -0
- package/dist/content/stage-playbooks.d.ts +8 -0
- package/dist/content/stage-playbooks.js +404 -0
- package/dist/content/start-command.d.ts +2 -12
- package/dist/content/start-command.js +221 -207
- package/dist/flow-state.d.ts +21 -178
- package/dist/flow-state.js +67 -170
- package/dist/fs-utils.d.ts +6 -26
- package/dist/fs-utils.js +29 -162
- package/dist/gitignore.d.ts +2 -1
- package/dist/gitignore.js +51 -34
- package/dist/harness-detect.d.ts +10 -0
- package/dist/harness-detect.js +29 -0
- package/dist/install.d.ts +27 -15
- package/dist/install.js +230 -1342
- package/dist/knowledge-store.d.ts +19 -163
- package/dist/knowledge-store.js +56 -590
- package/dist/logger.d.ts +8 -3
- package/dist/logger.js +13 -4
- package/dist/orchestrator-routing.d.ts +29 -0
- package/dist/orchestrator-routing.js +156 -0
- package/dist/run-persistence.d.ts +7 -118
- package/dist/run-persistence.js +29 -845
- package/dist/runtime/run-hook.entry.d.ts +1 -3
- package/dist/runtime/run-hook.entry.js +19 -4
- package/dist/runtime/run-hook.mjs +13 -1024
- package/dist/types.d.ts +25 -261
- package/dist/types.js +8 -36
- package/package.json +6 -3
- package/dist/artifact-linter/brainstorm.d.ts +0 -2
- package/dist/artifact-linter/brainstorm.js +0 -353
- package/dist/artifact-linter/design.d.ts +0 -18
- package/dist/artifact-linter/design.js +0 -444
- package/dist/artifact-linter/findings-dedup.d.ts +0 -56
- package/dist/artifact-linter/findings-dedup.js +0 -232
- package/dist/artifact-linter/plan.d.ts +0 -2
- package/dist/artifact-linter/plan.js +0 -826
- package/dist/artifact-linter/review-army.d.ts +0 -49
- package/dist/artifact-linter/review-army.js +0 -520
- package/dist/artifact-linter/review.d.ts +0 -2
- package/dist/artifact-linter/review.js +0 -113
- package/dist/artifact-linter/scope.d.ts +0 -2
- package/dist/artifact-linter/scope.js +0 -158
- package/dist/artifact-linter/shared.d.ts +0 -637
- package/dist/artifact-linter/shared.js +0 -2163
- package/dist/artifact-linter/ship.d.ts +0 -2
- package/dist/artifact-linter/ship.js +0 -250
- package/dist/artifact-linter/spec.d.ts +0 -2
- package/dist/artifact-linter/spec.js +0 -176
- package/dist/artifact-linter/tdd.d.ts +0 -118
- package/dist/artifact-linter/tdd.js +0 -1404
- package/dist/artifact-linter.d.ts +0 -15
- package/dist/artifact-linter.js +0 -517
- package/dist/codex-feature-flag.d.ts +0 -58
- package/dist/codex-feature-flag.js +0 -193
- package/dist/content/closeout-guidance.d.ts +0 -14
- package/dist/content/closeout-guidance.js +0 -44
- package/dist/content/diff-command.d.ts +0 -1
- package/dist/content/diff-command.js +0 -43
- package/dist/content/harness-doc.d.ts +0 -1
- package/dist/content/harness-doc.js +0 -65
- package/dist/content/hook-events.d.ts +0 -9
- package/dist/content/hook-events.js +0 -23
- package/dist/content/hook-manifest.d.ts +0 -81
- package/dist/content/hook-manifest.js +0 -156
- package/dist/content/hooks.d.ts +0 -11
- package/dist/content/hooks.js +0 -1972
- package/dist/content/idea.d.ts +0 -60
- package/dist/content/idea.js +0 -416
- package/dist/content/language-policy.d.ts +0 -2
- package/dist/content/language-policy.js +0 -13
- package/dist/content/learnings.d.ts +0 -6
- package/dist/content/learnings.js +0 -141
- package/dist/content/observe.d.ts +0 -19
- package/dist/content/observe.js +0 -86
- package/dist/content/opencode-plugin.d.ts +0 -1
- package/dist/content/opencode-plugin.js +0 -635
- package/dist/content/review-prompts.d.ts +0 -1
- package/dist/content/review-prompts.js +0 -104
- package/dist/content/runtime-shared-snippets.d.ts +0 -8
- package/dist/content/runtime-shared-snippets.js +0 -80
- package/dist/content/session-hooks.d.ts +0 -7
- package/dist/content/session-hooks.js +0 -107
- package/dist/content/skills-elicitation.d.ts +0 -1
- package/dist/content/skills-elicitation.js +0 -167
- package/dist/content/stage-command.d.ts +0 -2
- package/dist/content/stage-command.js +0 -17
- package/dist/content/stage-schema.d.ts +0 -117
- package/dist/content/stage-schema.js +0 -955
- package/dist/content/stages/_lint-metadata/index.d.ts +0 -2
- package/dist/content/stages/_lint-metadata/index.js +0 -97
- package/dist/content/stages/brainstorm.d.ts +0 -2
- package/dist/content/stages/brainstorm.js +0 -184
- package/dist/content/stages/design.d.ts +0 -2
- package/dist/content/stages/design.js +0 -288
- package/dist/content/stages/index.d.ts +0 -8
- package/dist/content/stages/index.js +0 -11
- package/dist/content/stages/plan.d.ts +0 -2
- package/dist/content/stages/plan.js +0 -191
- package/dist/content/stages/review.d.ts +0 -2
- package/dist/content/stages/review.js +0 -240
- package/dist/content/stages/schema-types.d.ts +0 -203
- package/dist/content/stages/schema-types.js +0 -1
- package/dist/content/stages/scope.d.ts +0 -2
- package/dist/content/stages/scope.js +0 -254
- package/dist/content/stages/ship.d.ts +0 -2
- package/dist/content/stages/ship.js +0 -159
- package/dist/content/stages/spec.d.ts +0 -2
- package/dist/content/stages/spec.js +0 -170
- package/dist/content/stages/tdd.d.ts +0 -4
- package/dist/content/stages/tdd.js +0 -273
- package/dist/content/state-contracts.d.ts +0 -1
- package/dist/content/state-contracts.js +0 -63
- package/dist/content/status-command.d.ts +0 -4
- package/dist/content/status-command.js +0 -109
- package/dist/content/subagent-context-skills.d.ts +0 -4
- package/dist/content/subagent-context-skills.js +0 -279
- package/dist/content/subagents.d.ts +0 -3
- package/dist/content/subagents.js +0 -997
- package/dist/content/templates.d.ts +0 -26
- package/dist/content/templates.js +0 -1692
- package/dist/content/track-render-context.d.ts +0 -18
- package/dist/content/track-render-context.js +0 -53
- package/dist/content/tree-command.d.ts +0 -1
- package/dist/content/tree-command.js +0 -64
- package/dist/content/utility-skills.d.ts +0 -30
- package/dist/content/utility-skills.js +0 -160
- package/dist/content/view-command.d.ts +0 -2
- package/dist/content/view-command.js +0 -92
- package/dist/delegation.d.ts +0 -649
- package/dist/delegation.js +0 -1539
- package/dist/early-loop.d.ts +0 -70
- package/dist/early-loop.js +0 -302
- package/dist/execution-topology.d.ts +0 -44
- package/dist/execution-topology.js +0 -95
- package/dist/gate-evidence.d.ts +0 -85
- package/dist/gate-evidence.js +0 -631
- package/dist/harness-adapters.d.ts +0 -151
- package/dist/harness-adapters.js +0 -756
- package/dist/harness-selection.d.ts +0 -31
- package/dist/harness-selection.js +0 -214
- package/dist/hook-schema.d.ts +0 -6
- package/dist/hook-schema.js +0 -114
- package/dist/hook-schemas/claude-hooks.v1.json +0 -10
- package/dist/hook-schemas/codex-hooks.v1.json +0 -10
- package/dist/hook-schemas/cursor-hooks.v1.json +0 -13
- package/dist/init-detect.d.ts +0 -2
- package/dist/init-detect.js +0 -50
- package/dist/internal/advance-stage/advance.d.ts +0 -89
- package/dist/internal/advance-stage/advance.js +0 -655
- package/dist/internal/advance-stage/cancel-run.d.ts +0 -8
- package/dist/internal/advance-stage/cancel-run.js +0 -19
- package/dist/internal/advance-stage/flow-state-coercion.d.ts +0 -3
- package/dist/internal/advance-stage/flow-state-coercion.js +0 -81
- package/dist/internal/advance-stage/helpers.d.ts +0 -14
- package/dist/internal/advance-stage/helpers.js +0 -145
- package/dist/internal/advance-stage/hook.d.ts +0 -8
- package/dist/internal/advance-stage/hook.js +0 -40
- package/dist/internal/advance-stage/parsers.d.ts +0 -72
- package/dist/internal/advance-stage/parsers.js +0 -357
- package/dist/internal/advance-stage/proactive-delegation-trace.d.ts +0 -24
- package/dist/internal/advance-stage/proactive-delegation-trace.js +0 -56
- package/dist/internal/advance-stage/review-loop.d.ts +0 -16
- package/dist/internal/advance-stage/review-loop.js +0 -199
- package/dist/internal/advance-stage/rewind.d.ts +0 -14
- package/dist/internal/advance-stage/rewind.js +0 -108
- package/dist/internal/advance-stage/start-flow.d.ts +0 -13
- package/dist/internal/advance-stage/start-flow.js +0 -241
- package/dist/internal/advance-stage/verify.d.ts +0 -21
- package/dist/internal/advance-stage/verify.js +0 -185
- package/dist/internal/advance-stage.d.ts +0 -7
- package/dist/internal/advance-stage.js +0 -138
- package/dist/internal/cohesion-contract-stub.d.ts +0 -24
- package/dist/internal/cohesion-contract-stub.js +0 -148
- package/dist/internal/compound-readiness.d.ts +0 -23
- package/dist/internal/compound-readiness.js +0 -102
- package/dist/internal/detect-public-api-changes.d.ts +0 -5
- package/dist/internal/detect-public-api-changes.js +0 -45
- package/dist/internal/detect-supply-chain-changes.d.ts +0 -6
- package/dist/internal/detect-supply-chain-changes.js +0 -138
- package/dist/internal/early-loop-status.d.ts +0 -7
- package/dist/internal/early-loop-status.js +0 -93
- package/dist/internal/envelope-validate.d.ts +0 -7
- package/dist/internal/envelope-validate.js +0 -66
- package/dist/internal/flow-state-repair.d.ts +0 -20
- package/dist/internal/flow-state-repair.js +0 -104
- package/dist/internal/plan-split-waves.d.ts +0 -190
- package/dist/internal/plan-split-waves.js +0 -764
- package/dist/internal/runtime-integrity.d.ts +0 -7
- package/dist/internal/runtime-integrity.js +0 -268
- package/dist/internal/slice-commit.d.ts +0 -7
- package/dist/internal/slice-commit.js +0 -619
- package/dist/internal/tdd-loop-status.d.ts +0 -14
- package/dist/internal/tdd-loop-status.js +0 -68
- package/dist/internal/tdd-red-evidence.d.ts +0 -7
- package/dist/internal/tdd-red-evidence.js +0 -153
- package/dist/internal/waiver-grant.d.ts +0 -62
- package/dist/internal/waiver-grant.js +0 -294
- package/dist/internal/wave-status.d.ts +0 -74
- package/dist/internal/wave-status.js +0 -506
- package/dist/managed-resources.d.ts +0 -53
- package/dist/managed-resources.js +0 -313
- package/dist/policy.d.ts +0 -10
- package/dist/policy.js +0 -167
- package/dist/retro-gate.d.ts +0 -9
- package/dist/retro-gate.js +0 -47
- package/dist/run-archive.d.ts +0 -61
- package/dist/run-archive.js +0 -391
- package/dist/runs.d.ts +0 -2
- package/dist/runs.js +0 -2
- package/dist/stack-detection.d.ts +0 -116
- package/dist/stack-detection.js +0 -489
- package/dist/streaming/event-stream.d.ts +0 -31
- package/dist/streaming/event-stream.js +0 -114
- package/dist/tdd-cycle.d.ts +0 -107
- package/dist/tdd-cycle.js +0 -289
- package/dist/tdd-verification-evidence.d.ts +0 -17
- package/dist/tdd-verification-evidence.js +0 -122
- package/dist/track-heuristics.d.ts +0 -27
- package/dist/track-heuristics.js +0 -154
- package/dist/util/slice-id.d.ts +0 -58
- package/dist/util/slice-id.js +0 -89
- package/dist/worktree-manager.d.ts +0 -20
- package/dist/worktree-manager.js +0 -108
|
@@ -1,232 +0,0 @@
|
|
|
1
|
-
import { createHash } from "node:crypto";
|
|
2
|
-
import fs from "node:fs/promises";
|
|
3
|
-
import path from "node:path";
|
|
4
|
-
import { RUNTIME_ROOT } from "../constants.js";
|
|
5
|
-
import { ensureDir, exists, withDirectoryLock, writeFileSafe } from "../fs-utils.js";
|
|
6
|
-
/**
|
|
7
|
-
* linter-dedup cache. The linter persists a per-stage
|
|
8
|
-
* fingerprint of each finding between runs so authors can tell at a
|
|
9
|
-
* glance what's `new`, `repeat`, or `resolved` relative to the last run.
|
|
10
|
-
*
|
|
11
|
-
* Fingerprint = `sha256(stage | rule | normalizedDetail).slice(0, 8)`.
|
|
12
|
-
* Details are normalized to stabilize the digest: whitespace collapsed,
|
|
13
|
-
* run-ids/hashes/timestamps replaced with placeholders, and enumeration
|
|
14
|
-
* counts (e.g. "3 approach detail card(s)") replaced with `<N>`.
|
|
15
|
-
*
|
|
16
|
-
* The cache is intentionally bounded by `MAX_PER_STAGE` so a noisy stage
|
|
17
|
-
* can't grow the sidecar without bound. When the active run trims the
|
|
18
|
-
* cache we drop the oldest `firstSeenAt` entries first.
|
|
19
|
-
*/
|
|
20
|
-
const FINDINGS_CACHE_REL_PATH = `${RUNTIME_ROOT}/.linter-findings.json`;
|
|
21
|
-
const FINDINGS_CACHE_LOCK_REL_PATH = `${RUNTIME_ROOT}/.linter-findings.json.lock`;
|
|
22
|
-
export const FINDINGS_CACHE_SCHEMA_VERSION = 1;
|
|
23
|
-
const MAX_PER_STAGE = 200;
|
|
24
|
-
function cachePath(projectRoot) {
|
|
25
|
-
return path.join(projectRoot, FINDINGS_CACHE_REL_PATH);
|
|
26
|
-
}
|
|
27
|
-
function cacheLockPath(projectRoot) {
|
|
28
|
-
return path.join(projectRoot, FINDINGS_CACHE_LOCK_REL_PATH);
|
|
29
|
-
}
|
|
30
|
-
function emptyStageCache() {
|
|
31
|
-
return { findings: [], lastRunAt: null };
|
|
32
|
-
}
|
|
33
|
-
function emptyCacheFile() {
|
|
34
|
-
return { schemaVersion: FINDINGS_CACHE_SCHEMA_VERSION, stages: {} };
|
|
35
|
-
}
|
|
36
|
-
/**
|
|
37
|
-
* Normalize a finding detail string so volatile tokens (run IDs,
|
|
38
|
-
* timestamps, counts, hex hashes, temp paths) don't cause a finding
|
|
39
|
-
* to appear "new" on every invocation.
|
|
40
|
-
*/
|
|
41
|
-
export function normalizeFindingDetail(detail) {
|
|
42
|
-
if (typeof detail !== "string" || detail.length === 0)
|
|
43
|
-
return "";
|
|
44
|
-
let normalized = detail;
|
|
45
|
-
normalized = normalized.replace(/\brun-[a-z0-9-]+\b/giu, "run-<id>");
|
|
46
|
-
normalized = normalized.replace(/\b[0-9a-f]{16,}\b/giu, "<hex>");
|
|
47
|
-
normalized = normalized.replace(/\b\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(?:\.\d+)?Z?\b/gu, "<ts>");
|
|
48
|
-
normalized = normalized.replace(/\b\d{10,}\b/gu, "<n>");
|
|
49
|
-
normalized = normalized.replace(/\b\d+\b/gu, "<n>");
|
|
50
|
-
normalized = normalized.replace(/[ \t]+/gu, " ");
|
|
51
|
-
normalized = normalized.replace(/\r?\n/gu, " ");
|
|
52
|
-
return normalized.trim().toLowerCase();
|
|
53
|
-
}
|
|
54
|
-
export function fingerprintFinding(stage, finding) {
|
|
55
|
-
const payload = `${stage}|${finding.rule.trim()}|${normalizeFindingDetail(finding.details)}`;
|
|
56
|
-
return createHash("sha256").update(payload, "utf8").digest("hex").slice(0, 8);
|
|
57
|
-
}
|
|
58
|
-
async function readCacheFile(projectRoot) {
|
|
59
|
-
const filePath = cachePath(projectRoot);
|
|
60
|
-
if (!(await exists(filePath)))
|
|
61
|
-
return emptyCacheFile();
|
|
62
|
-
let raw;
|
|
63
|
-
try {
|
|
64
|
-
raw = await fs.readFile(filePath, "utf8");
|
|
65
|
-
}
|
|
66
|
-
catch {
|
|
67
|
-
return emptyCacheFile();
|
|
68
|
-
}
|
|
69
|
-
let parsed;
|
|
70
|
-
try {
|
|
71
|
-
parsed = JSON.parse(raw);
|
|
72
|
-
}
|
|
73
|
-
catch {
|
|
74
|
-
return emptyCacheFile();
|
|
75
|
-
}
|
|
76
|
-
if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
|
|
77
|
-
return emptyCacheFile();
|
|
78
|
-
}
|
|
79
|
-
const typed = parsed;
|
|
80
|
-
const stages = (typed.stages ?? {});
|
|
81
|
-
const next = emptyCacheFile();
|
|
82
|
-
for (const [stageKey, value] of Object.entries(stages)) {
|
|
83
|
-
if (!value || typeof value !== "object" || Array.isArray(value))
|
|
84
|
-
continue;
|
|
85
|
-
const rawStage = value;
|
|
86
|
-
const findingsRaw = Array.isArray(rawStage.findings) ? rawStage.findings : [];
|
|
87
|
-
const findings = [];
|
|
88
|
-
for (const row of findingsRaw) {
|
|
89
|
-
if (!row || typeof row !== "object" || Array.isArray(row))
|
|
90
|
-
continue;
|
|
91
|
-
const r = row;
|
|
92
|
-
const fingerprint = typeof r.fingerprint === "string" ? r.fingerprint : "";
|
|
93
|
-
const rule = typeof r.rule === "string" ? r.rule : "";
|
|
94
|
-
const section = typeof r.section === "string" ? r.section : "";
|
|
95
|
-
const firstSeenAt = typeof r.firstSeenAt === "string" ? r.firstSeenAt : "";
|
|
96
|
-
const lastSeenAt = typeof r.lastSeenAt === "string" ? r.lastSeenAt : "";
|
|
97
|
-
const runCount = typeof r.runCount === "number" && Number.isFinite(r.runCount)
|
|
98
|
-
? Math.max(1, Math.floor(r.runCount))
|
|
99
|
-
: 1;
|
|
100
|
-
if (fingerprint.length === 0 || rule.length === 0)
|
|
101
|
-
continue;
|
|
102
|
-
findings.push({ fingerprint, rule, section, firstSeenAt, lastSeenAt, runCount });
|
|
103
|
-
}
|
|
104
|
-
next.stages[stageKey] = {
|
|
105
|
-
findings,
|
|
106
|
-
lastRunAt: typeof rawStage.lastRunAt === "string" ? rawStage.lastRunAt : null
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
return next;
|
|
110
|
-
}
|
|
111
|
-
async function writeCacheFile(projectRoot, cache) {
|
|
112
|
-
await ensureDir(path.dirname(cachePath(projectRoot)));
|
|
113
|
-
await writeFileSafe(cachePath(projectRoot), `${JSON.stringify(cache, null, 2)}\n`, { mode: 0o600 });
|
|
114
|
-
}
|
|
115
|
-
/**
|
|
116
|
-
* Classify each emitted finding as `new`, `repeat:N`, or `resolved`
|
|
117
|
-
* relative to the cached sidecar for this stage. Persists the updated
|
|
118
|
-
* fingerprint set under a directory lock so concurrent lint runs for
|
|
119
|
-
* the same project don't clobber each other.
|
|
120
|
-
*
|
|
121
|
-
* The returned `header` is a short human string intended for inclusion
|
|
122
|
-
* above the linter output; it's stable across runs when findings
|
|
123
|
-
* repeat. Empty string when there is nothing meaningful to report
|
|
124
|
-
* (no findings and no carry-over state).
|
|
125
|
-
*/
|
|
126
|
-
export async function classifyAndPersistFindings(projectRoot, stage, findings, options = {}) {
|
|
127
|
-
const nowIso = (options.now ?? new Date()).toISOString();
|
|
128
|
-
return withDirectoryLock(cacheLockPath(projectRoot), async () => {
|
|
129
|
-
const cache = await readCacheFile(projectRoot);
|
|
130
|
-
const previous = cache.stages[stage] ?? emptyStageCache();
|
|
131
|
-
const previousByFingerprint = new Map();
|
|
132
|
-
for (const entry of previous.findings) {
|
|
133
|
-
previousByFingerprint.set(entry.fingerprint, entry);
|
|
134
|
-
}
|
|
135
|
-
const currentFingerprints = new Set();
|
|
136
|
-
const classified = [];
|
|
137
|
-
const nextFindings = [];
|
|
138
|
-
let newCount = 0;
|
|
139
|
-
let repeatCount = 0;
|
|
140
|
-
for (const finding of findings) {
|
|
141
|
-
const fingerprint = fingerprintFinding(stage, finding);
|
|
142
|
-
currentFingerprints.add(fingerprint);
|
|
143
|
-
const prior = previousByFingerprint.get(fingerprint);
|
|
144
|
-
if (prior) {
|
|
145
|
-
const nextEntry = {
|
|
146
|
-
fingerprint,
|
|
147
|
-
rule: finding.rule,
|
|
148
|
-
section: finding.section,
|
|
149
|
-
firstSeenAt: prior.firstSeenAt || nowIso,
|
|
150
|
-
lastSeenAt: nowIso,
|
|
151
|
-
runCount: prior.runCount + 1
|
|
152
|
-
};
|
|
153
|
-
nextFindings.push(nextEntry);
|
|
154
|
-
repeatCount += 1;
|
|
155
|
-
classified.push({
|
|
156
|
-
finding,
|
|
157
|
-
fingerprint,
|
|
158
|
-
status: { kind: "repeat", count: nextEntry.runCount }
|
|
159
|
-
});
|
|
160
|
-
continue;
|
|
161
|
-
}
|
|
162
|
-
const nextEntry = {
|
|
163
|
-
fingerprint,
|
|
164
|
-
rule: finding.rule,
|
|
165
|
-
section: finding.section,
|
|
166
|
-
firstSeenAt: nowIso,
|
|
167
|
-
lastSeenAt: nowIso,
|
|
168
|
-
runCount: 1
|
|
169
|
-
};
|
|
170
|
-
nextFindings.push(nextEntry);
|
|
171
|
-
newCount += 1;
|
|
172
|
-
classified.push({
|
|
173
|
-
finding,
|
|
174
|
-
fingerprint,
|
|
175
|
-
status: { kind: "new" }
|
|
176
|
-
});
|
|
177
|
-
}
|
|
178
|
-
const resolved = [];
|
|
179
|
-
for (const entry of previous.findings) {
|
|
180
|
-
if (currentFingerprints.has(entry.fingerprint))
|
|
181
|
-
continue;
|
|
182
|
-
resolved.push({
|
|
183
|
-
fingerprint: entry.fingerprint,
|
|
184
|
-
rule: entry.rule,
|
|
185
|
-
lastSeenAt: entry.lastSeenAt
|
|
186
|
-
});
|
|
187
|
-
}
|
|
188
|
-
nextFindings.sort((a, b) => {
|
|
189
|
-
const aTime = Date.parse(a.firstSeenAt);
|
|
190
|
-
const bTime = Date.parse(b.firstSeenAt);
|
|
191
|
-
return Number.isFinite(aTime) && Number.isFinite(bTime) ? aTime - bTime : 0;
|
|
192
|
-
});
|
|
193
|
-
const trimmed = nextFindings.length > MAX_PER_STAGE
|
|
194
|
-
? nextFindings.slice(nextFindings.length - MAX_PER_STAGE)
|
|
195
|
-
: nextFindings;
|
|
196
|
-
cache.stages[stage] = {
|
|
197
|
-
findings: trimmed,
|
|
198
|
-
lastRunAt: nowIso
|
|
199
|
-
};
|
|
200
|
-
await writeCacheFile(projectRoot, cache);
|
|
201
|
-
const summary = {
|
|
202
|
-
newCount,
|
|
203
|
-
repeatCount,
|
|
204
|
-
resolvedCount: resolved.length,
|
|
205
|
-
resolved
|
|
206
|
-
};
|
|
207
|
-
const header = buildDedupHeader(stage, summary);
|
|
208
|
-
return { classified, summary, header };
|
|
209
|
-
});
|
|
210
|
-
}
|
|
211
|
-
export function buildDedupHeader(stage, summary) {
|
|
212
|
-
const parts = [];
|
|
213
|
-
if (summary.newCount > 0)
|
|
214
|
-
parts.push(`${summary.newCount} new`);
|
|
215
|
-
if (summary.repeatCount > 0)
|
|
216
|
-
parts.push(`${summary.repeatCount} repeat`);
|
|
217
|
-
if (summary.resolvedCount > 0)
|
|
218
|
-
parts.push(`${summary.resolvedCount} resolved`);
|
|
219
|
-
if (parts.length === 0)
|
|
220
|
-
return "";
|
|
221
|
-
return `linter findings (stage=${stage}): ${parts.join(", ")}.`;
|
|
222
|
-
}
|
|
223
|
-
export function formatFindingStatusTag(status) {
|
|
224
|
-
if (status.kind === "new")
|
|
225
|
-
return "[new]";
|
|
226
|
-
if (status.kind === "resolved")
|
|
227
|
-
return "[resolved]";
|
|
228
|
-
return `[repeat:${status.count}]`;
|
|
229
|
-
}
|
|
230
|
-
export function findingsDedupCachePathFor(projectRoot) {
|
|
231
|
-
return cachePath(projectRoot);
|
|
232
|
-
}
|