qa360 1.4.5 → 2.0.1

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 (209) hide show
  1. package/README.md +1 -1
  2. package/dist/commands/ai.d.ts +41 -0
  3. package/dist/commands/ai.js +499 -0
  4. package/dist/commands/ask.js +12 -12
  5. package/dist/commands/coverage.d.ts +8 -0
  6. package/dist/commands/coverage.js +252 -0
  7. package/dist/commands/explain.d.ts +27 -0
  8. package/dist/commands/explain.js +630 -0
  9. package/dist/commands/flakiness.d.ts +73 -0
  10. package/dist/commands/flakiness.js +435 -0
  11. package/dist/commands/generate.d.ts +66 -0
  12. package/dist/commands/generate.js +438 -0
  13. package/dist/commands/init.d.ts +56 -9
  14. package/dist/commands/init.js +217 -10
  15. package/dist/commands/monitor.d.ts +27 -0
  16. package/dist/commands/monitor.js +225 -0
  17. package/dist/commands/ollama.d.ts +40 -0
  18. package/dist/commands/ollama.js +301 -0
  19. package/dist/commands/pack.d.ts +37 -9
  20. package/dist/commands/pack.js +240 -141
  21. package/dist/commands/regression.d.ts +8 -0
  22. package/dist/commands/regression.js +340 -0
  23. package/dist/commands/repair.d.ts +26 -0
  24. package/dist/commands/repair.js +307 -0
  25. package/dist/commands/retry.d.ts +43 -0
  26. package/dist/commands/retry.js +275 -0
  27. package/dist/commands/run.d.ts +8 -3
  28. package/dist/commands/run.js +45 -31
  29. package/dist/commands/slo.d.ts +8 -0
  30. package/dist/commands/slo.js +327 -0
  31. package/dist/core/adapters/playwright-native-api.d.ts +183 -0
  32. package/dist/core/adapters/playwright-native-api.js +461 -0
  33. package/dist/core/adapters/playwright-ui.d.ts +7 -0
  34. package/dist/core/adapters/playwright-ui.js +29 -1
  35. package/dist/core/ai/anthropic-provider.d.ts +50 -0
  36. package/dist/core/ai/anthropic-provider.js +211 -0
  37. package/dist/core/ai/deepseek-provider.d.ts +81 -0
  38. package/dist/core/ai/deepseek-provider.js +254 -0
  39. package/dist/core/ai/index.d.ts +60 -0
  40. package/dist/core/ai/index.js +18 -0
  41. package/dist/core/ai/llm-client.d.ts +45 -0
  42. package/dist/core/ai/llm-client.js +7 -0
  43. package/dist/core/ai/mock-provider.d.ts +49 -0
  44. package/dist/core/ai/mock-provider.js +121 -0
  45. package/dist/core/ai/ollama-provider.d.ts +78 -0
  46. package/dist/core/ai/ollama-provider.js +192 -0
  47. package/dist/core/ai/openai-provider.d.ts +48 -0
  48. package/dist/core/ai/openai-provider.js +188 -0
  49. package/dist/core/ai/provider-factory.d.ts +160 -0
  50. package/dist/core/ai/provider-factory.js +269 -0
  51. package/dist/core/auth/api-key-provider.d.ts +16 -0
  52. package/dist/core/auth/api-key-provider.js +63 -0
  53. package/dist/core/auth/aws-iam-provider.d.ts +35 -0
  54. package/dist/core/auth/aws-iam-provider.js +177 -0
  55. package/dist/core/auth/azure-ad-provider.d.ts +15 -0
  56. package/dist/core/auth/azure-ad-provider.js +99 -0
  57. package/dist/core/auth/basic-auth-provider.d.ts +26 -0
  58. package/dist/core/auth/basic-auth-provider.js +111 -0
  59. package/dist/core/auth/gcp-adc-provider.d.ts +27 -0
  60. package/dist/core/auth/gcp-adc-provider.js +126 -0
  61. package/dist/core/auth/index.d.ts +238 -0
  62. package/dist/core/auth/index.js +82 -0
  63. package/dist/core/auth/jwt-provider.d.ts +19 -0
  64. package/dist/core/auth/jwt-provider.js +160 -0
  65. package/dist/core/auth/manager.d.ts +84 -0
  66. package/dist/core/auth/manager.js +230 -0
  67. package/dist/core/auth/oauth2-provider.d.ts +17 -0
  68. package/dist/core/auth/oauth2-provider.js +114 -0
  69. package/dist/core/auth/totp-provider.d.ts +31 -0
  70. package/dist/core/auth/totp-provider.js +134 -0
  71. package/dist/core/auth/ui-login-provider.d.ts +26 -0
  72. package/dist/core/auth/ui-login-provider.js +198 -0
  73. package/dist/core/cache/index.d.ts +7 -0
  74. package/dist/core/cache/index.js +6 -0
  75. package/dist/core/cache/lru-cache.d.ts +203 -0
  76. package/dist/core/cache/lru-cache.js +397 -0
  77. package/dist/core/coverage/analyzer.d.ts +101 -0
  78. package/dist/core/coverage/analyzer.js +415 -0
  79. package/dist/core/coverage/collector.d.ts +74 -0
  80. package/dist/core/coverage/collector.js +459 -0
  81. package/dist/core/coverage/config.d.ts +37 -0
  82. package/dist/core/coverage/config.js +156 -0
  83. package/dist/core/coverage/index.d.ts +11 -0
  84. package/dist/core/coverage/index.js +15 -0
  85. package/dist/core/coverage/types.d.ts +267 -0
  86. package/dist/core/coverage/types.js +6 -0
  87. package/dist/core/coverage/vault.d.ts +95 -0
  88. package/dist/core/coverage/vault.js +405 -0
  89. package/dist/core/dashboard/assets.d.ts +6 -0
  90. package/dist/core/dashboard/assets.js +690 -0
  91. package/dist/core/dashboard/index.d.ts +6 -0
  92. package/dist/core/dashboard/index.js +5 -0
  93. package/dist/core/dashboard/server.d.ts +72 -0
  94. package/dist/core/dashboard/server.js +354 -0
  95. package/dist/core/dashboard/types.d.ts +70 -0
  96. package/dist/core/dashboard/types.js +5 -0
  97. package/dist/core/discoverer/index.d.ts +115 -0
  98. package/dist/core/discoverer/index.js +250 -0
  99. package/dist/core/flakiness/index.d.ts +228 -0
  100. package/dist/core/flakiness/index.js +384 -0
  101. package/dist/core/generation/code-formatter.d.ts +111 -0
  102. package/dist/core/generation/code-formatter.js +307 -0
  103. package/dist/core/generation/code-generator.d.ts +144 -0
  104. package/dist/core/generation/code-generator.js +293 -0
  105. package/dist/core/generation/generator.d.ts +40 -0
  106. package/dist/core/generation/generator.js +76 -0
  107. package/dist/core/generation/index.d.ts +30 -0
  108. package/dist/core/generation/index.js +28 -0
  109. package/dist/core/generation/pack-generator.d.ts +107 -0
  110. package/dist/core/generation/pack-generator.js +416 -0
  111. package/dist/core/generation/prompt-builder.d.ts +132 -0
  112. package/dist/core/generation/prompt-builder.js +672 -0
  113. package/dist/core/generation/source-analyzer.d.ts +213 -0
  114. package/dist/core/generation/source-analyzer.js +657 -0
  115. package/dist/core/generation/test-optimizer.d.ts +117 -0
  116. package/dist/core/generation/test-optimizer.js +328 -0
  117. package/dist/core/generation/types.d.ts +214 -0
  118. package/dist/core/generation/types.js +4 -0
  119. package/dist/core/index.d.ts +23 -1
  120. package/dist/core/index.js +39 -0
  121. package/dist/core/pack/validator.js +31 -1
  122. package/dist/core/pack-v2/index.d.ts +9 -0
  123. package/dist/core/pack-v2/index.js +8 -0
  124. package/dist/core/pack-v2/loader.d.ts +62 -0
  125. package/dist/core/pack-v2/loader.js +231 -0
  126. package/dist/core/pack-v2/migrator.d.ts +56 -0
  127. package/dist/core/pack-v2/migrator.js +455 -0
  128. package/dist/core/pack-v2/validator.d.ts +61 -0
  129. package/dist/core/pack-v2/validator.js +577 -0
  130. package/dist/core/regression/detector.d.ts +107 -0
  131. package/dist/core/regression/detector.js +497 -0
  132. package/dist/core/regression/index.d.ts +9 -0
  133. package/dist/core/regression/index.js +11 -0
  134. package/dist/core/regression/trend-analyzer.d.ts +102 -0
  135. package/dist/core/regression/trend-analyzer.js +345 -0
  136. package/dist/core/regression/types.d.ts +222 -0
  137. package/dist/core/regression/types.js +7 -0
  138. package/dist/core/regression/vault.d.ts +87 -0
  139. package/dist/core/regression/vault.js +289 -0
  140. package/dist/core/repair/engine/fixer.d.ts +24 -0
  141. package/dist/core/repair/engine/fixer.js +226 -0
  142. package/dist/core/repair/engine/suggestion-engine.d.ts +18 -0
  143. package/dist/core/repair/engine/suggestion-engine.js +187 -0
  144. package/dist/core/repair/index.d.ts +10 -0
  145. package/dist/core/repair/index.js +13 -0
  146. package/dist/core/repair/repairer.d.ts +90 -0
  147. package/dist/core/repair/repairer.js +284 -0
  148. package/dist/core/repair/types.d.ts +91 -0
  149. package/dist/core/repair/types.js +6 -0
  150. package/dist/core/repair/utils/error-analyzer.d.ts +28 -0
  151. package/dist/core/repair/utils/error-analyzer.js +264 -0
  152. package/dist/core/retry/flakiness-integration.d.ts +60 -0
  153. package/dist/core/retry/flakiness-integration.js +228 -0
  154. package/dist/core/retry/index.d.ts +14 -0
  155. package/dist/core/retry/index.js +16 -0
  156. package/dist/core/retry/retry-engine.d.ts +80 -0
  157. package/dist/core/retry/retry-engine.js +296 -0
  158. package/dist/core/retry/types.d.ts +178 -0
  159. package/dist/core/retry/types.js +52 -0
  160. package/dist/core/retry/vault.d.ts +77 -0
  161. package/dist/core/retry/vault.js +304 -0
  162. package/dist/core/runner/e2e-helpers.d.ts +102 -0
  163. package/dist/core/runner/e2e-helpers.js +153 -0
  164. package/dist/core/runner/phase3-runner.d.ts +101 -2
  165. package/dist/core/runner/phase3-runner.js +559 -24
  166. package/dist/core/self-healing/assertion-healer.d.ts +97 -0
  167. package/dist/core/self-healing/assertion-healer.js +371 -0
  168. package/dist/core/self-healing/engine.d.ts +122 -0
  169. package/dist/core/self-healing/engine.js +538 -0
  170. package/dist/core/self-healing/index.d.ts +10 -0
  171. package/dist/core/self-healing/index.js +11 -0
  172. package/dist/core/self-healing/selector-healer.d.ts +103 -0
  173. package/dist/core/self-healing/selector-healer.js +372 -0
  174. package/dist/core/self-healing/types.d.ts +152 -0
  175. package/dist/core/self-healing/types.js +6 -0
  176. package/dist/core/slo/config.d.ts +107 -0
  177. package/dist/core/slo/config.js +360 -0
  178. package/dist/core/slo/index.d.ts +11 -0
  179. package/dist/core/slo/index.js +15 -0
  180. package/dist/core/slo/sli-calculator.d.ts +92 -0
  181. package/dist/core/slo/sli-calculator.js +364 -0
  182. package/dist/core/slo/slo-tracker.d.ts +148 -0
  183. package/dist/core/slo/slo-tracker.js +379 -0
  184. package/dist/core/slo/types.d.ts +281 -0
  185. package/dist/core/slo/types.js +7 -0
  186. package/dist/core/slo/vault.d.ts +102 -0
  187. package/dist/core/slo/vault.js +427 -0
  188. package/dist/core/tui/index.d.ts +7 -0
  189. package/dist/core/tui/index.js +6 -0
  190. package/dist/core/tui/monitor.d.ts +92 -0
  191. package/dist/core/tui/monitor.js +271 -0
  192. package/dist/core/tui/renderer.d.ts +33 -0
  193. package/dist/core/tui/renderer.js +218 -0
  194. package/dist/core/tui/types.d.ts +63 -0
  195. package/dist/core/tui/types.js +5 -0
  196. package/dist/core/types/pack-v2.d.ts +425 -0
  197. package/dist/core/types/pack-v2.js +8 -0
  198. package/dist/core/vault/index.d.ts +116 -0
  199. package/dist/core/vault/index.js +400 -5
  200. package/dist/core/watch/index.d.ts +7 -0
  201. package/dist/core/watch/index.js +6 -0
  202. package/dist/core/watch/watch-mode.d.ts +213 -0
  203. package/dist/core/watch/watch-mode.js +389 -0
  204. package/dist/index.js +68 -68
  205. package/dist/utils/config.d.ts +5 -0
  206. package/dist/utils/config.js +136 -0
  207. package/package.json +5 -1
  208. package/dist/core/adapters/playwright-api.d.ts +0 -82
  209. package/dist/core/adapters/playwright-api.js +0 -264
