qualty 0.1.7 → 0.1.8

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
+ }
package/bin/qualty.js CHANGED
@@ -3,6 +3,7 @@
3
3
  import { appendFileSync } from "node:fs";
4
4
  import { spawn } from "node:child_process";
5
5
  import process from "node:process";
6
+ import { runLocalCi } from "./local-runner.js";
6
7
 
7
8
  /** When unset, CLI uses Qualty development API. Override for production: QUALTY_API_URL=https://qualty-api-production.up.railway.app or local: http://localhost:8000 */
8
9
  const DEFAULT_QUALTY_API_URL = "https://qualty-api-development.up.railway.app";
@@ -38,9 +39,10 @@ function usage() {
38
39
  console.log(
39
40
  [
40
41
  "Usage:",
41
- " qualty connect --project <project-id> [--port 3000] [--api https://your-api] [--token <bearer-token>]",
42
- " qualty run --project <project-id> [--suite-id <suite-id>] [--ids <id1,id2>] [--api https://your-api] [--token <bearer-token>]",
42
+ " qualty connect --project <project-id> [--port 3000] [--api https://your-api] [--token <bearer-token>] [--org <org-id>]",
43
+ " qualty run --project <project-id> [--suite-id <suite-id>] [--ids <id1,id2>] [--local] [--api https://your-api] [--token <bearer-token>] [--org <org-id>]",
43
44
  " [--poll-interval 5] [--timeout 30] [--fail-on-failure true] [--no-view-logs]",
45
+ " [--local] [--device 1440x900] [--headed] [--local-concurrency 4]",
44
46
  "",
45
47
  " After each run, logs include step results, explanation, and final evaluator output (dashboard \"View logs\" data).",
46
48
  " Pass --no-view-logs for a short log only (e.g. very large evaluator text).",
@@ -49,16 +51,20 @@ function usage() {
49
51
  "Env vars:",
50
52
  ` QUALTY_API_URL Backend API URL (default: ${DEFAULT_QUALTY_API_URL}; set for local/self-hosted)`,
51
53
  " QUALTY_API_TOKEN Bearer token used for auth",
54
+ " QUALTY_ORG_ID Org context for scoped API requests",
55
+ " QUALTY_LOCAL_CONCURRENCY Local parallel workers for --local runs (default: 4)",
52
56
  ].join("\n")
53
57
  );
54
58
  }
55
59
 
56
- async function apiRequest({ apiUrl, token, path, method = "GET", body }) {
60
+ async function apiRequest({ apiUrl, token, orgId, path, method = "GET", body }) {
61
+ const normalizedOrgId = String(orgId || "").trim();
57
62
  const response = await fetch(`${apiUrl}${path}`, {
58
63
  method,
59
64
  headers: {
60
65
  "Content-Type": "application/json",
61
66
  Authorization: `Bearer ${token}`,
67
+ ...(normalizedOrgId ? { "X-Qualty-Org-Id": normalizedOrgId } : {}),
62
68
  },
63
69
  body: body ? JSON.stringify(body) : undefined,
64
70
  });
@@ -97,6 +103,7 @@ async function runConnect(args) {
97
103
  const port = Number(args.port || 3000);
98
104
  const apiUrl = resolveApiUrl(args);
99
105
  const token = args.token || process.env.QUALTY_API_TOKEN;
106
+ const orgId = args.org || process.env.QUALTY_ORG_ID;
100
107
 
101
108
  if (!projectId || !token) {
102
109
  usage();
@@ -109,6 +116,7 @@ async function runConnect(args) {
109
116
  const connect = await apiRequest({
110
117
  apiUrl,
111
118
  token,
119
+ orgId,
112
120
  path: "/api/v1/localhost/connect",
113
121
  method: "POST",
114
122
  body: { project_id: projectId, port },
@@ -129,6 +137,7 @@ async function runConnect(args) {
129
137
  await apiRequest({
130
138
  apiUrl,
131
139
  token,
140
+ orgId,
132
141
  path: "/api/v1/localhost/heartbeat",
133
142
  method: "POST",
134
143
  body: { project_id: projectId },
@@ -145,6 +154,7 @@ async function runConnect(args) {
145
154
  await apiRequest({
146
155
  apiUrl,
147
156
  token,
157
+ orgId,
148
158
  path: "/api/v1/localhost/disconnect",
149
159
  method: "POST",
150
160
  body: { project_id: projectId },
@@ -173,6 +183,17 @@ function parseBoolean(value, defaultValue) {
173
183
  throw new Error(`Invalid boolean value: ${value}`);
174
184
  }
175
185
 
186
+ function resolveLocalConcurrency(args) {
187
+ const raw = args["local-concurrency"] ?? process.env.QUALTY_LOCAL_CONCURRENCY ?? 4;
188
+ const n = Number(raw);
189
+ if (!Number.isInteger(n) || n < 1) {
190
+ throw new Error(
191
+ `Invalid local concurrency: ${raw}. Use --local-concurrency <integer>=1 or QUALTY_LOCAL_CONCURRENCY>=1`
192
+ );
193
+ }
194
+ return n;
195
+ }
196
+
176
197
  function sleep(ms) {
177
198
  return new Promise((resolve) => setTimeout(resolve, ms));
178
199
  }
@@ -500,6 +521,7 @@ function printQualtyViewLogsReport(executionId, status) {
500
521
  async function runCi(args) {
501
522
  const apiUrl = resolveApiUrl(args);
502
523
  const token = args.token || process.env.QUALTY_API_TOKEN;
524
+ const orgId = args.org || process.env.QUALTY_ORG_ID;
503
525
  const projectId = args.project;
504
526
  const suiteId = args["suite-id"];
505
527
  const explicitIds = String(args.ids || "")
@@ -524,7 +546,7 @@ async function runCi(args) {
524
546
  throw new Error("Invalid --timeout. Must be > 0.");
525
547
  }
526
548
 
527
- const savedJobs = await resolveSavedJobs({ apiUrl, token, projectId, suiteId, explicitIds });
549
+ const savedJobs = await resolveSavedJobs({ apiUrl, token, orgId, projectId, suiteId, explicitIds });
528
550
  if (!Array.isArray(savedJobs) || savedJobs.length === 0) {
529
551
  throw new Error(
530
552
  "No saved tests matched your selector. The test ID may be invalid, deleted, or outside your token authorization scope."
@@ -546,6 +568,7 @@ async function runCi(args) {
546
568
  const startResp = await apiRequest({
547
569
  apiUrl,
548
570
  token,
571
+ orgId,
549
572
  path: "/api/v1/ci/run",
550
573
  method: "POST",
551
574
  body: startPayload,
@@ -568,6 +591,7 @@ async function runCi(args) {
568
591
  const batch = await apiRequest({
569
592
  apiUrl,
570
593
  token,
594
+ orgId,
571
595
  path: "/api/v1/status/batch",
572
596
  method: "POST",
573
597
  body: { job_ids: executionJobIds },
@@ -633,7 +657,7 @@ async function runCi(args) {
633
657
  }
634
658
  }
635
659
 
636
- async function resolveSavedJobs({ apiUrl, token, projectId, suiteId, explicitIds }) {
660
+ async function resolveSavedJobs({ apiUrl, token, orgId, projectId, suiteId, explicitIds }) {
637
661
  const query = new URLSearchParams();
638
662
  if (projectId) query.set("project_id", String(projectId));
639
663
  if (suiteId) query.set("suite_id", String(suiteId));
@@ -641,6 +665,7 @@ async function resolveSavedJobs({ apiUrl, token, projectId, suiteId, explicitIds
641
665
  return apiRequest({
642
666
  apiUrl,
643
667
  token,
668
+ orgId,
644
669
  path: `/api/v1/saved-jobs?${query.toString()}`,
645
670
  });
646
671
  }
@@ -648,6 +673,7 @@ async function resolveSavedJobs({ apiUrl, token, projectId, suiteId, explicitIds
648
673
  async function runResolve(args) {
649
674
  const apiUrl = resolveApiUrl(args);
650
675
  const token = args.token || process.env.QUALTY_API_TOKEN;
676
+ const orgId = args.org || process.env.QUALTY_ORG_ID;
651
677
  const projectId = args.project;
652
678
  const suiteId = args["suite-id"];
653
679
  const explicitIds = String(args.ids || "")
@@ -663,7 +689,7 @@ async function runResolve(args) {
663
689
  throw new Error("Provide --project, --suite-id, or --ids to select tests.");
664
690
  }
665
691
 
666
- const savedJobs = await resolveSavedJobs({ apiUrl, token, projectId, suiteId, explicitIds });
692
+ const savedJobs = await resolveSavedJobs({ apiUrl, token, orgId, projectId, suiteId, explicitIds });
667
693
  if (!Array.isArray(savedJobs) || savedJobs.length === 0) {
668
694
  throw new Error(
669
695
  "No saved tests matched your selector. The test ID may be invalid, deleted, or outside your token authorization scope."
@@ -699,6 +725,26 @@ async function main() {
699
725
  return;
700
726
  }
701
727
  if (command === "run") {
728
+ if (parseBoolean(args.local, false)) {
729
+ await runLocalCi({
730
+ apiUrl: resolveApiUrl(args),
731
+ token: args.token || process.env.QUALTY_API_TOKEN,
732
+ orgId: args.org || process.env.QUALTY_ORG_ID,
733
+ projectId: args.project,
734
+ suiteId: args["suite-id"],
735
+ explicitIds: String(args.ids || "")
736
+ .split(",")
737
+ .map((id) => id.trim())
738
+ .filter(Boolean),
739
+ resolveSavedJobs,
740
+ apiRequest,
741
+ failOnFailure: parseBoolean(args["fail-on-failure"], true),
742
+ device: args.device || "1440x900",
743
+ headless: !parseBoolean(args.headed, false),
744
+ localConcurrency: resolveLocalConcurrency(args),
745
+ });
746
+ return;
747
+ }
702
748
  await runCi(args);
703
749
  return;
704
750
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "qualty",
3
- "version": "0.1.7",
3
+ "version": "0.1.8",
4
4
  "description": "Qualty CLI for localhost and CI test runs",
5
5
  "bin": {
6
6
  "qualty": "bin/qualty.js"
@@ -9,5 +9,8 @@
9
9
  "license": "UNLICENSED",
10
10
  "files": [
11
11
  "bin"
12
- ]
12
+ ],
13
+ "dependencies": {
14
+ "playwright": "1.60.0"
15
+ }
13
16
  }