shiplightai 0.1.72 → 0.1.73

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,7 @@
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 mn=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var yn=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ie=(e,t)=>{for(var r in t)mn(e,r,{get:t[r],enumerable:!0})};import*as C from"fs";import*as J from"path";import*as pt from"os";import{execFileSync as wn}from"child_process";function ut(){return J.join(pt.homedir(),".shiplight","version-check.json")}function _n(){return J.join(pt.homedir(),".shiplight","npm-prefix.json")}function xn(e=_n()){try{let r=C.readFileSync(e,"utf-8"),s=JSON.parse(r);if(typeof s.prefix=="string"&&typeof s.fetchedAt=="number"&&Date.now()-s.fetchedAt<vn)return s.prefix}catch{}let t;try{t=wn("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{C.mkdirSync(J.dirname(e),{recursive:!0}),C.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function qt(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>xn()),s=e.warn??(a=>console.warn(a)),n=e.error??(a=>console.error(a)),o=e.exit??(a=>process.exit(a));try{if(!t)return;let a=r();if(!a)return;let i,l;try{i=C.realpathSync(t),l=C.realpathSync(a)}catch{return}if(i.startsWith(l+J.sep)){n(`
4
+ var wn=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var bn=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ie=(e,t)=>{for(var r in t)wn(e,r,{get:t[r],enumerable:!0})};import*as N from"fs";import*as J from"path";import*as lt from"os";import{execFileSync as vn}from"child_process";function pt(){return J.join(lt.homedir(),".shiplight","version-check.json")}function Tn(){return J.join(lt.homedir(),".shiplight","npm-prefix.json")}function kn(e=Tn()){try{let r=N.readFileSync(e,"utf-8"),s=JSON.parse(r);if(typeof s.prefix=="string"&&typeof s.fetchedAt=="number"&&Date.now()-s.fetchedAt<xn)return s.prefix}catch{}let t;try{t=vn("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{N.mkdirSync(J.dirname(e),{recursive:!0}),N.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Qt(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>kn()),s=e.warn??(a=>console.warn(a)),n=e.error??(a=>console.error(a)),o=e.exit??(a=>process.exit(a));try{if(!t)return;let a=r();if(!a)return;let i,c;try{i=N.realpathSync(t),c=N.realpathSync(a)}catch{return}if(i.startsWith(c+J.sep)){n(`
5
5
  shiplightai cannot be run from a global install.
6
6
  Global installs don't auto-update and cause version skew.
7
7
 
@@ -10,14 +10,14 @@ Install it as a project dependency instead:
10
10
  cd <your-project>
11
11
  npm i -D shiplightai
12
12
  npx shiplight <command>
13
- `),o(1);return}let c=J.join(l,"lib","node_modules","shiplightai"),u=J.join(l,"node_modules","shiplightai"),d=C.existsSync(c)?c:C.existsSync(u)?u:null;d&&s(`
13
+ `),o(1);return}let l=J.join(c,"lib","node_modules","shiplightai"),u=J.join(c,"node_modules","shiplightai"),d=N.existsSync(l)?l:N.existsSync(u)?u:null;d&&s(`
14
14
  \x1B[33m\u26A0 A global shiplightai install was detected at ${d}.
15
15
  Global installs don't auto-update and can shadow the project-local CLI on PATH.
16
16
  Please remove it: npm uninstall -g shiplightai\x1B[0m
17
- `)}catch{}}function Tn(e=ut()){try{let t=C.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<Sn)return r}catch{}return null}function Pn(e,t=ut()){try{C.mkdirSync(J.dirname(t),{recursive:!0}),C.writeFileSync(t,JSON.stringify(e))}catch{}}async function kn(){try{let e=await fetch(bn,{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 En(e,t){let r=d=>{let f=d.indexOf("-");return f===-1?[d,!1]:[d.slice(0,f),!0]},s=d=>d.split(".").map(f=>parseInt(f,10)||0),[n,o]=r(e),[a,i]=r(t),l=s(n),c=s(a),u=Math.max(l.length,c.length);for(let d=0;d<u;d++){let f=l[d]??0,g=c[d]??0;if(f<g)return!0;if(f>g)return!1}return!!(o&&!i)}async function Zt(e={}){let t=e.runningVersion??lt,r=e.cwd??process.cwd(),s=e.cacheFile??ut(),n=e.fetchLatest??kn,o=e.env??process.env,a=e.warn??(c=>console.warn(c));if(o.CI||t==="dev"||!C.existsSync(J.join(r,"package-lock.json")))return;let i=null,l=Tn(s);if(l)i=l.latest;else{try{i=await n()}catch{i=null}i&&Pn({latest:i,fetchedAt:Date.now()},s)}i&&En(t,i)&&a(`
17
+ `)}catch{}}function Pn(e=pt()){try{let t=N.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<_n)return r}catch{}return null}function En(e,t=pt()){try{N.mkdirSync(J.dirname(t),{recursive:!0}),N.writeFileSync(t,JSON.stringify(e))}catch{}}async function An(){try{let e=await fetch(Sn,{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 $n(e,t){let r=d=>{let f=d.indexOf("-");return f===-1?[d,!1]:[d.slice(0,f),!0]},s=d=>d.split(".").map(f=>parseInt(f,10)||0),[n,o]=r(e),[a,i]=r(t),c=s(n),l=s(a),u=Math.max(c.length,l.length);for(let d=0;d<u;d++){let f=c[d]??0,g=l[d]??0;if(f<g)return!0;if(f>g)return!1}return!!(o&&!i)}async function er(e={}){let t=e.runningVersion??ct,r=e.cwd??process.cwd(),s=e.cacheFile??pt(),n=e.fetchLatest??An,o=e.env??process.env,a=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!N.existsSync(J.join(r,"package-lock.json")))return;let i=null,c=Pn(s);if(c)i=c.latest;else{try{i=await n()}catch{i=null}i&&En({latest:i,fetchedAt:Date.now()},s)}i&&$n(t,i)&&a(`
18
18
  \x1B[33m\u26A0 shiplightai ${i} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var lt,ke,bn,Sn,vn,Ne=_(()=>{"use strict";lt="0.1.72",ke=lt!=="dev"?lt:void 0,bn="https://registry.npmjs.org/shiplightai/latest",Sn=3600*1e3,vn=10080*60*1e3});import*as ae from"fs";import*as ue from"path";function dt(e){let{projectPath:t}=e,r=e.projectName??ue.basename(ue.resolve(t));if(ae.existsSync(t)&&ae.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);ae.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let a=ue.join(t,n);ae.mkdirSync(ue.dirname(a),{recursive:!0}),ae.writeFileSync(a,o)};return s("package.json",An.replace("{{name}}",r)),s("playwright.config.ts",$n),s(".gitignore",In),s(".env.example",Mn),s(".mcp.json",On),s("auth/example.login.ts",Ln),s("environments/example.env.yaml",Rn),s("tests/example.test.yaml",Cn),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var An,$n,In,Mn,On,Ln,Rn,Cn,Qt=_(()=>{"use strict";An=`{
20
+ `)}var ct,Pe,Sn,_n,xn,Ce=_(()=>{"use strict";ct="0.1.73",Pe=ct!=="dev"?ct:void 0,Sn="https://registry.npmjs.org/shiplightai/latest",_n=3600*1e3,xn=10080*60*1e3});import*as ae from"fs";import*as ue from"path";function ut(e){let{projectPath:t}=e,r=e.projectName??ue.basename(ue.resolve(t));if(ae.existsSync(t)&&ae.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);ae.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let a=ue.join(t,n);ae.mkdirSync(ue.dirname(a),{recursive:!0}),ae.writeFileSync(a,o)};return s("package.json",In.replace("{{name}}",r)),s("playwright.config.ts",Mn),s(".gitignore",On),s(".env.example",Ln),s(".mcp.json",Rn),s("auth/example.login.ts",Cn),s("environments/example.env.yaml",Nn),s("tests/example.test.yaml",Dn),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var In,Mn,On,Ln,Rn,Cn,Nn,Dn,tr=_(()=>{"use strict";In=`{
21
21
  "name": "{{name}}",
22
22
  "type": "module",
23
23
  "scripts": {
@@ -29,7 +29,7 @@ Install it as a project dependency instead:
29
29
  "dotenv": "^16.4.7"
30
30
  }
31
31
  }
32
- `,$n=`import { defineConfig, shiplightConfig } from 'shiplightai';
32
+ `,Mn=`import { defineConfig, shiplightConfig } from 'shiplightai';
33
33
 
34
34
  export default defineConfig({
35
35
  ...shiplightConfig(),
@@ -47,13 +47,13 @@ export default defineConfig({
47
47
  trace: 'on',
48
48
  },
49
49
  });
50
- `,In=`node_modules/
50
+ `,On=`node_modules/
51
51
  test-results/
52
52
  shiplight-report/
53
53
  .shiplight/
54
54
  .env
55
55
  *.yaml.spec.ts
56
- .auth`,Mn=`# Shiplight API token (optional) \u2014 enables cloud sync tools
56
+ .auth`,Ln=`# Shiplight API token (optional) \u2014 enables cloud sync tools
57
57
  # Get yours at https://app.shiplight.ai/settings/api-tokens
58
58
  # SHIPLIGHT_API_TOKEN=
59
59
 
@@ -77,7 +77,7 @@ shiplight-report/
77
77
 
78
78
  # Optional: override starting URL for all tests
79
79
  # PLAYWRIGHT_STARTING_URL=
80
- `,On=`{
80
+ `,Rn=`{
81
81
  "mcpServers": {
82
82
  "shiplight": {
83
83
  "command": "npx",
@@ -90,7 +90,7 @@ shiplight-report/
90
90
  }
91
91
  }
92
92
  }
93
- }`,Ln=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
93
+ }`,Cn=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
94
94
  import path from 'path';
95
95
  import fs from 'fs/promises';
96
96
  import { chromium } from '@playwright/test';
@@ -133,13 +133,13 @@ export async function login(args: Record<string, unknown> = {}): Promise<string>
133
133
 
134
134
  return sessionPath;
135
135
  }
136
- `,Rn=`# This is an example of environment. Don't use it.
136
+ `,Nn=`# This is an example of environment. Don't use it.
137
137
  name: example
138
138
  url: https://staging.example.com
139
139
  accounts:
140
140
  - username: test-user-1@example.com
141
141
  password: EXAMPLE_TEST_USER_1_PASSWORD
142
- 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,Cn=`goal: Verify the Shiplight homepage links to the quick-start docs
142
+ 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,Dn=`goal: Verify the Shiplight homepage links to the quick-start docs
143
143
  base_url: https://www.shiplight.ai
144
144
  statements:
145
145
  - URL: /
@@ -153,7 +153,7 @@ statements:
153
153
 
154
154
  - VERIFY: The browser lands on the docs site
155
155
  js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
156
- `});var er=_(()=>{"use strict";Qt()});var tr={};ie(tr,{runCreate:()=>Nn});import*as je from"path";import*as Fe from"fs";function De(){console.log(`
156
+ `});var rr=_(()=>{"use strict";tr()});var sr={};ie(sr,{runCreate:()=>jn});import*as De from"path";import*as je from"fs";function Ne(){console.log(`
157
157
  Usage: shiplight create <path> [options]
158
158
 
159
159
  Scaffold a new Shiplight test project at <path>. Creates package.json,
@@ -166,22 +166,22 @@ Options:
166
166
  Examples:
167
167
  shiplight create ./my-tests
168
168
  shiplight create ./my-tests --name acme-e2e
169
- `)}async function Nn(e){let t,r;for(let o=0;o<e.length;o++){let a=e[o];if(a==="--help"||a==="-h"){De();return}else a==="--name"?(r=e[++o],(!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.
169
+ `)}async function jn(e){let t,r;for(let o=0;o<e.length;o++){let a=e[o];if(a==="--help"||a==="-h"){Ne();return}else a==="--name"?(r=e[++o],(!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.
170
170
  npm package names must be lowercase.
171
171
  Use only lowercase letters, digits, hyphens, underscores, and dots;
172
- must start with a lowercase letter or digit.`),process.exit(1))):a.startsWith("--")?(console.error(`Error: Unknown option: ${a}`),De(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${a}`),De(),process.exit(1)):t=a}t||(console.error(`Error: missing required <path> argument
173
- `),De(),process.exit(1));let s=je.resolve(t);Fe.existsSync(s)&&Fe.readdirSync(s).length>0&&(console.error(`Error: target directory is not empty: ${s}
174
- Remove it or choose a different path.`),process.exit(1));let n=dt({projectPath:s,projectName:r});console.log(`
172
+ must start with a lowercase letter or digit.`),process.exit(1))):a.startsWith("--")?(console.error(`Error: Unknown option: ${a}`),Ne(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${a}`),Ne(),process.exit(1)):t=a}t||(console.error(`Error: missing required <path> argument
173
+ `),Ne(),process.exit(1));let s=De.resolve(t);je.existsSync(s)&&je.readdirSync(s).length>0&&(console.error(`Error: target directory is not empty: ${s}
174
+ Remove it or choose a different path.`),process.exit(1));let n=ut({projectPath:s,projectName:r});console.log(`
175
175
  Created Shiplight test project at ${n.projectPath}
176
176
  `),console.log(` Name: ${n.projectName}`),console.log(" Files:");for(let o of n.filesCreated)console.log(` - ${o}`);console.log(`
177
177
  Next steps:
178
178
 
179
- cd ${je.relative(process.cwd(),n.projectPath)||"."}
179
+ cd ${De.relative(process.cwd(),n.projectPath)||"."}
180
180
  cp .env.example .env # then edit .env and set an AI provider API key
181
181
  npm install
182
182
  npx playwright install chromium
183
183
  npx shiplight test
184
- `)}var rr=_(()=>{"use strict";er()});import*as X from"fs";import*as ce from"path";function sr(e){let t=ce.resolve(e);for(;;){if(X.existsSync(ce.join(t,"package.json"))||X.existsSync(ce.join(t,".shiplight")))return t;let r=ce.dirname(t);if(r===t)break;t=r}return null}function nr(e){return ce.join(e,Dn)}function or(e,t){return ce.join(nr(e),`${jn}${t}.json`)}function ir(e,t){let r=sr(t);if(!r)return;let s=nr(r);X.mkdirSync(s,{recursive:!0});let n={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};X.writeFileSync(or(r,e),JSON.stringify(n),"utf-8")}function ar(e,t){let r=sr(t);if(r)try{X.unlinkSync(or(r,e))}catch{}}var Dn,jn,cr=_(()=>{"use strict";Dn=".shiplight/run",jn="debug-"});import*as pr from"node:net";function lr(e,t){return new Promise(r=>{let s=pr.createServer();s.once("error",n=>{r(n.code!=="EADDRINUSE")}),s.once("listening",()=>{s.close(()=>r(!0))}),s.listen(e,t)})}async function Fn(e){return await lr(e,"127.0.0.1")?lr(e,"::1"):!1}async function ur(e,t){for(let r=e;r<e+t;r++)if(await Fn(r))return r;return null}var dr=_(()=>{"use strict"});var hr={};ie(hr,{findPlaywrightConfig:()=>Ee,makeIdempotentFileCleaner:()=>fr,spawnPlaywrightProcess:()=>ft});import*as le from"fs";import*as te from"path";import{spawn as Un}from"child_process";import{parse as Bn}from"yaml";function fr(e){let t=!1;return()=>{if(!t){t=!0;try{le.unlinkSync(e)}catch{}}}}function Ee(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=te.resolve(e);for(;;){for(let n of t){let o=te.join(r,n);if(le.existsSync(o))return o}let s=te.dirname(r);if(s===r)break;r=s}return null}function Hn(e,t,r,s){let n={...r},o=n.launchOptions?.args??[];return n.launchOptions={...n.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
184
+ `)}var nr=_(()=>{"use strict";rr()});import*as X from"fs";import*as ce from"path";function or(e){let t=ce.resolve(e);for(;;){if(X.existsSync(ce.join(t,"package.json"))||X.existsSync(ce.join(t,".shiplight")))return t;let r=ce.dirname(t);if(r===t)break;t=r}return null}function ir(e){return ce.join(e,Fn)}function ar(e,t){return ce.join(ir(e),`${Un}${t}.json`)}function cr(e,t){let r=or(t);if(!r)return;let s=ir(r);X.mkdirSync(s,{recursive:!0});let n={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};X.writeFileSync(ar(r,e),JSON.stringify(n),"utf-8")}function lr(e,t){let r=or(t);if(r)try{X.unlinkSync(ar(r,e))}catch{}}var Fn,Un,pr=_(()=>{"use strict";Fn=".shiplight/run",Un="debug-"});import*as dr from"node:net";function ur(e,t){return new Promise(r=>{let s=dr.createServer();s.once("error",n=>{r(n.code!=="EADDRINUSE")}),s.once("listening",()=>{s.close(()=>r(!0))}),s.listen(e,t)})}async function Bn(e){return await ur(e,"127.0.0.1")?ur(e,"::1"):!1}async function fr(e,t){for(let r=e;r<e+t;r++)if(await Bn(r))return r;return null}var hr=_(()=>{"use strict"});var mr={};ie(mr,{findPlaywrightConfig:()=>Ee,makeIdempotentFileCleaner:()=>gr,spawnPlaywrightProcess:()=>dt});import*as le from"fs";import*as te from"path";import{spawn as Hn}from"child_process";import{parse as Wn}from"yaml";function gr(e){let t=!1;return()=>{if(!t){t=!0;try{le.unlinkSync(e)}catch{}}}}function Ee(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=te.resolve(e);for(;;){for(let n of t){let o=te.join(r,n);if(le.existsSync(o))return o}let s=te.dirname(r);if(s===r)break;r=s}return null}function Gn(e,t,r,s){let n={...r},o=n.launchOptions?.args??[];return n.launchOptions={...n.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
185
185
  import { test } from 'shiplightai/fixture';
186
186
  ${`
187
187
  test.use(${JSON.stringify(n)});
@@ -205,30 +205,30 @@ test('__shiplight_debug__', async ({ page, agent }) => {
205
205
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
206
206
  await new Promise(() => {});
207
207
  });
208
- `}async function Gn(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let o=t();o.once("error",()=>n(!1)),o.once("listening",()=>{o.close(()=>n(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function ft(e){let{yamlFilePath:t,configPath:r,tempSuffix:s="",headed:n}=e,o=te.dirname(r),a=await Gn(16174),i;if(!le.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let w=Bn(le.readFileSync(t,"utf-8"));w?.use&&typeof w.use=="object"&&!Array.isArray(w.use)&&(i=w.use),w?.base_url&&!i?.baseURL&&(i={...i,baseURL:w.base_url})}catch(w){console.error("[debugger] Could not parse YAML for `use` block:",w)}let l=te.dirname(te.resolve(t)),c=s?`-${s}`:"",u=te.join(l,`.__shiplight_debug__${c}.yaml.spec.ts`),d=Hn(t,a,i,o);le.writeFileSync(u,d);let f=fr(u),g=["playwright","test",u,...n?["--headed"]:[]],h=Un("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});h.stdout?.on("data",w=>{process.stderr.write(w)}),h.stderr?.on("data",w=>{process.stderr.write(w)});let p=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),h.on("close",w=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),f(),w!==0&&w!==null&&console.error(`[debugger] Playwright process exited with code ${w}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let b=["127.0.0.1","::1"];async function y(w){try{let E=w.includes(":")?`[${w}]`:w,T=await fetch(`http://${E}:${a}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let m=null;for(let w=0;w<180;w++){if(h.exitCode!==null)throw f(),new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`);for(let E of b)if(await y(E)){m=E;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${a}`);break}if(w===179)throw p(),f(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(E=>setTimeout(E,1e3))}if(!m)throw p(),f(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:m,pid:h.pid??0,cleanup:async()=>{p(),f()}}}var ht=_(()=>{"use strict"});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});import{v4 as fa}from"uuid";var wr=_(()=>{"use strict"});import{z as S}from"zod";var br,gt,Sr,we,vr,_r,xr,B,Tr,Ue,mt,yt=_(()=>{"use strict";br=S.enum(["JS_CODE","AI_MODE"]),gt=S.object({type:br,expression:S.string()}),Sr=S.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=S.object({uid:S.string(),type:Sr,comment:S.string().optional()}),vr=S.object({action_data:S.object({action_name:S.string(),kwargs:S.record(S.any()).optional(),args:S.array(S.any()).optional()}),action_description:S.string().optional(),url:S.string().optional(),xpath:S.string().nullable().optional(),locator:S.string().nullable().optional(),css_selector:S.string().nullable().optional(),unique_selector:S.string().nullable().optional(),element_index:S.number().nullable().optional(),frame_path:S.array(S.any()).optional(),artifacts:S.record(S.any()).optional(),feedback:S.string().optional(),original_browser_use_action:S.any().optional()}).passthrough(),_r=we.extend({type:S.literal("DRAFT"),description:S.string()}),xr=we.extend({type:S.literal("ACTION"),description:S.string(),action_entity:vr.optional(),locator:S.string().optional(),use_pure_vision:S.boolean().optional()}),B=S.lazy(()=>S.union([_r,xr,we.extend({type:S.literal("STEP"),description:S.string().optional().default(""),statements:S.array(B),reference_id:S.number().optional(),template_path:S.string().optional(),template_params:S.record(S.string()).optional()}),we.extend({type:S.literal("IF_ELSE"),description:S.string().optional(),condition:gt,then:S.array(B),else:S.array(B).optional()}),we.extend({type:S.literal("WHILE_LOOP"),description:S.string().optional(),condition:gt,body:S.array(B),timeout_ms:S.number().optional()})])),Tr=S.object({name:S.string(),statements:S.array(B),teardown:S.array(B).optional(),skip:S.union([S.boolean(),S.string()]).optional(),timeout:S.number().optional(),fail:S.union([S.boolean(),S.string()]).optional(),only:S.boolean().optional(),slow:S.boolean().optional()}),Ue=S.object({tests:S.array(Tr).min(1),beforeAll:S.array(B).optional(),afterAll:S.array(B).optional(),beforeEach:S.array(B).optional(),afterEach:S.array(B).optional()}),mt=S.object({comment:S.string().optional(),version:S.string().optional(),goal:S.string().optional(),url:S.string().optional(),baseURL:S.string().optional(),final_feedback:S.string().optional(),completed:S.boolean().optional(),success:S.boolean().optional(),statements:S.array(B).optional(),teardown:S.array(B).optional(),last_modified_at:S.string().optional(),testGroup:Ue.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 ba,parse as Ar,parseAllDocuments as Sa,parseDocument as Wn,Document as $r,isMap as Se,isSeq as H}from"yaml";import{v4 as q}from"uuid";function Be(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(K)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(K)),r}function $e(e,t){if(e.testGroup)return Mr(e,t);let r=Be(e,t),s=new $r(r),n=s.contents?.get("tags",!0);return H(n)&&(n.flow=!0),e.comment&&(s.commentBefore=e.comment),Pr(s,e.statements??[]),e.teardown&&Pr(s,e.teardown,"teardown"),s.toString(Ir)}function Pr(e,t,r="statements"){let s=e.contents;if(!s||!Se(s))return;let n=s.get(r,!0);H(n)&&Ae(n,t)}function Ae(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),s.comment&&(r===0?e.commentBefore=s.comment:n.commentBefore=s.comment),Se(n)){let o=n;if(s.type==="STEP"){let a=o.get("statements",!0);H(a)&&Ae(a,s.statements)}else if(s.type==="IF_ELSE"){let a=o.get("THEN",!0);H(a)&&Ae(a,s.then);let i=o.get("ELSE",!0);H(i)&&s.else&&Ae(i,s.else)}else if(s.type==="WHILE_LOOP"){let a=o.get("DO",!0);H(a)&&Ae(a,s.body)}}}}function Mr(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let s={};t?.test_case_id!==void 0&&(s.test_case_id=t.test_case_id),t?.name&&(s.name=t.name),t?.tags&&t.tags.length>0&&(s.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(s.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(s.settings=t.settings);let n={};e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(K)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(K)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(K)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(K)),n.tests=r.tests.map(i=>{let l={name:i.name};return i.skip!==void 0&&(l.skip=i.skip),i.timeout!==void 0&&(l.timeout=i.timeout),i.fail!==void 0&&(l.fail=i.fail),i.only!==void 0&&(l.only=i.only),i.slow!==void 0&&(l.slow=i.slow),l.statements=i.statements.map(K),i.teardown&&i.teardown.length>0&&(l.teardown=i.teardown.map(K)),l}),s.suite=n;let o=new $r(s),a=o.contents?.get("tags",!0);return H(a)&&(a.flow=!0),o.toString(Ir)}function K(e){switch(e.type){case"DRAFT":return Kn(e);case"ACTION":return Vn(e);case"STEP":return zn(e);case"IF_ELSE":return Yn(e);case"WHILE_LOOP":return Jn(e)}}function Kn(e){return{intent:e.description}}function Vn(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 i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l=r?.code;return typeof l=="string"&&l.trim()?{VERIFY:i,js:l}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l={URL:i};return r?.new_tab===!0&&(l.new_tab=!0),typeof r?.timeout_seconds=="number"&&(l.timeout_seconds=r.timeout_seconds),l}}if(t==="js_action"){let i=r?.code;if(typeof i=="string"&&i.trim()&&e.description)return{description:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let l={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(l.timeout_seconds=r.timeout_seconds),l}}if(t==="wait"){let i=r?.seconds,c={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(c.seconds=i),c}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:i}}if(!e.action_entity)return{intent:e.description};let s=e.action_entity.action_data??e.action_entity.action;if(!s)return{intent:e.description};let n={intent:e.description,action:s.action_name},o=e.locator??e.action_entity.locator;o&&(n.locator=o);let a=e.action_entity.xpath;if(a&&(n.xpath=a),e.use_pure_vision&&(n.use_pure_vision=!0),s.kwargs&&Object.keys(s.kwargs).length>0)for(let[i,l]of Object.entries(s.kwargs))i!=="uid"&&(i==="statement"&&(t==="ai_action"||t==="ai_step")||(n[i]=l));return s.args&&s.args.length>0&&(n.args=s.args),n}function zn(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(K)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Yn(e){let t={IF:Or(e.condition),THEN:e.then.map(K)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(K)),t}function Jn(e){let t={WHILE:Or(e.condition),DO:e.body.map(K)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Or(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ie(e){try{let t=Ar(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()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(s=>typeof s=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function wt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(wt);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let s={};for(let[n,o]of Object.entries(t))s[n]=wt(o);return s}function j(e){if(e.length>kr)throw new Error(`YAML input too large (${e.length} bytes, max ${kr})`);let t=wt(Ar(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Xn(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:V(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=V(t.teardown));let s=mt.safeParse(r);if(!s.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(s.error.errors)}`);let n=s.data;return He(e,n),n}function Xn(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 n={tests:r.map(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let l={name:i.name,statements:V(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(l.teardown=V(i.teardown)),i.skip!==void 0&&(l.skip=i.skip),typeof i.timeout=="number"&&(l.timeout=i.timeout),i.fail!==void 0&&(l.fail=i.fail),i.only===!0&&(l.only=!0),i.slow===!0&&(l.slow=!0),l})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=V(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=V(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=V(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=V(t.afterEach));let o=Ue.safeParse(n);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function V(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(qn)}function qn(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 Zn(t);if("WHILE"in t)return Qn(t);if("STEP"in t)return eo(t);if("VERIFY"in t){let r=t.VERIFY,s={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(s.code=t.js),{uid:q(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:s}}}}if("URL"in t){let r=t.URL,s=t.new_tab===!0?!0:void 0,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return s&&(o.new_tab=!0),n!==void 0&&(o.timeout_seconds=n),{uid:q(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:q(),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:s}}}}}if("WAIT"in t){let r=t.WAIT,s=typeof t.seconds=="number"?t.seconds:3;return{uid:q(),type:"ACTION",description:typeof r=="string"?r:`Wait ${s}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${s}s`,action_data:{action_name:"wait",kwargs:{seconds:s}}}}}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.');let s=typeof t.description=="string"?t.description:"Code block";return{uid:q(),type:"ACTION",description:s,action_entity:{action_description:s,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",s=t.js;return{uid:q(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof s=="string"?s:String(s)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...s}=t;return Er({...s,action:"function",functionName:r})}if("action"in t)return Er(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:q(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Lr(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 Zn(e){let t=Lr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let s={uid:q(),type:"IF_ELSE",condition:t,then:V(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(s.else=V(e.ELSE)),s}function Qn(e){let t=Lr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let s={uid:q(),type:"WHILE_LOOP",condition:t,body:V(r)};return typeof e.timeout_ms=="number"&&(s.timeout_ms=e.timeout_ms),s}function eo(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:q(),type:"STEP",description:t,statements:V(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let s=e.template_params,n={};for(let[o,a]of Object.entries(s))n[o]=String(a);r.template_params=n}return r}function Er(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:"",s=typeof e.locator=="string"?e.locator:void 0,n=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[c,u]of Object.entries(e))to.has(c)||(a[c]=u);t==="verify"&&typeof a.js=="string"&&(a.code=a.js,delete a.js),(t==="ai_action"||t==="ai_step")&&a.statement===void 0&&(a.statement=r);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};s&&(i.locator=s),n&&(i.xpath=n);let l={uid:q(),type:"ACTION",description:r,action_entity:i};return o&&(l.use_pure_vision=!0),l}function He(e,t){let r;try{r=Wn(e)}catch{return}let s=r.contents;if(!s||!Se(s))return;if(r.commentBefore)t.comment=r.commentBefore;else{let l=s.items?.[0];l?.key&&l.key.commentBefore&&(t.comment=l.key.commentBefore)}let n=s,o=n.get("statements",!0);H(o)&&t.statements&&be(o,t.statements);let a=n.get("teardown",!0);H(a)&&t.teardown&&be(a,t.teardown)}function be(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 s=e.items[r];s.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=s.commentBefore);let n=t[r];if(n.type==="STEP"&&Se(s)){let o=s.get("statements",!0);H(o)&&be(o,n.statements)}else if(n.type==="IF_ELSE"&&Se(s)){let o=s.get("THEN",!0);H(o)&&be(o,n.then);let a=s.get("ELSE",!0);H(a)&&n.else&&be(a,n.else)}else if(n.type==="WHILE_LOOP"&&Se(s)){let o=s.get("DO",!0);H(o)&&be(o,n.body)}}}var Ir,kr,to,Ge=_(()=>{"use strict";yt();Ir={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};kr=1024*1024;to=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as Pa,stringify as ka}from"yaml";var Rr=_(()=>{"use strict";Ge()});var bt,We,Ke=_(()=>{"use strict";bt=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},We=e=>{let t=[],r=s=>{for(let n of s){t.push(n);let o=bt(n);for(let a of o)r(a.statements)}};return r(e),t}});function Fr(e){let t=0,r=0;for(let s of e)if(s.type==="DRAFT")r++;else if(s.type==="ACTION"){let n=s.action_entity?.action_data?.action_name??"";jr.has(n)||t++}return{action:t,draft:r}}function so(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Dr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function no(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,s=t.join(`
209
- `).match(/\bawait\b/g),n=s?s.length:0;return r<=Cr&&n<=Nr?null:`${r} non-blank line(s), ${n} await(s) \u2014 limits are ${Cr} lines and ${Nr} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function St(e,t){let r=t?.coverageThreshold??ro,s=[],n=[],o;try{o=j(e)}catch(f){return{valid:!1,errors:[`Invalid YAML: ${f.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||s.push('Missing required field: "goal"'),o.statements?.length||s.push('Missing required field: "statements"');let a=[...We(o.statements??[]),...o.teardown?We(o.teardown):[]],{action:i,draft:l}=Fr(a),c="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let f of a){let g=f;if(g.reference_id!==void 0){let h=g.description||f.uid;s.push(`Unresolved cloud template reference on statement "${h}" (reference_id: ${g.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(f.type==="ACTION"){let h=f,p=h.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let b=h.action_entity?.action_data?.kwargs?.code;if(typeof b=="string"){let y=so(b);if(y){let m=h.description||p;s.push(`Invalid JS in "${m}": ${y}. ${c}`)}else if(p==="verify"){let m=no(b);if(m){let w=h.description||p;s.push(`JS cache for "${w}" is too complex: ${m}`)}}}}}if(f.type==="IF_ELSE"){let h=f;if(h.condition.type==="JS_CODE"){let p=Dr(h.condition.expression);p&&s.push(`Invalid JS in IF condition "${h.condition.expression}": ${p}. ${c}`)}}if(f.type==="WHILE_LOOP"){let h=f;if(h.condition.type==="JS_CODE"){let p=Dr(h.condition.expression);p&&s.push(`Invalid JS in WHILE condition "${h.condition.expression}": ${p}. ${c}`)}}}let u=i+l,d=u>0?Math.round(i/u*100):0;return u>0&&d/100<r&&n.push(`Low action coverage: ${i}/${u} statements (${d}%) are enriched with action/js. ${l} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:s.length===0,errors:s,warnings:n,stats:{total:u,action:i,draft:l,coverage:d}}}var ro,Cr,Nr,jr,Ur=_(()=>{"use strict";Ge();Ke();ro=.5,Cr=5,Nr=3,jr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Ve=_(()=>{"use strict"});var Br=_(()=>{"use strict";Ve()});var Hr,io,Gr,Wr,Me,ao,co,Kr=_(()=>{"use strict";Hr=112,io=1080-Hr,Gr={"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"}},Wr={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"]},Me=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Wr[e].map(s=>Gr[s]).filter(s=>s.defaultBrowserType&&r.includes(s.defaultBrowserType))},ao={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile")}},co={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile",!0)}}});var Vr=_(()=>{"use strict"});function vt(){return{version:"1.0",entries:{}}}var zr=_(()=>{"use strict";Ke()});var Z,_t,Yr=_(()=>{"use strict";Z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(Z||{}),_t=18e4});var Jr=_(()=>{"use strict"});var Xr=_(()=>{"use strict"});var qr=_(()=>{"use strict"});var Zr=_(()=>{"use strict";Ve()});var de=_(()=>{"use strict";gr();mr();yr();wr();Rr();Ur();Ge();yt();Br();Kr();Vr();zr();Ke();Yr();Jr();Xr();qr();Zr();Ve()});import{stringify as po}from"yaml";import{createHash as $o}from"crypto";import{parse as Io,stringify as cs}from"yaml";import{readFileSync as Mo,existsSync as Oo}from"fs";import{resolve as Tt,dirname as Lo}from"path";import{parse as rs,stringify as Ro}from"yaml";import{readFileSync as Bo,writeFileSync as Ho,mkdirSync as Go}from"fs";import{dirname as Wo}from"path";function re(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function O(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 fo(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 Ye(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 s=fo(e);if(s){let n=JSON.stringify(s);return`${t}.locator(${n}).first()`}return null}function Qr(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:ho.includes(t)}function mo(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 v(e,t){N.set(e,t)}function yo(e){return N.get(e)}function ge(e,t,r=[]){let s=[...r];return t.locator?s.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&s.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&s.push(`frame_path: ${JSON.stringify(t.frame_path)}`),s.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...s.map(n=>` ${n},`),"});"]}function es(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 s=["page","testContext","request","agent"],n=["undefined","null","true","false"],o=r.map(a=>s.includes(a)||n.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${o.join(", ")})`}function se(e,t,r,s="main"){let n=[];for(let o=0;o<e.length;o++){let a=e[o],i=`${s}.${o}`,l=wo(a,t,i,r);l.length>0&&(n.push(...l),o<e.length-1&&n.push(""))}return n}function wo(e,t,r,s){let n=" ".repeat(t);switch(e.type){case"DRAFT":return bo(e,t,r,s);case"ACTION":return So(e,t,r,s);case"STEP":return vo(e,t,r,s);case"IF_ELSE":return _o(e,t,r,s);case"WHILE_LOOP":return xo(e,t,r,s);default:return[`${n}// Unknown statement type: ${e.type}`]}}function bo(e,t,r,s){let n=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let a=JSON.stringify(o);return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${a}, '${r}');`]}function So(e,t,r,s){let n=" ".repeat(t),o=e.description,a=e.uid,l=s.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!l){if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let w=JSON.stringify(o),E=!!e.use_pure_vision;return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${w}, '${r}', ${E});`]}let c=e.locator?{...l,locator:e.locator}:l;o&&o!==c.action_description&&(c={...c,action_description:o});let u=c.action_data?.action_name||"",d=c.action_description||"",f=yo(u);if(!f)return[`${n}// ${r}: Unknown action: ${u}`];let g={imports:s.imports},h=f(c,r,g);if(s.noAgent){if(Qr(c))return[`${n}// ${r}: ${O(d)}`,`${n}// AI action: ${O(d)} (requires agent - skipped in hook)`];let w=To(c,u,n,r);return w||[`${n}// ${r}: ${O(d)}`,...h.map(E=>`${n}${E}`)]}if(Qr(c))return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,...h.map(w=>`${n}${w}`)];let p=JSON.stringify(d),b=h.map(w=>`${n} ${w}`),y=mo(c),m=a?`'${a}'`:"undefined";return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...b,`${n}}, ${p}, '${r}', ${m}, ${y});`]}function vo(e,t,r,s){let n=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${n}// Step: ${O(e.description)}`);let a=se(e.statements,t,s,r);return o.push(...a),o}function _o(e,t,r,s){let n=" ".repeat(t),o=[];if(o.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${n}if (${e.condition.expression}) {`);else{o.push(`${n}// AI Condition: ${O(e.condition.expression)}`);let i=JSON.stringify(e.condition.expression);o.push(`${n}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=se(e.then,t+1,s,`${r}.then`);if(o.push(...a),e.else&&e.else.length>0){o.push(`${n}} else {`);let i=se(e.else,t+1,s,`${r}.else`);o.push(...i)}return o.push(`${n}}`),o}function xo(e,t,r,s){let n=" ".repeat(t),o=[];o.push(`${n}// ${r}: Loop`);let a=e.timeout_ms??_t,i=a/1e3,l=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,c=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${n}const ${c}_start = Date.now();`),o.push(`${n}const ${c}_timeout = ${a};`),o.push(`${n}const ${c}_check = () => {`),o.push(`${n} if (Date.now() - ${c}_start > ${c}_timeout) {`),o.push(`${n} throw new Error('${l}');`),o.push(`${n} }`),o.push(`${n} return true;`),o.push(`${n}};`),e.condition.type==="JS_CODE")o.push(`${n}while (${c}_check() && (${e.condition.expression})) {`);else{o.push(`${n}// AI Loop Condition: ${O(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${n}while (${c}_check() && await agent.evaluate(page, ${d}, "${r}")) {`)}let u=se(e.body,t+1,s,`${r}.body`);return o.push(...u),o.push(`${n}}`),o}function To(e,t,r,s){let n=e.action_description||"",o=e.action_data?.kwargs||{},a=o.timeout_ms??Pt;switch(t){case"go_to_url":case"open_tab":{let i=o.url||"";return[`${r}// ${s}: ${O(n)}`,`${r}await page.goto(${JSON.stringify(i)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goForward();`];case"input_text":{let i=o.text||"",l=Ye(e);return l?[`${r}// ${s}: ${O(n)}`,`${r}await ${l}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=o.text||o.label||"",l=Ye(e);return l?[`${r}// ${s}: ${O(n)}`,`${r}await ${l}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function Po(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...is()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...ts("beforeEach",t.beforeEach,o)),r.push(""));let a=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 i=t?.testName||e.goal||"Generated test",l=xt(t?.tags);for(let c of t.parameters){let u=os(e,c.values);r.push(...Xe(u,`${l}${re(i)} [${re(c.name)}]`,o,0,a,c.name)),r.push("")}}else{let i=t?.testName||e.goal||"Generated test",l=xt(t?.tags);r.push(...Xe(e,`${l}${re(i)}`,o,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...ts("afterEach",t.afterEach,o))),as(r,n),r.join(`
210
- `)}function ko(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...is()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},a=t?.testName||"Test Suite",i=xt(t?.tags);r.push(`test.describe.serial('${i}${re(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...ze("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...ze("beforeEach",e.beforeEach,o,1)),r.push(""));for(let c=0;c<e.tests.length;c++){let u=e.tests[c],d=u.timeout||u.skip!==void 0||u.fail!==void 0||u.only||u.slow?{timeout:u.timeout,skip:u.skip,fail:u.fail,only:u.only,slow:u.slow}:void 0;if(u.parameters&&u.parameters.length>0)for(let f of u.parameters){let g=os(u.testFlow,f.values);r.push(...Xe(g,`${re(u.name)} [${re(f.name)}]`,o,1,d,f.name)),r.push("")}else r.push(...Xe(u.testFlow,re(u.name),o,1,d)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...ze("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...ze("afterAll",e.afterAll,o,1)),r.push("});"),as(r,n),r.join(`
211
- `)}function xt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Je(e){let t=new Set;function r(s){for(let n of s)switch(n.type){case Z.ACTION:{let a=n.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&Eo.includes(i)&&t.add(i);break}case Z.STEP:r(n.statements);break;case Z.IF_ELSE:{let o=n;r(o.then),o.else&&r(o.else);break}case Z.WHILE_LOOP:r(n.body);break}}return r(e),t}function Ao(e){let t=Je(e.statements??[]);if(e.teardown)for(let r of Je(e.teardown))t.add(r);return t}function kt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Xe(e,t,r,s=0,n,o){let a=" ".repeat(s),i=[],l=Ao(e),c=kt(l),u=n?.only?"test.only":"test",d=o?`, { tag: '@${re(o)}' }`:"";i.push(`${a}${u}('${t}'${d}, async (${c}) => {`),n?.skip===!0?i.push(`${a} test.skip();`):typeof n?.skip=="string"&&i.push(`${a} test.skip(true, '${re(n.skip)}');`),n?.fail===!0?i.push(`${a} test.fail();`):typeof n?.fail=="string"&&i.push(`${a} test.fail(true, '${re(n.fail)}');`),n?.slow&&i.push(`${a} test.slow();`),n?.timeout&&i.push(`${a} test.setTimeout(${n.timeout});`);let f=e.teardown&&e.teardown.length>0,g=s+1;if(f){if(i.push(`${a} try {`),e.statements&&e.statements.length>0){i.push(`${a} // Test steps`);let p=se(e.statements,g+1,r);i.push(...p)}i.push(`${a} } finally {`),i.push(`${a} // Teardown`);let h=se(e.teardown,g+1,r,"teardown");i.push(...h),i.push(`${a} }`)}else if(e.statements&&e.statements.length>0){i.push(`${a} // Test steps`);let h=se(e.statements,g,r);i.push(...h)}return i.push(`${a}});`),i}function ts(e,t,r){let s=[],n=ns(t),o=Je(n),a=kt(o);return s.push(`test.${e}(async (${a}) => {`),s.push(...se(n,1,r,e)),s.push("});"),s}function ze(e,t,r,s){let n=" ".repeat(s),o=[],a=ns(t);if(e==="beforeAll"||e==="afterAll"){let l={...r,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...se(a,s+1,l,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let l=Je(a),c=kt(l);o.push(`${n}test.${e}(async (${c}) => {`),o.push(...se(a,s+1,r,e)),o.push(`${n}});`)}return o}function ns(e){let r=po({goal:"_hook",statements:e});return j(r).statements??[]}function os(e,t){let r=$e(e);for(let[s,n]of Object.entries(t))r=r.split(`<<${s}>>`).join(String(n));return j(r)}function is(){return["import { test, expect } from 'shiplightai/fixture';"]}function as(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let s=Array.from(t);e.splice(r,0,...s)}}function ls(e,t,r){let s={expandingPaths:new Set([Tt(t)]),depth:0,referencedPaths:new Set,basePath:r},n={...e};Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,s)),Array.isArray(n.teardown)&&(n.teardown=fe(n.teardown,t,s));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=fe(n[o],t,s));return{doc:n,referencedTemplatePaths:Array.from(s.referencedPaths)}}function fe(e,t,r){let s=[];for(let n of e)if(Co(n)){let o=No(n,t,r);s.push(o)}else s.push(Do(n,t,r));return s}function Co(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function No(e,t,r){if(r.depth>=ss)throw new Error(`Template expansion exceeded maximum depth of ${ss}. Check for deeply nested or circular template references.`);let s=Tt(Lo(t),e.template),n=!Oo(s)&&r.basePath?Tt(r.basePath,e.template):s;if(r.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);r.referencedPaths.add(n);let o;try{o=Mo(n,"utf-8")}catch(h){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${h.message}`)}let a=rs(o);if(!a||typeof a!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let i=a.params||[],l=e.params||{};for(let h of i)if(!(h in l))throw new Error(`Template ${e.template} requires param "${h}" but it was not provided. Required params: [${i.join(", ")}]`);let c=a.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let p=Ro(c);for(let[b,y]of Object.entries(l))p=p.split(`<<${b}>>`).join(String(y));c=rs(p)}let u={expandingPaths:new Set([...r.expandingPaths,n]),depth:r.depth+1,referencedPaths:r.referencedPaths},d=fe(c,n,u),g={STEP:a.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(l).length>0&&(g.template_params=l),g}function Do(e,t,r){if(typeof e!="object"||e===null)return e;let s={...e};return Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,r)),Array.isArray(s.THEN)&&(s.THEN=fe(s.THEN,t,r)),Array.isArray(s.ELSE)&&(s.ELSE=fe(s.ELSE,t,r)),Array.isArray(s.DO)&&(s.DO=fe(s.DO,t,r)),s}function At(e,t,r){let s=Io(e),n=s?.name,o=s?.tags,a=s?.use;if(s&&(s.name!==void 0||s.tags!==void 0||s.use!==void 0)&&(delete s.name,delete s.tags,delete s.use),s?.suite){if(s.goal||s.statements)throw new Et('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Fo(s,n,o,a,t,r)}return jo(s,n,o,a,t,r)}function jo(e,t,r,s,n,o){let a=e?.beforeEach,i=e?.afterEach,l=ps(e?.parameters),c=e?.timeout,u=e?.skip,d=e?.fail,f=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 Et(`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 h=[];if(n&&e&&typeof e=="object"){let y=ls(e,n,o);e=y.doc,h=y.referencedTemplatePaths}let p=cs(e),b=j(p);return n&&(ve(b.statements??[],n,"main"),b.teardown&&ve(b.teardown,n,"teardown")),{testFlow:b,name:t,tags:r,use:s,beforeEach:a,afterEach:i,parameters:l,timeout:c,skip:u,fail:d,only:f,slow:g,referencedTemplatePaths:h}}function Fo(e,t,r,s,n,o){let a=e.suite;if(!Array.isArray(a.tests)||a.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let i=a.beforeAll,l=a.afterAll,c=a.beforeEach,u=a.afterEach,d=[],f=a.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let b={goal:p.name,statements:p.statements};p.teardown&&(b.teardown=p.teardown);let y=[],m=b;if(n&&typeof b=="object"){let $=ls(b,n,o);m=$.doc,y=$.referencedTemplatePaths,d.push(...y)}let w=cs(m),E=j(w),T=ps(p.parameters);return{testFlow:E,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:T,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=a.base_url,h=g?{...s,baseURL:g}:s;return{suite:{beforeAll:i,afterAll:l,beforeEach:c,afterEach:u,tests:f},name:t,tags:r,use:h,referencedTemplatePaths:d}}function ps(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 ve(e,t,r){for(let s=0;s<e.length;s++){let n=e[s],o=`${r}.${s}`,a=n.description||"";if(n.uid=Uo(t,o,a),n.type===Z.STEP)ve(n.statements,t,o);else if(n.type===Z.IF_ELSE){let i=n;ve(i.then,t,`${o}.then`),i.else&&ve(i.else,t,`${o}.else`)}else n.type===Z.WHILE_LOOP&&ve(n.body,t,`${o}.body`)}}function Uo(e,t,r){let s=$o("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${s.slice(0,8)}-${s.slice(8,12)}-${s.slice(12,16)}-${s.slice(16,20)}-${s.slice(20,32)}`}function us(e,t){let r;try{r=Bo(e,"utf-8")}catch(s){return{valid:!1,errors:[`Failed to read file: ${s.message}`],warnings:[]}}return Ko(r,e,t)}function Ko(e,t,r){let s=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=s||n?null:St(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let a,i,l=[];try{let c=r?.parsed??At(e,t);l=c.referencedTemplatePaths;let u={version:r?.version,actionEntityStore:r?.actionEntityStore},d=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?a=ko(c.suite,{...u,testName:c.name,tags:c.tags,use:c.use}):a=Po(c.testFlow,{...u,testName:c.name,tags:c.tags,use:d,beforeEach:c.beforeEach,afterEach:c.afterEach,parameters:c.parameters,timeout:c.timeout,skip:c.skip,fail:c.fail,only:c.only,slow:c.slow});let f=a.split(`
208
+ `}async function Kn(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let o=t();o.once("error",()=>n(!1)),o.once("listening",()=>{o.close(()=>n(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function dt(e){let{yamlFilePath:t,configPath:r,tempSuffix:s="",headed:n}=e,o=te.dirname(r),a=await Kn(16174),i;if(!le.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let b=Wn(le.readFileSync(t,"utf-8"));b?.use&&typeof b.use=="object"&&!Array.isArray(b.use)&&(i=b.use),b?.base_url&&!i?.baseURL&&(i={...i,baseURL:b.base_url})}catch(b){console.error("[debugger] Could not parse YAML for `use` block:",b)}let c=te.dirname(te.resolve(t)),l=s?`-${s}`:"",u=te.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=Gn(t,a,i,o);le.writeFileSync(u,d);let f=gr(u),g=["playwright","test",u,...n?["--headed"]:[]],h=Hn("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});h.stdout?.on("data",b=>{process.stderr.write(b)}),h.stderr?.on("data",b=>{process.stderr.write(b)});let p=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),h.on("close",b=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),f(),b!==0&&b!==null&&console.error(`[debugger] Playwright process exited with code ${b}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let w=["127.0.0.1","::1"];async function y(b){try{let E=b.includes(":")?`[${b}]`:b,T=await fetch(`http://${E}:${a}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let m=null;for(let b=0;b<180;b++){if(h.exitCode!==null)throw f(),new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`);for(let E of w)if(await y(E)){m=E;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${a}`);break}if(b===179)throw p(),f(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(E=>setTimeout(E,1e3))}if(!m)throw p(),f(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:m,pid:h.pid??0,cleanup:async()=>{p(),f()}}}var ft=_(()=>{"use strict"});var yr=_(()=>{"use strict"});var wr=_(()=>{"use strict"});var br=_(()=>{"use strict"});import{v4 as ga}from"uuid";var vr=_(()=>{"use strict"});import{z as v}from"zod";var Sr,ht,_r,we,xr,Tr,kr,B,Pr,Fe,gt,mt=_(()=>{"use strict";Sr=v.enum(["JS_CODE","AI_MODE"]),ht=v.object({type:Sr,expression:v.string()}),_r=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=v.object({uid:v.string(),type:_r,comment:v.string().optional()}),xr=v.object({action_data:v.object({action_name:v.string(),kwargs:v.record(v.any()).optional(),args:v.array(v.any()).optional()}),action_description:v.string().optional(),url:v.string().optional(),xpath:v.string().nullable().optional(),locator:v.string().nullable().optional(),css_selector:v.string().nullable().optional(),unique_selector:v.string().nullable().optional(),element_index:v.number().nullable().optional(),frame_path:v.array(v.any()).optional(),artifacts:v.record(v.any()).optional(),feedback:v.string().optional(),original_browser_use_action:v.any().optional()}).passthrough(),Tr=we.extend({type:v.literal("DRAFT"),description:v.string()}),kr=we.extend({type:v.literal("ACTION"),description:v.string(),action_entity:xr.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),B=v.lazy(()=>v.union([Tr,kr,we.extend({type:v.literal("STEP"),description:v.string().optional().default(""),statements:v.array(B),reference_id:v.number().optional(),template_path:v.string().optional(),template_params:v.record(v.string()).optional()}),we.extend({type:v.literal("IF_ELSE"),description:v.string().optional(),condition:ht,then:v.array(B),else:v.array(B).optional()}),we.extend({type:v.literal("WHILE_LOOP"),description:v.string().optional(),condition:ht,body:v.array(B),timeout_ms:v.number().optional()})])),Pr=v.object({name:v.string(),statements:v.array(B),teardown:v.array(B).optional(),skip:v.union([v.boolean(),v.string()]).optional(),timeout:v.number().optional(),fail:v.union([v.boolean(),v.string()]).optional(),only:v.boolean().optional(),slow:v.boolean().optional()}),Fe=v.object({tests:v.array(Pr).min(1),beforeAll:v.array(B).optional(),afterAll:v.array(B).optional(),beforeEach:v.array(B).optional(),afterEach:v.array(B).optional()}),gt=v.object({comment:v.string().optional(),version:v.string().optional(),goal:v.string().optional(),url:v.string().optional(),baseURL:v.string().optional(),final_feedback:v.string().optional(),completed:v.boolean().optional(),success:v.boolean().optional(),statements:v.array(B).optional(),teardown:v.array(B).optional(),last_modified_at:v.string().optional(),testGroup:Fe.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 Sa,parse as Ir,parseAllDocuments as _a,parseDocument as Vn,Document as Mr,isMap as ve,isSeq as H}from"yaml";import{v4 as q}from"uuid";function Ue(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(K)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(K)),r}function wt(e,t){if(e.testGroup)return Lr(e,t);let r=Ue(e,t),s=new Mr(r),n=s.contents?.get("tags",!0);return H(n)&&(n.flow=!0),e.comment&&(s.commentBefore=e.comment),Er(s,e.statements??[]),e.teardown&&Er(s,e.teardown,"teardown"),s.toString(Or)}function Er(e,t,r="statements"){let s=e.contents;if(!s||!ve(s))return;let n=s.get(r,!0);H(n)&&Ae(n,t)}function Ae(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),s.comment&&(r===0?e.commentBefore=s.comment:n.commentBefore=s.comment),ve(n)){let o=n;if(s.type==="STEP"){let a=o.get("statements",!0);H(a)&&Ae(a,s.statements)}else if(s.type==="IF_ELSE"){let a=o.get("THEN",!0);H(a)&&Ae(a,s.then);let i=o.get("ELSE",!0);H(i)&&s.else&&Ae(i,s.else)}else if(s.type==="WHILE_LOOP"){let a=o.get("DO",!0);H(a)&&Ae(a,s.body)}}}}function Lr(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let s={};t?.test_case_id!==void 0&&(s.test_case_id=t.test_case_id),t?.name&&(s.name=t.name),t?.tags&&t.tags.length>0&&(s.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(s.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(s.settings=t.settings);let n={};e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(K)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(K)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(K)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(K)),n.tests=r.tests.map(i=>{let c={name:i.name};return i.skip!==void 0&&(c.skip=i.skip),i.timeout!==void 0&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only!==void 0&&(c.only=i.only),i.slow!==void 0&&(c.slow=i.slow),c.statements=i.statements.map(K),i.teardown&&i.teardown.length>0&&(c.teardown=i.teardown.map(K)),c}),s.suite=n;let o=new Mr(s),a=o.contents?.get("tags",!0);return H(a)&&(a.flow=!0),o.toString(Or)}function K(e){switch(e.type){case"DRAFT":return zn(e);case"ACTION":return Yn(e);case"STEP":return Jn(e);case"IF_ELSE":return Xn(e);case"WHILE_LOOP":return qn(e)}}function zn(e){return{intent:e.description}}function Yn(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 i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:i,js:c}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:i};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 i=r?.code;if(typeof i=="string"&&i.trim()&&e.description)return{description:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let c={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let i=r?.seconds,l={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(l.seconds=i),l}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:i}}if(!e.action_entity)return{intent:e.description};let s=e.action_entity.action_data??e.action_entity.action;if(!s)return{intent:e.description};let n={intent:e.description,action:s.action_name},o=e.locator??e.action_entity.locator;o&&(n.locator=o);let a=e.action_entity.xpath;if(a&&(n.xpath=a),e.use_pure_vision&&(n.use_pure_vision=!0),s.kwargs&&Object.keys(s.kwargs).length>0)for(let[i,c]of Object.entries(s.kwargs))i!=="uid"&&(i==="statement"&&(t==="ai_action"||t==="ai_step")||(n[i]=c));return s.args&&s.args.length>0&&(n.args=s.args),n}function Jn(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(K)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Xn(e){let t={IF:Rr(e.condition),THEN:e.then.map(K)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(K)),t}function qn(e){let t={WHILE:Rr(e.condition),DO:e.body.map(K)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Rr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function $e(e){try{let t=Ir(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()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(s=>typeof s=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function yt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(yt);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let s={};for(let[n,o]of Object.entries(t))s[n]=yt(o);return s}function W(e){if(e.length>Ar)throw new Error(`YAML input too large (${e.length} bytes, max ${Ar})`);let t=yt(Ir(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Zn(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:V(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=V(t.teardown));let s=gt.safeParse(r);if(!s.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(s.error.errors)}`);let n=s.data;return Be(e,n),n}function Zn(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 n={tests:r.map(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let c={name:i.name,statements:V(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(c.teardown=V(i.teardown)),i.skip!==void 0&&(c.skip=i.skip),typeof i.timeout=="number"&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only===!0&&(c.only=!0),i.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=V(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=V(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=V(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=V(t.afterEach));let o=Fe.safeParse(n);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function V(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Qn)}function Qn(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 to(t);if("STEP"in t)return ro(t);if("VERIFY"in t){let r=t.VERIFY,s={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(s.code=t.js),{uid:q(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:s}}}}if("URL"in t){let r=t.URL,s=t.new_tab===!0?!0:void 0,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return s&&(o.new_tab=!0),n!==void 0&&(o.timeout_seconds=n),{uid:q(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:q(),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:s}}}}}if("WAIT"in t){let r=t.WAIT,s=typeof t.seconds=="number"?t.seconds:3;return{uid:q(),type:"ACTION",description:typeof r=="string"?r:`Wait ${s}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${s}s`,action_data:{action_name:"wait",kwargs:{seconds:s}}}}}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.');let s=typeof t.description=="string"?t.description:"Code block";return{uid:q(),type:"ACTION",description:s,action_entity:{action_description:s,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",s=t.js;return{uid:q(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof s=="string"?s:String(s)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...s}=t;return $r({...s,action:"function",functionName:r})}if("action"in t)return $r(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:q(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Cr(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=Cr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let s={uid:q(),type:"IF_ELSE",condition:t,then:V(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(s.else=V(e.ELSE)),s}function to(e){let t=Cr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let s={uid:q(),type:"WHILE_LOOP",condition:t,body:V(r)};return typeof e.timeout_ms=="number"&&(s.timeout_ms=e.timeout_ms),s}function ro(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:q(),type:"STEP",description:t,statements:V(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let s=e.template_params,n={};for(let[o,a]of Object.entries(s))n[o]=String(a);r.template_params=n}return r}function $r(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:"",s=typeof e.locator=="string"?e.locator:void 0,n=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[l,u]of Object.entries(e))so.has(l)||(a[l]=u);t==="verify"&&typeof a.js=="string"&&(a.code=a.js,delete a.js),(t==="ai_action"||t==="ai_step")&&a.statement===void 0&&(a.statement=r);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};s&&(i.locator=s),n&&(i.xpath=n);let c={uid:q(),type:"ACTION",description:r,action_entity:i};return o&&(c.use_pure_vision=!0),c}function Be(e,t){let r;try{r=Vn(e)}catch{return}let s=r.contents;if(!s||!ve(s))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=s.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let n=s,o=n.get("statements",!0);H(o)&&t.statements&&be(o,t.statements);let a=n.get("teardown",!0);H(a)&&t.teardown&&be(a,t.teardown)}function be(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 s=e.items[r];s.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=s.commentBefore);let n=t[r];if(n.type==="STEP"&&ve(s)){let o=s.get("statements",!0);H(o)&&be(o,n.statements)}else if(n.type==="IF_ELSE"&&ve(s)){let o=s.get("THEN",!0);H(o)&&be(o,n.then);let a=s.get("ELSE",!0);H(a)&&n.else&&be(a,n.else)}else if(n.type==="WHILE_LOOP"&&ve(s)){let o=s.get("DO",!0);H(o)&&be(o,n.body)}}}var Or,Ar,so,He=_(()=>{"use strict";mt();Or={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Ar=1024*1024;so=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as Ea,stringify as Aa}from"yaml";var Nr=_(()=>{"use strict";He()});var bt,We,Ge=_(()=>{"use strict";bt=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},We=e=>{let t=[],r=s=>{for(let n of s){t.push(n);let o=bt(n);for(let a of o)r(a.statements)}};return r(e),t}});function Br(e){let t=0,r=0;for(let s of e)if(s.type==="DRAFT")r++;else if(s.type==="ACTION"){let n=s.action_entity?.action_data?.action_name??"";Ur.has(n)||t++}return{action:t,draft:r}}function oo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Fr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function io(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,s=t.join(`
209
+ `).match(/\bawait\b/g),n=s?s.length:0;return r<=Dr&&n<=jr?null:`${r} non-blank line(s), ${n} await(s) \u2014 limits are ${Dr} lines and ${jr} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function vt(e,t){let r=t?.coverageThreshold??no,s=[],n=[],o;try{o=W(e)}catch(f){return{valid:!1,errors:[`Invalid YAML: ${f.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||s.push('Missing required field: "goal"'),o.statements?.length||s.push('Missing required field: "statements"');let a=[...We(o.statements??[]),...o.teardown?We(o.teardown):[]],{action:i,draft:c}=Br(a),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let f of a){let g=f;if(g.reference_id!==void 0){let h=g.description||f.uid;s.push(`Unresolved cloud template reference on statement "${h}" (reference_id: ${g.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(f.type==="ACTION"){let h=f,p=h.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let w=h.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let y=oo(w);if(y){let m=h.description||p;s.push(`Invalid JS in "${m}": ${y}. ${l}`)}else if(p==="verify"){let m=io(w);if(m){let b=h.description||p;s.push(`JS cache for "${b}" is too complex: ${m}`)}}}}}if(f.type==="IF_ELSE"){let h=f;if(h.condition.type==="JS_CODE"){let p=Fr(h.condition.expression);p&&s.push(`Invalid JS in IF condition "${h.condition.expression}": ${p}. ${l}`)}}if(f.type==="WHILE_LOOP"){let h=f;if(h.condition.type==="JS_CODE"){let p=Fr(h.condition.expression);p&&s.push(`Invalid JS in WHILE condition "${h.condition.expression}": ${p}. ${l}`)}}}let u=i+c,d=u>0?Math.round(i/u*100):0;return u>0&&d/100<r&&n.push(`Low action coverage: ${i}/${u} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:s.length===0,errors:s,warnings:n,stats:{total:u,action:i,draft:c,coverage:d}}}var no,Dr,jr,Ur,Hr=_(()=>{"use strict";He();Ge();no=.5,Dr=5,jr=3,Ur=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Ke=_(()=>{"use strict"});var Wr=_(()=>{"use strict";Ke()});var Gr,co,Kr,Vr,Ie,lo,po,zr=_(()=>{"use strict";Gr=112,co=1080-Gr,Kr={"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"}},Vr={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"]},Ie=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Vr[e].map(s=>Kr[s]).filter(s=>s.defaultBrowserType&&r.includes(s.defaultBrowserType))},lo={desktop:{label:"Desktop",type:"desktop",devices:Ie("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Ie("mobile")}},po={desktop:{label:"Desktop",type:"desktop",devices:Ie("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Ie("mobile",!0)}}});var Yr=_(()=>{"use strict"});function St(){return{version:"1.0",entries:{}}}var Jr=_(()=>{"use strict";Ge()});var Z,_t,Xr=_(()=>{"use strict";Z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(Z||{}),_t=18e4});var qr=_(()=>{"use strict"});var Zr=_(()=>{"use strict"});var Qr=_(()=>{"use strict"});var es=_(()=>{"use strict";Ke()});var de=_(()=>{"use strict";yr();wr();br();vr();Nr();Hr();He();mt();Wr();zr();Yr();Jr();Ge();Xr();qr();Zr();Qr();es();Ke()});import{stringify as fo}from"yaml";import{createHash as Mo}from"crypto";import{parse as Oo,stringify as ls}from"yaml";import{readFileSync as Lo,existsSync as Ro}from"fs";import{resolve as Tt,dirname as Co}from"path";import{parse as ns,stringify as No}from"yaml";import{readFileSync as Wo,writeFileSync as Go,mkdirSync as Ko}from"fs";import{dirname as Vo}from"path";function re(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function O(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function ho(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function go(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 ze(e){let t=ho(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let s=go(e);if(s){let n=JSON.stringify(s);return`${t}.locator(${n}).first()`}return null}function ts(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:mo.includes(t)}function wo(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!yo.includes(t)}function S(e,t){D.set(e,t)}function bo(e){return D.get(e)}function ge(e,t,r=[]){let s=[...r];return t.locator?s.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&s.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&s.push(`frame_path: ${JSON.stringify(t.frame_path)}`),s.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...s.map(n=>` ${n},`),"});"]}function rs(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 s=["page","testContext","request","agent"],n=["undefined","null","true","false"],o=r.map(a=>s.includes(a)||n.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${o.join(", ")})`}function se(e,t,r,s="main"){let n=[];for(let o=0;o<e.length;o++){let a=e[o],i=`${s}.${o}`,c=vo(a,t,i,r);c.length>0&&(n.push(...c),o<e.length-1&&n.push(""))}return n}function vo(e,t,r,s){let n=" ".repeat(t);switch(e.type){case"DRAFT":return So(e,t,r,s);case"ACTION":return _o(e,t,r,s);case"STEP":return xo(e,t,r,s);case"IF_ELSE":return To(e,t,r,s);case"WHILE_LOOP":return ko(e,t,r,s);default:return[`${n}// Unknown statement type: ${e.type}`]}}function So(e,t,r,s){let n=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let a=JSON.stringify(o);return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${a}, '${r}');`]}function _o(e,t,r,s){let n=" ".repeat(t),o=e.description,a=e.uid,c=s.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let b=JSON.stringify(o),E=!!e.use_pure_vision;return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${b}, '${r}', ${E});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let u=l.action_data?.action_name||"",d=l.action_description||"",f=bo(u);if(!f)return[`${n}// ${r}: Unknown action: ${u}`];let g={imports:s.imports},h=f(l,r,g);if(s.noAgent){if(ts(l))return[`${n}// ${r}: ${O(d)}`,`${n}// AI action: ${O(d)} (requires agent - skipped in hook)`];let b=Po(l,u,n,r);return b||[`${n}// ${r}: ${O(d)}`,...h.map(E=>`${n}${E}`)]}if(ts(l))return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,...h.map(b=>`${n}${b}`)];let p=JSON.stringify(d),w=h.map(b=>`${n} ${b}`),y=wo(l),m=a?`'${a}'`:"undefined";return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...w,`${n}}, ${p}, '${r}', ${m}, ${y});`]}function xo(e,t,r,s){let n=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${n}// Step: ${O(e.description)}`);let a=se(e.statements,t,s,r);return o.push(...a),o}function To(e,t,r,s){let n=" ".repeat(t),o=[];if(o.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${n}if (${e.condition.expression}) {`);else{o.push(`${n}// AI Condition: ${O(e.condition.expression)}`);let i=JSON.stringify(e.condition.expression);o.push(`${n}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=se(e.then,t+1,s,`${r}.then`);if(o.push(...a),e.else&&e.else.length>0){o.push(`${n}} else {`);let i=se(e.else,t+1,s,`${r}.else`);o.push(...i)}return o.push(`${n}}`),o}function ko(e,t,r,s){let n=" ".repeat(t),o=[];o.push(`${n}// ${r}: Loop`);let a=e.timeout_ms??_t,i=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${n}const ${l}_start = Date.now();`),o.push(`${n}const ${l}_timeout = ${a};`),o.push(`${n}const ${l}_check = () => {`),o.push(`${n} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${n} throw new Error('${c}');`),o.push(`${n} }`),o.push(`${n} return true;`),o.push(`${n}};`),e.condition.type==="JS_CODE")o.push(`${n}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${n}// AI Loop Condition: ${O(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${n}while (${l}_check() && await agent.evaluate(page, ${d}, "${r}")) {`)}let u=se(e.body,t+1,s,`${r}.body`);return o.push(...u),o.push(`${n}}`),o}function Po(e,t,r,s){let n=e.action_description||"",o=e.action_data?.kwargs||{},a=o.timeout_ms??kt;switch(t){case"go_to_url":case"open_tab":{let i=o.url||"";return[`${r}// ${s}: ${O(n)}`,`${r}await page.goto(${JSON.stringify(i)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goForward();`];case"input_text":{let i=o.text||"",c=ze(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=o.text||o.label||"",c=ze(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function Eo(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...as()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...ss("beforeEach",t.beforeEach,o)),r.push(""));let a=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 i=t?.testName||e.goal||"Generated test",c=xt(t?.tags);for(let l of t.parameters)r.push(...Je(e,`${c}${re(i)} [${re(l.name)}]`,o,0,a,l.name,l.values)),r.push("")}else{let i=t?.testName||e.goal||"Generated test",c=xt(t?.tags);r.push(...Je(e,`${c}${re(i)}`,o,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...ss("afterEach",t.afterEach,o))),cs(r,n),r.join(`
210
+ `)}function Ao(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...as()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},a=t?.testName||"Test Suite",i=xt(t?.tags);r.push(`test.describe.serial('${i}${re(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ve("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ve("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let u=e.tests[l],d=u.timeout||u.skip!==void 0||u.fail!==void 0||u.only||u.slow?{timeout:u.timeout,skip:u.skip,fail:u.fail,only:u.only,slow:u.slow}:void 0;if(u.parameters&&u.parameters.length>0)for(let f of u.parameters)r.push(...Je(u.testFlow,`${re(u.name)} [${re(f.name)}]`,o,1,d,f.name,f.values)),r.push("");else r.push(...Je(u.testFlow,re(u.name),o,1,d)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ve("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ve("afterAll",e.afterAll,o,1)),r.push("});"),cs(r,n),r.join(`
211
+ `)}function xt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Ye(e){let t=new Set;function r(s){for(let n of s)switch(n.type){case Z.ACTION:{let a=n.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&$o.includes(i)&&t.add(i);break}case Z.STEP:r(n.statements);break;case Z.IF_ELSE:{let o=n;r(o.then),o.else&&r(o.else);break}case Z.WHILE_LOOP:r(n.body);break}}return r(e),t}function Io(e){let t=Ye(e.statements??[]);if(e.teardown)for(let r of Ye(e.teardown))t.add(r);return t}function Pt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Je(e,t,r,s=0,n,o,a){let i=" ".repeat(s),c=[],l=Io(e),u=Pt(l),d=n?.only?"test.only":"test",f=o?`, { tag: '@${re(o)}' }`:"";if(c.push(`${i}${d}('${t}'${f}, async (${u}) => {`),n?.skip===!0?c.push(`${i} test.skip();`):typeof n?.skip=="string"&&c.push(`${i} test.skip(true, '${re(n.skip)}');`),n?.fail===!0?c.push(`${i} test.fail();`):typeof n?.fail=="string"&&c.push(`${i} test.fail(true, '${re(n.fail)}');`),n?.slow&&c.push(`${i} test.slow();`),n?.timeout&&c.push(`${i} test.setTimeout(${n.timeout});`),a){for(let[p,w]of Object.entries(a))c.push(`${i} agent.agentServices.saveVariable(${JSON.stringify(p)}, ${JSON.stringify(w)});`);c.push("")}let g=e.teardown&&e.teardown.length>0,h=s+1;if(g){if(c.push(`${i} try {`),e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let w=se(e.statements,h+1,r);c.push(...w)}c.push(`${i} } finally {`),c.push(`${i} // Teardown`);let p=se(e.teardown,h+1,r,"teardown");c.push(...p),c.push(`${i} }`)}else if(e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let p=se(e.statements,h,r);c.push(...p)}return c.push(`${i}});`),c}function ss(e,t,r){let s=[],n=is(t),o=Ye(n),a=Pt(o);return s.push(`test.${e}(async (${a}) => {`),s.push(...se(n,1,r,e)),s.push("});"),s}function Ve(e,t,r,s){let n=" ".repeat(s),o=[],a=is(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...se(a,s+1,c,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let c=Ye(a),l=Pt(c);o.push(`${n}test.${e}(async (${l}) => {`),o.push(...se(a,s+1,r,e)),o.push(`${n}});`)}return o}function is(e){let r=fo({goal:"_hook",statements:e});return W(r).statements??[]}function as(){return["import { test, expect } from 'shiplightai/fixture';"]}function cs(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let s=Array.from(t);e.splice(r,0,...s)}}function ps(e,t,r){let s={expandingPaths:new Set([Tt(t)]),depth:0,referencedPaths:new Set,basePath:r},n={...e};Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,s)),Array.isArray(n.teardown)&&(n.teardown=fe(n.teardown,t,s));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=fe(n[o],t,s));return{doc:n,referencedTemplatePaths:Array.from(s.referencedPaths)}}function fe(e,t,r){let s=[];for(let n of e)if(Do(n)){let o=jo(n,t,r);s.push(o)}else s.push(Fo(n,t,r));return s}function Do(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function jo(e,t,r){if(r.depth>=os)throw new Error(`Template expansion exceeded maximum depth of ${os}. Check for deeply nested or circular template references.`);let s=Tt(Co(t),e.template),n=!Ro(s)&&r.basePath?Tt(r.basePath,e.template):s;if(r.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);r.referencedPaths.add(n);let o;try{o=Lo(n,"utf-8")}catch(h){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${h.message}`)}let a=ns(o);if(!a||typeof a!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let i=a.params||[],c=e.params||{};for(let h of i)if(!(h in c))throw new Error(`Template ${e.template} requires param "${h}" but it was not provided. Required params: [${i.join(", ")}]`);let l=a.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let p=No(l);for(let[w,y]of Object.entries(c))p=p.split(`<<${w}>>`).join(String(y));l=ns(p)}let u={expandingPaths:new Set([...r.expandingPaths,n]),depth:r.depth+1,referencedPaths:r.referencedPaths},d=fe(l,n,u),g={STEP:a.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(c).length>0&&(g.template_params=c),g}function Fo(e,t,r){if(typeof e!="object"||e===null)return e;let s={...e};return Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,r)),Array.isArray(s.THEN)&&(s.THEN=fe(s.THEN,t,r)),Array.isArray(s.ELSE)&&(s.ELSE=fe(s.ELSE,t,r)),Array.isArray(s.DO)&&(s.DO=fe(s.DO,t,r)),s}function At(e,t,r){let s=Oo(e),n=s?.name,o=s?.tags,a=s?.use;if(s&&(s.name!==void 0||s.tags!==void 0||s.use!==void 0)&&(delete s.name,delete s.tags,delete s.use),s?.suite){if(s.goal||s.statements)throw new Et('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Bo(s,n,o,a,t,r)}return Uo(s,n,o,a,t,r)}function Uo(e,t,r,s,n,o){let a=e?.beforeEach,i=e?.afterEach,c=us(e?.parameters),l=e?.timeout,u=e?.skip,d=e?.fail,f=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 Et(`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 h=[];if(n&&e&&typeof e=="object"){let y=ps(e,n,o);e=y.doc,h=y.referencedTemplatePaths}let p=ls(e),w=W(p);return n&&(Se(w.statements??[],n,"main"),w.teardown&&Se(w.teardown,n,"teardown")),{testFlow:w,name:t,tags:r,use:s,beforeEach:a,afterEach:i,parameters:c,timeout:l,skip:u,fail:d,only:f,slow:g,referencedTemplatePaths:h}}function Bo(e,t,r,s,n,o){let a=e.suite;if(!Array.isArray(a.tests)||a.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let i=a.beforeAll,c=a.afterAll,l=a.beforeEach,u=a.afterEach,d=[],f=a.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let w={goal:p.name,statements:p.statements};p.teardown&&(w.teardown=p.teardown);let y=[],m=w;if(n&&typeof w=="object"){let $=ps(w,n,o);m=$.doc,y=$.referencedTemplatePaths,d.push(...y)}let b=ls(m),E=W(b),T=us(p.parameters);return{testFlow:E,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:T,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=a.base_url,h=g?{...s,baseURL:g}:s;return{suite:{beforeAll:i,afterAll:c,beforeEach:l,afterEach:u,tests:f},name:t,tags:r,use:h,referencedTemplatePaths:d}}function us(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 Se(e,t,r){for(let s=0;s<e.length;s++){let n=e[s],o=`${r}.${s}`,a=n.description||"";if(n.uid=Ho(t,o,a),n.type===Z.STEP)Se(n.statements,t,o);else if(n.type===Z.IF_ELSE){let i=n;Se(i.then,t,`${o}.then`),i.else&&Se(i.else,t,`${o}.else`)}else n.type===Z.WHILE_LOOP&&Se(n.body,t,`${o}.body`)}}function Ho(e,t,r){let s=Mo("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${s.slice(0,8)}-${s.slice(8,12)}-${s.slice(12,16)}-${s.slice(16,20)}-${s.slice(20,32)}`}function ds(e,t){let r;try{r=Wo(e,"utf-8")}catch(s){return{valid:!1,errors:[`Failed to read file: ${s.message}`],warnings:[]}}return zo(r,e,t)}function zo(e,t,r){let s=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=s||n?null:vt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let a,i,c=[];try{let l=r?.parsed??At(e,t);c=l.referencedTemplatePaths;let u={version:r?.version,actionEntityStore:r?.actionEntityStore},d=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?a=Ao(l.suite,{...u,testName:l.name,tags:l.tags,use:l.use}):a=Eo(l.testFlow,{...u,testName:l.name,tags:l.tags,use:d,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 f=a.split(`
212
212
  `).filter(g=>!g.startsWith("import ")).join(`
213
- `);new Function(f),i=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Go(Wo(i),{recursive:!0}),Ho(i,a)}catch(c){let u=c instanceof Et?"":c.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: ${c.message}.${u}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:i,referencedTemplatePaths:l}}var Pt,ho,go,N,Eo,ss,Et,$t=_(()=>{"use strict";de();de();de();de();Pt=5e3;ho=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],go=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];N=new Map;v("click",e=>{let t=Ye(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.click({ timeout: ${r} });`]});v("click_element",N.get("click"));v("click_element_by_index",N.get("click"));v("double_click",e=>ge("double_click",e));v("double_click_on_element",N.get("double_click"));v("right_click",e=>ge("right_click",e));v("right_click_on_element",N.get("right_click"));v("hover",e=>ge("hover",e));v("hover_element_by_index",N.get("hover"));v("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("fill",N.get("input_text"));v("clear_input",e=>ge("clear_input",e));v("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});v("send_keys",N.get("press"));v("send_keys_on_element",e=>{let t=Ye(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 s=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${s} });`]});v("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("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)})');`]});v("scroll_down",N.get("scroll"));v("scroll_up",N.get("scroll"));v("scroll_element",N.get("scroll"));v("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});v("scroll_on_element",e=>ge("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));v("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)} } },`,"});"]});v("open_tab",N.get("go_to_url"));v("go_back",()=>['await agent.execAction("go_back", page, {});']);v("reload_page",()=>['await agent.execAction("reload_page", page, {});']);v("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);v("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);v("verify",(e,t)=>{let r=e.action_data?.kwargs,s=typeof r?.code=="string",n=s?r?.statement||e.action_description:e.action_description||r?.statement;if(s&&n){let a=r.code.split(`
214
- `),i=JSON.stringify(n);return["{ const _t = Date.now(); try {",...a.map(l=>` ${l}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${i}\`);`,"} 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: ${i}\`);`,` await agent.assert(page, ${i}, ${JSON.stringify(t||"")});`,"} }"]}return s?r.code.split(`
215
- `):n?[`await agent.assert(page, ${JSON.stringify(n)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});v("ai_assert",N.get("verify"));v("assert",N.get("verify"));v("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let s=JSON.stringify(r),n=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${s}, '${t||""}', ${n});`]});v("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"]});v("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,s=e.action_data?.kwargs?.variable_name;if(!r||!s)return["// Skipping ai_extract: missing element_description or variable_name"];let n=JSON.stringify(r),o=JSON.stringify(s);return[`await agent.extract(page, ${n}, ${o}, '${t||""}');`]});v("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,s=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${s}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});v("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)} } },`,"});"]});v("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],s=t.split(`
216
- `);for(let n of s)r.push(` ${n}`);return r.push("}"),r});v("function",(e,t,r)=>{let s=e.action_data?.kwargs||{},n=s.functionName;if(n&&n.includes("#")){let[a,i]=n.split("#");if(a&&i){let l=a.replace(/\.(ts|js|mjs)$/,""),c=`import { ${i} } from '${l}';`;r?.imports?.add(c);let u={...s,functionName:i},d=es(u);return d?[d.endsWith(";")?d:`${d};`]:["// Skipping function: invalid export pattern"]}}let o=es(s);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});v("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)} } },`,"});"]});v("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],s={};return t.paths?s.paths=t.paths:t.path&&(s.path=t.path),t.use_file_input&&(s.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(s)} }`),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(n=>` ${n},`),"});"]});v("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} } },`,"});"]);v("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} } },`,"});"]);v("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} } },`,"});"]});v("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(s=>` ${s},`),"});"]});v("done",()=>["// Done - no action needed"]);v("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});v("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
217
- `):["// Skipping js_action: missing code"]});Eo=["testContext","request"];ss=5;Et=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as Vo}from"express";import*as k from"fs/promises";import*as x from"path";import{stringify as qe,parse as Oe}from"yaml";function me(e){if(!e||e.length===0)return[];let r=qe({goal:"_hook",statements:e});return j(r).statements??[]}async function ds(e){try{let t=await k.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 ds(x.join(e,r.name))))return!0}catch{}return!1}function Ze(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function zo(e){let t=new Map,r;try{r=await k.readdir(e)}catch(s){if(s.code==="ENOENT")return t;throw s}for(let s of r.filter(n=>n.endsWith(".yaml")))try{let n=x.join(e,s),a=Oe(await k.readFile(n,"utf-8"))?.name??x.basename(s,".yaml");t.set(Ze(a),`templates/${s}`)}catch{}return t}function Mt(e,t){if(Array.isArray(e))return e.map(o=>Mt(o,t));if(!e||typeof e!="object")return e;let r=e,s={};for(let[o,a]of Object.entries(r))s[o]=Mt(a,t);let n=typeof r.reference_id=="number"?r.reference_id:void 0;if(n!==void 0&&typeof s.template_path!="string"){let o=t.get(n);o&&(s.template_path=o,delete s.reference_id)}return s}function It(e){return Be({statements:e}).statements}function fs(e){let{initialDir:t,initialFile:r,projectRoot:s,onFileSelected:n}=e,o=Vo();function a(){return x.join(s??t,"fixtures")}o.get("/api/files",async(l,c)=>{try{let u=typeof l.query.dir=="string"?l.query.dir:t,d=x.resolve(u),f=await k.readdir(d,{withFileTypes:!0}),g=[];for(let p of f)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let b=x.join(d,p.name);await ds(b)&&g.push({name:p.name,type:"directory",path:b})}else p.isFile()&&p.name.endsWith(".test.yaml")&&g.push({name:p.name,type:"file",path:x.join(d,p.name)});g.sort((p,b)=>p.type!==b.type?p.type==="directory"?-1:1:p.name.localeCompare(b.name));let h=x.dirname(d);c.json({dir:d,parent:h!==d?h:null,entries:g,initialFile:r??null,projectRoot:s??t})}catch(u){console.error("[debugger] Error listing files:",u),c.status(500).json({error:u.message})}});function i(l){if(typeof l=="string"&&l){let c=x.resolve(l);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 o.get("/api/test-flow",async(l,c)=>{let u=i(l.query.file);if("error"in u)return c.status(400).json({error:u.error});let d=u.filePath;try{n?.(d);let f=await k.readFile(d,"utf-8"),g=await k.stat(d),h=Ie(f),p=At(f,d,s);if(p.suite){let b=p.suite,y={tests:b.tests.map(w=>({name:w.name,statements:w.testFlow.statements??[],teardown:w.testFlow.teardown,skip:w.skip,timeout:w.timeout,fail:w.fail,only:w.only,slow:w.slow})),beforeAll:me(b.beforeAll),afterAll:me(b.afterAll),beforeEach:me(b.beforeEach),afterEach:me(b.afterEach)},m={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:y};c.json({isSuite:!0,testFlow:m,metadata:h,name:p.name,tags:p.tags,use:p.use,filePath:d,fileName:x.basename(d),lastModified:g.mtimeMs})}else{let b=p.testFlow;He(f,b),c.json({isSuite:!1,testFlow:b,metadata:h,name:p.name,tags:p.tags,use:p.use,filePath:d,fileName:x.basename(d),lastModified:g.mtimeMs})}}catch(f){if(f.code==="ENOENT")return c.status(404).json({error:`File not found: ${d}`});console.error("[debugger] Error loading test flow:",f),c.status(500).json({error:f.message})}}),o.put("/api/test-flow",async(l,c)=>{try{let u=i(l.query.file);if("error"in u)return c.status(400).json({error:u.error});let d=u.filePath,{testFlow:f,metadata:g}=l.body;if(!f)return c.status(400).json({error:"testFlow is required"});let h=x.join(s??t,"templates"),p=await zo(h),b=Mt(f,p),y=$e(b,g),m=d+".tmp";await k.writeFile(m,y,"utf-8"),await k.rename(m,d);let w=await k.stat(d);c.json({success:!0,lastModified:w.mtimeMs})}catch(u){console.error("[debugger] Error saving test flow:",u),c.status(500).json({error:u.message})}}),o.get("/api/fixtures",async(l,c)=>{try{let u=a();try{let f=(await k.readdir(u,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();c.json({files:f,dir:u})}catch(d){if(d.code==="ENOENT")c.json({files:[],dir:u});else throw d}}catch(u){console.error("[debugger] Error listing fixtures:",u),c.status(500).json({error:u.message})}}),o.post("/api/fixtures",async(l,c)=>{try{let u=a();await k.mkdir(u,{recursive:!0});let{name:d,content:f}=l.body;if(!d||!f)return c.status(400).json({error:"name and content are required"});let g=x.basename(d);if(!g)return c.status(400).json({error:"Invalid file name"});let h=x.join(u,g);await k.writeFile(h,Buffer.from(f,"base64")),c.json({fileName:g})}catch(u){console.error("[debugger] Error saving fixture:",u),c.status(500).json({error:u.message})}}),o.get("/api/functions",async(l,c)=>{let u=x.join(s??t,"helpers"),d;try{d=await k.readdir(u)}catch(p){return p.code==="ENOENT"?c.json([]):(console.error("[debugger] Error reading helpers dir:",p),c.status(500).json({error:p.message}))}let f=d.filter(p=>/\.(ts|js|mjs)$/.test(p)),g=[],h=1;for(let p of f){let b=await k.readFile(x.join(u,p),"utf-8"),y=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,m;for(;(m=y.exec(b))!==null;){let[,w,E]=m,T=E.replace(/\s*:\s*[^,)]+/g,"");g.push({id:h++,name:`helpers/${p}#${w}`,description:"",status:"Active",code:`async function ${w}${T} {}`})}}c.json(g)}),o.get("/api/reusable-steps",async(l,c)=>{let u=x.join(s??t,"templates"),d;try{d=await k.readdir(u)}catch(h){return h.code==="ENOENT"?c.json([]):(console.error("[debugger] Error reading templates dir:",h),c.status(500).json({error:h.message}))}let f=d.filter(h=>h.endsWith(".yaml")).sort(),g=[];for(let h of f)try{let p=await k.readFile(x.join(u,h),"utf-8"),b=Oe(p);if(!b||typeof b!="object")continue;let y=b.name??x.basename(h,".yaml"),m=b.description??"",w=me(b.statements);g.push({id:Ze(y),organizationId:"local",name:y,description:m,statements:w})}catch(p){console.error(`[debugger] Error parsing template ${h}:`,p)}c.json(g)}),o.post("/api/reusable-steps/exists/:name",async(l,c)=>{let u=x.join(s??t,"templates"),d=decodeURIComponent(l.params.name).toLowerCase(),f;try{f=await k.readdir(u)}catch(g){return g.code==="ENOENT"?c.json(!1):c.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=await k.readFile(x.join(u,g),"utf-8");if((Oe(h)?.name??x.basename(g,".yaml")).toLowerCase()===d)return c.json(!0)}catch{}c.json(!1)}),o.post("/api/reusable-steps",async(l,c)=>{let u=x.join(s??t,"templates"),{reusableStep:d}=l.body??{};if(!d?.name||!Array.isArray(d.statements))return c.status(400).json({error:"reusableStep.name and reusableStep.statements are required"});await k.mkdir(u,{recursive:!0});let f=d.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=x.join(u,`${f}.yaml`),h={name:d.name,statements:It(d.statements)};d.description&&(h.description=d.description),await k.writeFile(g,qe(h),"utf-8"),c.json({id:Ze(d.name),organizationId:"local",name:d.name,description:d.description??"",statements:d.statements})}),o.put("/api/reusable-steps/:id/update",async(l,c)=>{let u=parseInt(l.params.id,10),d=x.join(s??t,"templates"),f;try{f=await k.readdir(d)}catch(g){return g.code==="ENOENT"?c.status(404).json({error:"Templates directory not found"}):c.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=x.join(d,g),p=Oe(await k.readFile(h,"utf-8"))??{},b=p.name??x.basename(g,".yaml");if(Ze(b)!==u)continue;let y=l.body??{};return y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=It(y.statements)),await k.writeFile(h,qe(p),"utf-8"),c.json({id:u,organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}c.status(404).json({error:`Template with id ${u} not found`})}),o.put("/api/templates/:name",async(l,c)=>{let u=x.join(s??t,"templates"),d=decodeURIComponent(l.params.name),f;try{f=await k.readdir(u)}catch(g){return g.code==="ENOENT"?c.status(404).json({error:"Templates directory not found"}):c.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=x.join(u,g),p=Oe(await k.readFile(h,"utf-8"))??{};if((p.name??x.basename(g,".yaml"))!==d)continue;let y=l.body??{};return y.name!==void 0&&(p.name=y.name),y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=It(y.statements)),await k.writeFile(h,qe(p),"utf-8"),c.json({organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}c.status(404).json({error:`Template "${d}" not found`})}),o.get("/api/test-results",async(l,c)=>{let u=i(l.query.file);if("error"in u)return c.status(400).json({error:u.error});let d=s??t,f=x.basename(u.filePath,".test.yaml"),g=x.join(d,".shiplight","artifacts",f),h=[],p=null;try{let b=await k.readdir(g,{withFileTypes:!0});for(let y of b)if(y.isDirectory()){let m=x.join(g,y.name),w=await k.readdir(m);for(let E of w.filter(T=>/\.(png|jpe?g|webp)$/i.test(T))){let T=x.relative(g,x.join(m,E)),$=y.name.match(/^(.+?)_(before|after)$/),Y=$?$[1]:y.name,L=$?$[2]:"screenshot";h.push({url:`/api/report-assets/${f}/${T}`,label:L,stepId:Y})}}else/\.(webm|mp4)$/i.test(y.name)&&(p||(p=`/api/report-assets/${f}/${y.name}`));h.sort((y,m)=>y.stepId!==m.stepId?(y.stepId??"").localeCompare(m.stepId??""):y.label.localeCompare(m.label))}catch{}if(h.length===0&&!p)return c.status(404).json({error:"No test results found"});c.json({videoPath:p,screenshots:h})}),o}var hs=_(()=>{"use strict";de();$t()});import*as ws from"http";import*as bs from"net";import*as ne from"path";import*as Ss from"fs";import{randomUUID as Yo}from"crypto";function ms(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let s=[];e.on("data",n=>s.push(n)),e.on("end",()=>t(Buffer.concat(s))),e.on("error",r)})}function ys(e,t,r,s,n,o){return new Promise(a=>{let i=Array.isArray(o)?o[0]:o,l=ws.request({hostname:r,port:s,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(n.length)},timeout:3e5},c=>{t.writeHead(c.statusCode??502,c.headers),c.on("data",u=>t.write(u)),c.on("end",()=>{t.end(),a()}),c.on("error",()=>{t.writableEnded||t.end(),a()})});l.on("error",c=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+c.message}),a()}),l.end(n)})}function Ot(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var gs,Qe,vs=_(()=>{"use strict";ht();gs=1e4,Qe=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??ft,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}openSession(t){let r=ne.resolve(t),s=this.byYamlPath.get(r);if(s){let c=this.sessions.get(s);if(c&&c.session.status!=="ended")return{...c.session};this.byYamlPath.delete(r)}if(!Ss.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!Ee(ne.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Yo().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"idle",exitInfo:null},l={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,l),this.byYamlPath.set(r,o),this.notifyStateChange(i),this.log(`[manager] session ${o} created (idle) for ${ne.basename(r)}`),{...i}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let s=r.session.yamlPath,n=Ee(ne.dirname(s));if(!n)throw new Error(`No Playwright config found for ${s} (searched parents for playwright.config.{ts,js,mjs}).`);r.session.status="starting",this.notifyStateChange(r.session);let o=(async()=>{let a=gs+18e4,i,l=new Promise((d,f)=>{i=setTimeout(()=>f(new Error(`Timed out after ${a/1e3}s waiting for inner playwright to register on a port.`)),a)}),c;try{c=await Promise.race([this.spawner({yamlFilePath:s,configPath:n,tempSuffix:t,headed:this.headed}),l])}finally{i&&clearTimeout(i)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup;let u=d=>{r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo=d,this.notifyStateChange(r.session))};r.livenessTimer=this.startLivenessProbe(t,u),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ne.basename(s)})`)})();r.readyPromise=o;try{await o}catch(a){throw r.session.status="ended",r.session.exitInfo=a.message,this.notifyStateChange(r.session),a}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(s){this.log(`[manager] stopSandbox ${t} cleanup error: ${s.message}`)}r.session.innerPort=0,r.session.innerHost="127.0.0.1",r.session.pid=0,r.session.status="idle",r.session.exitInfo=null,r.cleanup=async()=>{},r.readyPromise=Promise.resolve(),this.notifyStateChange(r.session)}}startLivenessProbe(t,r){let o=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"){clearInterval(a);return}let l=!1;try{process.kill(i.session.pid,0),l=!0}catch(c){c?.code!=="ESRCH"&&(l=!0)}l?o=0:(o+=1,o>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(s){this.log(`[manager] closeSession ${t} cleanup error: ${s.message}`)}}}async restartInner(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to restart`);if(r.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let s=r.session.yamlPath,n=Ee(ne.dirname(s));if(!n)throw new Error(`No Playwright config found for ${s} (searched parents for playwright.config.{ts,js,mjs}).`);r.restartInProgress=!0;try{r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(c){this.log(`[manager] restartInner ${t} old cleanup error: ${c.message}`)}r.session.status="starting",r.session.innerPort=0,r.session.pid=0,this.notifyStateChange(r.session);let o=gs+18e4,a,i=new Promise((c,u)=>{a=setTimeout(()=>u(new Error(`Timed out after ${o/1e3}s waiting for inner playwright to respawn.`)),o)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:s,configPath:n,tempSuffix:t,headed:this.headed}),i])}catch(c){throw r.session.status="ended",r.session.exitInfo=c.message,this.notifyStateChange(r.session),c}finally{a&&clearTimeout(a)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup,r.livenessTimer=this.startLivenessProbe(t,c=>{r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo=c,this.notifyStateChange(r.session))}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} restarted on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ne.basename(s)})`)}finally{r.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:s}=r;return async(n,o,a)=>{let i=this.sessions.get(t);if(!i){o.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}if(i.session.status==="idle"){o.status(503).json({status:"error",message:"Sandbox not started"});return}let l=`${n.method} ${n.path}`;if(l==="POST /api/int-runner/create-session"){let u=await ms(n),d={};if(u.length)try{d=JSON.parse(u.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}d.testFilePath=i.session.yamlPath,await ys(n,o,i.session.innerHost,i.session.innerPort,Buffer.from(JSON.stringify(d),"utf-8"),"application/json");return}if(l==="POST /api/int-runner/terminate-session"){try{await this.stopSandbox(t),o.json({status:"success",details:"Sandbox stopped"})}catch(u){o.status(500).json({status:"error",message:u.message})}return}if(l==="POST /api/int-runner/liveview-url"){let u=s?.(n)??"";o.json({liveviewUrl:u,browserWsUrl:""});return}let c=await ms(n);await ys(n,o,i.session.innerHost,i.session.innerPort,c,n.headers["content-type"])}}wsUpgradeFor(t){return async(r,s,n,o)=>{let a=this.sessions.get(t);if(!a){Ot(s,`HTTP/1.1 404 Not Found\r
213
+ `);new Function(f),i=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Ko(Vo(i),{recursive:!0}),Go(i,a)}catch(l){let u=l instanceof Et?"":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}.${u}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:i,referencedTemplatePaths:c}}var kt,mo,yo,D,$o,os,Et,$t=_(()=>{"use strict";de();de();de();de();kt=5e3;mo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],yo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];D=new Map;S("click",e=>{let t=ze(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??kt;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",D.get("click"));S("click_element_by_index",D.get("click"));S("double_click",e=>ge("double_click",e));S("double_click_on_element",D.get("double_click"));S("right_click",e=>ge("right_click",e));S("right_click_on_element",D.get("right_click"));S("hover",e=>ge("hover",e));S("hover_element_by_index",D.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",D.get("input_text"));S("clear_input",e=>ge("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",D.get("press"));S("send_keys_on_element",e=>{let t=ze(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 s=e.action_data?.kwargs?.timeout_ms??kt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${s} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("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",D.get("scroll"));S("scroll_up",D.get("scroll"));S("scroll_element",D.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=>ge("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",D.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,s=typeof r?.code=="string",n=s?r?.statement||e.action_description:e.action_description||r?.statement;if(s&&n){let a=r.code.split(`
214
+ `),i=JSON.stringify(n);return["{ const _t = Date.now(); try {",...a.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${i}\`);`,"} 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: ${i}\`);`,` await agent.assert(page, ${i}, ${JSON.stringify(t||"")});`,"} }"]}return s?r.code.split(`
215
+ `):n?[`await agent.assert(page, ${JSON.stringify(n)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",D.get("verify"));S("assert",D.get("verify"));S("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let s=JSON.stringify(r),n=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${s}, '${t||""}', ${n});`]});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,s=e.action_data?.kwargs?.variable_name;if(!r||!s)return["// Skipping ai_extract: missing element_description or variable_name"];let n=JSON.stringify(r),o=JSON.stringify(s);return[`await agent.extract(page, ${n}, ${o}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,s=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${s}, '${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=["{"],s=t.split(`
216
+ `);for(let n of s)r.push(` ${n}`);return r.push("}"),r});S("function",(e,t,r)=>{let s=e.action_data?.kwargs||{},n=s.functionName;if(n&&n.includes("#")){let[a,i]=n.split("#");if(a&&i){let c=a.replace(/\.(ts|js|mjs)$/,""),l=`import { ${i} } from '${c}';`;r?.imports?.add(l);let u={...s,functionName:i},d=rs(u);return d?[d.endsWith(";")?d:`${d};`]:["// Skipping function: invalid export pattern"]}}let o=rs(s);return o?[o.endsWith(";")?o:`${o};`]:["// 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=[],s={};return t.paths?s.paths=t.paths:t.path&&(s.path=t.path),t.use_file_input&&(s.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(s)} }`),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(n=>` ${n},`),"});"]});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(s=>` ${s},`),"});"]});S("done",()=>["// Done - no action needed"]);S("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
217
+ `):["// Skipping js_action: missing code"]});$o=["testContext","request"];os=5;Et=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as Yo}from"express";import*as P from"fs/promises";import*as x from"path";import{stringify as Xe,parse as Me}from"yaml";function me(e){if(!e||e.length===0)return[];let r=Xe({goal:"_hook",statements:e});return W(r).statements??[]}async function fs(e){try{let t=await P.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 fs(x.join(e,r.name))))return!0}catch{}return!1}function qe(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function Jo(e){let t=new Map,r;try{r=await P.readdir(e)}catch(s){if(s.code==="ENOENT")return t;throw s}for(let s of r.filter(n=>n.endsWith(".yaml")))try{let n=x.join(e,s),a=Me(await P.readFile(n,"utf-8"))?.name??x.basename(s,".yaml");t.set(qe(a),`templates/${s}`)}catch{}return t}function Mt(e,t){if(Array.isArray(e))return e.map(o=>Mt(o,t));if(!e||typeof e!="object")return e;let r=e,s={};for(let[o,a]of Object.entries(r))s[o]=Mt(a,t);let n=typeof r.reference_id=="number"?r.reference_id:void 0;if(n!==void 0&&typeof s.template_path!="string"){let o=t.get(n);o&&(s.template_path=o,delete s.reference_id)}return s}function It(e){return Ue({statements:e}).statements}function hs(e){let{initialDir:t,initialFile:r,projectRoot:s,onFileSelected:n}=e,o=Yo();function a(){return x.join(s??t,"fixtures")}o.get("/api/files",async(c,l)=>{try{let u=typeof c.query.dir=="string"?c.query.dir:t,d=x.resolve(u),f=await P.readdir(d,{withFileTypes:!0}),g=[];for(let p of f)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let w=x.join(d,p.name);await fs(w)&&g.push({name:p.name,type:"directory",path:w})}else p.isFile()&&p.name.endsWith(".test.yaml")&&g.push({name:p.name,type:"file",path:x.join(d,p.name)});g.sort((p,w)=>p.type!==w.type?p.type==="directory"?-1:1:p.name.localeCompare(w.name));let h=x.dirname(d);l.json({dir:d,parent:h!==d?h:null,entries:g,initialFile:r??null,projectRoot:s??t})}catch(u){console.error("[debugger] Error listing files:",u),l.status(500).json({error:u.message})}});function i(c){if(typeof c=="string"&&c){let l=x.resolve(c);return l.endsWith(".test.yaml")?{filePath:l}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return o.get("/api/test-flow",async(c,l)=>{let u=i(c.query.file);if("error"in u)return l.status(400).json({error:u.error});let d=u.filePath;try{n?.(d);let f=await P.readFile(d,"utf-8"),g=await P.stat(d),h=$e(f),p=At(f,d,s);if(p.suite){let w=p.suite,y={tests:w.tests.map(b=>({name:b.name,statements:b.testFlow.statements??[],teardown:b.testFlow.teardown,skip:b.skip,timeout:b.timeout,fail:b.fail,only:b.only,slow:b.slow})),beforeAll:me(w.beforeAll),afterAll:me(w.afterAll),beforeEach:me(w.beforeEach),afterEach:me(w.afterEach)},m={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:y};l.json({isSuite:!0,testFlow:m,metadata:h,name:p.name,tags:p.tags,use:p.use,filePath:d,fileName:x.basename(d),lastModified:g.mtimeMs})}else{let w=p.testFlow;Be(f,w),l.json({isSuite:!1,testFlow:w,metadata:h,name:p.name,tags:p.tags,use:p.use,parameters:p.parameters,filePath:d,fileName:x.basename(d),lastModified:g.mtimeMs})}}catch(f){if(f.code==="ENOENT")return l.status(404).json({error:`File not found: ${d}`});console.error("[debugger] Error loading test flow:",f),l.status(500).json({error:f.message})}}),o.put("/api/test-flow",async(c,l)=>{try{let u=i(c.query.file);if("error"in u)return l.status(400).json({error:u.error});let d=u.filePath,{testFlow:f,metadata:g}=c.body;if(!f)return l.status(400).json({error:"testFlow is required"});let h=x.join(s??t,"templates"),p=await Jo(h),w=Mt(f,p),y=wt(w,g),m=d+".tmp";await P.writeFile(m,y,"utf-8"),await P.rename(m,d);let b=await P.stat(d);l.json({success:!0,lastModified:b.mtimeMs})}catch(u){console.error("[debugger] Error saving test flow:",u),l.status(500).json({error:u.message})}}),o.get("/api/fixtures",async(c,l)=>{try{let u=a();try{let f=(await P.readdir(u,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:f,dir:u})}catch(d){if(d.code==="ENOENT")l.json({files:[],dir:u});else throw d}}catch(u){console.error("[debugger] Error listing fixtures:",u),l.status(500).json({error:u.message})}}),o.post("/api/fixtures",async(c,l)=>{try{let u=a();await P.mkdir(u,{recursive:!0});let{name:d,content:f}=c.body;if(!d||!f)return l.status(400).json({error:"name and content are required"});let g=x.basename(d);if(!g)return l.status(400).json({error:"Invalid file name"});let h=x.join(u,g);await P.writeFile(h,Buffer.from(f,"base64")),l.json({fileName:g})}catch(u){console.error("[debugger] Error saving fixture:",u),l.status(500).json({error:u.message})}}),o.get("/api/functions",async(c,l)=>{let u=x.join(s??t,"helpers"),d;try{d=await P.readdir(u)}catch(p){return p.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading helpers dir:",p),l.status(500).json({error:p.message}))}let f=d.filter(p=>/\.(ts|js|mjs)$/.test(p)),g=[],h=1;for(let p of f){let w=await P.readFile(x.join(u,p),"utf-8"),y=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,m;for(;(m=y.exec(w))!==null;){let[,b,E]=m,T=E.replace(/\s*:\s*[^,)]+/g,"");g.push({id:h++,name:`helpers/${p}#${b}`,description:"",status:"Active",code:`async function ${b}${T} {}`})}}l.json(g)}),o.get("/api/reusable-steps",async(c,l)=>{let u=x.join(s??t,"templates"),d;try{d=await P.readdir(u)}catch(h){return h.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading templates dir:",h),l.status(500).json({error:h.message}))}let f=d.filter(h=>h.endsWith(".yaml")).sort(),g=[];for(let h of f)try{let p=await P.readFile(x.join(u,h),"utf-8"),w=Me(p);if(!w||typeof w!="object")continue;let y=w.name??x.basename(h,".yaml"),m=w.description??"",b=me(w.statements);g.push({id:qe(y),organizationId:"local",name:y,description:m,statements:b})}catch(p){console.error(`[debugger] Error parsing template ${h}:`,p)}l.json(g)}),o.post("/api/reusable-steps/exists/:name",async(c,l)=>{let u=x.join(s??t,"templates"),d=decodeURIComponent(c.params.name).toLowerCase(),f;try{f=await P.readdir(u)}catch(g){return g.code==="ENOENT"?l.json(!1):l.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=await P.readFile(x.join(u,g),"utf-8");if((Me(h)?.name??x.basename(g,".yaml")).toLowerCase()===d)return l.json(!0)}catch{}l.json(!1)}),o.post("/api/reusable-steps",async(c,l)=>{let u=x.join(s??t,"templates"),{reusableStep:d}=c.body??{};if(!d?.name||!Array.isArray(d.statements))return l.status(400).json({error:"reusableStep.name and reusableStep.statements are required"});await P.mkdir(u,{recursive:!0});let f=d.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=x.join(u,`${f}.yaml`),h={name:d.name,statements:It(d.statements)};d.description&&(h.description=d.description),await P.writeFile(g,Xe(h),"utf-8"),l.json({id:qe(d.name),organizationId:"local",name:d.name,description:d.description??"",statements:d.statements})}),o.put("/api/reusable-steps/:id/update",async(c,l)=>{let u=parseInt(c.params.id,10),d=x.join(s??t,"templates"),f;try{f=await P.readdir(d)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=x.join(d,g),p=Me(await P.readFile(h,"utf-8"))??{},w=p.name??x.basename(g,".yaml");if(qe(w)!==u)continue;let y=c.body??{};return y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=It(y.statements)),await P.writeFile(h,Xe(p),"utf-8"),l.json({id:u,organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template with id ${u} not found`})}),o.put("/api/templates/:name",async(c,l)=>{let u=x.join(s??t,"templates"),d=decodeURIComponent(c.params.name),f;try{f=await P.readdir(u)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of f.filter(h=>h.endsWith(".yaml")))try{let h=x.join(u,g),p=Me(await P.readFile(h,"utf-8"))??{};if((p.name??x.basename(g,".yaml"))!==d)continue;let y=c.body??{};return y.name!==void 0&&(p.name=y.name),y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=It(y.statements)),await P.writeFile(h,Xe(p),"utf-8"),l.json({organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template "${d}" not found`})}),o.get("/api/test-results",async(c,l)=>{let u=i(c.query.file);if("error"in u)return l.status(400).json({error:u.error});let d=s??t,f=x.basename(u.filePath,".test.yaml"),g=x.join(d,".shiplight","artifacts",f),h=[],p=null;try{let w=await P.readdir(g,{withFileTypes:!0});for(let y of w)if(y.isDirectory()){let m=x.join(g,y.name),b=await P.readdir(m);for(let E of b.filter(T=>/\.(png|jpe?g|webp)$/i.test(T))){let T=x.relative(g,x.join(m,E)),$=y.name.match(/^(.+?)_(before|after)$/),Y=$?$[1]:y.name,L=$?$[2]:"screenshot";h.push({url:`/api/report-assets/${f}/${T}`,label:L,stepId:Y})}}else/\.(webm|mp4)$/i.test(y.name)&&(p||(p=`/api/report-assets/${f}/${y.name}`));h.sort((y,m)=>y.stepId!==m.stepId?(y.stepId??"").localeCompare(m.stepId??""):y.label.localeCompare(m.label))}catch{}if(h.length===0&&!p)return l.status(404).json({error:"No test results found"});l.json({videoPath:p,screenshots:h})}),o}var gs=_(()=>{"use strict";de();$t()});import*as bs from"http";import*as vs from"net";import*as ne from"path";import*as Ss from"fs";import{randomUUID as Xo}from"crypto";function ys(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let s=[];e.on("data",n=>s.push(n)),e.on("end",()=>t(Buffer.concat(s))),e.on("error",r)})}function ws(e,t,r,s,n,o){return new Promise(a=>{let i=Array.isArray(o)?o[0]:o,c=bs.request({hostname:r,port:s,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(n.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",u=>t.write(u)),l.on("end",()=>{t.end(),a()}),l.on("error",()=>{t.writableEnded||t.end(),a()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),a()}),c.end(n)})}function Ot(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var ms,Ze,_s=_(()=>{"use strict";ft();ms=1e4,Ze=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??dt,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}openSession(t){let r=ne.resolve(t),s=this.byYamlPath.get(r);if(s){let l=this.sessions.get(s);if(l&&l.session.status!=="ended")return{...l.session};this.byYamlPath.delete(r)}if(!Ss.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!Ee(ne.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Xo().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"idle",exitInfo:null},c={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,c),this.byYamlPath.set(r,o),this.notifyStateChange(i),this.log(`[manager] session ${o} created (idle) for ${ne.basename(r)}`),{...i}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let s=r.session.yamlPath,n=Ee(ne.dirname(s));if(!n)throw new Error(`No Playwright config found for ${s} (searched parents for playwright.config.{ts,js,mjs}).`);r.session.status="starting",this.notifyStateChange(r.session);let o=(async()=>{let a=ms+18e4,i,c=new Promise((d,f)=>{i=setTimeout(()=>f(new Error(`Timed out after ${a/1e3}s waiting for inner playwright to register on a port.`)),a)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:s,configPath:n,tempSuffix:t,headed:this.headed}),c])}finally{i&&clearTimeout(i)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup;let u=d=>{r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo=d,this.notifyStateChange(r.session))};r.livenessTimer=this.startLivenessProbe(t,u),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ne.basename(s)})`)})();r.readyPromise=o;try{await o}catch(a){throw r.session.status="ended",r.session.exitInfo=a.message,this.notifyStateChange(r.session),a}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(s){this.log(`[manager] stopSandbox ${t} cleanup error: ${s.message}`)}r.session.innerPort=0,r.session.innerHost="127.0.0.1",r.session.pid=0,r.session.status="idle",r.session.exitInfo=null,r.cleanup=async()=>{},r.readyPromise=Promise.resolve(),this.notifyStateChange(r.session)}}startLivenessProbe(t,r){let o=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"){clearInterval(a);return}let c=!1;try{process.kill(i.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?o=0:(o+=1,o>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(s){this.log(`[manager] closeSession ${t} cleanup error: ${s.message}`)}}}async restartInner(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to restart`);if(r.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let s=r.session.yamlPath,n=Ee(ne.dirname(s));if(!n)throw new Error(`No Playwright config found for ${s} (searched parents for playwright.config.{ts,js,mjs}).`);r.restartInProgress=!0;try{r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(l){this.log(`[manager] restartInner ${t} old cleanup error: ${l.message}`)}r.session.status="starting",r.session.innerPort=0,r.session.pid=0,this.notifyStateChange(r.session);let o=ms+18e4,a,i=new Promise((l,u)=>{a=setTimeout(()=>u(new Error(`Timed out after ${o/1e3}s waiting for inner playwright to respawn.`)),o)}),c;try{c=await Promise.race([this.spawner({yamlFilePath:s,configPath:n,tempSuffix:t,headed:this.headed}),i])}catch(l){throw r.session.status="ended",r.session.exitInfo=l.message,this.notifyStateChange(r.session),l}finally{a&&clearTimeout(a)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup,r.livenessTimer=this.startLivenessProbe(t,l=>{r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo=l,this.notifyStateChange(r.session))}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} restarted on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ne.basename(s)})`)}finally{r.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:s}=r;return async(n,o,a)=>{let i=this.sessions.get(t);if(!i){o.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}if(i.session.status==="idle"){o.status(503).json({status:"error",message:"Sandbox not started"});return}let c=`${n.method} ${n.path}`;if(c==="POST /api/int-runner/create-session"){let u=await ys(n),d={};if(u.length)try{d=JSON.parse(u.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}d.testFilePath=i.session.yamlPath,await ws(n,o,i.session.innerHost,i.session.innerPort,Buffer.from(JSON.stringify(d),"utf-8"),"application/json");return}if(c==="POST /api/int-runner/terminate-session"){try{await this.stopSandbox(t),o.json({status:"success",details:"Sandbox stopped"})}catch(u){o.status(500).json({status:"error",message:u.message})}return}if(c==="POST /api/int-runner/liveview-url"){let u=s?.(n)??"";o.json({liveviewUrl:u,browserWsUrl:""});return}let l=await ys(n);await ws(n,o,i.session.innerHost,i.session.innerPort,l,n.headers["content-type"])}}wsUpgradeFor(t){return async(r,s,n,o)=>{let a=this.sessions.get(t);if(!a){Ot(s,`HTTP/1.1 404 Not Found\r
218
218
  \r
219
219
  `);return}if(a.session.status==="ended"){Ot(s,`HTTP/1.1 410 Gone\r
220
220
  \r
221
- `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,l=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!l.ok)throw new Error(`Inner /api/browser-cdp returned ${l.status}`);let{cdpUrl:c}=await l.json(),u=new URL(c.replace(/^ws/,"http")),d=parseInt(u.port||"80",10),f=u.hostname,g=u.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let h=bs.createConnection(d,f);h.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
221
+ `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,c=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),u=new URL(l.replace(/^ws/,"http")),d=parseInt(u.port||"80",10),f=u.hostname,g=u.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let h=vs.createConnection(d,f);h.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
222
222
  Host: ${f}:${d}\r