@@ -0,0 +1,289 @@
1
+ /**
2
+ * Regression Vault Integration
3
+ *
4
+ * Integrates regression data storage and retrieval with the Evidence Vault.
5
+ */
6
+ import { promisify } from 'util';
7
+ /**
8
+ * Regression Vault class
9
+ */
10
+ export class RegressionVault {
11
+ db;
12
+ dbRun;
13
+ dbAll;
14
+ dbGet;
15
+ constructor(db) {
16
+ this.db = db;
17
+ this.dbRun = promisify(db.run.bind(db));
18
+ this.dbAll = promisify(db.all.bind(db));
19
+ this.dbGet = promisify(db.get.bind(db));
20
+ }
21
+ /**
22
+ * Initialize regression tables
23
+ */
24
+ async initialize() {
25
+ const schema = `
26
+ -- Regression detections table
27
+ CREATE TABLE IF NOT EXISTS regression_detections (
28
+ id TEXT PRIMARY KEY,
29
+ run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
30
+ baseline_run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE SET NULL,
31
+ type TEXT NOT NULL,
32
+ severity TEXT NOT NULL,
33
+ status TEXT NOT NULL DEFAULT 'detected',
34
+ metric_name TEXT NOT NULL,
35
+ baseline_value REAL NOT NULL,
36
+ current_value REAL NOT NULL,
37
+ absolute_change REAL NOT NULL,
38
+ percent_change REAL NOT NULL,
39
+ significance REAL NOT NULL,
40
+ confidence REAL NOT NULL,
41
+ direction TEXT NOT NULL,
42
+ affected_component TEXT,
43
+ gate TEXT NOT NULL,
44
+ context_json TEXT,
45
+ suggestions_json TEXT NOT NULL,
46
+ commits_json TEXT,
47
+ timestamp INTEGER NOT NULL,
48
+ notes TEXT
49
+ );
50
+
51
+ CREATE INDEX IF NOT EXISTS idx_regression_run_id ON regression_detections(run_id);
52
+ CREATE INDEX IF NOT EXISTS idx_regression_baseline_run_id ON regression_detections(baseline_run_id);
53
+ CREATE INDEX IF NOT EXISTS idx_regression_type ON regression_detections(type);
54
+ CREATE INDEX IF NOT EXISTS idx_regression_severity ON regression_detections(severity);
55
+ CREATE INDEX IF NOT EXISTS idx_regression_status ON regression_detections(status);
56
+ CREATE INDEX IF NOT EXISTS idx_regression_metric ON regression_detections(metric_name);
57
+ CREATE INDEX IF NOT EXISTS idx_regression_timestamp ON regression_detections(timestamp);
58
+
59
+ -- Regression snapshots (aggregate metrics over time)
60
+ CREATE TABLE IF NOT EXISTS regression_snapshots (
61
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
62
+ run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
63
+ metric_name TEXT NOT NULL,
64
+ value REAL NOT NULL,
65
+ baseline_value REAL,
66
+ change_percent REAL,
67
+ is_regression INTEGER DEFAULT 0,
68
+ timestamp INTEGER NOT NULL,
69
+ UNIQUE(run_id, metric_name)
70
+ );
71
+
72
+ CREATE INDEX IF NOT EXISTS idx_regression_snapshots_run_id ON regression_snapshots(run_id);
73
+ CREATE INDEX IF NOT EXISTS idx_regression_snapshots_metric ON regression_snapshots(metric_name);
74
+
75
+ -- Regression history for trend analysis
76
+ CREATE TABLE IF NOT EXISTS regression_history (
77
+ id INTEGER PRIMARY KEY AUTOINCREMENT,
78
+ metric_name TEXT NOT NULL,
79
+ run_id TEXT NOT NULL REFERENCES runs(id) ON DELETE CASCADE,
80
+ value REAL NOT NULL,
81
+ mean REAL NOT NULL,
82
+ std_dev REAL NOT NULL,
83
+ sample_size INTEGER NOT NULL,
84
+ window_start INTEGER NOT NULL,
85
+ window_end INTEGER NOT NULL,
86
+ timestamp INTEGER NOT NULL
87
+ );
88
+
89
+ CREATE INDEX IF NOT EXISTS idx_regression_history_metric ON regression_history(metric_name);
90
+ CREATE INDEX IF NOT EXISTS idx_regression_history_timestamp ON regression_history(timestamp);
91
+ `;
92
+ const statements = schema
93
+ .split(';')
94
+ .map(s => s.trim())
95
+ .filter(s => s.length > 0);
96
+ for (const statement of statements) {
97
+ await this.dbRun(statement, []);
98
+ }
99
+ }
100
+ /**
101
+ * Store a regression detection
102
+ */
103
+ async storeRegression(regression) {
104
+ await this.dbRun(`INSERT INTO regression_detections (
105
+ id, run_id, baseline_run_id, type, severity, status,
106
+ metric_name, baseline_value, current_value, absolute_change,
107
+ percent_change, significance, confidence, direction,
108
+ affected_component, gate, context_json, suggestions_json,
109
+ commits_json, timestamp, notes
110
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`, [
111
+ regression.id,
112
+ regression.runId,
113
+ regression.baselineRunId,
114
+ regression.type,
115
+ regression.severity,
116
+ regression.status,
117
+ regression.metricName,
118
+ regression.baselineValue,
119
+ regression.currentValue,
120
+ regression.absoluteChange,
121
+ regression.percentChange,
122
+ regression.significance,
123
+ regression.confidence,
124
+ regression.direction,
125
+ regression.affectedComponent,
126
+ regression.gate,
127
+ regression.context ? JSON.stringify(regression.context) : null,
128
+ JSON.stringify(regression.suggestions),
129
+ regression.commits ? JSON.stringify(regression.commits) : null,
130
+ regression.timestamp,
131
+ regression.notes || null
132
+ ]);
133
+ }
134
+ /**
135
+ * Store multiple regressions
136
+ */
137
+ async storeRegressions(regressions) {
138
+ for (const regression of regressions) {
139
+ await this.storeRegression(regression);
140
+ }
141
+ }
142
+ /**
143
+ * Get regressions by run ID
144
+ */
145
+ async getRegressionsByRun(runId) {
146
+ const rows = await this.dbAll(`SELECT * FROM regression_detections WHERE run_id = ? ORDER BY severity DESC, percent_change DESC`, [runId]);
147
+ return this.mapRowsToDetections(rows);
148
+ }
149
+ /**
150
+ * Get regressions by type
151
+ */
152
+ async getRegressionsByType(type, limit = 100) {
153
+ const rows = await this.dbAll(`SELECT * FROM regression_detections WHERE type = ? ORDER BY timestamp DESC LIMIT ?`, [type, limit]);
154
+ return this.mapRowsToDetections(rows);
155
+ }
156
+ /**
157
+ * Get regressions by severity
158
+ */
159
+ async getRegressionsBySeverity(severity, limit = 100) {
160
+ const rows = await this.dbAll(`SELECT * FROM regression_detections WHERE severity = ? ORDER BY timestamp DESC LIMIT ?`, [severity, limit]);
161
+ return this.mapRowsToDetections(rows);
162
+ }
163
+ /**
164
+ * Get regressions by status
165
+ */
166
+ async getRegressionsByStatus(status, limit = 100) {
167
+ const rows = await this.dbAll(`SELECT * FROM regression_detections WHERE status = ? ORDER BY timestamp DESC LIMIT ?`, [status, limit]);
168
+ return this.mapRowsToDetections(rows);
169
+ }
170
+ /**
171
+ * Get recent regressions
172
+ */
173
+ async getRecentRegressions(limit = 50) {
174
+ const rows = await this.dbAll(`SELECT * FROM regression_detections ORDER BY timestamp DESC LIMIT ?`, [limit]);
175
+ return this.mapRowsToDetections(rows);
176
+ }
177
+ /**
178
+ * Get regressions for a metric
179
+ */
180
+ async getRegressionsByMetric(metricName, limit = 50) {
181
+ const rows = await this.dbAll(`SELECT * FROM regression_detections WHERE metric_name = ? ORDER BY timestamp DESC LIMIT ?`, [metricName, limit]);
182
+ return this.mapRowsToDetections(rows);
183
+ }
184
+ /**
185
+ * Update regression status
186
+ */
187
+ async updateRegressionStatus(id, status, notes) {
188
+ let sql = `UPDATE regression_detections SET status = ?`;
189
+ const params = [status];
190
+ if (notes) {
191
+ sql += `, notes = ?`;
192
+ params.push(notes);
193
+ }
194
+ sql += ` WHERE id = ?`;
195
+ params.push(id);
196
+ await this.dbRun(sql, params);
197
+ }
198
+ /**
199
+ * Store regression snapshot
200
+ */
201
+ async storeSnapshot(runId, metricName, value, baselineValue) {
202
+ const changePercent = baselineValue !== undefined
203
+ ? ((value - baselineValue) / baselineValue) * 100
204
+ : null;
205
+ await this.dbRun(`INSERT OR REPLACE INTO regression_snapshots (
206
+ run_id, metric_name, value, baseline_value, change_percent, timestamp
207
+ ) VALUES (?, ?, ?, ?, ?, ?)`, [runId, metricName, value, baselineValue, changePercent, Date.now()]);
208
+ }
209
+ /**
210
+ * Get metric history for trend analysis
211
+ */
212
+ async getMetricHistory(metricName, limit = 100) {
213
+ const rows = await this.dbAll(`SELECT run_id, value, timestamp FROM regression_snapshots
214
+ WHERE metric_name = ?
215
+ ORDER BY timestamp DESC
216
+ LIMIT ?`, [metricName, limit]);
217
+ return rows.map((r) => ({
218
+ runId: r.run_id,
219
+ value: r.value,
220
+ timestamp: r.timestamp
221
+ }));
222
+ }
223
+ /**
224
+ * Get regression statistics
225
+ */
226
+ async getStatistics() {
227
+ const total = await this.dbGet(`SELECT COUNT(*) as count FROM regression_detections`, []);
228
+ const bySeverity = await this.dbAll(`SELECT severity, COUNT(*) as count FROM regression_detections GROUP BY severity`, []);
229
+ const byType = await this.dbAll(`SELECT type, COUNT(*) as count FROM regression_detections GROUP BY type`, []);
230
+ const byStatus = await this.dbAll(`SELECT status, COUNT(*) as count FROM regression_detections GROUP BY status`, []);
231
+ const weekAgo = Date.now() - 7 * 24 * 60 * 60 * 1000;
232
+ const recent = await this.dbGet(`SELECT COUNT(*) as count FROM regression_detections WHERE timestamp >= ?`, [weekAgo]);
233
+ const severityMap = {};
234
+ for (const row of bySeverity) {
235
+ severityMap[row.severity] = row.count;
236
+ }
237
+ const typeMap = {};
238
+ for (const row of byType) {
239
+ typeMap[row.type] = row.count;
240
+ }
241
+ const statusMap = {};
242
+ for (const row of byStatus) {
243
+ statusMap[row.status] = row.count;
244
+ }
245
+ return {
246
+ total: total?.count || 0,
247
+ bySeverity: severityMap,
248
+ byType: typeMap,
249
+ byStatus: statusMap,
250
+ recent: recent?.count || 0
251
+ };
252
+ }
253
+ /**
254
+ * Delete old regressions
255
+ */
256
+ async deleteOldRegressions(olderThanMs) {
257
+ const cutoff = Date.now() - olderThanMs;
258
+ const result = await this.dbRun(`DELETE FROM regression_detections WHERE timestamp < ?`, [cutoff]);
259
+ return result.changes || 0;
260
+ }
261
+ /**
262
+ * Map database rows to RegressionDetection objects
263
+ */
264
+ mapRowsToDetections(rows) {
265
+ return rows.map((r) => ({
266
+ id: r.id,
267
+ runId: r.run_id,
268
+ baselineRunId: r.baseline_run_id,
269
+ type: r.type,
270
+ severity: r.severity,
271
+ status: r.status,
272
+ metricName: r.metric_name,
273
+ baselineValue: r.baseline_value,
274
+ currentValue: r.current_value,
275
+ absoluteChange: r.absolute_change,
276
+ percentChange: r.percent_change,
277
+ significance: r.significance,
278
+ confidence: r.confidence,
279
+ direction: r.direction,
280
+ affectedComponent: r.affected_component,
281
+ gate: r.gate,
282
+ context: r.context_json ? JSON.parse(r.context_json) : undefined,
283
+ suggestions: JSON.parse(r.suggestions_json),
284
+ commits: r.commits_json ? JSON.parse(r.commits_json) : undefined,
285
+ timestamp: r.timestamp,
286
+ notes: r.notes
287
+ }));
288
+ }
289
+ }
@@ -0,0 +1,24 @@
1
+ /**
2
+ * Code Fixer
3
+ *
4
+ * Applies fix suggestions to test files.
5
+ */
6
+ import type { FixSuggestion, FixResult } from '../types.js';
7
+ export interface FixerOptions {
8
+ backup?: boolean;
9
+ backupDir?: string;
10
+ dryRun?: boolean;
11
+ createPatch?: boolean;
12
+ }
13
+ /**
14
+ * Apply fix suggestions to files
15
+ */
16
+ export declare function applyFixes(suggestions: FixSuggestion[], options?: FixerOptions): Promise<FixResult>;
17
+ /**
18
+ * Generate unified diff format
19
+ */
20
+ export declare function generateDiff(original: string, modified: string): string;
21
+ /**
22
+ * Revert changes from backup
23
+ */
24
+ export declare function revertBackup(backupPath: string, originalPath: string): Promise<boolean>;
@@ -0,0 +1,226 @@
1
+ /**
2
+ * Code Fixer
3
+ *
4
+ * Applies fix suggestions to test files.
5
+ */
6
+ import { readFile, writeFile } from 'fs/promises';
7
+ import { join, basename } from 'path';
8
+ import { mkdirSync, existsSync } from 'fs';
9
+ /**
10
+ * Apply fix suggestions to files
11
+ */
12
+ export async function applyFixes(suggestions, options = {}) {
13
+ const changes = [];
14
+ let applied = 0;
15
+ let failed = 0;
16
+ // Group suggestions by file
17
+ const byFile = new Map();
18
+ for (const suggestion of suggestions) {
19
+ if (!byFile.has(suggestion.filePath)) {
20
+ byFile.set(suggestion.filePath, []);
21
+ }
22
+ byFile.get(suggestion.filePath).push(suggestion);
23
+ }
24
+ // Apply fixes per file
25
+ for (const [filePath, fileSuggestions] of byFile.entries()) {
26
+ try {
27
+ const result = await applyFixesToFile(filePath, fileSuggestions, options);
28
+ changes.push(...result.changes);
29
+ applied += result.applied;
30
+ failed += result.failed;
31
+ }
32
+ catch (error) {
33
+ failed++;
34
+ }
35
+ }
36
+ return {
37
+ success: failed === 0,
38
+ suggestions,
39
+ applied,
40
+ failed,
41
+ testsPassed: 0, // Will be updated by test runner
42
+ testsFailed: 0,
43
+ changes
44
+ };
45
+ }
46
+ /**
47
+ * Apply fixes to a single file
48
+ */
49
+ async function applyFixesToFile(filePath, suggestions, options) {
50
+ const fullPath = filePath; // Assume relative to cwd or already absolute
51
+ // Read original content
52
+ const original = await readFile(fullPath, 'utf-8');
53
+ let modified = original;
54
+ const changes = [];
55
+ // Create backup if requested
56
+ let backupPath = '';
57
+ if (options.backup) {
58
+ backupPath = await createBackup(fullPath, options.backupDir);
59
+ }
60
+ // Apply suggestions in reverse line order (to preserve line numbers)
61
+ const sortedSuggestions = [...suggestions].sort((a, b) => b.line - a.line);
62
+ for (const suggestion of sortedSuggestions) {
63
+ try {
64
+ const result = applySuggestion(modified, suggestion);
65
+ if (result.changed) {
66
+ changes.push({
67
+ path: fullPath,
68
+ original: modified,
69
+ modified: result.content,
70
+ backup: backupPath
71
+ });
72
+ modified = result.content;
73
+ }
74
+ }
75
+ catch (error) {
76
+ // Failed to apply this suggestion
77
+ }
78
+ }
79
+ // Write modified content
80
+ if (!options.dryRun && changes.length > 0) {
81
+ await writeFile(fullPath, modified, 'utf-8');
82
+ }
83
+ const applied = changes.length;
84
+ const failed = suggestions.length - applied;
85
+ return { changes, applied, failed };
86
+ }
87
+ /**
88
+ * Apply a single suggestion to content
89
+ */
90
+ function applySuggestion(content, suggestion) {
91
+ const lines = content.split('\n');
92
+ const lineIndex = suggestion.line - 1; // Convert to 0-based
93
+ if (lineIndex < 0 || lineIndex >= lines.length) {
94
+ return { changed: false, content };
95
+ }
96
+ // Apply different fix strategies
97
+ switch (suggestion.type) {
98
+ case 'fix_timeout':
99
+ return insertTimeout(lines, lineIndex, suggestion.code);
100
+ case 'fix_import':
101
+ return insertImport(lines, suggestion.code);
102
+ case 'add_assertion':
103
+ return insertAssertion(lines, lineIndex, suggestion.code);
104
+ case 'update_expectation':
105
+ return updateLine(lines, lineIndex, suggestion.code);
106
+ default:
107
+ return insertCodeAtLine(lines, lineIndex, suggestion.code);
108
+ }
109
+ }
110
+ /**
111
+ * Insert timeout at test level
112
+ */
113
+ function insertTimeout(lines, lineIndex, code) {
114
+ // Find the test function and insert timeout after it
115
+ for (let i = lineIndex; i >= 0 && i >= lineIndex - 5; i--) {
116
+ if (lines[i].includes('it(') || lines[i].includes('test(')) {
117
+ lines[i] = lines[i].replace(/,?\s*\)\s*=>/, `, { timeout: 10000 } =>`);
118
+ return { changed: true, content: lines.join('\n') };
119
+ }
120
+ }
121
+ return { changed: false, content: lines.join('\n') };
122
+ }
123
+ /**
124
+ * Insert import at top of file
125
+ */
126
+ function insertImport(lines, code) {
127
+ // Find last import and insert after it
128
+ let lastImportIndex = -1;
129
+ for (let i = 0; i < lines.length; i++) {
130
+ if (lines[i].trim().startsWith('import ')) {
131
+ lastImportIndex = i;
132
+ }
133
+ }
134
+ if (lastImportIndex >= 0) {
135
+ lines.splice(lastImportIndex + 1, 0, code);
136
+ return { changed: true, content: lines.join('\n') };
137
+ }
138
+ return { changed: false, content: lines.join('\n') };
139
+ }
140
+ /**
141
+ * Insert assertion in test
142
+ */
143
+ function insertAssertion(lines, lineIndex, code) {
144
+ // Insert before the closing brace
145
+ for (let i = lineIndex; i < Math.min(lines.length, lineIndex + 10); i++) {
146
+ if (lines[i].trim() === '});') {
147
+ lines.splice(i, 0, code);
148
+ return { changed: true, content: lines.join('\n') };
149
+ }
150
+ }
151
+ return { changed: false, content: lines.join('\n') };
152
+ }
153
+ /**
154
+ * Update a specific line
155
+ */
156
+ function updateLine(lines, lineIndex, code) {
157
+ if (lineIndex >= 0 && lineIndex < lines.length) {
158
+ lines[lineIndex] = code;
159
+ return { changed: true, content: lines.join('\n') };
160
+ }
161
+ return { changed: false, content: lines.join('\n') };
162
+ }
163
+ /**
164
+ * Insert code at specific line
165
+ */
166
+ function insertCodeAtLine(lines, lineIndex, code) {
167
+ lines.splice(lineIndex, 0, code);
168
+ return { changed: true, content: lines.join('\n') };
169
+ }
170
+ /**
171
+ * Create backup of a file
172
+ */
173
+ async function createBackup(filePath, backupDir) {
174
+ const timestamp = new Date().toISOString().replace(/[:.]/g, '-');
175
+ const backupFileName = `${filePath}.backup-${timestamp}`;
176
+ if (backupDir) {
177
+ if (!existsSync(backupDir)) {
178
+ mkdirSync(backupDir, { recursive: true });
179
+ }
180
+ const backupPath = join(backupDir, basename(filePath));
181
+ await writeFile(backupPath, await readFile(filePath, 'utf-8'));
182
+ return backupPath;
183
+ }
184
+ await writeFile(backupFileName, await readFile(filePath, 'utf-8'));
185
+ return backupFileName;
186
+ }
187
+ /**
188
+ * Generate unified diff format
189
+ */
190
+ export function generateDiff(original, modified) {
191
+ const origLines = original.split('\n');
192
+ const modLines = modified.split('\n');
193
+ const diff = [];
194
+ let i = 0;
195
+ let j = 0;
196
+ while (i < origLines.length || j < modLines.length) {
197
+ if (i < origLines.length && j < modLines.length && origLines[i] === modLines[j]) {
198
+ i++;
199
+ j++;
200
+ }
201
+ else {
202
+ if (i < origLines.length) {
203
+ diff.push(`- ${origLines[i]}`);
204
+ i++;
205
+ }
206
+ if (j < modLines.length) {
207
+ diff.push(`+ ${modLines[j]}`);
208
+ j++;
209
+ }
210
+ }
211
+ }
212
+ return diff.join('\n');
213
+ }
214
+ /**
215
+ * Revert changes from backup
216
+ */
217
+ export async function revertBackup(backupPath, originalPath) {
218
+ try {
219
+ const content = await readFile(backupPath, 'utf-8');
220
+ await writeFile(originalPath, content, 'utf-8');
221
+ return true;
222
+ }
223
+ catch {
224
+ return false;
225
+ }
226
+ }
@@ -0,0 +1,18 @@
1
+ /**
2
+ * Fix Suggestion Engine
3
+ *
4
+ * Generates fix suggestions using LLM.
5
+ */
6
+ import type { FixSuggestion, TestError, RepairOptions } from '../types.js';
7
+ export interface SuggestionOptions extends RepairOptions {
8
+ maxSuggestions?: number;
9
+ includeCode?: boolean;
10
+ }
11
+ /**
12
+ * Generate fix suggestions for test errors
13
+ */
14
+ export declare function generateFixSuggestions(errors: TestError[], testCode: string, sourceCode?: string, options?: SuggestionOptions): Promise<FixSuggestion[]>;
15
+ /**
16
+ * Get quick fix suggestions without LLM
17
+ */
18
+ export declare function getQuickFixes(error: TestError): FixSuggestion[];