@trailmark/core 0.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.
Files changed (94) hide show
  1. package/dist/baseline.d.ts +21 -0
  2. package/dist/baseline.d.ts.map +1 -0
  3. package/dist/baseline.js +187 -0
  4. package/dist/baseline.js.map +1 -0
  5. package/dist/diff.d.ts +3 -0
  6. package/dist/diff.d.ts.map +1 -0
  7. package/dist/diff.js +120 -0
  8. package/dist/diff.js.map +1 -0
  9. package/dist/fingerprint.d.ts +8 -0
  10. package/dist/fingerprint.d.ts.map +1 -0
  11. package/dist/fingerprint.js +85 -0
  12. package/dist/fingerprint.js.map +1 -0
  13. package/dist/hash.d.ts +4 -0
  14. package/dist/hash.d.ts.map +1 -0
  15. package/dist/hash.js +29 -0
  16. package/dist/hash.js.map +1 -0
  17. package/dist/impact.d.ts +5 -0
  18. package/dist/impact.d.ts.map +1 -0
  19. package/dist/impact.js +27 -0
  20. package/dist/impact.js.map +1 -0
  21. package/dist/index.d.ts +12 -0
  22. package/dist/index.d.ts.map +1 -0
  23. package/dist/index.js +28 -0
  24. package/dist/index.js.map +1 -0
  25. package/dist/issue.d.ts +10 -0
  26. package/dist/issue.d.ts.map +1 -0
  27. package/dist/issue.js +40 -0
  28. package/dist/issue.js.map +1 -0
  29. package/dist/normalize.d.ts +3 -0
  30. package/dist/normalize.d.ts.map +1 -0
  31. package/dist/normalize.js +59 -0
  32. package/dist/normalize.js.map +1 -0
  33. package/dist/report/buildReport.d.ts +3 -0
  34. package/dist/report/buildReport.d.ts.map +1 -0
  35. package/dist/report/buildReport.js +223 -0
  36. package/dist/report/buildReport.js.map +1 -0
  37. package/dist/report/formatConsole.d.ts +3 -0
  38. package/dist/report/formatConsole.d.ts.map +1 -0
  39. package/dist/report/formatConsole.js +112 -0
  40. package/dist/report/formatConsole.js.map +1 -0
  41. package/dist/report/index.d.ts +4 -0
  42. package/dist/report/index.d.ts.map +1 -0
  43. package/dist/report/index.js +20 -0
  44. package/dist/report/index.js.map +1 -0
  45. package/dist/report/sort.d.ts +9 -0
  46. package/dist/report/sort.d.ts.map +1 -0
  47. package/dist/report/sort.js +20 -0
  48. package/dist/report/sort.js.map +1 -0
  49. package/dist/report/types.d.ts +85 -0
  50. package/dist/report/types.d.ts.map +1 -0
  51. package/dist/report/types.js +3 -0
  52. package/dist/report/types.js.map +1 -0
  53. package/dist/run.d.ts +8 -0
  54. package/dist/run.d.ts.map +1 -0
  55. package/dist/run.js +20 -0
  56. package/dist/run.js.map +1 -0
  57. package/dist/scope.d.ts +5 -0
  58. package/dist/scope.d.ts.map +1 -0
  59. package/dist/scope.js +23 -0
  60. package/dist/scope.js.map +1 -0
  61. package/dist/types.d.ts +150 -0
  62. package/dist/types.d.ts.map +1 -0
  63. package/dist/types.js +3 -0
  64. package/dist/types.js.map +1 -0
  65. package/dist/url.d.ts +3 -0
  66. package/dist/url.d.ts.map +1 -0
  67. package/dist/url.js +82 -0
  68. package/dist/url.js.map +1 -0
  69. package/package.json +23 -0
  70. package/src/baseline.ts +217 -0
  71. package/src/diff.ts +150 -0
  72. package/src/fingerprint.ts +107 -0
  73. package/src/hash.ts +29 -0
  74. package/src/impact.ts +27 -0
  75. package/src/index.ts +11 -0
  76. package/src/issue.ts +49 -0
  77. package/src/normalize.ts +66 -0
  78. package/src/report/buildReport.ts +276 -0
  79. package/src/report/formatConsole.ts +137 -0
  80. package/src/report/index.ts +3 -0
  81. package/src/report/sort.ts +28 -0
  82. package/src/report/types.ts +101 -0
  83. package/src/run.ts +22 -0
  84. package/src/scope.ts +25 -0
  85. package/src/types.ts +167 -0
  86. package/src/url.ts +102 -0
  87. package/test/baseline.test.ts +151 -0
  88. package/test/diff.test.ts +113 -0
  89. package/test/fingerprint.test.ts +41 -0
  90. package/test/issue.test.ts +79 -0
  91. package/test/report.build.test.ts +191 -0
  92. package/test/report.console.test.ts +102 -0
  93. package/test/url.test.ts +25 -0
  94. package/tsconfig.json +8 -0
