qualty 0.1.7 → 0.1.9

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.
@@ -0,0 +1,1091 @@
1
+ import { mkdirSync, writeFileSync, mkdtempSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { tmpdir } from "node:os";
4
+
5
+ async function downloadProjectAttachments({ apiUrl, token, orgId, projectId, specs }) {
6
+ const labelToPath = {};
7
+ if (!projectId || !Array.isArray(specs) || specs.length === 0) return labelToPath;
8
+ const dir = mkdtempSync(join(tmpdir(), "qualty-attach-"));
9
+ const orgHeader = String(orgId || "").trim();
10
+ for (const spec of specs) {
11
+ const fileId = spec.id;
12
+ const label = spec.runner_key || spec.label;
13
+ if (!fileId || !label) continue;
14
+ const response = await fetch(
15
+ `${apiUrl}/api/v1/projects/${projectId}/upload-files/${encodeURIComponent(fileId)}`,
16
+ {
17
+ method: "GET",
18
+ headers: {
19
+ Authorization: `Bearer ${token}`,
20
+ ...(orgHeader ? { "X-Qualty-Org-Id": orgHeader } : {}),
21
+ },
22
+ }
23
+ );
24
+ if (!response.ok) continue;
25
+ const buf = Buffer.from(await response.arrayBuffer());
26
+ const name = spec.original_filename || `${label}.bin`;
27
+ const localPath = join(dir, name.replace(/[^a-zA-Z0-9._-]+/g, "_"));
28
+ writeFileSync(localPath, buf);
29
+ labelToPath[label] = localPath;
30
+ if (spec.label && spec.label !== label) labelToPath[spec.label] = localPath;
31
+ }
32
+ return labelToPath;
33
+ }
34
+
35
+ function attachNetworkCollector(page) {
36
+ const events = [];
37
+ const onRequest = (req) => {
38
+ try {
39
+ events.push({
40
+ url: req.url(),
41
+ method: req.method(),
42
+ resource_type: req.resourceType(),
43
+ failed: false,
44
+ });
45
+ } catch {
46
+ /* ignore */
47
+ }
48
+ };
49
+ const onRequestFailed = (req) => {
50
+ try {
51
+ events.push({
52
+ url: req.url(),
53
+ method: req.method(),
54
+ resource_type: req.resourceType(),
55
+ failed: true,
56
+ });
57
+ } catch {
58
+ /* ignore */
59
+ }
60
+ };
61
+ page.on("request", onRequest);
62
+ page.on("requestfailed", onRequestFailed);
63
+ return {
64
+ snapshot() {
65
+ return { events_app_related: events.slice(-500) };
66
+ },
67
+ };
68
+ }
69
+
70
+ function resolveFileUploadArguments(action, labelToPath) {
71
+ if (!action || !Array.isArray(action.arguments)) return action;
72
+ const method = String(action.method || "").toLowerCase();
73
+ if (method !== "file_upload" && method !== "set_input_files") return action;
74
+ const resolved = action.arguments.map((arg) => {
75
+ const key = String(arg || "").trim();
76
+ if (labelToPath[key]) return labelToPath[key];
77
+ return key;
78
+ });
79
+ return { ...action, arguments: resolved };
80
+ }
81
+
82
+ export async function capturePageContext(page) {
83
+ try {
84
+ const text = await page.evaluate(() => {
85
+ const body = document.body;
86
+ return body ? body.innerText.slice(0, 12000) : "";
87
+ });
88
+ return { page_text: String(text || ""), interactive_summary: "" };
89
+ } catch {
90
+ return { page_text: "", interactive_summary: "" };
91
+ }
92
+ }
93
+
94
+ function _actLog(stepIndex, attempt, message) {
95
+ const stepLabel = Number.isFinite(stepIndex) ? stepIndex + 1 : stepIndex;
96
+ console.log(`[qualty][act][step ${stepLabel}][attempt ${attempt}] ${message}`);
97
+ }
98
+
99
+ function _summarizeAction(action) {
100
+ if (!action || typeof action !== "object") return "no-action";
101
+ const method = String(action.method || "").trim() || "unknown";
102
+ const selector = String(action.selector || "").trim();
103
+ return selector ? `${method} selector=${selector}` : method;
104
+ }
105
+
106
+ async function capturePageContextViaApi({
107
+ page,
108
+ }) {
109
+ return capturePageContext(page);
110
+ }
111
+
112
+ async function _extractInteractiveCandidates(page, limit = 200) {
113
+ try {
114
+ const rows = await page.evaluate((maxRows) => {
115
+ const normalizeText = (v) => (v || "").replace(/\s+/g, " ").trim();
116
+ const getXPath = (element) => {
117
+ if (!element || !element.tagName) return "";
118
+ if (element.id) return `//*[@id="${element.id}"]`;
119
+ if (element === document.body) return "/html/body";
120
+ const parent = element.parentElement;
121
+ if (!parent) return "";
122
+ const tag = element.tagName.toLowerCase();
123
+ const sameTag = Array.from(parent.children).filter((c) => c.tagName === element.tagName);
124
+ const index = sameTag.length > 1 ? `[${sameTag.indexOf(element) + 1}]` : "";
125
+ const parentXPath = getXPath(parent);
126
+ return parentXPath ? `${parentXPath}/${tag}${index}` : `//${tag}${index}`;
127
+ };
128
+ const isVisible = (el) => {
129
+ if (!el) return false;
130
+ const style = window.getComputedStyle(el);
131
+ if (!style) return false;
132
+ if (style.display === "none" || style.visibility === "hidden" || style.opacity === "0") return false;
133
+ const rect = el.getBoundingClientRect();
134
+ return rect.width > 0 && rect.height > 0;
135
+ };
136
+ const inferRole = (el) => {
137
+ const explicit = el.getAttribute("role");
138
+ if (explicit) return explicit;
139
+ const tag = (el.tagName || "").toLowerCase();
140
+ if (tag === "a") return "link";
141
+ if (tag === "button") return "button";
142
+ if (tag === "input") return "textbox";
143
+ if (tag === "select") return "listbox";
144
+ if (tag === "textarea") return "textbox";
145
+ return null;
146
+ };
147
+ const getTestId = (el) =>
148
+ el.getAttribute("data-testid") ||
149
+ el.getAttribute("data-test-id") ||
150
+ el.getAttribute("data-test") ||
151
+ el.getAttribute("data-qa") ||
152
+ null;
153
+
154
+ const selector =
155
+ "a,button,input,select,textarea,[role],[tabindex],[onclick],[contenteditable='true']";
156
+ const nodes = Array.from(document.querySelectorAll(selector));
157
+ const seen = new Set();
158
+
159
+ const out = [];
160
+ for (const el of nodes) {
161
+ if (seen.has(el) || !isVisible(el) || out.length >= maxRows) continue;
162
+ seen.add(el);
163
+ if (!el || out.length >= maxRows) break;
164
+ const xpath = getXPath(el);
165
+ if (!xpath) continue;
166
+ const rect = el.getBoundingClientRect();
167
+ const tag = (el.tagName || "").toLowerCase();
168
+ const role = inferRole(el);
169
+ const text = normalizeText(el.innerText || el.textContent || "").slice(0, 120) || null;
170
+ const className =
171
+ typeof el.className === "string" ? normalizeText(el.className).slice(0, 120) : null;
172
+ const bbox = {
173
+ x: Math.round(rect.x),
174
+ y: Math.round(rect.y),
175
+ width: Math.round(rect.width),
176
+ height: Math.round(rect.height),
177
+ right: Math.round(rect.right),
178
+ bottom: Math.round(rect.bottom),
179
+ };
180
+ const midpoint = {
181
+ x: Math.round(rect.x + rect.width / 2),
182
+ y: Math.round(rect.y + rect.height / 2),
183
+ };
184
+ out.push({
185
+ selector: xpath,
186
+ id: el.id || null,
187
+ tag: tag || null,
188
+ role: role || null,
189
+ text,
190
+ type: el.getAttribute("type") || null,
191
+ title: el.getAttribute("title") || null,
192
+ test_id: getTestId(el),
193
+ name_attr: el.getAttribute("name") || null,
194
+ aria_label: el.getAttribute("aria-label") || null,
195
+ placeholder: el.getAttribute("placeholder") || null,
196
+ href: el.getAttribute("href") || null,
197
+ class_name: className,
198
+ bbox,
199
+ midpoint,
200
+ target_coordinate: `${midpoint.x},${midpoint.y}`,
201
+ });
202
+ }
203
+ return out.slice(0, 400);
204
+ }, Math.max(1, Math.min(500, Number(limit) || 200)));
205
+ return Array.isArray(rows) ? rows : [];
206
+ } catch {
207
+ return [];
208
+ }
209
+ }
210
+
211
+ /**
212
+ * Cloud ActHandler parity: attempt 1 (cache / bundle resolve) → 2 (selector LLM) → 3 (heal).
213
+ */
214
+ async function runActStep({
215
+ apiRequest,
216
+ apiUrl,
217
+ token,
218
+ orgId,
219
+ executionId,
220
+ stepIndex,
221
+ page,
222
+ credentials,
223
+ labelToPath,
224
+ step,
225
+ }) {
226
+ const attemptHistory = [];
227
+ const excluded = [];
228
+ let topCandidatesCache = null;
229
+ let lastErr = null;
230
+ let action = null;
231
+ const maxAttempts = 3;
232
+ let executionSource = null;
233
+
234
+ for (let attempt = 1; attempt <= maxAttempts; attempt += 1) {
235
+ _actLog(stepIndex, attempt, "observe: requesting plan");
236
+ let ctx = await capturePageContextViaApi({
237
+ page,
238
+ });
239
+
240
+ const observeRequest = async (extraBody = null) =>
241
+ apiRequest({
242
+ apiUrl,
243
+ token,
244
+ orgId,
245
+ path: `/api/v1/cli/executions/${executionId}/observe`,
246
+ method: "POST",
247
+ body: {
248
+ step_index: stepIndex,
249
+ attempt,
250
+ page_text: ctx.page_text,
251
+ interactive_summary: ctx.interactive_summary,
252
+ attempt_history: attemptHistory,
253
+ excluded_selectors: excluded,
254
+ top_candidates_cache: topCandidatesCache,
255
+ ...(extraBody && typeof extraBody === "object" ? extraBody : {}),
256
+ },
257
+ });
258
+
259
+ let observe = await observeRequest();
260
+
261
+ const status = String(observe.status || "");
262
+ const source = String(observe.resolution_source || "").trim();
263
+ _actLog(
264
+ stepIndex,
265
+ attempt,
266
+ `observe: status=${status || "unknown"}${source ? ` source=${source}` : ""}`
267
+ );
268
+ if (Array.isArray(observe.top_candidates_cache)) {
269
+ topCandidatesCache = observe.top_candidates_cache;
270
+ }
271
+ if (observe.step_kind === "done" || status === "done") {
272
+ break;
273
+ }
274
+ if (observe.step_kind === "agent" || observe.step_kind === "login" || status === "agent") {
275
+ throw new Error(observe.error || `${observe.step_kind} steps require agent path`);
276
+ }
277
+ if (observe.step_kind === "check" || status === "check") {
278
+ throw new Error(observe.error || "Check steps are not supported in local CLI yet");
279
+ }
280
+ if (status === "heal") {
281
+ _actLog(stepIndex, attempt, "escalating to agent heal");
282
+ const healResult = await runAgentStep({
283
+ apiRequest,
284
+ apiUrl,
285
+ token,
286
+ orgId,
287
+ executionId,
288
+ stepIndex,
289
+ instruction: String(observe.remainder_instruction || ""),
290
+ page,
291
+ credentials,
292
+ labelToPath,
293
+ instructionOverride: true,
294
+ });
295
+ return {
296
+ ok: healResult.ok,
297
+ error: healResult.error,
298
+ action: { method: "agent_heal", description: observe.remainder_instruction || "" },
299
+ skipRemainingSteps: healResult.ok,
300
+ source: "agent_heal",
301
+ };
302
+ }
303
+ if (status === "retry") {
304
+ _actLog(stepIndex, attempt, `retry: ${observe.error || "planner requested retry"}`);
305
+ attemptHistory.push({ attempt, error: observe.error || "retry" });
306
+ continue;
307
+ }
308
+ if (status === "needs_bundle_resolve") {
309
+ const collectedCandidates = await _extractInteractiveCandidates(page, 200);
310
+ _actLog(
311
+ stepIndex,
312
+ attempt,
313
+ `needs_bundle_resolve: collected ${collectedCandidates.length} candidates, retrying observe`
314
+ );
315
+ observe = await observeRequest({
316
+ collected_candidates: collectedCandidates,
317
+ collector_version: "candidate_js_v1",
318
+ });
319
+ if (Array.isArray(observe.top_candidates_cache)) {
320
+ topCandidatesCache = observe.top_candidates_cache;
321
+ }
322
+ const retryStatus = String(observe.status || "");
323
+ const retrySource = String(observe.resolution_source || "").trim();
324
+ _actLog(
325
+ stepIndex,
326
+ attempt,
327
+ `observe(after candidates): status=${retryStatus || "unknown"}${
328
+ retrySource ? ` source=${retrySource}` : ""
329
+ }`
330
+ );
331
+ if (retryStatus === "action" && observe.action) {
332
+ action = resolveFileUploadArguments(observe.action, labelToPath);
333
+ executionSource = retrySource || source || null;
334
+ _actLog(stepIndex, attempt, `execute: ${_summarizeAction(action)} source=${executionSource || "unknown"}`);
335
+ try {
336
+ await executeAction(page, action, credentials);
337
+ _actLog(stepIndex, attempt, "execute: success");
338
+ lastErr = null;
339
+ break;
340
+ } catch (err) {
341
+ lastErr = err.message || String(err);
342
+ _actLog(stepIndex, attempt, `execute: failed error=${lastErr}`);
343
+ attemptHistory.push({
344
+ attempt,
345
+ method: action.method,
346
+ selector: action.selector,
347
+ error: lastErr,
348
+ });
349
+ if (action.selector) excluded.push(action.selector);
350
+ continue;
351
+ }
352
+ }
353
+ if (retryStatus === "retry") {
354
+ _actLog(stepIndex, attempt, `retry(after candidates): ${observe.error || "planner requested retry"}`);
355
+ attemptHistory.push({ attempt, error: observe.error || "retry" });
356
+ continue;
357
+ }
358
+ if (retryStatus === "heal") {
359
+ _actLog(stepIndex, attempt, "escalating to agent heal");
360
+ const healResult = await runAgentStep({
361
+ apiRequest,
362
+ apiUrl,
363
+ token,
364
+ orgId,
365
+ executionId,
366
+ stepIndex,
367
+ instruction: String(observe.remainder_instruction || ""),
368
+ page,
369
+ credentials,
370
+ labelToPath,
371
+ instructionOverride: true,
372
+ });
373
+ return {
374
+ ok: healResult.ok,
375
+ error: healResult.error,
376
+ action: { method: "agent_heal", description: observe.remainder_instruction || "" },
377
+ skipRemainingSteps: healResult.ok,
378
+ source: "agent_heal",
379
+ };
380
+ }
381
+ lastErr = observe.error || "No action returned after bundle resolve";
382
+ _actLog(stepIndex, attempt, `retry(after candidates): ${lastErr}`);
383
+ attemptHistory.push({ attempt, error: lastErr });
384
+ continue;
385
+ }
386
+ if (status === "action" && observe.action) {
387
+ action = resolveFileUploadArguments(observe.action, labelToPath);
388
+ executionSource = source || null;
389
+ _actLog(stepIndex, attempt, `execute: ${_summarizeAction(action)} source=${executionSource || "unknown"}`);
390
+ try {
391
+ await executeAction(page, action, credentials);
392
+ _actLog(stepIndex, attempt, "execute: success");
393
+ lastErr = null;
394
+ break;
395
+ } catch (err) {
396
+ lastErr = err.message || String(err);
397
+ _actLog(stepIndex, attempt, `execute: failed error=${lastErr}`);
398
+ attemptHistory.push({
399
+ attempt,
400
+ method: action.method,
401
+ selector: action.selector,
402
+ error: lastErr,
403
+ });
404
+ if (action.selector) excluded.push(action.selector);
405
+ continue;
406
+ }
407
+ }
408
+ lastErr = observe.error || "No action returned";
409
+ _actLog(stepIndex, attempt, `retry: ${lastErr}`);
410
+ attemptHistory.push({ attempt, error: lastErr });
411
+ }
412
+
413
+ return {
414
+ ok: !lastErr,
415
+ error: lastErr,
416
+ action,
417
+ skipRemainingSteps: false,
418
+ source: executionSource,
419
+ };
420
+ }
421
+
422
+ async function clickAtCoordinate(page, coord, button = "left") {
423
+ const [x, y] = coord.split(",").map((v) => Number(v.trim()));
424
+ if (!Number.isFinite(x) || !Number.isFinite(y)) {
425
+ throw new Error(`Invalid target_coordinate: ${coord}`);
426
+ }
427
+ if (button === "right") {
428
+ await page.mouse.click(x, y, { button: "right" });
429
+ } else {
430
+ await page.mouse.click(x, y);
431
+ }
432
+ }
433
+
434
+ async function resolveFileInputAtCoordinate(page, x, y) {
435
+ return page.evaluate(
436
+ ({ px, py }) => {
437
+ const el = document.elementFromPoint(px, py);
438
+ if (!el) return null;
439
+ const walk = (node) => {
440
+ if (!node) return null;
441
+ if (node.tagName && String(node.tagName).toLowerCase() === "input") {
442
+ const t = (node.getAttribute("type") || "").toLowerCase();
443
+ if (t === "file") return node;
444
+ }
445
+ const label = node.closest && node.closest("label");
446
+ if (label && label.control && label.control.type === "file") return label.control;
447
+ return node.parentElement ? walk(node.parentElement) : null;
448
+ };
449
+ const found = walk(el);
450
+ if (!found) return null;
451
+ if (found.id) return `#${found.id}`;
452
+ return null;
453
+ },
454
+ { px: x, py: y }
455
+ );
456
+ }
457
+
458
+ export async function executeAction(page, action, credentials = null) {
459
+ const method = String(action.method || "click").toLowerCase();
460
+ const selector = String(action.selector || "").trim();
461
+ const args = Array.isArray(action.arguments) ? action.arguments.map(String) : [];
462
+ const coord = String(action.target_coordinate || "").trim();
463
+
464
+ if (method === "wait") {
465
+ const ms = Math.max(0, Number(args[0] || 500));
466
+ await page.waitForTimeout(ms);
467
+ return;
468
+ }
469
+
470
+ if (method === "scrollto" && coord) {
471
+ const [x, y] = coord.split(",").map((v) => Number(v.trim()));
472
+ await page.evaluate(
473
+ ({ x, y }) => window.scrollTo(Number(x) || 0, Number(y) || 0),
474
+ { x, y }
475
+ );
476
+ return;
477
+ }
478
+
479
+ if (method === "scroll") {
480
+ const [x, y] = coord
481
+ ? coord.split(",").map((v) => Number(v.trim()))
482
+ : [null, null];
483
+ const deltaX = Number(args[0] || 0);
484
+ const deltaY = Number(args[1] || 700);
485
+ if (Number.isFinite(x) && Number.isFinite(y)) {
486
+ await page.mouse.move(x, y);
487
+ }
488
+ await page.mouse.wheel(deltaX, deltaY);
489
+ return;
490
+ }
491
+
492
+ if (method === "focused_keypress") {
493
+ for (const key of args) {
494
+ if (key) await page.keyboard.press(key);
495
+ }
496
+ return;
497
+ }
498
+
499
+ if (method === "fill_sensitive_field" && credentials) {
500
+ const fieldType = String(action.field_type || "").toLowerCase();
501
+ const value =
502
+ fieldType === "password"
503
+ ? credentials.password || ""
504
+ : credentials.username || "";
505
+ if (!value) throw new Error(`No ${fieldType} credential available`);
506
+ if (coord) await clickAtCoordinate(page, coord);
507
+ await page.waitForTimeout(200);
508
+ const isMac = process.platform === "darwin";
509
+ const selectAll = isMac ? "Meta+A" : "Control+A";
510
+ await page.keyboard.press(selectAll);
511
+ await page.keyboard.press("Backspace");
512
+ await page.keyboard.type(value, { delay: 40 });
513
+ if (action.press_enter) await page.keyboard.press("Enter");
514
+ return;
515
+ }
516
+
517
+ const coordinateMethods = new Set([
518
+ "click",
519
+ "dblclick",
520
+ "rightclick",
521
+ "hover",
522
+ "dragdrop",
523
+ ]);
524
+ if (coord && coordinateMethods.has(method)) {
525
+ if (method === "click") {
526
+ await clickAtCoordinate(page, coord, "left");
527
+ return;
528
+ }
529
+ if (method === "rightclick") {
530
+ await clickAtCoordinate(page, coord, "right");
531
+ return;
532
+ }
533
+ if (method === "dblclick") {
534
+ const [x, y] = coord.split(",").map((v) => Number(v.trim()));
535
+ await page.mouse.dblclick(x, y);
536
+ return;
537
+ }
538
+ if (method === "hover") {
539
+ const [x, y] = coord.split(",").map((v) => Number(v.trim()));
540
+ await page.mouse.move(x, y);
541
+ return;
542
+ }
543
+ if (method === "dragdrop") {
544
+ const [sx, sy] = coord.split(",").map((v) => Number(v.trim()));
545
+ const ex = Number(args[0]);
546
+ const ey = Number(args[1]);
547
+ await page.mouse.move(sx, sy);
548
+ await page.mouse.down();
549
+ await page.mouse.move(ex, ey);
550
+ await page.mouse.up();
551
+ return;
552
+ }
553
+ }
554
+
555
+ if (method === "file_upload" && coord && !selector) {
556
+ const [x, y] = coord.split(",").map((v) => Number(v.trim()));
557
+ const sel = await resolveFileInputAtCoordinate(page, x, y);
558
+ if (!sel) throw new Error("file_upload: no file input at coordinate");
559
+ await page.locator(sel).first().setInputFiles(args, { timeout: 30000 });
560
+ return;
561
+ }
562
+
563
+ if (method === "type" && !selector && coord) {
564
+ await clickAtCoordinate(page, coord);
565
+ await page.waitForTimeout(200);
566
+ await page.keyboard.type(args[0] || "", { delay: 40 });
567
+ return;
568
+ }
569
+
570
+ if (method === "type" && !selector && !coord) {
571
+ await page.keyboard.type(args[0] || "", { delay: 40 });
572
+ return;
573
+ }
574
+
575
+ if (!selector && method !== "focused_keypress") {
576
+ throw new Error(`Action ${method} requires a selector or target_coordinate`);
577
+ }
578
+
579
+ const locator = page.locator(selector).first();
580
+
581
+ switch (method) {
582
+ case "click":
583
+ await locator.click({ timeout: 15000 });
584
+ break;
585
+ case "dblclick":
586
+ await locator.dblclick({ timeout: 15000 });
587
+ break;
588
+ case "rightclick":
589
+ await locator.click({ button: "right", timeout: 15000 });
590
+ break;
591
+ case "hover":
592
+ await locator.hover({ timeout: 15000 });
593
+ break;
594
+ case "fill":
595
+ case "type":
596
+ await locator.fill(args[0] || "", { timeout: 15000 });
597
+ break;
598
+ case "press":
599
+ for (const key of args) {
600
+ if (key) await locator.press(key, { timeout: 15000 });
601
+ }
602
+ break;
603
+ case "file_upload":
604
+ case "set_input_files":
605
+ await locator.setInputFiles(args, { timeout: 30000 });
606
+ break;
607
+ case "scroll":
608
+ await locator.scrollIntoViewIfNeeded({ timeout: 15000 });
609
+ break;
610
+ case "scrollto":
611
+ await locator.scrollIntoViewIfNeeded({ timeout: 15000 });
612
+ break;
613
+ case "dragdrop": {
614
+ const box = await locator.boundingBox();
615
+ if (!box) throw new Error("dragdrop: element not visible");
616
+ const sx = box.x + box.width / 2;
617
+ const sy = box.y + box.height / 2;
618
+ const ex = Number(args[0]);
619
+ const ey = Number(args[1]);
620
+ await page.mouse.move(sx, sy);
621
+ await page.mouse.down();
622
+ await page.mouse.move(ex, ey);
623
+ await page.mouse.up();
624
+ break;
625
+ }
626
+ default:
627
+ throw new Error(`Unsupported action method: ${method}`);
628
+ }
629
+ }
630
+
631
+ async function runAgentStep({
632
+ apiRequest,
633
+ apiUrl,
634
+ token,
635
+ orgId,
636
+ executionId,
637
+ stepIndex,
638
+ instruction,
639
+ page,
640
+ credentials,
641
+ labelToPath,
642
+ maxTurns = 100,
643
+ instructionOverride = false,
644
+ }) {
645
+ let reset = true;
646
+ let lastErr = null;
647
+ let turnCount = 0;
648
+ const pendingFunctionResults = [];
649
+
650
+ while (turnCount < maxTurns) {
651
+ const screenshotB64 = (await page.screenshot({ type: "png" })).toString("base64");
652
+ const turnBody = {
653
+ step_index: stepIndex,
654
+ screenshot_b64: screenshotB64,
655
+ page_url: page.url(),
656
+ reset,
657
+ function_results: pendingFunctionResults.length ? pendingFunctionResults : undefined,
658
+ };
659
+ if (instructionOverride && instruction) {
660
+ turnBody.instruction_override = instruction;
661
+ }
662
+ const turn = await apiRequest({
663
+ apiUrl,
664
+ token,
665
+ orgId,
666
+ path: `/api/v1/cli/executions/${executionId}/agent-turn`,
667
+ method: "POST",
668
+ body: turnBody,
669
+ });
670
+ reset = false;
671
+ pendingFunctionResults.length = 0;
672
+ turnCount += 1;
673
+
674
+ if (turn.status === "error") {
675
+ lastErr = turn.error || "Agent turn failed";
676
+ break;
677
+ }
678
+ if (turn.status === "done") {
679
+ lastErr = null;
680
+ break;
681
+ }
682
+ if (turn.status === "continue") {
683
+ continue;
684
+ }
685
+ if (turn.status !== "actions" || !Array.isArray(turn.actions) || !turn.actions.length) {
686
+ lastErr = turn.error || "Agent returned no actions";
687
+ break;
688
+ }
689
+
690
+ for (const raw of turn.actions) {
691
+ const action = resolveFileUploadArguments(raw, labelToPath);
692
+ let actionOk = true;
693
+ let actionErr = null;
694
+ try {
695
+ if (action.method === "fill_sensitive_field") {
696
+ await executeAction(page, action, credentials);
697
+ const fnId = action._function_call_id;
698
+ if (fnId) {
699
+ pendingFunctionResults.push({
700
+ call_id: fnId,
701
+ success: true,
702
+ message: "fill_sensitive_field executed locally",
703
+ });
704
+ }
705
+ } else {
706
+ await executeAction(page, action, credentials);
707
+ }
708
+ await page.waitForTimeout(200);
709
+ } catch (err) {
710
+ actionOk = false;
711
+ actionErr = err.message || String(err);
712
+ lastErr = actionErr;
713
+ }
714
+
715
+ const afterB64 = (await page.screenshot({ type: "png" })).toString("base64");
716
+ try {
717
+ await apiRequest({
718
+ apiUrl,
719
+ token,
720
+ orgId,
721
+ path: `/api/v1/cli/executions/${executionId}/agent-segment`,
722
+ method: "POST",
723
+ body: {
724
+ step_index: stepIndex,
725
+ turn: turnCount,
726
+ turn_thought: turn.turn_thought || "",
727
+ action,
728
+ executor_ok: actionOk,
729
+ error: actionErr || undefined,
730
+ screenshot_b64: afterB64,
731
+ page_url: page.url(),
732
+ },
733
+ });
734
+ } catch (segErr) {
735
+ console.warn("[CLI] agent-segment upload failed:", segErr.message || segErr);
736
+ }
737
+
738
+ if (!actionOk) break;
739
+ }
740
+ if (lastErr) break;
741
+ }
742
+
743
+ if (!lastErr && turnCount >= maxTurns) {
744
+ lastErr = `Agent exceeded ${maxTurns} turns`;
745
+ }
746
+ return { ok: !lastErr, error: lastErr, turns: turnCount };
747
+ }
748
+
749
+ export async function runLocalExecution({
750
+ apiRequest,
751
+ apiUrl,
752
+ token,
753
+ orgId,
754
+ executionId,
755
+ device,
756
+ headless,
757
+ }) {
758
+ const { chromium } = await import("playwright");
759
+
760
+ const claim = await apiRequest({
761
+ apiUrl,
762
+ token,
763
+ orgId,
764
+ path: `/api/v1/cli/executions/${executionId}/claim`,
765
+ method: "POST",
766
+ body: { device },
767
+ });
768
+
769
+ const url = claim.url;
770
+ const viewport = claim.viewport || { width: 1440, height: 900 };
771
+ const steps = Array.isArray(claim.steps) ? claim.steps : [];
772
+ const totalSteps = claim.total_steps || steps.length;
773
+
774
+ const browserServer = await chromium.launchServer({
775
+ headless: headless !== false,
776
+ host: "127.0.0.1",
777
+ });
778
+ const cdpEndpoint = browserServer.wsEndpoint();
779
+ const browser = await chromium.connect(cdpEndpoint);
780
+ const context = await browser.newContext({
781
+ viewport: { width: viewport.width || 1440, height: viewport.height || 900 },
782
+ });
783
+ const page = await context.newPage();
784
+ const networkCollector = attachNetworkCollector(page);
785
+ const labelToPath = await downloadProjectAttachments({
786
+ apiUrl,
787
+ token,
788
+ orgId,
789
+ projectId: claim.project_id,
790
+ specs: claim.file_attachments,
791
+ });
792
+ const credentials = claim.credentials || null;
793
+
794
+ const stepsJson = [];
795
+ let runSuccess = true;
796
+ let runError = null;
797
+ const started = Date.now();
798
+ let lastPrevScreenshotB64 = null;
799
+ let lastCurrScreenshotB64 = null;
800
+ let firstPrevScreenshotB64 = null;
801
+
802
+ try {
803
+ await page.goto(url, { waitUntil: "domcontentloaded", timeout: 60000 });
804
+ await page.waitForTimeout(500);
805
+
806
+ for (let stepIndex = 0; stepIndex < totalSteps; stepIndex += 1) {
807
+ const step = steps[stepIndex] || {};
808
+ const kind = String(step.kind || "act").toLowerCase();
809
+ const instruction = step.instruction || step.name || `Step ${stepIndex + 1}`;
810
+ let action = null;
811
+ let lastErr = null;
812
+ const prevBuf = await page.screenshot({ type: "png" });
813
+
814
+ if (kind === "agent" || kind === "login") {
815
+ console.log(`[qualty][agent][step ${stepIndex + 1}] mode=pure_agent kind=${kind}`);
816
+ const agentResult = await runAgentStep({
817
+ apiRequest,
818
+ apiUrl,
819
+ token,
820
+ orgId,
821
+ executionId,
822
+ stepIndex,
823
+ instruction,
824
+ page,
825
+ credentials,
826
+ labelToPath,
827
+ });
828
+ lastErr = agentResult.ok ? null : agentResult.error;
829
+ action = { method: "agent", description: instruction };
830
+ } else if (kind === "check") {
831
+ lastErr = "Check steps are not supported in local CLI yet; use cloud run or act steps.";
832
+ } else {
833
+ console.log(`[qualty][act][step ${stepIndex + 1}] mode=act`);
834
+ const actResult = await runActStep({
835
+ apiRequest,
836
+ apiUrl,
837
+ token,
838
+ orgId,
839
+ executionId,
840
+ stepIndex,
841
+ page,
842
+ credentials,
843
+ labelToPath,
844
+ step,
845
+ });
846
+ lastErr = actResult.ok ? null : actResult.error;
847
+ action = actResult.action;
848
+ if (actResult.source) {
849
+ console.log(`[qualty][act][step ${stepIndex + 1}] resolved_source=${actResult.source}`);
850
+ }
851
+ if (actResult.skipRemainingSteps && actResult.ok) {
852
+ const executorOk = true;
853
+ const currBuf = await page.screenshot({ type: "png" });
854
+ const prevB64 = prevBuf.toString("base64");
855
+ const currB64 = currBuf.toString("base64");
856
+ lastPrevScreenshotB64 = prevB64;
857
+ lastCurrScreenshotB64 = currB64;
858
+ if (firstPrevScreenshotB64 == null) firstPrevScreenshotB64 = prevB64;
859
+ let row = {
860
+ name: instruction,
861
+ description: instruction,
862
+ status: "passed",
863
+ action: action
864
+ ? `${action.method}${action.selector ? ` ${action.selector}` : ""}`
865
+ : "agent_heal",
866
+ };
867
+ const stepResp = await apiRequest({
868
+ apiUrl,
869
+ token,
870
+ orgId,
871
+ path: `/api/v1/cli/executions/${executionId}/step-result`,
872
+ method: "POST",
873
+ body: {
874
+ step_index: stepIndex,
875
+ executor_ok: executorOk,
876
+ action,
877
+ result: row,
878
+ prev_screenshot_b64: prevB64,
879
+ curr_screenshot_b64: currB64,
880
+ },
881
+ });
882
+ if (stepResp && stepResp.result && typeof stepResp.result === "object") {
883
+ row = { ...row, ...stepResp.result };
884
+ }
885
+ stepsJson.push(row);
886
+ break;
887
+ }
888
+ }
889
+
890
+ const executorOk = !lastErr;
891
+ const currBuf = await page.screenshot({ type: "png" });
892
+ const prevB64 = prevBuf.toString("base64");
893
+ const currB64 = currBuf.toString("base64");
894
+ lastPrevScreenshotB64 = prevB64;
895
+ lastCurrScreenshotB64 = currB64;
896
+ if (firstPrevScreenshotB64 == null) firstPrevScreenshotB64 = prevB64;
897
+
898
+ let row = {
899
+ name: instruction,
900
+ description: instruction,
901
+ status: executorOk ? "passed" : "failed",
902
+ action: action
903
+ ? `${action.method}${action.selector ? ` ${action.selector}` : ""}`
904
+ : "",
905
+ error: lastErr || undefined,
906
+ };
907
+
908
+ const stepResp = await apiRequest({
909
+ apiUrl,
910
+ token,
911
+ orgId,
912
+ path: `/api/v1/cli/executions/${executionId}/step-result`,
913
+ method: "POST",
914
+ body: {
915
+ step_index: stepIndex,
916
+ executor_ok: executorOk,
917
+ action,
918
+ result: row,
919
+ prev_screenshot_b64: prevB64,
920
+ curr_screenshot_b64: currB64,
921
+ },
922
+ });
923
+
924
+ if (stepResp && stepResp.result && typeof stepResp.result === "object") {
925
+ row = { ...row, ...stepResp.result };
926
+ }
927
+ const passed = row.status === "passed";
928
+ if (!passed) runSuccess = false;
929
+ stepsJson.push(row);
930
+
931
+ if (!passed) {
932
+ runError = lastErr || row.step_thought || "Step failed";
933
+ break;
934
+ }
935
+ await page.waitForTimeout(300);
936
+ }
937
+
938
+ try {
939
+ lastCurrScreenshotB64 = (await page.screenshot({ type: "png" })).toString("base64");
940
+ } catch {
941
+ /* keep last step screenshot */
942
+ }
943
+ } finally {
944
+ await context.close().catch(() => {});
945
+ await browser.close().catch(() => {});
946
+ await browserServer.close().catch(() => {});
947
+ }
948
+
949
+ const runtimeSec = (Date.now() - started) / 1000;
950
+ const completeResp = await apiRequest({
951
+ apiUrl,
952
+ token,
953
+ orgId,
954
+ path: `/api/v1/cli/executions/${executionId}/complete`,
955
+ method: "POST",
956
+ body: {
957
+ success: runSuccess,
958
+ steps_json: stepsJson,
959
+ explanation: runSuccess ? "Local run completed" : runError || "Local run failed",
960
+ runtime_sec: runtimeSec,
961
+ error_message: runError,
962
+ final_screenshot_b64: lastCurrScreenshotB64,
963
+ prev_screenshot_b64: firstPrevScreenshotB64 || lastPrevScreenshotB64,
964
+ network_report: networkCollector.snapshot(),
965
+ },
966
+ });
967
+
968
+ if (completeResp && completeResp.status) {
969
+ runSuccess = completeResp.status === "completed";
970
+ }
971
+
972
+ return { success: runSuccess, stepsJson, error: runError };
973
+ }
974
+
975
+ export async function runLocalCi({
976
+ apiUrl,
977
+ token,
978
+ orgId,
979
+ projectId,
980
+ suiteId,
981
+ explicitIds,
982
+ resolveSavedJobs,
983
+ apiRequest,
984
+ failOnFailure,
985
+ device,
986
+ headless,
987
+ localConcurrency = 4,
988
+ }) {
989
+ const savedJobs = await resolveSavedJobs({ apiUrl, token, orgId, projectId, suiteId, explicitIds });
990
+ if (!Array.isArray(savedJobs) || savedJobs.length === 0) {
991
+ throw new Error("No saved tests matched your selector.");
992
+ }
993
+
994
+ const startResp = await apiRequest({
995
+ apiUrl,
996
+ token,
997
+ orgId,
998
+ path: "/api/v1/ci/run",
999
+ method: "POST",
1000
+ body: {
1001
+ project_id: projectId ? String(projectId) : undefined,
1002
+ suite_id: suiteId ? String(suiteId) : undefined,
1003
+ job_ids: explicitIds.length > 0 ? explicitIds : savedJobs.map((j) => j.id),
1004
+ execution_backend: "cli_local",
1005
+ },
1006
+ });
1007
+
1008
+ const runs = Array.isArray(startResp.runs) ? startResp.runs : [];
1009
+ const started = runs.filter((r) => r.execution_job_id);
1010
+ if (started.length === 0) throw new Error("Failed to start any local runs.");
1011
+
1012
+ const concurrency = Math.max(1, Number(localConcurrency) || 4);
1013
+ const workerCount = Math.min(concurrency, started.length);
1014
+
1015
+ let passed = 0;
1016
+ let failed = 0;
1017
+
1018
+ // eslint-disable-next-line no-console
1019
+ console.log(`[qualty] Running ${started.length} local run(s) with concurrency=${workerCount}`);
1020
+
1021
+ let cursor = 0;
1022
+ const worker = async (workerId) => {
1023
+ while (true) {
1024
+ const idx = cursor;
1025
+ cursor += 1;
1026
+ const run = started[idx];
1027
+ if (!run) return;
1028
+
1029
+ const id = run.execution_job_id;
1030
+ // eslint-disable-next-line no-console
1031
+ console.log(
1032
+ `[qualty][worker ${workerId}] Local run ${id} (${run.saved_job_name || run.saved_job_id})`
1033
+ );
1034
+ try {
1035
+ const result = await runLocalExecution({
1036
+ apiRequest,
1037
+ apiUrl,
1038
+ token,
1039
+ orgId,
1040
+ executionId: id,
1041
+ device,
1042
+ headless,
1043
+ });
1044
+ if (result.success) {
1045
+ passed += 1;
1046
+ // eslint-disable-next-line no-console
1047
+ console.log(`[qualty] ${id} => passed`);
1048
+ } else {
1049
+ failed += 1;
1050
+ // eslint-disable-next-line no-console
1051
+ console.log(`[qualty] ${id} => failed: ${result.error || "unknown"}`);
1052
+ }
1053
+ } catch (err) {
1054
+ failed += 1;
1055
+ // eslint-disable-next-line no-console
1056
+ console.error(`[qualty] ${id} => error: ${err.message || err}`);
1057
+ try {
1058
+ await apiRequest({
1059
+ apiUrl,
1060
+ token,
1061
+ orgId,
1062
+ path: `/api/v1/cli/executions/${id}/complete`,
1063
+ method: "POST",
1064
+ body: {
1065
+ success: false,
1066
+ steps_json: [],
1067
+ explanation: String(err.message || err),
1068
+ runtime_sec: 0,
1069
+ error_message: String(err.message || err),
1070
+ },
1071
+ });
1072
+ } catch {
1073
+ /* best effort */
1074
+ }
1075
+ }
1076
+ }
1077
+ };
1078
+
1079
+ await Promise.all(
1080
+ Array.from({ length: workerCount }, (_, idx) => worker(idx + 1))
1081
+ );
1082
+
1083
+ if (started.length > 1) {
1084
+ // eslint-disable-next-line no-console
1085
+ console.log(`[qualty] Parallel local runs complete (${started.length} total).`);
1086
+ }
1087
+
1088
+ // eslint-disable-next-line no-console
1089
+ console.log(`[qualty] Local summary: ${passed} passed, ${failed} failed.`);
1090
+ if (failOnFailure && failed > 0) process.exit(1);
1091
+ }