@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.
Files changed (155) hide show
  1. package/dist/cjs/src/background/navigation/NavigationGraph.js +48 -0
  2. package/dist/cjs/src/background/navigation/NavigationGraph.js.map +1 -1
  3. package/dist/cjs/src/browser.js +42 -1
  4. package/dist/cjs/src/browser.js.map +1 -1
  5. package/dist/cjs/src/decorators/ContainerHelpers.js +16 -1
  6. package/dist/cjs/src/decorators/ContainerHelpers.js.map +1 -1
  7. package/dist/cjs/src/decorators/Tool.js +2 -3
  8. package/dist/cjs/src/decorators/Tool.js.map +1 -1
  9. package/dist/cjs/src/testing/graph-tester/core/GraphTester.js +339 -0
  10. package/dist/cjs/src/testing/graph-tester/core/GraphTester.js.map +1 -0
  11. package/dist/cjs/src/testing/graph-tester/core/TestFunction.js +189 -0
  12. package/dist/cjs/src/testing/graph-tester/core/TestFunction.js.map +1 -0
  13. package/dist/cjs/src/testing/graph-tester/core/types.js +24 -0
  14. package/dist/cjs/src/testing/graph-tester/core/types.js.map +1 -0
  15. package/dist/cjs/src/testing/graph-tester/fixtures/index.js +13 -0
  16. package/dist/cjs/src/testing/graph-tester/fixtures/index.js.map +1 -0
  17. package/dist/cjs/src/testing/graph-tester/fixtures/portPool.js +184 -0
  18. package/dist/cjs/src/testing/graph-tester/fixtures/portPool.js.map +1 -0
  19. package/dist/cjs/src/testing/graph-tester/index.js +98 -0
  20. package/dist/cjs/src/testing/graph-tester/index.js.map +1 -0
  21. package/dist/cjs/src/testing/graph-tester/modes/AccessibilityMode.js +230 -0
  22. package/dist/cjs/src/testing/graph-tester/modes/AccessibilityMode.js.map +1 -0
  23. package/dist/cjs/src/testing/graph-tester/modes/PerformanceMode.js +168 -0
  24. package/dist/cjs/src/testing/graph-tester/modes/PerformanceMode.js.map +1 -0
  25. package/dist/cjs/src/testing/graph-tester/modes/SEOMode.js +264 -0
  26. package/dist/cjs/src/testing/graph-tester/modes/SEOMode.js.map +1 -0
  27. package/dist/cjs/src/testing/graph-tester/modes/VisualRegressionMode.js +199 -0
  28. package/dist/cjs/src/testing/graph-tester/modes/VisualRegressionMode.js.map +1 -0
  29. package/dist/cjs/src/testing/graph-tester/modes/index.js +17 -0
  30. package/dist/cjs/src/testing/graph-tester/modes/index.js.map +1 -0
  31. package/dist/cjs/src/testing/graph-tester/reporters/HTMLReporter.js +411 -0
  32. package/dist/cjs/src/testing/graph-tester/reporters/HTMLReporter.js.map +1 -0
  33. package/dist/cjs/src/testing/graph-tester/reporters/JSONReporter.js +127 -0
  34. package/dist/cjs/src/testing/graph-tester/reporters/JSONReporter.js.map +1 -0
  35. package/dist/cjs/src/testing/graph-tester/reporters/MarkdownReporter.js +169 -0
  36. package/dist/cjs/src/testing/graph-tester/reporters/MarkdownReporter.js.map +1 -0
  37. package/dist/cjs/src/testing/graph-tester/reporters/UnifiedReporter.js +118 -0
  38. package/dist/cjs/src/testing/graph-tester/reporters/UnifiedReporter.js.map +1 -0
  39. package/dist/cjs/src/testing/graph-tester/reporters/index.js +17 -0
  40. package/dist/cjs/src/testing/graph-tester/reporters/index.js.map +1 -0
  41. package/dist/cjs/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js +47 -0
  42. package/dist/cjs/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js.map +1 -0
  43. package/dist/cjs/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js +584 -0
  44. package/dist/cjs/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js.map +1 -0
  45. package/dist/cjs/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js +376 -0
  46. package/dist/cjs/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js.map +1 -0
  47. package/dist/cjs/src/testing/graph-tester/screenshot/index.js +15 -0
  48. package/dist/cjs/src/testing/graph-tester/screenshot/index.js.map +1 -0
  49. package/dist/cjs/src/testing/graph-tester/screenshot/types.js +11 -0
  50. package/dist/cjs/src/testing/graph-tester/screenshot/types.js.map +1 -0
  51. package/dist/cjs/src/testing/selectors.js +1 -1
  52. package/dist/esm/src/background/navigation/INavigationGraph.d.ts +15 -0
  53. package/dist/esm/src/background/navigation/INavigationGraph.d.ts.map +1 -1
  54. package/dist/esm/src/background/navigation/NavigationGraph.d.ts +32 -0
  55. package/dist/esm/src/background/navigation/NavigationGraph.d.ts.map +1 -1
  56. package/dist/esm/src/background/navigation/NavigationGraph.js +48 -0
  57. package/dist/esm/src/background/navigation/NavigationGraph.js.map +1 -1
  58. package/dist/esm/src/browser.d.ts +9 -0
  59. package/dist/esm/src/browser.d.ts.map +1 -1
  60. package/dist/esm/src/browser.js +14 -0
  61. package/dist/esm/src/browser.js.map +1 -1
  62. package/dist/esm/src/decorators/ContainerHelpers.d.ts +2 -1
  63. package/dist/esm/src/decorators/ContainerHelpers.d.ts.map +1 -1
  64. package/dist/esm/src/decorators/ContainerHelpers.js +16 -1
  65. package/dist/esm/src/decorators/ContainerHelpers.js.map +1 -1
  66. package/dist/esm/src/decorators/Tool.d.ts.map +1 -1
  67. package/dist/esm/src/decorators/Tool.js +2 -3
  68. package/dist/esm/src/decorators/Tool.js.map +1 -1
  69. package/dist/esm/src/testing/graph-tester/core/GraphTester.d.ts +110 -0
  70. package/dist/esm/src/testing/graph-tester/core/GraphTester.d.ts.map +1 -0
  71. package/dist/esm/src/testing/graph-tester/core/GraphTester.js +335 -0
  72. package/dist/esm/src/testing/graph-tester/core/GraphTester.js.map +1 -0
  73. package/dist/esm/src/testing/graph-tester/core/TestFunction.d.ts +120 -0
  74. package/dist/esm/src/testing/graph-tester/core/TestFunction.d.ts.map +1 -0
  75. package/dist/esm/src/testing/graph-tester/core/TestFunction.js +184 -0
  76. package/dist/esm/src/testing/graph-tester/core/TestFunction.js.map +1 -0
  77. package/dist/esm/src/testing/graph-tester/core/types.d.ts +331 -0
  78. package/dist/esm/src/testing/graph-tester/core/types.d.ts.map +1 -0
  79. package/dist/esm/src/testing/graph-tester/core/types.js +21 -0
  80. package/dist/esm/src/testing/graph-tester/core/types.js.map +1 -0
  81. package/dist/esm/src/testing/graph-tester/fixtures/index.d.ts +8 -0
  82. package/dist/esm/src/testing/graph-tester/fixtures/index.d.ts.map +1 -0
  83. package/dist/esm/src/testing/graph-tester/fixtures/index.js +7 -0
  84. package/dist/esm/src/testing/graph-tester/fixtures/index.js.map +1 -0
  85. package/dist/esm/src/testing/graph-tester/fixtures/portPool.d.ts +40 -0
  86. package/dist/esm/src/testing/graph-tester/fixtures/portPool.d.ts.map +1 -0
  87. package/dist/esm/src/testing/graph-tester/fixtures/portPool.js +147 -0
  88. package/dist/esm/src/testing/graph-tester/fixtures/portPool.js.map +1 -0
  89. package/dist/esm/src/testing/graph-tester/index.d.ts +84 -0
  90. package/dist/esm/src/testing/graph-tester/index.d.ts.map +1 -0
  91. package/dist/esm/src/testing/graph-tester/index.js +77 -0
  92. package/dist/esm/src/testing/graph-tester/index.js.map +1 -0
  93. package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.d.ts +40 -0
  94. package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.d.ts.map +1 -0
  95. package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.js +193 -0
  96. package/dist/esm/src/testing/graph-tester/modes/AccessibilityMode.js.map +1 -0
  97. package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.d.ts +42 -0
  98. package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.d.ts.map +1 -0
  99. package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.js +131 -0
  100. package/dist/esm/src/testing/graph-tester/modes/PerformanceMode.js.map +1 -0
  101. package/dist/esm/src/testing/graph-tester/modes/SEOMode.d.ts +39 -0
  102. package/dist/esm/src/testing/graph-tester/modes/SEOMode.d.ts.map +1 -0
  103. package/dist/esm/src/testing/graph-tester/modes/SEOMode.js +227 -0
  104. package/dist/esm/src/testing/graph-tester/modes/SEOMode.js.map +1 -0
  105. package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.d.ts +83 -0
  106. package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.d.ts.map +1 -0
  107. package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.js +162 -0
  108. package/dist/esm/src/testing/graph-tester/modes/VisualRegressionMode.js.map +1 -0
  109. package/dist/esm/src/testing/graph-tester/modes/index.d.ts +14 -0
  110. package/dist/esm/src/testing/graph-tester/modes/index.d.ts.map +1 -0
  111. package/dist/esm/src/testing/graph-tester/modes/index.js +10 -0
  112. package/dist/esm/src/testing/graph-tester/modes/index.js.map +1 -0
  113. package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.d.ts +38 -0
  114. package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.d.ts.map +1 -0
  115. package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.js +374 -0
  116. package/dist/esm/src/testing/graph-tester/reporters/HTMLReporter.js.map +1 -0
  117. package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.d.ts +50 -0
  118. package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.d.ts.map +1 -0
  119. package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.js +90 -0
  120. package/dist/esm/src/testing/graph-tester/reporters/JSONReporter.js.map +1 -0
  121. package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.d.ts +33 -0
  122. package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.d.ts.map +1 -0
  123. package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.js +132 -0
  124. package/dist/esm/src/testing/graph-tester/reporters/MarkdownReporter.js.map +1 -0
  125. package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.d.ts +30 -0
  126. package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.d.ts.map +1 -0
  127. package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.js +81 -0
  128. package/dist/esm/src/testing/graph-tester/reporters/UnifiedReporter.js.map +1 -0
  129. package/dist/esm/src/testing/graph-tester/reporters/index.d.ts +14 -0
  130. package/dist/esm/src/testing/graph-tester/reporters/index.d.ts.map +1 -0
  131. package/dist/esm/src/testing/graph-tester/reporters/index.js +10 -0
  132. package/dist/esm/src/testing/graph-tester/reporters/index.js.map +1 -0
  133. package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.d.ts +33 -0
  134. package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.d.ts.map +1 -0
  135. package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js +43 -0
  136. package/dist/esm/src/testing/graph-tester/screenshot/CanvasAnnotationRenderer.js.map +1 -0
  137. package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.d.ts +70 -0
  138. package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.d.ts.map +1 -0
  139. package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js +547 -0
  140. package/dist/esm/src/testing/graph-tester/screenshot/HTMLAnnotationRenderer.js.map +1 -0
  141. package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.d.ts +83 -0
  142. package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.d.ts.map +1 -0
  143. package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js +339 -0
  144. package/dist/esm/src/testing/graph-tester/screenshot/ScreenshotAnnotator.js.map +1 -0
  145. package/dist/esm/src/testing/graph-tester/screenshot/index.d.ts +11 -0
  146. package/dist/esm/src/testing/graph-tester/screenshot/index.d.ts.map +1 -0
  147. package/dist/esm/src/testing/graph-tester/screenshot/index.js +9 -0
  148. package/dist/esm/src/testing/graph-tester/screenshot/index.js.map +1 -0
  149. package/dist/esm/src/testing/graph-tester/screenshot/types.d.ts +331 -0
  150. package/dist/esm/src/testing/graph-tester/screenshot/types.d.ts.map +1 -0
  151. package/dist/esm/src/testing/graph-tester/screenshot/types.js +10 -0
  152. package/dist/esm/src/testing/graph-tester/screenshot/types.js.map +1 -0
  153. package/dist/esm/src/testing/selectors.d.ts +1 -1
  154. package/dist/esm/src/testing/selectors.js +1 -1
  155. 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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"}