shiplightai 0.1.62 → 0.1.64

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 is=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var as=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ee=(e,t)=>{for(var n in t)is(e,n,{get:t[n],enumerable:!0})};import*as O from"fs";import*as W from"path";import*as it from"os";import{execFileSync as cs}from"child_process";function at(){return W.join(it.homedir(),".shiplight","version-check.json")}function us(){return W.join(it.homedir(),".shiplight","npm-prefix.json")}function hs(e=us()){try{let n=O.readFileSync(e,"utf-8"),r=JSON.parse(n);if(typeof r.prefix=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<ds)return r.prefix}catch{}let t;try{t=cs("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{O.mkdirSync(W.dirname(e),{recursive:!0}),O.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Vt(e={}){let t=e.scriptPath??process.argv[1],n=e.getPrefix??(()=>hs()),r=e.warn??(i=>console.warn(i)),s=e.error??(i=>console.error(i)),o=e.exit??(i=>process.exit(i));try{if(!t)return;let i=n();if(!i)return;let a,l;try{a=O.realpathSync(t),l=O.realpathSync(i)}catch{return}if(a.startsWith(l+W.sep)){s(`
4
+ var cs=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var ls=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),te=(e,t)=>{for(var n in t)cs(e,n,{get:t[n],enumerable:!0})};import*as O from"fs";import*as W from"path";import*as at from"os";import{execFileSync as ps}from"child_process";function ct(){return W.join(at.homedir(),".shiplight","version-check.json")}function fs(){return W.join(at.homedir(),".shiplight","npm-prefix.json")}function gs(e=fs()){try{let n=O.readFileSync(e,"utf-8"),r=JSON.parse(n);if(typeof r.prefix=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<hs)return r.prefix}catch{}let t;try{t=ps("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{O.mkdirSync(W.dirname(e),{recursive:!0}),O.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Vt(e={}){let t=e.scriptPath??process.argv[1],n=e.getPrefix??(()=>gs()),r=e.warn??(i=>console.warn(i)),s=e.error??(i=>console.error(i)),o=e.exit??(i=>process.exit(i));try{if(!t)return;let i=n();if(!i)return;let a,l;try{a=O.realpathSync(t),l=O.realpathSync(i)}catch{return}if(a.startsWith(l+W.sep)){s(`
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=W.join(l,"lib","node_modules","shiplightai"),d=W.join(l,"node_modules","shiplightai"),u=O.existsSync(c)?c:O.existsSync(d)?d:null;u&&r(`
14
- \x1B[33m\u26A0 A global shiplightai install was detected at ${u}.
13
+ `),o(1);return}let c=W.join(l,"lib","node_modules","shiplightai"),u=W.join(l,"node_modules","shiplightai"),d=O.existsSync(c)?c:O.existsSync(u)?u:null;d&&r(`
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 fs(e=at()){try{let t=O.readFileSync(e,"utf-8"),n=JSON.parse(t);if(typeof n.latest=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<ps)return n}catch{}return null}function gs(e,t=at()){try{O.mkdirSync(W.dirname(t),{recursive:!0}),O.writeFileSync(t,JSON.stringify(e))}catch{}}async function ms(){try{let e=await fetch(ls,{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 ys(e,t){let n=u=>{let h=u.indexOf("-");return h===-1?[u,!1]:[u.slice(0,h),!0]},r=u=>u.split(".").map(h=>parseInt(h,10)||0),[s,o]=n(e),[i,a]=n(t),l=r(s),c=r(i),d=Math.max(l.length,c.length);for(let u=0;u<d;u++){let h=l[u]??0,g=c[u]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!a)}async function zt(e={}){let t=e.runningVersion??ot,n=e.cwd??process.cwd(),r=e.cacheFile??at(),s=e.fetchLatest??ms,o=e.env??process.env,i=e.warn??(c=>console.warn(c));if(o.CI||t==="dev"||!O.existsSync(W.join(n,"package-lock.json")))return;let a=null,l=fs(r);if(l)a=l.latest;else{try{a=await s()}catch{a=null}a&&gs({latest:a,fetchedAt:Date.now()},r)}a&&ys(t,a)&&i(`
17
+ `)}catch{}}function ms(e=ct()){try{let t=O.readFileSync(e,"utf-8"),n=JSON.parse(t);if(typeof n.latest=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<ds)return n}catch{}return null}function ys(e,t=ct()){try{O.mkdirSync(W.dirname(t),{recursive:!0}),O.writeFileSync(t,JSON.stringify(e))}catch{}}async function ws(){try{let e=await fetch(us,{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 bs(e,t){let n=d=>{let f=d.indexOf("-");return f===-1?[d,!1]:[d.slice(0,f),!0]},r=d=>d.split(".").map(f=>parseInt(f,10)||0),[s,o]=n(e),[i,a]=n(t),l=r(s),c=r(i),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&&!a)}async function Yt(e={}){let t=e.runningVersion??it,n=e.cwd??process.cwd(),r=e.cacheFile??ct(),s=e.fetchLatest??ws,o=e.env??process.env,i=e.warn??(c=>console.warn(c));if(o.CI||t==="dev"||!O.existsSync(W.join(n,"package-lock.json")))return;let a=null,l=ms(r);if(l)a=l.latest;else{try{a=await s()}catch{a=null}a&&ys({latest:a,fetchedAt:Date.now()},r)}a&&bs(t,a)&&i(`
18
18
  \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var ot,Se,ls,ps,ds,$e=_(()=>{"use strict";ot="0.1.62",Se=ot!=="dev"?ot:void 0,ls="https://registry.npmjs.org/shiplightai/latest",ps=3600*1e3,ds=10080*60*1e3});import*as te from"fs";import*as ie from"path";function ct(e){let{projectPath:t}=e,n=e.projectName??ie.basename(ie.resolve(t));if(te.existsSync(t)&&te.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);te.mkdirSync(t,{recursive:!0});let r=(s,o)=>{let i=ie.join(t,s);te.mkdirSync(ie.dirname(i),{recursive:!0}),te.writeFileSync(i,o)};return r("package.json",ws.replace("{{name}}",n)),r("playwright.config.ts",bs),r(".gitignore",vs),r(".env.example",Ss),r(".mcp.json",_s),r(".claude/CLAUDE.md",xs),r(".claude/auth.md",Ts),r(".claude/creating-tests.md",ks),r(".claude/settings.local.json",Ps),r(".claude/test-spec-template.md",As),r(".claude/updating-tests.md",Es),r("auth/example.login.ts",$s),r("environments/example.env.yaml",Ms),r("tests/example.test.yaml",Is),{projectPath:t,projectName:n,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json",".claude/CLAUDE.md",".claude/auth.md",".claude/creating-tests.md",".claude/settings.local.json",".claude/test-spec-template.md",".claude/updating-tests.md","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var ws,bs,vs,Ss,_s,xs,Ts,ks,Ps,As,Es,$s,Ms,Is,Yt=_(()=>{"use strict";ws=`{
20
+ `)}var it,_e,us,ds,hs,Le=_(()=>{"use strict";it="0.1.64",_e=it!=="dev"?it:void 0,us="https://registry.npmjs.org/shiplightai/latest",ds=3600*1e3,hs=10080*60*1e3});import*as ne from"fs";import*as ae from"path";function lt(e){let{projectPath:t}=e,n=e.projectName??ae.basename(ae.resolve(t));if(ne.existsSync(t)&&ne.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);ne.mkdirSync(t,{recursive:!0});let r=(s,o)=>{let i=ae.join(t,s);ne.mkdirSync(ae.dirname(i),{recursive:!0}),ne.writeFileSync(i,o)};return r("package.json",vs.replace("{{name}}",n)),r("playwright.config.ts",Ss),r(".gitignore",_s),r(".env.example",xs),r(".mcp.json",Ts),r(".claude/CLAUDE.md",ks),r(".claude/auth.md",Ps),r(".claude/creating-tests.md",As),r(".claude/settings.local.json",Es),r(".claude/test-spec-template.md",$s),r(".claude/updating-tests.md",Ms),r("auth/example.login.ts",Is),r("environments/example.env.yaml",Os),r("tests/example.test.yaml",Ls),{projectPath:t,projectName:n,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json",".claude/CLAUDE.md",".claude/auth.md",".claude/creating-tests.md",".claude/settings.local.json",".claude/test-spec-template.md",".claude/updating-tests.md","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var vs,Ss,_s,xs,Ts,ks,Ps,As,Es,$s,Ms,Is,Os,Ls,Jt=_(()=>{"use strict";vs=`{
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
- `,bs=`import { defineConfig, shiplightConfig } from 'shiplightai';
32
+ `,Ss=`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
- `,vs=`node_modules/
50
+ `,_s=`node_modules/
51
51
  test-results/
52
52
  shiplight-report/
53
53
  .shiplight/
54
54
  .env
55
55
  *.yaml.spec.ts
56
- .auth`,Ss=`# Shiplight API token (optional) \u2014 enables cloud sync tools
56
+ .auth`,xs=`# 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
- `,_s=`{
80
+ `,Ts=`{
81
81
  "mcpServers": {
82
82
  "shiplight": {
83
83
  "command": "npx",
@@ -90,7 +90,7 @@ shiplight-report/
90
90
  }
91
91
  }
92
92
  }
93
- }`,xs=`# Shiplight Test Project
93
+ }`,ks=`# Shiplight Test Project
94
94
 
95
95
  This is a Shiplight E2E test project.
96
96
 
@@ -164,7 +164,7 @@ When sources disagree, this precedence applies:
164
164
  6. Agent inference
165
165
 
166
166
  If current app behavior conflicts with the test spec or test goal, report the mismatch \u2014 do not silently rewrite the test.
167
- `,Ts=`# Auth
167
+ `,Ps=`# Auth
168
168
 
169
169
  This guide explains how to reason about authentication, how accounts are defined in environment files, and how to write auth login modules for tests that require login.
170
170
 
@@ -280,7 +280,7 @@ Never commit real credentials to specs, tests, fixtures, environment files, or d
280
280
  - one-time codes
281
281
 
282
282
  Credentials belong in \`.env\`. Environment YAML files contain only env var names, not values. The \`.env\` file is gitignored and must not be edited unless the user explicitly asks.
283
- `,ks=`# Creating Tests
283
+ `,As=`# Creating Tests
284
284
 
285
285
  When asked to create a test, follow this workflow in order. Do not skip steps.
286
286
 
@@ -391,13 +391,13 @@ Do not close the task until this step is complete.
391
391
 
392
392
  - Update the spec status to \`Implemented\`.
393
393
  - Add the test file path to the \`Implementation\` section of the spec.
394
- `,Ps=`{
394
+ `,Es=`{
395
395
  "enableAllProjectMcpServers": true,
396
396
  "enabledMcpjsonServers": [
397
397
  "shiplight"
398
398
  ]
399
399
  }
400
- `,As=`# Test Spec: <Name>
400
+ `,$s=`# Test Spec: <Name>
401
401
 
402
402
  ## Status
403
403
 
@@ -459,7 +459,7 @@ Describe the user, account type, permission level, or auth state required.
459
459
  ## Implementation
460
460
 
461
461
  - Test file:
462
- `,Es=`# Updating Tests
462
+ `,Ms=`# Updating Tests
463
463
 
464
464
  Use this workflow when changing, debugging, or repairing existing tests.
465
465
 
@@ -521,7 +521,7 @@ After completing update work, report:
521
521
  - Behavior covered.
522
522
  - Command run and pass/fail result.
523
523
  - Any product/spec mismatch or unresolved blocker.
524
- `,$s=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
524
+ `,Is=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
525
525
  import path from 'path';
526
526
  import fs from 'fs/promises';
527
527
  import { chromium } from '@playwright/test';
@@ -563,13 +563,13 @@ export async function login(args: Record<string, unknown> = {}): Promise<string>
563
563
 
564
564
  return sessionPath;
565
565
  }
566
- `,Ms=`# This is an example of environment. Don't use it.
566
+ `,Os=`# This is an example of environment. Don't use it.
567
567
  name: example
568
568
  url: https://staging.example.com
569
569
  accounts:
570
570
  - username: test-user-1@example.com
571
571
  password: EXAMPLE_TEST_USER_1_PASSWORD
572
- 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,Is=`goal: Verify the Shiplight homepage links to the quick-start docs
572
+ 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,Ls=`goal: Verify the Shiplight homepage links to the quick-start docs
573
573
  base_url: https://www.shiplight.ai
574
574
  statements:
575
575
  - URL: /
@@ -583,7 +583,7 @@ statements:
583
583
 
584
584
  - VERIFY: The browser lands on the docs site
585
585
  js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
586
- `});var Jt=_(()=>{"use strict";Yt()});var qt={};ee(qt,{runCreate:()=>Os});import*as Ie from"path";import*as Oe from"fs";function Me(){console.log(`
586
+ `});var qt=_(()=>{"use strict";Jt()});var Xt={};te(Xt,{runCreate:()=>Rs});import*as Ce from"path";import*as Ne from"fs";function Re(){console.log(`
587
587
  Usage: shiplight create <path> [options]
588
588
 
589
589
  Scaffold a new Shiplight test project at <path>. Creates package.json,
@@ -596,22 +596,22 @@ Options:
596
596
  Examples:
597
597
  shiplight create ./my-tests
598
598
  shiplight create ./my-tests --name acme-e2e
599
- `)}async function Os(e){let t,n;for(let o=0;o<e.length;o++){let i=e[o];if(i==="--help"||i==="-h"){Me();return}else i==="--name"?(n=e[++o],(!n||!n.trim())&&(console.error("Error: --name requires a non-empty value"),process.exit(1)),/^[a-z0-9][a-z0-9._-]*$/.test(n)||(console.error(`Error: --name "${n}" is not a valid package name.
599
+ `)}async function Rs(e){let t,n;for(let o=0;o<e.length;o++){let i=e[o];if(i==="--help"||i==="-h"){Re();return}else i==="--name"?(n=e[++o],(!n||!n.trim())&&(console.error("Error: --name requires a non-empty value"),process.exit(1)),/^[a-z0-9][a-z0-9._-]*$/.test(n)||(console.error(`Error: --name "${n}" is not a valid package name.
600
600
  npm package names must be lowercase.
601
601
  Use only lowercase letters, digits, hyphens, underscores, and dots;
602
- must start with a lowercase letter or digit.`),process.exit(1))):i.startsWith("--")?(console.error(`Error: Unknown option: ${i}`),Me(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${i}`),Me(),process.exit(1)):t=i}t||(console.error(`Error: missing required <path> argument
603
- `),Me(),process.exit(1));let r=Ie.resolve(t);Oe.existsSync(r)&&Oe.readdirSync(r).length>0&&(console.error(`Error: target directory is not empty: ${r}
604
- Remove it or choose a different path.`),process.exit(1));let s=ct({projectPath:r,projectName:n});console.log(`
602
+ must start with a lowercase letter or digit.`),process.exit(1))):i.startsWith("--")?(console.error(`Error: Unknown option: ${i}`),Re(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${i}`),Re(),process.exit(1)):t=i}t||(console.error(`Error: missing required <path> argument
603
+ `),Re(),process.exit(1));let r=Ce.resolve(t);Ne.existsSync(r)&&Ne.readdirSync(r).length>0&&(console.error(`Error: target directory is not empty: ${r}
604
+ Remove it or choose a different path.`),process.exit(1));let s=lt({projectPath:r,projectName:n});console.log(`
605
605
  Created Shiplight test project at ${s.projectPath}
606
606
  `),console.log(` Name: ${s.projectName}`),console.log(" Files:");for(let o of s.filesCreated)console.log(` - ${o}`);console.log(`
607
607
  Next steps:
608
608
 
609
- cd ${Ie.relative(process.cwd(),s.projectPath)||"."}
609
+ cd ${Ce.relative(process.cwd(),s.projectPath)||"."}
610
610
  cp .env.example .env # then edit .env and set an AI provider API key
611
611
  npm install
612
612
  npx playwright install chromium
613
613
  npx shiplight test
614
- `)}var Xt=_(()=>{"use strict";Jt()});import*as G from"fs";import*as ne from"path";function Zt(e){let t=ne.resolve(e);for(;;){if(G.existsSync(ne.join(t,"package.json"))||G.existsSync(ne.join(t,".shiplight")))return t;let n=ne.dirname(t);if(n===t)break;t=n}return null}function Qt(e){return ne.join(e,Ls)}function en(e,t){return ne.join(Qt(e),`${Rs}${t}.json`)}function tn(e,t){let n=Zt(t);if(!n)return;let r=Qt(n);G.mkdirSync(r,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};G.writeFileSync(en(n,e),JSON.stringify(s),"utf-8")}function nn(e,t){let n=Zt(t);if(n)try{G.unlinkSync(en(n,e))}catch{}}var Ls,Rs,rn=_(()=>{"use strict";Ls=".shiplight/run",Rs="debug-"});import*as on from"node:net";function sn(e,t){return new Promise(n=>{let r=on.createServer();r.once("error",s=>{n(s.code!=="EADDRINUSE")}),r.once("listening",()=>{r.close(()=>n(!0))}),r.listen(e,t)})}async function Cs(e){return await sn(e,"127.0.0.1")?sn(e,"::1"):!1}async function an(e,t){for(let n=e;n<e+t;n++)if(await Cs(n))return n;return null}var cn=_(()=>{"use strict"});var pn={};ee(pn,{findPlaywrightConfig:()=>Le,makeIdempotentFileCleaner:()=>ln,spawnPlaywrightProcess:()=>lt});import*as re from"fs";import*as q from"path";import{spawn as Ns}from"child_process";import{parse as Ds}from"yaml";function ln(e){let t=!1;return()=>{if(!t){t=!0;try{re.unlinkSync(e)}catch{}}}}function Le(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],n=q.resolve(e);for(;;){for(let s of t){let o=q.join(n,s);if(re.existsSync(o))return o}let r=q.dirname(n);if(r===n)break;n=r}return null}function Fs(e,t,n,r){let s={...n},o=s.launchOptions?.args??[];return s.launchOptions={...s.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
614
+ `)}var Zt=_(()=>{"use strict";qt()});import*as G from"fs";import*as re from"path";function Qt(e){let t=re.resolve(e);for(;;){if(G.existsSync(re.join(t,"package.json"))||G.existsSync(re.join(t,".shiplight")))return t;let n=re.dirname(t);if(n===t)break;t=n}return null}function en(e){return re.join(e,Cs)}function tn(e,t){return re.join(en(e),`${Ns}${t}.json`)}function nn(e,t){let n=Qt(t);if(!n)return;let r=en(n);G.mkdirSync(r,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};G.writeFileSync(tn(n,e),JSON.stringify(s),"utf-8")}function rn(e,t){let n=Qt(t);if(n)try{G.unlinkSync(tn(n,e))}catch{}}var Cs,Ns,sn=_(()=>{"use strict";Cs=".shiplight/run",Ns="debug-"});import*as an from"node:net";function on(e,t){return new Promise(n=>{let r=an.createServer();r.once("error",s=>{n(s.code!=="EADDRINUSE")}),r.once("listening",()=>{r.close(()=>n(!0))}),r.listen(e,t)})}async function Ds(e){return await on(e,"127.0.0.1")?on(e,"::1"):!1}async function cn(e,t){for(let n=e;n<e+t;n++)if(await Ds(n))return n;return null}var ln=_(()=>{"use strict"});var un={};te(un,{findPlaywrightConfig:()=>xe,makeIdempotentFileCleaner:()=>pn,spawnPlaywrightProcess:()=>pt});import*as se from"fs";import*as q from"path";import{spawn as js}from"child_process";import{parse as Fs}from"yaml";function pn(e){let t=!1;return()=>{if(!t){t=!0;try{se.unlinkSync(e)}catch{}}}}function xe(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],n=q.resolve(e);for(;;){for(let s of t){let o=q.join(n,s);if(se.existsSync(o))return o}let r=q.dirname(n);if(r===n)break;n=r}return null}function Us(e,t,n,r){let s={...n},o=s.launchOptions?.args??[];return s.launchOptions={...s.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
615
615
  import { test } from 'shiplightai/fixture';
616
616
  ${`
617
617
  test.use(${JSON.stringify(s)});
@@ -635,29 +635,29 @@ test('__shiplight_debug__', async ({ page, agent }) => {
635
635
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
636
636
  await new Promise(() => {});
637
637
  });
638
- `}async function Us(e){let{createServer:t}=await import("net");for(let n=e;n<e+20;n++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(n,"127.0.0.1")}))return n;throw new Error(`No available port found in range ${e}-${e+19}`)}async function lt(e){let{yamlFilePath:t,configPath:n,tempSuffix:r="",headed:s}=e,o=q.dirname(n),i=await Us(16174),a;if(!re.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let v=Ds(re.readFileSync(t,"utf-8"));v?.use&&typeof v.use=="object"&&!Array.isArray(v.use)&&(a=v.use),v?.base_url&&!a?.baseURL&&(a={...a,baseURL:v.base_url}),v?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!v.settings.auto_dismiss_modal}),v?.settings?.browser_timezone!=null&&(a={...a,timezoneId:String(v.settings.browser_timezone)}),v?.settings?.browser_language!=null&&(a={...a,locale:String(v.settings.browser_language)}),v?.settings?.extra_http_headers!=null&&typeof v.settings.extra_http_headers=="object"&&(a={...a,extraHTTPHeaders:v.settings.extra_http_headers})}catch(v){console.error("[debugger] Could not parse YAML for `use` block:",v)}let l=q.dirname(q.resolve(t)),c=r?`-${r}`:"",d=q.join(l,`.__shiplight_debug__${c}.yaml.spec.ts`),u=Fs(t,i,a,o);re.writeFileSync(d,u);let h=ln(d),g=["playwright","test",d,...s?["--headed"]:[]],f=Ns("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});f.stdout?.on("data",v=>{process.stderr.write(v)}),f.stderr?.on("data",v=>{process.stderr.write(v)});let p=()=>{f.killed||f.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),f.on("close",v=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),h(),v!==0&&v!==null&&console.error(`[debugger] Playwright process exited with code ${v}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let w=["127.0.0.1","::1"];async function m(v){try{let P=v.includes(":")?`[${v}]`:v,A=await fetch(`http://${P}:${i}/api/test-flow`);if(A.ok){try{await A.text()}catch{}return!0}}catch{}return!1}let y=null;for(let v=0;v<180;v++){if(f.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${f.exitCode} before sandbox was ready`);for(let P of w)if(await m(P)){y=P;break}if(y){console.error(`[debugger] Playwright sandbox ready on ${y}:${i}`);break}if(v===179)throw p(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(P=>setTimeout(P,1e3))}if(!y)throw p(),h(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:y,pid:f.pid??0,cleanup:async()=>{p(),h()}}}var pt=_(()=>{"use strict"});var dn=_(()=>{"use strict"});var un=_(()=>{"use strict"});var hn=_(()=>{"use strict"});import{v4 as aa}from"uuid";var fn=_(()=>{"use strict"});import{z as b}from"zod";var gn,dt,mn,fe,yn,wn,bn,D,vn,Re,ut,ht=_(()=>{"use strict";gn=b.enum(["JS_CODE","AI_MODE"]),dt=b.object({type:gn,expression:b.string()}),mn=b.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),fe=b.object({uid:b.string(),type:mn,comment:b.string().optional()}),yn=b.object({action_data:b.object({action_name:b.string(),kwargs:b.record(b.any()).optional(),args:b.array(b.any()).optional()}),action_description:b.string().optional(),url:b.string().optional(),xpath:b.string().nullable().optional(),locator:b.string().nullable().optional(),css_selector:b.string().nullable().optional(),unique_selector:b.string().nullable().optional(),element_index:b.number().nullable().optional(),frame_path:b.array(b.any()).optional(),artifacts:b.record(b.any()).optional(),feedback:b.string().optional(),original_browser_use_action:b.any().optional()}).passthrough(),wn=fe.extend({type:b.literal("DRAFT"),description:b.string()}),bn=fe.extend({type:b.literal("ACTION"),description:b.string(),action_entity:yn.optional(),locator:b.string().optional(),use_pure_vision:b.boolean().optional()}),D=b.lazy(()=>b.union([wn,bn,fe.extend({type:b.literal("STEP"),description:b.string().optional().default(""),statements:b.array(D),reference_id:b.number().optional(),template_path:b.string().optional(),template_params:b.record(b.string()).optional()}),fe.extend({type:b.literal("IF_ELSE"),description:b.string().optional(),condition:dt,then:b.array(D),else:b.array(D).optional()}),fe.extend({type:b.literal("WHILE_LOOP"),description:b.string().optional(),condition:dt,body:b.array(D),timeout_ms:b.number().optional()})])),vn=b.object({name:b.string(),statements:b.array(D),teardown:b.array(D).optional(),skip:b.union([b.boolean(),b.string()]).optional(),timeout:b.number().optional(),fail:b.union([b.boolean(),b.string()]).optional(),only:b.boolean().optional(),slow:b.boolean().optional()}),Re=b.object({tests:b.array(vn).min(1),beforeAll:b.array(D).optional(),afterAll:b.array(D).optional(),beforeEach:b.array(D).optional(),afterEach:b.array(D).optional()}),ut=b.object({comment:b.string().optional(),version:b.string().optional(),goal:b.string().optional(),url:b.string().optional(),baseURL:b.string().optional(),final_feedback:b.string().optional(),completed:b.boolean().optional(),success:b.boolean().optional(),statements:b.array(D).optional(),teardown:b.array(D).optional(),last_modified_at:b.string().optional(),testGroup:Re.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 js,parse as Tn,parseAllDocuments as ha,parseDocument as Bs,Document as Hs,isMap as me,isSeq as V}from"yaml";import{v4 as K}from"uuid";function Ce(e,t){let n={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},statements:(e.statements??[]).map(B)};return e.final_feedback&&(n.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(n.teardown=e.teardown.map(B)),n}function xe(e,t){if(e.testGroup)return Pn(e,t);let n=Ce(e,t),r=new Hs(n);return e.comment&&(r.commentBefore=e.comment),Sn(r,e.statements??[]),e.teardown&&Sn(r,e.teardown,"teardown"),r.toString(kn)}function Sn(e,t,n="statements"){let r=e.contents;if(!r||!me(r))return;let s=r.get(n,!0);V(s)&&_e(s,t)}function _e(e,t){for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=t[n],s=e.items[n];if(n>0&&(s.spaceBefore=!0),r.comment&&(n===0?e.commentBefore=r.comment:s.commentBefore=r.comment),me(s)){let o=s;if(r.type==="STEP"){let i=o.get("statements",!0);V(i)&&_e(i,r.statements)}else if(r.type==="IF_ELSE"){let i=o.get("THEN",!0);V(i)&&_e(i,r.then);let a=o.get("ELSE",!0);V(a)&&r.else&&_e(a,r.else)}else if(r.type==="WHILE_LOOP"){let i=o.get("DO",!0);V(i)&&_e(i,r.body)}}}}function Pn(e,t){let n=e.testGroup;if(!n)throw new Error("suiteToYaml requires a TestFlow with testGroup");let r={};t?.test_case_id!==void 0&&(r.test_case_id=t.test_case_id),t?.name&&(r.name=t.name),t?.tags&&t.tags.length>0&&(r.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(r.use=t.use);let s={};return e.baseURL&&(s.base_url=e.baseURL),n.beforeAll&&n.beforeAll.length>0&&(s.beforeAll=n.beforeAll.map(B)),n.beforeEach&&n.beforeEach.length>0&&(s.beforeEach=n.beforeEach.map(B)),n.afterEach&&n.afterEach.length>0&&(s.afterEach=n.afterEach.map(B)),n.afterAll&&n.afterAll.length>0&&(s.afterAll=n.afterAll.map(B)),s.tests=n.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(B),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(B)),i}),r.suite=s,js(r,kn)}function B(e){switch(e.type){case"DRAFT":return Ws(e);case"ACTION":return Gs(e);case"STEP":return Ks(e);case"IF_ELSE":return Vs(e);case"WHILE_LOOP":return zs(e)}}function Ws(e){return{intent:e.description}}function Gs(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,n=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=n?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l=n?.code;return typeof l=="string"?{VERIFY:a,js:l}:{VERIFY:a}}}if(t==="go_to_url"){let a=n?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l={URL:a};return n?.new_tab===!0&&(l.new_tab=!0),typeof n?.timeout_seconds=="number"&&(l.timeout_seconds=n.timeout_seconds),l}}if(t==="js_action"){let a=n?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=n?.condition;if(typeof a=="string"){let l={WAIT_UNTIL:a};return typeof n?.timeout_seconds=="number"&&n.timeout_seconds!==60&&(l.timeout_seconds=n.timeout_seconds),l}}if(t==="wait"){let a=n?.seconds,c={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(c.seconds=a),c}if(t==="js_code"){let a=n?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let r=e.action_entity.action_data??e.action_entity.action;if(!r)return{intent:e.description};let s={intent:e.description,action:r.action_name},o=e.locator??e.action_entity.locator;o&&(s.locator=o);let i=e.action_entity.xpath;if(i&&(s.xpath=i),e.use_pure_vision&&(s.use_pure_vision=!0),r.kwargs&&Object.keys(r.kwargs).length>0)for(let[a,l]of Object.entries(r.kwargs))s[a]=l;return r.args&&r.args.length>0&&(s.args=r.args),s}function Ks(e){if(e.template_path){let n={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(n.params=e.template_params),n}let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Vs(e){let t={IF:An(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function zs(e){let t={WHILE:An(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function An(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Te(e){try{let t=Tn(e);if(!t||typeof t!="object")return{};let n={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(n.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(n.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(n.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(n.timeout=t.timeout),n}catch{return{}}}function ft(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(ft);let t=e,n=Object.keys(t);if(n.length===1){let s=n[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let r={};for(let[s,o]of Object.entries(t))r[s]=ft(o);return r}function C(e){if(e.length>_n)throw new Error(`YAML input too large (${e.length} bytes, max ${_n})`);let t=ft(Tn(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Ys(t);let n={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:H(t.statements??[])};t.final_feedback&&(n.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(n.teardown=H(t.teardown));let r=ut.safeParse(n);if(!r.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(r.error.errors)}`);let s=r.data;return Ne(e,s),s}function Ys(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let n=t.tests;if(!Array.isArray(n)||n.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:n.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let l={name:a.name,statements:H(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(l.teardown=H(a.teardown)),a.skip!==void 0&&(l.skip=a.skip),typeof a.timeout=="number"&&(l.timeout=a.timeout),a.fail!==void 0&&(l.fail=a.fail),a.only===!0&&(l.only=!0),a.slow===!0&&(l.slow=!0),l})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=H(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=H(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=H(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=H(t.afterEach));let o=Re.safeParse(s);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 H(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Js)}function Js(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 qs(t);if("WHILE"in t)return Xs(t);if("STEP"in t)return Zs(t);if("VERIFY"in t){let n=t.VERIFY,r={statement:typeof n=="string"?n:String(n)};return typeof t.js=="string"&&(r.code=t.js),{uid:K(),type:"ACTION",description:String(n),action_entity:{action_description:String(n),action_data:{action_name:"verify",kwargs:r}}}}if("URL"in t){let n=t.URL,r=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof n=="string"?n:String(n)};return r&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:K(),type:"ACTION",description:`Navigate to ${n}`,action_entity:{action_description:`Navigate to ${n}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let n=t.WAIT_UNTIL,r=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:K(),type:"ACTION",description:`Wait until: ${n}`,action_entity:{action_description:`Wait until: ${n}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof n=="string"?n:String(n),timeout_seconds:r}}}}}if("WAIT"in t){let n=t.WAIT,r=typeof t.seconds=="number"?t.seconds:3;return{uid:K(),type:"ACTION",description:typeof n=="string"?n:`Wait ${r}s`,action_entity:{action_description:typeof n=="string"?n:`Wait ${r}s`,action_data:{action_name:"wait",kwargs:{seconds:r}}}}}if("CODE"in t){let n=t.CODE;if(n==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:K(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)&&t.action!=="verify"){let n=t.js,r=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:K(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_action",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:n,...r}=t;return xn({...r,action:"function",functionName:n})}if("action"in t)return xn(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:K(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function En(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 qs(e){let t=En(e.IF),n=e.THEN;if(!Array.isArray(n))throw new Error("IF_ELSE requires a THEN array");let r={uid:K(),type:"IF_ELSE",condition:t,then:H(n)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(r.else=H(e.ELSE)),r}function Xs(e){let t=En(e.WHILE),n=e.DO;if(!Array.isArray(n))throw new Error("WHILE_LOOP requires a DO array");let r={uid:K(),type:"WHILE_LOOP",condition:t,body:H(n)};return typeof e.timeout_ms=="number"&&(r.timeout_ms=e.timeout_ms),r}function Zs(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let n={uid:K(),type:"STEP",description:t,statements:H(e.statements)};if(typeof e.reference_id=="number"&&(n.reference_id=e.reference_id),typeof e.template_path=="string"&&(n.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let r=e.template_params,s={};for(let[o,i]of Object.entries(r))s[o]=String(i);n.template_params=s}return n}function xn(e){let t=typeof e.action=="string"?e.action:String(e.action),n=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",r=typeof e.locator=="string"?e.locator:void 0,s=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[c,d]of Object.entries(e))Qs.has(c)||(i[c]=d);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js);let a={action_description:n,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};r&&(a.locator=r),s&&(a.xpath=s);let l={uid:K(),type:"ACTION",description:n,action_entity:a};return o&&(l.use_pure_vision=!0),l}function Ne(e,t){let n;try{n=Bs(e)}catch{return}let r=n.contents;if(!r||!me(r))return;if(n.commentBefore)t.comment=n.commentBefore;else{let l=r.items?.[0];l?.key&&l.key.commentBefore&&(t.comment=l.key.commentBefore)}let s=r,o=s.get("statements",!0);V(o)&&t.statements&&ge(o,t.statements);let i=s.get("teardown",!0);V(i)&&t.teardown&&ge(i,t.teardown)}function ge(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=e.items[n];r.commentBefore&&!(n===0&&e.commentBefore)&&(t[n].comment=r.commentBefore);let s=t[n];if(s.type==="STEP"&&me(r)){let o=r.get("statements",!0);V(o)&&ge(o,s.statements)}else if(s.type==="IF_ELSE"&&me(r)){let o=r.get("THEN",!0);V(o)&&ge(o,s.then);let i=r.get("ELSE",!0);V(i)&&s.else&&ge(i,s.else)}else if(s.type==="WHILE_LOOP"&&me(r)){let o=r.get("DO",!0);V(o)&&ge(o,s.body)}}}var kn,_n,Qs,De=_(()=>{"use strict";ht();kn={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};_n=1024*1024;Qs=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as wa,stringify as ba}from"yaml";var $n=_(()=>{"use strict";De()});var gt,Fe,Ue=_(()=>{"use strict";gt=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},Fe=e=>{let t=[],n=r=>{for(let s of r){t.push(s);let o=gt(s);for(let i of o)n(i.statements)}};return n(e),t}});function On(e){let t=0,n=0;for(let r of e)if(r.type==="DRAFT")n++;else if(r.type==="ACTION"){let s=r.action_entity?.action_data?.action_name??"";In.has(s)||t++}return{action:t,draft:n}}function to(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Mn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function mt(e,t){let n=t?.coverageThreshold??eo,r=[],s=[],o;try{o=C(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||r.push('Missing required field: "goal"'),o.statements?.length||r.push('Missing required field: "statements"');let i=[...Fe(o.statements??[]),...o.teardown?Fe(o.teardown):[]],{action:a,draft:l}=On(i),c="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let h of i){let g=h;if(g.reference_id!==void 0){let f=g.description||h.uid;r.push(`Unresolved cloud template reference on statement "${f}" (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(h.type==="ACTION"){let f=h,p=f.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let w=f.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let m=to(w);if(m){let y=f.description||p;r.push(`Invalid JS in "${y}": ${m}. ${c}`)}}}}if(h.type==="IF_ELSE"){let f=h;if(f.condition.type==="JS_CODE"){let p=Mn(f.condition.expression);p&&r.push(`Invalid JS in IF condition "${f.condition.expression}": ${p}. ${c}`)}}if(h.type==="WHILE_LOOP"){let f=h;if(f.condition.type==="JS_CODE"){let p=Mn(f.condition.expression);p&&r.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${p}. ${c}`)}}}let d=a+l,u=d>0?Math.round(a/d*100):0;return d>0&&u/100<n&&s.push(`Low action coverage: ${a}/${d} statements (${u}%) are enriched with action/js. ${l} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:r.length===0,errors:r,warnings:s,stats:{total:d,action:a,draft:l,coverage:u}}}var eo,In,Ln=_(()=>{"use strict";De();Ue();eo=.5,In=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var je=_(()=>{"use strict"});var Rn=_(()=>{"use strict";je()});var Cn,ro,Nn,Dn,ke,so,oo,Fn=_(()=>{"use strict";Cn=112,ro=1080-Cn,Nn={"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"}},Dn={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"]},ke=(e,t=!1)=>{let n=["chromium"];return t&&n.push("webkit"),Dn[e].map(r=>Nn[r]).filter(r=>r.defaultBrowserType&&n.includes(r.defaultBrowserType))},so={desktop:{label:"Desktop",type:"desktop",devices:ke("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:ke("mobile")}},oo={desktop:{label:"Desktop",type:"desktop",devices:ke("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:ke("mobile",!0)}}});var Un=_(()=>{"use strict"});function yt(){return{version:"1.0",entries:{}}}var jn=_(()=>{"use strict";Ue()});var z,wt,Bn=_(()=>{"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||{}),wt=18e4});var Hn=_(()=>{"use strict"});var Wn=_(()=>{"use strict"});var Gn=_(()=>{"use strict"});var Kn=_(()=>{"use strict";je()});var ae=_(()=>{"use strict";dn();un();hn();fn();$n();Ln();De();ht();Rn();Fn();Un();jn();Ue();Bn();Hn();Wn();Gn();Kn();je()});import{stringify as ao}from"yaml";import{createHash as Po}from"crypto";import{parse as Ao,stringify as tr}from"yaml";import{readFileSync as Eo,existsSync as $o}from"fs";import{resolve as vt,dirname as Mo}from"path";import{parse as Jn,stringify as Io}from"yaml";import{readFileSync as Fo,writeFileSync as Uo,mkdirSync as jo}from"fs";import{dirname as Bo}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function I(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function co(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function lo(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 He(e){let t=co(e),n=e.locator;if(typeof n=="string"&&n.trim())return n=n.trim(),n.endsWith("first()")?`${t}.${n}`:`${t}.${n}.first()`;let r=lo(e);if(r){let s=JSON.stringify(r);return`${t}.locator(${s}).first()`}return null}function Vn(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:po.includes(t)}function ho(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!uo.includes(t)}function S(e,t){R.set(e,t)}function fo(e){return R.get(e)}function de(e,t,n=[]){let r=[...n];return t.locator?r.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&r.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(t.frame_path)}`),r.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...r.map(s=>` ${s},`),"});"]}function zn(e){let t=e.functionName;if(!t)return null;let n=Array.isArray(e.args)?e.args.map(String):[];if(n.length===0)return`await ${t}()`;let r=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=n.map(i=>r.includes(i)||s.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function X(e,t,n,r="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${r}.${o}`,l=go(i,t,a,n);l.length>0&&(s.push(...l),o<e.length-1&&s.push(""))}return s}function go(e,t,n,r){let s=" ".repeat(t);switch(e.type){case"DRAFT":return mo(e,t,n,r);case"ACTION":return yo(e,t,n,r);case"STEP":return wo(e,t,n,r);case"IF_ELSE":return bo(e,t,n,r);case"WHILE_LOOP":return vo(e,t,n,r);default:return[`${s}// Unknown statement type: ${e.type}`]}}function mo(e,t,n,r){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.run(page, ${i}, '${n}');`]}function yo(e,t,n,r){let s=" ".repeat(t),o=e.description,i=e.uid,l=r.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!l){if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let v=JSON.stringify(o),P=!!e.use_pure_vision;return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.execute(page, ${v}, '${n}', ${P});`]}let c=e.locator?{...l,locator:e.locator}:l;o&&o!==c.action_description&&(c={...c,action_description:o});let d=c.action_data?.action_name||"",u=c.action_description||"",h=fo(d);if(!h)return[`${s}// ${n}: Unknown action: ${d}`];let g={imports:r.imports},f=h(c,n,g);if(r.noAgent){if(Vn(c))return[`${s}// ${n}: ${I(u)}`,`${s}// AI action: ${I(u)} (requires agent - skipped in hook)`];let v=So(c,d,s,n);return v||[`${s}// ${n}: ${I(u)}`,...f.map(P=>`${s}${P}`)]}if(Vn(c))return[`${s}// ${n}: ${I(u)}`,`${s}page = agent.agentServices.validatePage(page);`,...f.map(v=>`${s}${v}`)];let p=JSON.stringify(u),w=f.map(v=>`${s} ${v}`),m=ho(c),y=i?`'${i}'`:"undefined";return[`${s}// ${n}: ${I(u)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...w,`${s}}, ${p}, '${n}', ${y}, ${m});`]}function wo(e,t,n,r){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${I(e.description)}`);let i=X(e.statements,t,r,n);return o.push(...i),o}function bo(e,t,n,r){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${n}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${I(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${n}")) {`)}let i=X(e.then,t+1,r,`${n}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=X(e.else,t+1,r,`${n}.else`);o.push(...a)}return o.push(`${s}}`),o}function vo(e,t,n,r){let s=" ".repeat(t),o=[];o.push(`${s}// ${n}: Loop`);let i=e.timeout_ms??wt,a=i/1e3,l=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,c=`loop_${n.replace(/\./g,"_")}`;if(o.push(`${s}const ${c}_start = Date.now();`),o.push(`${s}const ${c}_timeout = ${i};`),o.push(`${s}const ${c}_check = () => {`),o.push(`${s} if (Date.now() - ${c}_start > ${c}_timeout) {`),o.push(`${s} throw new Error('${l}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${c}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${I(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);o.push(`${s}while (${c}_check() && await agent.evaluate(page, ${u}, "${n}")) {`)}let d=X(e.body,t+1,r,`${n}.body`);return o.push(...d),o.push(`${s}}`),o}function So(e,t,n,r){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??St;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${n}// ${r}: ${I(s)}`,`${n}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goBack();`];case"go_forward":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goForward();`];case"input_text":{let a=o.text||"",l=He(e);return l?[`${n}// ${r}: ${I(s)}`,`${n}await ${l}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",l=He(e);return l?[`${n}// ${r}: ${I(s)}`,`${n}await ${l}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function _o(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...Qn()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(n.push(...Yn("beforeEach",t.beforeEach,o)),n.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",l=bt(t?.tags);for(let c of t.parameters){let d=Zn(e,c.values);n.push(...Ge(d,`${l}${se(a)} [${se(c.name)}]`,o,0,i)),n.push("")}}else{let a=t?.testName||e.goal||"Generated test",l=bt(t?.tags);n.push(...Ge(e,`${l}${se(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(n.push(""),n.push(...Yn("afterEach",t.afterEach,o))),er(n,s),n.join(`
639
- `)}function xo(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...Qn()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore},i=t?.testName||"Test Suite",a=bt(t?.tags);n.push(`test.describe.serial('${a}${se(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(n.push(...Be("beforeAll",e.beforeAll,o,1)),n.push("")),e.beforeEach&&e.beforeEach.length>0&&(n.push(...Be("beforeEach",e.beforeEach,o,1)),n.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],u=d.timeout||d.skip!==void 0||d.fail!==void 0||d.only||d.slow?{timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}:void 0;if(d.parameters&&d.parameters.length>0)for(let h of d.parameters){let g=Zn(d.testFlow,h.values);n.push(...Ge(g,`${se(d.name)} [${se(h.name)}]`,o,1,u)),n.push("")}else n.push(...Ge(d.testFlow,se(d.name),o,1,u)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&n.push("")}return e.afterEach&&e.afterEach.length>0&&(n.push(...Be("afterEach",e.afterEach,o,1)),n.push("")),e.afterAll&&e.afterAll.length>0&&n.push(...Be("afterAll",e.afterAll,o,1)),n.push("});"),er(n,s),n.join(`
640
- `)}function bt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function We(e){let t=new Set;function n(r){for(let s of r)switch(s.type){case z.ACTION:{let i=s.action_entity?.action_data?.kwargs;if(i?.args&&Array.isArray(i.args))for(let a of i.args)typeof a=="string"&&To.includes(a)&&t.add(a);break}case z.STEP:n(s.statements);break;case z.IF_ELSE:{let o=s;n(o.then),o.else&&n(o.else);break}case z.WHILE_LOOP:n(s.body);break}}return n(e),t}function ko(e){let t=We(e.statements??[]);if(e.teardown)for(let n of We(e.teardown))t.add(n);return t}function _t(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ge(e,t,n,r=0,s){let o=" ".repeat(r),i=[],a=ko(e),l=_t(a),c=s?.only?"test.only":"test";i.push(`${o}${c}('${t}', async (${l}) => {`),s?.skip===!0?i.push(`${o} test.skip();`):typeof s?.skip=="string"&&i.push(`${o} test.skip(true, '${se(s.skip)}');`),s?.fail===!0?i.push(`${o} test.fail();`):typeof s?.fail=="string"&&i.push(`${o} test.fail(true, '${se(s.fail)}');`),s?.slow&&i.push(`${o} test.slow();`),s?.timeout&&i.push(`${o} test.setTimeout(${s.timeout});`);let d=e.teardown&&e.teardown.length>0,u=r+1;if(d){if(i.push(`${o} try {`),e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let g=X(e.statements,u+1,n);i.push(...g)}i.push(`${o} } finally {`),i.push(`${o} // Teardown`);let h=X(e.teardown,u+1,n,"teardown");i.push(...h),i.push(`${o} }`)}else if(e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let h=X(e.statements,u,n);i.push(...h)}return i.push(`${o}});`),i}function Yn(e,t,n){let r=[],s=Xn(t),o=We(s),i=_t(o);return r.push(`test.${e}(async (${i}) => {`),r.push(...X(s,1,n,e)),r.push("});"),r}function Be(e,t,n,r){let s=" ".repeat(r),o=[],i=Xn(t);if(e==="beforeAll"||e==="afterAll"){let l={...n,noAgent:!0};o.push(`${s}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${s} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...X(i,r+1,l,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let l=We(i),c=_t(l);o.push(`${s}test.${e}(async (${c}) => {`),o.push(...X(i,r+1,n,e)),o.push(`${s}});`)}return o}function Xn(e){let n=ao({goal:"_hook",statements:e});return C(n).statements??[]}function Zn(e,t){let n=xe(e);for(let[r,s]of Object.entries(t))n=n.split(`<<${r}>>`).join(String(s));return C(n)}function Qn(){return["import { test, expect } from 'shiplightai/fixture';"]}function er(e,t){if(t.size>0){let n=0;for(let s=0;s<e.length;s++)e[s].startsWith("import ")&&(n=s+1);let r=Array.from(t);e.splice(n,0,...r)}}function nr(e,t,n){let r={expandingPaths:new Set([vt(t)]),depth:0,referencedPaths:new Set,basePath:n},s={...e};Array.isArray(s.statements)&&(s.statements=ce(s.statements,t,r)),Array.isArray(s.teardown)&&(s.teardown=ce(s.teardown,t,r));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=ce(s[o],t,r));return{doc:s,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ce(e,t,n){let r=[];for(let s of e)if(Oo(s)){let o=Lo(s,t,n);r.push(o)}else r.push(Ro(s,t,n));return r}function Oo(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Lo(e,t,n){if(n.depth>=qn)throw new Error(`Template expansion exceeded maximum depth of ${qn}. Check for deeply nested or circular template references.`);let r=vt(Mo(t),e.template),s=!$o(r)&&n.basePath?vt(n.basePath,e.template):r;if(n.expandingPaths.has(s))throw new Error(`Circular template reference detected: ${s} is already being expanded. Stack: ${Array.from(n.expandingPaths).join(" \u2192 ")} \u2192 ${s}`);n.referencedPaths.add(s);let o;try{o=Eo(s,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${f.message}`)}let i=Jn(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let a=i.params||[],l=e.params||{};for(let f of a)if(!(f in l))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${a.join(", ")}]`);let c=i.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let p=Io(c);for(let[w,m]of Object.entries(l))p=p.split(`<<${w}>>`).join(String(m));c=Jn(p)}let d={expandingPaths:new Set([...n.expandingPaths,s]),depth:n.depth+1,referencedPaths:n.referencedPaths},u=ce(c,s,d),g={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:u};return Object.keys(l).length>0&&(g.template_params=l),g}function Ro(e,t,n){if(typeof e!="object"||e===null)return e;let r={...e};return Array.isArray(r.statements)&&(r.statements=ce(r.statements,t,n)),Array.isArray(r.THEN)&&(r.THEN=ce(r.THEN,t,n)),Array.isArray(r.ELSE)&&(r.ELSE=ce(r.ELSE,t,n)),Array.isArray(r.DO)&&(r.DO=ce(r.DO,t,n)),r}function Tt(e,t,n){let r=Ao(e),s=r?.name,o=r?.tags,i=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new xt('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return No(r,s,o,i,t,n)}return Co(r,s,o,i,t,n)}function Co(e,t,n,r,s,o){let i=e?.beforeEach,a=e?.afterEach,l=rr(e?.parameters),c=e?.timeout,d=e?.skip,u=e?.fail,h=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new xt(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(s&&e&&typeof e=="object"){let m=nr(e,s,o);e=m.doc,f=m.referencedTemplatePaths}let p=tr(e),w=C(p);return s&&(ye(w.statements??[],s,"main"),w.teardown&&ye(w.teardown,s,"teardown")),{testFlow:w,name:t,tags:n,use:r,beforeEach:i,afterEach:a,parameters:l,timeout:c,skip:d,fail:u,only:h,slow:g,referencedTemplatePaths:f}}function No(e,t,n,r,s,o){let i=e.suite;if(!Array.isArray(i.tests)||i.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=i.beforeAll,l=i.afterAll,c=i.beforeEach,d=i.afterEach,u=[],h=i.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 m=[],y=w;if(s&&typeof w=="object"){let M=nr(w,s,o);y=M.doc,m=M.referencedTemplatePaths,u.push(...m)}let v=tr(y),P=C(v),A=rr(p.parameters);return{testFlow:P,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:A,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=i.base_url,f=g?{...r,baseURL:g}:r;return{suite:{beforeAll:a,afterAll:l,beforeEach:c,afterEach:d,tests:h},name:t,tags:n,use:f,referencedTemplatePaths:u}}function rr(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,n)=>{if(!t.name)throw new Error(`Parameter set at index ${n} 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 ye(e,t,n){for(let r=0;r<e.length;r++){let s=e[r],o=`${n}.${r}`,i=s.description||"";if(s.uid=Do(t,o,i),s.type===z.STEP)ye(s.statements,t,o);else if(s.type===z.IF_ELSE){let a=s;ye(a.then,t,`${o}.then`),a.else&&ye(a.else,t,`${o}.else`)}else s.type===z.WHILE_LOOP&&ye(s.body,t,`${o}.body`)}}function Do(e,t,n){let r=Po("sha256").update(`${e}:${t}:${n}`).digest("hex");return`${r.slice(0,8)}-${r.slice(8,12)}-${r.slice(12,16)}-${r.slice(16,20)}-${r.slice(20,32)}`}function sr(e,t){let n;try{n=Fo(e,"utf-8")}catch(r){return{valid:!1,errors:[`Failed to read file: ${r.message}`],warnings:[]}}return Ho(n,e,t)}function Ho(e,t,n){let r=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=r||s?null:mt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,l=[];try{let c=n?.parsed??Tt(e,t);l=c.referencedTemplatePaths;let d={version:n?.version,actionEntityStore:n?.actionEntityStore},u=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?i=xo(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):i=_o(c.testFlow,{...d,testName:c.name,tags:c.tags,use:u,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 h=i.split(`
638
+ `}async function Bs(e){let{createServer:t}=await import("net");for(let n=e;n<e+20;n++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(n,"127.0.0.1")}))return n;throw new Error(`No available port found in range ${e}-${e+19}`)}async function pt(e){let{yamlFilePath:t,configPath:n,tempSuffix:r="",headed:s}=e,o=q.dirname(n),i=await Bs(16174),a;if(!se.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let y=Fs(se.readFileSync(t,"utf-8"));y?.use&&typeof y.use=="object"&&!Array.isArray(y.use)&&(a=y.use),y?.base_url&&!a?.baseURL&&(a={...a,baseURL:y.base_url}),y?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!y.settings.auto_dismiss_modal}),y?.settings?.browser_timezone!=null&&(a={...a,timezoneId:String(y.settings.browser_timezone)}),y?.settings?.browser_language!=null&&(a={...a,locale:String(y.settings.browser_language)}),y?.settings?.extra_http_headers!=null&&typeof y.settings.extra_http_headers=="object"&&(a={...a,extraHTTPHeaders:y.settings.extra_http_headers})}catch(y){console.error("[debugger] Could not parse YAML for `use` block:",y)}let l=q.dirname(q.resolve(t)),c=r?`-${r}`:"",u=q.join(l,`.__shiplight_debug__${c}.yaml.spec.ts`),d=Us(t,i,a,o);se.writeFileSync(u,d);let f=pn(u),g=["playwright","test",u,...s?["--headed"]:[]],h=js("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});h.stdout?.on("data",y=>{process.stderr.write(y)}),h.stderr?.on("data",y=>{process.stderr.write(y)});let p=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),h.on("close",y=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),f(),y!==0&&y!==null&&console.error(`[debugger] Playwright process exited with code ${y}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let v=["127.0.0.1","::1"];async function w(y){try{let k=y.includes(":")?`[${y}]`:y,P=await fetch(`http://${k}:${i}/api/test-flow`);if(P.ok){try{await P.text()}catch{}return!0}}catch{}return!1}let m=null;for(let y=0;y<180;y++){if(h.exitCode!==null)throw f(),new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`);for(let k of v)if(await w(k)){m=k;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${i}`);break}if(y===179)throw p(),f(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(k=>setTimeout(k,1e3))}if(!m)throw p(),f(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:m,pid:h.pid??0,cleanup:async()=>{p(),f()}}}var ut=_(()=>{"use strict"});import{z as b}from"zod";var dn,dt,hn,fe,fn,gn,mn,j,yn,De,ht,ft=_(()=>{"use strict";dn=b.enum(["JS_CODE","AI_MODE"]),dt=b.object({type:dn,expression:b.string()}),hn=b.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),fe=b.object({uid:b.string(),type:hn,comment:b.string().optional()}),fn=b.object({action_data:b.object({action_name:b.string(),kwargs:b.record(b.any()).optional(),args:b.array(b.any()).optional()}),action_description:b.string().optional(),url:b.string().optional(),xpath:b.string().nullable().optional(),locator:b.string().nullable().optional(),css_selector:b.string().nullable().optional(),unique_selector:b.string().nullable().optional(),element_index:b.number().nullable().optional(),frame_path:b.array(b.any()).optional(),artifacts:b.record(b.any()).optional(),feedback:b.string().optional(),original_browser_use_action:b.any().optional()}).passthrough(),gn=fe.extend({type:b.literal("DRAFT"),description:b.string()}),mn=fe.extend({type:b.literal("ACTION"),description:b.string(),action_entity:fn.optional(),locator:b.string().optional(),use_pure_vision:b.boolean().optional()}),j=b.lazy(()=>b.union([gn,mn,fe.extend({type:b.literal("STEP"),description:b.string().optional().default(""),statements:b.array(j),reference_id:b.number().optional(),template_path:b.string().optional(),template_params:b.record(b.string()).optional()}),fe.extend({type:b.literal("IF_ELSE"),description:b.string().optional(),condition:dt,then:b.array(j),else:b.array(j).optional()}),fe.extend({type:b.literal("WHILE_LOOP"),description:b.string().optional(),condition:dt,body:b.array(j),timeout_ms:b.number().optional()})])),yn=b.object({name:b.string(),statements:b.array(j),teardown:b.array(j).optional(),skip:b.union([b.boolean(),b.string()]).optional(),timeout:b.number().optional(),fail:b.union([b.boolean(),b.string()]).optional(),only:b.boolean().optional(),slow:b.boolean().optional()}),De=b.object({tests:b.array(yn).min(1),beforeAll:b.array(j).optional(),afterAll:b.array(j).optional(),beforeEach:b.array(j).optional(),afterEach:b.array(j).optional()}),ht=b.object({comment:b.string().optional(),version:b.string().optional(),goal:b.string().optional(),url:b.string().optional(),baseURL:b.string().optional(),final_feedback:b.string().optional(),completed:b.boolean().optional(),success:b.boolean().optional(),statements:b.array(j).optional(),teardown:b.array(j).optional(),last_modified_at:b.string().optional(),testGroup:De.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 Hs,parse as Sn,parseAllDocuments as pa,parseDocument as Ws,Document as Gs,isMap as me,isSeq as z}from"yaml";import{v4 as K}from"uuid";function je(e,t){let n={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},statements:(e.statements??[]).map(B)};return e.final_feedback&&(n.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(n.teardown=e.teardown.map(B)),n}function ke(e,t){if(e.testGroup)return xn(e,t);let n=je(e,t),r=new Gs(n);return e.comment&&(r.commentBefore=e.comment),wn(r,e.statements??[]),e.teardown&&wn(r,e.teardown,"teardown"),r.toString(_n)}function wn(e,t,n="statements"){let r=e.contents;if(!r||!me(r))return;let s=r.get(n,!0);z(s)&&Te(s,t)}function Te(e,t){for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=t[n],s=e.items[n];if(n>0&&(s.spaceBefore=!0),r.comment&&(n===0?e.commentBefore=r.comment:s.commentBefore=r.comment),me(s)){let o=s;if(r.type==="STEP"){let i=o.get("statements",!0);z(i)&&Te(i,r.statements)}else if(r.type==="IF_ELSE"){let i=o.get("THEN",!0);z(i)&&Te(i,r.then);let a=o.get("ELSE",!0);z(a)&&r.else&&Te(a,r.else)}else if(r.type==="WHILE_LOOP"){let i=o.get("DO",!0);z(i)&&Te(i,r.body)}}}}function xn(e,t){let n=e.testGroup;if(!n)throw new Error("suiteToYaml requires a TestFlow with testGroup");let r={};t?.test_case_id!==void 0&&(r.test_case_id=t.test_case_id),t?.name&&(r.name=t.name),t?.tags&&t.tags.length>0&&(r.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(r.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(r.settings=t.settings);let s={};return e.baseURL&&(s.base_url=e.baseURL),n.beforeAll&&n.beforeAll.length>0&&(s.beforeAll=n.beforeAll.map(B)),n.beforeEach&&n.beforeEach.length>0&&(s.beforeEach=n.beforeEach.map(B)),n.afterEach&&n.afterEach.length>0&&(s.afterEach=n.afterEach.map(B)),n.afterAll&&n.afterAll.length>0&&(s.afterAll=n.afterAll.map(B)),s.tests=n.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(B),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(B)),i}),r.suite=s,Hs(r,_n)}function B(e){switch(e.type){case"DRAFT":return Ks(e);case"ACTION":return zs(e);case"STEP":return Vs(e);case"IF_ELSE":return Ys(e);case"WHILE_LOOP":return Js(e)}}function Ks(e){return{intent:e.description}}function zs(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,n=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=n?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l=n?.code;return typeof l=="string"?{VERIFY:a,js:l}:{VERIFY:a}}}if(t==="go_to_url"){let a=n?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let l={URL:a};return n?.new_tab===!0&&(l.new_tab=!0),typeof n?.timeout_seconds=="number"&&(l.timeout_seconds=n.timeout_seconds),l}}if(t==="js_action"){let a=n?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=n?.condition;if(typeof a=="string"){let l={WAIT_UNTIL:a};return typeof n?.timeout_seconds=="number"&&n.timeout_seconds!==60&&(l.timeout_seconds=n.timeout_seconds),l}}if(t==="wait"){let a=n?.seconds,c={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(c.seconds=a),c}if(t==="js_code"){let a=n?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let r=e.action_entity.action_data??e.action_entity.action;if(!r)return{intent:e.description};let s={intent:e.description,action:r.action_name},o=e.locator??e.action_entity.locator;o&&(s.locator=o);let i=e.action_entity.xpath;if(i&&(s.xpath=i),e.use_pure_vision&&(s.use_pure_vision=!0),r.kwargs&&Object.keys(r.kwargs).length>0)for(let[a,l]of Object.entries(r.kwargs))s[a]=l;return r.args&&r.args.length>0&&(s.args=r.args),s}function Vs(e){if(e.template_path){let n={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(n.params=e.template_params),n}let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Ys(e){let t={IF:Tn(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function Js(e){let t={WHILE:Tn(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Tn(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Pe(e){try{let t=Sn(e);if(!t||typeof t!="object")return{};let n={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(n.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(n.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(n.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(n.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(n.settings=t.settings),n}catch{return{}}}function gt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(gt);let t=e,n=Object.keys(t);if(n.length===1){let s=n[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let r={};for(let[s,o]of Object.entries(t))r[s]=gt(o);return r}function C(e){if(e.length>bn)throw new Error(`YAML input too large (${e.length} bytes, max ${bn})`);let t=gt(Sn(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return qs(t);let n={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:H(t.statements??[])};t.final_feedback&&(n.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(n.teardown=H(t.teardown));let r=ht.safeParse(n);if(!r.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(r.error.errors)}`);let s=r.data;return Fe(e,s),s}function qs(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let n=t.tests;if(!Array.isArray(n)||n.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:n.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let l={name:a.name,statements:H(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(l.teardown=H(a.teardown)),a.skip!==void 0&&(l.skip=a.skip),typeof a.timeout=="number"&&(l.timeout=a.timeout),a.fail!==void 0&&(l.fail=a.fail),a.only===!0&&(l.only=!0),a.slow===!0&&(l.slow=!0),l})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=H(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=H(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=H(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=H(t.afterEach));let o=De.safeParse(s);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 H(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Xs)}function Xs(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 Zs(t);if("WHILE"in t)return Qs(t);if("STEP"in t)return eo(t);if("VERIFY"in t){let n=t.VERIFY,r={statement:typeof n=="string"?n:String(n)};return typeof t.js=="string"&&(r.code=t.js),{uid:K(),type:"ACTION",description:String(n),action_entity:{action_description:String(n),action_data:{action_name:"verify",kwargs:r}}}}if("URL"in t){let n=t.URL,r=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof n=="string"?n:String(n)};return r&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:K(),type:"ACTION",description:`Navigate to ${n}`,action_entity:{action_description:`Navigate to ${n}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let n=t.WAIT_UNTIL,r=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:K(),type:"ACTION",description:`Wait until: ${n}`,action_entity:{action_description:`Wait until: ${n}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof n=="string"?n:String(n),timeout_seconds:r}}}}}if("WAIT"in t){let n=t.WAIT,r=typeof t.seconds=="number"?t.seconds:3;return{uid:K(),type:"ACTION",description:typeof n=="string"?n:`Wait ${r}s`,action_entity:{action_description:typeof n=="string"?n:`Wait ${r}s`,action_data:{action_name:"wait",kwargs:{seconds:r}}}}}if("CODE"in t){let n=t.CODE;if(n==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:K(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)&&t.action!=="verify"){let n=t.js,r=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:K(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_action",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:n,...r}=t;return vn({...r,action:"function",functionName:n})}if("action"in t)return vn(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:K(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function kn(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 Zs(e){let t=kn(e.IF),n=e.THEN;if(!Array.isArray(n))throw new Error("IF_ELSE requires a THEN array");let r={uid:K(),type:"IF_ELSE",condition:t,then:H(n)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(r.else=H(e.ELSE)),r}function Qs(e){let t=kn(e.WHILE),n=e.DO;if(!Array.isArray(n))throw new Error("WHILE_LOOP requires a DO array");let r={uid:K(),type:"WHILE_LOOP",condition:t,body:H(n)};return typeof e.timeout_ms=="number"&&(r.timeout_ms=e.timeout_ms),r}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 n={uid:K(),type:"STEP",description:t,statements:H(e.statements)};if(typeof e.reference_id=="number"&&(n.reference_id=e.reference_id),typeof e.template_path=="string"&&(n.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let r=e.template_params,s={};for(let[o,i]of Object.entries(r))s[o]=String(i);n.template_params=s}return n}function vn(e){let t=typeof e.action=="string"?e.action:String(e.action),n=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",r=typeof e.locator=="string"?e.locator:void 0,s=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[c,u]of Object.entries(e))to.has(c)||(i[c]=u);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js);let a={action_description:n,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};r&&(a.locator=r),s&&(a.xpath=s);let l={uid:K(),type:"ACTION",description:n,action_entity:a};return o&&(l.use_pure_vision=!0),l}function Fe(e,t){let n;try{n=Ws(e)}catch{return}let r=n.contents;if(!r||!me(r))return;if(n.commentBefore)t.comment=n.commentBefore;else{let l=r.items?.[0];l?.key&&l.key.commentBefore&&(t.comment=l.key.commentBefore)}let s=r,o=s.get("statements",!0);z(o)&&t.statements&&ge(o,t.statements);let i=s.get("teardown",!0);z(i)&&t.teardown&&ge(i,t.teardown)}function ge(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=e.items[n];r.commentBefore&&!(n===0&&e.commentBefore)&&(t[n].comment=r.commentBefore);let s=t[n];if(s.type==="STEP"&&me(r)){let o=r.get("statements",!0);z(o)&&ge(o,s.statements)}else if(s.type==="IF_ELSE"&&me(r)){let o=r.get("THEN",!0);z(o)&&ge(o,s.then);let i=r.get("ELSE",!0);z(i)&&s.else&&ge(i,s.else)}else if(s.type==="WHILE_LOOP"&&me(r)){let o=r.get("DO",!0);z(o)&&ge(o,s.body)}}}var _n,bn,to,Ue=_(()=>{"use strict";ft();_n={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};bn=1024*1024;to=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as ga,stringify as ma}from"yaml";var Pn=_(()=>{"use strict";Ue()});var mt,Be,He=_(()=>{"use strict";mt=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},Be=e=>{let t=[],n=r=>{for(let s of r){t.push(s);let o=mt(s);for(let i of o)n(i.statements)}};return n(e),t}});function In(e){let t=0,n=0;for(let r of e)if(r.type==="DRAFT")n++;else if(r.type==="ACTION"){let s=r.action_entity?.action_data?.action_name??"";Mn.has(s)||t++}return{action:t,draft:n}}function ro(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function $n(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function so(e){let t=e.split(/\r?\n/).map(s=>s.trim()).filter(s=>s.length>0&&!s.startsWith("//")).length,n=e.match(/\bawait\b/g),r=n?n.length:0;return t<=An&&r<=En?null:`${t} non-blank line(s), ${r} await(s) \u2014 limits are ${An} lines and ${En} awaits. The js: field is a cache for one natural-language intent, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY or intent+js per UI interaction or assertion \u2014 so each step is visible in the debugger and self-healable.`}function yt(e,t){let n=t?.coverageThreshold??no,r=[],s=[],o;try{o=C(e)}catch(f){return{valid:!1,errors:[`Invalid YAML: ${f.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||r.push('Missing required field: "goal"'),o.statements?.length||r.push('Missing required field: "statements"');let i=[...Be(o.statements??[]),...o.teardown?Be(o.teardown):[]],{action:a,draft:l}=In(i),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 i){let g=f;if(g.reference_id!==void 0){let h=g.description||f.uid;r.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 v=h.action_entity?.action_data?.kwargs?.code;if(typeof v=="string"){let w=ro(v);if(w){let m=h.description||p;r.push(`Invalid JS in "${m}": ${w}. ${c}`)}else if(p==="verify"||p==="js_action"){let m=so(v);if(m){let y=h.description||p;r.push(`JS cache for "${y}" is too complex: ${m}`)}}}}}if(f.type==="IF_ELSE"){let h=f;if(h.condition.type==="JS_CODE"){let p=$n(h.condition.expression);p&&r.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=$n(h.condition.expression);p&&r.push(`Invalid JS in WHILE condition "${h.condition.expression}": ${p}. ${c}`)}}}let u=a+l,d=u>0?Math.round(a/u*100):0;return u>0&&d/100<n&&s.push(`Low action coverage: ${a}/${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:r.length===0,errors:r,warnings:s,stats:{total:u,action:a,draft:l,coverage:d}}}var no,An,En,Mn,On=_(()=>{"use strict";Ue();He();no=.5,An=5,En=3,Mn=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});function wt(){return{version:"1.0",entries:{}}}var Ln=_(()=>{"use strict";He()});import{v4 as ka}from"uuid";var Rn=_(()=>{"use strict"});var V,bt,Cn=_(()=>{"use strict";V=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(V||{}),bt=18e4});var We=_(()=>{"use strict"});var Nn=_(()=>{"use strict";We()});var Dn,io,jn,Fn,Ae,ao,co,Un=_(()=>{"use strict";Dn=112,io=1080-Dn,jn={"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"}},Fn={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"]},Ae=(e,t=!1)=>{let n=["chromium"];return t&&n.push("webkit"),Fn[e].map(r=>jn[r]).filter(r=>r.defaultBrowserType&&n.includes(r.defaultBrowserType))},ao={desktop:{label:"Desktop",type:"desktop",devices:Ae("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Ae("mobile")}},co={desktop:{label:"Desktop",type:"desktop",devices:Ae("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Ae("mobile",!0)}}});var Bn=_(()=>{"use strict"});var Hn=_(()=>{"use strict"});var Wn=_(()=>{"use strict"});var Gn=_(()=>{"use strict"});var Kn=_(()=>{"use strict"});var zn=_(()=>{"use strict"});var Vn=_(()=>{"use strict"});var Yn=_(()=>{"use strict";We()});var ce=_(()=>{"use strict";Pn();On();Ue();ft();Ln();Rn();He();Cn();Nn();Un();Bn();Hn();Wn();Gn();Kn();zn();Vn();Yn();We()});import{stringify as po}from"yaml";import{createHash as $o}from"crypto";import{parse as Mo,stringify as sr}from"yaml";import{readFileSync as Io,existsSync as Oo}from"fs";import{resolve as St,dirname as Lo}from"path";import{parse as Zn,stringify as Ro}from"yaml";import{readFileSync as Bo,writeFileSync as Ho,mkdirSync as Wo}from"fs";import{dirname as Go}from"path";function oe(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function I(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 ho(e){let t=e.xpath;return typeof t=="string"&&t.trim()?!t.startsWith("xpath=")&&!t.startsWith("/")&&!t.startsWith("//")?`xpath=//${t}`:t.startsWith("xpath=")?t:`xpath=${t}`:null}function Ke(e){let t=uo(e),n=e.locator;if(typeof n=="string"&&n.trim())return n=n.trim(),n.endsWith("first()")?`${t}.${n}`:`${t}.${n}.first()`;let r=ho(e);if(r){let s=JSON.stringify(r);return`${t}.locator(${s}).first()`}return null}function Jn(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:fo.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 S(e,t){R.set(e,t)}function yo(e){return R.get(e)}function ue(e,t,n=[]){let r=[...n];return t.locator?r.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&r.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(t.frame_path)}`),r.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...r.map(s=>` ${s},`),"});"]}function qn(e){let t=e.functionName;if(!t)return null;let n=Array.isArray(e.args)?e.args.map(String):[];if(n.length===0)return`await ${t}()`;let r=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=n.map(i=>r.includes(i)||s.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function X(e,t,n,r="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${r}.${o}`,l=wo(i,t,a,n);l.length>0&&(s.push(...l),o<e.length-1&&s.push(""))}return s}function wo(e,t,n,r){let s=" ".repeat(t);switch(e.type){case"DRAFT":return bo(e,t,n,r);case"ACTION":return vo(e,t,n,r);case"STEP":return So(e,t,n,r);case"IF_ELSE":return _o(e,t,n,r);case"WHILE_LOOP":return xo(e,t,n,r);default:return[`${s}// Unknown statement type: ${e.type}`]}}function bo(e,t,n,r){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.run(page, ${i}, '${n}');`]}function vo(e,t,n,r){let s=" ".repeat(t),o=e.description,i=e.uid,l=r.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!l){if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let y=JSON.stringify(o),k=!!e.use_pure_vision;return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.execute(page, ${y}, '${n}', ${k});`]}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[`${s}// ${n}: Unknown action: ${u}`];let g={imports:r.imports},h=f(c,n,g);if(r.noAgent){if(Jn(c))return[`${s}// ${n}: ${I(d)}`,`${s}// AI action: ${I(d)} (requires agent - skipped in hook)`];let y=To(c,u,s,n);return y||[`${s}// ${n}: ${I(d)}`,...h.map(k=>`${s}${k}`)]}if(Jn(c))return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,...h.map(y=>`${s}${y}`)];let p=JSON.stringify(d),v=h.map(y=>`${s} ${y}`),w=mo(c),m=i?`'${i}'`:"undefined";return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...v,`${s}}, ${p}, '${n}', ${m}, ${w});`]}function So(e,t,n,r){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${I(e.description)}`);let i=X(e.statements,t,r,n);return o.push(...i),o}function _o(e,t,n,r){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${n}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${I(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${n}")) {`)}let i=X(e.then,t+1,r,`${n}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=X(e.else,t+1,r,`${n}.else`);o.push(...a)}return o.push(`${s}}`),o}function xo(e,t,n,r){let s=" ".repeat(t),o=[];o.push(`${s}// ${n}: Loop`);let i=e.timeout_ms??bt,a=i/1e3,l=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,c=`loop_${n.replace(/\./g,"_")}`;if(o.push(`${s}const ${c}_start = Date.now();`),o.push(`${s}const ${c}_timeout = ${i};`),o.push(`${s}const ${c}_check = () => {`),o.push(`${s} if (Date.now() - ${c}_start > ${c}_timeout) {`),o.push(`${s} throw new Error('${l}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${c}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${I(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${s}while (${c}_check() && await agent.evaluate(page, ${d}, "${n}")) {`)}let u=X(e.body,t+1,r,`${n}.body`);return o.push(...u),o.push(`${s}}`),o}function To(e,t,n,r){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??_t;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${n}// ${r}: ${I(s)}`,`${n}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goBack();`];case"go_forward":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goForward();`];case"input_text":{let a=o.text||"",l=Ke(e);return l?[`${n}// ${r}: ${I(s)}`,`${n}await ${l}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",l=Ke(e);return l?[`${n}// ${r}: ${I(s)}`,`${n}await ${l}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function ko(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...nr()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(n.push(...Xn("beforeEach",t.beforeEach,o)),n.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",l=vt(t?.tags);for(let c of t.parameters){let u=tr(e,c.values);n.push(...Ve(u,`${l}${oe(a)} [${oe(c.name)}]`,o,0,i)),n.push("")}}else{let a=t?.testName||e.goal||"Generated test",l=vt(t?.tags);n.push(...Ve(e,`${l}${oe(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(n.push(""),n.push(...Xn("afterEach",t.afterEach,o))),rr(n,s),n.join(`
639
+ `)}function Po(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...nr()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore},i=t?.testName||"Test Suite",a=vt(t?.tags);n.push(`test.describe.serial('${a}${oe(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(n.push(...Ge("beforeAll",e.beforeAll,o,1)),n.push("")),e.beforeEach&&e.beforeEach.length>0&&(n.push(...Ge("beforeEach",e.beforeEach,o,1)),n.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=tr(u.testFlow,f.values);n.push(...Ve(g,`${oe(u.name)} [${oe(f.name)}]`,o,1,d)),n.push("")}else n.push(...Ve(u.testFlow,oe(u.name),o,1,d)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&n.push("")}return e.afterEach&&e.afterEach.length>0&&(n.push(...Ge("afterEach",e.afterEach,o,1)),n.push("")),e.afterAll&&e.afterAll.length>0&&n.push(...Ge("afterAll",e.afterAll,o,1)),n.push("});"),rr(n,s),n.join(`
640
+ `)}function vt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function ze(e){let t=new Set;function n(r){for(let s of r)switch(s.type){case V.ACTION:{let i=s.action_entity?.action_data?.kwargs;if(i?.args&&Array.isArray(i.args))for(let a of i.args)typeof a=="string"&&Ao.includes(a)&&t.add(a);break}case V.STEP:n(s.statements);break;case V.IF_ELSE:{let o=s;n(o.then),o.else&&n(o.else);break}case V.WHILE_LOOP:n(s.body);break}}return n(e),t}function Eo(e){let t=ze(e.statements??[]);if(e.teardown)for(let n of ze(e.teardown))t.add(n);return t}function xt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ve(e,t,n,r=0,s){let o=" ".repeat(r),i=[],a=Eo(e),l=xt(a),c=s?.only?"test.only":"test";i.push(`${o}${c}('${t}', async (${l}) => {`),s?.skip===!0?i.push(`${o} test.skip();`):typeof s?.skip=="string"&&i.push(`${o} test.skip(true, '${oe(s.skip)}');`),s?.fail===!0?i.push(`${o} test.fail();`):typeof s?.fail=="string"&&i.push(`${o} test.fail(true, '${oe(s.fail)}');`),s?.slow&&i.push(`${o} test.slow();`),s?.timeout&&i.push(`${o} test.setTimeout(${s.timeout});`);let u=e.teardown&&e.teardown.length>0,d=r+1;if(u){if(i.push(`${o} try {`),e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let g=X(e.statements,d+1,n);i.push(...g)}i.push(`${o} } finally {`),i.push(`${o} // Teardown`);let f=X(e.teardown,d+1,n,"teardown");i.push(...f),i.push(`${o} }`)}else if(e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let f=X(e.statements,d,n);i.push(...f)}return i.push(`${o}});`),i}function Xn(e,t,n){let r=[],s=er(t),o=ze(s),i=xt(o);return r.push(`test.${e}(async (${i}) => {`),r.push(...X(s,1,n,e)),r.push("});"),r}function Ge(e,t,n,r){let s=" ".repeat(r),o=[],i=er(t);if(e==="beforeAll"||e==="afterAll"){let l={...n,noAgent:!0};o.push(`${s}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${s} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...X(i,r+1,l,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let l=ze(i),c=xt(l);o.push(`${s}test.${e}(async (${c}) => {`),o.push(...X(i,r+1,n,e)),o.push(`${s}});`)}return o}function er(e){let n=po({goal:"_hook",statements:e});return C(n).statements??[]}function tr(e,t){let n=ke(e);for(let[r,s]of Object.entries(t))n=n.split(`<<${r}>>`).join(String(s));return C(n)}function nr(){return["import { test, expect } from 'shiplightai/fixture';"]}function rr(e,t){if(t.size>0){let n=0;for(let s=0;s<e.length;s++)e[s].startsWith("import ")&&(n=s+1);let r=Array.from(t);e.splice(n,0,...r)}}function or(e,t,n){let r={expandingPaths:new Set([St(t)]),depth:0,referencedPaths:new Set,basePath:n},s={...e};Array.isArray(s.statements)&&(s.statements=le(s.statements,t,r)),Array.isArray(s.teardown)&&(s.teardown=le(s.teardown,t,r));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=le(s[o],t,r));return{doc:s,referencedTemplatePaths:Array.from(r.referencedPaths)}}function le(e,t,n){let r=[];for(let s of e)if(Co(s)){let o=No(s,t,n);r.push(o)}else r.push(Do(s,t,n));return r}function Co(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function No(e,t,n){if(n.depth>=Qn)throw new Error(`Template expansion exceeded maximum depth of ${Qn}. Check for deeply nested or circular template references.`);let r=St(Lo(t),e.template),s=!Oo(r)&&n.basePath?St(n.basePath,e.template):r;if(n.expandingPaths.has(s))throw new Error(`Circular template reference detected: ${s} is already being expanded. Stack: ${Array.from(n.expandingPaths).join(" \u2192 ")} \u2192 ${s}`);n.referencedPaths.add(s);let o;try{o=Io(s,"utf-8")}catch(h){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${h.message}`)}let i=Zn(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let a=i.params||[],l=e.params||{};for(let h of a)if(!(h in l))throw new Error(`Template ${e.template} requires param "${h}" but it was not provided. Required params: [${a.join(", ")}]`);let c=i.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[v,w]of Object.entries(l))p=p.split(`<<${v}>>`).join(String(w));c=Zn(p)}let u={expandingPaths:new Set([...n.expandingPaths,s]),depth:n.depth+1,referencedPaths:n.referencedPaths},d=le(c,s,u),g={STEP:i.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,n){if(typeof e!="object"||e===null)return e;let r={...e};return Array.isArray(r.statements)&&(r.statements=le(r.statements,t,n)),Array.isArray(r.THEN)&&(r.THEN=le(r.THEN,t,n)),Array.isArray(r.ELSE)&&(r.ELSE=le(r.ELSE,t,n)),Array.isArray(r.DO)&&(r.DO=le(r.DO,t,n)),r}function kt(e,t,n){let r=Mo(e),s=r?.name,o=r?.tags,i=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new Tt('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Fo(r,s,o,i,t,n)}return jo(r,s,o,i,t,n)}function jo(e,t,n,r,s,o){let i=e?.beforeEach,a=e?.afterEach,l=ir(e?.parameters),c=e?.timeout,u=e?.skip,d=e?.fail,f=e?.only,g=e?.slow,h=e?.settings,p=r;if(h){let y={};h.auto_dismiss_modal!==void 0&&(y.autoDismissModal=!!h.auto_dismiss_modal),h.browser_timezone!==void 0&&h.browser_timezone!==null&&(y.timezoneId=String(h.browser_timezone)),h.browser_language!==void 0&&h.browser_language!==null&&(y.locale=String(h.browser_language)),h.extra_http_headers!==void 0&&h.extra_http_headers!==null&&typeof h.extra_http_headers=="object"&&(y.extraHTTPHeaders=h.extra_http_headers),Object.keys(y).length>0&&(p={...p,...y})}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,delete e.settings),e?.url)throw new Tt(`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 v=[];if(s&&e&&typeof e=="object"){let y=or(e,s,o);e=y.doc,v=y.referencedTemplatePaths}let w=sr(e),m=C(w);return s&&(ye(m.statements??[],s,"main"),m.teardown&&ye(m.teardown,s,"teardown")),{testFlow:m,name:t,tags:n,use:p,beforeEach:i,afterEach:a,parameters:l,timeout:c,skip:u,fail:d,only:f,slow:g,referencedTemplatePaths:v}}function Fo(e,t,n,r,s,o){let i=e.suite;if(!Array.isArray(i.tests)||i.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=i.beforeAll,l=i.afterAll,c=i.beforeEach,u=i.afterEach,d=[],f=i.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 v={goal:p.name,statements:p.statements};p.teardown&&(v.teardown=p.teardown);let w=[],m=v;if(s&&typeof v=="object"){let E=or(v,s,o);m=E.doc,w=E.referencedTemplatePaths,d.push(...w)}let y=sr(m),k=C(y),P=ir(p.parameters);return{testFlow:k,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:P,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=i.base_url,h=g?{...r,baseURL:g}:r;return{suite:{beforeAll:a,afterAll:l,beforeEach:c,afterEach:u,tests:f},name:t,tags:n,use:h,referencedTemplatePaths:d}}function ir(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,n)=>{if(!t.name)throw new Error(`Parameter set at index ${n} 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 ye(e,t,n){for(let r=0;r<e.length;r++){let s=e[r],o=`${n}.${r}`,i=s.description||"";if(s.uid=Uo(t,o,i),s.type===V.STEP)ye(s.statements,t,o);else if(s.type===V.IF_ELSE){let a=s;ye(a.then,t,`${o}.then`),a.else&&ye(a.else,t,`${o}.else`)}else s.type===V.WHILE_LOOP&&ye(s.body,t,`${o}.body`)}}function Uo(e,t,n){let r=$o("sha256").update(`${e}:${t}:${n}`).digest("hex");return`${r.slice(0,8)}-${r.slice(8,12)}-${r.slice(12,16)}-${r.slice(16,20)}-${r.slice(20,32)}`}function ar(e,t){let n;try{n=Bo(e,"utf-8")}catch(r){return{valid:!1,errors:[`Failed to read file: ${r.message}`],warnings:[]}}return Ko(n,e,t)}function Ko(e,t,n){let r=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=r||s?null:yt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,l=[];try{let c=n?.parsed??kt(e,t);l=c.referencedTemplatePaths;let u={version:n?.version,actionEntityStore:n?.actionEntityStore},d=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?i=Po(c.suite,{...u,testName:c.name,tags:c.tags,use:c.use}):i=ko(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=i.split(`
641
641
  `).filter(g=>!g.startsWith("import ")).join(`
642
- `);new Function(h),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),jo(Bo(a),{recursive:!0}),Uo(a,i)}catch(c){let d=c instanceof xt?"":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}.${d}`],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:a,referencedTemplatePaths:l}}var St,po,uo,R,To,qn,xt,kt=_(()=>{"use strict";ae();ae();ae();ae();St=5e3;po=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],uo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];R=new Map;S("click",e=>{let t=He(e);if(!t)return['await agent.execAction("click", page, {});'];let n=e.action_data?.kwargs?.timeout_ms??St;return[`await ${t}.click({ timeout: ${n} });`]});S("click_element",R.get("click"));S("click_element_by_index",R.get("click"));S("double_click",e=>de("double_click",e));S("double_click_on_element",R.get("double_click"));S("right_click",e=>de("right_click",e));S("right_click_on_element",R.get("right_click"));S("hover",e=>de("hover",e));S("hover_element_by_index",R.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return de("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",R.get("input_text"));S("clear_input",e=>de("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",R.get("press"));S("send_keys_on_element",e=>{let t=He(e),n=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(n)} } },`,"});"];let r=e.action_data?.kwargs?.timeout_ms??St;return[`await ${t}.press(${JSON.stringify(n)}, { timeout: ${r} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return de("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",R.get("scroll"));S("scroll_up",R.get("scroll"));S("scroll_element",R.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=>de("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",R.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 n=e.action_data?.kwargs,r=typeof n?.code=="string",s=r?n?.statement||e.action_description:e.action_description||n?.statement;if(r&&s){let i=n.code.split(`
642
+ `);new Function(f),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Wo(Go(a),{recursive:!0}),Ho(a,i)}catch(c){let u=c instanceof Tt?"":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:a,referencedTemplatePaths:l}}var _t,fo,go,R,Ao,Qn,Tt,Pt=_(()=>{"use strict";ce();ce();ce();ce();_t=5e3;fo=["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"];R=new Map;S("click",e=>{let t=Ke(e);if(!t)return['await agent.execAction("click", page, {});'];let n=e.action_data?.kwargs?.timeout_ms??_t;return[`await ${t}.click({ timeout: ${n} });`]});S("click_element",R.get("click"));S("click_element_by_index",R.get("click"));S("double_click",e=>ue("double_click",e));S("double_click_on_element",R.get("double_click"));S("right_click",e=>ue("right_click",e));S("right_click_on_element",R.get("right_click"));S("hover",e=>ue("hover",e));S("hover_element_by_index",R.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ue("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",R.get("input_text"));S("clear_input",e=>ue("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",R.get("press"));S("send_keys_on_element",e=>{let t=Ke(e),n=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(n)} } },`,"});"];let r=e.action_data?.kwargs?.timeout_ms??_t;return[`await ${t}.press(${JSON.stringify(n)}, { timeout: ${r} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ue("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",R.get("scroll"));S("scroll_up",R.get("scroll"));S("scroll_element",R.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=>ue("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",R.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 n=e.action_data?.kwargs,r=typeof n?.code=="string",s=r?n?.statement||e.action_description:e.action_description||n?.statement;if(r&&s){let i=n.code.split(`
643
643
  `),a=JSON.stringify(s);return["{ const _t = Date.now(); try {",...i.map(l=>` ${l}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${a}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${a}\`);`,` await agent.assert(page, ${a}, ${JSON.stringify(t||"")});`,"} }"]}return r?n.code.split(`
644
644
  `):s?[`await agent.assert(page, ${JSON.stringify(s)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",R.get("verify"));S("assert",R.get("verify"));S("ai_action",(e,t)=>{let n=e.action_data?.kwargs?.statement;if(!n)return["// Skipping ai_action: missing statement"];let r=JSON.stringify(n),s=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${r}, '${t||""}', ${s});`]});S("ai_step",(e,t)=>{let n=e.action_data?.kwargs?.statement;return n?[`await agent.run(page, ${JSON.stringify(n)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});S("ai_extract",(e,t)=>{let n=e.action_data?.kwargs?.element_description,r=e.action_data?.kwargs?.variable_name;if(!n||!r)return["// Skipping ai_extract: missing element_description or variable_name"];let s=JSON.stringify(n),o=JSON.stringify(r);return[`await agent.extract(page, ${s}, ${o}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let n=e.action_data?.kwargs?.condition,r=e.action_data?.kwargs?.timeout_seconds||60;return n?[`await agent.waitUntilCondition(page, ${JSON.stringify(n)}, ${r}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});S("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",n=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(n)} } },`,"});"]});S("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let n=["{"],r=t.split(`
645
- `);for(let s of r)n.push(` ${s}`);return n.push("}"),n});S("function",(e,t,n)=>{let r=e.action_data?.kwargs||{},s=r.functionName;if(s&&s.includes("#")){let[i,a]=s.split("#");if(i&&a){let l=i.replace(/\.(ts|js|mjs)$/,""),c=`import { ${a} } from '${l}';`;n?.imports?.add(c);let d={...r,functionName:a},u=zn(d);return u?[u.endsWith(";")?u:`${u};`]:["// Skipping function: invalid export pattern"]}}let o=zn(r);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||{},n=[],r={};return t.paths?r.paths=t.paths:t.path&&(r.path=t.path),t.use_file_input&&(r.use_file_input=!0),n.push(`action_data: { kwargs: ${JSON.stringify(r)} }`),e.locator?n.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&n.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...n.map(s=>` ${s},`),"});"]});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??"",n=[];return n.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?n.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&n.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...n.map(r=>` ${r},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
646
- `):["// Skipping js_action: missing code"]});To=["testContext","request"];qn=5;xt=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as Wo}from"express";import*as T from"fs/promises";import*as x from"path";import{stringify as Ke,parse as Pe}from"yaml";function ue(e){if(!e||e.length===0)return[];let n=Ke({goal:"_hook",statements:e});return C(n).statements??[]}async function or(e){try{let t=await T.readdir(e,{withFileTypes:!0});for(let n of t)if(!(n.name==="node_modules"||n.name.startsWith("."))&&(n.isFile()&&n.name.endsWith(".test.yaml")||n.isDirectory()&&await or(x.join(e,n.name))))return!0}catch{}return!1}function Ve(e){let t=2166136261;for(let n=0;n<e.length;n++)t^=e.charCodeAt(n),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function Go(e){let t=new Map,n;try{n=await T.readdir(e)}catch(r){if(r.code==="ENOENT")return t;throw r}for(let r of n.filter(s=>s.endsWith(".yaml")))try{let s=x.join(e,r),i=Pe(await T.readFile(s,"utf-8"))?.name??x.basename(r,".yaml");t.set(Ve(i),`templates/${r}`)}catch{}return t}function At(e,t){if(Array.isArray(e))return e.map(o=>At(o,t));if(!e||typeof e!="object")return e;let n=e,r={};for(let[o,i]of Object.entries(n))r[o]=At(i,t);let s=typeof n.reference_id=="number"?n.reference_id:void 0;if(s!==void 0&&typeof r.template_path!="string"){let o=t.get(s);o&&(r.template_path=o,delete r.reference_id)}return r}function Pt(e){return Ce({statements:e}).statements}function ir(e){let{initialDir:t,initialFile:n,projectRoot:r,onFileSelected:s}=e,o=Wo();function i(){return x.join(r??t,"fixtures")}o.get("/api/files",async(l,c)=>{try{let d=typeof l.query.dir=="string"?l.query.dir:t,u=x.resolve(d),h=await T.readdir(u,{withFileTypes:!0}),g=[];for(let p of h)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let w=x.join(u,p.name);await or(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(u,p.name)});g.sort((p,w)=>p.type!==w.type?p.type==="directory"?-1:1:p.name.localeCompare(w.name));let f=x.dirname(u);c.json({dir:u,parent:f!==u?f:null,entries:g,initialFile:n??null,projectRoot:r??t})}catch(d){console.error("[debugger] Error listing files:",d),c.status(500).json({error:d.message})}});function a(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 n?{filePath:n}:{error:"No file specified. Pass ?file= parameter."}}return o.get("/api/test-flow",async(l,c)=>{let d=a(l.query.file);if("error"in d)return c.status(400).json({error:d.error});let u=d.filePath;try{s?.(u);let h=await T.readFile(u,"utf-8"),g=await T.stat(u),f=Te(h),p=Tt(h,u,r);if(p.suite){let w=p.suite,m={tests:w.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:ue(w.beforeAll),afterAll:ue(w.afterAll),beforeEach:ue(w.beforeEach),afterEach:ue(w.afterEach)},y={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:m};c.json({isSuite:!0,testFlow:y,metadata:f,name:p.name,tags:p.tags,use:p.use,filePath:u,fileName:x.basename(u),lastModified:g.mtimeMs})}else{let w=p.testFlow;Ne(h,w),c.json({isSuite:!1,testFlow:w,metadata:f,name:p.name,tags:p.tags,use:p.use,filePath:u,fileName:x.basename(u),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return c.status(404).json({error:`File not found: ${u}`});console.error("[debugger] Error loading test flow:",h),c.status(500).json({error:h.message})}}),o.put("/api/test-flow",async(l,c)=>{try{let d=a(l.query.file);if("error"in d)return c.status(400).json({error:d.error});let u=d.filePath,{testFlow:h,metadata:g}=l.body;if(!h)return c.status(400).json({error:"testFlow is required"});let f=x.join(r??t,"templates"),p=await Go(f),w=At(h,p),m=xe(w,g),y=u+".tmp";await T.writeFile(y,m,"utf-8"),await T.rename(y,u);let v=await T.stat(u);c.json({success:!0,lastModified:v.mtimeMs})}catch(d){console.error("[debugger] Error saving test flow:",d),c.status(500).json({error:d.message})}}),o.get("/api/fixtures",async(l,c)=>{try{let d=i();try{let h=(await T.readdir(d,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();c.json({files:h,dir:d})}catch(u){if(u.code==="ENOENT")c.json({files:[],dir:d});else throw u}}catch(d){console.error("[debugger] Error listing fixtures:",d),c.status(500).json({error:d.message})}}),o.post("/api/fixtures",async(l,c)=>{try{let d=i();await T.mkdir(d,{recursive:!0});let{name:u,content:h}=l.body;if(!u||!h)return c.status(400).json({error:"name and content are required"});let g=x.basename(u);if(!g)return c.status(400).json({error:"Invalid file name"});let f=x.join(d,g);await T.writeFile(f,Buffer.from(h,"base64")),c.json({fileName:g})}catch(d){console.error("[debugger] Error saving fixture:",d),c.status(500).json({error:d.message})}}),o.get("/api/functions",async(l,c)=>{let d=x.join(r??t,"helpers"),u;try{u=await T.readdir(d)}catch(p){return p.code==="ENOENT"?c.json([]):(console.error("[debugger] Error reading helpers dir:",p),c.status(500).json({error:p.message}))}let h=u.filter(p=>/\.(ts|js|mjs)$/.test(p)),g=[],f=1;for(let p of h){let w=await T.readFile(x.join(d,p),"utf-8"),m=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,y;for(;(y=m.exec(w))!==null;){let[,v,P]=y,A=P.replace(/\s*:\s*[^,)]+/g,"");g.push({id:f++,name:`helpers/${p}#${v}`,description:"",status:"Active",code:`async function ${v}${A} {}`})}}c.json(g)}),o.get("/api/reusable-steps",async(l,c)=>{let d=x.join(r??t,"templates"),u;try{u=await T.readdir(d)}catch(f){return f.code==="ENOENT"?c.json([]):(console.error("[debugger] Error reading templates dir:",f),c.status(500).json({error:f.message}))}let h=u.filter(f=>f.endsWith(".yaml")).sort(),g=[];for(let f of h)try{let p=await T.readFile(x.join(d,f),"utf-8"),w=Pe(p);if(!w||typeof w!="object")continue;let m=w.name??x.basename(f,".yaml"),y=w.description??"",v=ue(w.statements);g.push({id:Ve(m),organizationId:"local",name:m,description:y,statements:v})}catch(p){console.error(`[debugger] Error parsing template ${f}:`,p)}c.json(g)}),o.post("/api/reusable-steps/exists/:name",async(l,c)=>{let d=x.join(r??t,"templates"),u=decodeURIComponent(l.params.name).toLowerCase(),h;try{h=await T.readdir(d)}catch(g){return g.code==="ENOENT"?c.json(!1):c.status(500).json({error:g.message})}for(let g of h.filter(f=>f.endsWith(".yaml")))try{let f=await T.readFile(x.join(d,g),"utf-8");if((Pe(f)?.name??x.basename(g,".yaml")).toLowerCase()===u)return c.json(!0)}catch{}c.json(!1)}),o.post("/api/reusable-steps",async(l,c)=>{let d=x.join(r??t,"templates"),{reusableStep:u}=l.body??{};if(!u?.name||!Array.isArray(u.statements))return c.status(400).json({error:"reusableStep.name and reusableStep.statements are required"});await T.mkdir(d,{recursive:!0});let h=u.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=x.join(d,`${h}.yaml`),f={name:u.name,statements:Pt(u.statements)};u.description&&(f.description=u.description),await T.writeFile(g,Ke(f),"utf-8"),c.json({id:Ve(u.name),organizationId:"local",name:u.name,description:u.description??"",statements:u.statements})}),o.put("/api/reusable-steps/:id/update",async(l,c)=>{let d=parseInt(l.params.id,10),u=x.join(r??t,"templates"),h;try{h=await T.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 h.filter(f=>f.endsWith(".yaml")))try{let f=x.join(u,g),p=Pe(await T.readFile(f,"utf-8"))??{},w=p.name??x.basename(g,".yaml");if(Ve(w)!==d)continue;let m=l.body??{};return m.description!==void 0&&(p.description=m.description),m.statements!==void 0&&(p.statements=Pt(m.statements)),await T.writeFile(f,Ke(p),"utf-8"),c.json({id:d,organizationId:"local",...p,statements:m.statements!==void 0?m.statements:ue(p.statements)})}catch{}c.status(404).json({error:`Template with id ${d} not found`})}),o.put("/api/templates/:name",async(l,c)=>{let d=x.join(r??t,"templates"),u=decodeURIComponent(l.params.name),h;try{h=await T.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 h.filter(f=>f.endsWith(".yaml")))try{let f=x.join(d,g),p=Pe(await T.readFile(f,"utf-8"))??{};if((p.name??x.basename(g,".yaml"))!==u)continue;let m=l.body??{};return m.name!==void 0&&(p.name=m.name),m.description!==void 0&&(p.description=m.description),m.statements!==void 0&&(p.statements=Pt(m.statements)),await T.writeFile(f,Ke(p),"utf-8"),c.json({organizationId:"local",...p,statements:m.statements!==void 0?m.statements:ue(p.statements)})}catch{}c.status(404).json({error:`Template "${u}" not found`})}),o}var ar=_(()=>{"use strict";ae();kt()});import*as dr from"http";import*as ur from"net";import*as le from"path";import*as hr from"fs";import{randomUUID as Ko}from"crypto";function lr(e){return new Promise((t,n)=>{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 r=[];e.on("data",s=>r.push(s)),e.on("end",()=>t(Buffer.concat(r))),e.on("error",n)})}function pr(e,t,n,r,s,o){return new Promise(i=>{let a=Array.isArray(o)?o[0]:o,l=dr.request({hostname:n,port:r,path:e.originalUrl,method:e.method,headers:{"content-type":a||"application/json","content-length":String(s.length)},timeout:3e5},c=>{t.writeHead(c.statusCode??502,c.headers),c.on("data",d=>t.write(d)),c.on("end",()=>{t.end(),i()}),c.on("error",()=>{t.writableEnded||t.end(),i()})});l.on("error",c=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+c.message}),i()}),l.end(s)})}function Et(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var cr,ze,fr=_(()=>{"use strict";pt();cr=1e4,ze=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??lt,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(n){this.log(`[manager] onSessionStateChange listener threw: ${n.message}`)}}async openSession(t){let n=le.resolve(t),r=this.byYamlPath.get(n);if(r){let c=this.sessions.get(r);if(c&&c.session.status!=="ended")return{...c.session};this.byYamlPath.delete(n)}if(!hr.existsSync(n))throw new Error(`YAML file not found: ${n}`);let s=Le(le.dirname(n));if(!s)throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Ko().slice(0,8)}`,i=new Date().toISOString(),a={sessionId:o,yamlPath:n,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:i,status:"starting",exitInfo:null},l={session:a,cleanup:async()=>{},readyPromise:Promise.resolve()};this.sessions.set(o,l),this.byYamlPath.set(n,o),this.notifyStateChange(a);try{let c=cr+18e4,d,u=new Promise((f,p)=>{d=setTimeout(()=>p(new Error(`Timed out after ${c/1e3}s waiting for inner playwright to register on a port.`)),c)}),h;try{h=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:o,headed:this.headed}),u])}finally{d&&clearTimeout(d)}a.innerPort=h.port,a.innerHost=h.host,a.pid=h.pid,a.status="running",l.cleanup=h.cleanup;let g=f=>{a.status!=="ended"&&(a.status="ended",a.exitInfo=f,this.notifyStateChange(a))};return l.livenessTimer=this.startLivenessProbe(o,g),this.notifyStateChange(a),this.log(`[manager] session ${o} running on ${a.innerHost}:${a.innerPort} (yaml=${le.basename(n)})`),{...a}}catch(c){throw this.sessions.delete(o),this.byYamlPath.get(n)===o&&this.byYamlPath.delete(n),a.status="ended",a.exitInfo=c.message,this.notifyStateChange(a),c}}startLivenessProbe(t,n){let o=0,i=setInterval(()=>{let a=this.sessions.get(t);if(!a||a.session.status==="ended"){clearInterval(i);return}let l=!1;try{process.kill(a.session.pid,0),l=!0}catch(c){c?.code!=="ESRCH"&&(l=!0)}l?o=0:(o+=1,o>=3&&(clearInterval(i),n("process-exited")))},3e3);return i.unref(),i}async closeSession(t){let n=this.sessions.get(t);if(n){this.sessions.delete(t),this.byYamlPath.get(n.session.yamlPath)===t&&this.byYamlPath.delete(n.session.yamlPath),n.livenessTimer&&(clearInterval(n.livenessTimer),n.livenessTimer=void 0),n.session.status!=="ended"&&(n.session.status="ended",n.session.exitInfo="SIGTERM",this.notifyStateChange(n.session));try{await n.cleanup()}catch(r){this.log(`[manager] closeSession ${t} cleanup error: ${r.message}`)}}}async restartInner(t){let n=this.sessions.get(t);if(!n)throw new Error(`No session ${t} to restart`);if(n.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let r=n.session.yamlPath,s=Le(le.dirname(r));if(!s)throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);n.restartInProgress=!0;try{n.livenessTimer&&(clearInterval(n.livenessTimer),n.livenessTimer=void 0);try{await n.cleanup()}catch(c){this.log(`[manager] restartInner ${t} old cleanup error: ${c.message}`)}n.session.status="starting",n.session.innerPort=0,n.session.pid=0,this.notifyStateChange(n.session);let o=cr+18e4,i,a=new Promise((c,d)=>{i=setTimeout(()=>d(new Error(`Timed out after ${o/1e3}s waiting for inner playwright to respawn.`)),o)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:r,configPath:s,tempSuffix:t,headed:this.headed}),a])}catch(c){throw n.session.status="ended",n.session.exitInfo=c.message,this.notifyStateChange(n.session),c}finally{i&&clearTimeout(i)}n.session.innerPort=l.port,n.session.innerHost=l.host,n.session.pid=l.pid,n.session.status="running",n.cleanup=l.cleanup,n.livenessTimer=this.startLivenessProbe(t,c=>{n.session.status!=="ended"&&(n.session.status="ended",n.session.exitInfo=c,this.notifyStateChange(n.session))}),this.notifyStateChange(n.session),this.log(`[manager] session ${t} restarted on ${n.session.innerHost}:${n.session.innerPort} (yaml=${le.basename(r)})`)}finally{n.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let n=this.sessions.get(t);return n?{...n.session}:void 0}httpProxyFor(t,n={}){let{liveviewUrlBuilder:r}=n;return async(s,o,i)=>{let a=this.sessions.get(t);if(!a){o.status(404).json({status:"error",message:"Session not found"});return}if(a.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:a.session.exitInfo});return}let l=`${s.method} ${s.path}`;if(l==="POST /api/int-runner/create-session"){let d=await lr(s),u={};if(d.length)try{u=JSON.parse(d.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}u.testFilePath=a.session.yamlPath,await pr(s,o,a.session.innerHost,a.session.innerPort,Buffer.from(JSON.stringify(u),"utf-8"),"application/json");return}if(l==="POST /api/int-runner/terminate-session"){try{await this.restartInner(t),o.json({status:"success",details:"Session restarted"})}catch(d){o.status(500).json({status:"error",message:d.message})}return}if(l==="POST /api/int-runner/liveview-url"){let d=r?.(s)??"";o.json({liveviewUrl:d,browserWsUrl:""});return}let c=await lr(s);await pr(s,o,a.session.innerHost,a.session.innerPort,c,s.headers["content-type"])}}wsUpgradeFor(t){return async(n,r,s,o)=>{let i=this.sessions.get(t);if(!i){Et(r,`HTTP/1.1 404 Not Found\r
645
+ `);for(let s of r)n.push(` ${s}`);return n.push("}"),n});S("function",(e,t,n)=>{let r=e.action_data?.kwargs||{},s=r.functionName;if(s&&s.includes("#")){let[i,a]=s.split("#");if(i&&a){let l=i.replace(/\.(ts|js|mjs)$/,""),c=`import { ${a} } from '${l}';`;n?.imports?.add(c);let u={...r,functionName:a},d=qn(u);return d?[d.endsWith(";")?d:`${d};`]:["// Skipping function: invalid export pattern"]}}let o=qn(r);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||{},n=[],r={};return t.paths?r.paths=t.paths:t.path&&(r.path=t.path),t.use_file_input&&(r.use_file_input=!0),n.push(`action_data: { kwargs: ${JSON.stringify(r)} }`),e.locator?n.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&n.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...n.map(s=>` ${s},`),"});"]});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??"",n=[];return n.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?n.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&n.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...n.map(r=>` ${r},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
646
+ `):["// Skipping js_action: missing code"]});Ao=["testContext","request"];Qn=5;Tt=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as zo}from"express";import*as T from"fs/promises";import*as x from"path";import{stringify as Ye,parse as Ee}from"yaml";function de(e){if(!e||e.length===0)return[];let n=Ye({goal:"_hook",statements:e});return C(n).statements??[]}async function cr(e){try{let t=await T.readdir(e,{withFileTypes:!0});for(let n of t)if(!(n.name==="node_modules"||n.name.startsWith("."))&&(n.isFile()&&n.name.endsWith(".test.yaml")||n.isDirectory()&&await cr(x.join(e,n.name))))return!0}catch{}return!1}function Je(e){let t=2166136261;for(let n=0;n<e.length;n++)t^=e.charCodeAt(n),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function Vo(e){let t=new Map,n;try{n=await T.readdir(e)}catch(r){if(r.code==="ENOENT")return t;throw r}for(let r of n.filter(s=>s.endsWith(".yaml")))try{let s=x.join(e,r),i=Ee(await T.readFile(s,"utf-8"))?.name??x.basename(r,".yaml");t.set(Je(i),`templates/${r}`)}catch{}return t}function Et(e,t){if(Array.isArray(e))return e.map(o=>Et(o,t));if(!e||typeof e!="object")return e;let n=e,r={};for(let[o,i]of Object.entries(n))r[o]=Et(i,t);let s=typeof n.reference_id=="number"?n.reference_id:void 0;if(s!==void 0&&typeof r.template_path!="string"){let o=t.get(s);o&&(r.template_path=o,delete r.reference_id)}return r}function At(e){return je({statements:e}).statements}function lr(e){let{initialDir:t,initialFile:n,projectRoot:r,onFileSelected:s}=e,o=zo();function i(){return x.join(r??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 T.readdir(d,{withFileTypes:!0}),g=[];for(let p of f)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let v=x.join(d,p.name);await cr(v)&&g.push({name:p.name,type:"directory",path:v})}else p.isFile()&&p.name.endsWith(".test.yaml")&&g.push({name:p.name,type:"file",path:x.join(d,p.name)});g.sort((p,v)=>p.type!==v.type?p.type==="directory"?-1:1:p.name.localeCompare(v.name));let h=x.dirname(d);c.json({dir:d,parent:h!==d?h:null,entries:g,initialFile:n??null,projectRoot:r??t})}catch(u){console.error("[debugger] Error listing files:",u),c.status(500).json({error:u.message})}});function a(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 n?{filePath:n}:{error:"No file specified. Pass ?file= parameter."}}return o.get("/api/test-flow",async(l,c)=>{let u=a(l.query.file);if("error"in u)return c.status(400).json({error:u.error});let d=u.filePath;try{s?.(d);let f=await T.readFile(d,"utf-8"),g=await T.stat(d),h=Pe(f),p=kt(f,d,r);if(p.suite){let v=p.suite,w={tests:v.tests.map(y=>({name:y.name,statements:y.testFlow.statements??[],teardown:y.testFlow.teardown,skip:y.skip,timeout:y.timeout,fail:y.fail,only:y.only,slow:y.slow})),beforeAll:de(v.beforeAll),afterAll:de(v.afterAll),beforeEach:de(v.beforeEach),afterEach:de(v.afterEach)},m={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:w};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 v=p.testFlow;Fe(f,v),c.json({isSuite:!1,testFlow:v,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=a(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(r??t,"templates"),p=await Vo(h),v=Et(f,p),w=ke(v,g),m=d+".tmp";await T.writeFile(m,w,"utf-8"),await T.rename(m,d);let y=await T.stat(d);c.json({success:!0,lastModified:y.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=i();try{let f=(await T.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=i();await T.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 T.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(r??t,"helpers"),d;try{d=await T.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 v=await T.readFile(x.join(u,p),"utf-8"),w=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,m;for(;(m=w.exec(v))!==null;){let[,y,k]=m,P=k.replace(/\s*:\s*[^,)]+/g,"");g.push({id:h++,name:`helpers/${p}#${y}`,description:"",status:"Active",code:`async function ${y}${P} {}`})}}c.json(g)}),o.get("/api/reusable-steps",async(l,c)=>{let u=x.join(r??t,"templates"),d;try{d=await T.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 T.readFile(x.join(u,h),"utf-8"),v=Ee(p);if(!v||typeof v!="object")continue;let w=v.name??x.basename(h,".yaml"),m=v.description??"",y=de(v.statements);g.push({id:Je(w),organizationId:"local",name:w,description:m,statements:y})}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(r??t,"templates"),d=decodeURIComponent(l.params.name).toLowerCase(),f;try{f=await T.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 T.readFile(x.join(u,g),"utf-8");if((Ee(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(r??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 T.mkdir(u,{recursive:!0});let f=d.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=x.join(u,`${f}.yaml`),h={name:d.name,statements:At(d.statements)};d.description&&(h.description=d.description),await T.writeFile(g,Ye(h),"utf-8"),c.json({id:Je(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(r??t,"templates"),f;try{f=await T.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=Ee(await T.readFile(h,"utf-8"))??{},v=p.name??x.basename(g,".yaml");if(Je(v)!==u)continue;let w=l.body??{};return w.description!==void 0&&(p.description=w.description),w.statements!==void 0&&(p.statements=At(w.statements)),await T.writeFile(h,Ye(p),"utf-8"),c.json({id:u,organizationId:"local",...p,statements:w.statements!==void 0?w.statements:de(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(r??t,"templates"),d=decodeURIComponent(l.params.name),f;try{f=await T.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=Ee(await T.readFile(h,"utf-8"))??{};if((p.name??x.basename(g,".yaml"))!==d)continue;let w=l.body??{};return w.name!==void 0&&(p.name=w.name),w.description!==void 0&&(p.description=w.description),w.statements!==void 0&&(p.statements=At(w.statements)),await T.writeFile(h,Ye(p),"utf-8"),c.json({organizationId:"local",...p,statements:w.statements!==void 0?w.statements:de(p.statements)})}catch{}c.status(404).json({error:`Template "${d}" not found`})}),o.get("/api/test-results",async(l,c)=>{let u=a(l.query.file);if("error"in u)return c.status(400).json({error:u.error});let d=r??t,f=x.basename(u.filePath,".test.yaml"),g=x.join(d,".shiplight","artifacts",f),h=[],p=null;try{let v=await T.readdir(g,{withFileTypes:!0});for(let w of v)if(w.isDirectory()){let m=x.join(g,w.name),y=await T.readdir(m);for(let k of y.filter(P=>/\.(png|jpe?g|webp)$/i.test(P))){let P=x.relative(g,x.join(m,k)),E=w.name.match(/^(.+?)_(before|after)$/),Se=E?E[1]:w.name,Oe=E?E[2]:"screenshot";h.push({url:`/api/report-assets/${f}/${P}`,label:Oe,stepId:Se})}}else/\.(webm|mp4)$/i.test(w.name)&&(p||(p=`/api/report-assets/${f}/${w.name}`));h.sort((w,m)=>w.stepId!==m.stepId?(w.stepId??"").localeCompare(m.stepId??""):w.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 pr=_(()=>{"use strict";ce();Pt()});import*as fr from"http";import*as gr from"net";import*as Z from"path";import*as mr from"fs";import{randomUUID as Yo}from"crypto";function dr(e){return new Promise((t,n)=>{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 r=[];e.on("data",s=>r.push(s)),e.on("end",()=>t(Buffer.concat(r))),e.on("error",n)})}function hr(e,t,n,r,s,o){return new Promise(i=>{let a=Array.isArray(o)?o[0]:o,l=fr.request({hostname:n,port:r,path:e.originalUrl,method:e.method,headers:{"content-type":a||"application/json","content-length":String(s.length)},timeout:3e5},c=>{t.writeHead(c.statusCode??502,c.headers),c.on("data",u=>t.write(u)),c.on("end",()=>{t.end(),i()}),c.on("error",()=>{t.writableEnded||t.end(),i()})});l.on("error",c=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+c.message}),i()}),l.end(s)})}function $t(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var ur,qe,yr=_(()=>{"use strict";ut();ur=1e4,qe=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??pt,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(n){this.log(`[manager] onSessionStateChange listener threw: ${n.message}`)}}openSession(t){let n=Z.resolve(t),r=this.byYamlPath.get(n);if(r){let c=this.sessions.get(r);if(c&&c.session.status!=="ended")return{...c.session};this.byYamlPath.delete(n)}if(!mr.existsSync(n))throw new Error(`YAML file not found: ${n}`);if(!xe(Z.dirname(n)))throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Yo().slice(0,8)}`,i=new Date().toISOString(),a={sessionId:o,yamlPath:n,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:i,status:"idle",exitInfo:null},l={session:a,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,l),this.byYamlPath.set(n,o),this.notifyStateChange(a),this.log(`[manager] session ${o} created (idle) for ${Z.basename(n)}`),{...a}}async startSandbox(t){let n=this.sessions.get(t);if(!n)throw new Error(`No session ${t} to start sandbox for`);if(n.session.status==="running"||n.session.status==="starting"){n.readyPromise&&await n.readyPromise;return}if(n.session.status==="ended")throw new Error(`Session ${t} has ended`);let r=n.session.yamlPath,s=xe(Z.dirname(r));if(!s)throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);n.session.status="starting",this.notifyStateChange(n.session);let o=(async()=>{let i=ur+18e4,a,l=new Promise((d,f)=>{a=setTimeout(()=>f(new Error(`Timed out after ${i/1e3}s waiting for inner playwright to register on a port.`)),i)}),c;try{c=await Promise.race([this.spawner({yamlFilePath:r,configPath:s,tempSuffix:t,headed:this.headed}),l])}finally{a&&clearTimeout(a)}n.session.innerPort=c.port,n.session.innerHost=c.host,n.session.pid=c.pid,n.session.status="running",n.cleanup=c.cleanup;let u=d=>{n.session.status!=="ended"&&(n.session.status="ended",n.session.exitInfo=d,this.notifyStateChange(n.session))};n.livenessTimer=this.startLivenessProbe(t,u),this.notifyStateChange(n.session),this.log(`[manager] session ${t} running on ${n.session.innerHost}:${n.session.innerPort} (yaml=${Z.basename(r)})`)})();n.readyPromise=o;try{await o}catch(i){throw n.session.status="ended",n.session.exitInfo=i.message,this.notifyStateChange(n.session),i}}async stopSandbox(t){let n=this.sessions.get(t);if(n&&n.session.status!=="idle"&&n.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),n.livenessTimer&&(clearInterval(n.livenessTimer),n.livenessTimer=void 0);try{await n.cleanup()}catch(r){this.log(`[manager] stopSandbox ${t} cleanup error: ${r.message}`)}n.session.innerPort=0,n.session.innerHost="127.0.0.1",n.session.pid=0,n.session.status="idle",n.session.exitInfo=null,n.cleanup=async()=>{},n.readyPromise=Promise.resolve(),this.notifyStateChange(n.session)}}startLivenessProbe(t,n){let o=0,i=setInterval(()=>{let a=this.sessions.get(t);if(!a||a.session.status==="ended"){clearInterval(i);return}let l=!1;try{process.kill(a.session.pid,0),l=!0}catch(c){c?.code!=="ESRCH"&&(l=!0)}l?o=0:(o+=1,o>=3&&(clearInterval(i),n("process-exited")))},3e3);return i.unref(),i}async closeSession(t){let n=this.sessions.get(t);if(n){this.log(`[manager] closeSession ${t} (status=${n.session.status})`),this.sessions.delete(t),this.byYamlPath.get(n.session.yamlPath)===t&&this.byYamlPath.delete(n.session.yamlPath),n.livenessTimer&&(clearInterval(n.livenessTimer),n.livenessTimer=void 0),n.session.status!=="ended"&&(n.session.status="ended",n.session.exitInfo="SIGTERM",this.notifyStateChange(n.session));try{await n.cleanup()}catch(r){this.log(`[manager] closeSession ${t} cleanup error: ${r.message}`)}}}async restartInner(t){let n=this.sessions.get(t);if(!n)throw new Error(`No session ${t} to restart`);if(n.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let r=n.session.yamlPath,s=xe(Z.dirname(r));if(!s)throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);n.restartInProgress=!0;try{n.livenessTimer&&(clearInterval(n.livenessTimer),n.livenessTimer=void 0);try{await n.cleanup()}catch(c){this.log(`[manager] restartInner ${t} old cleanup error: ${c.message}`)}n.session.status="starting",n.session.innerPort=0,n.session.pid=0,this.notifyStateChange(n.session);let o=ur+18e4,i,a=new Promise((c,u)=>{i=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:r,configPath:s,tempSuffix:t,headed:this.headed}),a])}catch(c){throw n.session.status="ended",n.session.exitInfo=c.message,this.notifyStateChange(n.session),c}finally{i&&clearTimeout(i)}n.session.innerPort=l.port,n.session.innerHost=l.host,n.session.pid=l.pid,n.session.status="running",n.cleanup=l.cleanup,n.livenessTimer=this.startLivenessProbe(t,c=>{n.session.status!=="ended"&&(n.session.status="ended",n.session.exitInfo=c,this.notifyStateChange(n.session))}),this.notifyStateChange(n.session),this.log(`[manager] session ${t} restarted on ${n.session.innerHost}:${n.session.innerPort} (yaml=${Z.basename(r)})`)}finally{n.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let n=this.sessions.get(t);return n?{...n.session}:void 0}httpProxyFor(t,n={}){let{liveviewUrlBuilder:r}=n;return async(s,o,i)=>{let a=this.sessions.get(t);if(!a){o.status(404).json({status:"error",message:"Session not found"});return}if(a.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:a.session.exitInfo});return}if(a.session.status==="idle"){o.status(503).json({status:"error",message:"Sandbox not started"});return}let l=`${s.method} ${s.path}`;if(l==="POST /api/int-runner/create-session"){let u=await dr(s),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=a.session.yamlPath,await hr(s,o,a.session.innerHost,a.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=r?.(s)??"";o.json({liveviewUrl:u,browserWsUrl:""});return}let c=await dr(s);await hr(s,o,a.session.innerHost,a.session.innerPort,c,s.headers["content-type"])}}wsUpgradeFor(t){return async(n,r,s,o)=>{let i=this.sessions.get(t);if(!i){$t(r,`HTTP/1.1 404 Not Found\r
647
647
  \r
648
- `);return}if(i.session.status==="ended"){Et(r,`HTTP/1.1 410 Gone\r
648
+ `);return}if(i.session.status==="ended"){$t(r,`HTTP/1.1 410 Gone\r
649
649
  \r
650
- `);return}try{let a=i.session.innerHost.includes(":")?`[${i.session.innerHost}]`:i.session.innerHost,l=await fetch(`http://${a}:${i.session.innerPort}/api/browser-cdp`);if(!l.ok)throw new Error(`Inner /api/browser-cdp returned ${l.status}`);let{cdpUrl:c}=await l.json(),d=new URL(c.replace(/^ws/,"http")),u=parseInt(d.port||"80",10),h=d.hostname,g=d.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let f=ur.createConnection(u,h);f.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
651
- Host: ${h}:${u}\r
652
- `;for(let[w,m]of Object.entries(n.headers)){let y=w.toLowerCase();y!=="host"&&y!=="origin"&&(p+=`${w}: ${Array.isArray(m)?m.join(", "):m}\r
650
+ `);return}try{let a=i.session.innerHost.includes(":")?`[${i.session.innerHost}]`:i.session.innerHost,l=await fetch(`http://${a}:${i.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=gr.createConnection(d,f);h.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
651
+ Host: ${f}:${d}\r
652
+ `;for(let[v,w]of Object.entries(n.headers)){let m=v.toLowerCase();m!=="host"&&m!=="origin"&&(p+=`${v}: ${Array.isArray(w)?w.join(", "):w}\r
653
653
  `)}p+=`\r
654
- `,f.write(p),s.length&&f.write(s),f.pipe(r),r.pipe(f)}),f.on("error",()=>{(!("destroyed"in r)||!r.destroyed)&&r.destroy()}),r.on("error",()=>{f.destroyed||f.destroy()}),r.on("close",()=>{f.destroyed||f.destroy()})}catch(a){this.log(`[manager] WS upgrade for ${t} failed: ${a.message}`),Et(r,`HTTP/1.1 502 Bad Gateway\r
654
+ `,h.write(p),s.length&&h.write(s),h.pipe(r),r.pipe(h)}),h.on("error",()=>{(!("destroyed"in r)||!r.destroyed)&&r.destroy()}),r.on("error",()=>{h.destroyed||h.destroy()}),r.on("close",()=>{h.destroyed||h.destroy()})}catch(a){this.log(`[manager] WS upgrade for ${t} failed: ${a.message}`),$t(r,`HTTP/1.1 502 Bad Gateway\r
655
655
  \r
656
- `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(n=>this.closeSession(n)))}}});import{Router as Vo}from"express";import gr from"express";import*as we from"fs";import*as Z from"path";function zo(e){return e?Z.isAbsolute(e)?Z.normalize(e):Z.resolve(e):null}function Yo(e,t){let n=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${n}/ws/debugger/${e}/cdp-browser`}function Jo(e){let t=Z.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function mr(e){let{manager:t,staticDir:n,resolveYamlPath:r=zo,liveviewUrlBuilder:s=Yo}=e,o=Vo();o.post("/api/debugger/sessions",gr.json(),async(a,l)=>{let c=a.body?.yamlPath;if(typeof c!="string"||!c){l.status(400).json({error:"yamlPath is required"});return}let d=r(c);if(!d){l.status(403).json({error:"Path outside allowed roots"});return}if(!Jo(d)){l.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!we.existsSync(d)){l.status(404).json({error:"File not found"});return}let u=t.listSessions().find(h=>h.yamlPath===d&&h.status!=="ended");try{let h=await t.openSession(d);l.status(u?200:201).json({sessionId:h.sessionId,yamlPath:h.yamlPath,startedAt:h.startedAt,status:h.status})}catch(h){l.status(500).json({error:h.message})}}),o.get("/api/debugger/sessions",(a,l)=>{l.setHeader("Cache-Control","no-store"),l.json({sessions:t.listSessions().map(c=>({sessionId:c.sessionId,yamlPath:c.yamlPath,startedAt:c.startedAt,status:c.status}))})}),o.delete("/api/debugger/sessions/:sessionId",async(a,l)=>{let c=a.params.sessionId,d=!!t.getSession(c);await t.closeSession(c),l.json({deleted:!0,alreadyGone:!d})}),we.existsSync(n)?o.use("/debugger/static",gr.static(n)):console.error(`[debugger] WARNING: debugger static dir missing at ${n} \u2014 iframe routes will 404`),o.get("/debugger/:sessionId/",(a,l)=>{let c=a.params.sessionId;if(!t.getSession(c)){l.status(404).send("Debugger session not found");return}let u=Z.join(n,"index.html");if(!we.existsSync(u)){l.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let g=we.readFileSync(u,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');l.type("html").send(g)});let i=(a,l,c)=>{let d=a.params.sessionId,u=a.originalUrl,h=`/debugger/${d}`;if(u.startsWith(h)){let f=u.slice(h.length)||"/";a.url=f,Object.defineProperty(a,"originalUrl",{value:f,configurable:!0})}t.httpProxyFor(d,{liveviewUrlBuilder:()=>s(d,a)})(a,l,c)};return o.use("/debugger/:sessionId",i),o}function yr(e,t,n,r){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){n.write(`HTTP/1.1 404 Not Found\r
656
+ `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(n=>this.closeSession(n)))}}});import{Router as Jo}from"express";import Mt from"express";import*as we from"fs";import*as Q from"path";function qo(e){return e?Q.isAbsolute(e)?Q.normalize(e):Q.resolve(e):null}function Xo(e,t){let n=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${n}/ws/debugger/${e}/cdp-browser`}function Zo(e){let t=Q.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function wr(e){let{manager:t,staticDir:n,resolveYamlPath:r=qo,liveviewUrlBuilder:s=Xo,fallbackRouter:o,artifactsDir:i}=e,a=Jo(),l=i?Mt.static(i):null;a.post("/api/debugger/sessions",Mt.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=r(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(!we.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})}}),a.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}))})}),a.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})}),we.existsSync(n)?a.use("/debugger/static",Mt.static(n)):console.error(`[debugger] WARNING: debugger static dir missing at ${n} \u2014 iframe routes will 404`),a.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=Q.join(n,"index.html");if(!we.existsSync(h)){d.status(500).send(`Debugger SPA bundle missing at ${h}`);return}let v=we.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(v)});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,v=`/debugger/${g}`;if(p.startsWith(v)){let m=p.slice(v.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(y){d.status(500).json({status:"error",message:y.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:()=>s(g,u)})(u,d,f)};return a.use("/debugger/:sessionId",c),a}function br(e,t,n,r){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){n.write(`HTTP/1.1 404 Not Found\r
657
657
  \r
658
- `),n.destroy();return}let i=o[1],a=o[2]||"";a.startsWith("/cdp-browser/page/")?a=a.slice(12):(a==="/cdp-browser"||a==="/cdp-browser/")&&(a=""),e.wsUpgradeFor(i)(t,n,r,a||void 0)}var wr=_(()=>{"use strict"});var vr={};ee(vr,{startDebuggerServer:()=>Zo});import Ye from"express";import*as F from"path";import{createRequire as qo}from"node:module";async function Zo(e){let{initialDir:t,initialFile:n,projectRoot:r,port:s,headed:o=!1}=e,i=new ze({headed:o}),a=Ye();a.use((p,w,m)=>{if(w.setHeader("Access-Control-Allow-Origin","*"),w.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),w.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),p.method==="OPTIONS")return w.sendStatus(204);m()}),a.use(Ye.json({limit:"10mb"})),a.use(ir({initialDir:t,initialFile:n,projectRoot:r}));let l=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,c=l.includes(F.sep+"src"+F.sep),d=c?F.resolve(l,"../../dist/static"):F.join(l,"static"),u=c?F.resolve(l,"../../dist/static-embedded"):F.join(l,"static-embedded");a.use("/devtools",Ye.static(br)),a.use(Ye.static(d)),a.use(mr({manager:i,staticDir:u})),a.get("*path",(p,w,m)=>{if(p.path.startsWith("/api/"))return m();w.sendFile(F.join(d,"index.html"),y=>{y&&w.send(Qo(t,s))})});let h=await new Promise((p,w)=>{let m=a.listen(s,"localhost",()=>{p(m)});m.on("error",w)});h.on("upgrade",(p,w,m)=>{if((p.url??"").startsWith("/ws/debugger/")){yr(i,p,w,m);return}w.write(`HTTP/1.1 404 Not Found\r
658
+ `),n.destroy();return}let i=o[1],a=o[2]||"";a.startsWith("/cdp-browser/page/")?a=a.slice(12):(a==="/cdp-browser"||a==="/cdp-browser/")&&(a=""),e.wsUpgradeFor(i)(t,n,r,a||void 0)}var vr=_(()=>{"use strict"});var _r={};te(_r,{startDebuggerServer:()=>ti});import $e from"express";import*as N from"path";import{createRequire as Qo}from"node:module";async function ti(e){let{initialDir:t,initialFile:n,projectRoot:r,port:s,headed:o=!1}=e,i=new qe({headed:o}),a=$e();a.use((w,m,y)=>{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"),w.method==="OPTIONS")return m.sendStatus(204);y()}),a.use($e.json({limit:"10mb"}));let l=lr({initialDir:t,initialFile:n,projectRoot:r});a.use(l);let c=N.join(r??t,".shiplight","artifacts");a.use("/api/report-assets",$e.static(c));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,d=u.includes(N.sep+"src"+N.sep),f=d?N.resolve(u,"../../dist/static"):N.join(u,"static"),g=d?N.resolve(u,"../../dist/static-embedded"):N.join(u,"static-embedded");a.use("/devtools",$e.static(Sr)),a.use($e.static(f)),a.use(wr({manager:i,staticDir:g,fallbackRouter:l,artifactsDir:c})),a.get("*path",(w,m,y)=>{if(w.path.startsWith("/api/"))return y();m.sendFile(N.join(f,"index.html"),k=>{k&&m.send(ni(t,s))})});let h=await new Promise((w,m)=>{let y=a.listen(s,"localhost",()=>{w(y)});y.on("error",m)});h.on("upgrade",(w,m,y)=>{if((w.url??"").startsWith("/ws/debugger/")){br(i,w,m,y);return}m.write(`HTTP/1.1 404 Not Found\r
659
659
  \r
660
- `),w.destroy()});let g=`http://localhost:${s}`,f=n?`${g}/?open=${encodeURIComponent(n)}`:g;return console.error(`[debugger] Server running at ${g}`),console.error(`[debugger] Directory: ${t}`),n&&console.error(`[debugger] File: ${n}`),{url:f,close:async()=>{await i.shutdown(),await new Promise((p,w)=>{h.close(m=>m?w(m):p())})}}}function Qo(e,t){return`<!DOCTYPE html>
660
+ `),m.destroy()});let p=`http://localhost:${s}`,v=n?`${p}/?open=${encodeURIComponent(n)}`:p;return console.error(`[debugger] Server running at ${p}`),console.error(`[debugger] Directory: ${t}`),n&&console.error(`[debugger] File: ${n}`),{url:v,close:async()=>{await i.shutdown(),await new Promise((w,m)=>{h.close(y=>y?m(y):w())})}}}function ni(e,t){return`<!DOCTYPE html>
661
661
  <html>
662
662
  <head>
663
663
  <meta charset="utf-8">
@@ -675,27 +675,27 @@ Host: ${h}:${u}\r
675
675
  <p>Directory: <code>${e}</code></p>
676
676
  <p>Server: localhost:${t}</p>
677
677
  </body>
678
- </html>`}var Xo,br,Sr=_(()=>{"use strict";ar();fr();wr();Xo=qo(import.meta.url);try{br=F.join(F.dirname(Xo.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 xr={};ee(xr,{startDebugger:()=>ti});import*as Y from"fs";import*as pe from"path";function Je(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function ti(e){let t,n=be,r=!1,s,o=!1,i=!0,a=!1;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(n=parseInt(e[y+1],10),r=!0,y++):e[y]==="--url"&&e[y+1]?(s=e[y+1],y++):e[y]==="--new"?o=!0:e[y]==="--open"?i=!1:e[y]==="--no-open"?i=!0:e[y]==="--headed"?a=!0:e[y]==="--help"||e[y]==="-h"?(ni(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let l,c;if(t){let y=pe.resolve(t);Y.existsSync(y)&&Y.statSync(y).isDirectory()?l=y:(l=pe.dirname(y),c=y)}else l=process.cwd();if(o&&c&&!Y.existsSync(c)){let y=pe.dirname(c);Y.existsSync(y)||Y.mkdirSync(y,{recursive:!0}),Y.writeFileSync(c,ei(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${c}`)}c&&!Y.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:d}=await Promise.resolve().then(()=>(pt(),pn)),u=d(l),h=u?pe.dirname(u):l;(await import("dotenv")).config({path:pe.join(h,".env"),override:!0}),u||(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:f}=await Promise.resolve().then(()=>(Sr(),vr));if(console.log(c?`Starting debugger for: ${c}`:`Starting debugger in: ${l}`),u&&console.log(`Using Playwright config: ${u}`),!r){let y=await an(be,_r);y===null&&(console.error(Je(`Error: No available port found in range ${be}-${be+_r-1}.`)),console.error(Je("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),y!==be&&console.log(`Port ${be} is in use; using port ${y} instead.`),n=y}let p;try{p=await f({initialDir:l,initialFile:c,projectRoot:h,port:n,headed:a})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(Je(`Error: Port ${n} is already in use.`)),console.error(Je(r?"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)),y}if(tn(n,l),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!i)try{let{default:y}=await import("open");await y(p.url)}catch{}let w=!1,m=async()=>{w&&(console.log(`
679
- Force exiting...`),process.exit(1)),w=!0,console.log(`
680
- Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{nn(n,l),await p.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",m),process.on("SIGTERM",m)}function ni(){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 be,_r,ei,Tr=_(()=>{"use strict";rn();cn();be=6174,_r=10;ei=e=>`goal: New test
678
+ </html>`}var ei,Sr,xr=_(()=>{"use strict";pr();yr();vr();ei=Qo(import.meta.url);try{Sr=N.join(N.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 kr={};te(kr,{startDebugger:()=>si});import*as Y from"fs";import*as pe from"path";function Xe(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function si(e){let t,n=be,r=!1,s,o=!1,i=!0,a=!1;for(let m=0;m<e.length;m++)e[m]==="--port"&&e[m+1]?(n=parseInt(e[m+1],10),r=!0,m++):e[m]==="--url"&&e[m+1]?(s=e[m+1],m++):e[m]==="--new"?o=!0:e[m]==="--open"?i=!1:e[m]==="--no-open"?i=!0:e[m]==="--headed"?a=!0:e[m]==="--help"||e[m]==="-h"?(oi(),process.exit(0)):e[m].startsWith("--")||(t=e[m]);let l,c;if(t){let m=pe.resolve(t);Y.existsSync(m)&&Y.statSync(m).isDirectory()?l=m:(l=pe.dirname(m),c=m)}else l=process.cwd();if(o&&c&&!Y.existsSync(c)){let m=pe.dirname(c);Y.existsSync(m)||Y.mkdirSync(m,{recursive:!0}),Y.writeFileSync(c,ri(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${c}`)}c&&!Y.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(()=>(ut(),un)),d=u(l),f=d?pe.dirname(d):l;(await import("dotenv")).config({path:pe.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(()=>(xr(),_r));if(console.log(c?`Starting debugger for: ${c}`:`Starting debugger in: ${l}`),d&&console.log(`Using Playwright config: ${d}`),!r){let m=await cn(be,Tr);m===null&&(console.error(Xe(`Error: No available port found in range ${be}-${be+Tr-1}.`)),console.error(Xe("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),m!==be&&console.log(`Port ${be} is in use; using port ${m} instead.`),n=m}let p;try{p=await h({initialDir:l,initialFile:c,projectRoot:f,port:n,headed:a})}catch(m){throw m?.code==="EADDRINUSE"&&(console.error(Xe(`Error: Port ${n} is already in use.`)),console.error(Xe(r?"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(nn(n,l),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!i)try{let{default:m}=await import("open");await m(p.url)}catch{}let v=!1,w=async()=>{v&&(console.log(`
679
+ Force exiting...`),process.exit(1)),v=!0,console.log(`
680
+ Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{rn(n,l),await p.close()}catch{}clearTimeout(m),process.exit(0)};process.on("SIGINT",w),process.on("SIGTERM",w)}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 be,Tr,ri,Pr=_(()=>{"use strict";sn();ln();be=6174,Tr=10;ri=e=>`goal: New test
681
681
  statements:
682
682
  - URL: ${e}
683
683
  - "TODO: Add your first step"
684
- `});import mp from"dotenv";function si(){return globalThis[ri]}function qe(){let e=si();return e===void 0?process.env:e}var ri,$t=_(()=>{"use strict";ri="__shiplightDotenvCache__"});function Xe(e,t){let n=t?.trim();return n?n.endsWith("/")?n.slice(0,-1):n:e.startsWith(oi)?ii:ai}var oi,ii,ai,Mt=_(()=>{"use strict";oi="shp_",ii="https://nova-api.shiplight.ai",ai="https://api.shiplight.ai"});var It={};ee(It,{lookupActionStores:()=>pi,updateActionStores:()=>di});function kr(){let e=qe(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Xe(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function pi(e){let t=kr();if(!t||e.length===0)return new Map;try{let n=new AbortController,r=setTimeout(()=>n.abort(),ci),s=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:n.signal});if(clearTimeout(r),!s.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${s.status}`),new Map;let o=await s.json(),i=new Map;for(let[a,l]of Object.entries(o.stores??{}))i.set(a,l);return i}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",n.message),new Map}}async function di(e){let t=kr();if(!t||e.size===0)return 0;try{let n=new AbortController,r=setTimeout(()=>n.abort(),li),s={};for(let[a,l]of e)s[a]=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:s}),signal:n.signal});return clearTimeout(r),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",n.message),0}}var ci,li,Ot=_(()=>{"use strict";Mt();$t();ci=2e3,li=5e3});import*as U from"fs";import*as Ae from"path";import{globSync as ui}from"glob";function Pr(e){let t=qe().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Rt:new Lt(e)}function Ee(e){return e.replace(/\//g,"__")+".json"}function fi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var hi,Lt,Rt,Ar=_(()=>{"use strict";ae();$t();hi=".shiplight/action-cache";Lt=class{constructor(t){this.cwd=t;this.cacheDir=Ae.join(t,hi)}isCloud=!1;cacheDir;async lookup(t){let n=new Map;if(t.length===0||!U.existsSync(this.cacheDir))return n;for(let r of t){let s=Ae.join(this.cacheDir,Ee(r));try{if(U.existsSync(s)){let o=U.readFileSync(s,"utf-8");n.set(r,JSON.parse(o))}}catch{}}return n}async update(t){if(t.size===0)return 0;U.mkdirSync(this.cacheDir,{recursive:!0});let n=0;for(let[r,s]of t)try{let o=Ae.join(this.cacheDir,Ee(r)),i=yt();if(U.existsSync(o))try{i=JSON.parse(U.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};U.writeFileSync(o,JSON.stringify(a,null,2)),n++}catch{}return n}loadAll(){if(!U.existsSync(this.cacheDir))return;let t=ui("*.json",{cwd:this.cacheDir});if(t.length===0)return;let n=new Map,r=0;for(let s of t)try{let o=U.readFileSync(Ae.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=fi(s);n.set(a,i),r+=Object.keys(i.entries??{}).length}catch{}if(n.size!==0)return console.log(`[shiplight] Cache: loaded ${r} cached action entit${r===1?"y":"ies"} for ${n.size} test file${n.size!==1?"s":""}`),n}},Rt=class{isCloud=!0;async lookup(t){let{lookupActionStores:n}=await Promise.resolve().then(()=>(Ot(),It));return n(t)}async update(t){let{updateActionStores:n}=await Promise.resolve().then(()=>(Ot(),It));return n(t)}loadAll(){}}});var Or={};ee(Or,{buildPlaywrightSpawnOptions:()=>$r,buildTestPathsFromGitInfo:()=>Ir,cloudKeyToCwdRelPath:()=>Dt,runTests:()=>mi});import{spawn as gi,execFileSync as Er}from"child_process";import*as j from"fs";import*as N from"path";import{globSync as Ct}from"glob";async function mi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(u=>j.existsSync(N.join(t,u)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
685
- `));let s=e.includes("--magic"),i=e.filter(u=>u!=="--magic").map(u=>u.endsWith(".test.yaml")?u.replace(/\.test\.yaml$/,".yaml.spec.ts"):u),a=Pr(t);await wi(t,a),Se&&process.stdout.write(`shiplightai v${Se}
686
- `);let l={...process.env};s&&(l.SHIPLIGHT_MAGIC="1");let c=gi("npx",["playwright","test",...i],$r(t,l)),d=await new Promise(u=>{c.on("close",h=>u(h??1))});await bi(t,a),process.exit(d)}function $r(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Nt(){try{return Er("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function yi(){try{let e=Er("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Mr(e,t,n){if(!n)return{testPaths:[...e],branchPrefix:""};let r=yi(),s=Nt();return Ir(e,t,r,s)}function Ir(e,t,n,r){return{testPaths:e.map(o=>{let i=r?N.relative(r,N.resolve(t,o)):o;return`${n}${i}`}),branchPrefix:n}}function Dt(e,t,n,r){let s=t&&e.startsWith(t)?e.slice(t.length):e;return n?N.relative(r,N.resolve(n,s)):s}async function wi(e,t){try{let n=Ct("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(n.length===0)return;let{testPaths:r,branchPrefix:s}=Mr(n,e,t.isCloud),o=await t.lookup(r);if(o.size===0)return;let i=N.join(e,".shiplight","action-cache");j.mkdirSync(i,{recursive:!0});let a=Nt(),l=0;for(let[c,d]of o){let u=t.isCloud?Dt(c,s,a,e):c,h=N.join(i,Ee(u));j.writeFileSync(h,JSON.stringify(d,null,2)),l++}console.log(`[shiplight] Cache: downloaded ${l} action store${l!==1?"s":""}`)}catch(n){console.warn("[shiplight] Cache download failed:",n.message)}}async function bi(e,t){try{let n=N.join(e,"test-results");if(!j.existsSync(n))return;let r=Ct("**/new-action-entities.json",{cwd:n});if(r.length===0)return;let s={};for(let u of r)try{let h=j.readFileSync(N.join(n,u),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(s,g.entries)}catch{}if(Object.keys(s).length===0)return;let o=Ct("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=Mr(o,e,t.isCloud),l=new Map;for(let u=0;u<o.length;u++){let h=o[u],g=i[u],f=N.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!j.existsSync(f))continue;let p=j.readFileSync(f,"utf-8"),w={};for(let[m,y]of Object.entries(s))p.includes(m)&&(w[m]=y);if(Object.keys(w).length>0){let m=t.isCloud?Dt(g,a,Nt(),e):g,y=N.join(e,".shiplight","action-cache",Ee(m)),v={};if(j.existsSync(y))try{v=JSON.parse(j.readFileSync(y,"utf-8")).entries}catch{}l.set(g,{version:"1.0",entries:{...v,...w}})}}if(l.size===0)return;let c=await t.update(l),d=Array.from(l.values()).reduce((u,h)=>u+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${d} action entit${d!==1?"ies":"y"} for ${c} test${c!==1?"s":""}`)}catch(n){console.warn("[shiplight] Cache upload failed:",n.message)}}var Lr=_(()=>{"use strict";Ar();$e()});function $(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Qe(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),n=(e%6e4/1e3).toFixed(0);return`${t}m ${n}s`}function Rr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Ze(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 vi(e){return`<span class="badge badge-${e}">${e}</span>`}function Si(e){if(!e.code)return"";let t=e.code.split(`
687
- `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((s,o)=>{let i=e.codeStartLine+o,a=i===e.codeLine,l=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${l} \u2502 ${$(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(r=>`<span class="code-line code-line-body">${$(r)}</span>`).join("")}</pre></div>`}function _i(e){let t=e.duration!=null?`<span class="step-duration">${Qe(e.duration)}</span>`:"",n=Ze(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${$(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":Si(e),i="";e.error&&(i=`<div class="step-error"><pre>${$(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${$(e.message)}</div>`);let l=e.status==="failure"?" open":"";return`
684
+ `});import bp from"dotenv";function ai(){return globalThis[ii]}function Ze(){let e=ai();return e===void 0?process.env:e}var ii,It=_(()=>{"use strict";ii="__shiplightDotenvCache__"});function Qe(e,t){let n=t?.trim();return n?n.endsWith("/")?n.slice(0,-1):n:e.startsWith(ci)?li:pi}var ci,li,pi,Ot=_(()=>{"use strict";ci="shp_",li="https://nova-api.shiplight.ai",pi="https://api.shiplight.ai"});var Lt={};te(Lt,{lookupActionStores:()=>hi,updateActionStores:()=>fi});function Ar(){let e=Ze(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Qe(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function hi(e){let t=Ar();if(!t||e.length===0)return new Map;try{let n=new AbortController,r=setTimeout(()=>n.abort(),ui),s=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:n.signal});if(clearTimeout(r),!s.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${s.status}`),new Map;let o=await s.json(),i=new Map;for(let[a,l]of Object.entries(o.stores??{}))i.set(a,l);return i}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",n.message),new Map}}async function fi(e){let t=Ar();if(!t||e.size===0)return 0;try{let n=new AbortController,r=setTimeout(()=>n.abort(),di),s={};for(let[a,l]of e)s[a]=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:s}),signal:n.signal});return clearTimeout(r),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",n.message),0}}var ui,di,Rt=_(()=>{"use strict";Ot();It();ui=2e3,di=5e3});import*as F from"fs";import*as Me from"path";import{globSync as gi}from"glob";function Er(e){let t=Ze().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Nt:new Ct(e)}function Ie(e){return e.replace(/\//g,"__")+".json"}function yi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var mi,Ct,Nt,$r=_(()=>{"use strict";ce();It();mi=".shiplight/action-cache";Ct=class{constructor(t){this.cwd=t;this.cacheDir=Me.join(t,mi)}isCloud=!1;cacheDir;async lookup(t){let n=new Map;if(t.length===0||!F.existsSync(this.cacheDir))return n;for(let r of t){let s=Me.join(this.cacheDir,Ie(r));try{if(F.existsSync(s)){let o=F.readFileSync(s,"utf-8");n.set(r,JSON.parse(o))}}catch{}}return n}async update(t){if(t.size===0)return 0;F.mkdirSync(this.cacheDir,{recursive:!0});let n=0;for(let[r,s]of t)try{let o=Me.join(this.cacheDir,Ie(r)),i=wt();if(F.existsSync(o))try{i=JSON.parse(F.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};F.writeFileSync(o,JSON.stringify(a,null,2)),n++}catch{}return n}loadAll(){if(!F.existsSync(this.cacheDir))return;let t=gi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let n=new Map,r=0;for(let s of t)try{let o=F.readFileSync(Me.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=yi(s);n.set(a,i),r+=Object.keys(i.entries??{}).length}catch{}if(n.size!==0)return console.log(`[shiplight] Cache: loaded ${r} cached action entit${r===1?"y":"ies"} for ${n.size} test file${n.size!==1?"s":""}`),n}},Nt=class{isCloud=!0;async lookup(t){let{lookupActionStores:n}=await Promise.resolve().then(()=>(Rt(),Lt));return n(t)}async update(t){let{updateActionStores:n}=await Promise.resolve().then(()=>(Rt(),Lt));return n(t)}loadAll(){}}});var Rr={};te(Rr,{buildPlaywrightSpawnOptions:()=>Ir,buildTestPathsFromGitInfo:()=>Lr,cloudKeyToCwdRelPath:()=>Ft,runTests:()=>bi});import{spawn as wi,execFileSync as Mr}from"child_process";import*as U from"fs";import*as D from"path";import{globSync as Dt}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=>U.existsSync(D.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.
685
+ `));let s=e.includes("--magic"),i=e.filter(d=>d!=="--magic").map(d=>d.endsWith(".test.yaml")?d.replace(/\.test\.yaml$/,".yaml.spec.ts"):d),a=Er(t);await Si(t,a),_e&&process.stdout.write(`shiplightai v${_e}
686
+ `);let l={...process.env};s&&(l.SHIPLIGHT_MAGIC="1");let c=wi("npx",["playwright","test",...i],Ir(t,l)),u=await new Promise(d=>{c.on("close",f=>d(f??1))});await _i(t,a),process.exit(u)}function Ir(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function jt(){try{return Mr("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function vi(){try{let e=Mr("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Or(e,t,n){if(!n)return{testPaths:[...e],branchPrefix:""};let r=vi(),s=jt();return Lr(e,t,r,s)}function Lr(e,t,n,r){return{testPaths:e.map(o=>{let i=r?D.relative(r,D.resolve(t,o)):o;return`${n}${i}`}),branchPrefix:n}}function Ft(e,t,n,r){let s=t&&e.startsWith(t)?e.slice(t.length):e;return n?D.relative(r,D.resolve(n,s)):s}async function Si(e,t){try{let n=Dt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(n.length===0)return;let{testPaths:r,branchPrefix:s}=Or(n,e,t.isCloud),o=await t.lookup(r);if(o.size===0)return;let i=D.join(e,".shiplight","action-cache");U.mkdirSync(i,{recursive:!0});let a=jt(),l=0;for(let[c,u]of o){let d=t.isCloud?Ft(c,s,a,e):c,f=D.join(i,Ie(d));U.writeFileSync(f,JSON.stringify(u,null,2)),l++}console.log(`[shiplight] Cache: downloaded ${l} action store${l!==1?"s":""}`)}catch(n){console.warn("[shiplight] Cache download failed:",n.message)}}async function _i(e,t){try{let n=D.join(e,"test-results");if(!U.existsSync(n))return;let r=Dt("**/new-action-entities.json",{cwd:n});if(r.length===0)return;let s={};for(let d of r)try{let f=U.readFileSync(D.join(n,d),"utf-8"),g=JSON.parse(f);g?.entries&&Object.assign(s,g.entries)}catch{}if(Object.keys(s).length===0)return;let o=Dt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=Or(o,e,t.isCloud),l=new Map;for(let d=0;d<o.length;d++){let f=o[d],g=i[d],h=D.join(e,f.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!U.existsSync(h))continue;let p=U.readFileSync(h,"utf-8"),v={};for(let[w,m]of Object.entries(s))p.includes(w)&&(v[w]=m);if(Object.keys(v).length>0){let w=t.isCloud?Ft(g,a,jt(),e):g,m=D.join(e,".shiplight","action-cache",Ie(w)),y={};if(U.existsSync(m))try{y=JSON.parse(U.readFileSync(m,"utf-8")).entries}catch{}l.set(g,{version:"1.0",entries:{...y,...v}})}}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(n){console.warn("[shiplight] Cache upload failed:",n.message)}}var Cr=_(()=>{"use strict";$r();Le()});function M(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function tt(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),n=(e%6e4/1e3).toFixed(0);return`${t}m ${n}s`}function Nr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function et(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 xi(e){return`<span class="badge badge-${e}">${e}</span>`}function Ti(e){if(!e.code)return"";let t=e.code.split(`
687
+ `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((s,o)=>{let i=e.codeStartLine+o,a=i===e.codeLine,l=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${l} \u2502 ${M(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(r=>`<span class="code-line code-line-body">${M(r)}</span>`).join("")}</pre></div>`}function ki(e){let t=e.duration!=null?`<span class="step-duration">${tt(e.duration)}</span>`:"",n=et(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${M(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":Ti(e),i="";e.error&&(i=`<div class="step-error"><pre>${M(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${M(e.message)}</div>`);let l=e.status==="failure"?" open":"";return`
688
688
  <details class="step-details step step-${e.status}"${l}>
689
689
  <summary class="step-header">
690
690
  ${n}
691
- <span class="step-id">${$(e.stepId)}</span>
692
- <span class="step-description-collapsed">${$(e.description)}</span>
691
+ <span class="step-id">${M(e.stepId)}</span>
692
+ <span class="step-description-collapsed">${M(e.description)}</span>
693
693
  ${t}
694
694
  </summary>
695
695
  <div class="step-expanded">
696
696
  ${s}
697
697
  ${o}
698
- <div class="step-description-full">${$(e.description)}</div>
698
+ <div class="step-description-full">${M(e.description)}</div>
699
699
  ${a}
700
700
  ${i}
701
701
  </div>
@@ -703,32 +703,32 @@ statements:
703
703
  <div class="step step-${e.status}">
704
704
  <div class="step-header">
705
705
  ${n}
706
- <span class="step-id">${$(e.stepId)}</span>
707
- <span class="step-description">${$(e.description)}</span>
706
+ <span class="step-id">${M(e.stepId)}</span>
707
+ <span class="step-description">${M(e.description)}</span>
708
708
  ${t}
709
709
  </div>
710
- </div>`}function xi(e,t,n,r){let s=[];e&&s.push(`
710
+ </div>`}function Pi(e,t,n,r){let s=[];e&&s.push(`
711
711
  <details class="artifact-section">
712
712
  <summary class="artifact-summary">Video</summary>
713
713
  <div class="artifact-content">
714
714
  <div class="video-container">
715
715
  <video controls preload="metadata" class="artifact-video">
716
- <source src="${$(e)}" type="video/webm" />
716
+ <source src="${M(e)}" type="video/webm" />
717
717
  </video>
718
718
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
719
719
  </div>
720
720
  </div>
721
- </details>`);let o=n.filter(i=>i.screenshot).map(i=>({src:i.screenshot,stepId:i.stepId,description:i.description,status:i.status,message:i.message||i.error||""}));if(o.length>0){let i=$(JSON.stringify(o)),a=o.map((l,c)=>`
721
+ </details>`);let o=n.filter(i=>i.screenshot).map(i=>({src:i.screenshot,stepId:i.stepId,description:i.description,status:i.status,message:i.message||i.error||""}));if(o.length>0){let i=M(JSON.stringify(o)),a=o.map((l,c)=>`
722
722
  <div class="screenshot-thumb" onclick="openGalleryAt(this, ${c})" data-gallery="${i}">
723
- <img src="${$(l.src)}" alt="${$(l.stepId)}" />
724
- <span class="thumb-label">${$(l.stepId)}</span>
723
+ <img src="${M(l.src)}" alt="${M(l.stepId)}" />
724
+ <span class="thumb-label">${M(l.stepId)}</span>
725
725
  </div>`).join("");s.push(`
726
726
  <details class="artifact-section">
727
727
  <summary class="artifact-summary">Screenshots (${o.length})</summary>
728
728
  <div class="artifact-content">
729
729
  <div class="screenshot-grid">${a}</div>
730
730
  </div>
731
- </details>`)}if(t){let i=$(t);s.push(`
731
+ </details>`)}if(t){let i=M(t);s.push(`
732
732
  <details class="artifact-section">
733
733
  <summary class="artifact-summary">Trace</summary>
734
734
  <div class="artifact-content">
@@ -739,40 +739,40 @@ statements:
739
739
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
740
740
  <p class="trace-hint"><a href="${i}" class="attachment-link" download>Download trace.zip</a></p>
741
741
  </div>
742
- </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function Cr(e,t,n,r,s){let o=e.map(_i).join(`
743
- `),i="";t&&!e.some(l=>l.error)&&(i=`<div class="test-error"><pre>${$(t)}</pre></div>`);let a=xi(n,r,e,s);return`
742
+ </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function Dr(e,t,n,r,s){let o=e.map(ki).join(`
743
+ `),i="";t&&!e.some(l=>l.error)&&(i=`<div class="test-error"><pre>${M(t)}</pre></div>`);let a=Pi(n,r,e,s);return`
744
744
  ${i}
745
745
  <div class="steps-list">
746
746
  ${o||'<div class="no-steps">No YAML step details available</div>'}
747
747
  </div>
748
- ${a}`}function Ti(e,t){let n=e.flaky?"flaky":e.status,r=Ze(n),s;if(e.flaky&&e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length,a=e.attempts.map((c,d)=>{let u=d===i-1,h=c.status==="passed"?"passed":"failed",g=`Attempt ${c.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${h}"
749
- onclick="switchAttemptTab('${o}', ${d})"
750
- data-tab-index="${d}">${Ze(h)} ${g} <span class="attempt-tab-badge badge-${h}">${c.status}</span></button>`}).join(""),l=e.attempts.map((c,d)=>{let u=d===i-1,h=Cr(c.steps,c.error,c.videoPath,c.tracePath,`${t}-attempt-${d}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${d}">
748
+ ${a}`}function Ai(e,t){let n=e.flaky?"flaky":e.status,r=et(n),s;if(e.flaky&&e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length,a=e.attempts.map((c,u)=>{let d=u===i-1,f=c.status==="passed"?"passed":"failed",g=`Attempt ${c.attemptNumber}`;return`<button class="attempt-tab ${d?"active":""} attempt-tab-${f}"
749
+ onclick="switchAttemptTab('${o}', ${u})"
750
+ data-tab-index="${u}">${et(f)} ${g} <span class="attempt-tab-badge badge-${f}">${c.status}</span></button>`}).join(""),l=e.attempts.map((c,u)=>{let d=u===i-1,f=Dr(c.steps,c.error,c.videoPath,c.tracePath,`${t}-attempt-${u}`);return`<div class="attempt-panel ${d?"active":""}" data-panel-index="${u}">
751
751
  <div class="attempt-meta">
752
- ${Ze(c.status==="passed"?"passed":"failed")}
753
- <span class="attempt-meta-text">Attempt ${c.attemptNumber} &mdash; ${c.status} in ${Qe(c.duration)}</span>
752
+ ${et(c.status==="passed"?"passed":"failed")}
753
+ <span class="attempt-meta-text">Attempt ${c.attemptNumber} &mdash; ${c.status} in ${tt(c.duration)}</span>
754
754
  </div>
755
- ${h}
755
+ ${f}
756
756
  </div>`}).join("");s=`
757
- <div class="flaky-note">Flaky &mdash; ${Rr(e.retries)}</div>
757
+ <div class="flaky-note">Flaky &mdash; ${Nr(e.retries)}</div>
758
758
  <div class="attempt-tabs" id="${o}">
759
759
  <div class="attempt-tab-bar">${a}</div>
760
760
  ${l}
761
761
  </div>`}else s=`
762
- ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${Rr(e.retries)}</div>`:""}
763
- ${Cr(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
762
+ ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${Nr(e.retries)}</div>`:""}
763
+ ${Dr(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
764
764
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
765
765
  <summary class="test-summary test-${n}">
766
766
  ${r}
767
- <span class="test-title">${$(e.title)}</span>
768
- <span class="test-file">${$(e.file)}</span>
769
- ${vi(n)}
770
- <span class="test-duration">${Qe(e.duration)}</span>
767
+ <span class="test-title">${M(e.title)}</span>
768
+ <span class="test-file">${M(e.file)}</span>
769
+ ${xi(n)}
770
+ <span class="test-duration">${tt(e.duration)}</span>
771
771
  </summary>
772
772
  <div class="test-body">
773
773
  ${s}
774
774
  </div>
775
- </details>`}function Ft(e){let t=e.tests.filter(a=>a.flaky).length,n=e.tests.filter(a=>a.status==="passed"&&!a.flaky).length,r=e.tests.filter(a=>a.status==="failed"||a.status==="timedOut").length,s=e.tests.filter(a=>a.status==="skipped").length,o=e.tests.length,i=e.tests.map((a,l)=>Ti(a,l)).join(`
775
+ </details>`}function Ut(e){let t=e.tests.filter(a=>a.flaky).length,n=e.tests.filter(a=>a.status==="passed"&&!a.flaky).length,r=e.tests.filter(a=>a.status==="failed"||a.status==="timedOut").length,s=e.tests.filter(a=>a.status==="skipped").length,o=e.tests.length,i=e.tests.map((a,l)=>Ai(a,l)).join(`
776
776
  `);return`<!DOCTYPE html>
777
777
  <html lang="en">
778
778
  <head>
@@ -1410,7 +1410,7 @@ statements:
1410
1410
  ${t>0?`<span class="summary-stat flaky">${t} flaky</span>`:""}
1411
1411
  ${r>0?`<span class="summary-stat failed">${r} failed</span>`:""}
1412
1412
  ${s>0?`<span class="summary-stat skipped">${s} skipped</span>`:""}
1413
- <span class="summary-stat">${Qe(e.totalDuration)}</span>
1413
+ <span class="summary-stat">${tt(e.totalDuration)}</span>
1414
1414
  ${e.timestamp?`<span class="summary-stat" style="margin-left:auto;color:var(--color-text-secondary)">${new Date(e.timestamp).toLocaleString()}</span>`:""}
1415
1415
  </div>
1416
1416
  </div>${e.cacheSummary?`
@@ -1427,7 +1427,7 @@ statements:
1427
1427
  ${i}
1428
1428
  </div>
1429
1429
  <div class="footer">
1430
- Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${$(e.shiplightVersion)}`:""}
1430
+ Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${M(e.shiplightVersion)}`:""}
1431
1431
  </div>
1432
1432
  </div>
1433
1433
 
@@ -1554,9 +1554,9 @@ statements:
1554
1554
  });
1555
1555
  </script>
1556
1556
  </body>
1557
- </html>`}var Nr=_(()=>{"use strict"});import*as J from"fs";import*as Q from"path";import{execFileSync as ki}from"child_process";import{createHash as Pi}from"crypto";import ve from"axios";function Dr(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let n=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(n)return`https://${n[1]}/${n[2]}`}function oe(...e){try{return ki("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Ai(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=J.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Ei(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",n=process.env.GITHUB_REPOSITORY??"",r=process.env.GITHUB_RUN_ID??"",s=process.env.GITHUB_EVENT_NAME??"",i=Ai().pull_request,a=i?.head?.sha,l=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",c=process.env.GITHUB_REF??"",d=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??c.match(/^refs\/pull\/(\d+)\//)?.[1],u=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,h=process.env.SHIPLIGHT_PR_TITLE??i?.title,g=oe("log","-1","--pretty=%s"),f=oe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:l,gitBranch:u,gitRepo:n,commitMessage:g,authorEmail:f,prNumber:d,prTitle:h,prUrl:d&&n?`${t}/${n}/pull/${d}`:void 0,ciBuildId:r,ciBuildUrl:r&&n?`${t}/${n}/actions/runs/${r}`:void 0,commitUrl:l&&n?`${t}/${n}/commit/${l}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:s,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",n=process.env.CI_COMMIT_SHA??"",r=process.env.CI_MERGE_REQUEST_IID,s=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:n,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:s,prNumber:r,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:r&&t?`${t}/-/merge_requests/${r}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:n&&t?`${t}/commit/${n}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",n=process.env.CIRCLE_PROJECT_USERNAME??"",r=process.env.CIRCLE_PROJECT_REPONAME??"",s=n&&r?`${n}/${r}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,i=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,l=a?Dr(a):void 0,c=oe("log","-1","--pretty=%s"),d=oe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:c,authorEmail:d,prNumber:i,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=oe("rev-parse","HEAD"),n=oe("rev-parse","--abbrev-ref","HEAD"),r=oe("log","-1","--pretty=%s"),s=oe("log","-1","--pretty=%ae"),o=oe("remote","get-url","origin"),i=o?Dr(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:n,commitMessage:r,authorEmail:s,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function $i(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 Fr(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Mi(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 Ii(e,t){let n=new Map;for(let r of t){r.screenshotS3Uris={};let s=n.get(r.testCaseName);s?s.push(r):n.set(r.testCaseName,[r])}return e.map(r=>n.get(r.title)?.shift())}function et(e,t){return Q.isAbsolute(t)?t:Q.join(e,t)}function Oi(e,t,n,r){let s={};for(let o of e.steps)s[o.stepId]={description:o.description,status:o.status,duration:o.duration,message:o.error??o.message,screenshotS3Uri:t[o.stepId]};return{schemaVersion:2,result:Fr(e.status),flaky:!1,segments:[{outcome:Fr(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:s,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:n,traceS3Uri:r,actionStepsMap:e.actionStepsMap??{}}]}}function Bt(e){return Pi("md5").update(e).digest("base64")}function Ut(e){return Bt(J.readFileSync(e))}async function jt(e,t){let n=J.readFileSync(t),r=Q.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[r]??"application/octet-stream";await ve.put(e,n,{headers:{"Content-Type":o,"Content-MD5":Bt(n)}})}async function Ur(e,t,n,r){let s=Xe(r,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${r}`,"Content-Type":"application/json"},i=Ei(),a=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 m=et(t,p.videoPath);J.existsSync(m)&&(w.videoMd5=Ut(m))}if(p.tracePath){let m=et(t,p.tracePath);J.existsSync(m)&&(w.traceMd5=Ut(m))}return w}),l=e.tests.length;console.log(`[reporter] Uploading ${l} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let d=(await ve.post(`${s}/v1/local-runs`,{trigger:i.ciProvider,startTime:n,metadata:i,tests:a},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${d.testRunId})`);let u=Ii(e.tests,d.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,w)=>{let m=u[w];if(!m)return;let y=p.steps.filter(A=>A.screenshot);if(!y.length)return;let v=y.map(A=>A.stepId),P={};for(let A of y){let M=Q.isAbsolute(A.screenshot)?A.screenshot:Q.join(t,A.screenshot);J.existsSync(M)&&(P[A.stepId]=Ut(M))}try{let A=await ve.post(`${s}/v1/local-runs/${d.testRunId}/results/${m.testCaseResultId}/screenshot-urls`,{stepIds:v,md5s:P},{headers:o});m.uploadUrls.screenshots=A.data.screenshots,m.screenshotS3Uris=A.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${v.length} screenshot URL(s) for "${p.title}"`)}catch(A){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,A)}})),console.log("[reporter] [3/4] Uploading assets...");let h=(await Promise.all(e.tests.map(async(p,w)=>{let m=u[w];if(!m){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let y=m.uploadUrls,v={},P=0;await Promise.all(p.steps.map(async L=>{if(L.screenshot&&y.screenshots?.[L.stepId]){let he=Q.isAbsolute(L.screenshot)?L.screenshot:Q.join(t,L.screenshot);if(J.existsSync(he))try{await jt(y.screenshots[L.stepId],he),v[L.stepId]=m.screenshotS3Uris[L.stepId],P++}catch(os){console.warn(`[reporter] Screenshot upload failed for step ${L.stepId}:`,os)}}})),P>0&&console.log(`[reporter] [3/4] Uploaded ${P} screenshot(s) for "${p.title}"`);let A;if(p.videoPath&&y.video){let L=et(t,p.videoPath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await jt(y.video,L),A=m.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(he){console.warn("[reporter] Video upload failed:",he)}}}let M;if(p.tracePath&&y.trace){let L=et(t,p.tracePath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await jt(y.trace,L),M=m.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(he){console.warn("[reporter] Trace upload failed:",he)}}}console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let st=Oi(p,v,A,M),Wt=Buffer.from(JSON.stringify(st)),Gt=Bt(Wt),Kt=await ve.post(`${s}/v1/local-runs/${d.testRunId}/results/${m.testCaseResultId}/report-url`,{md5:Gt},{headers:o}),rs=Kt.data.reportUrl,ss=Kt.data.reportS3Uri;return await ve.put(rs,Wt,{headers:{"Content-Type":"application/json","Content-MD5":Gt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:m.testCaseResultId,result:$i(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:ss,videoS3Uri:A,traceS3Uri:M,metadata:{suiteName:p.suiteName,file:p.file}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=Mi(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let f=await ve.put(`${s}/v1/local-runs/${d.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:o});console.log(`
1558
- Shiplight cloud report: ${Li(f.data.reportUrl,s)}`)}function Li(e,t){if(/^https?:\/\//.test(e))return e;let n=e.startsWith("/")?e:`/${e}`;return`${Ri(t)}${n}`}function Ri(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 jr=_(()=>{"use strict";Mt()});var Kr={};ee(Kr,{buildGitHubSummary:()=>Gr,runReport:()=>Ci});import*as E from"fs";import*as k from"path";async function Ci(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"),n=e.includes("--merge"),r=e.includes("--github-summary");!n&&r&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),n?await Di(e,t,r):await Ni(e,t)}async function Ni(e,t){let n=e.find(a=>!a.startsWith("--"))||"shiplight-report",r=k.isAbsolute(n)?n:k.join(process.cwd(),n),s=k.join(r,"report-data.json");E.existsSync(s)||(console.error(`Error: ${s} 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(E.readFileSync(s,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${s}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=k.join(r,"index.html");if(E.writeFileSync(i,Ft(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Wr(o,r),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Di(e,t,n){let r=k.join(process.cwd(),"shiplight-report"),s=e.findIndex(g=>g==="-o"||g==="--output");if(s!==-1&&e[s+1]){let g=e[s+1];r=k.isAbsolute(g)?g:k.join(process.cwd(),g)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let g=0;g<e.length;g++){let f=e[g];if(o.has(f)){g++;continue}if(i.has(f))continue;let p=k.isAbsolute(f)?f:k.join(process.cwd(),f);a.push(p)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let l=[],c=0,d=0;E.mkdirSync(k.join(r,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let f=a[g],p=`shard-${g}`,w=k.join(f,"report-data.json");if(!E.existsSync(w)){console.warn(`Warning: No report-data.json found in ${f}, skipping.`);continue}let m;try{m=JSON.parse(E.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${p}: ${m.tests.length} tests from ${f}`),c+=m.totalDuration||0;let y=k.join(f,"screenshots");E.existsSync(y)&&Hr(y,k.join(r,"screenshots",p));let v=k.join(r,p);for(let P of m.tests){let A=[P,...P.attempts||[]];for(let M of A)Fi(M.steps,p),M.videoPath&&Br(f,M.videoPath,v)&&(M.videoPath=`${p}/${M.videoPath}`),M.tracePath&&Br(f,M.tracePath,v)&&(M.tracePath=`${p}/${M.tracePath}`);l.push(P)}d++}l.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let u={tests:l,totalDuration:c,timestamp:new Date().toISOString(),shiplightVersion:Se};E.writeFileSync(k.join(r,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let h=k.join(r,"index.html");if(E.writeFileSync(h,Ft(u),"utf-8"),console.log(`
1559
- Merged ${l.length} tests from ${d} shards into: ${h}`),await Wr(u,r),n&&Ui(l),t)try{let g=(await import("open")).default;await g(h)}catch{}}function Br(e,t,n){let r=k.resolve(e,t);return r.startsWith(k.resolve(e)+k.sep)?E.existsSync(r)?(E.mkdirSync(n,{recursive:!0}),E.copyFileSync(r,k.join(n,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Hr(e,t){E.mkdirSync(t,{recursive:!0});for(let n of E.readdirSync(e,{withFileTypes:!0})){let r=k.join(e,n.name),s=k.join(t,n.name);n.isDirectory()?Hr(r,s):E.copyFileSync(r,s)}}function Fi(e,t){for(let n of e)n.screenshot?.startsWith("screenshots/")&&(n.screenshot=n.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function Wr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let r=e.tests.map(o=>o.startTime).filter(o=>!!o),s=r.length>0?r.sort()[0]:e.timestamp??new Date().toISOString();try{await Ur(e,t,s,n)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function Ht(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),n=k.join("tests",k.basename(t));return{name:e.title||k.basename(t),yamlPath:n}}function Gr(e){let t=e.filter(a=>!a.file.includes("auth.setup")),n=t.filter(a=>a.flaky),r=t.filter(a=>a.status==="passed"&&!a.flaky),s=t.filter(a=>a.status!=="passed"),o=t.length,i=`## Test Results
1557
+ </html>`}var jr=_(()=>{"use strict"});import*as J from"fs";import*as ee from"path";import{execFileSync as Ei}from"child_process";import{createHash as $i}from"crypto";import ve from"axios";function Fr(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let n=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(n)return`https://${n[1]}/${n[2]}`}function ie(...e){try{return Ei("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Mi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=J.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Ii(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",n=process.env.GITHUB_REPOSITORY??"",r=process.env.GITHUB_RUN_ID??"",s=process.env.GITHUB_EVENT_NAME??"",i=Mi().pull_request,a=i?.head?.sha,l=process.env.SHIPLIGHT_GIT_SHA??a??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??i?.title,g=ie("log","-1","--pretty=%s"),h=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:l,gitBranch:d,gitRepo:n,commitMessage:g,authorEmail:h,prNumber:u,prTitle:f,prUrl:u&&n?`${t}/${n}/pull/${u}`:void 0,ciBuildId:r,ciBuildUrl:r&&n?`${t}/${n}/actions/runs/${r}`:void 0,commitUrl:l&&n?`${t}/${n}/commit/${l}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:s,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",n=process.env.CI_COMMIT_SHA??"",r=process.env.CI_MERGE_REQUEST_IID,s=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:n,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:s,prNumber:r,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:r&&t?`${t}/-/merge_requests/${r}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:n&&t?`${t}/commit/${n}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",n=process.env.CIRCLE_PROJECT_USERNAME??"",r=process.env.CIRCLE_PROJECT_REPONAME??"",s=n&&r?`${n}/${r}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,i=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,l=a?Fr(a):void 0,c=ie("log","-1","--pretty=%s"),u=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:c,authorEmail:u,prNumber:i,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=ie("rev-parse","HEAD"),n=ie("rev-parse","--abbrev-ref","HEAD"),r=ie("log","-1","--pretty=%s"),s=ie("log","-1","--pretty=%ae"),o=ie("remote","get-url","origin"),i=o?Fr(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:n,commitMessage:r,authorEmail:s,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function Oi(e){switch(e){case"passed":return"Passed";case"failed":return"Failed";case"timedOut":return"TimedOut";case"skipped":return"Skipped";case"interrupted":return"Failed";default:return"Failed"}}function Ur(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Li(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 Ri(e,t){let n=new Map;for(let r of t){r.screenshotS3Uris={};let s=n.get(r.testCaseName);s?s.push(r):n.set(r.testCaseName,[r])}return e.map(r=>n.get(r.title)?.shift())}function nt(e,t){return ee.isAbsolute(t)?t:ee.join(e,t)}function Ci(e,t,n,r){let s={};for(let o of e.steps)s[o.stepId]={description:o.description,status:o.status,duration:o.duration,message:o.error??o.message,screenshotS3Uri:t[o.stepId]};return{schemaVersion:2,result:Ur(e.status),flaky:!1,segments:[{outcome:Ur(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:s,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:n,traceS3Uri:r,actionStepsMap:e.actionStepsMap??{}}]}}function Wt(e){return $i("md5").update(e).digest("base64")}function Bt(e){return Wt(J.readFileSync(e))}async function Ht(e,t){let n=J.readFileSync(t),r=ee.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[r]??"application/octet-stream";await ve.put(e,n,{headers:{"Content-Type":o,"Content-MD5":Wt(n)}})}async function Br(e,t,n,r){let s=Qe(r,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${r}`,"Content-Type":"application/json"},i=Ii(),a=e.tests.map(p=>{let v={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 w=nt(t,p.videoPath);J.existsSync(w)&&(v.videoMd5=Bt(w))}if(p.tracePath){let w=nt(t,p.tracePath);J.existsSync(w)&&(v.traceMd5=Bt(w))}return v}),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 ve.post(`${s}/v1/local-runs`,{trigger:i.ciProvider,startTime:n,metadata:i,tests:a},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Ri(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,v)=>{let w=d[v];if(!w)return;let m=p.steps.filter(P=>P.screenshot);if(!m.length)return;let y=m.map(P=>P.stepId),k={};for(let P of m){let E=ee.isAbsolute(P.screenshot)?P.screenshot:ee.join(t,P.screenshot);J.existsSync(E)&&(k[P.stepId]=Bt(E))}try{let P=await ve.post(`${s}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:k},{headers:o});w.uploadUrls.screenshots=P.data.screenshots,w.screenshotS3Uris=P.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${p.title}"`)}catch(P){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,P)}})),console.log("[reporter] [3/4] Uploading assets...");let f=(await Promise.all(e.tests.map(async(p,v)=>{let w=d[v];if(!w){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=w.uploadUrls,y={},k=0;await Promise.all(p.steps.map(async L=>{if(L.screenshot&&m.screenshots?.[L.stepId]){let he=ee.isAbsolute(L.screenshot)?L.screenshot:ee.join(t,L.screenshot);if(J.existsSync(he))try{await Ht(m.screenshots[L.stepId],he),y[L.stepId]=w.screenshotS3Uris[L.stepId],k++}catch(as){console.warn(`[reporter] Screenshot upload failed for step ${L.stepId}:`,as)}}})),k>0&&console.log(`[reporter] [3/4] Uploaded ${k} screenshot(s) for "${p.title}"`);let P;if(p.videoPath&&m.video){let L=nt(t,p.videoPath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await Ht(m.video,L),P=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(he){console.warn("[reporter] Video upload failed:",he)}}}let E;if(p.tracePath&&m.trace){let L=nt(t,p.tracePath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await Ht(m.trace,L),E=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(he){console.warn("[reporter] Trace upload failed:",he)}}}console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let Se=Ci(p,y,P,E),Oe=Buffer.from(JSON.stringify(Se)),Kt=Wt(Oe),zt=await ve.post(`${s}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:Kt},{headers:o}),os=zt.data.reportUrl,is=zt.data.reportS3Uri;return await ve.put(os,Oe,{headers:{"Content-Type":"application/json","Content-MD5":Kt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:w.testCaseResultId,result:Oi(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:is,videoS3Uri:P,traceS3Uri:E,metadata:{suiteName:p.suiteName,file:p.file}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=Li(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let h=await ve.put(`${s}/v1/local-runs/${u.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:f},{headers:o});console.log(`
1558
+ Shiplight cloud report: ${Ni(h.data.reportUrl,s)}`)}function Ni(e,t){if(/^https?:\/\//.test(e))return e;let n=e.startsWith("/")?e:`/${e}`;return`${Di(t)}${n}`}function Di(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 Hr=_(()=>{"use strict";Ot()});var Vr={};te(Vr,{buildGitHubSummary:()=>zr,runReport:()=>ji});import*as $ from"fs";import*as A from"path";async function ji(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"),n=e.includes("--merge"),r=e.includes("--github-summary");!n&&r&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),n?await Ui(e,t,r):await Fi(e,t)}async function Fi(e,t){let n=e.find(a=>!a.startsWith("--"))||"shiplight-report",r=A.isAbsolute(n)?n:A.join(process.cwd(),n),s=A.join(r,"report-data.json");$.existsSync(s)||(console.error(`Error: ${s} 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($.readFileSync(s,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${s}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=A.join(r,"index.html");if($.writeFileSync(i,Ut(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Kr(o,r),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Ui(e,t,n){let r=A.join(process.cwd(),"shiplight-report"),s=e.findIndex(g=>g==="-o"||g==="--output");if(s!==-1&&e[s+1]){let g=e[s+1];r=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let g=0;g<e.length;g++){let h=e[g];if(o.has(h)){g++;continue}if(i.has(h))continue;let p=A.isAbsolute(h)?h:A.join(process.cwd(),h);a.push(p)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let l=[],c=0,u=0;$.mkdirSync(A.join(r,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let h=a[g],p=`shard-${g}`,v=A.join(h,"report-data.json");if(!$.existsSync(v)){console.warn(`Warning: No report-data.json found in ${h}, skipping.`);continue}let w;try{w=JSON.parse($.readFileSync(v,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${v}, skipping.`);continue}console.log(`Merging ${p}: ${w.tests.length} tests from ${h}`),c+=w.totalDuration||0;let m=A.join(h,"screenshots");$.existsSync(m)&&Gr(m,A.join(r,"screenshots",p));let y=A.join(r,p);for(let k of w.tests){let P=[k,...k.attempts||[]];for(let E of P)Bi(E.steps,p),E.videoPath&&Wr(h,E.videoPath,y)&&(E.videoPath=`${p}/${E.videoPath}`),E.tracePath&&Wr(h,E.tracePath,y)&&(E.tracePath=`${p}/${E.tracePath}`);l.push(k)}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:_e};$.writeFileSync(A.join(r,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let f=A.join(r,"index.html");if($.writeFileSync(f,Ut(d),"utf-8"),console.log(`
1559
+ Merged ${l.length} tests from ${u} shards into: ${f}`),await Kr(d,r),n&&Hi(l),t)try{let g=(await import("open")).default;await g(f)}catch{}}function Wr(e,t,n){let r=A.resolve(e,t);return r.startsWith(A.resolve(e)+A.sep)?$.existsSync(r)?($.mkdirSync(n,{recursive:!0}),$.copyFileSync(r,A.join(n,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Gr(e,t){$.mkdirSync(t,{recursive:!0});for(let n of $.readdirSync(e,{withFileTypes:!0})){let r=A.join(e,n.name),s=A.join(t,n.name);n.isDirectory()?Gr(r,s):$.copyFileSync(r,s)}}function Bi(e,t){for(let n of e)n.screenshot?.startsWith("screenshots/")&&(n.screenshot=n.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function Kr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let r=e.tests.map(o=>o.startTime).filter(o=>!!o),s=r.length>0?r.sort()[0]:e.timestamp??new Date().toISOString();try{await Br(e,t,s,n)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function Gt(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),n=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:n}}function zr(e){let t=e.filter(a=>!a.file.includes("auth.setup")),n=t.filter(a=>a.flaky),r=t.filter(a=>a.status==="passed"&&!a.flaky),s=t.filter(a=>a.status!=="passed"),o=t.length,i=`## Test Results
1560
1560
 
1561
1561
  `;if(s.length===0&&n.length===0?i+=`\u2705 All ${o} tests passed
1562
1562
 
@@ -1564,7 +1564,7 @@ Merged ${l.length} tests from ${d} shards into: ${h}`),await Wr(u,r),n&&Ui(l),t)
1564
1564
 
1565
1565
  `:i+=`\u274C ${s.length} failed, \u26A0\uFE0F ${n.length} flaky, \u2705 ${r.length} passed / ${o} total
1566
1566
 
1567
- `,s.length>0){let a=[...new Set(s.map(l=>Ht(l).yamlPath))];i+=`### Failed
1567
+ `,s.length>0){let a=[...new Set(s.map(l=>Gt(l).yamlPath))];i+=`### Failed
1568
1568
 
1569
1569
  `;for(let l of a)i+=`- \`npx shiplight test ${l}\`
1570
1570
  `;i+=`
@@ -1572,18 +1572,18 @@ Merged ${l.length} tests from ${d} shards into: ${h}`),await Wr(u,r),n&&Ui(l),t)
1572
1572
 
1573
1573
  `,i+="```sh\n",i+=`npx shiplight test ${a.map(l=>`"${l}"`).join(` \\
1574
1574
  `)}
1575
- `,i+="```\n\n"}if(n.length>0){let a=[...new Set(n.map(l=>Ht(l).yamlPath))];i+=`### Flaky (${a.length})
1575
+ `,i+="```\n\n"}if(n.length>0){let a=[...new Set(n.map(l=>Gt(l).yamlPath))];i+=`### Flaky (${a.length})
1576
1576
 
1577
1577
  `;for(let l of a)i+=`- \`${l}\`
1578
1578
  `;i+=`
1579
- `}if(r.length>0){let a=[...new Set(r.map(l=>Ht(l).yamlPath))];i+=`<details><summary>Passed (${a.length})</summary>
1579
+ `}if(r.length>0){let a=[...new Set(r.map(l=>Gt(l).yamlPath))];i+=`<details><summary>Passed (${a.length})</summary>
1580
1580
 
1581
1581
  `;for(let l of a)i+=`- ${l}
1582
1582
  `;i+=`
1583
1583
  </details>
1584
- `}return i}function Ui(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}E.appendFileSync(t,Gr(e)),console.log("GitHub step summary written.")}var Vr=_(()=>{"use strict";Nr();jr();$e()});var zr,Yr=_(()=>{"use strict";zr="0.1.62"});var Jr={};ee(Jr,{runTranspile:()=>Bi});import*as tt from"path";import{glob as ji}from"glob";async function Bi(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",n=process.cwd(),r=await ji(t,{cwd:n,ignore:["node_modules/**","*.yaml.spec.ts"]});r.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of r.sort()){let l=tt.resolve(n,a),c=sr(l,{version:zr});if(!c.valid){s++,console.log(`
1585
- \u2717 ${a}`);for(let u of c.errors)console.log(` ERROR: ${u}`);continue}i++;let d=tt.basename(c.specFile);if(c.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${d}`);for(let u of c.warnings)console.log(` WARNING: ${u}`)}else console.log(`\u2713 ${a} \u2192 ${d}`)}console.log(`
1586
- ${r.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var qr=_(()=>{"use strict";kt();Yr()});var Qr={};ee(Qr,{runInspect:()=>Hi});import*as nt from"fs";import*as Xr from"path";async function Hi(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"),n=e.includes("--stats"),r=e.find(i=>!i.startsWith("--"));r||(console.error("Error: no file specified"),process.exit(1));let s=Xr.resolve(process.cwd(),r);nt.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=nt.readFileSync(s,"utf-8");try{let i=Te(o),a=C(o);if(n)Wi(a,i);else{let l={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(l,null,t?0:2))}}catch(i){console.error(`Error parsing ${r}: ${i.message}`),process.exit(1)}}function Wi(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 n=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${n.tests.length}`);for(let r of n.tests){let s=r.skip?` [SKIP${typeof r.skip=="string"?`: ${r.skip}`:""}]`:"";console.log(` - ${r.name}: ${r.statements.length} statements${r.teardown?`, ${r.teardown.length} teardown`:""}${s}`)}n.beforeAll?.length&&console.log(`Hooks: beforeAll (${n.beforeAll.length})`),n.afterAll?.length&&console.log(`Hooks: afterAll (${n.afterAll.length})`),n.beforeEach?.length&&console.log(`Hooks: beforeEach (${n.beforeEach.length})`),n.afterEach?.length&&console.log(`Hooks: afterEach (${n.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 n=Zr(e.statements??[]);console.log(` DRAFT: ${n.drafts}, ACTION: ${n.actions}, STEP: ${n.steps}`)}}function Zr(e){let t={drafts:0,actions:0,steps:0};for(let n of e)if(n.type==="DRAFT")t.drafts++;else if(n.type==="ACTION")t.actions++;else if(n.type==="STEP"){t.steps++;let r=Zr(n.statements??[]);t.drafts+=r.drafts,t.actions+=r.actions,t.steps+=r.steps}return t}var es=_(()=>{"use strict";ae()});var ts=as((Gp,Gi)=>{Gi.exports={name:"shiplightai",version:"0.1.62",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:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@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"}});$e();import Ki from"dotenv";Ki.config();Vt();zt();function ns(){console.log(`
1584
+ `}return i}function Hi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}$.appendFileSync(t,zr(e)),console.log("GitHub step summary written.")}var Yr=_(()=>{"use strict";jr();Hr();Le()});var Jr,qr=_(()=>{"use strict";Jr="0.1.64"});var Xr={};te(Xr,{runTranspile:()=>Gi});import*as rt from"path";import{glob as Wi}from"glob";async function Gi(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",n=process.cwd(),r=await Wi(t,{cwd:n,ignore:["node_modules/**","*.yaml.spec.ts"]});r.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of r.sort()){let l=rt.resolve(n,a),c=ar(l,{version:Jr});if(!c.valid){s++,console.log(`
1585
+ \u2717 ${a}`);for(let d of c.errors)console.log(` ERROR: ${d}`);continue}i++;let u=rt.basename(c.specFile);if(c.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${u}`);for(let d of c.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${a} \u2192 ${u}`)}console.log(`
1586
+ ${r.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var Zr=_(()=>{"use strict";Pt();qr()});var ts={};te(ts,{runInspect:()=>Ki});import*as st from"fs";import*as Qr from"path";async function Ki(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"),n=e.includes("--stats"),r=e.find(i=>!i.startsWith("--"));r||(console.error("Error: no file specified"),process.exit(1));let s=Qr.resolve(process.cwd(),r);st.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=st.readFileSync(s,"utf-8");try{let i=Pe(o),a=C(o);if(n)zi(a,i);else{let l={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(l,null,t?0:2))}}catch(i){console.error(`Error parsing ${r}: ${i.message}`),process.exit(1)}}function zi(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 n=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${n.tests.length}`);for(let r of n.tests){let s=r.skip?` [SKIP${typeof r.skip=="string"?`: ${r.skip}`:""}]`:"";console.log(` - ${r.name}: ${r.statements.length} statements${r.teardown?`, ${r.teardown.length} teardown`:""}${s}`)}n.beforeAll?.length&&console.log(`Hooks: beforeAll (${n.beforeAll.length})`),n.afterAll?.length&&console.log(`Hooks: afterAll (${n.afterAll.length})`),n.beforeEach?.length&&console.log(`Hooks: beforeEach (${n.beforeEach.length})`),n.afterEach?.length&&console.log(`Hooks: afterEach (${n.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 n=es(e.statements??[]);console.log(` DRAFT: ${n.drafts}, ACTION: ${n.actions}, STEP: ${n.steps}`)}}function es(e){let t={drafts:0,actions:0,steps:0};for(let n of e)if(n.type==="DRAFT")t.drafts++;else if(n.type==="ACTION")t.actions++;else if(n.type==="STEP"){t.steps++;let r=es(n.statements??[]);t.drafts+=r.drafts,t.actions+=r.actions,t.steps+=r.steps}return t}var ns=_(()=>{"use strict";ce()});var rs=ls((Vp,Vi)=>{Vi.exports={name:"shiplightai",version:"0.1.64",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:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@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"}});Le();import Yi from"dotenv";Yi.config();Vt();Yt();function ss(){console.log(`
1587
1587
  Usage: shiplight <command> [options]
1588
1588
 
1589
1589
  Commands:
@@ -1605,5 +1605,5 @@ Examples:
1605
1605
  shiplight transpile
1606
1606
  shiplight transpile "tests/**/*.test.yaml"
1607
1607
  shiplight debug tests/login.test.yaml
1608
- `)}var rt=process.argv[2];switch(rt){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(Xt(),qt));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Tr(),xr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Lr(),Or));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(Vr(),Kr));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(qr(),Jr));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(es(),Qr));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=ts().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":ns();break;default:rt&&console.error(`Unknown command: ${rt}
1609
- `),ns(),process.exit(rt?1:0)}
1608
+ `)}var ot=process.argv[2];switch(ot){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(Zt(),Xt));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Pr(),kr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Cr(),Rr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(Yr(),Vr));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(Zr(),Xr));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(ns(),ts));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=rs().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":ss();break;default:ot&&console.error(`Unknown command: ${ot}
1609
+ `),ss(),process.exit(ot?1:0)}