@@ -0,0 +1,21 @@
1
+ import { BaselineFile, TrailmarkRun } from "./types";
2
+ export declare const BASELINE_SCHEMA_VERSION = 1;
3
+ export declare class BaselineFormatError extends Error {
4
+ constructor(message: string);
5
+ }
6
+ export declare function computeConfigHash(config: unknown): string;
7
+ export declare function parseBaselineFile(raw: string): {
8
+ data: null;
9
+ error: true;
10
+ message: string;
11
+ } | {
12
+ error: false;
13
+ data: BaselineFile;
14
+ message?: undefined;
15
+ };
16
+ export declare function buildBaselineFromRun(run: TrailmarkRun, options: {
17
+ createdAt?: string;
18
+ configHash: string;
19
+ previousBaseline?: BaselineFile;
20
+ }): BaselineFile;
21
+ //# sourceMappingURL=baseline.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"baseline.d.ts","sourceRoot":"","sources":["../src/baseline.ts"],"names":[],"mappings":"AAIA,OAAO,EAAiB,YAAY,EAAmB,YAAY,EAAE,MAAM,SAAS,CAAA;AAGpF,eAAO,MAAM,uBAAuB,IAAI,CAAA;AAExC,qBAAa,mBAAoB,SAAQ,KAAK;gBAChC,OAAO,EAAE,MAAM;CAI5B;AAED,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,OAAO,GAAG,MAAM,CAEzD;AAsFD,wBAAgB,iBAAiB,CAAC,GAAG,EAAE,MAAM;;;;;;;;EAsC5C;AAED,wBAAgB,oBAAoB,CAClC,GAAG,EAAE,YAAY,EACjB,OAAO,EAAE;IACP,SAAS,CAAC,EAAE,MAAM,CAAA;IAClB,UAAU,EAAE,MAAM,CAAA;IAClB,gBAAgB,CAAC,EAAE,YAAY,CAAA;CAChC,GACA,YAAY,CAiEd"}
@@ -0,0 +1,187 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.BaselineFormatError = exports.BASELINE_SCHEMA_VERSION = void 0;
4
+ exports.computeConfigHash = computeConfigHash;
5
+ exports.parseBaselineFile = parseBaselineFile;
6
+ exports.buildBaselineFromRun = buildBaselineFromRun;
7
+ const hash_1 = require("./hash");
8
+ const impact_1 = require("./impact");
9
+ const run_1 = require("./run");
10
+ const scope_1 = require("./scope");
11
+ const zod_1 = require("zod");
12
+ exports.BASELINE_SCHEMA_VERSION = 1;
13
+ class BaselineFormatError extends Error {
14
+ constructor(message) {
15
+ super(message);
16
+ this.name = "BaselineFormatError";
17
+ }
18
+ }
19
+ exports.BaselineFormatError = BaselineFormatError;
20
+ function computeConfigHash(config) {
21
+ return (0, hash_1.hashObject)(config);
22
+ }
23
+ function isRecord(value) {
24
+ return typeof value === "object" && value !== null && !Array.isArray(value);
25
+ }
26
+ function normalizeTags(tags) {
27
+ return Array.from(new Set(tags)).sort((a, b) => a.localeCompare(b));
28
+ }
29
+ const baselineEntrySchema = zod_1.z.object({
30
+ issueId: zod_1.z.string(),
31
+ scopeKey: zod_1.z.string().optional(),
32
+ projectName: zod_1.z.string().optional(),
33
+ relaxedIssueId: zod_1.z.string(),
34
+ ruleId: zod_1.z.string(),
35
+ impact: zod_1.z
36
+ .enum(impact_1.IMPACT_ORDER.filter((value) => value !== "unknown"))
37
+ .nullable()
38
+ .optional(),
39
+ pageKey: zod_1.z.string(),
40
+ pageUrl: zod_1.z.string(),
41
+ targetHash: zod_1.z.string(),
42
+ selectorHint: zod_1.z.string().optional(),
43
+ sampleTarget: zod_1.z.string().optional(),
44
+ help: zod_1.z.string().optional(),
45
+ helpUrl: zod_1.z.string().optional(),
46
+ tags: zod_1.z.array(zod_1.z.string()).transform((tags) => normalizeTags(tags)),
47
+ failureHint: zod_1.z.string().optional(),
48
+ occurrences: zod_1.z.number().finite().min(0),
49
+ firstSeenAt: zod_1.z.string(),
50
+ lastSeenAt: zod_1.z.string(),
51
+ });
52
+ const baselineFileSchema = zod_1.z.object({
53
+ version: zod_1.z.literal(exports.BASELINE_SCHEMA_VERSION),
54
+ createdAt: zod_1.z.string(),
55
+ tool: zod_1.z.object({
56
+ name: zod_1.z.string(),
57
+ version: zod_1.z.string(),
58
+ scanner: zod_1.z.string(),
59
+ }),
60
+ configHash: zod_1.z.string(),
61
+ findings: zod_1.z.record(zod_1.z.string(), baselineEntrySchema),
62
+ });
63
+ function formatPath(path) {
64
+ if (path.length === 0) {
65
+ return "baseline";
66
+ }
67
+ return path
68
+ .map((segment) => (typeof segment === "number" ? `[${String(segment)}]` : segment))
69
+ .join(".");
70
+ }
71
+ function comparableEntry(entry) {
72
+ return {
73
+ issueId: entry.issueId,
74
+ scopeKey: entry.scopeKey,
75
+ projectName: entry.projectName,
76
+ relaxedIssueId: entry.relaxedIssueId,
77
+ ruleId: entry.ruleId,
78
+ impact: entry.impact,
79
+ pageKey: entry.pageKey,
80
+ pageUrl: entry.pageUrl,
81
+ targetHash: entry.targetHash,
82
+ selectorHint: entry.selectorHint,
83
+ sampleTarget: entry.sampleTarget,
84
+ help: entry.help,
85
+ helpUrl: entry.helpUrl,
86
+ tags: entry.tags,
87
+ failureHint: entry.failureHint,
88
+ occurrences: entry.occurrences,
89
+ };
90
+ }
91
+ function isUnchangedEntry(previous, next) {
92
+ return (0, hash_1.hashObject)(comparableEntry(previous)) === (0, hash_1.hashObject)(comparableEntry(next));
93
+ }
94
+ function parseBaselineFile(raw) {
95
+ let parsed;
96
+ try {
97
+ parsed = JSON.parse(raw);
98
+ }
99
+ catch {
100
+ return {
101
+ data: null,
102
+ error: true,
103
+ message: "trailmark: baseline file is not a valid JSON.",
104
+ };
105
+ }
106
+ if (isRecord(parsed) &&
107
+ typeof parsed.version === "number" &&
108
+ parsed.version !== exports.BASELINE_SCHEMA_VERSION) {
109
+ return {
110
+ data: null,
111
+ error: true,
112
+ message: `trailmark: unsupported baseline version ${parsed.version}. Expected ${exports.BASELINE_SCHEMA_VERSION}.`,
113
+ };
114
+ }
115
+ const result = baselineFileSchema.safeParse(parsed);
116
+ if (!result.success) {
117
+ const issue = result.error.issues[0];
118
+ return {
119
+ data: null,
120
+ error: true,
121
+ message: `trailmark: invalid baseline format at "${formatPath(issue.path)}": ${issue.message}`,
122
+ };
123
+ }
124
+ return {
125
+ error: false,
126
+ data: result.data,
127
+ };
128
+ }
129
+ function buildBaselineFromRun(run, options) {
130
+ const createdAt = options.createdAt ?? options.previousBaseline?.createdAt ?? new Date().toISOString();
131
+ const previousFindings = options.previousBaseline?.findings ?? {};
132
+ const findings = {};
133
+ const sortedFindings = [...run.findings].sort((a, b) => {
134
+ const aKey = (0, scope_1.scopedIssueKey)(a.issueId, (0, scope_1.scopeKeyFromMeta)(a.meta));
135
+ const bKey = (0, scope_1.scopedIssueKey)(b.issueId, (0, scope_1.scopeKeyFromMeta)(b.meta));
136
+ return aKey.localeCompare(bKey);
137
+ });
138
+ for (const finding of sortedFindings) {
139
+ const scopeKey = (0, scope_1.scopeKeyFromMeta)(finding.meta);
140
+ const entryKey = (0, scope_1.scopedIssueKey)(finding.issueId, scopeKey);
141
+ const existing = findings[entryKey];
142
+ if (existing) {
143
+ existing.occurrences += 1;
144
+ continue;
145
+ }
146
+ const candidate = {
147
+ issueId: finding.issueId,
148
+ scopeKey,
149
+ projectName: finding.meta?.projectName,
150
+ relaxedIssueId: finding.relaxedIssueId,
151
+ ruleId: finding.ruleId,
152
+ impact: finding.impact,
153
+ pageKey: finding.pageKey,
154
+ pageUrl: finding.pageUrl,
155
+ targetHash: finding.targetHash,
156
+ selectorHint: finding.selectorHint,
157
+ sampleTarget: finding.target[0],
158
+ help: finding.help,
159
+ helpUrl: finding.helpUrl,
160
+ tags: normalizeTags([...finding.tags]),
161
+ failureHint: finding.failureHint,
162
+ occurrences: 1,
163
+ firstSeenAt: createdAt,
164
+ lastSeenAt: createdAt,
165
+ };
166
+ const previous = previousFindings[entryKey];
167
+ if (previous) {
168
+ candidate.firstSeenAt = previous.firstSeenAt;
169
+ if (isUnchangedEntry(previous, candidate)) {
170
+ candidate.lastSeenAt = previous.lastSeenAt;
171
+ }
172
+ }
173
+ findings[entryKey] = candidate;
174
+ }
175
+ const orderedFindings = {};
176
+ for (const key of Object.keys(findings).sort((a, b) => a.localeCompare(b))) {
177
+ orderedFindings[key] = findings[key];
178
+ }
179
+ return {
180
+ version: exports.BASELINE_SCHEMA_VERSION,
181
+ createdAt,
182
+ tool: run.tool ?? run_1.DEFAULT_TOOL_INFO,
183
+ configHash: options.configHash,
184
+ findings: orderedFindings,
185
+ };
186
+ }
187
+ //# sourceMappingURL=baseline.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"baseline.js","sourceRoot":"","sources":["../src/baseline.ts"],"names":[],"mappings":"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"}
package/dist/diff.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ import { BaselineFile, DiffOptions, DiffResult, TrailmarkRun } from "./types";
2
+ export declare function diffRuns(baseline: BaselineFile, current: TrailmarkRun, options?: DiffOptions): DiffResult;
3
+ //# sourceMappingURL=diff.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"diff.d.ts","sourceRoot":"","sources":["../src/diff.ts"],"names":[],"mappings":"AAAA,OAAO,EAEL,YAAY,EACZ,WAAW,EACX,UAAU,EAEV,YAAY,EACb,MAAM,SAAS,CAAA;AA4ChB,wBAAgB,QAAQ,CACtB,QAAQ,EAAE,YAAY,EACtB,OAAO,EAAE,YAAY,EACrB,OAAO,GAAE,WAAgB,GACxB,UAAU,CA8FZ"}
package/dist/diff.js ADDED
@@ -0,0 +1,120 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.diffRuns = diffRuns;
4
+ const scope_1 = require("./scope");
5
+ const impact_1 = require("./impact");
6
+ function dedupeFindings(findings) {
7
+ const unique = new Map();
8
+ for (const finding of findings) {
9
+ const key = (0, scope_1.scopedIssueKey)(finding.issueId, (0, scope_1.scopeKeyFromMeta)(finding.meta));
10
+ if (!unique.has(key)) {
11
+ unique.set(key, finding);
12
+ }
13
+ }
14
+ return Array.from(unique.entries())
15
+ .sort(([left], [right]) => left.localeCompare(right))
16
+ .map(([, finding]) => finding);
17
+ }
18
+ function makeRelaxedMap(baselineEntries) {
19
+ const map = new Map();
20
+ for (const entry of baselineEntries) {
21
+ const key = (0, scope_1.scopedIssueKey)(entry.relaxedIssueId, (0, scope_1.scopeKeyFromBaselineEntry)(entry));
22
+ const list = map.get(key) ?? [];
23
+ list.push(entry);
24
+ map.set(key, list);
25
+ }
26
+ return map;
27
+ }
28
+ function makeExactKey(entry) {
29
+ return (0, scope_1.scopedIssueKey)(entry.issueId, (0, scope_1.scopeKeyFromBaselineEntry)(entry));
30
+ }
31
+ function pickUnmatched(list, matchedBaselineIds) {
32
+ if (!list || list.length === 0) {
33
+ return undefined;
34
+ }
35
+ return list.find((entry) => !matchedBaselineIds.has(makeExactKey(entry)));
36
+ }
37
+ function diffRuns(baseline, current, options = {}) {
38
+ const debug = options.onDebug;
39
+ const failOnImpact = new Set((options.failOnImpact ?? ["serious", "critical"]).map((value) => value.toLowerCase()));
40
+ const baselineEntries = Object.values(baseline.findings);
41
+ const baselineByIssueId = new Map(baselineEntries.map((entry) => [makeExactKey(entry), entry]));
42
+ const baselineByRelaxed = makeRelaxedMap(baselineEntries);
43
+ const newFindings = [];
44
+ const existing = [];
45
+ const matchedBaselineIds = new Set();
46
+ for (const finding of dedupeFindings(current.findings)) {
47
+ const scopeKey = (0, scope_1.scopeKeyFromMeta)(finding.meta);
48
+ const scopedFindingIssueKey = (0, scope_1.scopedIssueKey)(finding.issueId, scopeKey);
49
+ const exact = baselineByIssueId.get(scopedFindingIssueKey);
50
+ if (exact) {
51
+ const baselineKey = makeExactKey(exact);
52
+ existing.push({
53
+ finding,
54
+ matchedBy: "exact",
55
+ baselineIssueId: exact.issueId,
56
+ baselineScopeKey: (0, scope_1.scopeKeyFromBaselineEntry)(exact),
57
+ });
58
+ matchedBaselineIds.add(baselineKey);
59
+ debug?.("exact-match", {
60
+ findingIssueId: finding.issueId,
61
+ findingScopeKey: scopeKey,
62
+ baselineIssueId: exact.issueId,
63
+ baselineScopeKey: (0, scope_1.scopeKeyFromBaselineEntry)(exact),
64
+ });
65
+ continue;
66
+ }
67
+ const relaxedMatches = baselineByRelaxed.get((0, scope_1.scopedIssueKey)(finding.relaxedIssueId, scopeKey));
68
+ const relaxed = pickUnmatched(relaxedMatches, matchedBaselineIds);
69
+ if (relaxed) {
70
+ const baselineKey = makeExactKey(relaxed);
71
+ existing.push({
72
+ finding,
73
+ matchedBy: "relaxed",
74
+ baselineIssueId: relaxed.issueId,
75
+ baselineScopeKey: (0, scope_1.scopeKeyFromBaselineEntry)(relaxed),
76
+ });
77
+ matchedBaselineIds.add(baselineKey);
78
+ debug?.("relaxed-match", {
79
+ findingIssueId: finding.issueId,
80
+ findingScopeKey: scopeKey,
81
+ baselineIssueId: relaxed.issueId,
82
+ baselineScopeKey: (0, scope_1.scopeKeyFromBaselineEntry)(relaxed),
83
+ });
84
+ continue;
85
+ }
86
+ newFindings.push(finding);
87
+ debug?.("new-finding", {
88
+ findingIssueId: finding.issueId,
89
+ findingScopeKey: scopeKey,
90
+ });
91
+ }
92
+ const resolved = baselineEntries.filter((entry) => !matchedBaselineIds.has(makeExactKey(entry)));
93
+ for (const entry of resolved) {
94
+ debug?.("resolved-finding", {
95
+ baselineIssueId: entry.issueId,
96
+ baselineScopeKey: (0, scope_1.scopeKeyFromBaselineEntry)(entry),
97
+ });
98
+ }
99
+ const newByImpact = {};
100
+ for (const finding of newFindings) {
101
+ const key = (0, impact_1.impactKey)(finding.impact);
102
+ newByImpact[key] = (newByImpact[key] ?? 0) + 1;
103
+ }
104
+ const blockingNew = newFindings.filter((finding) => failOnImpact.has((0, impact_1.impactKey)(finding.impact)));
105
+ return {
106
+ newFindings,
107
+ existing,
108
+ resolved,
109
+ blockingNew,
110
+ stats: {
111
+ totalCurrent: dedupeFindings(current.findings).length,
112
+ totalBaseline: baselineEntries.length,
113
+ newCount: newFindings.length,
114
+ existingCount: existing.length,
115
+ resolvedCount: resolved.length,
116
+ newByImpact,
117
+ },
118
+ };
119
+ }
120
+ //# sourceMappingURL=diff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"diff.js","sourceRoot":"","sources":["../src/diff.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import { AxeNodeResult, TargetFingerprintCanonical } from "./types";
2
+ export interface TargetFingerprintResult {
3
+ canonical: TargetFingerprintCanonical;
4
+ targetHash: string;
5
+ selectorHint: string | null;
6
+ }
7
+ export declare function buildTargetFingerprint(node: Pick<AxeNodeResult, "target" | "html">): TargetFingerprintResult;
8
+ //# sourceMappingURL=fingerprint.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fingerprint.d.ts","sourceRoot":"","sources":["../src/fingerprint.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,0BAA0B,EAAE,MAAM,SAAS,CAAA;AAkFnE,MAAM,WAAW,uBAAuB;IACtC,SAAS,EAAE,0BAA0B,CAAA;IACrC,UAAU,EAAE,MAAM,CAAA;IAClB,YAAY,EAAE,MAAM,GAAG,IAAI,CAAA;CAC5B;AAED,wBAAgB,sBAAsB,CACpC,IAAI,EAAE,IAAI,CAAC,aAAa,EAAE,QAAQ,GAAG,MAAM,CAAC,GAC3C,uBAAuB,CAezB"}
@@ -0,0 +1,85 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.buildTargetFingerprint = buildTargetFingerprint;
4
+ const hash_1 = require("./hash");
5
+ const STABLE_ATTR_NAMES = [
6
+ "data-testid",
7
+ "data-test",
8
+ "data-qa",
9
+ "aria-label",
10
+ "aria-labelledby",
11
+ "id",
12
+ "name",
13
+ "role",
14
+ "type",
15
+ "href",
16
+ "for",
17
+ ];
18
+ const STABLE_ATTR_SET = new Set(STABLE_ATTR_NAMES);
19
+ function truncate(value, maxLength = 120) {
20
+ if (value.length <= maxLength) {
21
+ return value;
22
+ }
23
+ return value.slice(0, maxLength);
24
+ }
25
+ function parseTag(html) {
26
+ if (!html) {
27
+ return null;
28
+ }
29
+ const match = html.match(/^\s*<\s*([a-zA-Z0-9:-]+)/);
30
+ return match ? match[1].toLowerCase() : null;
31
+ }
32
+ function parseStableAttrs(html) {
33
+ if (!html) {
34
+ return {};
35
+ }
36
+ const attrs = {};
37
+ const attrRegex = /([a-zA-Z_:][-a-zA-Z0-9_:.]*)(?:\s*=\s*("([^"]*)"|'([^']*)'|([^\s"'=<>`]+)))?/g;
38
+ for (const match of html.matchAll(attrRegex)) {
39
+ const rawName = match[1];
40
+ const name = rawName.toLowerCase();
41
+ if (!STABLE_ATTR_SET.has(name)) {
42
+ continue;
43
+ }
44
+ const value = match[3] ?? match[4] ?? match[5] ?? "";
45
+ attrs[name] = truncate(value.trim());
46
+ }
47
+ return attrs;
48
+ }
49
+ function sanitizeSelectorHint(selector) {
50
+ let output = selector
51
+ .replace(/:nth-[a-z-]+\([^)]*\)/gi, "")
52
+ .replace(/\[(\w[\w-]*)\s*=\s*["']?\d{2,}["']?\s*\]/g, "[$1]")
53
+ .replace(/\b\d{2,}\b/g, "")
54
+ .replace(/\s+/g, " ")
55
+ .trim();
56
+ return truncate(output, 120);
57
+ }
58
+ function selectorHintFromTarget(target) {
59
+ const first = target?.[0];
60
+ if (!first) {
61
+ return null;
62
+ }
63
+ const chunks = first
64
+ .split(/\s+|>|\+|~/)
65
+ .map((part) => part.trim())
66
+ .filter(Boolean);
67
+ const last = chunks.length > 0 ? chunks[chunks.length - 1] : first;
68
+ const hint = sanitizeSelectorHint(last);
69
+ return hint || null;
70
+ }
71
+ function buildTargetFingerprint(node) {
72
+ const selectorHint = selectorHintFromTarget(node.target);
73
+ const stableAttrs = parseStableAttrs(node.html);
74
+ const canonical = {
75
+ tag: parseTag(node.html),
76
+ stableAttrs,
77
+ structuralHint: Object.keys(stableAttrs).length === 0 ? selectorHint : null,
78
+ };
79
+ return {
80
+ canonical,
81
+ selectorHint,
82
+ targetHash: (0, hash_1.hashObject)(canonical),
83
+ };
84
+ }
85
+ //# sourceMappingURL=fingerprint.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fingerprint.js","sourceRoot":"","sources":["../src/fingerprint.ts"],"names":[],"mappings":"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"}
package/dist/hash.d.ts ADDED
@@ -0,0 +1,4 @@
1
+ export declare function stableStringify(value: unknown): string;
2
+ export declare function sha1Hex(input: string): string;
3
+ export declare function hashObject(input: unknown): string;
4
+ //# sourceMappingURL=hash.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hash.d.ts","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":"AAkBA,wBAAgB,eAAe,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAEtD;AAED,wBAAgB,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAE7C;AAED,wBAAgB,UAAU,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAEjD"}
package/dist/hash.js ADDED
@@ -0,0 +1,29 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.stableStringify = stableStringify;
4
+ exports.sha1Hex = sha1Hex;
5
+ exports.hashObject = hashObject;
6
+ const node_crypto_1 = require("node:crypto");
7
+ function sortValue(value) {
8
+ if (Array.isArray(value)) {
9
+ return value.map((item) => sortValue(item));
10
+ }
11
+ if (value && typeof value === "object") {
12
+ const sorted = {};
13
+ for (const key of Object.keys(value).sort()) {
14
+ sorted[key] = sortValue(value[key]);
15
+ }
16
+ return sorted;
17
+ }
18
+ return value;
19
+ }
20
+ function stableStringify(value) {
21
+ return JSON.stringify(sortValue(value));
22
+ }
23
+ function sha1Hex(input) {
24
+ return (0, node_crypto_1.createHash)("sha1").update(input).digest("hex");
25
+ }
26
+ function hashObject(input) {
27
+ return sha1Hex(stableStringify(input));
28
+ }
29
+ //# sourceMappingURL=hash.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hash.js","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":";;AAkBA,0CAEC;AAED,0BAEC;AAED,gCAEC;AA5BD,6CAAwC;AAExC,SAAS,SAAS,CAAC,KAAc;IAC/B,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACzB,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAA;IAC7C,CAAC;IAED,IAAI,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QACvC,MAAM,MAAM,GAA4B,EAAE,CAAA;QAC1C,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,KAAgC,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;YACvE,MAAM,CAAC,GAAG,CAAC,GAAG,SAAS,CAAE,KAAiC,CAAC,GAAG,CAAC,CAAC,CAAA;QAClE,CAAC;QACD,OAAO,MAAM,CAAA;IACf,CAAC;IAED,OAAO,KAAK,CAAA;AACd,CAAC;AAED,SAAgB,eAAe,CAAC,KAAc;IAC5C,OAAO,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAA;AACzC,CAAC;AAED,SAAgB,OAAO,CAAC,KAAa;IACnC,OAAO,IAAA,wBAAU,EAAC,MAAM,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAA;AACvD,CAAC;AAED,SAAgB,UAAU,CAAC,KAAc;IACvC,OAAO,OAAO,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAA;AACxC,CAAC"}
@@ -0,0 +1,5 @@
1
+ import { TrailmarkImpact, TrailmarkImpactKey } from "./types";
2
+ export declare const IMPACT_ORDER: TrailmarkImpactKey[];
3
+ export declare function impactKey(impact: TrailmarkImpact | TrailmarkImpactKey): TrailmarkImpactKey;
4
+ export declare function impactRank(impact: TrailmarkImpact | TrailmarkImpactKey): number;
5
+ //# sourceMappingURL=impact.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"impact.d.ts","sourceRoot":"","sources":["../src/impact.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,kBAAkB,EAAE,MAAM,SAAS,CAAA;AAE7D,eAAO,MAAM,YAAY,EAAE,kBAAkB,EAM5C,CAAA;AAED,wBAAgB,SAAS,CAAC,MAAM,EAAE,eAAe,GAAG,kBAAkB,GAAG,kBAAkB,CAM1F;AAED,wBAAgB,UAAU,CAAC,MAAM,EAAE,eAAe,GAAG,kBAAkB,GAAG,MAAM,CAQ/E"}
package/dist/impact.js ADDED
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.IMPACT_ORDER = void 0;
4
+ exports.impactKey = impactKey;
5
+ exports.impactRank = impactRank;
6
+ exports.IMPACT_ORDER = [
7
+ "critical",
8
+ "serious",
9
+ "moderate",
10
+ "minor",
11
+ "unknown",
12
+ ];
13
+ function impactKey(impact) {
14
+ if (!impact) {
15
+ return "unknown";
16
+ }
17
+ return impact;
18
+ }
19
+ function impactRank(impact) {
20
+ const key = impactKey(impact);
21
+ const index = exports.IMPACT_ORDER.indexOf(key);
22
+ if (index >= 0) {
23
+ return index;
24
+ }
25
+ return exports.IMPACT_ORDER.length + 1;
26
+ }
27
+ //# sourceMappingURL=impact.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"impact.js","sourceRoot":"","sources":["../src/impact.ts"],"names":[],"mappings":";;;AAUA,8BAMC;AAED,gCAQC;AAxBY,QAAA,YAAY,GAAyB;IAChD,UAAU;IACV,SAAS;IACT,UAAU;IACV,OAAO;IACP,SAAS;CACV,CAAA;AAED,SAAgB,SAAS,CAAC,MAA4C;IACpE,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO,SAAS,CAAA;IAClB,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC;AAED,SAAgB,UAAU,CAAC,MAA4C;IACrE,MAAM,GAAG,GAAG,SAAS,CAAC,MAAM,CAAC,CAAA;IAC7B,MAAM,KAAK,GAAG,oBAAY,CAAC,OAAO,CAAC,GAAG,CAAC,CAAA;IACvC,IAAI,KAAK,IAAI,CAAC,EAAE,CAAC;QACf,OAAO,KAAK,CAAA;IACd,CAAC;IAED,OAAO,oBAAY,CAAC,MAAM,GAAG,CAAC,CAAA;AAChC,CAAC"}
@@ -0,0 +1,12 @@
1
+ export * from "./types";
2
+ export * from "./hash";
3
+ export * from "./url";
4
+ export * from "./issue";
5
+ export * from "./fingerprint";
6
+ export * from "./normalize";
7
+ export * from "./run";
8
+ export * from "./baseline";
9
+ export * from "./scope";
10
+ export * from "./diff";
11
+ export * from "./report";
12
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,SAAS,CAAA;AACvB,cAAc,QAAQ,CAAA;AACtB,cAAc,OAAO,CAAA;AACrB,cAAc,SAAS,CAAA;AACvB,cAAc,eAAe,CAAA;AAC7B,cAAc,aAAa,CAAA;AAC3B,cAAc,OAAO,CAAA;AACrB,cAAc,YAAY,CAAA;AAC1B,cAAc,SAAS,CAAA;AACvB,cAAc,QAAQ,CAAA;AACtB,cAAc,UAAU,CAAA"}
package/dist/index.js ADDED
@@ -0,0 +1,28 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./types"), exports);
18
+ __exportStar(require("./hash"), exports);
19
+ __exportStar(require("./url"), exports);
20
+ __exportStar(require("./issue"), exports);
21
+ __exportStar(require("./fingerprint"), exports);
22
+ __exportStar(require("./normalize"), exports);
23
+ __exportStar(require("./run"), exports);
24
+ __exportStar(require("./baseline"), exports);
25
+ __exportStar(require("./scope"), exports);
26
+ __exportStar(require("./diff"), exports);
27
+ __exportStar(require("./report"), exports);
28
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;AAAA,0CAAuB;AACvB,yCAAsB;AACtB,wCAAqB;AACrB,0CAAuB;AACvB,gDAA6B;AAC7B,8CAA2B;AAC3B,wCAAqB;AACrB,6CAA0B;AAC1B,0CAAuB;AACvB,yCAAsB;AACtB,2CAAwB"}
@@ -0,0 +1,10 @@
1
+ export interface IssueIdInput {
2
+ ruleId: string;
3
+ pageKey: string;
4
+ targetHash: string;
5
+ failureHint?: string;
6
+ }
7
+ export declare function normalizeFailureHint(summary: string | undefined): string | undefined;
8
+ export declare function buildRelaxedIssueId(input: Omit<IssueIdInput, "failureHint">): string;
9
+ export declare function buildIssueId(input: IssueIdInput): string;
10
+ //# sourceMappingURL=issue.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"issue.d.ts","sourceRoot":"","sources":["../src/issue.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,YAAY;IAC3B,MAAM,EAAE,MAAM,CAAA;IACd,OAAO,EAAE,MAAM,CAAA;IACf,UAAU,EAAE,MAAM,CAAA;IAClB,WAAW,CAAC,EAAE,MAAM,CAAA;CACrB;AAED,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,MAAM,GAAG,SAAS,GAAG,MAAM,GAAG,SAAS,CAiBpF;AAED,wBAAgB,mBAAmB,CAAC,KAAK,EAAE,IAAI,CAAC,YAAY,EAAE,aAAa,CAAC,GAAG,MAAM,CAMpF;AAED,wBAAgB,YAAY,CAAC,KAAK,EAAE,YAAY,GAAG,MAAM,CAYxD"}