223
- `;for(let[b,y]of Object.entries(r.headers)){let m=b.toLowerCase();m!=="host"&&m!=="origin"&&(p+=`${b}: ${Array.isArray(y)?y.join(", "):y}\r
223
+ `;for(let[w,y]of Object.entries(r.headers)){let m=w.toLowerCase();m!=="host"&&m!=="origin"&&(p+=`${w}: ${Array.isArray(y)?y.join(", "):y}\r
224
224
  `)}p+=`\r
225
225
  `,h.write(p),n.length&&h.write(n),h.pipe(s),s.pipe(h)}),h.on("error",()=>{(!("destroyed"in s)||!s.destroyed)&&s.destroy()}),s.on("error",()=>{h.destroyed||h.destroy()}),s.on("close",()=>{h.destroyed||h.destroy()})}catch(i){this.log(`[manager] WS upgrade for ${t} failed: ${i.message}`),Ot(s,`HTTP/1.1 502 Bad Gateway\r
226
226
  \r
227
- `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as Jo}from"express";import Lt from"express";import*as _e from"fs";import*as oe from"path";function Xo(e){return e?oe.isAbsolute(e)?oe.normalize(e):oe.resolve(e):null}function qo(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function Zo(e){let t=oe.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function _s(e){let{manager:t,staticDir:r,resolveYamlPath:s=Xo,liveviewUrlBuilder:n=qo,fallbackRouter:o,artifactsDir:a}=e,i=Jo(),l=a?Lt.static(a):null;i.post("/api/debugger/sessions",Lt.json(),async(u,d)=>{let f=u.body?.yamlPath;if(typeof f!="string"||!f){d.status(400).json({error:"yamlPath is required"});return}let g=s(f);if(!g){d.status(403).json({error:"Path outside allowed roots"});return}if(!Zo(g)){d.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!_e.existsSync(g)){d.status(404).json({error:"File not found"});return}let h=t.listSessions().find(p=>p.yamlPath===g&&p.status!=="ended");try{let p=t.openSession(g);d.status(h?200:201).json({sessionId:p.sessionId,yamlPath:p.yamlPath,startedAt:p.startedAt,status:p.status})}catch(p){d.status(500).json({error:p.message})}}),i.get("/api/debugger/sessions",(u,d)=>{d.setHeader("Cache-Control","no-store"),d.json({sessions:t.listSessions().map(f=>({sessionId:f.sessionId,yamlPath:f.yamlPath,startedAt:f.startedAt,status:f.status}))})}),i.delete("/api/debugger/sessions/:sessionId",async(u,d)=>{let f=u.params.sessionId,g=!!t.getSession(f);await t.closeSession(f),d.json({deleted:!0,alreadyGone:!g})}),_e.existsSync(r)?i.use("/debugger/static",Lt.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),i.get("/debugger/:sessionId/",(u,d)=>{let f=u.params.sessionId;if(!t.getSession(f)){d.status(404).send("Debugger session not found");return}let h=oe.join(r,"index.html");if(!_e.existsSync(h)){d.status(500).send(`Debugger SPA bundle missing at ${h}`);return}let b=_e.readFileSync(h,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');d.type("html").send(b)});let c=async(u,d,f)=>{let g=u.params.sessionId;if(g==="static")return f();let h=t.getSession(g);if(!h){d.status(404).json({status:"error",message:"Session not found"});return}let p=u.originalUrl,b=`/debugger/${g}`;if(p.startsWith(b)){let m=p.slice(b.length)||"/";u.url=m,Object.defineProperty(u,"originalUrl",{value:m,configurable:!0})}if(l&&u.path.startsWith("/api/report-assets/")){u.url=u.url.replace(/^\/api\/report-assets/,""),l(u,d,f);return}if(h.status==="idle"||h.status==="starting"){let m=`${u.method} ${u.path}`;if(m==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(w){d.status(500).json({status:"error",message:w.message});return}else if(m==="POST /api/int-runner/liveview-url"){d.json({liveviewUrl:"",browserWsUrl:""});return}else if(u.path.startsWith("/api/int-runner/")){d.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(u,d,f);return}else{d.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>n(g,u)})(u,d,f)};return i.use("/debugger/:sessionId",c),i}function xs(e,t,r,s){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
227
+ `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as qo}from"express";import Lt from"express";import*as _e from"fs";import*as oe from"path";function Zo(e){return e?oe.isAbsolute(e)?oe.normalize(e):oe.resolve(e):null}function Qo(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function ei(e){let t=oe.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function xs(e){let{manager:t,staticDir:r,resolveYamlPath:s=Zo,liveviewUrlBuilder:n=Qo,fallbackRouter:o,artifactsDir:a}=e,i=qo(),c=a?Lt.static(a):null;i.post("/api/debugger/sessions",Lt.json(),async(u,d)=>{let f=u.body?.yamlPath;if(typeof f!="string"||!f){d.status(400).json({error:"yamlPath is required"});return}let g=s(f);if(!g){d.status(403).json({error:"Path outside allowed roots"});return}if(!ei(g)){d.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!_e.existsSync(g)){d.status(404).json({error:"File not found"});return}let h=t.listSessions().find(p=>p.yamlPath===g&&p.status!=="ended");try{let p=t.openSession(g);d.status(h?200:201).json({sessionId:p.sessionId,yamlPath:p.yamlPath,startedAt:p.startedAt,status:p.status})}catch(p){d.status(500).json({error:p.message})}}),i.get("/api/debugger/sessions",(u,d)=>{d.setHeader("Cache-Control","no-store"),d.json({sessions:t.listSessions().map(f=>({sessionId:f.sessionId,yamlPath:f.yamlPath,startedAt:f.startedAt,status:f.status}))})}),i.delete("/api/debugger/sessions/:sessionId",async(u,d)=>{let f=u.params.sessionId,g=!!t.getSession(f);await t.closeSession(f),d.json({deleted:!0,alreadyGone:!g})}),_e.existsSync(r)?i.use("/debugger/static",Lt.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),i.get("/debugger/:sessionId/",(u,d)=>{let f=u.params.sessionId;if(!t.getSession(f)){d.status(404).send("Debugger session not found");return}let h=oe.join(r,"index.html");if(!_e.existsSync(h)){d.status(500).send(`Debugger SPA bundle missing at ${h}`);return}let w=_e.readFileSync(h,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');d.type("html").send(w)});let l=async(u,d,f)=>{let g=u.params.sessionId;if(g==="static")return f();let h=t.getSession(g);if(!h){d.status(404).json({status:"error",message:"Session not found"});return}let p=u.originalUrl,w=`/debugger/${g}`;if(p.startsWith(w)){let m=p.slice(w.length)||"/";u.url=m,Object.defineProperty(u,"originalUrl",{value:m,configurable:!0})}if(c&&u.path.startsWith("/api/report-assets/")){u.url=u.url.replace(/^\/api\/report-assets/,""),c(u,d,f);return}if(h.status==="idle"||h.status==="starting"){let m=`${u.method} ${u.path}`;if(m==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(b){d.status(500).json({status:"error",message:b.message});return}else if(m==="POST /api/int-runner/liveview-url"){d.json({liveviewUrl:"",browserWsUrl:""});return}else if(u.path.startsWith("/api/int-runner/")){d.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(u,d,f);return}else{d.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>n(g,u)})(u,d,f)};return i.use("/debugger/:sessionId",l),i}function Ts(e,t,r,s){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
228
228
  \r
229
- `),r.destroy();return}let a=o[1],i=o[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,s,i||void 0)}var Ts=_(()=>{"use strict"});var ks={};ie(ks,{startDebuggerServer:()=>ti});import Le from"express";import*as F from"path";import{createRequire as Qo}from"node:module";async function ti(e){let{initialDir:t,initialFile:r,projectRoot:s,port:n,headed:o=!1}=e,a=new Qe({headed:o}),i=Le();i.use((y,m,w)=>{if(m.setHeader("Access-Control-Allow-Origin","*"),m.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),m.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),y.method==="OPTIONS")return m.sendStatus(204);w()}),i.use(Le.json({limit:"10mb"}));let l=fs({initialDir:t,initialFile:r,projectRoot:s});i.use(l);let c=F.join(s??t,".shiplight","artifacts");i.use("/api/report-assets",Le.static(c));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,d=u.includes(F.sep+"src"+F.sep),f=d?F.resolve(u,"../../dist/static"):F.join(u,"static"),g=d?F.resolve(u,"../../dist/static-embedded"):F.join(u,"static-embedded");i.use("/devtools",Le.static(Ps)),i.use(Le.static(f)),i.use(_s({manager:a,staticDir:g,fallbackRouter:l,artifactsDir:c})),i.get("*path",(y,m,w)=>{if(y.path.startsWith("/api/"))return w();m.sendFile(F.join(f,"index.html"),E=>{E&&m.send(ri(t,n))})});let h=await new Promise((y,m)=>{let w=i.listen(n,"localhost",()=>{y(w)});w.on("error",m)});h.on("upgrade",(y,m,w)=>{if((y.url??"").startsWith("/ws/debugger/")){xs(a,y,m,w);return}m.write(`HTTP/1.1 404 Not Found\r
229
+ `),r.destroy();return}let a=o[1],i=o[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,s,i||void 0)}var ks=_(()=>{"use strict"});var Es={};ie(Es,{startDebuggerServer:()=>si});import Oe from"express";import*as U from"path";import{createRequire as ti}from"node:module";async function si(e){let{initialDir:t,initialFile:r,projectRoot:s,port:n,headed:o=!1}=e,a=new Ze({headed:o}),i=Oe();i.use((y,m,b)=>{if(m.setHeader("Access-Control-Allow-Origin","*"),m.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),m.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),y.method==="OPTIONS")return m.sendStatus(204);b()}),i.use(Oe.json({limit:"10mb"}));let c=hs({initialDir:t,initialFile:r,projectRoot:s});i.use(c);let l=U.join(s??t,".shiplight","artifacts");i.use("/api/report-assets",Oe.static(l));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,d=u.includes(U.sep+"src"+U.sep),f=d?U.resolve(u,"../../dist/static"):U.join(u,"static"),g=d?U.resolve(u,"../../dist/static-embedded"):U.join(u,"static-embedded");i.use("/devtools",Oe.static(Ps)),i.use(Oe.static(f)),i.use(xs({manager:a,staticDir:g,fallbackRouter:c,artifactsDir:l})),i.get("*path",(y,m,b)=>{if(y.path.startsWith("/api/"))return b();m.sendFile(U.join(f,"index.html"),E=>{E&&m.send(ni(t,n))})});let h=await new Promise((y,m)=>{let b=i.listen(n,"localhost",()=>{y(b)});b.on("error",m)});h.on("upgrade",(y,m,b)=>{if((y.url??"").startsWith("/ws/debugger/")){Ts(a,y,m,b);return}m.write(`HTTP/1.1 404 Not Found\r
230
230
  \r
231
- `),m.destroy()});let p=`http://localhost:${n}`,b=r?`${p}/?open=${encodeURIComponent(r)}`:p;return console.error(`[debugger] Server running at ${p}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:b,close:async()=>{await a.shutdown(),await new Promise((y,m)=>{h.close(w=>w?m(w):y())})}}}function ri(e,t){return`<!DOCTYPE html>
231
+ `),m.destroy()});let p=`http://localhost:${n}`,w=r?`${p}/?open=${encodeURIComponent(r)}`:p;return console.error(`[debugger] Server running at ${p}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:w,close:async()=>{await a.shutdown(),await new Promise((y,m)=>{h.close(b=>b?m(b):y())})}}}function ni(e,t){return`<!DOCTYPE html>
232
232
  <html>
233
233
  <head>
234
234
  <meta charset="utf-8">
@@ -246,17 +246,17 @@ Host: ${f}:${d}\r
246
246
  <p>Directory: <code>${e}</code></p>
247
247
  <p>Server: localhost:${t}</p>
248
248
  </body>
249
- </html>`}var ei,Ps,Es=_(()=>{"use strict";hs();vs();Ts();ei=Qo(import.meta.url);try{Ps=F.join(F.dirname(ei.resolve("@shiplightai/devtools-assets/package.json")),"dist")}catch{console.error("[debugger] Required peer package @shiplightai/devtools-assets is not installed. Reinstall shiplightai (it pulls the assets package as a dependency)."),process.exit(1)}});var $s={};ie($s,{startDebugger:()=>ni});import*as Q from"fs";import*as he from"path";function et(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function ni(e){let t,r=xe,s=!1,n,o=!1,a=!0,i=!1;for(let m=0;m<e.length;m++)e[m]==="--port"&&e[m+1]?(r=parseInt(e[m+1],10),s=!0,m++):e[m]==="--url"&&e[m+1]?(n=e[m+1],m++):e[m]==="--new"?o=!0:e[m]==="--open"?a=!1:e[m]==="--no-open"?a=!0:e[m]==="--headed"?i=!0:e[m]==="--help"||e[m]==="-h"?(oi(),process.exit(0)):e[m].startsWith("--")||(t=e[m]);let l,c;if(t){let m=he.resolve(t);Q.existsSync(m)&&Q.statSync(m).isDirectory()?l=m:(l=he.dirname(m),c=m)}else l=process.cwd();if(o&&c&&!Q.existsSync(c)){let m=he.dirname(c);Q.existsSync(m)||Q.mkdirSync(m,{recursive:!0}),Q.writeFileSync(c,si(n||"https://example.com"),"utf-8"),console.log(`Created new test file: ${c}`)}c&&!Q.existsSync(c)&&(console.error(`Error: File not found: ${c}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:u}=await Promise.resolve().then(()=>(ht(),hr)),d=u(l),f=d?he.dirname(d):l;(await import("dotenv")).config({path:he.join(f,".env"),override:!0}),d||(console.error("Error: No Playwright config found in "+l),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:h}=await Promise.resolve().then(()=>(Es(),ks));if(console.log(c?`Starting debugger for: ${c}`:`Starting debugger in: ${l}`),d&&console.log(`Using Playwright config: ${d}`),!s){let m=await ur(xe,As);m===null&&(console.error(et(`Error: No available port found in range ${xe}-${xe+As-1}.`)),console.error(et("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),m!==xe&&console.log(`Port ${xe} is in use; using port ${m} instead.`),r=m}let p;try{p=await h({initialDir:l,initialFile:c,projectRoot:f,port:r,headed:i})}catch(m){throw m?.code==="EADDRINUSE"&&(console.error(et(`Error: Port ${r} is already in use.`)),console.error(et(s?"Try a different port number, or omit --port to let shiplight auto-pick one.":"All probed ports became busy after selection \u2014 re-run shiplight debug to retry.")),process.exit(1)),m}if(ir(r,l),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!a)try{let{default:m}=await import("open");await m(p.url)}catch{}let b=!1,y=async()=>{b&&(console.log(`
250
- Force exiting...`),process.exit(1)),b=!0,console.log(`
251
- Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{ar(r,l),await p.close()}catch{}clearTimeout(m),process.exit(0)};process.on("SIGINT",y),process.on("SIGTERM",y)}function oi(){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(" --headed Force a visible Chromium window (overrides use.headless)"),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 xe,As,si,Is=_(()=>{"use strict";cr();dr();xe=6174,As=10;si=e=>`goal: New test
249
+ </html>`}var ri,Ps,As=_(()=>{"use strict";gs();_s();ks();ri=ti(import.meta.url);try{Ps=U.join(U.dirname(ri.resolve("@shiplightai/devtools-assets/package.json")),"dist")}catch{console.error("[debugger] Required peer package @shiplightai/devtools-assets is not installed. Reinstall shiplightai (it pulls the assets package as a dependency)."),process.exit(1)}});var Is={};ie(Is,{startDebugger:()=>ii});import*as Q from"fs";import*as he from"path";function Qe(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function ii(e){let t,r=xe,s=!1,n,o=!1,a=!0,i=!1;for(let m=0;m<e.length;m++)e[m]==="--port"&&e[m+1]?(r=parseInt(e[m+1],10),s=!0,m++):e[m]==="--url"&&e[m+1]?(n=e[m+1],m++):e[m]==="--new"?o=!0:e[m]==="--open"?a=!1:e[m]==="--no-open"?a=!0:e[m]==="--headed"?i=!0:e[m]==="--help"||e[m]==="-h"?(ai(),process.exit(0)):e[m].startsWith("--")||(t=e[m]);let c,l;if(t){let m=he.resolve(t);Q.existsSync(m)&&Q.statSync(m).isDirectory()?c=m:(c=he.dirname(m),l=m)}else c=process.cwd();if(o&&l&&!Q.existsSync(l)){let m=he.dirname(l);Q.existsSync(m)||Q.mkdirSync(m,{recursive:!0}),Q.writeFileSync(l,oi(n||"https://example.com"),"utf-8"),console.log(`Created new test file: ${l}`)}l&&!Q.existsSync(l)&&(console.error(`Error: File not found: ${l}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:u}=await Promise.resolve().then(()=>(ft(),mr)),d=u(c),f=d?he.dirname(d):c;(await import("dotenv")).config({path:he.join(f,".env"),override:!0}),d||(console.error("Error: No Playwright config found in "+c),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:h}=await Promise.resolve().then(()=>(As(),Es));if(console.log(l?`Starting debugger for: ${l}`:`Starting debugger in: ${c}`),d&&console.log(`Using Playwright config: ${d}`),!s){let m=await fr(xe,$s);m===null&&(console.error(Qe(`Error: No available port found in range ${xe}-${xe+$s-1}.`)),console.error(Qe("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),m!==xe&&console.log(`Port ${xe} is in use; using port ${m} instead.`),r=m}let p;try{p=await h({initialDir:c,initialFile:l,projectRoot:f,port:r,headed:i})}catch(m){throw m?.code==="EADDRINUSE"&&(console.error(Qe(`Error: Port ${r} is already in use.`)),console.error(Qe(s?"Try a different port number, or omit --port to let shiplight auto-pick one.":"All probed ports became busy after selection \u2014 re-run shiplight debug to retry.")),process.exit(1)),m}if(cr(r,c),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!a)try{let{default:m}=await import("open");await m(p.url)}catch{}let w=!1,y=async()=>{w&&(console.log(`
250
+ Force exiting...`),process.exit(1)),w=!0,console.log(`
251
+ Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{lr(r,c),await p.close()}catch{}clearTimeout(m),process.exit(0)};process.on("SIGINT",y),process.on("SIGTERM",y)}function ai(){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(" --headed Force a visible Chromium window (overrides use.headless)"),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 xe,$s,oi,Ms=_(()=>{"use strict";pr();hr();xe=6174,$s=10;oi=e=>`goal: New test
252
252
  base_url: ${e}
253
253
  statements:
254
254
  - URL: /
255
- `});import xp from"dotenv";function ai(){return globalThis[ii]}function tt(){let e=ai();return e===void 0?process.env:e}var ii,Rt=_(()=>{"use strict";ii="__shiplightDotenvCache__"});function rt(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith(ci)?li:pi}var ci,li,pi,Ct=_(()=>{"use strict";ci="shp_",li="https://nova-api.shiplight.ai",pi="https://api.shiplight.ai"});var Nt={};ie(Nt,{lookupActionStores:()=>fi,updateActionStores:()=>hi});function Ms(){let e=tt(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:rt(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function fi(e){let t=Ms();if(!t||e.length===0)return new Map;try{let r=new AbortController,s=setTimeout(()=>r.abort(),ui),n=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(s),!n.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${n.status}`),new Map;let o=await n.json(),a=new Map;for(let[i,l]of Object.entries(o.stores??{}))a.set(i,l);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function hi(e){let t=Ms();if(!t||e.size===0)return 0;try{let r=new AbortController,s=setTimeout(()=>r.abort(),di),n={};for(let[i,l]of e)n[i]=l;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:n}),signal:r.signal});return clearTimeout(s),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var ui,di,Dt=_(()=>{"use strict";Ct();Rt();ui=2e3,di=5e3});import*as G from"fs";import*as Re from"path";import{globSync as gi}from"glob";function Os(e){let t=tt().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ft:new jt(e)}function Ce(e){return e.replace(/\//g,"__")+".json"}function yi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var mi,jt,Ft,Ls=_(()=>{"use strict";de();Rt();mi=".shiplight/action-cache";jt=class{constructor(t){this.cwd=t;this.cacheDir=Re.join(t,mi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!G.existsSync(this.cacheDir))return r;for(let s of t){let n=Re.join(this.cacheDir,Ce(s));try{if(G.existsSync(n)){let o=G.readFileSync(n,"utf-8");r.set(s,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;G.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[s,n]of t)try{let o=Re.join(this.cacheDir,Ce(s)),a=vt();if(G.existsSync(o))try{a=JSON.parse(G.readFileSync(o,"utf-8"))}catch{}let i={...a,entries:{...a.entries,...n.entries}};G.writeFileSync(o,JSON.stringify(i,null,2)),r++}catch{}return r}loadAll(){if(!G.existsSync(this.cacheDir))return;let t=gi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,s=0;for(let n of t)try{let o=G.readFileSync(Re.join(this.cacheDir,n),"utf-8"),a=JSON.parse(o),i=yi(n);r.set(i,a),s+=Object.keys(a.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${s} cached action entit${s===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},Ft=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}loadAll(){}}});var js={};ie(js,{buildPlaywrightSpawnOptions:()=>Cs,buildTestPathsFromGitInfo:()=>Ds,cloudKeyToCwdRelPath:()=>Ht,runTests:()=>bi});import{spawn as wi,execFileSync as Rs}from"child_process";import*as W from"fs";import*as U from"path";import{globSync as Ut}from"glob";async function bi(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(d=>W.existsSync(U.join(t,d)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
256
- `));let n=e.includes("--magic"),a=e.filter(d=>d!=="--magic").map(d=>d.endsWith(".test.yaml")?d.replace(/\.test\.yaml$/,".yaml.spec.ts"):d),i=Os(t);await vi(t,i),ke&&process.stdout.write(`shiplightai v${ke}
257
- `);let l={...process.env};n&&(l.SHIPLIGHT_MAGIC="1");let c=wi("npx",["playwright","test",...a],Cs(t,l)),u=await new Promise(d=>{c.on("close",f=>d(f??1))});await _i(t,i),process.exit(u)}function Cs(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Bt(){try{return Rs("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Si(){try{let e=Rs("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Ns(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let s=Si(),n=Bt();return Ds(e,t,s,n)}function Ds(e,t,r,s){return{testPaths:e.map(o=>{let a=s?U.relative(s,U.resolve(t,o)):o;return`${r}${a}`}),branchPrefix:r}}function Ht(e,t,r,s){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?U.relative(s,U.resolve(r,n)):n}async function vi(e,t){try{let r=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:s,branchPrefix:n}=Ns(r,e,t.isCloud),o=await t.lookup(s);if(o.size===0)return;let a=U.join(e,".shiplight","action-cache");W.mkdirSync(a,{recursive:!0});let i=Bt(),l=0;for(let[c,u]of o){let d=t.isCloud?Ht(c,n,i,e):c,f=U.join(a,Ce(d));W.writeFileSync(f,JSON.stringify(u,null,2)),l++}console.log(`[shiplight] Cache: downloaded ${l} action store${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function _i(e,t){try{let r=U.join(e,"test-results");if(!W.existsSync(r))return;let s=Ut("**/new-action-entities.json",{cwd:r});if(s.length===0)return;let n={};for(let d of s)try{let f=W.readFileSync(U.join(r,d),"utf-8"),g=JSON.parse(f);g?.entries&&Object.assign(n,g.entries)}catch{}if(Object.keys(n).length===0)return;let o=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=Ns(o,e,t.isCloud),l=new Map;for(let d=0;d<o.length;d++){let f=o[d],g=a[d],h=U.join(e,f.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!W.existsSync(h))continue;let p=W.readFileSync(h,"utf-8"),b={};for(let[y,m]of Object.entries(n))p.includes(y)&&(b[y]=m);if(Object.keys(b).length>0){let y=t.isCloud?Ht(g,i,Bt(),e):g,m=U.join(e,".shiplight","action-cache",Ce(y)),w={};if(W.existsSync(m))try{w=JSON.parse(W.readFileSync(m,"utf-8")).entries}catch{}l.set(g,{version:"1.0",entries:{...w,...b}})}}if(l.size===0)return;let c=await t.update(l),u=Array.from(l.values()).reduce((d,f)=>d+Object.keys(f.entries).length,0);console.log(`[shiplight] Cache: saved ${u} action entit${u!==1?"ies":"y"} for ${c} test${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Fs=_(()=>{"use strict";Ls();Ne()});import Us from"node:path";function M(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function nt(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 xi(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Ti(e,t){let r=e||"?",s=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${s}`:`failed after ${r} ${s}`}function st(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 Pi(e){return`<span class="badge badge-${e}">${e}</span>`}function ki(e){if(!e.code)return"";let t=e.code.split(`
258
- `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((n,o)=>{let a=e.codeStartLine+o,i=a===e.codeLine,l=String(a).padStart(4);return`<span class="code-line${i?" code-line-active":""}">${l} \u2502 ${M(n)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(s=>`<span class="code-line code-line-body">${M(s)}</span>`).join("")}</pre></div>`}function Ei(e){let t=e.duration!=null?`<span class="step-duration">${nt(e.duration)}</span>`:"",r=st(e.status);if(e.screenshot||e.message||e.error||e.code){let n="";e.screenshot&&(n=`<img src="${M(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":ki(e),a="";e.error&&(a=`<div class="step-error"><pre>${M(e.error)}</pre></div>`);let i="";e.message&&!e.error&&(i=`<div class="step-message">${M(e.message)}</div>`);let l=e.status==="failure"?" open":"";return`
259
- <details class="step-details step step-${e.status}"${l}>
255
+ `});import kp from"dotenv";function li(){return globalThis[ci]}function et(){let e=li();return e===void 0?process.env:e}var ci,Rt=_(()=>{"use strict";ci="__shiplightDotenvCache__"});function tt(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith(pi)?ui:di}var pi,ui,di,Ct=_(()=>{"use strict";pi="shp_",ui="https://nova-api.shiplight.ai",di="https://api.shiplight.ai"});var Nt={};ie(Nt,{lookupActionStores:()=>gi,updateActionStores:()=>mi});function Os(){let e=et(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:tt(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function gi(e){let t=Os();if(!t||e.length===0)return new Map;try{let r=new AbortController,s=setTimeout(()=>r.abort(),fi),n=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(s),!n.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${n.status}`),new Map;let o=await n.json(),a=new Map;for(let[i,c]of Object.entries(o.stores??{}))a.set(i,c);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function mi(e){let t=Os();if(!t||e.size===0)return 0;try{let r=new AbortController,s=setTimeout(()=>r.abort(),hi),n={};for(let[i,c]of e)n[i]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:n}),signal:r.signal});return clearTimeout(s),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var fi,hi,Dt=_(()=>{"use strict";Ct();Rt();fi=2e3,hi=5e3});import*as G from"fs";import*as Le from"path";import{globSync as yi}from"glob";function Ls(e){let t=et().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ft:new jt(e)}function Re(e){return e.replace(/\//g,"__")+".json"}function bi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var wi,jt,Ft,Rs=_(()=>{"use strict";de();Rt();wi=".shiplight/action-cache";jt=class{constructor(t){this.cwd=t;this.cacheDir=Le.join(t,wi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!G.existsSync(this.cacheDir))return r;for(let s of t){let n=Le.join(this.cacheDir,Re(s));try{if(G.existsSync(n)){let o=G.readFileSync(n,"utf-8");r.set(s,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;G.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[s,n]of t)try{let o=Le.join(this.cacheDir,Re(s)),a=St();if(G.existsSync(o))try{a=JSON.parse(G.readFileSync(o,"utf-8"))}catch{}let i={...a,entries:{...a.entries,...n.entries}};G.writeFileSync(o,JSON.stringify(i,null,2)),r++}catch{}return r}loadAll(){if(!G.existsSync(this.cacheDir))return;let t=yi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,s=0;for(let n of t)try{let o=G.readFileSync(Le.join(this.cacheDir,n),"utf-8"),a=JSON.parse(o),i=bi(n);r.set(i,a),s+=Object.keys(a.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${s} cached action entit${s===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},Ft=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}loadAll(){}}});var Us={};ie(Us,{buildPlaywrightSpawnOptions:()=>Ns,buildTestPathsFromGitInfo:()=>js,cloudKeyToCwdRelPath:()=>Gt,extractVarOverrideArgs:()=>Fs,loadVarsFile:()=>Ht,parseVarsArg:()=>Bt,runTests:()=>Si});import{spawn as vi,execFileSync as Cs}from"child_process";import*as j from"fs";import*as R from"path";import{globSync as Ut}from"glob";async function Si(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("Shiplight options:"),console.log(" --vars KEY=VAL[,KEY=VAL...] Override runtime {{VAR}} values (repeatable)."),console.log(" Comma separates pairs, so values cannot"),console.log(" contain commas \u2014 use --vars-file for those."),console.log(" In the space-separated form, the value cannot"),console.log(" start with `--` (it is treated as the next"),console.log(" flag); use --vars=KEY=--value instead."),console.log(' --vars-file <path> JSON file of overrides ({ KEY: "value", ... }).'),console.log(" Values must be strings \u2014 quote numbers."),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"),console.log(" shiplight test --vars SAUCE_USER=standard_user"),console.log(" shiplight test --vars-file ./local-vars.json"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(f=>j.existsSync(R.join(t,f)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
256
+ `));let n=e.includes("--magic"),o=e.filter(f=>f!=="--magic"),a={};try{let{extracted:f,remaining:g}=Fs(o,t);o=g,a=f}catch(f){console.error(`[shiplight] ${f.message}`),process.exit(2)}let i=o.map(f=>f.endsWith(".test.yaml")?f.replace(/\.test\.yaml$/,".yaml.spec.ts"):f),c=Ls(t);await xi(t,c),Pe&&process.stdout.write(`shiplightai v${Pe}
257
+ `);let l={...process.env};n&&(l.SHIPLIGHT_MAGIC="1"),Object.keys(a).length>0&&(l.SHIPLIGHT_VARS_OVERRIDE=JSON.stringify(a));let u=vi("npx",["playwright","test",...i],Ns(t,l)),d=await new Promise(f=>{u.on("close",g=>f(g??1))});await Ti(t,c),process.exit(d)}function Ns(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Wt(){try{return Cs("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function _i(){try{let e=Cs("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Ds(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let s=_i(),n=Wt();return js(e,t,s,n)}function js(e,t,r,s){return{testPaths:e.map(o=>{let a=s?R.relative(s,R.resolve(t,o)):o;return`${r}${a}`}),branchPrefix:r}}function Gt(e,t,r,s){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?R.relative(s,R.resolve(r,n)):n}async function xi(e,t){try{let r=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:s,branchPrefix:n}=Ds(r,e,t.isCloud),o=await t.lookup(s);if(o.size===0)return;let a=R.join(e,".shiplight","action-cache");j.mkdirSync(a,{recursive:!0});let i=Wt(),c=0;for(let[l,u]of o){let d=t.isCloud?Gt(l,n,i,e):l,f=R.join(a,Re(d));j.writeFileSync(f,JSON.stringify(u,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 Ti(e,t){try{let r=R.join(e,"test-results");if(!j.existsSync(r))return;let s=Ut("**/new-action-entities.json",{cwd:r});if(s.length===0)return;let n={};for(let d of s)try{let f=j.readFileSync(R.join(r,d),"utf-8"),g=JSON.parse(f);g?.entries&&Object.assign(n,g.entries)}catch{}if(Object.keys(n).length===0)return;let o=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=Ds(o,e,t.isCloud),c=new Map;for(let d=0;d<o.length;d++){let f=o[d],g=a[d],h=R.join(e,f.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!j.existsSync(h))continue;let p=j.readFileSync(h,"utf-8"),w={};for(let[y,m]of Object.entries(n))p.includes(y)&&(w[y]=m);if(Object.keys(w).length>0){let y=t.isCloud?Gt(g,i,Wt(),e):g,m=R.join(e,".shiplight","action-cache",Re(y)),b={};if(j.existsSync(m))try{b=JSON.parse(j.readFileSync(m,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...b,...w}})}}if(c.size===0)return;let l=await t.update(c),u=Array.from(c.values()).reduce((d,f)=>d+Object.keys(f.entries).length,0);console.log(`[shiplight] Cache: saved ${u} action entit${u!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}function Bt(e){let t={};for(let r of e.split(",")){let s=r.trim();if(s==="")continue;let n=s.indexOf("=");if(n<=0)throw new Error(`Invalid --vars entry "${s}": expected KEY=VALUE`);let o=s.slice(0,n).trim();t[o]=s.slice(n+1)}return t}function Ht(e,t){let r=R.isAbsolute(e)?e:R.join(t,e);if(!j.existsSync(r))throw new Error(`--vars-file not found: ${r}`);let s;try{s=JSON.parse(j.readFileSync(r,"utf-8"))}catch(o){throw new Error(`--vars-file ${r} is not valid JSON: ${o.message}`)}if(s===null||typeof s!="object"||Array.isArray(s))throw new Error(`--vars-file ${r} must be a JSON object`);let n={};for(let[o,a]of Object.entries(s)){if(typeof a!="string")throw new Error(`--vars-file ${r}: key "${o}" must be a string (got ${typeof a})`);n[o]=a}return n}function Fs(e,t){let r={},s={},n=[];for(let o=0;o<e.length;o++){let a=e[o];if(a==="--vars"||a==="--vars-file"){let i=e[o+1];if(i===void 0||i.startsWith("--"))throw new Error(`${a} requires a value`);a==="--vars"?Object.assign(s,Bt(i)):Object.assign(r,Ht(i,t)),o+=1;continue}if(a.startsWith("--vars=")){Object.assign(s,Bt(a.slice(7)));continue}if(a.startsWith("--vars-file=")){Object.assign(r,Ht(a.slice(12),t));continue}n.push(a)}return{extracted:{...r,...s},remaining:n}}var Bs=_(()=>{"use strict";Rs();Ce()});import Hs from"node:path";function M(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function st(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 ki(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Pi(e,t){let r=e||"?",s=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${s}`:`failed after ${r} ${s}`}function rt(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 Ei(e){return`<span class="badge badge-${e}">${e}</span>`}function Ai(e){if(!e.code)return"";let t=e.code.split(`
258
+ `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((n,o)=>{let a=e.codeStartLine+o,i=a===e.codeLine,c=String(a).padStart(4);return`<span class="code-line${i?" code-line-active":""}">${c} \u2502 ${M(n)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(s=>`<span class="code-line code-line-body">${M(s)}</span>`).join("")}</pre></div>`}function $i(e){let t=e.duration!=null?`<span class="step-duration">${st(e.duration)}</span>`:"",r=rt(e.status);if(e.screenshot||e.message||e.error||e.code){let n="";e.screenshot&&(n=`<img src="${M(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":Ai(e),a="";e.error&&(a=`<div class="step-error"><pre>${M(e.error)}</pre></div>`);let i="";e.message&&!e.error&&(i=`<div class="step-message">${M(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
259
+ <details class="step-details step step-${e.status}"${c}>
260
260
  <summary class="step-header">
261
261
  ${r}
262
262
  <span class="step-id">${M(e.stepId)}</span>
@@ -278,7 +278,7 @@ statements:
278
278
  <span class="step-description">${M(e.description)}</span>
279
279
  ${t}
280
280
  </div>
281
- </div>`}function Ai(e,t,r,s){let n=[];e&&n.push(`
281
+ </div>`}function Ii(e,t,r,s){let n=[];e&&n.push(`
282
282
  <details class="artifact-section">
283
283
  <summary class="artifact-summary">Video</summary>
284
284
  <div class="artifact-content">
@@ -289,10 +289,10 @@ statements:
289
289
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
290
290
  </div>
291
291
  </div>
292
- </details>`);let o=r.filter(a=>a.screenshot).map(a=>({src:a.screenshot,stepId:a.stepId,description:a.description,status:a.status,message:a.message||a.error||""}));if(o.length>0){let a=M(JSON.stringify(o)),i=o.map((l,c)=>`
293
- <div class="screenshot-thumb" onclick="openGalleryAt(this, ${c})" data-gallery="${a}">
294
- <img src="${M(l.src)}" alt="${M(l.stepId)}" />
295
- <span class="thumb-label">${M(l.stepId)}</span>
292
+ </details>`);let o=r.filter(a=>a.screenshot).map(a=>({src:a.screenshot,stepId:a.stepId,description:a.description,status:a.status,message:a.message||a.error||""}));if(o.length>0){let a=M(JSON.stringify(o)),i=o.map((c,l)=>`
293
+ <div class="screenshot-thumb" onclick="openGalleryAt(this, ${l})" data-gallery="${a}">
294
+ <img src="${M(c.src)}" alt="${M(c.stepId)}" />
295
+ <span class="thumb-label">${M(c.stepId)}</span>
296
296
  </div>`).join("");n.push(`
297
297
  <details class="artifact-section">
298
298
  <summary class="artifact-summary">Screenshots (${o.length})</summary>
@@ -310,38 +310,38 @@ statements:
310
310
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
311
311
  <p class="trace-hint"><a href="${M(t)}" class="attachment-link" download>Download trace.zip</a></p>
312
312
  </div>
313
- </details>`)}return n.length===0?"":`<div class="test-artifacts">${n.join("")}</div>`}function Bs(e,t,r,s,n){let o=e.map(Ei).join(`
314
- `),a="";t&&!e.some(l=>l.error)&&(a=`<div class="test-error"><pre>${M(t)}</pre></div>`);let i=Ai(r,s,e,n);return`
313
+ </details>`)}return n.length===0?"":`<div class="test-artifacts">${n.join("")}</div>`}function Ws(e,t,r,s,n){let o=e.map($i).join(`
314
+ `),a="";t&&!e.some(c=>c.error)&&(a=`<div class="test-error"><pre>${M(t)}</pre></div>`);let i=Ii(r,s,e,n);return`
315
315
  ${a}
316
316
  <div class="steps-list">
317
317
  ${o||'<div class="no-steps">No YAML step details available</div>'}
318
318
  </div>
319
- ${i}`}function $i(e,t){let r=e.flaky?"flaky":e.status,s=st(r),n;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length-1,l=e.attempts.map((f,g)=>{let h=g===i,p=f.status==="passed"?"passed":"failed",b=`Attempt ${f.attemptNumber}`;return`<button class="attempt-tab ${h?"active":""} attempt-tab-${p}"
319
+ ${i}`}function Mi(e,t){let r=e.flaky?"flaky":e.status,s=rt(r),n;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length-1,c=e.attempts.map((f,g)=>{let h=g===i,p=f.status==="passed"?"passed":"failed",w=`Attempt ${f.attemptNumber}`;return`<button class="attempt-tab ${h?"active":""} attempt-tab-${p}"
320
320
  onclick="switchAttemptTab('${o}', ${g})"
321
- data-tab-index="${g}">${st(p)} ${b} <span class="attempt-tab-badge badge-${p}">${f.status}</span></button>`}).join(""),c=e.attempts.map((f,g)=>{let h=g===i,p=Bs(f.steps,f.error,f.videoPath,f.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${h?"active":""}" data-panel-index="${g}">
321
+ data-tab-index="${g}">${rt(p)} ${w} <span class="attempt-tab-badge badge-${p}">${f.status}</span></button>`}).join(""),l=e.attempts.map((f,g)=>{let h=g===i,p=Ws(f.steps,f.error,f.videoPath,f.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${h?"active":""}" data-panel-index="${g}">
322
322
  <div class="attempt-meta">
323
- ${st(f.status==="passed"?"passed":"failed")}
324
- <span class="attempt-meta-text">Attempt ${f.attemptNumber} &mdash; ${f.status} in ${nt(f.duration)}</span>
323
+ ${rt(f.status==="passed"?"passed":"failed")}
324
+ <span class="attempt-meta-text">Attempt ${f.attemptNumber} &mdash; ${f.status} in ${st(f.duration)}</span>
325
325
  </div>
326
326
  ${p}
327
- </div>`}).join(""),u=e.flaky?`Flaky &mdash; ${xi(e.retries)}`:`Retried &mdash; ${Ti(e.retries,e.status)}`;n=`
327
+ </div>`}).join(""),u=e.flaky?`Flaky &mdash; ${ki(e.retries)}`:`Retried &mdash; ${Pi(e.retries,e.status)}`;n=`
328
328
  <div class="${e.flaky?"flaky-note":"retried-note"}">${u}</div>
329
329
  <div class="attempt-tabs" id="${o}">
330
- <div class="attempt-tab-bar">${l}</div>
331
- ${c}
332
- </div>`}else n=Bs(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
330
+ <div class="attempt-tab-bar">${c}</div>
331
+ ${l}
332
+ </div>`}else n=Ws(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
333
333
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
334
334
  <summary class="test-summary test-${r}">
335
335
  ${s}
336
336
  <span class="test-title">${M(e.title)}</span>
337
337
  <span class="test-file">${M(e.file)}</span>
338
- ${Pi(r)}
339
- <span class="test-duration">${nt(e.duration)}</span>
338
+ ${Ei(r)}
339
+ <span class="test-duration">${st(e.duration)}</span>
340
340
  </summary>
341
341
  <div class="test-body">
342
342
  ${n}
343
343
  </div>
344
- </details>`}function Hs(e,t){return!e||!t||Us.isAbsolute(e)?e:Us.join(t,e)}function Ii(e,t){return t?{...e,tracePath:Hs(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:Hs(r.tracePath,t)}))}:e}function Gt(e){let t=e.tests.filter(c=>c.flaky).length,r=e.tests.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0).length,s=e.tests.filter(c=>c.status==="passed"&&!c.flaky).length,n=e.tests.filter(c=>c.status==="failed"||c.status==="timedOut").length,o=e.tests.filter(c=>c.status==="skipped").length,a=e.tests.length,l=e.tests.map(c=>Ii(c,e.outputDir)).map((c,u)=>$i(c,u)).join(`
344
+ </details>`}function Gs(e,t){return!e||!t||Hs.isAbsolute(e)?e:Hs.join(t,e)}function Oi(e,t){return t?{...e,tracePath:Gs(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:Gs(r.tracePath,t)}))}:e}function Kt(e){let t=e.tests.filter(l=>l.flaky).length,r=e.tests.filter(l=>!l.flaky&&l.retries!=null&&l.retries>0).length,s=e.tests.filter(l=>l.status==="passed"&&!l.flaky).length,n=e.tests.filter(l=>l.status==="failed"||l.status==="timedOut").length,o=e.tests.filter(l=>l.status==="skipped").length,a=e.tests.length,c=e.tests.map(l=>Oi(l,e.outputDir)).map((l,u)=>Mi(l,u)).join(`
345
345
  `);return`<!DOCTYPE html>
346
346
  <html lang="en">
347
347
  <head>
@@ -991,7 +991,7 @@ statements:
991
991
  ${r>0?`<span class="summary-stat retried">${r} retried</span>`:""}
992
992
  ${n>0?`<span class="summary-stat failed">${n} failed</span>`:""}
993
993
  ${o>0?`<span class="summary-stat skipped">${o} skipped</span>`:""}
994
- <span class="summary-stat">${nt(e.totalDuration)}</span>
994
+ <span class="summary-stat">${st(e.totalDuration)}</span>
995
995
  ${e.timestamp?`<span class="summary-stat" style="margin-left:auto;color:var(--color-text-secondary)">${new Date(e.timestamp).toLocaleString()}</span>`:""}
996
996
  </div>
997
997
  </div>${e.cacheSummary?`
@@ -1005,7 +1005,7 @@ statements:
1005
1005
  </div>
1006
1006
  </div>`:""}
1007
1007
  <div class="test-list">
1008
- ${l}
1008
+ ${c}
1009
1009
  </div>
1010
1010
  <div class="footer">
1011
1011
  Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${M(e.shiplightVersion)}`:""}
@@ -1135,9 +1135,9 @@ statements:
1135
1135
  });
1136
1136
  </script>
1137
1137
  </body>
1138
- </html>`}var Gs=_(()=>{"use strict"});import*as D from"fs";import*as Pe from"path";import{execFileSync as Mi}from"child_process";import{createHash as Oi}from"crypto";import Te from"axios";function Ws(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 pe(...e){try{return Mi("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Li(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=D.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Ri(){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??"",s=process.env.GITHUB_RUN_ID??"",n=process.env.GITHUB_EVENT_NAME??"",a=Li().pull_request,i=a?.head?.sha,l=process.env.SHIPLIGHT_GIT_SHA??i??process.env.GITHUB_SHA??"",c=process.env.GITHUB_REF??"",u=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??c.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,f=process.env.SHIPLIGHT_PR_TITLE??a?.title,g=pe("log","-1","--pretty=%s"),h=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:l,gitBranch:d,gitRepo:r,commitMessage:g,authorEmail:h,prNumber:u,prTitle:f,prUrl:u&&r?`${t}/${r}/pull/${u}`:void 0,ciBuildId:s,ciBuildUrl:s&&r?`${t}/${r}/actions/runs/${s}`:void 0,commitUrl:l&&r?`${t}/${r}/commit/${l}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:n,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",s=process.env.CI_MERGE_REQUEST_IID,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,prNumber:s,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:s&&t?`${t}/-/merge_requests/${s}`: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??"",s=process.env.CIRCLE_PROJECT_REPONAME??"",n=r&&s?`${r}/${s}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,a=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,l=i?Ws(i):void 0,c=pe("log","-1","--pretty=%s"),u=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:n,commitMessage:c,authorEmail:u,prNumber:a,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&l?`${l}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=pe("rev-parse","HEAD"),r=pe("rev-parse","--abbrev-ref","HEAD"),s=pe("log","-1","--pretty=%s"),n=pe("log","-1","--pretty=%ae"),o=pe("remote","get-url","origin"),a=o?Ws(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:s,authorEmail:n,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function Ci(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 Wt(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Ni(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 Di(e,t){let r=new Map;for(let s of t){s.screenshotS3Uris={};let n=r.get(s.testCaseName);n?n.push(s):r.set(s.testCaseName,[s])}return e.map(s=>r.get(s.title)?.shift())}function z(e,t){return Pe.isAbsolute(t)?t:Pe.join(e,t)}function Ks(e,t){let r={};for(let s of e)r[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return r}function ji(e,t,r,s){let n=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let a=o.attemptNumber-1,i=t[a]??{},l={outcome:Wt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Ks(o.steps,i),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[a],traceS3Uri:s[a],actionStepsMap:e.actionStepsMap??{}};o.error&&(l.error={message:o.error}),o.status==="timedOut"&&(l.timedOut=!0),n.push(l)}else n.push({outcome:Wt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Ks(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:s[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:Wt(e.status),flaky:e.flaky??!1,segments:n}}function zt(e){return Oi("md5").update(e).digest("base64")}function Kt(e){return zt(D.readFileSync(e))}async function Vt(e,t){let r=D.readFileSync(t),s=Pe.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await Te.put(e,r,{headers:{"Content-Type":o,"Content-MD5":zt(r)}})}async function Vs(e,t,r,s){let n=rt(s,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${s}`,"Content-Type":"application/json"},a=Ri(),i=e.tests.map(p=>{let b={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let y=z(t,p.videoPath);D.existsSync(y)&&(b.videoMd5=Kt(y))}if(p.tracePath){let y=z(t,p.tracePath);D.existsSync(y)&&(b.traceMd5=Kt(y))}return b}),l=e.tests.length;console.log(`[reporter] Uploading ${l} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let u=(await Te.post(`${n}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Di(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,b)=>{let y=d[b];if(!y)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let T of p.attempts){let $=T.attemptNumber-1,Y=T.attemptNumber===p.attempts.length;for(let L of T.steps)L.screenshot&&m.push({key:`attempt-${$}.${L.stepId}`,filePath:z(t,L.screenshot)});if(!Y){if(T.videoPath){let L=z(t,T.videoPath);D.existsSync(L)&&m.push({key:`attempt-${$}.__video__`,filePath:L})}if(T.tracePath){let L=z(t,T.tracePath);D.existsSync(L)&&m.push({key:`attempt-${$}.__trace__`,filePath:L})}}}else for(let T of p.steps)T.screenshot&&m.push({key:T.stepId,filePath:z(t,T.screenshot)});if(!m.length)return;let w=m.map(T=>T.key),E={};for(let{key:T,filePath:$}of m)D.existsSync($)&&(E[T]=Kt($));try{let T=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:w,md5s:E},{headers:o});y.uploadUrls.screenshots=T.data.screenshots,y.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${w.length} screenshot URL(s) for "${p.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,T)}})),console.log("[reporter] [3/4] Uploading assets...");let f=(await Promise.all(e.tests.map(async(p,b)=>{let y=d[b];if(!y){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=y.uploadUrls,w={},E={},T={},$=0,Y=[];if(p.attempts&&p.attempts.length>1)for(let P of p.attempts){let R=P.attemptNumber-1,gn=P.attemptNumber===p.attempts.length;for(let ee of P.steps)ee.screenshot&&Y.push({key:`attempt-${R}.${ee.stepId}`,filePath:z(t,ee.screenshot),attemptIdx:R,assetType:"screenshot",originalStepId:ee.stepId});if(!gn){if(P.videoPath){let ee=z(t,P.videoPath);D.existsSync(ee)&&Y.push({key:`attempt-${R}.__video__`,filePath:ee,attemptIdx:R,assetType:"video"})}if(P.tracePath){let ee=z(t,P.tracePath);D.existsSync(ee)&&Y.push({key:`attempt-${R}.__trace__`,filePath:ee,attemptIdx:R,assetType:"trace"})}}}else for(let P of p.steps)P.screenshot&&Y.push({key:P.stepId,filePath:z(t,P.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:P.stepId});await Promise.all(Y.map(async P=>{if(m.screenshots?.[P.key]&&D.existsSync(P.filePath))try{await Vt(m.screenshots[P.key],P.filePath);let R=y.screenshotS3Uris[P.key];switch(P.assetType){case"screenshot":w[P.attemptIdx]||(w[P.attemptIdx]={}),w[P.attemptIdx][P.originalStepId]=R;break;case"video":E[P.attemptIdx]=R;break;case"trace":T[P.attemptIdx]=R;break}$++}catch(R){console.warn(`[reporter] Asset upload failed for ${P.key}:`,R)}})),$>0&&console.log(`[reporter] [3/4] Uploaded ${$} asset(s) for "${p.title}"`);let L;if(p.videoPath&&m.video){let P=z(t,p.videoPath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await Vt(m.video,P),L=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Video upload failed:",R)}}}let ye;if(p.tracePath&&m.trace){let P=z(t,p.tracePath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await Vt(m.trace,P),ye=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Trace upload failed:",R)}}}if(p.attempts&&p.attempts.length>1){let P=p.attempts.length-1;L&&(E[P]=L),ye&&(T[P]=ye)}else L&&(E[0]=L),ye&&(T[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let dn=ji(p,w,E,T),Yt=Buffer.from(JSON.stringify(dn)),Jt=zt(Yt),Xt=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:Jt},{headers:o}),fn=Xt.data.reportUrl,hn=Xt.data.reportS3Uri;return await Te.put(fn,Yt,{headers:{"Content-Type":"application/json","Content-MD5":Jt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:y.testCaseResultId,result:Ci(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:hn,videoS3Uri:L,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=Ni(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let h=await Te.put(`${n}/v1/local-runs/${u.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:f},{headers:o});console.log(`
1139
- Shiplight cloud report: ${Fi(h.data.reportUrl,n)}`)}function Fi(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${Ui(t)}${r}`}function Ui(e){let t=e.endsWith("/")?e.slice(0,-1):e;return t==="https://api.shiplight.ai"?"https://app.shiplight.ai":t==="https://nova-api.shiplight.ai"?"https://nova.shiplight.ai":t}var zs=_(()=>{"use strict";Ct()});var Qs={};ie(Qs,{buildGitHubSummary:()=>Zs,isReportToCloudEnabled:()=>Xs,runReport:()=>Bi});import*as I from"fs";import*as A from"path";async function Bi(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"),s=e.includes("--github-summary");!r&&s&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Gi(e,t,s):await Hi(e,t)}async function Hi(e,t){let r=e.find(i=>!i.startsWith("--"))||"shiplight-report",s=A.isAbsolute(r)?r:A.join(process.cwd(),r),n=A.join(s,"report-data.json");I.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(I.readFileSync(n,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${n}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=A.join(s,"index.html");if(I.writeFileSync(a,Gt({...o,outputDir:s}),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await qs(o,s),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function Gi(e,t,r){let s=A.join(process.cwd(),"shiplight-report"),n=e.findIndex(g=>g==="-o"||g==="--output");if(n!==-1&&e[n+1]){let g=e[n+1];s=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let g=0;g<e.length;g++){let h=e[g];if(o.has(h)){g++;continue}if(a.has(h))continue;let p=A.isAbsolute(h)?h:A.join(process.cwd(),h);i.push(p)}i.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 l=[],c=0,u=0;I.mkdirSync(A.join(s,"screenshots"),{recursive:!0});for(let g=0;g<i.length;g++){let h=i[g],p=`shard-${g}`,b=A.join(h,"report-data.json");if(!I.existsSync(b)){console.warn(`Warning: No report-data.json found in ${h}, skipping.`);continue}let y;try{y=JSON.parse(I.readFileSync(b,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${b}, skipping.`);continue}console.log(`Merging ${p}: ${y.tests.length} tests from ${h}`),c+=y.totalDuration||0;let m=A.join(h,"screenshots");I.existsSync(m)&&Js(m,A.join(s,"screenshots",p));let w=A.join(s,p);for(let E of y.tests){let T=[E,...E.attempts||[]];for(let $ of T)Wi($.steps,p),$.videoPath&&Ys(h,$.videoPath,w)&&($.videoPath=`${p}/${$.videoPath}`),$.tracePath&&Ys(h,$.tracePath,w)&&($.tracePath=`${p}/${$.tracePath}`);l.push(E)}u++}l.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:l,totalDuration:c,timestamp:new Date().toISOString(),shiplightVersion:ke};I.writeFileSync(A.join(s,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let f=A.join(s,"index.html");if(I.writeFileSync(f,Gt({...d,outputDir:s}),"utf-8"),console.log(`
1140
- Merged ${l.length} tests from ${u} shards into: ${f}`),await qs(d,s),r&&Ki(l),t)try{let g=(await import("open")).default;await g(f)}catch{}}function Ys(e,t,r){let s=A.resolve(e,t);return s.startsWith(A.resolve(e)+A.sep)?I.existsSync(s)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(s,A.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Js(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let s=A.join(e,r.name),n=A.join(t,r.name);r.isDirectory()?Js(s,n):I.copyFileSync(s,n)}}function Wi(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function Xs(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function qs(e,t){if(!Xs())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let s=e.tests.map(o=>o.startTime).filter(o=>!!o),n=s.length>0?s.sort()[0]:e.timestamp??new Date().toISOString();try{await Vs(e,t,n,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ot(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:r}}function Zs(e){let t=e.filter(l=>!l.file.includes("auth.setup")),r=t.filter(l=>l.flaky),s=t.filter(l=>!l.flaky&&l.retries!=null&&l.retries>0),n=t.filter(l=>l.status==="passed"&&!l.flaky),o=t.filter(l=>l.status!=="passed"),a=t.length,i=`## Test Results
1138
+ </html>`}var Ks=_(()=>{"use strict"});import*as F from"fs";import*as ke from"path";import{execFileSync as Li}from"child_process";import{createHash as Ri}from"crypto";import Te from"axios";function Vs(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 pe(...e){try{return Li("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Ci(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=F.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Ni(){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??"",s=process.env.GITHUB_RUN_ID??"",n=process.env.GITHUB_EVENT_NAME??"",a=Ci().pull_request,i=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??i??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",u=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,f=process.env.SHIPLIGHT_PR_TITLE??a?.title,g=pe("log","-1","--pretty=%s"),h=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:d,gitRepo:r,commitMessage:g,authorEmail:h,prNumber:u,prTitle:f,prUrl:u&&r?`${t}/${r}/pull/${u}`:void 0,ciBuildId:s,ciBuildUrl:s&&r?`${t}/${r}/actions/runs/${s}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:n,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",s=process.env.CI_MERGE_REQUEST_IID,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,prNumber:s,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:s&&t?`${t}/-/merge_requests/${s}`: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??"",s=process.env.CIRCLE_PROJECT_REPONAME??"",n=r&&s?`${r}/${s}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,a=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,c=i?Vs(i):void 0,l=pe("log","-1","--pretty=%s"),u=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:n,commitMessage:l,authorEmail:u,prNumber:a,prUrl:o,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=pe("rev-parse","HEAD"),r=pe("rev-parse","--abbrev-ref","HEAD"),s=pe("log","-1","--pretty=%s"),n=pe("log","-1","--pretty=%ae"),o=pe("remote","get-url","origin"),a=o?Vs(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:s,authorEmail:n,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function Di(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 Vt(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 Fi(e,t){let r=new Map;for(let s of t){s.screenshotS3Uris={};let n=r.get(s.testCaseName);n?n.push(s):r.set(s.testCaseName,[s])}return e.map(s=>r.get(s.title)?.shift())}function z(e,t){return ke.isAbsolute(t)?t:ke.join(e,t)}function zs(e,t){let r={};for(let s of e)r[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return r}function Ui(e,t,r,s){let n=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let a=o.attemptNumber-1,i=t[a]??{},c={outcome:Vt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:zs(o.steps,i),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[a],traceS3Uri:s[a],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),n.push(c)}else n.push({outcome:Vt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:zs(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:s[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:Vt(e.status),flaky:e.flaky??!1,segments:n}}function Jt(e){return Ri("md5").update(e).digest("base64")}function zt(e){return Jt(F.readFileSync(e))}async function Yt(e,t){let r=F.readFileSync(t),s=ke.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await Te.put(e,r,{headers:{"Content-Type":o,"Content-MD5":Jt(r)}})}async function Ys(e,t,r,s){let n=tt(s,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${s}`,"Content-Type":"application/json"},a=Ni(),i=e.tests.map(p=>{let w={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let y=z(t,p.videoPath);F.existsSync(y)&&(w.videoMd5=zt(y))}if(p.tracePath){let y=z(t,p.tracePath);F.existsSync(y)&&(w.traceMd5=zt(y))}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 u=(await Te.post(`${n}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Fi(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,w)=>{let y=d[w];if(!y)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let T of p.attempts){let $=T.attemptNumber-1,Y=T.attemptNumber===p.attempts.length;for(let L of T.steps)L.screenshot&&m.push({key:`attempt-${$}.${L.stepId}`,filePath:z(t,L.screenshot)});if(!Y){if(T.videoPath){let L=z(t,T.videoPath);F.existsSync(L)&&m.push({key:`attempt-${$}.__video__`,filePath:L})}if(T.tracePath){let L=z(t,T.tracePath);F.existsSync(L)&&m.push({key:`attempt-${$}.__trace__`,filePath:L})}}}else for(let T of p.steps)T.screenshot&&m.push({key:T.stepId,filePath:z(t,T.screenshot)});if(!m.length)return;let b=m.map(T=>T.key),E={};for(let{key:T,filePath:$}of m)F.existsSync($)&&(E[T]=zt($));try{let T=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:b,md5s:E},{headers:o});y.uploadUrls.screenshots=T.data.screenshots,y.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${b.length} screenshot URL(s) for "${p.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,T)}})),console.log("[reporter] [3/4] Uploading assets...");let f=(await Promise.all(e.tests.map(async(p,w)=>{let y=d[w];if(!y){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=y.uploadUrls,b={},E={},T={},$=0,Y=[];if(p.attempts&&p.attempts.length>1)for(let k of p.attempts){let C=k.attemptNumber-1,yn=k.attemptNumber===p.attempts.length;for(let ee of k.steps)ee.screenshot&&Y.push({key:`attempt-${C}.${ee.stepId}`,filePath:z(t,ee.screenshot),attemptIdx:C,assetType:"screenshot",originalStepId:ee.stepId});if(!yn){if(k.videoPath){let ee=z(t,k.videoPath);F.existsSync(ee)&&Y.push({key:`attempt-${C}.__video__`,filePath:ee,attemptIdx:C,assetType:"video"})}if(k.tracePath){let ee=z(t,k.tracePath);F.existsSync(ee)&&Y.push({key:`attempt-${C}.__trace__`,filePath:ee,attemptIdx:C,assetType:"trace"})}}}else for(let k of p.steps)k.screenshot&&Y.push({key:k.stepId,filePath:z(t,k.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:k.stepId});await Promise.all(Y.map(async k=>{if(m.screenshots?.[k.key]&&F.existsSync(k.filePath))try{await Yt(m.screenshots[k.key],k.filePath);let C=y.screenshotS3Uris[k.key];switch(k.assetType){case"screenshot":b[k.attemptIdx]||(b[k.attemptIdx]={}),b[k.attemptIdx][k.originalStepId]=C;break;case"video":E[k.attemptIdx]=C;break;case"trace":T[k.attemptIdx]=C;break}$++}catch(C){console.warn(`[reporter] Asset upload failed for ${k.key}:`,C)}})),$>0&&console.log(`[reporter] [3/4] Uploaded ${$} asset(s) for "${p.title}"`);let L;if(p.videoPath&&m.video){let k=z(t,p.videoPath);if(F.existsSync(k)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await Yt(m.video,k),L=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(C){console.warn("[reporter] Video upload failed:",C)}}}let ye;if(p.tracePath&&m.trace){let k=z(t,p.tracePath);if(F.existsSync(k)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await Yt(m.trace,k),ye=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(C){console.warn("[reporter] Trace upload failed:",C)}}}if(p.attempts&&p.attempts.length>1){let k=p.attempts.length-1;L&&(E[k]=L),ye&&(T[k]=ye)}else L&&(E[0]=L),ye&&(T[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let hn=Ui(p,b,E,T),Xt=Buffer.from(JSON.stringify(hn)),qt=Jt(Xt),Zt=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:qt},{headers:o}),gn=Zt.data.reportUrl,mn=Zt.data.reportS3Uri;return await Te.put(gn,Xt,{headers:{"Content-Type":"application/json","Content-MD5":qt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:y.testCaseResultId,result:Di(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:mn,videoS3Uri:L,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=ji(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let h=await Te.put(`${n}/v1/local-runs/${u.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:f},{headers:o});console.log(`
1139
+ Shiplight cloud report: ${Bi(h.data.reportUrl,n)}`)}function Bi(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${Hi(t)}${r}`}function Hi(e){let t=e.endsWith("/")?e.slice(0,-1):e;return t==="https://api.shiplight.ai"?"https://app.shiplight.ai":t==="https://nova-api.shiplight.ai"?"https://nova.shiplight.ai":t}var Js=_(()=>{"use strict";Ct()});var tn={};ie(tn,{buildGitHubSummary:()=>en,isReportToCloudEnabled:()=>Zs,runReport:()=>Wi});import*as I from"fs";import*as A from"path";async function Wi(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"),s=e.includes("--github-summary");!r&&s&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Ki(e,t,s):await Gi(e,t)}async function Gi(e,t){let r=e.find(i=>!i.startsWith("--"))||"shiplight-report",s=A.isAbsolute(r)?r:A.join(process.cwd(),r),n=A.join(s,"report-data.json");I.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(I.readFileSync(n,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${n}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=A.join(s,"index.html");if(I.writeFileSync(a,Kt({...o,outputDir:s}),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await Qs(o,s),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function Ki(e,t,r){let s=A.join(process.cwd(),"shiplight-report"),n=e.findIndex(g=>g==="-o"||g==="--output");if(n!==-1&&e[n+1]){let g=e[n+1];s=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let g=0;g<e.length;g++){let h=e[g];if(o.has(h)){g++;continue}if(a.has(h))continue;let p=A.isAbsolute(h)?h:A.join(process.cwd(),h);i.push(p)}i.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,u=0;I.mkdirSync(A.join(s,"screenshots"),{recursive:!0});for(let g=0;g<i.length;g++){let h=i[g],p=`shard-${g}`,w=A.join(h,"report-data.json");if(!I.existsSync(w)){console.warn(`Warning: No report-data.json found in ${h}, skipping.`);continue}let y;try{y=JSON.parse(I.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${p}: ${y.tests.length} tests from ${h}`),l+=y.totalDuration||0;let m=A.join(h,"screenshots");I.existsSync(m)&&qs(m,A.join(s,"screenshots",p));let b=A.join(s,p);for(let E of y.tests){let T=[E,...E.attempts||[]];for(let $ of T)Vi($.steps,p),$.videoPath&&Xs(h,$.videoPath,b)&&($.videoPath=`${p}/${$.videoPath}`),$.tracePath&&Xs(h,$.tracePath,b)&&($.tracePath=`${p}/${$.tracePath}`);c.push(E)}u++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:Pe};I.writeFileSync(A.join(s,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let f=A.join(s,"index.html");if(I.writeFileSync(f,Kt({...d,outputDir:s}),"utf-8"),console.log(`
1140
+ Merged ${c.length} tests from ${u} shards into: ${f}`),await Qs(d,s),r&&zi(c),t)try{let g=(await import("open")).default;await g(f)}catch{}}function Xs(e,t,r){let s=A.resolve(e,t);return s.startsWith(A.resolve(e)+A.sep)?I.existsSync(s)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(s,A.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function qs(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let s=A.join(e,r.name),n=A.join(t,r.name);r.isDirectory()?qs(s,n):I.copyFileSync(s,n)}}function Vi(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function Zs(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function Qs(e,t){if(!Zs())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let s=e.tests.map(o=>o.startTime).filter(o=>!!o),n=s.length>0?s.sort()[0]:e.timestamp??new Date().toISOString();try{await Ys(e,t,n,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function nt(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:r}}function en(e){let t=e.filter(c=>!c.file.includes("auth.setup")),r=t.filter(c=>c.flaky),s=t.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0),n=t.filter(c=>c.status==="passed"&&!c.flaky),o=t.filter(c=>c.status!=="passed"),a=t.length,i=`## Test Results
1141
1141
 
1142
1142
  `;if(o.length===0&&r.length===0?i+=`\u2705 All ${a} tests passed
1143
1143
 
@@ -1145,30 +1145,30 @@ Merged ${l.length} tests from ${u} shards into: ${f}`),await qs(d,s),r&&Ki(l),t)
1145
1145
 
1146
1146
  `:i+=`\u274C ${o.length} failed, \u26A0\uFE0F ${r.length} flaky, \u2705 ${n.length} passed / ${a} total
1147
1147
 
1148
- `,o.length>0){let l=[...new Set(o.map(c=>ot(c).yamlPath))];i+=`### Failed
1148
+ `,o.length>0){let c=[...new Set(o.map(l=>nt(l).yamlPath))];i+=`### Failed
1149
1149
 
1150
- `;for(let c of l)i+=`- \`npx shiplight test ${c}\`
1150
+ `;for(let l of c)i+=`- \`npx shiplight test ${l}\`
1151
1151
  `;i+=`
1152
1152
  **Run all failed tests**
1153
1153
 
1154
- `,i+="```sh\n",i+=`npx shiplight test ${l.map(c=>`"${c}"`).join(` \\
1154
+ `,i+="```sh\n",i+=`npx shiplight test ${c.map(l=>`"${l}"`).join(` \\
1155
1155
  `)}
1156
- `,i+="```\n\n"}if(r.length>0){let l=[...new Set(r.map(c=>ot(c).yamlPath))];i+=`### Flaky (${l.length})
1156
+ `,i+="```\n\n"}if(r.length>0){let c=[...new Set(r.map(l=>nt(l).yamlPath))];i+=`### Flaky (${c.length})
1157
1157
 
1158
- `;for(let c of l)i+=`- \`${c}\`
1158
+ `;for(let l of c)i+=`- \`${l}\`
1159
1159
  `;i+=`
1160
- `}if(s.length>0){let l=[...new Set(s.map(c=>ot(c).yamlPath))];i+=`### Retried (${l.length})
1160
+ `}if(s.length>0){let c=[...new Set(s.map(l=>nt(l).yamlPath))];i+=`### Retried (${c.length})
1161
1161
 
1162
- `;for(let c of l)i+=`- \`${c}\`
1162
+ `;for(let l of c)i+=`- \`${l}\`
1163
1163
  `;i+=`
1164
- `}if(n.length>0){let l=[...new Set(n.map(c=>ot(c).yamlPath))];i+=`<details><summary>Passed (${l.length})</summary>
1164
+ `}if(n.length>0){let c=[...new Set(n.map(l=>nt(l).yamlPath))];i+=`<details><summary>Passed (${c.length})</summary>
1165
1165
 
1166
- `;for(let c of l)i+=`- ${c}
1166
+ `;for(let l of c)i+=`- ${l}
1167
1167
  `;i+=`
1168
1168
  </details>
1169
- `}return i}function Ki(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,Zs(e)),console.log("GitHub step summary written.")}var en=_(()=>{"use strict";Gs();zs();Ne()});var tn,rn=_(()=>{"use strict";tn="0.1.72"});var sn={};ie(sn,{runTranspile:()=>zi});import*as it from"path";import{glob as Vi}from"glob";async function zi(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(),s=await Vi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,a=0;for(let i of s.sort()){let l=it.resolve(r,i),c=us(l,{version:tn});if(!c.valid){n++,console.log(`
1170
- \u2717 ${i}`);for(let d of c.errors)console.log(` ERROR: ${d}`);continue}a++;let u=it.basename(c.specFile);if(c.warnings.length>0){o++,console.log(`\u26A0 ${i} \u2192 ${u}`);for(let d of c.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${i} \u2192 ${u}`)}console.log(`
1171
- ${s.length} file(s): ${a} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var nn=_(()=>{"use strict";$t();rn()});var cn={};ie(cn,{runInspect:()=>Yi});import*as at from"fs";import*as on from"path";async function Yi(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"),s=e.find(a=>!a.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=on.resolve(process.cwd(),s);at.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=at.readFileSync(n,"utf-8");try{let a=Ie(o),i=j(o);if(r)Ji(i,a);else{let l={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(l,null,t?0:2))}}catch(a){console.error(`Error parsing ${s}: ${a.message}`),process.exit(1)}}function Ji(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 s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}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=an(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function an(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 s=an(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var ln=_(()=>{"use strict";de()});var pn=yn((qp,Xi)=>{Xi.exports={name:"shiplightai",version:"0.1.72",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"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},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","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.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","@shiplightai/devtools-assets":"workspace:*",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.60.0","@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.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Ne();import qi from"dotenv";qi.config();qt();Zt();function un(){console.log(`
1169
+ `}return i}function zi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,en(e)),console.log("GitHub step summary written.")}var rn=_(()=>{"use strict";Ks();Js();Ce()});var sn,nn=_(()=>{"use strict";sn="0.1.73"});var on={};ie(on,{runTranspile:()=>Ji});import*as ot from"path";import{glob as Yi}from"glob";async function Ji(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(),s=await Yi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,a=0;for(let i of s.sort()){let c=ot.resolve(r,i),l=ds(c,{version:sn});if(!l.valid){n++,console.log(`
1170
+ \u2717 ${i}`);for(let d of l.errors)console.log(` ERROR: ${d}`);continue}a++;let u=ot.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${i} \u2192 ${u}`);for(let d of l.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${i} \u2192 ${u}`)}console.log(`
1171
+ ${s.length} file(s): ${a} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var an=_(()=>{"use strict";$t();nn()});var pn={};ie(pn,{runInspect:()=>Xi});import*as it from"fs";import*as cn from"path";async function Xi(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"),s=e.find(a=>!a.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=cn.resolve(process.cwd(),s);it.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=it.readFileSync(n,"utf-8");try{let a=$e(o),i=W(o);if(r)qi(i,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${s}: ${a.message}`),process.exit(1)}}function qi(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 s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}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=ln(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function ln(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 s=ln(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var un=_(()=>{"use strict";de()});var dn=bn((Qp,Zi)=>{Zi.exports={name:"shiplightai",version:"0.1.73",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"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},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","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.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","@shiplightai/devtools-assets":"workspace:*",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.60.0","@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.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Ce();import Qi from"dotenv";Qi.config();Qt();er();function fn(){console.log(`
1172
1172
  Usage: shiplight <command> [options]
1173
1173
 
1174
1174
  Commands:
@@ -1190,5 +1190,5 @@ Examples:
1190
1190
  shiplight transpile
1191
1191
  shiplight transpile "tests/**/*.test.yaml"
1192
1192
  shiplight debug tests/login.test.yaml
1193
- `)}var ct=process.argv[2];switch(ct){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(rr(),tr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Is(),$s));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Fs(),js));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(en(),Qs));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(nn(),sn));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(ln(),cn));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=pn().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":un();break;default:ct&&console.error(`Unknown command: ${ct}
1194
- `),un(),process.exit(ct?1:0)}
1193
+ `)}var at=process.argv[2];switch(at){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(nr(),sr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Ms(),Is));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Bs(),Us));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(rn(),tn));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(an(),on));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(un(),pn));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=dn().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":fn();break;default:at&&console.error(`Unknown command: ${at}
1194
+ `),fn(),process.exit(at?1:0)}