shiplightai 0.1.47 → 0.1.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/cli.js CHANGED
@@ -1,7 +1,70 @@
1
1
  #!/usr/bin/env node
2
2
  import { createRequire as __cli_createRequire } from "module";
3
3
  const require = __cli_createRequire(import.meta.url);
4
- var ao=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var co=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),te=(e,t)=>{for(var r in t)ao(e,r,{get:t[r],enumerable:!0})};import*as W from"fs";import*as q from"path";function ft(e){let t=q.resolve(e);for(;;){if(W.existsSync(q.join(t,"package.json"))||W.existsSync(q.join(t,".shiplight")))return t;let r=q.dirname(t);if(r===t)break;t=r}return null}function gt(e){return q.join(e,lo)}function mt(e,t){return q.join(gt(e),`${po}${t}.json`)}function yt(e,t){let r=ft(t);if(!r)return;let o=gt(r);W.mkdirSync(o,{recursive:!0});let i={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};W.writeFileSync(mt(r,e),JSON.stringify(i),"utf-8")}function bt(e,t){let r=ft(t);if(r)try{W.unlinkSync(mt(r,e))}catch{}}var lo,po,wt=_(()=>{"use strict";lo=".shiplight/run",po="debug-"});var Ne={};te(Ne,{findPlaywrightConfig:()=>fo,spawnPlaywrightProcess:()=>mo});import*as N from"fs";import*as Y from"path";import{spawn as uo}from"child_process";import{parse as ho}from"yaml";function fo(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=Y.resolve(e);for(;;){for(let i of t){let n=Y.join(r,i);if(N.existsSync(n))return n}let o=Y.dirname(r);if(o===r)break;r=o}return null}function go(e,t,r,o){let i={...o},n=i.launchOptions?.args??[];return i.launchOptions={...i.launchOptions??{},args:[...n,`--remote-debugging-port=${r}`]},`// @generated by shiplightai \u2014 temporary debug test
4
+ var To=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var ko=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),te=(e,t)=>{for(var r in t)To(e,r,{get:t[r],enumerable:!0})};import*as R from"fs";import*as K from"path";function ze(e){let{projectPath:t}=e,r=e.projectName??K.basename(K.resolve(t));if(R.existsSync(t)&&R.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);R.mkdirSync(t,{recursive:!0});let o={name:r,type:"module",scripts:{test:"shiplight test","test:headed":"shiplight test --headed"},dependencies:{shiplightai:"latest",dotenv:"^16.4.7"}};R.writeFileSync(K.join(t,"package.json"),JSON.stringify(o,null,2)+`
5
+ `),R.writeFileSync(K.join(t,"playwright.config.ts"),`import { defineConfig, shiplightConfig } from 'shiplightai';
6
+
7
+ export default defineConfig({
8
+ ...shiplightConfig(),
9
+ testDir: '.',
10
+ testMatch: ['**/*.test.ts', '**/*.yaml.spec.ts'],
11
+ timeout: 120_000,
12
+ expect: { timeout: 10_000 },
13
+ retries: 0,
14
+ use: {
15
+ headless: true,
16
+ viewport: { width: 1280, height: 720 },
17
+ actionTimeout: 15_000,
18
+ video: 'on',
19
+ screenshot: 'on',
20
+ trace: 'on',
21
+ },
22
+ });
23
+ `);let n=["node_modules/","test-results/","shiplight-report/",".shiplight/",".env","*.yaml.spec.ts",""].join(`
24
+ `);R.writeFileSync(K.join(t,".gitignore"),n);let s=["# Shiplight API token (optional) \u2014 enables cloud sync tools","# Get yours at https://app.shiplight.ai/settings/api-tokens","# SHIPLIGHT_API_TOKEN=","","# AI provider API key (at least one required).","# Uncomment the one you want to use \u2014 shiplightai auto-selects the","# default model for that provider (names below match the current","# defaults in packages/types/src/organization.ts):","# GOOGLE_API_KEY= # default model: gemini-3.1-flash-lite-preview","# ANTHROPIC_API_KEY= # default model: claude-haiku-4-5","# OPENAI_API_KEY= # default model: gpt-5.4-mini","","# Optional: override the default AI model.","# Supports provider:model prefix for Azure, Bedrock, Vertex AI.","# WEB_AGENT_MODEL=claude-sonnet-4-6","# WEB_AGENT_MODEL=azure:gpt-4o","# WEB_AGENT_MODEL=bedrock:anthropic.claude-sonnet-4-6-v1","# WEB_AGENT_MODEL=vertex:gemini-2.5-pro","","# Optional: custom endpoint for OpenAI-compatible APIs","# OPENAI_BASE_URL=http://localhost:11434/v1","","# Optional: override starting URL for all tests","# PLAYWRIGHT_STARTING_URL=",""].join(`
25
+ `);R.writeFileSync(K.join(t,".env.example"),s);let a=K.join(t,"tests");return R.mkdirSync(a,{recursive:!0}),R.writeFileSync(K.join(a,"example.test.yaml"),`goal: Verify the Shiplight homepage links to the quick-start docs
26
+ base_url: https://www.shiplight.ai
27
+ statements:
28
+ - URL: /
29
+
30
+ - VERIFY: The page title contains "Shiplight"
31
+ js: "await expect(page).toHaveTitle(/Shiplight/)"
32
+
33
+ - intent: Click the Install Plugin link in the navigation
34
+ action: click
35
+ locator: "getByRole('link', { name: 'Install Plugin' })"
36
+
37
+ - VERIFY: The browser lands on the docs site
38
+ js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
39
+ `),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example","tests/example.test.yaml"]}}var At=_(()=>{"use strict"});var Et=_(()=>{"use strict";At()});var Pt={};te(Pt,{runCreate:()=>Do});import*as xe from"path";import*as Te from"fs";function _e(){console.log(`
40
+ Usage: shiplight create <path> [options]
41
+
42
+ Scaffold a new Shiplight test project at <path>. Creates package.json,
43
+ playwright.config.ts, .gitignore, and .env.example.
44
+
45
+ Options:
46
+ --name <name> Override the package.json "name" field (default: directory basename)
47
+ --help, -h Show this help message
48
+
49
+ Examples:
50
+ shiplight create ./my-tests
51
+ shiplight create ./my-tests --name acme-e2e
52
+ `)}async function Do(e){let t,r;for(let n=0;n<e.length;n++){let s=e[n];if(s==="--help"||s==="-h"){_e();return}else s==="--name"?(r=e[++n],(!r||!r.trim())&&(console.error("Error: --name requires a non-empty value"),process.exit(1)),/^[a-z0-9][a-z0-9._-]*$/.test(r)||(console.error(`Error: --name "${r}" is not a valid package name.
53
+ npm package names must be lowercase.
54
+ Use only lowercase letters, digits, hyphens, underscores, and dots;
55
+ must start with a lowercase letter or digit.`),process.exit(1))):s.startsWith("--")?(console.error(`Error: Unknown option: ${s}`),_e(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${s}`),_e(),process.exit(1)):t=s}t||(console.error(`Error: missing required <path> argument
56
+ `),_e(),process.exit(1));let o=xe.resolve(t);Te.existsSync(o)&&Te.readdirSync(o).length>0&&(console.error(`Error: target directory is not empty: ${o}
57
+ Remove it or choose a different path.`),process.exit(1));let i=ze({projectPath:o,projectName:r});console.log(`
58
+ Created Shiplight test project at ${i.projectPath}
59
+ `),console.log(` Name: ${i.projectName}`),console.log(" Files:");for(let n of i.filesCreated)console.log(` - ${n}`);console.log(`
60
+ Next steps:
61
+
62
+ cd ${xe.relative(process.cwd(),i.projectPath)||"."}
63
+ cp .env.example .env # then edit .env and set an AI provider API key
64
+ npm install
65
+ npx playwright install chromium
66
+ npx shiplight test
67
+ `)}var $t=_(()=>{"use strict";Et()});import*as z from"fs";import*as re from"path";function Mt(e){let t=re.resolve(e);for(;;){if(z.existsSync(re.join(t,"package.json"))||z.existsSync(re.join(t,".shiplight")))return t;let r=re.dirname(t);if(r===t)break;t=r}return null}function It(e){return re.join(e,Fo)}function Ot(e,t){return re.join(It(e),`${Uo}${t}.json`)}function Lt(e,t){let r=Mt(t);if(!r)return;let o=It(r);z.mkdirSync(o,{recursive:!0});let i={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};z.writeFileSync(Ot(r,e),JSON.stringify(i),"utf-8")}function Ct(e,t){let r=Mt(t);if(r)try{z.unlinkSync(Ot(r,e))}catch{}}var Fo,Uo,Rt=_(()=>{"use strict";Fo=".shiplight/run",Uo="debug-"});var Ve={};te(Ve,{findPlaywrightConfig:()=>jo,spawnPlaywrightProcess:()=>Wo});import*as U from"fs";import*as Z from"path";import{spawn as Bo}from"child_process";import{parse as Go}from"yaml";function jo(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=Z.resolve(e);for(;;){for(let i of t){let n=Z.join(r,i);if(U.existsSync(n))return n}let o=Z.dirname(r);if(o===r)break;r=o}return null}function Ho(e,t,r,o){let i={...o},n=i.launchOptions?.args??[];return i.launchOptions={...i.launchOptions??{},args:[...n,`--remote-debugging-port=${r}`]},`// @generated by shiplightai \u2014 temporary debug test
5
68
  import { test } from 'shiplightai/fixture';
6
69
  ${`
7
70
  test.use(${JSON.stringify(i)});
@@ -25,15 +88,15 @@ test('__shiplight_debug__', async ({ page, agent }) => {
25
88
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
26
89
  await new Promise(() => {});
27
90
  });
28
- `}async function St(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function mo(e){let{yamlFilePath:t,configPath:r}=e,o=Y.dirname(r),i=await St(16174),n=await St(9222),s;if(!N.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let u=ho(N.readFileSync(t,"utf-8"));u?.use&&typeof u.use=="object"&&!Array.isArray(u.use)&&(s=u.use),u?.base_url&&!s?.baseURL&&(s={...s,baseURL:u.base_url}),u?.settings?.auto_dismiss_modal!==void 0&&(s={...s,autoDismissModal:!!u.settings.auto_dismiss_modal})}catch(u){console.error("[debugger] Could not parse YAML for `use` block:",u)}let a=Y.dirname(Y.resolve(t)),c=Y.join(a,".__shiplight_debug__.yaml.spec.ts"),l=go(t,i,n,s);N.writeFileSync(c,l);let h=uo("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});h.stdout?.on("data",u=>{process.stderr.write(u)}),h.stderr?.on("data",u=>{process.stderr.write(u)});let g=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),h.on("close",u=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{N.unlinkSync(c)}catch{}u!==0&&u!==null&&console.error(`[debugger] Playwright process exited with code ${u}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let u=0;u<180;u++){if(h.exitCode!==null){try{N.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(u===179){g();try{N.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(d=>setTimeout(d,1e3))}return{port:i,cleanup:async()=>{g();try{N.unlinkSync(c)}catch{}}}}var Fe=_(()=>{"use strict"});import{z as m}from"zod";var vt,Ue,_t,ae,xt,Tt,kt,C,At,be,Be,Ge=_(()=>{"use strict";vt=m.enum(["JS_CODE","AI_MODE"]),Ue=m.object({type:vt,expression:m.string()}),_t=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ae=m.object({uid:m.string(),type:_t,comment:m.string().optional()}),xt=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Tt=ae.extend({type:m.literal("DRAFT"),description:m.string()}),kt=ae.extend({type:m.literal("ACTION"),description:m.string(),action_entity:xt.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),C=m.lazy(()=>m.union([Tt,kt,ae.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(C),reference_id:m.number().optional()}),ae.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Ue,then:m.array(C),else:m.array(C).optional()}),ae.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Ue,body:m.array(C),timeout_ms:m.number().optional()})])),At=m.object({name:m.string(),statements:m.array(C),teardown:m.array(C).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),be=m.object({tests:m.array(At).min(1),beforeAll:m.array(C).optional(),afterAll:m.array(C).optional(),beforeEach:m.array(C).optional(),afterEach:m.array(C).optional()}),Be=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(C).optional(),teardown:m.array(C).optional(),last_modified_at:m.string().optional(),testGroup:be.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as yo,parse as Mt,parseAllDocuments as hn,parseDocument as bo,Document as wo,isMap as le,isSeq as j}from"yaml";import{v4 as H}from"uuid";function It(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(F)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(F)),r}function he(e,t){if(e.testGroup)return Lt(e,t);let r=It(e,t),o=new wo(r);return e.comment&&(o.commentBefore=e.comment),Et(o,e.statements??[]),e.teardown&&Et(o,e.teardown,"teardown"),o.toString(Ot)}function Et(e,t,r="statements"){let o=e.contents;if(!o||!le(o))return;let i=o.get(r,!0);j(i)&&de(i,t)}function de(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),le(i)){let n=i;if(o.type==="STEP"){let s=n.get("statements",!0);j(s)&&de(s,o.statements)}else if(o.type==="IF_ELSE"){let s=n.get("THEN",!0);j(s)&&de(s,o.then);let a=n.get("ELSE",!0);j(a)&&o.else&&de(a,o.else)}else if(o.type==="WHILE_LOOP"){let s=n.get("DO",!0);j(s)&&de(s,o.body)}}}}function Lt(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(F)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(F)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(F)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(F)),i.tests=r.tests.map(n=>{let s={name:n.name};return n.skip!==void 0&&(s.skip=n.skip),n.timeout!==void 0&&(s.timeout=n.timeout),n.fail!==void 0&&(s.fail=n.fail),n.only!==void 0&&(s.only=n.only),n.slow!==void 0&&(s.slow=n.slow),s.statements=n.statements.map(F),n.teardown&&n.teardown.length>0&&(s.teardown=n.teardown.map(F)),s}),o.suite=i,yo(o,Ot)}function F(e){switch(e.type){case"DRAFT":return So(e);case"ACTION":return vo(e);case"STEP":return _o(e);case"IF_ELSE":return xo(e);case"WHILE_LOOP":return To(e)}}function So(e){return{intent:e.description}}function vo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let s=e.action_entity.xpath;if(s&&(i.xpath=s),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[a,c]of Object.entries(o.kwargs))i[a]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function _o(e){let t={STEP:e.description,statements:e.statements.map(F)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function xo(e){let t={IF:Ct(e.condition),THEN:e.then.map(F)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(F)),t}function To(e){let t={WHILE:Ct(e.condition),DO:e.body.map(F)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Ct(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function fe(e){try{let t=Mt(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),r}catch{return{}}}function We(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(We);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=We(n);return o}function O(e){if(e.length>$t)throw new Error(`YAML input too large (${e.length} bytes, max ${$t})`);let t=We(Mt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return ko(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:U(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=U(t.teardown));let o=Be.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return we(e,i),i}function ko(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let i={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:U(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=U(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=U(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=U(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=U(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=U(t.afterEach));let n=be.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function U(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Ao)}function Ao(e){if(typeof e=="string")throw new Error(`Plain string statements are not supported. Use an object with a "desc" key instead. Example: { "desc": "${e}" }`);if(typeof e!="object"||e===null)throw new Error(`Invalid statement: expected object, got ${typeof e}`);let t=e;if("IF"in t)return Eo(t);if("WHILE"in t)return $o(t);if("STEP"in t)return Po(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:H(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:H(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:H(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:H(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:H(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:H(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Pt({...o,action:"function",functionName:r})}if("action"in t)return Pt(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:H(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Rt(e){if(typeof e!="string")throw new Error(`Condition must be a string, got ${typeof e}`);return e.startsWith("js:")?{type:"JS_CODE",expression:e.slice(3)}:{type:"AI_MODE",expression:e}}function Eo(e){let t=Rt(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:H(),type:"IF_ELSE",condition:t,then:U(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=U(e.ELSE)),o}function $o(e){let t=Rt(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:H(),type:"WHILE_LOOP",condition:t,body:U(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function Po(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:H(),type:"STEP",description:t,statements:U(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Pt(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[l,p]of Object.entries(e))Mo.has(l)||(s[l]=p);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};o&&(a.locator=o),i&&(a.xpath=i);let c={uid:H(),type:"ACTION",description:r,action_entity:a};return n&&(c.use_pure_vision=!0),c}function we(e,t){let r;try{r=bo(e)}catch{return}let o=r.contents;if(!o||!le(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);j(n)&&t.statements&&ce(n,t.statements);let s=i.get("teardown",!0);j(s)&&t.teardown&&ce(s,t.teardown)}function ce(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&le(o)){let n=o.get("statements",!0);j(n)&&ce(n,i.statements)}else if(i.type==="IF_ELSE"&&le(o)){let n=o.get("THEN",!0);j(n)&&ce(n,i.then);let s=o.get("ELSE",!0);j(s)&&i.else&&ce(s,i.else)}else if(i.type==="WHILE_LOOP"&&le(o)){let n=o.get("DO",!0);j(n)&&ce(n,i.body)}}}var Ot,$t,Mo,Se=_(()=>{"use strict";Ge();Ot={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};$t=1024*1024;Mo=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as bn,stringify as wn}from"yaml";var Dt=_(()=>{"use strict";Se()});var He,ve,_e=_(()=>{"use strict";He=e=>{let t=[];switch(e.type){case"STEP":e.statements&&t.push({key:"statements",statements:e.statements});break;case"IF_ELSE":e.then&&t.push({key:"then",statements:e.then}),e.else&&t.push({key:"else",statements:e.else});break;case"WHILE_LOOP":e.body&&t.push({key:"body",statements:e.body});break}return t},ve=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=He(i);for(let s of n)r(s.statements)}};return r(e),t}});function Ut(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";Ft.has(i)||t++}return{action:t,draft:r}}function Oo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Nt(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function je(e,t){let r=t?.coverageThreshold??Io,o=[],i=[],n;try{n=O(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let s=[...ve(n.statements??[]),...n.teardown?ve(n.teardown):[]],{action:a,draft:c}=Ut(s),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of s){if(g.type==="ACTION"){let f=g,u=f.action_entity?.action_data?.action_name??"";if(u==="js_code"||u==="js_action"||u==="verify"||u==="ai_assert"){let d=f.action_entity?.action_data?.kwargs?.code;if(typeof d=="string"){let w=Oo(d);if(w){let b=f.description||u;o.push(`Invalid JS in "${b}": ${w}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let u=Nt(f.condition.expression);u&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${u}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let u=Nt(f.condition.expression);u&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${u}. ${l}`)}}}let p=a+c,h=p>0?Math.round(a/p*100):0;return p>0&&h/100<r&&i.push(`Low action coverage: ${a}/${p} statements (${h}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:a,draft:c,coverage:h}}}var Io,Ft,Bt=_(()=>{"use strict";Se();_e();Io=.5,Ft=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});function Ke(){return{version:"1.0",entries:{}}}var Gt=_(()=>{"use strict";_e()});import{v4 as $n}from"uuid";var Wt=_(()=>{"use strict"});var K,ze,Ht=_(()=>{"use strict";K=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(K||{}),ze=18e4});var xe=_(()=>{"use strict"});var jt=_(()=>{"use strict";xe()});var Kt,Co,zt,Vt,ge,Ro,Do,Yt=_(()=>{"use strict";Kt=112,Co=1080-Kt,zt={"Blackberry PlayBook":{name:"Blackberry PlayBook",userAgent:"Mozilla/5.0 (PlayBook; U; RIM Tablet OS 2.1.0; en-US) AppleWebKit/536.2+ (KHTML like Gecko) Version/26.0 Safari/536.2+",screen:{width:600,height:1024},viewport:{width:600,height:1024},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"BlackBerry Z30":{name:"BlackBerry Z30",userAgent:"Mozilla/5.0 (BB10; Touch) AppleWebKit/537.10+ (KHTML, like Gecko) Version/26.0 Mobile Safari/537.10+",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note 3":{name:"Galaxy Note 3",userAgent:"Mozilla/5.0 (Linux; U; Android 4.3; en-us; SM-N900T Build/JSS15J) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note II":{name:"Galaxy Note II",userAgent:"Mozilla/5.0 (Linux; U; Android 4.1; en-us; GT-N7100 Build/JRO03C) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S III":{name:"Galaxy S III",userAgent:"Mozilla/5.0 (Linux; U; Android 4.0; en-us; GT-I9300 Build/IMM76D) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S5":{name:"Galaxy S5",userAgent:"Mozilla/5.0 (Linux; Android 5.0; SM-G900P Build/LRX21T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S8":{name:"Galaxy S8",userAgent:"Mozilla/5.0 (Linux; Android 7.0; SM-G950U Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:740},viewport:{width:360,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S9+":{name:"Galaxy S9+",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; SM-G965U Build/R16NW) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:320,height:658},viewport:{width:320,height:658},deviceScaleFactor:4.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S24":{name:"Galaxy S24",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-S921U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:780},viewport:{width:360,height:780},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy A55":{name:"Galaxy A55",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-A556B) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:480,height:1040},viewport:{width:480,height:1040},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S4":{name:"Galaxy Tab S4",userAgent:"Mozilla/5.0 (Linux; Android 8.1.0; SM-T837A) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:712,height:1138},viewport:{width:712,height:1138},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S9":{name:"Galaxy Tab S9",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-X710) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:640,height:1024},viewport:{width:640,height:1024},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"iPad (gen 5)":{name:"iPad (gen 5)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 6)":{name:"iPad (gen 6)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 7)":{name:"iPad (gen 7)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:810,height:1080},viewport:{width:810,height:1080},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 11)":{name:"iPad (gen 11)",userAgent:"Mozilla/5.0 (iPad; CPU OS 18_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/604.1",screen:{width:656,height:944},viewport:{width:656,height:944},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Mini":{name:"iPad Mini",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Pro 11":{name:"iPad Pro 11",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:834,height:1194},viewport:{width:834,height:1194},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6":{name:"iPhone 6",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6 Plus":{name:"iPhone 6 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7":{name:"iPhone 7",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7 Plus":{name:"iPhone 7 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8":{name:"iPhone 8",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8 Plus":{name:"iPhone 8 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE":{name:"iPhone SE",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 10_3_1 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/14E304 Safari/602.1",screen:{width:320,height:568},viewport:{width:320,height:568},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE (3rd gen)":{name:"iPhone SE (3rd gen)",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 18_5 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/602.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone X":{name:"iPhone X",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:812},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone XR":{name:"iPhone XR",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:896},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11":{name:"iPhone 11",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro":{name:"iPhone 11 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:635},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro Max":{name:"iPhone 11 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12":{name:"iPhone 12",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro":{name:"iPhone 12 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro Max":{name:"iPhone 12 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Mini":{name:"iPhone 12 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13":{name:"iPhone 13",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro":{name:"iPhone 13 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro Max":{name:"iPhone 13 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Mini":{name:"iPhone 13 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14":{name:"iPhone 14",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Plus":{name:"iPhone 14 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro":{name:"iPhone 14 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:660},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro Max":{name:"iPhone 14 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15":{name:"iPhone 15",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Plus":{name:"iPhone 15 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro":{name:"iPhone 15 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro Max":{name:"iPhone 15 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Kindle Fire HDX":{name:"Kindle Fire HDX",userAgent:"Mozilla/5.0 (Linux; U; en-us; KFAPWI Build/JDQ39) AppleWebKit/535.19 (KHTML, like Gecko) Silk/3.13 Safari/535.19 Silk-Accelerated=true",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"LG Optimus L70":{name:"LG Optimus L70",userAgent:"Mozilla/5.0 (Linux; U; Android 4.4.2; en-us; LGMS323 Build/KOT49I.MS32310c) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:1.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 550":{name:"Microsoft Lumia 550",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 550) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 950":{name:"Microsoft Lumia 950",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 950) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:4,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 10":{name:"Nexus 10",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 10 Build/MOB31T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 4":{name:"Nexus 4",userAgent:"Mozilla/5.0 (Linux; Android 4.4.2; Nexus 4 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5":{name:"Nexus 5",userAgent:"Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5X":{name:"Nexus 5X",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 5X Build/OPR4.170623.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6":{name:"Nexus 6",userAgent:"Mozilla/5.0 (Linux; Android 7.1.1; Nexus 6 Build/N6F26U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6P":{name:"Nexus 6P",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 6P Build/OPP3.170518.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 7":{name:"Nexus 7",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 7 Build/MOB30X) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:600,height:960},viewport:{width:600,height:960},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia Lumia 520":{name:"Nokia Lumia 520",userAgent:"Mozilla/5.0 (compatible; MSIE 10.0; Windows Phone 8.0; Trident/6.0; IEMobile/10.0; ARM; Touch; NOKIA; Lumia 520)",screen:{width:320,height:533},viewport:{width:320,height:533},deviceScaleFactor:1.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia N9":{name:"Nokia N9",userAgent:"Mozilla/5.0 (MeeGo; NokiaN9) AppleWebKit/534.13 (KHTML, like Gecko) NokiaBrowser/8.5.0 Mobile Safari/534.13",screen:{width:480,height:854},viewport:{width:480,height:854},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Pixel 2":{name:"Pixel 2",userAgent:"Mozilla/5.0 (Linux; Android 8.0; Pixel 2 Build/OPD3.170816.012) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:731},viewport:{width:411,height:731},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 2 XL":{name:"Pixel 2 XL",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Pixel 2 XL Build/OPD1.170816.004) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:823},viewport:{width:411,height:823},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 3":{name:"Pixel 3",userAgent:"Mozilla/5.0 (Linux; Android 9; Pixel 3 Build/PQ1A.181105.017.A1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:786},viewport:{width:393,height:786},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4":{name:"Pixel 4",userAgent:"Mozilla/5.0 (Linux; Android 10; Pixel 4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:353,height:745},viewport:{width:353,height:745},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4a (5G)":{name:"Pixel 4a (5G)",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 4a (5G)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:892},viewport:{width:412,height:765},deviceScaleFactor:2.63,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 5":{name:"Pixel 5",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:851},viewport:{width:393,height:727},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 7":{name:"Pixel 7",userAgent:"Mozilla/5.0 (Linux; Android 14; Pixel 7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:915},viewport:{width:412,height:839},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Moto G4":{name:"Moto G4",userAgent:"Mozilla/5.0 (Linux; Android 7.0; Moto G (4)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Desktop Chrome HiDPI":{name:"Desktop Chrome HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge HiDPI":{name:"Desktop Edge HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Firefox HiDPI":{name:"Desktop Firefox HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"},"Desktop Safari":{name:"Desktop Safari",userAgent:"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Safari/605.1.15",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"webkit"},"Desktop Chrome":{name:"Desktop Chrome",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome Medium Resolution":{name:"Desktop Chrome Medium Resolution",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome (Branded)":{name:"Desktop Chrome (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Chrome Medium Resolution (Branded)":{name:"Desktop Chrome Medium Resolution (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Edge":{name:"Desktop Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge (Branded)":{name:"Desktop Edge (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Edge Medium Resolution (Branded)":{name:"Desktop Edge Medium Resolution (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Firefox":{name:"Desktop Firefox",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"}},Vt={desktop:["Desktop Chrome","Desktop Chrome Medium Resolution","Desktop Chrome (Branded)","Desktop Chrome Medium Resolution (Branded)","Desktop Edge (Branded)","Desktop Edge Medium Resolution (Branded)","Desktop Safari"],mobile:["iPhone 15 Pro Max","iPhone 15 Pro","iPhone 15 Plus","iPhone 15","iPhone 14 Pro Max","iPhone 14 Pro","iPhone 14 Plus","iPhone 14","iPhone 13 Pro Max","iPhone 13 Pro","iPhone 13","iPhone 13 Mini","iPhone 12 Pro Max","iPhone 12 Pro","iPhone 12","iPhone 12 Mini","iPhone 11 Pro Max","iPhone 11 Pro","iPhone 11","iPhone XR","iPhone X","iPhone SE (3rd gen)","iPhone SE","iPhone 8 Plus","iPhone 8","iPhone 7 Plus","iPhone 7","iPhone 6 Plus","iPhone 6","Galaxy S24","Galaxy A55","Galaxy S9+","Galaxy S8","Galaxy S5","Galaxy Note 3","Galaxy Note II","Galaxy S III","Pixel 7","Pixel 5","Pixel 4a (5G)","Pixel 4","Pixel 3","Pixel 2 XL","Pixel 2","Nexus 6P","Nexus 6","Nexus 5X","Nexus 5","Nexus 4","Moto G4","LG Optimus L70","Microsoft Lumia 950","Microsoft Lumia 550","Nokia Lumia 520","Nokia N9","BlackBerry Z30"]},ge=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Vt[e].map(o=>zt[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},Ro={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile")}},Do={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile",!0)}}});var Jt=_(()=>{"use strict"});var Xt=_(()=>{"use strict"});var qt=_(()=>{"use strict"});var Zt=_(()=>{"use strict"});var Qt=_(()=>{"use strict"});var er=_(()=>{"use strict"});var tr=_(()=>{"use strict"});var rr=_(()=>{"use strict";xe()});var re=_(()=>{"use strict";Dt();Bt();Se();Ge();Gt();Wt();_e();Ht();jt();Yt();Jt();Xt();qt();Zt();Qt();er();tr();rr();xe()});import{stringify as Fo}from"yaml";import{createHash as ri}from"crypto";import{parse as oi,stringify as dr}from"yaml";import{readFileSync as ii}from"fs";import{resolve as hr,dirname as ni}from"path";import{parse as sr,stringify as si}from"yaml";import{readFileSync as hi,writeFileSync as fi,mkdirSync as gi}from"fs";import{dirname as mi}from"path";function Z(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function E(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Uo(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Bo(e){let t=e.xpath;return typeof t=="string"&&t.trim()?!t.startsWith("xpath=")&&!t.startsWith("/")&&!t.startsWith("//")?`xpath=//${t}`:t.startsWith("xpath=")?t:`xpath=${t}`:null}function ke(e){let t=Uo(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=Bo(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function or(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Go.includes(t)}function Ho(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!Wo.includes(t)}function S(e,t){I.set(e,t)}function jo(e){return I.get(e)}function ne(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function ir(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(s=>o.includes(s)||i.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}function J(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let s=e[n],a=`${o}.${n}`,c=Ko(s,t,a,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function Ko(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return zo(e,t,r,o);case"ACTION":return Vo(e,t,r,o);case"STEP":return Yo(e,t,r,o);case"IF_ELSE":return Jo(e,t,r,o);case"WHILE_LOOP":return Xo(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function zo(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let s=JSON.stringify(n);return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${s}, '${r}');`]}function Vo(e,t,r,o){let i=" ".repeat(t),n=e.description,s=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",h=l.action_description||"",g=jo(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},u=g(l,r,f);if(o.noAgent){if(or(l))return[`${i}// ${r}: ${E(h)}`,`${i}// AI action: ${E(h)} (requires agent - skipped in hook)`];let y=qo(l,p,i,r);return y||[`${i}// ${r}: ${E(h)}`,...u.map(A=>`${i}${A}`)]}if(or(l))return[`${i}// ${r}: ${E(h)}`,`${i}page = agent.agentServices.validatePage(page);`,...u.map(y=>`${i}${y}`)];let d=JSON.stringify(h),w=u.map(y=>`${i} ${y}`),b=Ho(l),v=s?`'${s}'`:"undefined";return[`${i}// ${r}: ${E(h)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...w,`${i}}, ${d}, '${r}', ${v}, ${b});`]}function Yo(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${E(e.description)}`);let s=J(e.statements,t,o,r);return n.push(...s),n}function Jo(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${E(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let s=J(e.then,t+1,o,`${r}.then`);if(n.push(...s),e.else&&e.else.length>0){n.push(`${i}} else {`);let a=J(e.else,t+1,o,`${r}.else`);n.push(...a)}return n.push(`${i}}`),n}function Xo(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let s=e.timeout_ms??ze,a=s/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${s};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${E(e.condition.expression)}`);let h=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${h}, "${r}")) {`)}let p=J(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function qo(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},s=n.timeout_ms??Ye;switch(t){case"go_to_url":case"open_tab":{let a=n.url||"";return[`${r}// ${o}: ${E(i)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goForward();`];case"input_text":{let a=n.text||"",c=ke(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${s} });`]:null}case"select_dropdown_option":{let a=n.text||n.label||"",c=ke(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${s} });`]:null}default:return null}}function Zo(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...pr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...nr("beforeEach",t.beforeEach,n)),r.push(""));let s=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=Ve(t?.tags);for(let l of t.parameters){let p=lr(e,l.values);r.push(...Ee(p,`${c}${Z(a)} [${Z(l.name)}]`,n,0,s)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=Ve(t?.tags);r.push(...Ee(e,`${c}${Z(a)}`,n,0,s))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...nr("afterEach",t.afterEach,n))),ur(r,i),r.join(`
29
- `)}function Qo(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...pr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore},s=t?.testName||"Test Suite",a=Ve(t?.tags);r.push(`test.describe.serial('${a}${Z(s)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Te("beforeAll",e.beforeAll,n,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Te("beforeEach",e.beforeEach,n,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],h=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let g of p.parameters){let f=lr(p.testFlow,g.values);r.push(...Ee(f,`${Z(p.name)} [${Z(g.name)}]`,n,1,h)),r.push("")}else r.push(...Ee(p.testFlow,Z(p.name),n,1,h)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Te("afterEach",e.afterEach,n,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Te("afterAll",e.afterAll,n,1)),r.push("});"),ur(r,i),r.join(`
30
- `)}function Ve(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Ae(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case K.ACTION:{let s=i.action_entity?.action_data?.kwargs;if(s?.args&&Array.isArray(s.args))for(let a of s.args)typeof a=="string"&&ei.includes(a)&&t.add(a);break}case K.STEP:r(i.statements);break;case K.IF_ELSE:{let n=i;r(n.then),n.else&&r(n.else);break}case K.WHILE_LOOP:r(i.body);break}}return r(e),t}function ti(e){let t=Ae(e.statements??[]);if(e.teardown)for(let r of Ae(e.teardown))t.add(r);return t}function Je(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ee(e,t,r,o=0,i){let n=" ".repeat(o),s=[],a=ti(e),c=Je(a),l=i?.only?"test.only":"test";s.push(`${n}${l}('${t}', async (${c}) => {`),i?.skip===!0?s.push(`${n} test.skip();`):typeof i?.skip=="string"&&s.push(`${n} test.skip(true, '${Z(i.skip)}');`),i?.fail===!0?s.push(`${n} test.fail();`):typeof i?.fail=="string"&&s.push(`${n} test.fail(true, '${Z(i.fail)}');`),i?.slow&&s.push(`${n} test.slow();`),i?.timeout&&s.push(`${n} test.setTimeout(${i.timeout});`);let p=e.teardown&&e.teardown.length>0,h=o+1;if(p){if(s.push(`${n} try {`),e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let f=J(e.statements,h+1,r);s.push(...f)}s.push(`${n} } finally {`),s.push(`${n} // Teardown`);let g=J(e.teardown,h+1,r,"teardown");s.push(...g),s.push(`${n} }`)}else if(e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let g=J(e.statements,h,r);s.push(...g)}return s.push(`${n}});`),s}function nr(e,t,r){let o=[],i=cr(t),n=Ae(i),s=Je(n);return o.push(`test.${e}(async (${s}) => {`),o.push(...J(i,1,r,e)),o.push("});"),o}function Te(e,t,r,o){let i=" ".repeat(o),n=[],s=cr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};n.push(`${i}test.${e}(async ({ browser }, workerInfo) => {`),n.push(`${i} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),n.push(...J(s,o+1,c,e)),n.push(`${i} await page.close();`),n.push(`${i}});`)}else{let c=Ae(s),l=Je(c);n.push(`${i}test.${e}(async (${l}) => {`),n.push(...J(s,o+1,r,e)),n.push(`${i}});`)}return n}function cr(e){let r=Fo({goal:"_hook",statements:e});return O(r).statements??[]}function lr(e,t){let r=he(e);for(let[o,i]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(i));return O(r)}function pr(){return["import { test, expect } from 'shiplightai/fixture';"]}function ur(e,t){if(t.size>0){let r=0;for(let i=0;i<e.length;i++)e[i].startsWith("import ")&&(r=i+1);let o=Array.from(t);e.splice(r,0,...o)}}function fr(e,t){let r={expandingPaths:new Set([hr(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=oe(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=oe(o.teardown,t,r));for(let i of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[i])&&(o[i]=oe(o[i],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function oe(e,t,r){let o=[];for(let i of e)if(ai(i)){let n=ci(i,t,r);o.push(...n)}else o.push(li(i,t,r));return o}function ai(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function ci(e,t,r){if(r.depth>=ar)throw new Error(`Template expansion exceeded maximum depth of ${ar}. Check for deeply nested or circular template references.`);let o=hr(ni(t),e.template);if(r.expandingPaths.has(o))throw new Error(`Circular template reference detected: ${o} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${o}`);r.referencedPaths.add(o);let i;try{i=ii(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let n=sr(i);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let s=n.params||[],a=e.params||{};for(let p of s)if(!(p in a))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${s.join(", ")}]`);let c=n.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(a).length>0){let h=si(c);for(let[g,f]of Object.entries(a))h=h.split(`<<${g}>>`).join(String(f));c=sr(h)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return oe(c,o,l)}function li(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=oe(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=oe(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=oe(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=oe(o.DO,t,r)),o}function qe(e,t){let r=oi(e),o=r?.name,i=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new Xe('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ui(r,o,i,n,t)}return pi(r,o,i,n,t)}function pi(e,t,r,o,i){let n=e?.beforeEach,s=e?.afterEach,a=gr(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,h=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new Xe(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(i&&e&&typeof e=="object"){let w=fr(e,i);e=w.doc,f=w.referencedTemplatePaths}let u=dr(e),d=O(u);return i&&(pe(d.statements??[],i,"main"),d.teardown&&pe(d.teardown,i,"teardown")),{testFlow:d,name:t,tags:r,use:o,beforeEach:n,afterEach:s,parameters:a,timeout:c,skip:l,fail:p,only:h,slow:g,referencedTemplatePaths:f}}function ui(e,t,r,o,i){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,a=n.afterAll,c=n.beforeEach,l=n.afterEach,p=[],h=n.tests.map(u=>{if(!u.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(u.statements)||u.statements.length===0)throw new Error(`Suite test "${u.name}" must have a non-empty "statements" array.`);let d={goal:u.name,statements:u.statements};u.teardown&&(d.teardown=u.teardown);let w=[],b=d;if(i&&typeof d=="object"){let k=fr(d,i);b=k.doc,w=k.referencedTemplatePaths,p.push(...w)}let v=dr(b),y=O(v),A=gr(u.parameters);return{testFlow:y,name:u.name,tags:Array.isArray(u.tags)?u.tags:void 0,parameters:A,timeout:u.timeout,skip:u.skip,fail:u.fail,only:u.only,slow:u.slow}}),g=n.base_url,f=g?{...o,baseURL:g}:o;return{suite:{beforeAll:s,afterAll:a,beforeEach:c,afterEach:l,tests:h},name:t,tags:r,use:f,referencedTemplatePaths:p}}function gr(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function pe(e,t,r){for(let o=0;o<e.length;o++){let i=e[o],n=`${r}.${o}`,s=i.description||"";if(i.uid=di(t,n,s),i.type===K.STEP)pe(i.statements,t,n);else if(i.type===K.IF_ELSE){let a=i;pe(a.then,t,`${n}.then`),a.else&&pe(a.else,t,`${n}.else`)}else i.type===K.WHILE_LOOP&&pe(i.body,t,`${n}.body`)}}function di(e,t,r){let o=ri("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${o.slice(0,8)}-${o.slice(8,12)}-${o.slice(12,16)}-${o.slice(16,20)}-${o.slice(20,32)}`}function mr(e,t){let r;try{r=hi(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return yi(r,e,t)}function yi(e,t,r){let o=/\btemplate:\s/.test(e),i=/^suite:/m.test(e),n=o||i?null:je(e);if(n&&!n.valid)return{valid:!1,errors:n.errors,warnings:[],stats:n.stats};let s,a,c=[];try{let l=r?.parsed??qe(e,t);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore},h=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?s=Qo(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):s=Zo(l.testFlow,{...p,testName:l.name,tags:l.tags,use:h,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let g=s.split(`
91
+ `}async function Nt(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function Wo(e){let{yamlFilePath:t,configPath:r}=e,o=Z.dirname(r),i=await Nt(16174),n=await Nt(9222),s;if(!U.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let d=Go(U.readFileSync(t,"utf-8"));d?.use&&typeof d.use=="object"&&!Array.isArray(d.use)&&(s=d.use),d?.base_url&&!s?.baseURL&&(s={...s,baseURL:d.base_url}),d?.settings?.auto_dismiss_modal!==void 0&&(s={...s,autoDismissModal:!!d.settings.auto_dismiss_modal})}catch(d){console.error("[debugger] Could not parse YAML for `use` block:",d)}let a=Z.dirname(Z.resolve(t)),c=Z.join(a,".__shiplight_debug__.yaml.spec.ts"),l=Ho(t,i,n,s);U.writeFileSync(c,l);let u=Bo("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});u.stdout?.on("data",d=>{process.stderr.write(d)}),u.stderr?.on("data",d=>{process.stderr.write(d)});let g=()=>{u.killed||u.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),u.on("close",d=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{U.unlinkSync(c)}catch{}d!==0&&d!==null&&console.error(`[debugger] Playwright process exited with code ${d}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let d=0;d<180;d++){if(u.exitCode!==null){try{U.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${u.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(d===179){g();try{U.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(h=>setTimeout(h,1e3))}return{port:i,cleanup:async()=>{g();try{U.unlinkSync(c)}catch{}}}}var Ye=_(()=>{"use strict"});var Dt=_(()=>{"use strict"});var Ft=_(()=>{"use strict"});var Ut=_(()=>{"use strict"});import{v4 as Yn}from"uuid";var Bt=_(()=>{"use strict"});import{z as m}from"zod";var Gt,Je,jt,ue,Ht,Wt,Kt,N,zt,ke,Xe,qe=_(()=>{"use strict";Gt=m.enum(["JS_CODE","AI_MODE"]),Je=m.object({type:Gt,expression:m.string()}),jt=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ue=m.object({uid:m.string(),type:jt,comment:m.string().optional()}),Ht=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Wt=ue.extend({type:m.literal("DRAFT"),description:m.string()}),Kt=ue.extend({type:m.literal("ACTION"),description:m.string(),action_entity:Ht.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),N=m.lazy(()=>m.union([Wt,Kt,ue.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(N),reference_id:m.number().optional()}),ue.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Je,then:m.array(N),else:m.array(N).optional()}),ue.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Je,body:m.array(N),timeout_ms:m.number().optional()})])),zt=m.object({name:m.string(),statements:m.array(N),teardown:m.array(N).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),ke=m.object({tests:m.array(zt).min(1),beforeAll:m.array(N).optional(),afterAll:m.array(N).optional(),beforeEach:m.array(N).optional(),afterEach:m.array(N).optional()}),Xe=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(N).optional(),teardown:m.array(N).optional(),last_modified_at:m.string().optional(),testGroup:ke.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as Ko,parse as Xt,parseAllDocuments as es,parseDocument as zo,Document as Vo,isMap as he,isSeq as Y}from"yaml";import{v4 as V}from"uuid";function qt(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(B)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(B)),r}function ye(e,t){if(e.testGroup)return Qt(e,t);let r=qt(e,t),o=new Vo(r);return e.comment&&(o.commentBefore=e.comment),Vt(o,e.statements??[]),e.teardown&&Vt(o,e.teardown,"teardown"),o.toString(Zt)}function Vt(e,t,r="statements"){let o=e.contents;if(!o||!he(o))return;let i=o.get(r,!0);Y(i)&&me(i,t)}function me(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),he(i)){let n=i;if(o.type==="STEP"){let s=n.get("statements",!0);Y(s)&&me(s,o.statements)}else if(o.type==="IF_ELSE"){let s=n.get("THEN",!0);Y(s)&&me(s,o.then);let a=n.get("ELSE",!0);Y(a)&&o.else&&me(a,o.else)}else if(o.type==="WHILE_LOOP"){let s=n.get("DO",!0);Y(s)&&me(s,o.body)}}}}function Qt(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(B)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(B)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(B)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(B)),i.tests=r.tests.map(n=>{let s={name:n.name};return n.skip!==void 0&&(s.skip=n.skip),n.timeout!==void 0&&(s.timeout=n.timeout),n.fail!==void 0&&(s.fail=n.fail),n.only!==void 0&&(s.only=n.only),n.slow!==void 0&&(s.slow=n.slow),s.statements=n.statements.map(B),n.teardown&&n.teardown.length>0&&(s.teardown=n.teardown.map(B)),s}),o.suite=i,Ko(o,Zt)}function B(e){switch(e.type){case"DRAFT":return Yo(e);case"ACTION":return Jo(e);case"STEP":return Xo(e);case"IF_ELSE":return qo(e);case"WHILE_LOOP":return Zo(e)}}function Yo(e){return{intent:e.description}}function Jo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let s=e.action_entity.xpath;if(s&&(i.xpath=s),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[a,c]of Object.entries(o.kwargs))i[a]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function Xo(e){let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function qo(e){let t={IF:er(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function Zo(e){let t={WHILE:er(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function er(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function we(e){try{let t=Xt(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),r}catch{return{}}}function Ze(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(Ze);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=Ze(n);return o}function L(e){if(e.length>Yt)throw new Error(`YAML input too large (${e.length} bytes, max ${Yt})`);let t=Ze(Xt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Qo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:G(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=G(t.teardown));let o=Xe.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return Ae(e,i),i}function Qo(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let i={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:G(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=G(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=G(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=G(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=G(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=G(t.afterEach));let n=ke.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function G(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(ei)}function ei(e){if(typeof e=="string")throw new Error(`Plain string statements are not supported. Use an object with a "desc" key instead. Example: { "desc": "${e}" }`);if(typeof e!="object"||e===null)throw new Error(`Invalid statement: expected object, got ${typeof e}`);let t=e;if("IF"in t)return ti(t);if("WHILE"in t)return ri(t);if("STEP"in t)return oi(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:V(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:V(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:V(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:V(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:V(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:V(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Jt({...o,action:"function",functionName:r})}if("action"in t)return Jt(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:V(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function tr(e){if(typeof e!="string")throw new Error(`Condition must be a string, got ${typeof e}`);return e.startsWith("js:")?{type:"JS_CODE",expression:e.slice(3)}:{type:"AI_MODE",expression:e}}function ti(e){let t=tr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:V(),type:"IF_ELSE",condition:t,then:G(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=G(e.ELSE)),o}function ri(e){let t=tr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:V(),type:"WHILE_LOOP",condition:t,body:G(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function oi(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:V(),type:"STEP",description:t,statements:G(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Jt(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[l,p]of Object.entries(e))ii.has(l)||(s[l]=p);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};o&&(a.locator=o),i&&(a.xpath=i);let c={uid:V(),type:"ACTION",description:r,action_entity:a};return n&&(c.use_pure_vision=!0),c}function Ae(e,t){let r;try{r=zo(e)}catch{return}let o=r.contents;if(!o||!he(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);Y(n)&&t.statements&&de(n,t.statements);let s=i.get("teardown",!0);Y(s)&&t.teardown&&de(s,t.teardown)}function de(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&he(o)){let n=o.get("statements",!0);Y(n)&&de(n,i.statements)}else if(i.type==="IF_ELSE"&&he(o)){let n=o.get("THEN",!0);Y(n)&&de(n,i.then);let s=o.get("ELSE",!0);Y(s)&&i.else&&de(s,i.else)}else if(i.type==="WHILE_LOOP"&&he(o)){let n=o.get("DO",!0);Y(n)&&de(n,i.body)}}}var Zt,Yt,ii,Ee=_(()=>{"use strict";qe();Zt={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Yt=1024*1024;ii=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as ns,stringify as ss}from"yaml";var rr=_(()=>{"use strict";Ee()});var Qe,Pe,$e=_(()=>{"use strict";Qe=e=>{let t=[];switch(e.type){case"STEP":e.statements&&t.push({key:"statements",statements:e.statements});break;case"IF_ELSE":e.then&&t.push({key:"then",statements:e.then}),e.else&&t.push({key:"else",statements:e.else});break;case"WHILE_LOOP":e.body&&t.push({key:"body",statements:e.body});break}return t},Pe=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=Qe(i);for(let s of n)r(s.statements)}};return r(e),t}});function nr(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";ir.has(i)||t++}return{action:t,draft:r}}function si(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function or(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function et(e,t){let r=t?.coverageThreshold??ni,o=[],i=[],n;try{n=L(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let s=[...Pe(n.statements??[]),...n.teardown?Pe(n.teardown):[]],{action:a,draft:c}=nr(s),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of s){if(g.type==="ACTION"){let f=g,d=f.action_entity?.action_data?.action_name??"";if(d==="js_code"||d==="js_action"||d==="verify"||d==="ai_assert"){let h=f.action_entity?.action_data?.kwargs?.code;if(typeof h=="string"){let b=si(h);if(b){let w=f.description||d;o.push(`Invalid JS in "${w}": ${b}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${d}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${d}. ${l}`)}}}let p=a+c,u=p>0?Math.round(a/p*100):0;return p>0&&u/100<r&&i.push(`Low action coverage: ${a}/${p} statements (${u}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:a,draft:c,coverage:u}}}var ni,ir,sr=_(()=>{"use strict";Ee();$e();ni=.5,ir=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Me=_(()=>{"use strict"});var ar=_(()=>{"use strict";Me()});var cr,ci,lr,pr,be,li,pi,ur=_(()=>{"use strict";cr=112,ci=1080-cr,lr={"Blackberry PlayBook":{name:"Blackberry PlayBook",userAgent:"Mozilla/5.0 (PlayBook; U; RIM Tablet OS 2.1.0; en-US) AppleWebKit/536.2+ (KHTML like Gecko) Version/26.0 Safari/536.2+",screen:{width:600,height:1024},viewport:{width:600,height:1024},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"BlackBerry Z30":{name:"BlackBerry Z30",userAgent:"Mozilla/5.0 (BB10; Touch) AppleWebKit/537.10+ (KHTML, like Gecko) Version/26.0 Mobile Safari/537.10+",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note 3":{name:"Galaxy Note 3",userAgent:"Mozilla/5.0 (Linux; U; Android 4.3; en-us; SM-N900T Build/JSS15J) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note II":{name:"Galaxy Note II",userAgent:"Mozilla/5.0 (Linux; U; Android 4.1; en-us; GT-N7100 Build/JRO03C) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S III":{name:"Galaxy S III",userAgent:"Mozilla/5.0 (Linux; U; Android 4.0; en-us; GT-I9300 Build/IMM76D) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S5":{name:"Galaxy S5",userAgent:"Mozilla/5.0 (Linux; Android 5.0; SM-G900P Build/LRX21T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S8":{name:"Galaxy S8",userAgent:"Mozilla/5.0 (Linux; Android 7.0; SM-G950U Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:740},viewport:{width:360,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S9+":{name:"Galaxy S9+",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; SM-G965U Build/R16NW) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:320,height:658},viewport:{width:320,height:658},deviceScaleFactor:4.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S24":{name:"Galaxy S24",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-S921U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:780},viewport:{width:360,height:780},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy A55":{name:"Galaxy A55",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-A556B) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:480,height:1040},viewport:{width:480,height:1040},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S4":{name:"Galaxy Tab S4",userAgent:"Mozilla/5.0 (Linux; Android 8.1.0; SM-T837A) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:712,height:1138},viewport:{width:712,height:1138},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S9":{name:"Galaxy Tab S9",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-X710) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:640,height:1024},viewport:{width:640,height:1024},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"iPad (gen 5)":{name:"iPad (gen 5)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 6)":{name:"iPad (gen 6)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 7)":{name:"iPad (gen 7)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:810,height:1080},viewport:{width:810,height:1080},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 11)":{name:"iPad (gen 11)",userAgent:"Mozilla/5.0 (iPad; CPU OS 18_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/604.1",screen:{width:656,height:944},viewport:{width:656,height:944},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Mini":{name:"iPad Mini",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Pro 11":{name:"iPad Pro 11",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:834,height:1194},viewport:{width:834,height:1194},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6":{name:"iPhone 6",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6 Plus":{name:"iPhone 6 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7":{name:"iPhone 7",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7 Plus":{name:"iPhone 7 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8":{name:"iPhone 8",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8 Plus":{name:"iPhone 8 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE":{name:"iPhone SE",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 10_3_1 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/14E304 Safari/602.1",screen:{width:320,height:568},viewport:{width:320,height:568},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE (3rd gen)":{name:"iPhone SE (3rd gen)",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 18_5 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/602.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone X":{name:"iPhone X",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:812},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone XR":{name:"iPhone XR",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:896},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11":{name:"iPhone 11",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro":{name:"iPhone 11 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:635},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro Max":{name:"iPhone 11 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12":{name:"iPhone 12",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro":{name:"iPhone 12 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro Max":{name:"iPhone 12 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Mini":{name:"iPhone 12 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13":{name:"iPhone 13",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro":{name:"iPhone 13 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro Max":{name:"iPhone 13 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Mini":{name:"iPhone 13 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14":{name:"iPhone 14",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Plus":{name:"iPhone 14 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro":{name:"iPhone 14 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:660},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro Max":{name:"iPhone 14 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15":{name:"iPhone 15",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Plus":{name:"iPhone 15 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro":{name:"iPhone 15 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro Max":{name:"iPhone 15 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Kindle Fire HDX":{name:"Kindle Fire HDX",userAgent:"Mozilla/5.0 (Linux; U; en-us; KFAPWI Build/JDQ39) AppleWebKit/535.19 (KHTML, like Gecko) Silk/3.13 Safari/535.19 Silk-Accelerated=true",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"LG Optimus L70":{name:"LG Optimus L70",userAgent:"Mozilla/5.0 (Linux; U; Android 4.4.2; en-us; LGMS323 Build/KOT49I.MS32310c) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:1.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 550":{name:"Microsoft Lumia 550",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 550) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 950":{name:"Microsoft Lumia 950",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 950) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:4,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 10":{name:"Nexus 10",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 10 Build/MOB31T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 4":{name:"Nexus 4",userAgent:"Mozilla/5.0 (Linux; Android 4.4.2; Nexus 4 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5":{name:"Nexus 5",userAgent:"Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5X":{name:"Nexus 5X",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 5X Build/OPR4.170623.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6":{name:"Nexus 6",userAgent:"Mozilla/5.0 (Linux; Android 7.1.1; Nexus 6 Build/N6F26U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6P":{name:"Nexus 6P",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 6P Build/OPP3.170518.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 7":{name:"Nexus 7",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 7 Build/MOB30X) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:600,height:960},viewport:{width:600,height:960},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia Lumia 520":{name:"Nokia Lumia 520",userAgent:"Mozilla/5.0 (compatible; MSIE 10.0; Windows Phone 8.0; Trident/6.0; IEMobile/10.0; ARM; Touch; NOKIA; Lumia 520)",screen:{width:320,height:533},viewport:{width:320,height:533},deviceScaleFactor:1.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia N9":{name:"Nokia N9",userAgent:"Mozilla/5.0 (MeeGo; NokiaN9) AppleWebKit/534.13 (KHTML, like Gecko) NokiaBrowser/8.5.0 Mobile Safari/534.13",screen:{width:480,height:854},viewport:{width:480,height:854},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Pixel 2":{name:"Pixel 2",userAgent:"Mozilla/5.0 (Linux; Android 8.0; Pixel 2 Build/OPD3.170816.012) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:731},viewport:{width:411,height:731},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 2 XL":{name:"Pixel 2 XL",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Pixel 2 XL Build/OPD1.170816.004) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:823},viewport:{width:411,height:823},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 3":{name:"Pixel 3",userAgent:"Mozilla/5.0 (Linux; Android 9; Pixel 3 Build/PQ1A.181105.017.A1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:786},viewport:{width:393,height:786},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4":{name:"Pixel 4",userAgent:"Mozilla/5.0 (Linux; Android 10; Pixel 4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:353,height:745},viewport:{width:353,height:745},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4a (5G)":{name:"Pixel 4a (5G)",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 4a (5G)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:892},viewport:{width:412,height:765},deviceScaleFactor:2.63,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 5":{name:"Pixel 5",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:851},viewport:{width:393,height:727},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 7":{name:"Pixel 7",userAgent:"Mozilla/5.0 (Linux; Android 14; Pixel 7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:915},viewport:{width:412,height:839},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Moto G4":{name:"Moto G4",userAgent:"Mozilla/5.0 (Linux; Android 7.0; Moto G (4)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Desktop Chrome HiDPI":{name:"Desktop Chrome HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge HiDPI":{name:"Desktop Edge HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Firefox HiDPI":{name:"Desktop Firefox HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"},"Desktop Safari":{name:"Desktop Safari",userAgent:"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Safari/605.1.15",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"webkit"},"Desktop Chrome":{name:"Desktop Chrome",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome Medium Resolution":{name:"Desktop Chrome Medium Resolution",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome (Branded)":{name:"Desktop Chrome (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Chrome Medium Resolution (Branded)":{name:"Desktop Chrome Medium Resolution (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Edge":{name:"Desktop Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge (Branded)":{name:"Desktop Edge (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Edge Medium Resolution (Branded)":{name:"Desktop Edge Medium Resolution (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Firefox":{name:"Desktop Firefox",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"}},pr={desktop:["Desktop Chrome","Desktop Chrome Medium Resolution","Desktop Chrome (Branded)","Desktop Chrome Medium Resolution (Branded)","Desktop Edge (Branded)","Desktop Edge Medium Resolution (Branded)","Desktop Safari"],mobile:["iPhone 15 Pro Max","iPhone 15 Pro","iPhone 15 Plus","iPhone 15","iPhone 14 Pro Max","iPhone 14 Pro","iPhone 14 Plus","iPhone 14","iPhone 13 Pro Max","iPhone 13 Pro","iPhone 13","iPhone 13 Mini","iPhone 12 Pro Max","iPhone 12 Pro","iPhone 12","iPhone 12 Mini","iPhone 11 Pro Max","iPhone 11 Pro","iPhone 11","iPhone XR","iPhone X","iPhone SE (3rd gen)","iPhone SE","iPhone 8 Plus","iPhone 8","iPhone 7 Plus","iPhone 7","iPhone 6 Plus","iPhone 6","Galaxy S24","Galaxy A55","Galaxy S9+","Galaxy S8","Galaxy S5","Galaxy Note 3","Galaxy Note II","Galaxy S III","Pixel 7","Pixel 5","Pixel 4a (5G)","Pixel 4","Pixel 3","Pixel 2 XL","Pixel 2","Nexus 6P","Nexus 6","Nexus 5X","Nexus 5","Nexus 4","Moto G4","LG Optimus L70","Microsoft Lumia 950","Microsoft Lumia 550","Nokia Lumia 520","Nokia N9","BlackBerry Z30"]},be=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),pr[e].map(o=>lr[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},li={desktop:{label:"Desktop",type:"desktop",devices:be("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:be("mobile")}},pi={desktop:{label:"Desktop",type:"desktop",devices:be("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:be("mobile",!0)}}});var dr=_(()=>{"use strict"});function tt(){return{version:"1.0",entries:{}}}var hr=_(()=>{"use strict";$e()});var J,rt,fr=_(()=>{"use strict";J=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(J||{}),rt=18e4});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});var wr=_(()=>{"use strict";Me()});var se=_(()=>{"use strict";Dt();Ft();Ut();Bt();rr();sr();Ee();qe();ar();ur();dr();hr();$e();fr();gr();mr();yr();wr();Me()});import{stringify as di}from"yaml";import{createHash as Mi}from"crypto";import{parse as Ii,stringify as Pr}from"yaml";import{readFileSync as Oi}from"fs";import{resolve as $r,dirname as Li}from"path";import{parse as _r,stringify as Ci}from"yaml";import{readFileSync as Gi,writeFileSync as ji,mkdirSync as Hi}from"fs";import{dirname as Wi}from"path";function oe(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function E(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function hi(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function fi(e){let t=e.xpath;return typeof t=="string"&&t.trim()?!t.startsWith("xpath=")&&!t.startsWith("/")&&!t.startsWith("//")?`xpath=//${t}`:t.startsWith("xpath=")?t:`xpath=${t}`:null}function Oe(e){let t=hi(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=fi(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function br(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:gi.includes(t)}function yi(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!mi.includes(t)}function S(e,t){O.set(e,t)}function wi(e){return O.get(e)}function le(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function Sr(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(s=>o.includes(s)||i.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}function Q(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let s=e[n],a=`${o}.${n}`,c=bi(s,t,a,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function bi(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return Si(e,t,r,o);case"ACTION":return vi(e,t,r,o);case"STEP":return _i(e,t,r,o);case"IF_ELSE":return xi(e,t,r,o);case"WHILE_LOOP":return Ti(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function Si(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let s=JSON.stringify(n);return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${s}, '${r}');`]}function vi(e,t,r,o){let i=" ".repeat(t),n=e.description,s=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",u=l.action_description||"",g=wi(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},d=g(l,r,f);if(o.noAgent){if(br(l))return[`${i}// ${r}: ${E(u)}`,`${i}// AI action: ${E(u)} (requires agent - skipped in hook)`];let y=ki(l,p,i,r);return y||[`${i}// ${r}: ${E(u)}`,...d.map(A=>`${i}${A}`)]}if(br(l))return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,...d.map(y=>`${i}${y}`)];let h=JSON.stringify(u),b=d.map(y=>`${i} ${y}`),w=yi(l),v=s?`'${s}'`:"undefined";return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...b,`${i}}, ${h}, '${r}', ${v}, ${w});`]}function _i(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${E(e.description)}`);let s=Q(e.statements,t,o,r);return n.push(...s),n}function xi(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${E(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let s=Q(e.then,t+1,o,`${r}.then`);if(n.push(...s),e.else&&e.else.length>0){n.push(`${i}} else {`);let a=Q(e.else,t+1,o,`${r}.else`);n.push(...a)}return n.push(`${i}}`),n}function Ti(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let s=e.timeout_ms??rt,a=s/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${s};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${E(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${u}, "${r}")) {`)}let p=Q(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function ki(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},s=n.timeout_ms??it;switch(t){case"go_to_url":case"open_tab":{let a=n.url||"";return[`${r}// ${o}: ${E(i)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goForward();`];case"input_text":{let a=n.text||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${s} });`]:null}case"select_dropdown_option":{let a=n.text||n.label||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${s} });`]:null}default:return null}}function Ai(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...vr("beforeEach",t.beforeEach,n)),r.push(""));let s=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);for(let l of t.parameters){let p=kr(e,l.values);r.push(...Ce(p,`${c}${oe(a)} [${oe(l.name)}]`,n,0,s)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);r.push(...Ce(e,`${c}${oe(a)}`,n,0,s))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...vr("afterEach",t.afterEach,n))),Er(r,i),r.join(`
92
+ `)}function Ei(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore},s=t?.testName||"Test Suite",a=ot(t?.tags);r.push(`test.describe.serial('${a}${oe(s)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ie("beforeAll",e.beforeAll,n,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ie("beforeEach",e.beforeEach,n,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],u=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let g of p.parameters){let f=kr(p.testFlow,g.values);r.push(...Ce(f,`${oe(p.name)} [${oe(g.name)}]`,n,1,u)),r.push("")}else r.push(...Ce(p.testFlow,oe(p.name),n,1,u)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ie("afterEach",e.afterEach,n,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ie("afterAll",e.afterAll,n,1)),r.push("});"),Er(r,i),r.join(`
93
+ `)}function ot(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Le(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case J.ACTION:{let s=i.action_entity?.action_data?.kwargs;if(s?.args&&Array.isArray(s.args))for(let a of s.args)typeof a=="string"&&Pi.includes(a)&&t.add(a);break}case J.STEP:r(i.statements);break;case J.IF_ELSE:{let n=i;r(n.then),n.else&&r(n.else);break}case J.WHILE_LOOP:r(i.body);break}}return r(e),t}function $i(e){let t=Le(e.statements??[]);if(e.teardown)for(let r of Le(e.teardown))t.add(r);return t}function nt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ce(e,t,r,o=0,i){let n=" ".repeat(o),s=[],a=$i(e),c=nt(a),l=i?.only?"test.only":"test";s.push(`${n}${l}('${t}', async (${c}) => {`),i?.skip===!0?s.push(`${n} test.skip();`):typeof i?.skip=="string"&&s.push(`${n} test.skip(true, '${oe(i.skip)}');`),i?.fail===!0?s.push(`${n} test.fail();`):typeof i?.fail=="string"&&s.push(`${n} test.fail(true, '${oe(i.fail)}');`),i?.slow&&s.push(`${n} test.slow();`),i?.timeout&&s.push(`${n} test.setTimeout(${i.timeout});`);let p=e.teardown&&e.teardown.length>0,u=o+1;if(p){if(s.push(`${n} try {`),e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let f=Q(e.statements,u+1,r);s.push(...f)}s.push(`${n} } finally {`),s.push(`${n} // Teardown`);let g=Q(e.teardown,u+1,r,"teardown");s.push(...g),s.push(`${n} }`)}else if(e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let g=Q(e.statements,u,r);s.push(...g)}return s.push(`${n}});`),s}function vr(e,t,r){let o=[],i=Tr(t),n=Le(i),s=nt(n);return o.push(`test.${e}(async (${s}) => {`),o.push(...Q(i,1,r,e)),o.push("});"),o}function Ie(e,t,r,o){let i=" ".repeat(o),n=[],s=Tr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};n.push(`${i}test.${e}(async ({ browser }, workerInfo) => {`),n.push(`${i} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),n.push(...Q(s,o+1,c,e)),n.push(`${i} await page.close();`),n.push(`${i}});`)}else{let c=Le(s),l=nt(c);n.push(`${i}test.${e}(async (${l}) => {`),n.push(...Q(s,o+1,r,e)),n.push(`${i}});`)}return n}function Tr(e){let r=di({goal:"_hook",statements:e});return L(r).statements??[]}function kr(e,t){let r=ye(e);for(let[o,i]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(i));return L(r)}function Ar(){return["import { test, expect } from 'shiplightai/fixture';"]}function Er(e,t){if(t.size>0){let r=0;for(let i=0;i<e.length;i++)e[i].startsWith("import ")&&(r=i+1);let o=Array.from(t);e.splice(r,0,...o)}}function Mr(e,t){let r={expandingPaths:new Set([$r(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=ae(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=ae(o.teardown,t,r));for(let i of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[i])&&(o[i]=ae(o[i],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ae(e,t,r){let o=[];for(let i of e)if(Ri(i)){let n=Ni(i,t,r);o.push(...n)}else o.push(Di(i,t,r));return o}function Ri(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Ni(e,t,r){if(r.depth>=xr)throw new Error(`Template expansion exceeded maximum depth of ${xr}. Check for deeply nested or circular template references.`);let o=$r(Li(t),e.template);if(r.expandingPaths.has(o))throw new Error(`Circular template reference detected: ${o} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${o}`);r.referencedPaths.add(o);let i;try{i=Oi(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let n=_r(i);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let s=n.params||[],a=e.params||{};for(let p of s)if(!(p in a))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${s.join(", ")}]`);let c=n.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(a).length>0){let u=Ci(c);for(let[g,f]of Object.entries(a))u=u.split(`<<${g}>>`).join(String(f));c=_r(u)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return ae(c,o,l)}function Di(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=ae(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=ae(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=ae(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=ae(o.DO,t,r)),o}function at(e,t){let r=Ii(e),o=r?.name,i=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new st('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Ui(r,o,i,n,t)}return Fi(r,o,i,n,t)}function Fi(e,t,r,o,i){let n=e?.beforeEach,s=e?.afterEach,a=Ir(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,u=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new st(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(i&&e&&typeof e=="object"){let b=Mr(e,i);e=b.doc,f=b.referencedTemplatePaths}let d=Pr(e),h=L(d);return i&&(fe(h.statements??[],i,"main"),h.teardown&&fe(h.teardown,i,"teardown")),{testFlow:h,name:t,tags:r,use:o,beforeEach:n,afterEach:s,parameters:a,timeout:c,skip:l,fail:p,only:u,slow:g,referencedTemplatePaths:f}}function Ui(e,t,r,o,i){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,a=n.afterAll,c=n.beforeEach,l=n.afterEach,p=[],u=n.tests.map(d=>{if(!d.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(d.statements)||d.statements.length===0)throw new Error(`Suite test "${d.name}" must have a non-empty "statements" array.`);let h={goal:d.name,statements:d.statements};d.teardown&&(h.teardown=d.teardown);let b=[],w=h;if(i&&typeof h=="object"){let k=Mr(h,i);w=k.doc,b=k.referencedTemplatePaths,p.push(...b)}let v=Pr(w),y=L(v),A=Ir(d.parameters);return{testFlow:y,name:d.name,tags:Array.isArray(d.tags)?d.tags:void 0,parameters:A,timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}}),g=n.base_url,f=g?{...o,baseURL:g}:o;return{suite:{beforeAll:s,afterAll:a,beforeEach:c,afterEach:l,tests:u},name:t,tags:r,use:f,referencedTemplatePaths:p}}function Ir(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function fe(e,t,r){for(let o=0;o<e.length;o++){let i=e[o],n=`${r}.${o}`,s=i.description||"";if(i.uid=Bi(t,n,s),i.type===J.STEP)fe(i.statements,t,n);else if(i.type===J.IF_ELSE){let a=i;fe(a.then,t,`${n}.then`),a.else&&fe(a.else,t,`${n}.else`)}else i.type===J.WHILE_LOOP&&fe(i.body,t,`${n}.body`)}}function Bi(e,t,r){let o=Mi("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${o.slice(0,8)}-${o.slice(8,12)}-${o.slice(12,16)}-${o.slice(16,20)}-${o.slice(20,32)}`}function Or(e,t){let r;try{r=Gi(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return Ki(r,e,t)}function Ki(e,t,r){let o=/\btemplate:\s/.test(e),i=/^suite:/m.test(e),n=o||i?null:et(e);if(n&&!n.valid)return{valid:!1,errors:n.errors,warnings:[],stats:n.stats};let s,a,c=[];try{let l=r?.parsed??at(e,t);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore},u=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?s=Ei(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):s=Ai(l.testFlow,{...p,testName:l.name,tags:l.tags,use:u,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let g=s.split(`
31
94
  `).filter(f=>!f.startsWith("import ")).join(`
32
- `);new Function(g),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),gi(mi(a),{recursive:!0}),fi(a,s)}catch(l){let p=l instanceof Xe?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:n?.warnings??[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var Ye,Go,Wo,I,ei,ar,Xe,Ze=_(()=>{"use strict";re();re();re();re();Ye=5e3;Go=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],Wo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];I=new Map;S("click",e=>{let t=ke(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Ye;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",I.get("click"));S("click_element_by_index",I.get("click"));S("double_click",e=>ne("double_click",e));S("double_click_on_element",I.get("double_click"));S("right_click",e=>ne("right_click",e));S("right_click_on_element",I.get("right_click"));S("hover",e=>ne("hover",e));S("hover_element_by_index",I.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ne("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",I.get("input_text"));S("clear_input",e=>ne("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",I.get("press"));S("send_keys_on_element",e=>{let t=ke(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let o=e.action_data?.kwargs?.timeout_ms??Ye;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ne("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",I.get("scroll"));S("scroll_up",I.get("scroll"));S("scroll_element",I.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>ne("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",I.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",i=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&i){let s=r.code.split(`
95
+ `);new Function(g),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Hi(Wi(a),{recursive:!0}),ji(a,s)}catch(l){let p=l instanceof st?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:n?.warnings??[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var it,gi,mi,O,Pi,xr,st,ct=_(()=>{"use strict";se();se();se();se();it=5e3;gi=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],mi=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];O=new Map;S("click",e=>{let t=Oe(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",O.get("click"));S("click_element_by_index",O.get("click"));S("double_click",e=>le("double_click",e));S("double_click_on_element",O.get("double_click"));S("right_click",e=>le("right_click",e));S("right_click_on_element",O.get("right_click"));S("hover",e=>le("hover",e));S("hover_element_by_index",O.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return le("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",O.get("input_text"));S("clear_input",e=>le("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",O.get("press"));S("send_keys_on_element",e=>{let t=Oe(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let o=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return le("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",O.get("scroll"));S("scroll_up",O.get("scroll"));S("scroll_element",O.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>le("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",O.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",i=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&i){let s=r.code.split(`
33
96
  `),a=JSON.stringify(i);return["{ const _t = Date.now(); try {",...s.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${a}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${a}\`);`,` await agent.assert(page, ${a}, ${JSON.stringify(t||"")});`,"} }"]}return o?r.code.split(`
34
- `):i?[`await agent.assert(page, ${JSON.stringify(i)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",I.get("verify"));S("assert",I.get("verify"));S("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let o=JSON.stringify(r),i=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${o}, '${t||""}', ${i});`]});S("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});S("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,o=e.action_data?.kwargs?.variable_name;if(!r||!o)return["// Skipping ai_extract: missing element_description or variable_name"];let i=JSON.stringify(r),n=JSON.stringify(o);return[`await agent.extract(page, ${i}, ${n}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,o=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${o}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});S("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});S("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],o=t.split(`
35
- `);for(let i of o)r.push(` ${i}`);return r.push("}"),r});S("function",(e,t,r)=>{let o=e.action_data?.kwargs||{},i=o.functionName;if(i&&i.includes("#")){let[s,a]=i.split("#");if(s&&a){let c=s.replace(/\.(ts|js|mjs)$/,""),l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let p={...o,functionName:a},h=ir(p);return h?[h.endsWith(";")?h:`${h};`]:["// Skipping function: invalid export pattern"]}}let n=ir(o);return n?[n.endsWith(";")?n:`${n};`]:["// Skipping function: missing functionName"]});S("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});S("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],o={};return t.paths?o.paths=t.paths:t.path&&(o.path=t.path),t.use_file_input&&(o.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(o)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(i=>` ${i},`),"});"]});S("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);S("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);S("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});S("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(o=>` ${o},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
36
- `):["// Skipping js_action: missing code"]});ei=["testContext","request"];ar=5;Xe=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as bi}from"express";import*as B from"fs/promises";import*as G from"path";import{stringify as wi}from"yaml";function $e(e){if(!e||e.length===0)return[];let r=wi({goal:"_hook",statements:e});return O(r).statements??[]}async function yr(e){try{let t=await B.readdir(e,{withFileTypes:!0});for(let r of t)if(!(r.name==="node_modules"||r.name.startsWith("."))&&(r.isFile()&&r.name.endsWith(".test.yaml")||r.isDirectory()&&await yr(G.join(e,r.name))))return!0}catch{}return!1}function br(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:i}=e,n=bi();n.get("/api/files",async(a,c)=>{try{let l=typeof a.query.dir=="string"?a.query.dir:t,p=G.resolve(l),h=await B.readdir(p,{withFileTypes:!0}),g=[];for(let u of h)if(u.name!=="node_modules"&&!u.name.startsWith("."))if(u.isDirectory()){let d=G.join(p,u.name);await yr(d)&&g.push({name:u.name,type:"directory",path:d})}else u.isFile()&&u.name.endsWith(".test.yaml")&&g.push({name:u.name,type:"file",path:G.join(p,u.name)});g.sort((u,d)=>u.type!==d.type?u.type==="directory"?-1:1:u.name.localeCompare(d.name));let f=G.dirname(p);c.json({dir:p,parent:f!==p?f:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(l){console.error("[debugger] Error listing files:",l),c.status(500).json({error:l.message})}});function s(a){if(typeof a=="string"&&a){let c=G.resolve(a);return c.endsWith(".test.yaml")?{filePath:c}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return n.get("/api/test-flow",async(a,c)=>{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath;try{i?.(p);let h=await B.readFile(p,"utf-8"),g=await B.stat(p),f=fe(h),u=qe(h,p);if(u.suite){let d=u.suite,w={tests:d.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:$e(d.beforeAll),afterAll:$e(d.afterAll),beforeEach:$e(d.beforeEach),afterEach:$e(d.afterEach)},b={version:"1.3.0",baseURL:u.use?.baseURL,testGroup:w};c.json({isSuite:!0,testFlow:b,metadata:f,name:u.name,tags:u.tags,use:u.use,filePath:p,fileName:G.basename(p),lastModified:g.mtimeMs})}else{let d=u.testFlow;we(h,d),c.json({isSuite:!1,testFlow:d,metadata:f,name:u.name,tags:u.tags,use:u.use,filePath:p,fileName:G.basename(p),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return c.status(404).json({error:`File not found: ${p}`});console.error("[debugger] Error loading test flow:",h),c.status(500).json({error:h.message})}}),n.put("/api/test-flow",async(a,c)=>{try{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath,{testFlow:h,metadata:g}=a.body;if(!h)return c.status(400).json({error:"testFlow is required"});let f=he(h,g),u=p+".tmp";await B.writeFile(u,f,"utf-8"),await B.rename(u,p);let d=await B.stat(p);c.json({success:!0,lastModified:d.mtimeMs})}catch(l){console.error("[debugger] Error saving test flow:",l),c.status(500).json({error:l.message})}}),n}var wr=_(()=>{"use strict";re();Ze()});import*as Sr from"http";function vr(e){let{getPort:t,ensureReady:r}=e,o=null,i=0,n=5e3,s=async(a,c,l)=>{let p=t();if(p===null){if(!o&&Date.now()-i<n)return c.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});o||(o=r().catch(f=>{throw i=Date.now(),o=null,f}));try{p=await o}catch(f){return c.status(503).json({status:"error",message:"Failed to start sandbox: "+f.message})}}let h=await new Promise((f,u)=>{let d=[];a.on("data",w=>d.push(w)),a.on("end",()=>f(Buffer.concat(d))),a.on("error",u),a.readableEnded&&f(Buffer.alloc(0))}),g=Sr.request({hostname:"localhost",port:p,path:a.originalUrl,method:a.method,headers:{"content-type":a.headers["content-type"]||"application/json","content-length":String(h.length)},timeout:3e5},f=>{c.writeHead(f.statusCode,f.headers),f.on("data",u=>{c.write(u)}),f.on("end",()=>{c.end()}),f.on("error",()=>{c.writableEnded||c.end()})});g.on("error",f=>{c.headersSent||c.status(502).json({status:"error",message:"Inner server unavailable: "+f.message})}),g.end(h)};return s.reset=()=>{o=null},s.getPort=()=>t(),s.ensureReady=async()=>{let a=t();return a!==null?a:(o||(o=r().catch(c=>{throw i=Date.now(),o=null,c})),o)},s}var _r=_(()=>{"use strict"});var xr={};te(xr,{startDebuggerServer:()=>Si});import Pe from"express";import*as ie from"path";async function Si(e){let{initialDir:t,initialFile:r,projectRoot:o,port:i}=e,n=r,s=d=>{n=d,e.onFileSelected?.(d)},a=null,c=null,l=vr({getPort:()=>a,ensureReady:async()=>{console.error("[debugger] Initializing sandbox on first use...");let d=await e.innerServerFactory();return a=d.port,c=d.cleanup,d.port}}),p=Pe();p.use((d,w,b)=>{if(w.setHeader("Access-Control-Allow-Origin","*"),w.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),w.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),d.method==="OPTIONS")return w.sendStatus(204);b()}),p.post("/api/int-runner/create-session",Pe.json(),async(d,w)=>{n&&(d.body.testFilePath=n);try{let b=l.getPort();b||(b=await l.ensureReady());let v=await fetch(`http://localhost:${b}/api/int-runner/create-session`,{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify(d.body)}),y=await v.json();w.status(v.status).json(y)}catch(b){w.status(502).json({status:"error",message:b.message})}}),p.post("/api/int-runner/terminate-session",async(d,w)=>{let b=c;a=null,c=null,l.reset(),b&&(console.error("[debugger] Tearing down inner Playwright process for fresh restart..."),await b().catch(v=>console.error("[debugger] Cleanup error:",v.message))),w.json({status:"success",details:"Session terminated"})}),p.use("/api/int-runner",l),p.use("/api/browser-cdp",l),p.use(Pe.json({limit:"10mb"})),p.use(br({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:s}));let h=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=h.includes(ie.sep+"src"+ie.sep)?ie.resolve(h,"../../dist/static"):ie.join(h,"static");p.use(Pe.static(g)),p.get("*path",(d,w,b)=>{if(d.path.startsWith("/api/"))return b();w.sendFile(ie.join(g,"index.html"),v=>{v&&w.send(vi(t,i))})});let f=await new Promise((d,w)=>{let b=p.listen(i,"localhost",()=>{d(b)});b.on("error",w)}),u=`http://localhost:${i}`;return console.error(`[debugger] Server running at ${u}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:u,close:async()=>{c&&await c(),await new Promise((d,w)=>{f.close(b=>b?w(b):d())})}}}function vi(e,t){return`<!DOCTYPE html>
97
+ `):i?[`await agent.assert(page, ${JSON.stringify(i)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",O.get("verify"));S("assert",O.get("verify"));S("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let o=JSON.stringify(r),i=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${o}, '${t||""}', ${i});`]});S("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});S("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,o=e.action_data?.kwargs?.variable_name;if(!r||!o)return["// Skipping ai_extract: missing element_description or variable_name"];let i=JSON.stringify(r),n=JSON.stringify(o);return[`await agent.extract(page, ${i}, ${n}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,o=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${o}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});S("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});S("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],o=t.split(`
98
+ `);for(let i of o)r.push(` ${i}`);return r.push("}"),r});S("function",(e,t,r)=>{let o=e.action_data?.kwargs||{},i=o.functionName;if(i&&i.includes("#")){let[s,a]=i.split("#");if(s&&a){let c=s.replace(/\.(ts|js|mjs)$/,""),l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let p={...o,functionName:a},u=Sr(p);return u?[u.endsWith(";")?u:`${u};`]:["// Skipping function: invalid export pattern"]}}let n=Sr(o);return n?[n.endsWith(";")?n:`${n};`]:["// Skipping function: missing functionName"]});S("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});S("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],o={};return t.paths?o.paths=t.paths:t.path&&(o.path=t.path),t.use_file_input&&(o.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(o)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(i=>` ${i},`),"});"]});S("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);S("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);S("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});S("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(o=>` ${o},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
99
+ `):["// Skipping js_action: missing code"]});Pi=["testContext","request"];xr=5;st=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as zi}from"express";import*as j from"fs/promises";import*as H from"path";import{stringify as Vi}from"yaml";function Re(e){if(!e||e.length===0)return[];let r=Vi({goal:"_hook",statements:e});return L(r).statements??[]}async function Lr(e){try{let t=await j.readdir(e,{withFileTypes:!0});for(let r of t)if(!(r.name==="node_modules"||r.name.startsWith("."))&&(r.isFile()&&r.name.endsWith(".test.yaml")||r.isDirectory()&&await Lr(H.join(e,r.name))))return!0}catch{}return!1}function Cr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:i}=e,n=zi();n.get("/api/files",async(a,c)=>{try{let l=typeof a.query.dir=="string"?a.query.dir:t,p=H.resolve(l),u=await j.readdir(p,{withFileTypes:!0}),g=[];for(let d of u)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let h=H.join(p,d.name);await Lr(h)&&g.push({name:d.name,type:"directory",path:h})}else d.isFile()&&d.name.endsWith(".test.yaml")&&g.push({name:d.name,type:"file",path:H.join(p,d.name)});g.sort((d,h)=>d.type!==h.type?d.type==="directory"?-1:1:d.name.localeCompare(h.name));let f=H.dirname(p);c.json({dir:p,parent:f!==p?f:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(l){console.error("[debugger] Error listing files:",l),c.status(500).json({error:l.message})}});function s(a){if(typeof a=="string"&&a){let c=H.resolve(a);return c.endsWith(".test.yaml")?{filePath:c}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return n.get("/api/test-flow",async(a,c)=>{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath;try{i?.(p);let u=await j.readFile(p,"utf-8"),g=await j.stat(p),f=we(u),d=at(u,p);if(d.suite){let h=d.suite,b={tests:h.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:Re(h.beforeAll),afterAll:Re(h.afterAll),beforeEach:Re(h.beforeEach),afterEach:Re(h.afterEach)},w={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:b};c.json({isSuite:!0,testFlow:w,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:H.basename(p),lastModified:g.mtimeMs})}else{let h=d.testFlow;Ae(u,h),c.json({isSuite:!1,testFlow:h,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:H.basename(p),lastModified:g.mtimeMs})}}catch(u){if(u.code==="ENOENT")return c.status(404).json({error:`File not found: ${p}`});console.error("[debugger] Error loading test flow:",u),c.status(500).json({error:u.message})}}),n.put("/api/test-flow",async(a,c)=>{try{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath,{testFlow:u,metadata:g}=a.body;if(!u)return c.status(400).json({error:"testFlow is required"});let f=ye(u,g),d=p+".tmp";await j.writeFile(d,f,"utf-8"),await j.rename(d,p);let h=await j.stat(p);c.json({success:!0,lastModified:h.mtimeMs})}catch(l){console.error("[debugger] Error saving test flow:",l),c.status(500).json({error:l.message})}}),n}var Rr=_(()=>{"use strict";se();ct()});import*as Nr from"http";function Dr(e){let{getPort:t,ensureReady:r}=e,o=null,i=0,n=5e3,s=async(a,c,l)=>{let p=t();if(p===null){if(!o&&Date.now()-i<n)return c.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});o||(o=r().catch(f=>{throw i=Date.now(),o=null,f}));try{p=await o}catch(f){return c.status(503).json({status:"error",message:"Failed to start sandbox: "+f.message})}}let u=await new Promise((f,d)=>{let h=[];a.on("data",b=>h.push(b)),a.on("end",()=>f(Buffer.concat(h))),a.on("error",d),a.readableEnded&&f(Buffer.alloc(0))}),g=Nr.request({hostname:"localhost",port:p,path:a.originalUrl,method:a.method,headers:{"content-type":a.headers["content-type"]||"application/json","content-length":String(u.length)},timeout:3e5},f=>{c.writeHead(f.statusCode,f.headers),f.on("data",d=>{c.write(d)}),f.on("end",()=>{c.end()}),f.on("error",()=>{c.writableEnded||c.end()})});g.on("error",f=>{c.headersSent||c.status(502).json({status:"error",message:"Inner server unavailable: "+f.message})}),g.end(u)};return s.reset=()=>{o=null},s.getPort=()=>t(),s.ensureReady=async()=>{let a=t();return a!==null?a:(o||(o=r().catch(c=>{throw i=Date.now(),o=null,c})),o)},s}var Fr=_(()=>{"use strict"});var Ur={};te(Ur,{startDebuggerServer:()=>Yi});import Ne from"express";import*as ce from"path";async function Yi(e){let{initialDir:t,initialFile:r,projectRoot:o,port:i}=e,n=r,s=h=>{n=h,e.onFileSelected?.(h)},a=null,c=null,l=Dr({getPort:()=>a,ensureReady:async()=>{console.error("[debugger] Initializing sandbox on first use...");let h=await e.innerServerFactory();return a=h.port,c=h.cleanup,h.port}}),p=Ne();p.use((h,b,w)=>{if(b.setHeader("Access-Control-Allow-Origin","*"),b.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),b.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),h.method==="OPTIONS")return b.sendStatus(204);w()}),p.post("/api/int-runner/create-session",Ne.json(),async(h,b)=>{n&&(h.body.testFilePath=n);try{let w=l.getPort();w||(w=await l.ensureReady());let v=await fetch(`http://localhost:${w}/api/int-runner/create-session`,{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify(h.body)}),y=await v.json();b.status(v.status).json(y)}catch(w){b.status(502).json({status:"error",message:w.message})}}),p.post("/api/int-runner/terminate-session",async(h,b)=>{let w=c;a=null,c=null,l.reset(),w&&(console.error("[debugger] Tearing down inner Playwright process for fresh restart..."),await w().catch(v=>console.error("[debugger] Cleanup error:",v.message))),b.json({status:"success",details:"Session terminated"})}),p.use("/api/int-runner",l),p.use("/api/browser-cdp",l),p.use(Ne.json({limit:"10mb"})),p.use(Cr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:s}));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=u.includes(ce.sep+"src"+ce.sep)?ce.resolve(u,"../../dist/static"):ce.join(u,"static");p.use(Ne.static(g)),p.get("*path",(h,b,w)=>{if(h.path.startsWith("/api/"))return w();b.sendFile(ce.join(g,"index.html"),v=>{v&&b.send(Ji(t,i))})});let f=await new Promise((h,b)=>{let w=p.listen(i,"localhost",()=>{h(w)});w.on("error",b)}),d=`http://localhost:${i}`;return console.error(`[debugger] Server running at ${d}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:d,close:async()=>{c&&await c(),await new Promise((h,b)=>{f.close(w=>w?b(w):h())})}}}function Ji(e,t){return`<!DOCTYPE html>
37
100
  <html>
38
101
  <head>
39
102
  <meta charset="utf-8">
@@ -51,57 +114,59 @@ test('__shiplight_debug__', async ({ page, agent }) => {
51
114
  <p>Directory: <code>${e}</code></p>
52
115
  <p>Server: localhost:${t}</p>
53
116
  </body>
54
- </html>`}var Tr=_(()=>{"use strict";wr();_r()});var kr={};te(kr,{startDebugger:()=>xi});import*as z from"fs";import*as Q from"path";async function xi(e){let t,r=6174,o,i=!1,n=!0;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),y++):e[y]==="--url"&&e[y+1]?(o=e[y+1],y++):e[y]==="--new"?i=!0:e[y]==="--open"?n=!1:e[y]==="--no-open"?n=!0:e[y]==="--help"||e[y]==="-h"?(Ti(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let s,a;if(t){let y=Q.resolve(t);z.existsSync(y)&&z.statSync(y).isDirectory()?s=y:(s=Q.dirname(y),a=y)}else s=process.cwd();if(i&&a&&!z.existsSync(a)){let y=Q.dirname(a);z.existsSync(y)||z.mkdirSync(y,{recursive:!0}),z.writeFileSync(a,_i(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${a}`)}a&&!z.existsSync(a)&&(console.error(`Error: File not found: ${a}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(Fe(),Ne)),l=c(s),p=l?Q.dirname(l):s;(await import("dotenv")).config({path:Q.join(p,".env"),override:!0});let g=a;l||(console.error("Error: No Playwright config found in "+s),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let f=async()=>{let{spawnPlaywrightProcess:y}=await Promise.resolve().then(()=>(Fe(),Ne));console.error("[debugger] Starting Playwright sandbox...");let A=g||Q.join(s,"_debug.test.yaml");return y({yamlFilePath:A,configPath:l})},u=y=>{g=y},{startDebuggerServer:d}=await Promise.resolve().then(()=>(Tr(),xr));console.log(a?`Starting debugger for: ${a}`:`Starting debugger in: ${s}`),l&&console.log(`Using Playwright config: ${l}`);let w;try{w=await d({initialDir:s,initialFile:a,projectRoot:p,port:r,innerServerFactory:f,onFileSelected:u})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),y}if(yt(r,s),console.log(`Debugger running at: ${w.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!n)try{let{default:y}=await import("open");await y(w.url)}catch{}let b=!1,v=async()=>{b&&(console.log(`
55
- Force exiting...`),process.exit(1)),b=!0,console.log(`
56
- Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{bt(r,s),await w.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function Ti(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var _i,Ar=_(()=>{"use strict";wt();_i=e=>`goal: New test
117
+ </html>`}var Br=_(()=>{"use strict";Rr();Fr()});var Gr={};te(Gr,{startDebugger:()=>qi});import*as X from"fs";import*as ie from"path";async function qi(e){let t,r=6174,o,i=!1,n=!0;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),y++):e[y]==="--url"&&e[y+1]?(o=e[y+1],y++):e[y]==="--new"?i=!0:e[y]==="--open"?n=!1:e[y]==="--no-open"?n=!0:e[y]==="--help"||e[y]==="-h"?(Zi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let s,a;if(t){let y=ie.resolve(t);X.existsSync(y)&&X.statSync(y).isDirectory()?s=y:(s=ie.dirname(y),a=y)}else s=process.cwd();if(i&&a&&!X.existsSync(a)){let y=ie.dirname(a);X.existsSync(y)||X.mkdirSync(y,{recursive:!0}),X.writeFileSync(a,Xi(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${a}`)}a&&!X.existsSync(a)&&(console.error(`Error: File not found: ${a}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(Ye(),Ve)),l=c(s),p=l?ie.dirname(l):s;(await import("dotenv")).config({path:ie.join(p,".env"),override:!0});let g=a;l||(console.error("Error: No Playwright config found in "+s),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let f=async()=>{let{spawnPlaywrightProcess:y}=await Promise.resolve().then(()=>(Ye(),Ve));console.error("[debugger] Starting Playwright sandbox...");let A=g||ie.join(s,"_debug.test.yaml");return y({yamlFilePath:A,configPath:l})},d=y=>{g=y},{startDebuggerServer:h}=await Promise.resolve().then(()=>(Br(),Ur));console.log(a?`Starting debugger for: ${a}`:`Starting debugger in: ${s}`),l&&console.log(`Using Playwright config: ${l}`);let b;try{b=await h({initialDir:s,initialFile:a,projectRoot:p,port:r,innerServerFactory:f,onFileSelected:d})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),y}if(Lt(r,s),console.log(`Debugger running at: ${b.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!n)try{let{default:y}=await import("open");await y(b.url)}catch{}let w=!1,v=async()=>{w&&(console.log(`
118
+ Force exiting...`),process.exit(1)),w=!0,console.log(`
119
+ Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Ct(r,s),await b.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function Zi(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var Xi,jr=_(()=>{"use strict";Rt();Xi=e=>`goal: New test
57
120
  statements:
58
121
  - URL: ${e}
59
122
  - "TODO: Add your first step"
60
- `});var Qe={};te(Qe,{lookupActionStores:()=>ki,updateActionStores:()=>Ai});function Er(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function ki(e){let t=Er();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),i=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(o),!i.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${i.status}`),new Map;let n=await i.json(),s=new Map;for(let[a,c]of Object.entries(n.stores??{}))s.set(a,c);return s}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Ai(e){let t=Er();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),i={};for(let[a,c]of e)i[a]=c;let n=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:i}),signal:r.signal});return clearTimeout(o),n.ok?(await n.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${n.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var et=_(()=>{"use strict"});import*as R from"fs";import*as me from"path";import{globSync as Ei}from"glob";function $r(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new rt:new tt(e)}function ye(e){return e.replace(/\//g,"__")+".json"}function Pi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var $i,tt,rt,Pr=_(()=>{"use strict";re();$i=".shiplight/action-cache";tt=class{constructor(t){this.cwd=t;this.cacheDir=me.join(t,$i)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!R.existsSync(this.cacheDir))return r;for(let o of t){let i=me.join(this.cacheDir,ye(o));try{if(R.existsSync(i)){let n=R.readFileSync(i,"utf-8");r.set(o,JSON.parse(n))}}catch{}}return r}async update(t){if(t.size===0)return 0;R.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,i]of t)try{let n=me.join(this.cacheDir,ye(o)),s=Ke();if(R.existsSync(n))try{s=JSON.parse(R.readFileSync(n,"utf-8"))}catch{}let a={...s,entries:{...s.entries,...i.entries}};R.writeFileSync(n,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!R.existsSync(this.cacheDir))return;let t=Ei("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let i of t)try{let n=R.readFileSync(me.join(this.cacheDir,i),"utf-8"),s=JSON.parse(n),a=Pi(i);r.set(a,s),o+=Object.keys(s.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${o} cached action entit${o===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},rt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(et(),Qe));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(et(),Qe));return r(t)}loadAll(){}}});var Lr={};te(Lr,{buildTestPathsFromGitInfo:()=>Or,cloudKeyToCwdRelPath:()=>nt,runTests:()=>Ii});import{spawn as Mi,execFileSync as Mr}from"child_process";import*as D from"fs";import*as L from"path";import{globSync as ot}from"glob";async function Ii(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(h=>D.existsSync(L.join(t,h)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
61
- `));let i=e.includes("--magic"),s=e.filter(h=>h!=="--magic").map(h=>h.endsWith(".test.yaml")?h.replace(/\.test\.yaml$/,".yaml.spec.ts"):h),a=$r(t);await Li(t,a);let c={...process.env};i&&(c.SHIPLIGHT_MAGIC="1");let l=Mi("npx",["playwright","test",...s],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(h=>{l.on("close",g=>h(g??1))});await Ci(t,a),process.exit(p)}function it(){try{return Mr("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Oi(){try{let e=Mr("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Ir(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=Oi(),i=it();return Or(e,t,o,i)}function Or(e,t,r,o){return{testPaths:e.map(n=>{let s=o?L.relative(o,L.resolve(t,n)):n;return`${r}${s}`}),branchPrefix:r}}function nt(e,t,r,o){let i=t&&e.startsWith(t)?e.slice(t.length):e;return r?L.relative(o,L.resolve(r,i)):i}async function Li(e,t){try{let r=ot("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:i}=Ir(r,e,t.isCloud),n=await t.lookup(o);if(n.size===0)return;let s=L.join(e,".shiplight","action-cache");D.mkdirSync(s,{recursive:!0});let a=it(),c=0;for(let[l,p]of n){let h=t.isCloud?nt(l,i,a,e):l,g=L.join(s,ye(h));D.writeFileSync(g,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function Ci(e,t){try{let r=L.join(e,"test-results");if(!D.existsSync(r))return;let o=ot("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let i={};for(let h of o)try{let g=D.readFileSync(L.join(r,h),"utf-8"),f=JSON.parse(g);f?.entries&&Object.assign(i,f.entries)}catch{}if(Object.keys(i).length===0)return;let n=ot("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:s,branchPrefix:a}=Ir(n,e,t.isCloud),c=new Map;for(let h=0;h<n.length;h++){let g=n[h],f=s[h],u=L.join(e,g.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!D.existsSync(u))continue;let d=D.readFileSync(u,"utf-8"),w={};for(let[b,v]of Object.entries(i))d.includes(b)&&(w[b]=v);if(Object.keys(w).length>0){let b=t.isCloud?nt(f,a,it(),e):f,v=L.join(e,".shiplight","action-cache",ye(b)),y={};if(D.existsSync(v))try{y=JSON.parse(D.readFileSync(v,"utf-8")).entries}catch{}c.set(f,{version:"1.0",entries:{...y,...w}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((h,g)=>h+Object.keys(g.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Cr=_(()=>{"use strict";Pr()});function $(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Ie(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function Rr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Me(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function Ri(e){return`<span class="badge badge-${e}">${e}</span>`}function Di(e){let t=e.duration!=null?`<span class="step-duration">${Ie(e.duration)}</span>`:"",r=Me(e.status);if(e.screenshot||e.message||e.error){let i="";e.screenshot&&(i=`<img src="${$(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n="";e.error&&(n=`<div class="step-error"><pre>${$(e.error)}</pre></div>`);let s="";e.message&&!e.error&&(s=`<div class="step-message">${$(e.message)}</div>`);let a=e.status==="failure"?" open":"";return`
62
- <details class="step-details step step-${e.status}"${a}>
123
+ `});var lt={};te(lt,{lookupActionStores:()=>Qi,updateActionStores:()=>en});function Hr(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function Qi(e){let t=Hr();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),i=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(o),!i.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${i.status}`),new Map;let n=await i.json(),s=new Map;for(let[a,c]of Object.entries(n.stores??{}))s.set(a,c);return s}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function en(e){let t=Hr();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),i={};for(let[a,c]of e)i[a]=c;let n=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:i}),signal:r.signal});return clearTimeout(o),n.ok?(await n.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${n.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var pt=_(()=>{"use strict"});import*as D from"fs";import*as Se from"path";import{globSync as tn}from"glob";function Wr(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new dt:new ut(e)}function ve(e){return e.replace(/\//g,"__")+".json"}function on(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var rn,ut,dt,Kr=_(()=>{"use strict";se();rn=".shiplight/action-cache";ut=class{constructor(t){this.cwd=t;this.cacheDir=Se.join(t,rn)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!D.existsSync(this.cacheDir))return r;for(let o of t){let i=Se.join(this.cacheDir,ve(o));try{if(D.existsSync(i)){let n=D.readFileSync(i,"utf-8");r.set(o,JSON.parse(n))}}catch{}}return r}async update(t){if(t.size===0)return 0;D.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,i]of t)try{let n=Se.join(this.cacheDir,ve(o)),s=tt();if(D.existsSync(n))try{s=JSON.parse(D.readFileSync(n,"utf-8"))}catch{}let a={...s,entries:{...s.entries,...i.entries}};D.writeFileSync(n,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!D.existsSync(this.cacheDir))return;let t=tn("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let i of t)try{let n=D.readFileSync(Se.join(this.cacheDir,i),"utf-8"),s=JSON.parse(n),a=on(i);r.set(a,s),o+=Object.keys(s.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${o} cached action entit${o===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},dt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}loadAll(){}}});var Jr={};te(Jr,{buildTestPathsFromGitInfo:()=>Yr,cloudKeyToCwdRelPath:()=>gt,runTests:()=>sn});import{spawn as nn,execFileSync as zr}from"child_process";import*as F from"fs";import*as C from"path";import{globSync as ht}from"glob";async function sn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(u=>F.existsSync(C.join(t,u)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
124
+ `));let i=e.includes("--magic"),s=e.filter(u=>u!=="--magic").map(u=>u.endsWith(".test.yaml")?u.replace(/\.test\.yaml$/,".yaml.spec.ts"):u),a=Wr(t);await cn(t,a);let c={...process.env};i&&(c.SHIPLIGHT_MAGIC="1");let l=nn("npx",["playwright","test",...s],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(u=>{l.on("close",g=>u(g??1))});await ln(t,a),process.exit(p)}function ft(){try{return zr("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function an(){try{let e=zr("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Vr(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=an(),i=ft();return Yr(e,t,o,i)}function Yr(e,t,r,o){return{testPaths:e.map(n=>{let s=o?C.relative(o,C.resolve(t,n)):n;return`${r}${s}`}),branchPrefix:r}}function gt(e,t,r,o){let i=t&&e.startsWith(t)?e.slice(t.length):e;return r?C.relative(o,C.resolve(r,i)):i}async function cn(e,t){try{let r=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:i}=Vr(r,e,t.isCloud),n=await t.lookup(o);if(n.size===0)return;let s=C.join(e,".shiplight","action-cache");F.mkdirSync(s,{recursive:!0});let a=ft(),c=0;for(let[l,p]of n){let u=t.isCloud?gt(l,i,a,e):l,g=C.join(s,ve(u));F.writeFileSync(g,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function ln(e,t){try{let r=C.join(e,"test-results");if(!F.existsSync(r))return;let o=ht("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let i={};for(let u of o)try{let g=F.readFileSync(C.join(r,u),"utf-8"),f=JSON.parse(g);f?.entries&&Object.assign(i,f.entries)}catch{}if(Object.keys(i).length===0)return;let n=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:s,branchPrefix:a}=Vr(n,e,t.isCloud),c=new Map;for(let u=0;u<n.length;u++){let g=n[u],f=s[u],d=C.join(e,g.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!F.existsSync(d))continue;let h=F.readFileSync(d,"utf-8"),b={};for(let[w,v]of Object.entries(i))h.includes(w)&&(b[w]=v);if(Object.keys(b).length>0){let w=t.isCloud?gt(f,a,ft(),e):f,v=C.join(e,".shiplight","action-cache",ve(w)),y={};if(F.existsSync(v))try{y=JSON.parse(F.readFileSync(v,"utf-8")).entries}catch{}c.set(f,{version:"1.0",entries:{...y,...b}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((u,g)=>u+Object.keys(g.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Xr=_(()=>{"use strict";Kr()});function P(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Fe(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function qr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function De(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function pn(e){return`<span class="badge badge-${e}">${e}</span>`}function un(e){if(!e.code)return"";let t=e.code.split(`
125
+ `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((i,n)=>{let s=e.codeStartLine+n,a=s===e.codeLine,c=String(s).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${P(i)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(o=>`<span class="code-line code-line-body">${P(o)}</span>`).join("")}</pre></div>`}function dn(e){let t=e.duration!=null?`<span class="step-duration">${Fe(e.duration)}</span>`:"",r=De(e.status);if(e.screenshot||e.message||e.error||e.code){let i="";e.screenshot&&(i=`<img src="${P(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n=e.screenshot?"":un(e),s="";e.error&&(s=`<div class="step-error"><pre>${P(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${P(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
126
+ <details class="step-details step step-${e.status}"${c}>
63
127
  <summary class="step-header">
64
128
  ${r}
65
- <span class="step-id">${$(e.stepId)}</span>
66
- <span class="step-description-collapsed">${$(e.description)}</span>
129
+ <span class="step-id">${P(e.stepId)}</span>
130
+ <span class="step-description-collapsed">${P(e.description)}</span>
67
131
  ${t}
68
132
  </summary>
69
133
  <div class="step-expanded">
70
134
  ${i}
71
- <div class="step-description-full">${$(e.description)}</div>
72
- ${s}
73
135
  ${n}
136
+ <div class="step-description-full">${P(e.description)}</div>
137
+ ${a}
138
+ ${s}
74
139
  </div>
75
140
  </details>`}return`
76
141
  <div class="step step-${e.status}">
77
142
  <div class="step-header">
78
143
  ${r}
79
- <span class="step-id">${$(e.stepId)}</span>
80
- <span class="step-description">${$(e.description)}</span>
144
+ <span class="step-id">${P(e.stepId)}</span>
145
+ <span class="step-description">${P(e.description)}</span>
81
146
  ${t}
82
147
  </div>
83
- </div>`}function Ni(e,t,r,o){let i=[];e&&i.push(`
148
+ </div>`}function hn(e,t,r,o){let i=[];e&&i.push(`
84
149
  <details class="artifact-section">
85
150
  <summary class="artifact-summary">Video</summary>
86
151
  <div class="artifact-content">
87
152
  <div class="video-container">
88
153
  <video controls preload="metadata" class="artifact-video">
89
- <source src="${$(e)}" type="video/webm" />
154
+ <source src="${P(e)}" type="video/webm" />
90
155
  </video>
91
156
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
92
157
  </div>
93
158
  </div>
94
- </details>`);let n=r.filter(s=>s.screenshot).map(s=>({src:s.screenshot,stepId:s.stepId,description:s.description,status:s.status,message:s.message||s.error||""}));if(n.length>0){let s=$(JSON.stringify(n)),a=n.map((c,l)=>`
159
+ </details>`);let n=r.filter(s=>s.screenshot).map(s=>({src:s.screenshot,stepId:s.stepId,description:s.description,status:s.status,message:s.message||s.error||""}));if(n.length>0){let s=P(JSON.stringify(n)),a=n.map((c,l)=>`
95
160
  <div class="screenshot-thumb" onclick="openGalleryAt(this, ${l})" data-gallery="${s}">
96
- <img src="${$(c.src)}" alt="${$(c.stepId)}" />
97
- <span class="thumb-label">${$(c.stepId)}</span>
161
+ <img src="${P(c.src)}" alt="${P(c.stepId)}" />
162
+ <span class="thumb-label">${P(c.stepId)}</span>
98
163
  </div>`).join("");i.push(`
99
164
  <details class="artifact-section">
100
165
  <summary class="artifact-summary">Screenshots (${n.length})</summary>
101
166
  <div class="artifact-content">
102
167
  <div class="screenshot-grid">${a}</div>
103
168
  </div>
104
- </details>`)}if(t){let s=$(t);i.push(`
169
+ </details>`)}if(t){let s=P(t);i.push(`
105
170
  <details class="artifact-section">
106
171
  <summary class="artifact-summary">Trace</summary>
107
172
  <div class="artifact-content">
@@ -112,40 +177,40 @@ statements:
112
177
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
113
178
  <p class="trace-hint"><a href="${s}" class="attachment-link" download>Download trace.zip</a></p>
114
179
  </div>
115
- </details>`)}return i.length===0?"":`<div class="test-artifacts">${i.join("")}</div>`}function Dr(e,t,r,o,i){let n=e.map(Di).join(`
116
- `),s="";t&&!e.some(c=>c.error)&&(s=`<div class="test-error"><pre>${$(t)}</pre></div>`);let a=Ni(r,o,e,i);return`
180
+ </details>`)}return i.length===0?"":`<div class="test-artifacts">${i.join("")}</div>`}function Zr(e,t,r,o,i){let n=e.map(dn).join(`
181
+ `),s="";t&&!e.some(c=>c.error)&&(s=`<div class="test-error"><pre>${P(t)}</pre></div>`);let a=hn(r,o,e,i);return`
117
182
  ${s}
118
183
  <div class="steps-list">
119
184
  ${n||'<div class="no-steps">No YAML step details available</div>'}
120
185
  </div>
121
- ${a}`}function Fi(e,t){let r=e.flaky?"flaky":e.status,o=Me(r),i;if(e.flaky&&e.attempts&&e.attempts.length>1){let n=`tabs-${t}`,s=e.attempts.length,a=e.attempts.map((l,p)=>{let h=p===s-1,g=l.status==="passed"?"passed":"failed",f=`Attempt ${l.attemptNumber}`;return`<button class="attempt-tab ${h?"active":""} attempt-tab-${g}"
186
+ ${a}`}function fn(e,t){let r=e.flaky?"flaky":e.status,o=De(r),i;if(e.flaky&&e.attempts&&e.attempts.length>1){let n=`tabs-${t}`,s=e.attempts.length,a=e.attempts.map((l,p)=>{let u=p===s-1,g=l.status==="passed"?"passed":"failed",f=`Attempt ${l.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${g}"
122
187
  onclick="switchAttemptTab('${n}', ${p})"
123
- data-tab-index="${p}">${Me(g)} ${f} <span class="attempt-tab-badge badge-${g}">${l.status}</span></button>`}).join(""),c=e.attempts.map((l,p)=>{let h=p===s-1,g=Dr(l.steps,l.error,l.videoPath,l.tracePath,`${t}-attempt-${p}`);return`<div class="attempt-panel ${h?"active":""}" data-panel-index="${p}">
188
+ data-tab-index="${p}">${De(g)} ${f} <span class="attempt-tab-badge badge-${g}">${l.status}</span></button>`}).join(""),c=e.attempts.map((l,p)=>{let u=p===s-1,g=Zr(l.steps,l.error,l.videoPath,l.tracePath,`${t}-attempt-${p}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${p}">
124
189
  <div class="attempt-meta">
125
- ${Me(l.status==="passed"?"passed":"failed")}
126
- <span class="attempt-meta-text">Attempt ${l.attemptNumber} &mdash; ${l.status} in ${Ie(l.duration)}</span>
190
+ ${De(l.status==="passed"?"passed":"failed")}
191
+ <span class="attempt-meta-text">Attempt ${l.attemptNumber} &mdash; ${l.status} in ${Fe(l.duration)}</span>
127
192
  </div>
128
193
  ${g}
129
194
  </div>`}).join("");i=`
130
- <div class="flaky-note">Flaky &mdash; ${Rr(e.retries)}</div>
195
+ <div class="flaky-note">Flaky &mdash; ${qr(e.retries)}</div>
131
196
  <div class="attempt-tabs" id="${n}">
132
197
  <div class="attempt-tab-bar">${a}</div>
133
198
  ${c}
134
199
  </div>`}else i=`
135
- ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${Rr(e.retries)}</div>`:""}
136
- ${Dr(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
200
+ ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${qr(e.retries)}</div>`:""}
201
+ ${Zr(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
137
202
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
138
203
  <summary class="test-summary test-${r}">
139
204
  ${o}
140
- <span class="test-title">${$(e.title)}</span>
141
- <span class="test-file">${$(e.file)}</span>
142
- ${Ri(r)}
143
- <span class="test-duration">${Ie(e.duration)}</span>
205
+ <span class="test-title">${P(e.title)}</span>
206
+ <span class="test-file">${P(e.file)}</span>
207
+ ${pn(r)}
208
+ <span class="test-duration">${Fe(e.duration)}</span>
144
209
  </summary>
145
210
  <div class="test-body">
146
211
  ${i}
147
212
  </div>
148
- </details>`}function st(e){let t=e.tests.filter(a=>a.flaky).length,r=e.tests.filter(a=>a.status==="passed"&&!a.flaky).length,o=e.tests.filter(a=>a.status==="failed"||a.status==="timedOut").length,i=e.tests.filter(a=>a.status==="skipped").length,n=e.tests.length,s=e.tests.map((a,c)=>Fi(a,c)).join(`
213
+ </details>`}function mt(e){let t=e.tests.filter(a=>a.flaky).length,r=e.tests.filter(a=>a.status==="passed"&&!a.flaky).length,o=e.tests.filter(a=>a.status==="failed"||a.status==="timedOut").length,i=e.tests.filter(a=>a.status==="skipped").length,n=e.tests.length,s=e.tests.map((a,c)=>fn(a,c)).join(`
149
214
  `);return`<!DOCTYPE html>
150
215
  <html lang="en">
151
216
  <head>
@@ -739,6 +804,38 @@ statements:
739
804
  .gallery-status.failure { background: var(--color-failed); }
740
805
  .gallery-status.skipped { background: var(--color-skipped); }
741
806
  .gallery-status.pending { background: var(--color-pending); }
807
+
808
+ .step-code {
809
+ margin: 6px 0 4px;
810
+ border-radius: 4px;
811
+ overflow: hidden;
812
+ border: 1px solid var(--color-border);
813
+ }
814
+ .code-block {
815
+ background: #0d1117;
816
+ padding: 4px 0;
817
+ font-family: 'SFMono-Regular', Consolas, 'Liberation Mono', Menlo, monospace;
818
+ font-size: 12px;
819
+ line-height: 1.6;
820
+ overflow-x: auto;
821
+ display: block;
822
+ margin: 0;
823
+ }
824
+ .code-line {
825
+ display: block;
826
+ padding: 0 12px;
827
+ white-space: pre;
828
+ color: var(--color-text-secondary);
829
+ }
830
+ .code-line-active {
831
+ background: rgba(100, 181, 246, 0.1);
832
+ color: var(--color-text);
833
+ border-left: 2px solid var(--color-accent);
834
+ padding-left: 10px;
835
+ }
836
+ .code-line-body {
837
+ color: var(--color-text);
838
+ }
742
839
  </style>
743
840
  </head>
744
841
  <body>
@@ -751,7 +848,7 @@ statements:
751
848
  ${t>0?`<span class="summary-stat flaky">${t} flaky</span>`:""}
752
849
  ${o>0?`<span class="summary-stat failed">${o} failed</span>`:""}
753
850
  ${i>0?`<span class="summary-stat skipped">${i} skipped</span>`:""}
754
- <span class="summary-stat">${Ie(e.totalDuration)}</span>
851
+ <span class="summary-stat">${Fe(e.totalDuration)}</span>
755
852
  ${e.timestamp?`<span class="summary-stat" style="margin-left:auto;color:var(--color-text-secondary)">${new Date(e.timestamp).toLocaleString()}</span>`:""}
756
853
  </div>
757
854
  </div>${e.cacheSummary?`
@@ -895,9 +992,9 @@ statements:
895
992
  });
896
993
  </script>
897
994
  </body>
898
- </html>`}var Nr=_(()=>{"use strict"});import*as V from"fs";import*as X from"path";import{execFileSync as Ui}from"child_process";import{createHash as Bi}from"crypto";import ue from"axios";function Fr(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function ee(...e){try{return Ui("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Gi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=V.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Wi(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",o=process.env.GITHUB_RUN_ID??"",i=process.env.GITHUB_EVENT_NAME??"",s=Gi().pull_request,a=s?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],h=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,g=process.env.SHIPLIGHT_PR_TITLE??s?.title,f=ee("log","-1","--pretty=%s"),u=ee("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Actions",gitCommit:c,gitBranch:h,gitRepo:r,commitMessage:f,authorEmail:u,prNumber:p,prTitle:g,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:o,ciBuildUrl:o&&r?`${t}/${r}/actions/runs/${o}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:i,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",o=process.env.CI_MERGE_REQUEST_IID,i=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:i,prNumber:o,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:o&&t?`${t}/-/merge_requests/${o}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",o=process.env.CIRCLE_PROJECT_REPONAME??"",i=r&&o?`${r}/${o}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?Fr(a):void 0,l=ee("log","-1","--pretty=%s"),p=ee("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:i,commitMessage:l,authorEmail:p,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=ee("rev-parse","HEAD"),r=ee("rev-parse","--abbrev-ref","HEAD"),o=ee("log","-1","--pretty=%s"),i=ee("log","-1","--pretty=%ae"),n=ee("remote","get-url","origin"),s=n?Fr(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:i,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function Hi(e){switch(e){case"passed":return"Passed";case"failed":return"Failed";case"timedOut":return"TimedOut";case"skipped":return"Skipped";case"interrupted":return"Failed";default:return"Failed"}}function Ur(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function ji(e){return e.length===0||e.every(t=>t.status==="skipped")?"Skipped":e.some(t=>t.status==="failed"||t.status==="timedOut"||t.status==="interrupted")?"Failed":"Passed"}function Ki(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let i=r.get(o.testCaseName);i?i.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Oe(e,t){return X.isAbsolute(t)?t:X.join(e,t)}function zi(e,t,r,o){let i={};for(let n of e.steps)i[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:Ur(e.status),flaky:!1,segments:[{outcome:Ur(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:i,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function lt(e){return Bi("md5").update(e).digest("base64")}function at(e){return lt(V.readFileSync(e))}async function ct(e,t){let r=V.readFileSync(t),o=X.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await ue.put(e,r,{headers:{"Content-Type":n,"Content-MD5":lt(r)}})}async function Br(e,t,r,o){let i=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},s=Wi(),a=e.tests.map(d=>{let w={testCaseName:d.title,testCaseBaseName:d.baseTitle,suiteName:d.suiteName,file:d.file,tags:d.tags,suiteTags:d.suiteTags,baseUrl:d.baseUrl,skip:d.skip,slow:d.slow,timeout:d.timeout,parameterSetName:d.parameterSetName,flaky:d.flaky,retries:d.retries};if(d.videoPath){let b=Oe(t,d.videoPath);V.existsSync(b)&&(w.videoMd5=at(b))}if(d.tracePath){let b=Oe(t,d.tracePath);V.existsSync(b)&&(w.traceMd5=at(b))}return w}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let p=(await ue.post(`${i}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:a},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let h=Ki(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(d,w)=>{let b=h[w];if(!b)return;let v=d.steps.filter(k=>k.screenshot);if(!v.length)return;let y=v.map(k=>k.stepId),A={};for(let k of v){let P=X.isAbsolute(k.screenshot)?k.screenshot:X.join(t,k.screenshot);V.existsSync(P)&&(A[k.stepId]=at(P))}try{let k=await ue.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:A},{headers:n});b.uploadUrls.screenshots=k.data.screenshots,b.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${d.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${d.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let g=(await Promise.all(e.tests.map(async(d,w)=>{let b=h[w];if(!b){console.warn(`[reporter] No result slot found for test "${d.title}", skipping.`);return}let v=b.uploadUrls,y={},A=0;await Promise.all(d.steps.map(async M=>{if(M.screenshot&&v.screenshots?.[M.stepId]){let se=X.isAbsolute(M.screenshot)?M.screenshot:X.join(t,M.screenshot);if(V.existsSync(se))try{await ct(v.screenshots[M.stepId],se),y[M.stepId]=b.screenshotS3Uris[M.stepId],A++}catch(so){console.warn(`[reporter] Screenshot upload failed for step ${M.stepId}:`,so)}}})),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} screenshot(s) for "${d.title}"`);let k;if(d.videoPath&&v.video){let M=Oe(t,d.videoPath);if(V.existsSync(M)){console.log(`[reporter] [3/4] Uploading video for "${d.title}"...`);try{await ct(v.video,M),k=b.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${d.title}"`)}catch(se){console.warn("[reporter] Video upload failed:",se)}}}let P;if(d.tracePath&&v.trace){let M=Oe(t,d.tracePath);if(V.existsSync(M)){console.log(`[reporter] [3/4] Uploading trace for "${d.title}"...`);try{await ct(v.trace,M),P=b.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${d.title}"`)}catch(se){console.warn("[reporter] Trace upload failed:",se)}}}console.log(`[reporter] [3/4] Uploading report for "${d.title}"...`);let De=zi(d,y,k,P),ut=Buffer.from(JSON.stringify(De)),dt=lt(ut),ht=await ue.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/report-url`,{md5:dt},{headers:n}),io=ht.data.reportUrl,no=ht.data.reportS3Uri;return await ue.put(io,ut,{headers:{"Content-Type":"application/json","Content-MD5":dt}}),console.log(`[reporter] [3/4] Report uploaded for "${d.title}"`),{testCaseResultId:b.testCaseResultId,result:Hi(d.status),durationMs:d.duration,startTime:d.startTime,endTime:d.endTime,error:d.error,reportS3Uri:no,videoS3Uri:k,traceS3Uri:P,metadata:{suiteName:d.suiteName,file:d.file}}}))).filter(d=>!!d);console.log("[reporter] [4/4] Finalising run...");let f=ji(e.tests),u=await ue.put(`${i}/v1/local-runs/${p.testRunId}/complete`,{status:f,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:g},{headers:n});console.log(`
899
- Shiplight cloud report: ${u.data.reportUrl}`)}var Gr=_(()=>{"use strict"});var zr={};te(zr,{buildGitHubSummary:()=>Kr,runReport:()=>Vi});import*as T from"fs";import*as x from"path";async function Vi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),o=e.includes("--github-summary");!r&&o&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Ji(e,t,o):await Yi(e,t)}async function Yi(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",o=x.isAbsolute(r)?r:x.join(process.cwd(),r),i=x.join(o,"report-data.json");T.existsSync(i)||(console.error(`Error: ${i} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let n;try{n=JSON.parse(T.readFileSync(i,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${i}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let s=x.join(o,"index.html");if(T.writeFileSync(s,st(n),"utf-8"),console.log(`Shiplight report regenerated: ${s}`),await jr(n,o),t)try{let a=(await import("open")).default;await a(s)}catch{}}async function Ji(e,t,r){let o=x.join(process.cwd(),"shiplight-report"),i=e.findIndex(f=>f==="-o"||f==="--output");if(i!==-1&&e[i+1]){let f=e[i+1];o=x.isAbsolute(f)?f:x.join(process.cwd(),f)}let n=new Set(["-o","--output"]),s=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let f=0;f<e.length;f++){let u=e[f];if(n.has(u)){f++;continue}if(s.has(u))continue;let d=x.isAbsolute(u)?u:x.join(process.cwd(),u);a.push(d)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;T.mkdirSync(x.join(o,"screenshots"),{recursive:!0});for(let f=0;f<a.length;f++){let u=a[f],d=`shard-${f}`,w=x.join(u,"report-data.json");if(!T.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let b;try{b=JSON.parse(T.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${d}: ${b.tests.length} tests from ${u}`),l+=b.totalDuration||0;let v=x.join(u,"screenshots");T.existsSync(v)&&Hr(v,x.join(o,"screenshots",d));let y=x.join(o,d);for(let A of b.tests){let k=[A,...A.attempts||[]];for(let P of k)Xi(P.steps,d),P.videoPath&&Wr(u,P.videoPath,y)&&(P.videoPath=`${d}/${P.videoPath}`),P.tracePath&&Wr(u,P.tracePath,y)&&(P.tracePath=`${d}/${P.tracePath}`);c.push(A)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let h={tests:c,totalDuration:l,timestamp:new Date().toISOString()};T.writeFileSync(x.join(o,"report-data.json"),JSON.stringify(h,null,2),"utf-8");let g=x.join(o,"index.html");if(T.writeFileSync(g,st(h),"utf-8"),console.log(`
900
- Merged ${c.length} tests from ${p} shards into: ${g}`),await jr(h,o),r&&qi(c),t)try{let f=(await import("open")).default;await f(g)}catch{}}function Wr(e,t,r){let o=x.resolve(e,t);return o.startsWith(x.resolve(e)+x.sep)?T.existsSync(o)?(T.mkdirSync(r,{recursive:!0}),T.copyFileSync(o,x.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Hr(e,t){T.mkdirSync(t,{recursive:!0});for(let r of T.readdirSync(e,{withFileTypes:!0})){let o=x.join(e,r.name),i=x.join(t,r.name);r.isDirectory()?Hr(o,i):T.copyFileSync(o,i)}}function Xi(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function jr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN||process.env.__SHIPLIGHT_API_KEY;if(!r){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let o=e.tests.map(n=>n.startTime).filter(n=>!!n),i=o.length>0?o.sort()[0]:e.timestamp??new Date().toISOString();try{await Br(e,t,i,r)}catch(n){console.warn("[report] Cloud upload failed:",n)}}function pt(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=x.join("tests",x.basename(t));return{name:e.title||x.basename(t),yamlPath:r}}function Kr(e){let t=e.filter(a=>!a.file.includes("auth.setup")),r=t.filter(a=>a.flaky),o=t.filter(a=>a.status==="passed"&&!a.flaky),i=t.filter(a=>a.status!=="passed"),n=t.length,s=`## Test Results
995
+ </html>`}var Qr=_(()=>{"use strict"});import*as q from"fs";import*as ee from"path";import{execFileSync as gn}from"child_process";import{createHash as mn}from"crypto";import ge from"axios";function eo(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function ne(...e){try{return gn("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function yn(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=q.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function wn(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",o=process.env.GITHUB_RUN_ID??"",i=process.env.GITHUB_EVENT_NAME??"",s=yn().pull_request,a=s?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],u=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,g=process.env.SHIPLIGHT_PR_TITLE??s?.title,f=ne("log","-1","--pretty=%s"),d=ne("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:u,gitRepo:r,commitMessage:f,authorEmail:d,prNumber:p,prTitle:g,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:o,ciBuildUrl:o&&r?`${t}/${r}/actions/runs/${o}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:i,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",o=process.env.CI_MERGE_REQUEST_IID,i=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:i,prNumber:o,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:o&&t?`${t}/-/merge_requests/${o}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",o=process.env.CIRCLE_PROJECT_REPONAME??"",i=r&&o?`${r}/${o}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?eo(a):void 0,l=ne("log","-1","--pretty=%s"),p=ne("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:i,commitMessage:l,authorEmail:p,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=ne("rev-parse","HEAD"),r=ne("rev-parse","--abbrev-ref","HEAD"),o=ne("log","-1","--pretty=%s"),i=ne("log","-1","--pretty=%ae"),n=ne("remote","get-url","origin"),s=n?eo(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:i,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function bn(e){switch(e){case"passed":return"Passed";case"failed":return"Failed";case"timedOut":return"TimedOut";case"skipped":return"Skipped";case"interrupted":return"Failed";default:return"Failed"}}function to(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Sn(e){return e.length===0||e.every(t=>t.status==="skipped")?"Skipped":e.some(t=>t.status==="failed"||t.status==="timedOut"||t.status==="interrupted")?"Failed":"Passed"}function vn(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let i=r.get(o.testCaseName);i?i.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Ue(e,t){return ee.isAbsolute(t)?t:ee.join(e,t)}function _n(e,t,r,o){let i={};for(let n of e.steps)i[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:to(e.status),flaky:!1,segments:[{outcome:to(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:i,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function bt(e){return mn("md5").update(e).digest("base64")}function yt(e){return bt(q.readFileSync(e))}async function wt(e,t){let r=q.readFileSync(t),o=ee.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await ge.put(e,r,{headers:{"Content-Type":n,"Content-MD5":bt(r)}})}async function ro(e,t,r,o){let i=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},s=wn(),a=e.tests.map(h=>{let b={testCaseName:h.title,testCaseBaseName:h.baseTitle,suiteName:h.suiteName,file:h.file,tags:h.tags,suiteTags:h.suiteTags,baseUrl:h.baseUrl,skip:h.skip,slow:h.slow,timeout:h.timeout,parameterSetName:h.parameterSetName,flaky:h.flaky,retries:h.retries};if(h.videoPath){let w=Ue(t,h.videoPath);q.existsSync(w)&&(b.videoMd5=yt(w))}if(h.tracePath){let w=Ue(t,h.tracePath);q.existsSync(w)&&(b.traceMd5=yt(w))}return b}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let p=(await ge.post(`${i}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:a},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let u=vn(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(h,b)=>{let w=u[b];if(!w)return;let v=h.steps.filter(k=>k.screenshot);if(!v.length)return;let y=v.map(k=>k.stepId),A={};for(let k of v){let $=ee.isAbsolute(k.screenshot)?k.screenshot:ee.join(t,k.screenshot);q.existsSync($)&&(A[k.stepId]=yt($))}try{let k=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:A},{headers:n});w.uploadUrls.screenshots=k.data.screenshots,w.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${h.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${h.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let g=(await Promise.all(e.tests.map(async(h,b)=>{let w=u[b];if(!w){console.warn(`[reporter] No result slot found for test "${h.title}", skipping.`);return}let v=w.uploadUrls,y={},A=0;await Promise.all(h.steps.map(async I=>{if(I.screenshot&&v.screenshots?.[I.stepId]){let pe=ee.isAbsolute(I.screenshot)?I.screenshot:ee.join(t,I.screenshot);if(q.existsSync(pe))try{await wt(v.screenshots[I.stepId],pe),y[I.stepId]=w.screenshotS3Uris[I.stepId],A++}catch(xo){console.warn(`[reporter] Screenshot upload failed for step ${I.stepId}:`,xo)}}})),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} screenshot(s) for "${h.title}"`);let k;if(h.videoPath&&v.video){let I=Ue(t,h.videoPath);if(q.existsSync(I)){console.log(`[reporter] [3/4] Uploading video for "${h.title}"...`);try{await wt(v.video,I),k=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Video upload failed:",pe)}}}let $;if(h.tracePath&&v.trace){let I=Ue(t,h.tracePath);if(q.existsSync(I)){console.log(`[reporter] [3/4] Uploading trace for "${h.title}"...`);try{await wt(v.trace,I),$=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Trace upload failed:",pe)}}}console.log(`[reporter] [3/4] Uploading report for "${h.title}"...`);let He=_n(h,y,k,$),vt=Buffer.from(JSON.stringify(He)),_t=bt(vt),xt=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:_t},{headers:n}),vo=xt.data.reportUrl,_o=xt.data.reportS3Uri;return await ge.put(vo,vt,{headers:{"Content-Type":"application/json","Content-MD5":_t}}),console.log(`[reporter] [3/4] Report uploaded for "${h.title}"`),{testCaseResultId:w.testCaseResultId,result:bn(h.status),durationMs:h.duration,startTime:h.startTime,endTime:h.endTime,error:h.error,reportS3Uri:_o,videoS3Uri:k,traceS3Uri:$,metadata:{suiteName:h.suiteName,file:h.file}}}))).filter(h=>!!h);console.log("[reporter] [4/4] Finalising run...");let f=Sn(e.tests),d=await ge.put(`${i}/v1/local-runs/${p.testRunId}/complete`,{status:f,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:g},{headers:n});console.log(`
996
+ Shiplight cloud report: ${d.data.reportUrl}`)}var oo=_(()=>{"use strict"});var co={};te(co,{buildGitHubSummary:()=>ao,runReport:()=>xn});import*as T from"fs";import*as x from"path";async function xn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),o=e.includes("--github-summary");!r&&o&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await kn(e,t,o):await Tn(e,t)}async function Tn(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",o=x.isAbsolute(r)?r:x.join(process.cwd(),r),i=x.join(o,"report-data.json");T.existsSync(i)||(console.error(`Error: ${i} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let n;try{n=JSON.parse(T.readFileSync(i,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${i}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let s=x.join(o,"index.html");if(T.writeFileSync(s,mt(n),"utf-8"),console.log(`Shiplight report regenerated: ${s}`),await so(n,o),t)try{let a=(await import("open")).default;await a(s)}catch{}}async function kn(e,t,r){let o=x.join(process.cwd(),"shiplight-report"),i=e.findIndex(f=>f==="-o"||f==="--output");if(i!==-1&&e[i+1]){let f=e[i+1];o=x.isAbsolute(f)?f:x.join(process.cwd(),f)}let n=new Set(["-o","--output"]),s=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let f=0;f<e.length;f++){let d=e[f];if(n.has(d)){f++;continue}if(s.has(d))continue;let h=x.isAbsolute(d)?d:x.join(process.cwd(),d);a.push(h)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;T.mkdirSync(x.join(o,"screenshots"),{recursive:!0});for(let f=0;f<a.length;f++){let d=a[f],h=`shard-${f}`,b=x.join(d,"report-data.json");if(!T.existsSync(b)){console.warn(`Warning: No report-data.json found in ${d}, skipping.`);continue}let w;try{w=JSON.parse(T.readFileSync(b,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${b}, skipping.`);continue}console.log(`Merging ${h}: ${w.tests.length} tests from ${d}`),l+=w.totalDuration||0;let v=x.join(d,"screenshots");T.existsSync(v)&&no(v,x.join(o,"screenshots",h));let y=x.join(o,h);for(let A of w.tests){let k=[A,...A.attempts||[]];for(let $ of k)An($.steps,h),$.videoPath&&io(d,$.videoPath,y)&&($.videoPath=`${h}/${$.videoPath}`),$.tracePath&&io(d,$.tracePath,y)&&($.tracePath=`${h}/${$.tracePath}`);c.push(A)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let u={tests:c,totalDuration:l,timestamp:new Date().toISOString()};T.writeFileSync(x.join(o,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let g=x.join(o,"index.html");if(T.writeFileSync(g,mt(u),"utf-8"),console.log(`
997
+ Merged ${c.length} tests from ${p} shards into: ${g}`),await so(u,o),r&&En(c),t)try{let f=(await import("open")).default;await f(g)}catch{}}function io(e,t,r){let o=x.resolve(e,t);return o.startsWith(x.resolve(e)+x.sep)?T.existsSync(o)?(T.mkdirSync(r,{recursive:!0}),T.copyFileSync(o,x.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function no(e,t){T.mkdirSync(t,{recursive:!0});for(let r of T.readdirSync(e,{withFileTypes:!0})){let o=x.join(e,r.name),i=x.join(t,r.name);r.isDirectory()?no(o,i):T.copyFileSync(o,i)}}function An(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function so(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN||process.env.__SHIPLIGHT_API_KEY;if(!r){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let o=e.tests.map(n=>n.startTime).filter(n=>!!n),i=o.length>0?o.sort()[0]:e.timestamp??new Date().toISOString();try{await ro(e,t,i,r)}catch(n){console.warn("[report] Cloud upload failed:",n)}}function St(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=x.join("tests",x.basename(t));return{name:e.title||x.basename(t),yamlPath:r}}function ao(e){let t=e.filter(a=>!a.file.includes("auth.setup")),r=t.filter(a=>a.flaky),o=t.filter(a=>a.status==="passed"&&!a.flaky),i=t.filter(a=>a.status!=="passed"),n=t.length,s=`## Test Results
901
998
 
902
999
  `;if(i.length===0&&r.length===0?s+=`\u2705 All ${n} tests passed
903
1000
 
@@ -905,7 +1002,7 @@ Merged ${c.length} tests from ${p} shards into: ${g}`),await jr(h,o),r&&qi(c),t)
905
1002
 
906
1003
  `:s+=`\u274C ${i.length} failed, \u26A0\uFE0F ${r.length} flaky, \u2705 ${o.length} passed / ${n} total
907
1004
 
908
- `,i.length>0){let a=[...new Set(i.map(c=>pt(c).yamlPath))];s+=`### Failed
1005
+ `,i.length>0){let a=[...new Set(i.map(c=>St(c).yamlPath))];s+=`### Failed
909
1006
 
910
1007
  `;for(let c of a)s+=`- \`npx shiplight test ${c}\`
911
1008
  `;s+=`
@@ -913,21 +1010,38 @@ Merged ${c.length} tests from ${p} shards into: ${g}`),await jr(h,o),r&&qi(c),t)
913
1010
 
914
1011
  `,s+="```sh\n",s+=`npx shiplight test ${a.map(c=>`"${c}"`).join(` \\
915
1012
  `)}
916
- `,s+="```\n\n"}if(r.length>0){let a=[...new Set(r.map(c=>pt(c).yamlPath))];s+=`### Flaky (${a.length})
1013
+ `,s+="```\n\n"}if(r.length>0){let a=[...new Set(r.map(c=>St(c).yamlPath))];s+=`### Flaky (${a.length})
917
1014
 
918
1015
  `;for(let c of a)s+=`- \`${c}\`
919
1016
  `;s+=`
920
- `}if(o.length>0){let a=[...new Set(o.map(c=>pt(c).yamlPath))];s+=`<details><summary>Passed (${a.length})</summary>
1017
+ `}if(o.length>0){let a=[...new Set(o.map(c=>St(c).yamlPath))];s+=`<details><summary>Passed (${a.length})</summary>
921
1018
 
922
1019
  `;for(let c of a)s+=`- ${c}
923
1020
  `;s+=`
924
1021
  </details>
925
- `}return s}function qi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}T.appendFileSync(t,Kr(e)),console.log("GitHub step summary written.")}var Vr=_(()=>{"use strict";Nr();Gr()});var Yr,Jr=_(()=>{"use strict";Yr="0.1.47"});var Xr={};te(Xr,{runTranspile:()=>Qi});import*as Le from"path";import{glob as Zi}from"glob";async function Qi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),o=await Zi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let i=0,n=0,s=0;for(let a of o.sort()){let c=Le.resolve(r,a),l=mr(c,{version:Yr});if(!l.valid){i++,console.log(`
926
- \u2717 ${a}`);for(let h of l.errors)console.log(` ERROR: ${h}`);continue}s++;let p=Le.basename(l.specFile);if(l.warnings.length>0){n++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let h of l.warnings)console.log(` WARNING: ${h}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
927
- ${o.length} file(s): ${s} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var qr=_(()=>{"use strict";Ze();Jr()});var eo={};te(eo,{runInspect:()=>en});import*as Ce from"fs";import*as Zr from"path";async function en(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),o=e.find(s=>!s.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let i=Zr.resolve(process.cwd(),o);Ce.existsSync(i)||(console.error(`Error: file not found: ${i}`),process.exit(1));let n=Ce.readFileSync(i,"utf-8");try{let s=fe(n),a=O(n);if(r)tn(a,s);else{let c={...s.test_case_id!==void 0?{test_case_id:s.test_case_id}:{},...s.name?{name:s.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(s){console.error(`Error parsing ${o}: ${s.message}`),process.exit(1)}}function tn(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let o of r.tests){let i=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${i}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Qr(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Qr(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let o=Qr(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var to=_(()=>{"use strict";re()});var ro=co((Lc,rn)=>{rn.exports={name:"shiplightai",version:"0.1.47",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"}},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{build:"tsup && cd ../frontend && npx vite build --config vite.debugger.config.ts",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.58.2","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"1.58.2"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});import on from"dotenv";on.config();function oo(){console.log(`
1022
+ `}return s}function En(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}T.appendFileSync(t,ao(e)),console.log("GitHub step summary written.")}var lo=_(()=>{"use strict";Qr();oo()});var po,uo=_(()=>{"use strict";po="0.1.49"});var ho={};te(ho,{runTranspile:()=>$n});import*as Be from"path";import{glob as Pn}from"glob";async function $n(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),o=await Pn(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let i=0,n=0,s=0;for(let a of o.sort()){let c=Be.resolve(r,a),l=Or(c,{version:po});if(!l.valid){i++,console.log(`
1023
+ \u2717 ${a}`);for(let u of l.errors)console.log(` ERROR: ${u}`);continue}s++;let p=Be.basename(l.specFile);if(l.warnings.length>0){n++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let u of l.warnings)console.log(` WARNING: ${u}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
1024
+ ${o.length} file(s): ${s} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var fo=_(()=>{"use strict";ct();uo()});var yo={};te(yo,{runInspect:()=>Mn});import*as Ge from"fs";import*as go from"path";async function Mn(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),o=e.find(s=>!s.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let i=go.resolve(process.cwd(),o);Ge.existsSync(i)||(console.error(`Error: file not found: ${i}`),process.exit(1));let n=Ge.readFileSync(i,"utf-8");try{let s=we(n),a=L(n);if(r)In(a,s);else{let c={...s.test_case_id!==void 0?{test_case_id:s.test_case_id}:{},...s.name?{name:s.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(s){console.error(`Error parsing ${o}: ${s.message}`),process.exit(1)}}function In(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let o of r.tests){let i=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${i}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=mo(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function mo(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let o=mo(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var wo=_(()=>{"use strict";se()});var bo=ko((hl,On)=>{On.exports={name:"shiplightai",version:"0.1.49",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"}},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.58.2","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"1.58.2"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});import Ln from"dotenv";import*as M from"fs";import*as W from"path";import*as We from"os";import{execFileSync as Ao}from"child_process";var Eo="0.1.49",Po="https://registry.npmjs.org/shiplightai/latest",$o=3600*1e3,Mo=10080*60*1e3;function Ke(){return W.join(We.homedir(),".shiplight","version-check.json")}function Io(){return W.join(We.homedir(),".shiplight","npm-prefix.json")}function Oo(e=Io()){try{let r=M.readFileSync(e,"utf-8"),o=JSON.parse(r);if(typeof o.prefix=="string"&&typeof o.fetchedAt=="number"&&Date.now()-o.fetchedAt<Mo)return o.prefix}catch{}let t;try{t=Ao("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{M.mkdirSync(W.dirname(e),{recursive:!0}),M.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Tt(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Oo()),o=e.warn??(s=>console.warn(s)),i=e.error??(s=>console.error(s)),n=e.exit??(s=>process.exit(s));try{if(!t)return;let s=r();if(!s)return;let a,c;try{a=M.realpathSync(t),c=M.realpathSync(s)}catch{return}if(a.startsWith(c+W.sep)){i(`
1025
+ shiplightai cannot be run from a global install.
1026
+ Global installs don't auto-update and cause version skew.
1027
+
1028
+ Install it as a project dependency instead:
1029
+
1030
+ cd <your-project>
1031
+ npm i -D shiplightai
1032
+ npx shiplight <command>
1033
+ `),n(1);return}let l=W.join(c,"lib","node_modules","shiplightai"),p=W.join(c,"node_modules","shiplightai"),u=M.existsSync(l)?l:M.existsSync(p)?p:null;u&&o(`
1034
+ \x1B[33m\u26A0 A global shiplightai install was detected at ${u}.
1035
+ Global installs don't auto-update and can shadow the project-local CLI on PATH.
1036
+ Please remove it: npm uninstall -g shiplightai\x1B[0m
1037
+ `)}catch{}}function Lo(e=Ke()){try{let t=M.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<$o)return r}catch{}return null}function Co(e,t=Ke()){try{M.mkdirSync(W.dirname(t),{recursive:!0}),M.writeFileSync(t,JSON.stringify(e))}catch{}}async function Ro(){try{let e=await fetch(Po,{headers:{Accept:"application/json"},signal:AbortSignal.timeout(3e3)});if(!e.ok)return null;let t=await e.json();return typeof t.version=="string"?t.version:null}catch{return null}}function No(e,t){let r=u=>{let g=u.indexOf("-");return g===-1?[u,!1]:[u.slice(0,g),!0]},o=u=>u.split(".").map(g=>parseInt(g,10)||0),[i,n]=r(e),[s,a]=r(t),c=o(i),l=o(s),p=Math.max(c.length,l.length);for(let u=0;u<p;u++){let g=c[u]??0,f=l[u]??0;if(g<f)return!0;if(g>f)return!1}return!!(n&&!a)}async function kt(e={}){let t=e.runningVersion??Eo,r=e.cwd??process.cwd(),o=e.cacheFile??Ke(),i=e.fetchLatest??Ro,n=e.env??process.env,s=e.warn??(l=>console.warn(l));if(n.CI||t==="dev"||!M.existsSync(W.join(r,"package-lock.json")))return;let a=null,c=Lo(o);if(c)a=c.latest;else{try{a=await i()}catch{a=null}a&&Co({latest:a,fetchedAt:Date.now()},o)}a&&No(t,a)&&s(`
1038
+ \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
1039
+ Run: npm update shiplightai\x1B[0m
1040
+ `)}Ln.config();Tt();kt();function So(){console.log(`
928
1041
  Usage: shiplight <command> [options]
929
1042
 
930
1043
  Commands:
1044
+ create <path> Scaffold a new Shiplight test project at <path>
931
1045
  test [args...] Run Playwright tests (delegates to npx playwright test)
932
1046
  transpile [glob] Transpile YAML test files to Playwright specs (default: **/*.test.yaml)
933
1047
  inspect <file> Parse a YAML test file and output TestFlow JSON
@@ -940,9 +1054,10 @@ Options:
940
1054
  --version, -v Show version number
941
1055
 
942
1056
  Examples:
1057
+ shiplight create ./my-tests
943
1058
  shiplight test --headed
944
1059
  shiplight transpile
945
1060
  shiplight transpile "tests/**/*.test.yaml"
946
1061
  shiplight debug tests/login.test.yaml
947
- `)}var Re=process.argv[2];switch(Re){case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Ar(),kr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Cr(),Lr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(Vr(),zr));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(qr(),Xr));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(to(),eo));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=ro().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":oo();break;default:Re&&console.error(`Unknown command: ${Re}
948
- `),oo(),process.exit(Re?1:0)}
1062
+ `)}var je=process.argv[2];switch(je){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>($t(),Pt));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(jr(),Gr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Xr(),Jr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(lo(),co));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(fo(),ho));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(wo(),yo));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=bo().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":So();break;default:je&&console.error(`Unknown command: ${je}
1063
+ `),So(),process.exit(je?1:0)}