@supernal/interface 1.0.9 → 1.0.12
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/dist/cjs/src/background/navigation/NavigationGraph.js +48 -0
- package/dist/cjs/src/background/navigation/NavigationGraph.js.map +1 -1
- package/dist/cjs/src/browser.js +42 -1
- package/dist/cjs/src/browser.js.map +1 -1
- package/dist/cjs/src/decorators/ContainerHelpers.js +16 -1
- package/dist/cjs/src/decorators/ContainerHelpers.js.map +1 -1
- package/dist/cjs/src/decorators/Tool.js +2 -3
- package/dist/cjs/src/decorators/Tool.js.map +1 -1
- package/dist/cjs/src/testing/graph-tester/core/GraphTester.js +339 -0
- package/dist/cjs/src/testing/graph-tester/core/GraphTester.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/core/TestFunction.js +189 -0
- package/dist/cjs/src/testing/graph-tester/core/TestFunction.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/core/types.js +24 -0
- package/dist/cjs/src/testing/graph-tester/core/types.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/fixtures/index.js +13 -0
- package/dist/cjs/src/testing/graph-tester/fixtures/index.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/fixtures/portPool.js +184 -0
- package/dist/cjs/src/testing/graph-tester/fixtures/portPool.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/index.js +98 -0
- package/dist/cjs/src/testing/graph-tester/index.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/modes/AccessibilityMode.js +230 -0
- package/dist/cjs/src/testing/graph-tester/modes/AccessibilityMode.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/modes/PerformanceMode.js +168 -0
- package/dist/cjs/src/testing/graph-tester/modes/PerformanceMode.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/modes/SEOMode.js +264 -0
- package/dist/cjs/src/testing/graph-tester/modes/SEOMode.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/modes/VisualRegressionMode.js +199 -0
- package/dist/cjs/src/testing/graph-tester/modes/VisualRegressionMode.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/modes/index.js +17 -0
- package/dist/cjs/src/testing/graph-tester/modes/index.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/reporters/HTMLReporter.js +411 -0
- package/dist/cjs/src/testing/graph-tester/reporters/HTMLReporter.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/reporters/JSONReporter.js +127 -0
- package/dist/cjs/src/testing/graph-tester/reporters/JSONReporter.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/reporters/MarkdownReporter.js +169 -0
- package/dist/cjs/src/testing/graph-tester/reporters/MarkdownReporter.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/reporters/UnifiedReporter.js +118 -0
- package/dist/cjs/src/testing/graph-tester/reporters/UnifiedReporter.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/reporters/index.js +17 -0
- package/dist/cjs/src/testing/graph-tester/reporters/index.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js +47 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js +584 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js +376 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/index.js +15 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/index.js.map +1 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/types.js +11 -0
- package/dist/cjs/src/testing/graph-tester/screenshot/types.js.map +1 -0
- package/dist/cjs/src/testing/selectors.js +1 -1
- package/dist/esm/src/background/navigation/INavigationGraph.d.ts +15 -0
- package/dist/esm/src/background/navigation/INavigationGraph.d.ts.map +1 -1
- package/dist/esm/src/background/navigation/NavigationGraph.d.ts +32 -0
- package/dist/esm/src/background/navigation/NavigationGraph.d.ts.map +1 -1
- package/dist/esm/src/background/navigation/NavigationGraph.js +48 -0
- package/dist/esm/src/background/navigation/NavigationGraph.js.map +1 -1
- package/dist/esm/src/browser.d.ts +9 -0
- package/dist/esm/src/browser.d.ts.map +1 -1
- package/dist/esm/src/browser.js +14 -0
- package/dist/esm/src/browser.js.map +1 -1
- package/dist/esm/src/decorators/ContainerHelpers.d.ts +2 -1
- package/dist/esm/src/decorators/ContainerHelpers.d.ts.map +1 -1
- package/dist/esm/src/decorators/ContainerHelpers.js +16 -1
- package/dist/esm/src/decorators/ContainerHelpers.js.map +1 -1
- package/dist/esm/src/decorators/Tool.d.ts.map +1 -1
- package/dist/esm/src/decorators/Tool.js +2 -3
- package/dist/esm/src/decorators/Tool.js.map +1 -1
- package/dist/esm/src/testing/graph-tester/core/GraphTester.d.ts +110 -0
- package/dist/esm/src/testing/graph-tester/core/GraphTester.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/core/GraphTester.js +335 -0
- package/dist/esm/src/testing/graph-tester/core/GraphTester.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/core/TestFunction.d.ts +120 -0
- package/dist/esm/src/testing/graph-tester/core/TestFunction.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/core/TestFunction.js +184 -0
- package/dist/esm/src/testing/graph-tester/core/TestFunction.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/core/types.d.ts +331 -0
- package/dist/esm/src/testing/graph-tester/core/types.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/core/types.js +21 -0
- package/dist/esm/src/testing/graph-tester/core/types.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/fixtures/index.d.ts +8 -0
- package/dist/esm/src/testing/graph-tester/fixtures/index.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/fixtures/index.js +7 -0
- package/dist/esm/src/testing/graph-tester/fixtures/index.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/fixtures/portPool.d.ts +40 -0
- package/dist/esm/src/testing/graph-tester/fixtures/portPool.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/fixtures/portPool.js +147 -0
- package/dist/esm/src/testing/graph-tester/fixtures/portPool.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/index.d.ts +84 -0
- package/dist/esm/src/testing/graph-tester/index.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/index.js +77 -0
- package/dist/esm/src/testing/graph-tester/index.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.d.ts +40 -0
- package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.js +193 -0
- package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.d.ts +42 -0
- package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.js +131 -0
- package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/SEOMode.d.ts +39 -0
- package/dist/esm/src/testing/graph-tester/modes/SEOMode.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/SEOMode.js +227 -0
- package/dist/esm/src/testing/graph-tester/modes/SEOMode.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.d.ts +83 -0
- package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.js +162 -0
- package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/index.d.ts +14 -0
- package/dist/esm/src/testing/graph-tester/modes/index.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/modes/index.js +10 -0
- package/dist/esm/src/testing/graph-tester/modes/index.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.d.ts +38 -0
- package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.js +374 -0
- package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.d.ts +50 -0
- package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.js +90 -0
- package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.d.ts +33 -0
- package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.js +132 -0
- package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.d.ts +30 -0
- package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.js +81 -0
- package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/index.d.ts +14 -0
- package/dist/esm/src/testing/graph-tester/reporters/index.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/reporters/index.js +10 -0
- package/dist/esm/src/testing/graph-tester/reporters/index.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.d.ts +33 -0
- package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js +43 -0
- package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.d.ts +70 -0
- package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js +547 -0
- package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.d.ts +83 -0
- package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js +339 -0
- package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/index.d.ts +11 -0
- package/dist/esm/src/testing/graph-tester/screenshot/index.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/index.js +9 -0
- package/dist/esm/src/testing/graph-tester/screenshot/index.js.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/types.d.ts +331 -0
- package/dist/esm/src/testing/graph-tester/screenshot/types.d.ts.map +1 -0
- package/dist/esm/src/testing/graph-tester/screenshot/types.js +10 -0
- package/dist/esm/src/testing/graph-tester/screenshot/types.js.map +1 -0
- package/dist/esm/src/testing/selectors.d.ts +1 -1
- package/dist/esm/src/testing/selectors.js +1 -1
- package/package.json +19 -6
|
@@ -0,0 +1,168 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Performance testing mode (stub - to be implemented in Phase 2).
|
|
4
|
+
*
|
|
5
|
+
* Will collect Core Web Vitals and Lighthouse metrics.
|
|
6
|
+
*
|
|
7
|
+
* @packageDocumentation
|
|
8
|
+
*/
|
|
9
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
12
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
13
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
14
|
+
}
|
|
15
|
+
Object.defineProperty(o, k2, desc);
|
|
16
|
+
}) : (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
o[k2] = m[k];
|
|
19
|
+
}));
|
|
20
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
21
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
22
|
+
}) : function(o, v) {
|
|
23
|
+
o["default"] = v;
|
|
24
|
+
});
|
|
25
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
26
|
+
var ownKeys = function(o) {
|
|
27
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
28
|
+
var ar = [];
|
|
29
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
30
|
+
return ar;
|
|
31
|
+
};
|
|
32
|
+
return ownKeys(o);
|
|
33
|
+
};
|
|
34
|
+
return function (mod) {
|
|
35
|
+
if (mod && mod.__esModule) return mod;
|
|
36
|
+
var result = {};
|
|
37
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
38
|
+
__setModuleDefault(result, mod);
|
|
39
|
+
return result;
|
|
40
|
+
};
|
|
41
|
+
})();
|
|
42
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
43
|
+
exports.PerformanceMode = void 0;
|
|
44
|
+
const TestFunction_1 = require("../core/TestFunction");
|
|
45
|
+
/**
|
|
46
|
+
* Performance testing mode (stub).
|
|
47
|
+
*/
|
|
48
|
+
class PerformanceMode extends TestFunction_1.TestFunction {
|
|
49
|
+
constructor(config) {
|
|
50
|
+
super();
|
|
51
|
+
this.mode = 'performance';
|
|
52
|
+
this.name = 'Performance';
|
|
53
|
+
this.description = 'Collects Core Web Vitals and performance metrics';
|
|
54
|
+
this.config = config;
|
|
55
|
+
}
|
|
56
|
+
async execute(page, context) {
|
|
57
|
+
const startTime = Date.now();
|
|
58
|
+
const errors = [];
|
|
59
|
+
try {
|
|
60
|
+
// Collect Core Web Vitals and performance metrics
|
|
61
|
+
const metrics = await page.evaluate(() => {
|
|
62
|
+
return new Promise((resolve) => {
|
|
63
|
+
const observer = new PerformanceObserver((list) => {
|
|
64
|
+
const entries = list.getEntries();
|
|
65
|
+
// Process entries as they come
|
|
66
|
+
});
|
|
67
|
+
// Use performance API to get navigation timing
|
|
68
|
+
const perfData = performance.getEntriesByType('navigation')[0];
|
|
69
|
+
const paintEntries = performance.getEntriesByType('paint');
|
|
70
|
+
const fcp = paintEntries.find((entry) => entry.name === 'first-contentful-paint')?.startTime;
|
|
71
|
+
// Get layout shift metrics
|
|
72
|
+
const layoutShifts = performance.getEntriesByType('layout-shift');
|
|
73
|
+
const cls = layoutShifts.reduce((sum, entry) => sum + (entry.value || 0), 0);
|
|
74
|
+
// Calculate TTI approximation (when main thread is idle)
|
|
75
|
+
const tti = perfData ? perfData.domInteractive : 0;
|
|
76
|
+
// Get resource timing
|
|
77
|
+
const resources = performance.getEntriesByType('resource');
|
|
78
|
+
const totalSize = resources.reduce((sum, r) => sum + (r.transferSize || 0), 0);
|
|
79
|
+
resolve({
|
|
80
|
+
// Core Web Vitals
|
|
81
|
+
fcp: fcp || 0,
|
|
82
|
+
lcp: 0, // Will be collected via observer
|
|
83
|
+
cls: cls || 0,
|
|
84
|
+
tti: tti || 0,
|
|
85
|
+
tbt: 0, // Total Blocking Time - approximation
|
|
86
|
+
// Additional metrics
|
|
87
|
+
domContentLoaded: perfData?.domContentLoadedEventEnd || 0,
|
|
88
|
+
loadComplete: perfData?.loadEventEnd || 0,
|
|
89
|
+
totalResourceSize: totalSize,
|
|
90
|
+
resourceCount: resources.length,
|
|
91
|
+
// Timing breakdown
|
|
92
|
+
dns: perfData ? perfData.domainLookupEnd - perfData.domainLookupStart : 0,
|
|
93
|
+
tcp: perfData ? perfData.connectEnd - perfData.connectStart : 0,
|
|
94
|
+
ttfb: perfData ? perfData.responseStart - perfData.requestStart : 0,
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
});
|
|
98
|
+
// Check thresholds
|
|
99
|
+
const thresholdViolations = [];
|
|
100
|
+
if (this.config.thresholds) {
|
|
101
|
+
if (this.config.thresholds.fcp && metrics.fcp > this.config.thresholds.fcp) {
|
|
102
|
+
thresholdViolations.push({
|
|
103
|
+
severity: 'warning',
|
|
104
|
+
message: `FCP ${metrics.fcp.toFixed(2)}ms exceeds threshold ${this.config.thresholds.fcp}ms`,
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
if (this.config.thresholds.lcp && metrics.lcp > this.config.thresholds.lcp) {
|
|
108
|
+
thresholdViolations.push({
|
|
109
|
+
severity: 'warning',
|
|
110
|
+
message: `LCP ${metrics.lcp.toFixed(2)}ms exceeds threshold ${this.config.thresholds.lcp}ms`,
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
if (this.config.thresholds.cls && metrics.cls > this.config.thresholds.cls) {
|
|
114
|
+
thresholdViolations.push({
|
|
115
|
+
severity: 'warning',
|
|
116
|
+
message: `CLS ${metrics.cls.toFixed(3)} exceeds threshold ${this.config.thresholds.cls}`,
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
if (this.config.thresholds.tti && metrics.tti > this.config.thresholds.tti) {
|
|
120
|
+
thresholdViolations.push({
|
|
121
|
+
severity: 'warning',
|
|
122
|
+
message: `TTI ${metrics.tti.toFixed(2)}ms exceeds threshold ${this.config.thresholds.tti}ms`,
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
// Save results
|
|
127
|
+
const outputPath = `${this.config.outputDir}/${this.sanitizeRoute(context.route)}-performance.json`;
|
|
128
|
+
await this.ensureDirectory(this.config.outputDir);
|
|
129
|
+
await this.writeFile(outputPath, JSON.stringify({ route: context.route, metrics, thresholdViolations }, null, 2));
|
|
130
|
+
const duration = Date.now() - startTime;
|
|
131
|
+
return {
|
|
132
|
+
passed: thresholdViolations.length === 0,
|
|
133
|
+
duration,
|
|
134
|
+
errors: thresholdViolations,
|
|
135
|
+
metadata: {
|
|
136
|
+
performance: metrics,
|
|
137
|
+
outputPath,
|
|
138
|
+
},
|
|
139
|
+
};
|
|
140
|
+
}
|
|
141
|
+
catch (error) {
|
|
142
|
+
errors.push({
|
|
143
|
+
severity: 'critical',
|
|
144
|
+
message: `Performance collection failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
145
|
+
stack: error instanceof Error ? error.stack : undefined,
|
|
146
|
+
});
|
|
147
|
+
return {
|
|
148
|
+
passed: false,
|
|
149
|
+
duration: Date.now() - startTime,
|
|
150
|
+
errors,
|
|
151
|
+
metadata: {},
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
sanitizeRoute(route) {
|
|
156
|
+
return route.replace(/^\//, '').replace(/\//g, '-') || 'home';
|
|
157
|
+
}
|
|
158
|
+
async ensureDirectory(dir) {
|
|
159
|
+
const fs = await Promise.resolve().then(() => __importStar(require('fs/promises')));
|
|
160
|
+
await fs.mkdir(dir, { recursive: true });
|
|
161
|
+
}
|
|
162
|
+
async writeFile(path, content) {
|
|
163
|
+
const fs = await Promise.resolve().then(() => __importStar(require('fs/promises')));
|
|
164
|
+
await fs.writeFile(path, content, 'utf-8');
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
exports.PerformanceMode = PerformanceMode;
|
|
168
|
+
//# sourceMappingURL=PerformanceMode.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"PerformanceMode.js","sourceRoot":"","sources":["../../../../../../src/testing/graph-tester/modes/PerformanceMode.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* SEO testing mode (stub - to be implemented in Phase 2).
|
|
4
|
+
*
|
|
5
|
+
* Will validate meta tags, OpenGraph, and structured data.
|
|
6
|
+
*
|
|
7
|
+
* @packageDocumentation
|
|
8
|
+
*/
|
|
9
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
12
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
13
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
14
|
+
}
|
|
15
|
+
Object.defineProperty(o, k2, desc);
|
|
16
|
+
}) : (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
o[k2] = m[k];
|
|
19
|
+
}));
|
|
20
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
21
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
22
|
+
}) : function(o, v) {
|
|
23
|
+
o["default"] = v;
|
|
24
|
+
});
|
|
25
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
26
|
+
var ownKeys = function(o) {
|
|
27
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
28
|
+
var ar = [];
|
|
29
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
30
|
+
return ar;
|
|
31
|
+
};
|
|
32
|
+
return ownKeys(o);
|
|
33
|
+
};
|
|
34
|
+
return function (mod) {
|
|
35
|
+
if (mod && mod.__esModule) return mod;
|
|
36
|
+
var result = {};
|
|
37
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
38
|
+
__setModuleDefault(result, mod);
|
|
39
|
+
return result;
|
|
40
|
+
};
|
|
41
|
+
})();
|
|
42
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
43
|
+
exports.SEOMode = void 0;
|
|
44
|
+
const TestFunction_1 = require("../core/TestFunction");
|
|
45
|
+
/**
|
|
46
|
+
* SEO testing mode (stub).
|
|
47
|
+
*/
|
|
48
|
+
class SEOMode extends TestFunction_1.TestFunction {
|
|
49
|
+
constructor(config) {
|
|
50
|
+
super();
|
|
51
|
+
this.mode = 'seo';
|
|
52
|
+
this.name = 'SEO';
|
|
53
|
+
this.description = 'Validates meta tags and SEO elements';
|
|
54
|
+
this.config = config;
|
|
55
|
+
}
|
|
56
|
+
async execute(page, context) {
|
|
57
|
+
const startTime = Date.now();
|
|
58
|
+
const errors = [];
|
|
59
|
+
try {
|
|
60
|
+
// Extract SEO metadata
|
|
61
|
+
const seoData = await page.evaluate(() => {
|
|
62
|
+
// Helper to get meta content
|
|
63
|
+
const getMeta = (name) => {
|
|
64
|
+
const meta = document.querySelector(`meta[name="${name}"], meta[property="${name}"]`);
|
|
65
|
+
return meta?.getAttribute('content') || null;
|
|
66
|
+
};
|
|
67
|
+
// Get all meta tags
|
|
68
|
+
const allMeta = {};
|
|
69
|
+
Array.from(document.querySelectorAll('meta')).forEach((meta) => {
|
|
70
|
+
const name = meta.getAttribute('name') || meta.getAttribute('property');
|
|
71
|
+
const content = meta.getAttribute('content');
|
|
72
|
+
if (name && content) {
|
|
73
|
+
allMeta[name] = content;
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
// Title and description
|
|
77
|
+
const title = document.querySelector('title')?.textContent || '';
|
|
78
|
+
const description = getMeta('description');
|
|
79
|
+
// OpenGraph
|
|
80
|
+
const og = {
|
|
81
|
+
title: getMeta('og:title'),
|
|
82
|
+
description: getMeta('og:description'),
|
|
83
|
+
image: getMeta('og:image'),
|
|
84
|
+
url: getMeta('og:url'),
|
|
85
|
+
type: getMeta('og:type'),
|
|
86
|
+
siteName: getMeta('og:site_name'),
|
|
87
|
+
};
|
|
88
|
+
// Twitter Cards
|
|
89
|
+
const twitter = {
|
|
90
|
+
card: getMeta('twitter:card'),
|
|
91
|
+
title: getMeta('twitter:title'),
|
|
92
|
+
description: getMeta('twitter:description'),
|
|
93
|
+
image: getMeta('twitter:image'),
|
|
94
|
+
site: getMeta('twitter:site'),
|
|
95
|
+
creator: getMeta('twitter:creator'),
|
|
96
|
+
};
|
|
97
|
+
// Canonical URL
|
|
98
|
+
const canonical = document.querySelector('link[rel="canonical"]')?.getAttribute('href');
|
|
99
|
+
// Heading structure
|
|
100
|
+
const headings = {
|
|
101
|
+
h1: Array.from(document.querySelectorAll('h1')).map((h) => h.textContent?.trim() || ''),
|
|
102
|
+
h2: Array.from(document.querySelectorAll('h2')).map((h) => h.textContent?.trim() || ''),
|
|
103
|
+
h3Count: document.querySelectorAll('h3').length,
|
|
104
|
+
h4Count: document.querySelectorAll('h4').length,
|
|
105
|
+
};
|
|
106
|
+
// Structured data (JSON-LD)
|
|
107
|
+
const structuredData = [];
|
|
108
|
+
Array.from(document.querySelectorAll('script[type="application/ld+json"]')).forEach((script) => {
|
|
109
|
+
try {
|
|
110
|
+
const data = JSON.parse(script.textContent || '{}');
|
|
111
|
+
structuredData.push(data);
|
|
112
|
+
}
|
|
113
|
+
catch {
|
|
114
|
+
// Ignore invalid JSON
|
|
115
|
+
}
|
|
116
|
+
});
|
|
117
|
+
// Images
|
|
118
|
+
const images = Array.from(document.querySelectorAll('img'));
|
|
119
|
+
const imagesWithoutAlt = images.filter((img) => !img.getAttribute('alt')).length;
|
|
120
|
+
// Links
|
|
121
|
+
const links = Array.from(document.querySelectorAll('a'));
|
|
122
|
+
const externalLinks = links.filter((a) => {
|
|
123
|
+
const href = a.getAttribute('href');
|
|
124
|
+
return href && (href.startsWith('http://') || href.startsWith('https://'));
|
|
125
|
+
}).length;
|
|
126
|
+
return {
|
|
127
|
+
title,
|
|
128
|
+
description,
|
|
129
|
+
canonical,
|
|
130
|
+
allMeta,
|
|
131
|
+
og,
|
|
132
|
+
twitter,
|
|
133
|
+
headings,
|
|
134
|
+
structuredData,
|
|
135
|
+
images: {
|
|
136
|
+
total: images.length,
|
|
137
|
+
withoutAlt: imagesWithoutAlt,
|
|
138
|
+
},
|
|
139
|
+
links: {
|
|
140
|
+
total: links.length,
|
|
141
|
+
external: externalLinks,
|
|
142
|
+
},
|
|
143
|
+
};
|
|
144
|
+
});
|
|
145
|
+
// Validate required meta tags
|
|
146
|
+
const missingTags = [];
|
|
147
|
+
if (this.config.requiredMetaTags) {
|
|
148
|
+
for (const tag of this.config.requiredMetaTags) {
|
|
149
|
+
if (!seoData.allMeta[tag]) {
|
|
150
|
+
missingTags.push(tag);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
// Basic SEO validations
|
|
155
|
+
const issues = [];
|
|
156
|
+
if (!seoData.title) {
|
|
157
|
+
issues.push({ severity: 'critical', message: 'Missing page title' });
|
|
158
|
+
}
|
|
159
|
+
else if (seoData.title.length < 30 || seoData.title.length > 60) {
|
|
160
|
+
issues.push({
|
|
161
|
+
severity: 'warning',
|
|
162
|
+
message: `Title length ${seoData.title.length} characters (recommended: 30-60)`,
|
|
163
|
+
});
|
|
164
|
+
}
|
|
165
|
+
if (!seoData.description) {
|
|
166
|
+
issues.push({ severity: 'critical', message: 'Missing meta description' });
|
|
167
|
+
}
|
|
168
|
+
else if (seoData.description.length < 120 || seoData.description.length > 160) {
|
|
169
|
+
issues.push({
|
|
170
|
+
severity: 'warning',
|
|
171
|
+
message: `Description length ${seoData.description.length} characters (recommended: 120-160)`,
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
if (seoData.headings.h1.length === 0) {
|
|
175
|
+
issues.push({ severity: 'warning', message: 'Missing H1 heading' });
|
|
176
|
+
}
|
|
177
|
+
else if (seoData.headings.h1.length > 1) {
|
|
178
|
+
issues.push({
|
|
179
|
+
severity: 'warning',
|
|
180
|
+
message: `Multiple H1 headings found (${seoData.headings.h1.length})`,
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
if (seoData.images.withoutAlt > 0) {
|
|
184
|
+
issues.push({
|
|
185
|
+
severity: 'warning',
|
|
186
|
+
message: `${seoData.images.withoutAlt} images without alt text`,
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
// OpenGraph validation
|
|
190
|
+
if (this.config.validateOpenGraph !== false) {
|
|
191
|
+
if (!seoData.og.title)
|
|
192
|
+
issues.push({ severity: 'warning', message: 'Missing og:title' });
|
|
193
|
+
if (!seoData.og.description)
|
|
194
|
+
issues.push({ severity: 'warning', message: 'Missing og:description' });
|
|
195
|
+
if (!seoData.og.image)
|
|
196
|
+
issues.push({ severity: 'warning', message: 'Missing og:image' });
|
|
197
|
+
}
|
|
198
|
+
// Twitter Cards validation
|
|
199
|
+
if (this.config.validateTwitterCards !== false) {
|
|
200
|
+
if (!seoData.twitter.card)
|
|
201
|
+
issues.push({ severity: 'info', message: 'Missing twitter:card' });
|
|
202
|
+
}
|
|
203
|
+
if (missingTags.length > 0) {
|
|
204
|
+
issues.push({
|
|
205
|
+
severity: 'critical',
|
|
206
|
+
message: `Missing required meta tags: ${missingTags.join(', ')}`,
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
// Save results
|
|
210
|
+
const outputPath = `${this.config.outputDir}/${this.sanitizeRoute(context.route)}-seo.json`;
|
|
211
|
+
await this.ensureDirectory(this.config.outputDir);
|
|
212
|
+
await this.writeFile(outputPath, JSON.stringify({
|
|
213
|
+
route: context.route,
|
|
214
|
+
seo: seoData,
|
|
215
|
+
issues,
|
|
216
|
+
score: this.calculateScore(issues),
|
|
217
|
+
}, null, 2));
|
|
218
|
+
const duration = Date.now() - startTime;
|
|
219
|
+
return {
|
|
220
|
+
passed: issues.length === 0,
|
|
221
|
+
duration,
|
|
222
|
+
errors: issues,
|
|
223
|
+
metadata: {
|
|
224
|
+
seo: {
|
|
225
|
+
issueCount: issues.length,
|
|
226
|
+
score: this.calculateScore(issues),
|
|
227
|
+
hasStructuredData: seoData.structuredData.length > 0,
|
|
228
|
+
},
|
|
229
|
+
outputPath,
|
|
230
|
+
},
|
|
231
|
+
};
|
|
232
|
+
}
|
|
233
|
+
catch (error) {
|
|
234
|
+
errors.push({
|
|
235
|
+
severity: 'critical',
|
|
236
|
+
message: `SEO validation failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
237
|
+
stack: error instanceof Error ? error.stack : undefined,
|
|
238
|
+
});
|
|
239
|
+
return {
|
|
240
|
+
passed: false,
|
|
241
|
+
duration: Date.now() - startTime,
|
|
242
|
+
errors,
|
|
243
|
+
metadata: {},
|
|
244
|
+
};
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
calculateScore(issues) {
|
|
248
|
+
// Simple scoring: 100 - (10 points per issue)
|
|
249
|
+
return Math.max(0, 100 - issues.length * 10);
|
|
250
|
+
}
|
|
251
|
+
sanitizeRoute(route) {
|
|
252
|
+
return route.replace(/^\//, '').replace(/\//g, '-') || 'home';
|
|
253
|
+
}
|
|
254
|
+
async ensureDirectory(dir) {
|
|
255
|
+
const fs = await Promise.resolve().then(() => __importStar(require('fs/promises')));
|
|
256
|
+
await fs.mkdir(dir, { recursive: true });
|
|
257
|
+
}
|
|
258
|
+
async writeFile(path, content) {
|
|
259
|
+
const fs = await Promise.resolve().then(() => __importStar(require('fs/promises')));
|
|
260
|
+
await fs.writeFile(path, content, 'utf-8');
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
exports.SEOMode = SEOMode;
|
|
264
|
+
//# sourceMappingURL=SEOMode.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SEOMode.js","sourceRoot":"","sources":["../../../../../../src/testing/graph-tester/modes/SEOMode.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Visual regression testing mode.
|
|
4
|
+
*
|
|
5
|
+
* Captures screenshots with element annotations for visual regression testing and AI analysis.
|
|
6
|
+
*
|
|
7
|
+
* @packageDocumentation
|
|
8
|
+
*/
|
|
9
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
12
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
13
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
14
|
+
}
|
|
15
|
+
Object.defineProperty(o, k2, desc);
|
|
16
|
+
}) : (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
o[k2] = m[k];
|
|
19
|
+
}));
|
|
20
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
21
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
22
|
+
}) : function(o, v) {
|
|
23
|
+
o["default"] = v;
|
|
24
|
+
});
|
|
25
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
26
|
+
var ownKeys = function(o) {
|
|
27
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
28
|
+
var ar = [];
|
|
29
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
30
|
+
return ar;
|
|
31
|
+
};
|
|
32
|
+
return ownKeys(o);
|
|
33
|
+
};
|
|
34
|
+
return function (mod) {
|
|
35
|
+
if (mod && mod.__esModule) return mod;
|
|
36
|
+
var result = {};
|
|
37
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
38
|
+
__setModuleDefault(result, mod);
|
|
39
|
+
return result;
|
|
40
|
+
};
|
|
41
|
+
})();
|
|
42
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
43
|
+
exports.VisualRegressionMode = void 0;
|
|
44
|
+
const TestFunction_1 = require("../core/TestFunction");
|
|
45
|
+
const screenshot_1 = require("../screenshot");
|
|
46
|
+
const path = __importStar(require("path"));
|
|
47
|
+
/**
|
|
48
|
+
* Visual regression testing mode.
|
|
49
|
+
*
|
|
50
|
+
* Captures full-page screenshots with element annotations, bounding boxes,
|
|
51
|
+
* and generates interactive HTML overlays for review.
|
|
52
|
+
*
|
|
53
|
+
* @example
|
|
54
|
+
* ```typescript
|
|
55
|
+
* const mode = new VisualRegressionMode({
|
|
56
|
+
* outputDir: './screenshots',
|
|
57
|
+
* annotate: true,
|
|
58
|
+
* generateHTML: true
|
|
59
|
+
* });
|
|
60
|
+
*
|
|
61
|
+
* const tester = new GraphTester({
|
|
62
|
+
* baseUrl: 'http://localhost:3000',
|
|
63
|
+
* routes: [{ route: '/' }],
|
|
64
|
+
* modes: []
|
|
65
|
+
* });
|
|
66
|
+
*
|
|
67
|
+
* tester.registerTestFunction(mode);
|
|
68
|
+
* const results = await tester.runTests();
|
|
69
|
+
* ```
|
|
70
|
+
*/
|
|
71
|
+
class VisualRegressionMode extends TestFunction_1.TestFunction {
|
|
72
|
+
constructor(config) {
|
|
73
|
+
super();
|
|
74
|
+
this.mode = 'visual';
|
|
75
|
+
this.name = 'Visual Regression';
|
|
76
|
+
this.description = 'Captures screenshots with element annotations for visual regression testing';
|
|
77
|
+
this.annotator = null;
|
|
78
|
+
this.config = {
|
|
79
|
+
fullPage: true,
|
|
80
|
+
annotate: true,
|
|
81
|
+
generateHTML: true,
|
|
82
|
+
includeInvisible: false,
|
|
83
|
+
minElementSize: 5,
|
|
84
|
+
aiAnalyze: false,
|
|
85
|
+
...config,
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Setup: Initialize screenshot annotator.
|
|
90
|
+
*/
|
|
91
|
+
async setup(page) {
|
|
92
|
+
if (this.config.annotate) {
|
|
93
|
+
this.annotator = new screenshot_1.ScreenshotAnnotator({
|
|
94
|
+
outputDir: this.config.outputDir,
|
|
95
|
+
fullPage: this.config.fullPage,
|
|
96
|
+
includeInvisible: this.config.includeInvisible,
|
|
97
|
+
minElementSize: this.config.minElementSize,
|
|
98
|
+
...this.config.annotatorConfig,
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Execute: Capture screenshot and annotations.
|
|
104
|
+
*/
|
|
105
|
+
async execute(page, context) {
|
|
106
|
+
const startTime = Date.now();
|
|
107
|
+
const errors = [];
|
|
108
|
+
const metadata = {};
|
|
109
|
+
try {
|
|
110
|
+
// Capture annotated screenshot
|
|
111
|
+
if (this.annotator) {
|
|
112
|
+
const viewportLabel = context.viewport ? ` [${context.viewport.name}]` : '';
|
|
113
|
+
console.log(`Capturing annotated screenshot for ${context.route}${viewportLabel}...`);
|
|
114
|
+
const annotatedScreenshot = await this.annotator.capture(page, context.route, context.viewport?.name);
|
|
115
|
+
metadata.screenshot = {
|
|
116
|
+
path: annotatedScreenshot.screenshotPath,
|
|
117
|
+
annotationCount: annotatedScreenshot.annotations.length,
|
|
118
|
+
interactiveElementCount: annotatedScreenshot.annotations.filter((a) => a.isInteractive)
|
|
119
|
+
.length,
|
|
120
|
+
pageSize: annotatedScreenshot.pageMetadata.pageSize,
|
|
121
|
+
viewport: annotatedScreenshot.pageMetadata.viewport,
|
|
122
|
+
};
|
|
123
|
+
// Generate HTML overlay
|
|
124
|
+
if (this.config.generateHTML) {
|
|
125
|
+
const htmlPath = annotatedScreenshot.screenshotPath.replace('.png', '.html');
|
|
126
|
+
const renderer = new screenshot_1.HTMLAnnotationRenderer({
|
|
127
|
+
outputPath: htmlPath,
|
|
128
|
+
includeSidebar: true,
|
|
129
|
+
enableHover: true,
|
|
130
|
+
enableClickToHighlight: true,
|
|
131
|
+
...this.config.htmlRendererConfig,
|
|
132
|
+
});
|
|
133
|
+
const renderResult = await renderer.render(annotatedScreenshot);
|
|
134
|
+
metadata.htmlOverlay = {
|
|
135
|
+
path: renderResult.outputPath,
|
|
136
|
+
fileSize: renderResult.fileSize,
|
|
137
|
+
duration: renderResult.duration,
|
|
138
|
+
};
|
|
139
|
+
console.log(`Generated HTML overlay: ${renderResult.outputPath}`);
|
|
140
|
+
}
|
|
141
|
+
// AI analysis (enterprise only)
|
|
142
|
+
if (this.config.aiAnalyze) {
|
|
143
|
+
errors.push({
|
|
144
|
+
severity: 'info',
|
|
145
|
+
message: 'AI analysis requires @supernalintelligence/interface-enterprise',
|
|
146
|
+
});
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
else {
|
|
150
|
+
// Simple screenshot without annotations
|
|
151
|
+
const screenshotPath = path.join(this.config.outputDir, `${this.sanitizeRoute(context.route)}.png`);
|
|
152
|
+
await page.screenshot({
|
|
153
|
+
path: screenshotPath,
|
|
154
|
+
fullPage: this.config.fullPage,
|
|
155
|
+
});
|
|
156
|
+
metadata.screenshot = {
|
|
157
|
+
path: screenshotPath,
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
return {
|
|
161
|
+
passed: true,
|
|
162
|
+
duration: Date.now() - startTime,
|
|
163
|
+
errors,
|
|
164
|
+
metadata,
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
catch (error) {
|
|
168
|
+
errors.push({
|
|
169
|
+
severity: 'critical',
|
|
170
|
+
message: `Visual regression failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
171
|
+
stack: error instanceof Error ? error.stack : undefined,
|
|
172
|
+
});
|
|
173
|
+
return {
|
|
174
|
+
passed: false,
|
|
175
|
+
duration: Date.now() - startTime,
|
|
176
|
+
errors,
|
|
177
|
+
metadata,
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Sanitize route for filename.
|
|
183
|
+
*/
|
|
184
|
+
sanitizeRoute(route) {
|
|
185
|
+
return route.replace(/^\//, '').replace(/\//g, '-') || 'home';
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Validate configuration.
|
|
189
|
+
*/
|
|
190
|
+
validate() {
|
|
191
|
+
const errors = super.validate();
|
|
192
|
+
if (!this.config.outputDir) {
|
|
193
|
+
errors.push('outputDir is required');
|
|
194
|
+
}
|
|
195
|
+
return errors;
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
exports.VisualRegressionMode = VisualRegressionMode;
|
|
199
|
+
//# sourceMappingURL=VisualRegressionMode.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"VisualRegressionMode.js","sourceRoot":"","sources":["../../../../../../src/testing/graph-tester/modes/VisualRegressionMode.ts"],"names":[],"mappings":"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"}
|