shiplightai 0.1.84 → 0.1.85

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 Hs=Object.defineProperty;var x=(e,t)=>()=>(e&&(t=e(e=0)),t);var Bs=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ue=(e,t)=>{for(var r in t)Hs(e,r,{get:t[r],enumerable:!0})};import*as U from"fs";import*as ee from"path";import*as mt from"os";import{execFileSync as Ws}from"child_process";function yt(){return ee.join(mt.homedir(),".shiplight","version-check.json")}function zs(){return ee.join(mt.homedir(),".shiplight","npm-prefix.json")}function Ys(e=zs()){try{let r=U.readFileSync(e,"utf-8"),n=JSON.parse(r);if(typeof n.prefix=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<Vs)return n.prefix}catch{}let t;try{t=Ws("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{U.mkdirSync(ee.dirname(e),{recursive:!0}),U.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function dr(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Ys()),n=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=r();if(!i)return;let a,c;try{a=U.realpathSync(t),c=U.realpathSync(i)}catch{return}if(a.startsWith(c+ee.sep)){s(`
4
+ var Ws=Object.defineProperty;var x=(e,t)=>()=>(e&&(t=e(e=0)),t);var Gs=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ue=(e,t)=>{for(var r in t)Ws(e,r,{get:t[r],enumerable:!0})};import*as F from"fs";import*as ee from"path";import*as mt from"os";import{execFileSync as Ks}from"child_process";function yt(){return ee.join(mt.homedir(),".shiplight","version-check.json")}function Js(){return ee.join(mt.homedir(),".shiplight","npm-prefix.json")}function Xs(e=Js()){try{let r=F.readFileSync(e,"utf-8"),n=JSON.parse(r);if(typeof n.prefix=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<Ys)return n.prefix}catch{}let t;try{t=Ks("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{F.mkdirSync(ee.dirname(e),{recursive:!0}),F.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function fr(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Xs()),n=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=r();if(!i)return;let a,c;try{a=F.realpathSync(t),c=F.realpathSync(i)}catch{return}if(a.startsWith(c+ee.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,21 +10,21 @@ 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 l=ee.join(c,"lib","node_modules","shiplightai"),p=ee.join(c,"node_modules","shiplightai"),f=U.existsSync(l)?l:U.existsSync(p)?p:null;f&&n(`
14
- \x1B[33m\u26A0 A global shiplightai install was detected at ${f}.
13
+ `),o(1);return}let l=ee.join(c,"lib","node_modules","shiplightai"),p=ee.join(c,"node_modules","shiplightai"),d=F.existsSync(l)?l:F.existsSync(p)?p:null;d&&n(`
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 Js(e=yt()){try{let t=U.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<Ks)return r}catch{}return null}function Xs(e,t=yt()){try{U.mkdirSync(ee.dirname(t),{recursive:!0}),U.writeFileSync(t,JSON.stringify(e))}catch{}}async function qs(){try{let e=await fetch(Gs,{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 Zs(e,t){let r=f=>{let h=f.indexOf("-");return h===-1?[f,!1]:[f.slice(0,h),!0]},n=f=>f.split(".").map(h=>parseInt(h,10)||0),[s,o]=r(e),[i,a]=r(t),c=n(s),l=n(i),p=Math.max(c.length,l.length);for(let f=0;f<p;f++){let h=c[f]??0,g=l[f]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!a)}async function fr(e={}){let t=e.runningVersion??gt,r=e.cwd??process.cwd(),n=e.cacheFile??yt(),s=e.fetchLatest??qs,o=e.env??process.env,i=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!U.existsSync(ee.join(r,"package-lock.json")))return;let a=null,c=Js(n);if(c)a=c.latest;else{try{a=await s()}catch{a=null}a&&Xs({latest:a,fetchedAt:Date.now()},n)}a&&Zs(t,a)&&i(`
17
+ `)}catch{}}function qs(e=yt()){try{let t=F.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<zs)return r}catch{}return null}function Zs(e,t=yt()){try{F.mkdirSync(ee.dirname(t),{recursive:!0}),F.writeFileSync(t,JSON.stringify(e))}catch{}}async function Qs(){try{let e=await fetch(Vs,{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 eo(e,t){let r=d=>{let h=d.indexOf("-");return h===-1?[d,!1]:[d.slice(0,h),!0]},n=d=>d.split(".").map(h=>parseInt(h,10)||0),[s,o]=r(e),[i,a]=r(t),c=n(s),l=n(i),p=Math.max(c.length,l.length);for(let d=0;d<p;d++){let h=c[d]??0,g=l[d]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!a)}async function hr(e={}){let t=e.runningVersion??gt,r=e.cwd??process.cwd(),n=e.cacheFile??yt(),s=e.fetchLatest??Qs,o=e.env??process.env,i=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!F.existsSync(ee.join(r,"package-lock.json")))return;let a=null,c=qs(n);if(c)a=c.latest;else{try{a=await s()}catch{a=null}a&&Zs({latest:a,fetchedAt:Date.now()},n)}a&&eo(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 gt,Me,Gs,Ks,Vs,Ge=x(()=>{"use strict";gt="0.1.84",Me=gt!=="dev"?gt:void 0,Gs="https://registry.npmjs.org/shiplightai/latest",Ks=3600*1e3,Vs=10080*60*1e3});import*as N from"fs";import*as O from"path";import{builtinModules as Qs}from"node:module";function Oe(e){let t;try{t=N.lstatSync(e)}catch(r){return r.code==="ENOENT"?"absent":"inaccessible"}if(t.isSymbolicLink())try{return N.statSync(e).isDirectory()?"directory":"non-directory"}catch(r){return r.code==="ENOENT"?"broken-symlink":"inaccessible"}return t.isDirectory()?"directory":"non-directory"}function gr(e){let t=Oe(e);return t==="directory"||t==="non-directory"}function bt(e,t){try{return N.realpathSync(e)}catch(r){throw new Error(`Cannot scaffold: failed to resolve real path of ${e} (${t}): ${r.message}. This is required for the project-root containment check that prevents writes outside the project.`)}}function co(e){let t="shiplight-test-project",r=e.trim();if(!r)return{name:t,original:e};let n=r.toLowerCase().replace(/[^a-z0-9._-]+/g,"-").replace(/^[^a-z0-9]+/,"").replace(/[-_.]+$/,"");return n?io.has(n)?{name:t,original:e}:n.length>ao?{name:t,original:e}:{name:n,original:e}:{name:t,original:e}}function lo(e){let t=Oe(e);if(!(t==="absent"||t==="directory"))throw t==="broken-symlink"?new Error(`Cannot scaffold into ${e}: path is a broken symlink. Remove or fix the link before scaffolding.`):t==="inaccessible"?new Error(`Cannot scaffold into ${e}: cannot stat path (permission denied or other I/O error).`):new Error(`Cannot scaffold into ${e}: path exists and is not a directory.`)}function uo(e){let t=O.join(e,"package.json"),r;try{r=N.statSync(t)}catch(i){return i.code==="ENOENT"?"absent":"unknown"}if(!r.isFile()||r.size>po)return"unknown";let n;try{n=N.readFileSync(t,"utf-8")}catch{return"unknown"}let s;try{s=JSON.parse(n)}catch{return"unknown"}if(Array.isArray(s)||!s||typeof s!="object")return"unknown";let o=s;return o.type==="module"?"module":o.type==="commonjs"?"commonjs":"unknown"}function vt(e){return e.some(t=>!mr.has(t))}function yr(e,t){let r=O.relative(e,t);return r===""?!0:!(r===".."||r.startsWith(".."+O.sep)||O.isAbsolute(r))}function ho(e,t,r,n){let s=Oe(e);if(s==="absent")return;let o=null;try{o=N.lstatSync(e)}catch{}let i=o?.isSymbolicLink()??!1;if(s==="broken-symlink")throw new Error(`Cannot scaffold ${r}: ${e} is a broken symlink. Remove or fix the link before scaffolding.`);if(s==="inaccessible")throw new Error(`Cannot scaffold ${r}: cannot stat ${e} (permission denied or other I/O error). Resolve the access problem before scaffolding.`);if(s==="directory"){let c=i?"a symlink that resolves to a directory":"a directory",l=i?"Remove the symlink (or point it at a file) before scaffolding.":"Remove or rename the directory before scaffolding.";throw new Error(`Cannot scaffold ${r}: ${e} is ${c}, but a file is expected here. ${l}`)}if(!i||n==="skip")return;let a=bt(e,`leaf symlink at ${r}`);if(!yr(t,a))throw new Error(`Cannot scaffold ${r}: ${e} is a symlink that resolves to ${a}, which is outside the project root ${t}. Writes through this link would escape the project.`)}function St(e){let t=O.resolve(e.projectPath),r=co(e.projectName??O.basename(t)),n=r.name;lo(t);let s=[];if(Oe(t)==="directory")try{s=N.readdirSync(t)}catch{}N.mkdirSync(t,{recursive:!0});let o=bt(t,"project root"),i=hr.split(`
21
- `).map(u=>u.trim()).filter(u=>u.length>0&&!u.startsWith("#")),a=fo.map(u=>({rel:u,abs:O.join(t,u)})).filter(u=>gr(u.abs)),c=uo(t),l=a.length===0&&c==="commonjs",p=[];if(p.push({kind:"agent_merge",relPath:"package.json",content:eo.replace(/\{\{name\}\}/g,()=>n),merge:{strategy:"json_merge_deps_and_scripts",humanSummary:"Add shiplightai to dependencies, add test scripts, set type:module and engines.node>=22 (with confirmation).",instructions:`An existing package.json was found. Merge the Shiplight template into it WITHOUT clobbering user fields:
20
+ `)}var gt,Me,Vs,zs,Ys,Ge=x(()=>{"use strict";gt="0.1.85",Me=gt!=="dev"?gt:void 0,Vs="https://registry.npmjs.org/shiplightai/latest",zs=3600*1e3,Ys=10080*60*1e3});import*as N from"fs";import*as O from"path";import{builtinModules as to}from"node:module";function Oe(e){let t;try{t=N.lstatSync(e)}catch(r){return r.code==="ENOENT"?"absent":"inaccessible"}if(t.isSymbolicLink())try{return N.statSync(e).isDirectory()?"directory":"non-directory"}catch(r){return r.code==="ENOENT"?"broken-symlink":"inaccessible"}return t.isDirectory()?"directory":"non-directory"}function mr(e){let t=Oe(e);return t==="directory"||t==="non-directory"}function bt(e,t){try{return N.realpathSync(e)}catch(r){throw new Error(`Cannot scaffold: failed to resolve real path of ${e} (${t}): ${r.message}. This is required for the project-root containment check that prevents writes outside the project.`)}}function po(e){let t="shiplight-test-project",r=e.trim();if(!r)return{name:t,original:e};let n=r.toLowerCase().replace(/[^a-z0-9._-]+/g,"-").replace(/^[^a-z0-9]+/,"").replace(/[-_.]+$/,"");return n?co.has(n)?{name:t,original:e}:n.length>lo?{name:t,original:e}:{name:n,original:e}:{name:t,original:e}}function uo(e){let t=Oe(e);if(!(t==="absent"||t==="directory"))throw t==="broken-symlink"?new Error(`Cannot scaffold into ${e}: path is a broken symlink. Remove or fix the link before scaffolding.`):t==="inaccessible"?new Error(`Cannot scaffold into ${e}: cannot stat path (permission denied or other I/O error).`):new Error(`Cannot scaffold into ${e}: path exists and is not a directory.`)}function ho(e){let t=O.join(e,"package.json"),r;try{r=N.statSync(t)}catch(i){return i.code==="ENOENT"?"absent":"unknown"}if(!r.isFile()||r.size>fo)return"unknown";let n;try{n=N.readFileSync(t,"utf-8")}catch{return"unknown"}let s;try{s=JSON.parse(n)}catch{return"unknown"}if(Array.isArray(s)||!s||typeof s!="object")return"unknown";let o=s;return o.type==="module"?"module":o.type==="commonjs"?"commonjs":"unknown"}function vt(e){return e.some(t=>!yr.has(t))}function wr(e,t){let r=O.relative(e,t);return r===""?!0:!(r===".."||r.startsWith(".."+O.sep)||O.isAbsolute(r))}function mo(e,t,r,n){let s=Oe(e);if(s==="absent")return;let o=null;try{o=N.lstatSync(e)}catch{}let i=o?.isSymbolicLink()??!1;if(s==="broken-symlink")throw new Error(`Cannot scaffold ${r}: ${e} is a broken symlink. Remove or fix the link before scaffolding.`);if(s==="inaccessible")throw new Error(`Cannot scaffold ${r}: cannot stat ${e} (permission denied or other I/O error). Resolve the access problem before scaffolding.`);if(s==="directory"){let c=i?"a symlink that resolves to a directory":"a directory",l=i?"Remove the symlink (or point it at a file) before scaffolding.":"Remove or rename the directory before scaffolding.";throw new Error(`Cannot scaffold ${r}: ${e} is ${c}, but a file is expected here. ${l}`)}if(!i||n==="skip")return;let a=bt(e,`leaf symlink at ${r}`);if(!wr(t,a))throw new Error(`Cannot scaffold ${r}: ${e} is a symlink that resolves to ${a}, which is outside the project root ${t}. Writes through this link would escape the project.`)}function St(e){let t=O.resolve(e.projectPath),r=po(e.projectName??O.basename(t)),n=r.name;uo(t);let s=[];if(Oe(t)==="directory")try{s=N.readdirSync(t)}catch{}N.mkdirSync(t,{recursive:!0});let o=bt(t,"project root"),i=gr.split(`
21
+ `).map(u=>u.trim()).filter(u=>u.length>0&&!u.startsWith("#")),a=go.map(u=>({rel:u,abs:O.join(t,u)})).filter(u=>mr(u.abs)),c=ho(t),l=a.length===0&&c==="commonjs",p=[];if(p.push({kind:"agent_merge",relPath:"package.json",content:ro.replace(/\{\{name\}\}/g,()=>n),merge:{strategy:"json_merge_deps_and_scripts",humanSummary:"Add shiplightai to dependencies, add test scripts, set type:module and engines.node>=22 (with confirmation).",instructions:`An existing package.json was found. Merge the Shiplight template into it WITHOUT clobbering user fields:
22
22
  1. Under .dependencies: add any key from the template's dependencies that is not already present. Do NOT change the version of a key the user already pinned.
23
23
  2. Under .scripts: add "test" and "test:headed" only if missing. If a script of the same name already exists, do not overwrite \u2014 instead surface the conflict to the user and suggest renaming.
24
24
  3. The template sets "type": "module". If the existing package.json does not, ASK the user before changing it. (If the user keeps CommonJS, the playwright.config.* merge entry's own instructions handle translating the config \u2014 you do not need to chase the config from here.)
25
25
  4. The template sets "engines": { "node": ">=22" }. If the existing package.json has no engines.node, add it. If it pins an older Node, ASK the user \u2014 Shiplight requires Node 22 and will fail at runtime on older versions.
26
26
  5. Leave "name", "version", "description", "author", "license", and every other field alone.
27
- Use Read + Edit to perform a minimal in-place merge \u2014 do not rewrite the whole file.`}}),a.length>0)for(let u of a)p.push({kind:"agent_merge",relPath:u.rel,content:wt,merge:{strategy:"review_and_decide",humanSummary:`An existing ${u.rel} is already in place. We did not modify it.`,instructions:`An existing ${u.rel} was found. Show the user the Shiplight template and ask whether to (a) replace it entirely, (b) spread \`...shiplightConfig()\` from \`shiplightai\` into their existing defineConfig, or (c) leave their config as-is. Do not modify the file without explicit confirmation. If the user's package.json explicitly declares \`"type": "commonjs"\` (NOT just "no type field" \u2014 Shiplight treats missing type as ESM), translate the import-statement portion of the template to \`require()\` before offering option (a) or (b).`}});else l?p.push({kind:"agent_merge",relPath:"playwright.config.ts",content:wt,forceMerge:!0,merge:{strategy:"review_and_decide",humanSummary:"Your package.json declares CommonJS. We did not write the ESM playwright.config.ts \u2014 decide CJS vs flipping to type:module first.",instructions:'Note: this entry describes a file that DOES NOT YET EXIST on disk. Do not attempt to Read `abs_path` \u2014 instead, ASK the user before writing anything.\nThe user\'s package.json explicitly declares CommonJS ("type": "commonjs"). The Shiplight template uses ESM-only syntax (`import { defineConfig, shiplightConfig } from \'shiplightai\'; export default defineConfig({ ...shiplightConfig() })`), which would crash at load time with `Cannot use import statement outside a module`.\nASK the user whether to (a) flip package.json to `"type": "module"` (then write the template as-is to `abs_path`), or (b) keep CommonJS and write a CJS variant to `abs_path`: `const { defineConfig, shiplightConfig } = require(\'shiplightai\'); module.exports = defineConfig({ ...shiplightConfig() });`. Do not write the file without explicit confirmation.'}}):p.push({kind:"agent_merge",relPath:"playwright.config.ts",content:wt,merge:{strategy:"review_and_decide",humanSummary:"An existing playwright.config.ts is already in place. We did not modify it.",instructions:"An existing playwright.config.ts was found. Show the user the Shiplight template and ask whether to (a) replace it entirely, (b) spread `...shiplightConfig()` from `shiplightai` into their existing defineConfig, or (c) leave their config as-is. Do not modify the file without explicit confirmation."}});p.push({kind:"agent_merge",relPath:".gitignore",content:hr,merge:{strategy:"append_missing_lines",linesToEnsure:[...i],humanSummary:"Append Shiplight-required ignore patterns (node_modules/, .env, test-results/, ...) to .gitignore.",instructions:"An existing .gitignore was found. For each line in `linesToEnsure`, append it to the file only if no semantically-equivalent pattern is already present. Treat `node_modules`, `node_modules/`, `/node_modules`, and `**/node_modules` as equivalent (same goes for the other patterns) \u2014 if any form is present, skip the append. Group new entries under a `# Shiplight` comment block at the end of the file. Preserve all existing content, ordering, and comments."}}),p.push({kind:"agent_merge",relPath:".env.example",content:to,merge:{strategy:"append_missing_env_keys",humanSummary:"Append missing AI-provider keys GOOGLE_API_KEY / ANTHROPIC_API_KEY / OPENAI_API_KEY to .env.example.",instructions:"An existing .env.example was found. For each variable KEY in the template (lines matching `KEY=` or `# KEY=`), check whether KEY appears anywhere in the existing file. If not, append the corresponding line (preserving commented vs uncommented form). Group additions under a `# --- Shiplight ---` divider at the end. Preserve all existing content."}}),p.push({kind:"agent_merge",relPath:".mcp.json",content:ro,merge:{strategy:"json_merge_under_key",mergeKey:"mcpServers",humanSummary:"Add the shiplight MCP server entry under .mcpServers, or nested-merge missing fields if one already exists.",instructions:"An existing .mcp.json was found. Parse it, then merge the template under .mcpServers:\n1. If the existing file has no \"shiplight\" key under .mcpServers, add the entire template entry.\n2. If a \"shiplight\" key already exists, perform a NESTED merge \u2014 for each field in the template's shiplight entry (`command`, `args`, `env`, ...), add the field if missing, and within `env` add any missing keys (e.g. `PWDEBUG`) without overwriting values the user already set. Do NOT replace the user's existing values.\n3. Do not modify or overwrite any other top-level mcpServers entries.\nPreserve overall formatting where possible.\nNote: .mcp.json is for the user's coding agent's MCP discovery \u2014 Shiplight CLI tests run fine without it. Treat this merge as integration setup, not a hard prerequisite."}}),p.push({kind:"skip",relPath:"auth/example.login.ts",content:no}),p.push({kind:"skip",relPath:"environments/example.env.yaml",content:so}),p.push({kind:"skip",relPath:"tests/example.test.yaml",content:oo});for(let u of p){let d=O.join(t,u.relPath),w=O.dirname(d);for(;;){let b=O.relative(t,w);if(b===""||b===".."||b.startsWith(".."+O.sep)||O.isAbsolute(b))break;let y=Oe(w);if(y!=="absent"&&y!=="directory"){let S=` (project root: ${o}). If multiple plans share this ancestor, the same issue blocks them all.`;throw new Error(y==="broken-symlink"?`Cannot scaffold ${u.relPath}: parent path ${w} is a broken symlink.${S}`:y==="inaccessible"?`Cannot scaffold ${u.relPath}: cannot stat parent path ${w} (permission denied or other I/O error).${S}`:`Cannot scaffold ${u.relPath}: parent path ${w} exists and is not a directory.${S}`)}if(y==="directory"){let S=bt(w,`parent of ${u.relPath}`);if(!yr(o,S))throw new Error(`Cannot scaffold ${u.relPath}: parent path ${w} resolves to ${S}, which is outside the project root ${o}.`)}let m=O.dirname(w);if(m===w)break;w=m}ho(d,o,u.relPath,u.kind)}let f=[],h=[],g=[];for(let u of p){let d=O.join(t,u.relPath),w=gr(d),b=u.kind==="agent_merge"&&u.forceMerge===!0;if(!w&&!b){N.mkdirSync(O.dirname(d),{recursive:!0}),N.writeFileSync(d,u.content),f.push(u.relPath);continue}if(u.kind==="skip"){h.push(u.relPath);continue}let y={path:u.relPath,absPath:d,mergeStrategy:u.merge.strategy,template:u.content,instructions:u.merge.instructions,humanSummary:u.merge.humanSummary};u.merge.mergeKey!==void 0&&(y.mergeKey=u.merge.mergeKey),u.merge.linesToEnsure!==void 0&&(y.linesToEnsure=u.merge.linesToEnsure),g.push(y)}return{projectPath:t,projectName:n,originalProjectName:r.original,filesCreated:f,filesSkipped:h,filesNeedingAgentMerge:g,preExistingTopLevelEntries:s}}var eo,wt,hr,to,ro,no,so,oo,io,ao,po,fo,mr,wr=x(()=>{"use strict";eo=`{
27
+ Use Read + Edit to perform a minimal in-place merge \u2014 do not rewrite the whole file.`}}),a.length>0)for(let u of a)p.push({kind:"agent_merge",relPath:u.rel,content:wt,merge:{strategy:"review_and_decide",humanSummary:`An existing ${u.rel} is already in place. We did not modify it.`,instructions:`An existing ${u.rel} was found. Show the user the Shiplight template and ask whether to (a) replace it entirely, (b) spread \`...shiplightConfig()\` from \`shiplightai\` into their existing defineConfig, or (c) leave their config as-is. Do not modify the file without explicit confirmation. If the user's package.json explicitly declares \`"type": "commonjs"\` (NOT just "no type field" \u2014 Shiplight treats missing type as ESM), translate the import-statement portion of the template to \`require()\` before offering option (a) or (b).`}});else l?p.push({kind:"agent_merge",relPath:"playwright.config.ts",content:wt,forceMerge:!0,merge:{strategy:"review_and_decide",humanSummary:"Your package.json declares CommonJS. We did not write the ESM playwright.config.ts \u2014 decide CJS vs flipping to type:module first.",instructions:'Note: this entry describes a file that DOES NOT YET EXIST on disk. Do not attempt to Read `abs_path` \u2014 instead, ASK the user before writing anything.\nThe user\'s package.json explicitly declares CommonJS ("type": "commonjs"). The Shiplight template uses ESM-only syntax (`import { defineConfig, shiplightConfig } from \'shiplightai\'; export default defineConfig({ ...shiplightConfig() })`), which would crash at load time with `Cannot use import statement outside a module`.\nASK the user whether to (a) flip package.json to `"type": "module"` (then write the template as-is to `abs_path`), or (b) keep CommonJS and write a CJS variant to `abs_path`: `const { defineConfig, shiplightConfig } = require(\'shiplightai\'); module.exports = defineConfig({ ...shiplightConfig() });`. Do not write the file without explicit confirmation.'}}):p.push({kind:"agent_merge",relPath:"playwright.config.ts",content:wt,merge:{strategy:"review_and_decide",humanSummary:"An existing playwright.config.ts is already in place. We did not modify it.",instructions:"An existing playwright.config.ts was found. Show the user the Shiplight template and ask whether to (a) replace it entirely, (b) spread `...shiplightConfig()` from `shiplightai` into their existing defineConfig, or (c) leave their config as-is. Do not modify the file without explicit confirmation."}});p.push({kind:"agent_merge",relPath:".gitignore",content:gr,merge:{strategy:"append_missing_lines",linesToEnsure:[...i],humanSummary:"Append Shiplight-required ignore patterns (node_modules/, .env, test-results/, ...) to .gitignore.",instructions:"An existing .gitignore was found. For each line in `linesToEnsure`, append it to the file only if no semantically-equivalent pattern is already present. Treat `node_modules`, `node_modules/`, `/node_modules`, and `**/node_modules` as equivalent (same goes for the other patterns) \u2014 if any form is present, skip the append. Group new entries under a `# Shiplight` comment block at the end of the file. Preserve all existing content, ordering, and comments."}}),p.push({kind:"agent_merge",relPath:".env.example",content:no,merge:{strategy:"append_missing_env_keys",humanSummary:"Append missing AI-provider keys GOOGLE_API_KEY / ANTHROPIC_API_KEY / OPENAI_API_KEY to .env.example.",instructions:"An existing .env.example was found. For each variable KEY in the template (lines matching `KEY=` or `# KEY=`), check whether KEY appears anywhere in the existing file. If not, append the corresponding line (preserving commented vs uncommented form). Group additions under a `# --- Shiplight ---` divider at the end. Preserve all existing content."}}),p.push({kind:"agent_merge",relPath:".mcp.json",content:so,merge:{strategy:"json_merge_under_key",mergeKey:"mcpServers",humanSummary:"Add the shiplight MCP server entry under .mcpServers, or nested-merge missing fields if one already exists.",instructions:"An existing .mcp.json was found. Parse it, then merge the template under .mcpServers:\n1. If the existing file has no \"shiplight\" key under .mcpServers, add the entire template entry.\n2. If a \"shiplight\" key already exists, perform a NESTED merge \u2014 for each field in the template's shiplight entry (`command`, `args`, `env`, ...), add the field if missing, and within `env` add any missing keys (e.g. `PWDEBUG`) without overwriting values the user already set. Do NOT replace the user's existing values.\n3. Do not modify or overwrite any other top-level mcpServers entries.\nPreserve overall formatting where possible.\nNote: .mcp.json is for the user's coding agent's MCP discovery \u2014 Shiplight CLI tests run fine without it. Treat this merge as integration setup, not a hard prerequisite."}}),p.push({kind:"skip",relPath:"auth/example.login.ts",content:oo}),p.push({kind:"skip",relPath:"environments/example.env.yaml",content:io}),p.push({kind:"skip",relPath:"tests/example.test.yaml",content:ao});for(let u of p){let f=O.join(t,u.relPath),w=O.dirname(f);for(;;){let b=O.relative(t,w);if(b===""||b===".."||b.startsWith(".."+O.sep)||O.isAbsolute(b))break;let y=Oe(w);if(y!=="absent"&&y!=="directory"){let S=` (project root: ${o}). If multiple plans share this ancestor, the same issue blocks them all.`;throw new Error(y==="broken-symlink"?`Cannot scaffold ${u.relPath}: parent path ${w} is a broken symlink.${S}`:y==="inaccessible"?`Cannot scaffold ${u.relPath}: cannot stat parent path ${w} (permission denied or other I/O error).${S}`:`Cannot scaffold ${u.relPath}: parent path ${w} exists and is not a directory.${S}`)}if(y==="directory"){let S=bt(w,`parent of ${u.relPath}`);if(!wr(o,S))throw new Error(`Cannot scaffold ${u.relPath}: parent path ${w} resolves to ${S}, which is outside the project root ${o}.`)}let m=O.dirname(w);if(m===w)break;w=m}mo(f,o,u.relPath,u.kind)}let d=[],h=[],g=[];for(let u of p){let f=O.join(t,u.relPath),w=mr(f),b=u.kind==="agent_merge"&&u.forceMerge===!0;if(!w&&!b){N.mkdirSync(O.dirname(f),{recursive:!0}),N.writeFileSync(f,u.content),d.push(u.relPath);continue}if(u.kind==="skip"){h.push(u.relPath);continue}let y={path:u.relPath,absPath:f,mergeStrategy:u.merge.strategy,template:u.content,instructions:u.merge.instructions,humanSummary:u.merge.humanSummary};u.merge.mergeKey!==void 0&&(y.mergeKey=u.merge.mergeKey),u.merge.linesToEnsure!==void 0&&(y.linesToEnsure=u.merge.linesToEnsure),g.push(y)}return{projectPath:t,projectName:n,originalProjectName:r.original,filesCreated:d,filesSkipped:h,filesNeedingAgentMerge:g,preExistingTopLevelEntries:s}}var ro,wt,gr,no,so,oo,io,ao,co,lo,fo,go,yr,br=x(()=>{"use strict";ro=`{
28
28
  "name": "{{name}}",
29
29
  "type": "module",
30
30
  "engines": {
@@ -40,6 +40,11 @@ Use Read + Edit to perform a minimal in-place merge \u2014 do not rewrite the wh
40
40
  }
41
41
  `,wt=`import { defineConfig, shiplightConfig } from 'shiplightai';
42
42
 
43
+ // testDir '.' scans the whole project; shiplightConfig() defaults its YAML scan
44
+ // to the same project root, so nothing extra is needed here. If you narrow
45
+ // testDir to a subfolder, pass the same path as scanDir so the two stay in sync:
46
+ // const testDir = './e2e';
47
+ // export default defineConfig({ ...shiplightConfig({ scanDir: testDir }), testDir, ... });
43
48
  export default defineConfig({
44
49
  ...shiplightConfig(),
45
50
  testDir: '.',
@@ -56,13 +61,13 @@ export default defineConfig({
56
61
  trace: 'on',
57
62
  },
58
63
  });
59
- `,hr=`node_modules/
64
+ `,gr=`node_modules/
60
65
  test-results/
61
66
  shiplight-report/
62
67
  .shiplight/
63
68
  .env
64
69
  *.yaml.spec.ts
65
- .auth`,to=`# Shiplight API token
70
+ .auth`,no=`# Shiplight API token
66
71
  # Get yours at https://nova.shiplight.ai/api-tokens
67
72
  # If set, no AI provider API key required.
68
73
  # SHIPLIGHT_API_TOKEN=
@@ -74,7 +79,7 @@ shiplight-report/
74
79
 
75
80
  # Optional: override the default AI model.
76
81
  # WEB_AGENT_MODEL=claude-sonnet-4-6
77
- `,ro=`{
82
+ `,so=`{
78
83
  "mcpServers": {
79
84
  "shiplight": {
80
85
  "command": "npx",
@@ -87,7 +92,7 @@ shiplight-report/
87
92
  }
88
93
  }
89
94
  }
90
- }`,no=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
95
+ }`,oo=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
91
96
  import path from 'path';
92
97
  import fs from 'fs/promises';
93
98
  import { chromium } from '@playwright/test';
@@ -130,13 +135,13 @@ export async function login(args: Record<string, unknown> = {}): Promise<string>
130
135
 
131
136
  return sessionPath;
132
137
  }
133
- `,so=`# This is an example of environment. Don't use it.
138
+ `,io=`# This is an example of environment. Don't use it.
134
139
  name: example
135
140
  url: https://staging.example.com
136
141
  accounts:
137
142
  - username: test-user-1@example.com
138
143
  password: EXAMPLE_TEST_USER_1_PASSWORD
139
- 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,oo=`goal: Verify the Shiplight homepage links to the quick-start docs
144
+ 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,ao=`goal: Verify the Shiplight homepage links to the quick-start docs
140
145
  base_url: https://www.shiplight.ai
141
146
  statements:
142
147
  - URL: /
@@ -150,7 +155,7 @@ statements:
150
155
 
151
156
  - VERIFY: The browser lands on the docs site
152
157
  js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
153
- `;io=new Set(["node_modules","favicon.ico","node","npm","js","test","sea",...Qs]),ao=214;po=1048576;fo=["playwright.config.ts","playwright.config.js","playwright.config.mjs","playwright.config.cjs"],mr=new Set([".git",".gitkeep",".DS_Store","Thumbs.db",".idea",".vscode",".husky","node_modules",".next",".turbo",".cache",".pnpm-store"])});var br=x(()=>{"use strict";wr()});var vr={};ue(vr,{runCreate:()=>mo});import*as Ve from"path";function go(e){return/^[A-Za-z0-9_./~+-]+$/.test(e)?e:`'${e.replace(/'/g,"'\\''")}'`}function Ke(){console.log(`
158
+ `;co=new Set(["node_modules","favicon.ico","node","npm","js","test","sea",...to]),lo=214;fo=1048576;go=["playwright.config.ts","playwright.config.js","playwright.config.mjs","playwright.config.cjs"],yr=new Set([".git",".gitkeep",".DS_Store","Thumbs.db",".idea",".vscode",".husky","node_modules",".next",".turbo",".cache",".pnpm-store"])});var vr=x(()=>{"use strict";br()});var Sr={};ue(Sr,{runCreate:()=>wo});import*as Ve from"path";function yo(e){return/^[A-Za-z0-9_./~+-]+$/.test(e)?e:`'${e.replace(/'/g,"'\\''")}'`}function Ke(){console.log(`
154
159
  Usage: shiplight create <path> [options]
155
160
 
156
161
  Scaffold a Shiplight test project at <path>. Writes package.json,
@@ -167,11 +172,11 @@ Examples:
167
172
  shiplight create ./my-tests
168
173
  shiplight create ./my-tests --name acme-e2e
169
174
  shiplight create . # scaffold into the current repo
170
- `)}async function mo(e){let t,r;for(let u=0;u<e.length;u++){let d=e[u];if(d==="--help"||d==="-h"){Ke();return}else d==="--name"?(r=e[++u],(!r||!r.trim())&&(console.error("Error: --name requires a non-empty value"),process.exit(1)),/^[a-z0-9][a-z0-9._-]*$/.test(r)||(console.error(`Error: --name "${r}" is not a valid package name.
175
+ `)}async function wo(e){let t,r;for(let u=0;u<e.length;u++){let f=e[u];if(f==="--help"||f==="-h"){Ke();return}else f==="--name"?(r=e[++u],(!r||!r.trim())&&(console.error("Error: --name requires a non-empty value"),process.exit(1)),/^[a-z0-9][a-z0-9._-]*$/.test(r)||(console.error(`Error: --name "${r}" is not a valid package name.
171
176
  npm package names must be lowercase.
172
177
  Use only lowercase letters, digits, hyphens, underscores, and dots;
173
- must start with a lowercase letter or digit.`),process.exit(1))):d.startsWith("--")?(console.error(`Error: Unknown option: ${d}`),Ke(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${d}`),Ke(),process.exit(1)):t=d}t||(console.error(`Error: missing required <path> argument
174
- `),Ke(),process.exit(1));let n=Ve.resolve(t),s=Number(process.versions.node.split(".")[0]);Number.isFinite(s)&&s<22&&console.warn(`Warning: you are on Node ${process.versions.node}. Shiplight requires Node 22 or newer; tests will fail at runtime on older versions.`);let o;try{o=St({projectPath:n,projectName:r})}catch(u){let d=u instanceof Error?u.message:String(u);console.error(`Error: ${d}`),process.exit(1)}let i=vt(o.preExistingTopLevelEntries);if(o.projectName!==o.originalProjectName&&o.originalProjectName.trim().length>0&&console.warn(`Note: project name '${o.originalProjectName}' was normalized to '${o.projectName}'. Override with --name <name> if you want a different value.`),console.log(`
178
+ must start with a lowercase letter or digit.`),process.exit(1))):f.startsWith("--")?(console.error(`Error: Unknown option: ${f}`),Ke(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${f}`),Ke(),process.exit(1)):t=f}t||(console.error(`Error: missing required <path> argument
179
+ `),Ke(),process.exit(1));let n=Ve.resolve(t),s=Number(process.versions.node.split(".")[0]);Number.isFinite(s)&&s<22&&console.warn(`Warning: you are on Node ${process.versions.node}. Shiplight requires Node 22 or newer; tests will fail at runtime on older versions.`);let o;try{o=St({projectPath:n,projectName:r})}catch(u){let f=u instanceof Error?u.message:String(u);console.error(`Error: ${f}`),process.exit(1)}let i=vt(o.preExistingTopLevelEntries);if(o.projectName!==o.originalProjectName&&o.originalProjectName.trim().length>0&&console.warn(`Note: project name '${o.originalProjectName}' was normalized to '${o.projectName}'. Override with --name <name> if you want a different value.`),console.log(`
175
180
  ${i?"Added Shiplight files to existing project at":"Scaffolded Shiplight test project at"} ${o.projectPath}
176
181
  `),console.log(` Name: ${o.projectName}`),o.filesCreated.length>0){console.log(`
177
182
  Files created:`);for(let u of o.filesCreated)console.log(` + ${u}`)}if(o.filesSkipped.length>0){console.log(`
@@ -179,16 +184,16 @@ ${i?"Added Shiplight files to existing project at":"Scaffolded Shiplight test pr
179
184
  Files that already exist \u2014 left alone (you need to merge them):`);for(let u of o.filesNeedingAgentMerge)console.log(` ! ${u.path}`),console.log(` ${u.humanSummary}`);console.log(`
180
185
  Open each file and apply the change yourself, or let your coding
181
186
  agent do it: the MCP \`scaffold_project\` tool returns the full template
182
- content + per-file merge_strategy in its response.`)}let c=o.preExistingTopLevelEntries.includes(".env"),l=o.filesNeedingAgentMerge.some(u=>u.path===".env.example"),p=o.filesCreated.includes(".env.example"),f=c&&l?"# .env AND .env.example both exist: merge .env.example first, then add any newly-required keys to your .env; do NOT cp over your existing .env":c&&p?"# A fresh .env.example was just written next to your existing .env \u2014 diff them and add any new required keys (e.g. GOOGLE_API_KEY) to your .env; do NOT overwrite":c?"# .env already exists \u2014 review it and ensure an AI provider key (GOOGLE_API_KEY or ANTHROPIC_API_KEY) is set; do NOT overwrite":l?"# AFTER merging .env.example above, run: cp .env.example .env # then set an AI provider key":"cp .env.example .env # then edit .env and set an AI provider API key",h=go(Ve.relative(process.cwd(),o.projectPath)||"."),g=o.filesNeedingAgentMerge.length>0?`
187
+ content + per-file merge_strategy in its response.`)}let c=o.preExistingTopLevelEntries.includes(".env"),l=o.filesNeedingAgentMerge.some(u=>u.path===".env.example"),p=o.filesCreated.includes(".env.example"),d=c&&l?"# .env AND .env.example both exist: merge .env.example first, then add any newly-required keys to your .env; do NOT cp over your existing .env":c&&p?"# A fresh .env.example was just written next to your existing .env \u2014 diff them and add any new required keys (e.g. GOOGLE_API_KEY) to your .env; do NOT overwrite":c?"# .env already exists \u2014 review it and ensure an AI provider key (GOOGLE_API_KEY or ANTHROPIC_API_KEY) is set; do NOT overwrite":l?"# AFTER merging .env.example above, run: cp .env.example .env # then set an AI provider key":"cp .env.example .env # then edit .env and set an AI provider API key",h=yo(Ve.relative(process.cwd(),o.projectPath)||"."),g=o.filesNeedingAgentMerge.length>0?`
183
188
  # resolve the merge-required files listed above first`:"";console.log(`
184
189
  Next steps:
185
190
 
186
191
  cd ${h}${g}
187
- ${f}
192
+ ${d}
188
193
  npm install
189
194
  npx playwright install chromium
190
195
  npx shiplight test
191
- `)}var Sr=x(()=>{"use strict";br()});import*as te from"fs";import*as de from"path";function _r(e){let t=de.resolve(e);for(;;){if(te.existsSync(de.join(t,"package.json"))||te.existsSync(de.join(t,".shiplight")))return t;let r=de.dirname(t);if(r===t)break;t=r}return null}function xr(e){return de.join(e,yo)}function Tr(e,t){return de.join(xr(e),`${wo}${t}.json`)}function kr(e,t){let r=_r(t);if(!r)return;let n=xr(r);te.mkdirSync(n,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};te.writeFileSync(Tr(r,e),JSON.stringify(s),"utf-8")}function Pr(e,t){let r=_r(t);if(r)try{te.unlinkSync(Tr(r,e))}catch{}}var yo,wo,Er=x(()=>{"use strict";yo=".shiplight/run",wo="debug-"});import*as $r from"node:net";function Ar(e,t){return new Promise(r=>{let n=$r.createServer();n.once("error",s=>{r(s.code!=="EADDRINUSE")}),n.once("listening",()=>{n.close(()=>r(!0))}),n.listen(e,t)})}async function bo(e){return await Ar(e,"127.0.0.1")?Ar(e,"::1"):!1}async function Ir(e,t){for(let r=e;r<e+t;r++)if(await bo(r))return r;return null}var Mr=x(()=>{"use strict"});var Rr={};ue(Rr,{findPlaywrightConfig:()=>Re,makeIdempotentFileCleaner:()=>Or,spawnPlaywrightProcess:()=>_t});import*as fe from"fs";import*as X from"path";import{spawn as vo}from"child_process";import{parse as So}from"yaml";function Or(e){let t=!1;return()=>{if(!t){t=!0;try{fe.unlinkSync(e)}catch{}}}}function Re(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=X.resolve(e);for(;;){for(let s of t){let o=X.join(r,s);if(fe.existsSync(o))return o}let n=X.dirname(r);if(n===r)break;r=n}return null}function _o(e,t,r,n){let s={...r},o=s.launchOptions?.args??[];return s.launchOptions={...s.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
196
+ `)}var _r=x(()=>{"use strict";vr()});import*as te from"fs";import*as de from"path";function xr(e){let t=de.resolve(e);for(;;){if(te.existsSync(de.join(t,"package.json"))||te.existsSync(de.join(t,".shiplight")))return t;let r=de.dirname(t);if(r===t)break;t=r}return null}function Tr(e){return de.join(e,bo)}function kr(e,t){return de.join(Tr(e),`${vo}${t}.json`)}function Pr(e,t){let r=xr(t);if(!r)return;let n=Tr(r);te.mkdirSync(n,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};te.writeFileSync(kr(r,e),JSON.stringify(s),"utf-8")}function Er(e,t){let r=xr(t);if(r)try{te.unlinkSync(kr(r,e))}catch{}}var bo,vo,Ar=x(()=>{"use strict";bo=".shiplight/run",vo="debug-"});import*as Ir from"node:net";function $r(e,t){return new Promise(r=>{let n=Ir.createServer();n.once("error",s=>{r(s.code!=="EADDRINUSE")}),n.once("listening",()=>{n.close(()=>r(!0))}),n.listen(e,t)})}async function So(e){return await $r(e,"127.0.0.1")?$r(e,"::1"):!1}async function Mr(e,t){for(let r=e;r<e+t;r++)if(await So(r))return r;return null}var Or=x(()=>{"use strict"});var Lr={};ue(Lr,{findPlaywrightConfig:()=>Re,makeIdempotentFileCleaner:()=>Rr,spawnPlaywrightProcess:()=>_t});import*as fe from"fs";import*as X from"path";import{spawn as _o}from"child_process";import{parse as xo}from"yaml";function Rr(e){let t=!1;return()=>{if(!t){t=!0;try{fe.unlinkSync(e)}catch{}}}}function Re(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=X.resolve(e);for(;;){for(let s of t){let o=X.join(r,s);if(fe.existsSync(o))return o}let n=X.dirname(r);if(n===r)break;r=n}return null}function To(e,t,r,n){let s={...r},o=s.launchOptions?.args??[];return s.launchOptions={...s.launchOptions??{},args:[...o,"--remote-debugging-port=0"]},`// @generated by shiplightai \u2014 temporary debug test
192
197
  import { test } from 'shiplightai/fixture';
193
198
  ${`
194
199
  test.use(${JSON.stringify(s)});
@@ -212,30 +217,30 @@ test('__shiplight_debug__', async ({ page, agent }) => {
212
217
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
213
218
  await new Promise(() => {});
214
219
  });
215
- `}async function xo(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function _t(e){let{yamlFilePath:t,configPath:r,tempSuffix:n="",headed:s}=e,o=X.dirname(r),i=await xo(16174),a;if(!fe.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let S=So(fe.readFileSync(t,"utf-8"));S?.use&&typeof S.use=="object"&&!Array.isArray(S.use)&&(a=S.use),S?.base_url&&!a?.baseURL&&(a={...a,baseURL:S.base_url})}catch(S){console.error("[debugger] Could not parse YAML for `use` block:",S)}let c=X.dirname(X.resolve(t)),l=n?`-${n}`:"",p=X.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),f=_o(t,i,a,o);fe.writeFileSync(p,f);let h=Or(p),u=["playwright","test",X.relative(o,p).replace(/\\/g,"/"),...s?["--headed"]:[]],d=vo("npx",u,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",OMNITERM_BROWSER_REGISTRY_URL:""}});d.stdout?.on("data",S=>{process.stderr.write(S)}),d.stderr?.on("data",S=>{process.stderr.write(S)});let w=()=>{d.killed||d.kill("SIGTERM")};process.on("SIGTERM",w),process.on("SIGINT",w),process.on("exit",w),d.on("close",S=>{process.removeListener("SIGTERM",w),process.removeListener("SIGINT",w),process.removeListener("exit",w),h(),S!==0&&S!==null&&console.error(`[debugger] Playwright process exited with code ${S}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let b=["127.0.0.1","::1"];async function y(S){try{let P=S.includes(":")?`[${S}]`:S,k=await fetch(`http://${P}:${i}/api/test-flow`);if(k.ok){try{await k.text()}catch{}return!0}}catch{}return!1}let m=null;for(let S=0;S<180;S++){if(d.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${d.exitCode} before sandbox was ready`);for(let P of b)if(await y(P)){m=P;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${i}`);break}if(S===179)throw w(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(P=>setTimeout(P,1e3))}if(!m)throw w(),h(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:m,pid:d.pid??0,cleanup:async()=>{w(),h()}}}var xt=x(()=>{"use strict"});var Lr=x(()=>{"use strict"});var Cr=x(()=>{"use strict"});var Nr=x(()=>{"use strict"});var Tt,ze,Ye=x(()=>{"use strict";Tt=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},ze=e=>{let t=[],r=n=>{for(let s of n){t.push(s);let o=Tt(s);for(let i of o)r(i.statements)}};return r(e),t}});function kt(){return{version:"1.0",entries:{}}}var jr=x(()=>{"use strict";Ye()});import{v4 as pc}from"uuid";var Dr=x(()=>{"use strict"});import{z as v}from"zod";var Ur,Pt,Fr,xe,Hr,Br,Wr,V,Gr,Je,Et,At=x(()=>{"use strict";Ur=v.enum(["JS_CODE","AI_MODE"]),Pt=v.object({type:Ur,expression:v.string()}),Fr=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),xe=v.object({uid:v.string(),type:Fr,comment:v.string().optional()}),Hr=v.object({action_data:v.object({action_name:v.string(),kwargs:v.record(v.any()).optional(),args:v.array(v.any()).optional()}),action_description:v.string().optional(),url:v.string().optional(),xpath:v.string().nullable().optional(),locator:v.string().nullable().optional(),css_selector:v.string().nullable().optional(),unique_selector:v.string().nullable().optional(),element_index:v.number().nullable().optional(),frame_path:v.array(v.any()).optional(),artifacts:v.record(v.any()).optional(),feedback:v.string().optional(),original_browser_use_action:v.any().optional()}).passthrough(),Br=xe.extend({type:v.literal("DRAFT"),description:v.string()}),Wr=xe.extend({type:v.literal("ACTION"),description:v.string(),action_entity:Hr.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),V=v.lazy(()=>v.union([Br,Wr,xe.extend({type:v.literal("STEP"),description:v.string().optional().default(""),statements:v.array(V),reference_id:v.number().optional(),template_path:v.string().optional(),template_params:v.record(v.string()).optional()}),xe.extend({type:v.literal("IF_ELSE"),description:v.string().optional(),condition:Pt,then:v.array(V),else:v.array(V).optional()}),xe.extend({type:v.literal("WHILE_LOOP"),description:v.string().optional(),condition:Pt,body:v.array(V),timeout_ms:v.number().optional()})])),Gr=v.object({name:v.string(),statements:v.array(V),teardown:v.array(V).optional(),skip:v.union([v.boolean(),v.string()]).optional(),timeout:v.number().optional(),fail:v.union([v.boolean(),v.string()]).optional(),only:v.boolean().optional(),slow:v.boolean().optional()}),Je=v.object({tests:v.array(Gr).min(1),beforeAll:v.array(V).optional(),afterAll:v.array(V).optional(),beforeEach:v.array(V).optional(),afterEach:v.array(V).optional()}),Et=v.object({comment:v.string().optional(),version:v.string().optional(),goal:v.string().optional(),url:v.string().optional(),baseURL:v.string().optional(),final_feedback:v.string().optional(),completed:v.boolean().optional(),success:v.boolean().optional(),statements:v.array(V).optional(),teardown:v.array(V).optional(),last_modified_at:v.string().optional(),testGroup:Je.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 mc,parse as Yr,parseAllDocuments as yc,parseDocument as To,Document as Jr,isMap as ke,isSeq as z}from"yaml";import{v4 as re}from"uuid";function Xe(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(q)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(q)),r}function It(e,t){if(e.testGroup)return qr(e,t);let r=Xe(e,t),n=new Jr(r),s=n.contents?.get("tags",!0);return z(s)&&(s.flow=!0),e.comment&&(n.commentBefore=e.comment),Kr(n,e.statements??[]),e.teardown&&Kr(n,e.teardown,"teardown"),n.toString(Xr)}function Kr(e,t,r="statements"){let n=e.contents;if(!n||!ke(n))return;let s=n.get(r,!0);z(s)&&Le(s,t)}function Le(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=t[r],s=e.items[r];if(r>0&&(s.spaceBefore=!0),n.comment&&(r===0?e.commentBefore=n.comment:s.commentBefore=n.comment),ke(s)){let o=s;if(n.type==="STEP"){let i=o.get("statements",!0);z(i)&&Le(i,n.statements)}else if(n.type==="IF_ELSE"){let i=o.get("THEN",!0);z(i)&&Le(i,n.then);let a=o.get("ELSE",!0);z(a)&&n.else&&Le(a,n.else)}else if(n.type==="WHILE_LOOP"){let i=o.get("DO",!0);z(i)&&Le(i,n.body)}}}}function qr(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let n={};t?.test_case_id!==void 0&&(n.test_case_id=t.test_case_id),t?.name&&(n.name=t.name),t?.tags&&t.tags.length>0&&(n.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(n.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(n.settings=t.settings);let s={};e.baseURL&&(s.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(s.beforeAll=r.beforeAll.map(q)),r.beforeEach&&r.beforeEach.length>0&&(s.beforeEach=r.beforeEach.map(q)),r.afterEach&&r.afterEach.length>0&&(s.afterEach=r.afterEach.map(q)),r.afterAll&&r.afterAll.length>0&&(s.afterAll=r.afterAll.map(q)),s.tests=r.tests.map(a=>{let c={name:a.name};return a.skip!==void 0&&(c.skip=a.skip),a.timeout!==void 0&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only!==void 0&&(c.only=a.only),a.slow!==void 0&&(c.slow=a.slow),c.statements=a.statements.map(q),a.teardown&&a.teardown.length>0&&(c.teardown=a.teardown.map(q)),c}),n.suite=s;let o=new Jr(n),i=o.contents?.get("tags",!0);return z(i)&&(i.flow=!0),o.toString(Xr)}function q(e){switch(e.type){case"DRAFT":return ko(e);case"ACTION":return Po(e);case"STEP":return Eo(e);case"IF_ELSE":return Ao(e);case"WHILE_LOOP":return $o(e)}}function ko(e){return{intent:e.description}}function Po(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&a.trim()&&e.description)return{description:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:a}}if(!e.action_entity)return{intent:e.description};let n=e.action_entity.action_data??e.action_entity.action;if(!n)return{intent:e.description};let s={intent:e.description,action:n.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),n.kwargs&&Object.keys(n.kwargs).length>0)for(let[a,c]of Object.entries(n.kwargs))a!=="uid"&&(a==="statement"&&(t==="ai_action"||t==="ai_step")||(s[a]=c));return n.args&&n.args.length>0&&(s.args=n.args),s}function Eo(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(q)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Ao(e){let t={IF:Zr(e.condition),THEN:e.then.map(q)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(q)),t}function $o(e){let t={WHILE:Zr(e.condition),DO:e.body.map(q)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Zr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ce(e){try{let t=Yr(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(n=>typeof n=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function $t(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map($t);let t=e,r=Object.keys(t);if(r.length===1){let s=r[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let n={};for(let[s,o]of Object.entries(t))n[s]=$t(o);return n}function Y(e){if(e.length>Vr)throw new Error(`YAML input too large (${e.length} bytes, max ${Vr})`);let t=$t(Yr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Io(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:Z(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=Z(t.teardown));let n=Et.safeParse(r);if(!n.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(n.error.errors)}`);let s=n.data;return qe(e,s),s}function Io(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:Z(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=Z(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=Z(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=Z(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=Z(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=Z(t.afterEach));let o=Je.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 Z(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Mo)}function Mo(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 Oo(t);if("WHILE"in t)return Ro(t);if("STEP"in t)return Lo(t);if("VERIFY"in t){let r=t.VERIFY,n={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(n.code=t.js),{uid:re(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:n}}}}if("URL"in t){let r=t.URL,n=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return n&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:re(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:re(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:n}}}}}if("WAIT"in t){let r=t.WAIT,n=typeof t.seconds=="number"?t.seconds:3;return{uid:re(),type:"ACTION",description:typeof r=="string"?r:`Wait ${n}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${n}s`,action_data:{action_name:"wait",kwargs:{seconds:n}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');let n=typeof t.description=="string"?t.description:"Code block";return{uid:re(),type:"ACTION",description:n,action_entity:{action_description:n,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",n=t.js;return{uid:re(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...n}=t;return zr({...n,action:"function",functionName:r})}if("action"in t)return zr(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:re(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Qr(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 Oo(e){let t=Qr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let n={uid:re(),type:"IF_ELSE",condition:t,then:Z(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(n.else=Z(e.ELSE)),n}function Ro(e){let t=Qr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let n={uid:re(),type:"WHILE_LOOP",condition:t,body:Z(r)};return typeof e.timeout_ms=="number"&&(n.timeout_ms=e.timeout_ms),n}function Lo(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:re(),type:"STEP",description:t,statements:Z(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let n=e.template_params,s={};for(let[o,i]of Object.entries(n))s[o]=String(i);r.template_params=s}return r}function zr(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",n=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[l,p]of Object.entries(e))Co.has(l)||(i[l]=p);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js),(t==="ai_action"||t==="ai_step")&&i.statement===void 0&&(i.statement=r);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};n&&(a.locator=n),s&&(a.xpath=s);let c={uid:re(),type:"ACTION",description:r,action_entity:a};return o&&(c.use_pure_vision=!0),c}function qe(e,t){let r;try{r=To(e)}catch{return}let n=r.contents;if(!n||!ke(n))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=n.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let s=n,o=s.get("statements",!0);z(o)&&t.statements&&Te(o,t.statements);let i=s.get("teardown",!0);z(i)&&t.teardown&&Te(i,t.teardown)}function Te(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=e.items[r];n.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=n.commentBefore);let s=t[r];if(s.type==="STEP"&&ke(n)){let o=n.get("statements",!0);z(o)&&Te(o,s.statements)}else if(s.type==="IF_ELSE"&&ke(n)){let o=n.get("THEN",!0);z(o)&&Te(o,s.then);let i=n.get("ELSE",!0);z(i)&&s.else&&Te(i,s.else)}else if(s.type==="WHILE_LOOP"&&ke(n)){let o=n.get("DO",!0);z(o)&&Te(o,s.body)}}}var Xr,Vr,Co,Ze=x(()=>{"use strict";At();Xr={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Vr=1024*1024;Co=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as _c,stringify as xc}from"yaml";var en=x(()=>{"use strict";Ze()});function on(e){let t=0,r=0;for(let n of e)if(n.type==="DRAFT")r++;else if(n.type==="ACTION"){let s=n.action_entity?.action_data?.action_name??"";sn.has(s)||t++}return{action:t,draft:r}}function jo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function nn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Do(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,n=t.join(`
216
- `).match(/\bawait\b/g),s=n?n.length:0;return r<=tn&&s<=rn?null:`${r} non-blank line(s), ${s} await(s) \u2014 limits are ${tn} lines and ${rn} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function Mt(e,t){let r=t?.coverageThreshold??No,n=[],s=[],o;try{o=Y(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||n.push('Missing required field: "goal"'),o.statements?.length||n.push('Missing required field: "statements"');let i=[...ze(o.statements??[]),...o.teardown?ze(o.teardown):[]],{action:a,draft:c}=on(i),l="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 u=g.description||h.uid;n.push(`Unresolved cloud template reference on statement "${u}" (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 u=h,d=u.action_entity?.action_data?.action_name??"";if(d==="js_code"||d==="js_action"||d==="verify"||d==="ai_assert"){let w=u.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let b=jo(w);if(b){let y=u.description||d;n.push(`Invalid JS in "${y}": ${b}. ${l}`)}else if(d==="verify"){let y=Do(w);if(y){let m=u.description||d;n.push(`JS cache for "${m}" is too complex: ${y}`)}}}}}if(h.type==="IF_ELSE"){let u=h;if(u.condition.type==="JS_CODE"){let d=nn(u.condition.expression);d&&n.push(`Invalid JS in IF condition "${u.condition.expression}": ${d}. ${l}`)}}if(h.type==="WHILE_LOOP"){let u=h;if(u.condition.type==="JS_CODE"){let d=nn(u.condition.expression);d&&n.push(`Invalid JS in WHILE condition "${u.condition.expression}": ${d}. ${l}`)}}}let p=a+c,f=p>0?Math.round(a/p*100):0;return p>0&&f/100<r&&s.push(`Low action coverage: ${a}/${p} statements (${f}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:n.length===0,errors:n,warnings:s,stats:{total:p,action:a,draft:c,coverage:f}}}var No,tn,rn,sn,an=x(()=>{"use strict";Ye();Ze();No=.5,tn=5,rn=3,sn=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var ne,Ot,cn=x(()=>{"use strict";ne=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(ne||{}),Ot=18e4});var Qe=x(()=>{"use strict"});var ln=x(()=>{"use strict";Qe()});var pn=x(()=>{"use strict";Qe()});var un,Ho,dn,fn,Ne,Bo,Wo,hn=x(()=>{"use strict";un=112,Ho=1080-un,dn={"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"]},Ne=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),fn[e].map(n=>dn[n]).filter(n=>n.defaultBrowserType&&r.includes(n.defaultBrowserType))},Bo={desktop:{label:"Desktop",type:"desktop",devices:Ne("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Ne("mobile")}},Wo={desktop:{label:"Desktop",type:"desktop",devices:Ne("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Ne("mobile",!0)}}});var gn=x(()=>{"use strict"});var mn=x(()=>{"use strict"});var yn=x(()=>{"use strict"});var wn=x(()=>{"use strict"});var ge=x(()=>{"use strict";Lr();Cr();Nr();jr();Dr();en();an();Ye();Ze();cn();At();ln();pn();Qe();hn();gn();mn();yn();wn()});import{stringify as Go}from"yaml";import Rt from"node:path";import{createHash as ci}from"crypto";import{parse as li,stringify as An}from"yaml";import{readFileSync as pi,existsSync as ui}from"fs";import{resolve as Ct,dirname as di}from"path";import{parse as _n,stringify as fi}from"yaml";import{readFileSync as vi,writeFileSync as Si,mkdirSync as _i}from"fs";import{dirname as Tn}from"path";function ae(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function L(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Ko(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Vo(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 tt(e){let t=Ko(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let n=Vo(e);if(n){let s=JSON.stringify(n);return`${t}.locator(${s}).first()`}return null}function bn(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:zo.includes(t)}function Jo(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!Yo.includes(t)}function _(e,t){H.set(e,t)}function Xo(e){return H.get(e)}function be(e,t,r=[]){let n=[...r];return t.locator?n.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&n.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(t.frame_path)}`),n.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...n.map(s=>` ${s},`),"});"]}function jt(e,t){let r=t.action_data?.kwargs||{},n=[];return typeof r.relative_x=="number"&&typeof r.relative_y=="number"?(n.push(`action_data: { kwargs: { relative_x: ${r.relative_x}, relative_y: ${r.relative_y} } }`),t.locator?n.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&n.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(t.frame_path)}`)):typeof r.x=="number"&&typeof r.y=="number"&&n.push(`action_data: { kwargs: { x: ${r.x}, y: ${r.y} } }`),[`await agent.execAction("${e}", page, {`,...n.map(s=>` ${s},`),"});"]}function vn(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):Array.isArray(e.parameterValues)?e.parameterValues.map(String):[];if(r.length===0)return`await ${t}()`;let n=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=r.map(i=>n.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 ce(e,t,r,n="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${n}.${o}`,c=qo(i,t,a,r);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function qo(e,t,r,n){let s=" ".repeat(t);switch(e.type){case"DRAFT":return Zo(e,t,r,n);case"ACTION":return Qo(e,t,r,n);case"STEP":return ei(e,t,r,n);case"IF_ELSE":return ti(e,t,r,n);case"WHILE_LOOP":return ri(e,t,r,n);default:return[`${s}// Unknown statement type: ${e.type}`]}}function Zo(e,t,r,n){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${r}: Skipping - no description`];if(n.noAgent)return[`${s}// ${r}: ${L(o)}`,`${s}// DRAFT: ${L(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${r}: ${L(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}, '${r}');`]}function Qo(e,t,r,n){let s=" ".repeat(t),o=e.description,i=e.uid,c=n.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${s}// ${r}: Skipping - no description`];if(n.noAgent)return[`${s}// ${r}: ${L(o)}`,`${s}// DRAFT: ${L(o)} (requires agent - skipped in hook)`];let m=JSON.stringify(o),S=!!e.use_pure_vision;return[`${s}// ${r}: ${L(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, ${m}, '${r}', ${S});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let p=l.action_data?.action_name||"",f=l.action_description||"",h=Xo(p);if(!h)return[`${s}// ${r}: Unknown action: ${p}`,`${s}throw new Error(${JSON.stringify(`Unknown action: ${p}`)});`];let g={imports:n.imports,yamlDir:n.yamlDir,projectRoot:n.projectRoot},u=h(l,r,g);if(n.noAgent){if(bn(l))return[`${s}// ${r}: ${L(f)}`,`${s}// AI action: ${L(f)} (requires agent - skipped in hook)`];let m=ni(l,p,s,r);return m||[`${s}// ${r}: ${L(f)}`,...u.map(S=>`${s}${S}`)]}if(bn(l))return[`${s}// ${r}: ${L(f)}`,`${s}page = agent.agentServices.validatePage(page);`,...u.map(m=>`${s}${m}`)];let d=JSON.stringify(f),w=u.map(m=>`${s} ${m}`),b=Jo(l),y=i?`'${i}'`:"undefined";return[`${s}// ${r}: ${L(f)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...w,`${s}}, ${d}, '${r}', ${y}, ${b});`]}function ei(e,t,r,n){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${L(e.description)}`);let i=ce(e.statements,t,n,r);return o.push(...i),o}function ti(e,t,r,n){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${L(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let i=ce(e.then,t+1,n,`${r}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=ce(e.else,t+1,n,`${r}.else`);o.push(...a)}return o.push(`${s}}`),o}function ri(e,t,r,n){let s=" ".repeat(t),o=[];o.push(`${s}// ${r}: Loop`);let i=e.timeout_ms??Ot,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${i};`),o.push(`${s}const ${l}_check = () => {`),o.push(`${s} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${s} throw new Error('${c}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${L(e.condition.expression)}`);let f=JSON.stringify(e.condition.expression);o.push(`${s}while (${l}_check() && await agent.evaluate(page, ${f}, "${r}")) {`)}let p=ce(e.body,t+1,n,`${r}.body`);return o.push(...p),o.push(`${s}}`),o}function ni(e,t,r,n){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??Nt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${r}// ${n}: ${L(s)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${n}: ${L(s)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${n}: ${L(s)}`,`${r}await page.goForward();`];case"input_text":{let a=o.text||"",c=tt(e);return c?[`${r}// ${n}: ${L(s)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=tt(e);return c?[`${r}// ${n}: ${L(s)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function si(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...Pn()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore,yamlDir:t?.yamlDir,projectRoot:t?.projectRoot};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...Sn("beforeEach",t.beforeEach,o)),r.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",c=Lt(t?.tags);for(let l of t.parameters)r.push(...nt(e,`${c}${ae(a)} [${ae(l.name)}]`,o,0,i,l.name,l.values)),r.push("")}else{let a=t?.testName||e.goal||"Generated test",c=Lt(t?.tags);r.push(...nt(e,`${c}${ae(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...Sn("afterEach",t.afterEach,o))),En(r,s),r.join(`
217
- `)}function oi(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...Pn()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore,yamlDir:t?.yamlDir,projectRoot:t?.projectRoot},i=t?.testName||"Test Suite",a=Lt(t?.tags);r.push(`test.describe.serial('${a}${ae(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...et("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...et("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],f=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let h of p.parameters)r.push(...nt(p.testFlow,`${ae(p.name)} [${ae(h.name)}]`,o,1,f,h.name,h.values)),r.push("");else r.push(...nt(p.testFlow,ae(p.name),o,1,f)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...et("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...et("afterAll",e.afterAll,o,1)),r.push("});"),En(r,s),r.join(`
218
- `)}function Lt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function rt(e){let t=new Set;function r(n){for(let s of n)switch(s.type){case ne.ACTION:{let i=s.action_entity?.action_data?.kwargs,a=[...Array.isArray(i?.args)?i.args:[],...Array.isArray(i?.parameterValues)?i.parameterValues:[]];for(let c of a)typeof c=="string"&&ii.includes(c)&&t.add(c);break}case ne.STEP:r(s.statements);break;case ne.IF_ELSE:{let o=s;r(o.then),o.else&&r(o.else);break}case ne.WHILE_LOOP:r(s.body);break}}return r(e),t}function ai(e){let t=rt(e.statements??[]);if(e.teardown)for(let r of rt(e.teardown))t.add(r);return t}function Dt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function nt(e,t,r,n=0,s,o,i){let a=" ".repeat(n),c=[],l=ai(e),p=Dt(l),f=s?.only?"test.only":"test",h=o?`, { tag: '@${ae(o)}' }`:"";if(c.push(`${a}${f}('${t}'${h}, async (${p}) => {`),s?.skip===!0?c.push(`${a} test.skip();`):typeof s?.skip=="string"&&c.push(`${a} test.skip(true, '${ae(s.skip)}');`),s?.fail===!0?c.push(`${a} test.fail();`):typeof s?.fail=="string"&&c.push(`${a} test.fail(true, '${ae(s.fail)}');`),s?.slow&&c.push(`${a} test.slow();`),s?.timeout&&c.push(`${a} test.setTimeout(${s.timeout});`),i){for(let[d,w]of Object.entries(i))c.push(`${a} agent.agentServices.saveVariable(${JSON.stringify(d)}, ${JSON.stringify(w)});`);c.push("")}let g=e.teardown&&e.teardown.length>0,u=n+1;if(g){if(c.push(`${a} try {`),e.statements&&e.statements.length>0){c.push(`${a} // Test steps`);let w=ce(e.statements,u+1,r);c.push(...w)}c.push(`${a} } finally {`),c.push(`${a} // Teardown`);let d=ce(e.teardown,u+1,r,"teardown");c.push(...d),c.push(`${a} }`)}else if(e.statements&&e.statements.length>0){c.push(`${a} // Test steps`);let d=ce(e.statements,u,r);c.push(...d)}return c.push(`${a}});`),c}function Sn(e,t,r){let n=[],s=kn(t),o=rt(s),i=Dt(o);return n.push(`test.${e}(async (${i}) => {`),n.push(...ce(s,1,r,e)),n.push("});"),n}function et(e,t,r,n){let s=" ".repeat(n),o=[],i=kn(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,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(...ce(i,n+1,c,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let c=rt(i),l=Dt(c);o.push(`${s}test.${e}(async (${l}) => {`),o.push(...ce(i,n+1,r,e)),o.push(`${s}});`)}return o}function kn(e){let r=Go({goal:"_hook",statements:e});return Y(r).statements??[]}function Pn(){return["import { test, expect } from 'shiplightai/fixture';"]}function En(e,t){if(t.size>0){let r=0;for(let s=0;s<e.length;s++)e[s].startsWith("import ")&&(r=s+1);let n=Array.from(t);e.splice(r,0,...n)}}function $n(e,t,r){let n={expandingPaths:new Set([Ct(t)]),depth:0,referencedPaths:new Set,basePath:r},s={...e};Array.isArray(s.statements)&&(s.statements=me(s.statements,t,n)),Array.isArray(s.teardown)&&(s.teardown=me(s.teardown,t,n));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=me(s[o],t,n));return{doc:s,referencedTemplatePaths:Array.from(n.referencedPaths)}}function me(e,t,r){let n=[];for(let s of e)if(hi(s)){let o=gi(s,t,r);n.push(o)}else n.push(mi(s,t,r));return n}function hi(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function gi(e,t,r){if(r.depth>=xn)throw new Error(`Template expansion exceeded maximum depth of ${xn}. Check for deeply nested or circular template references.`);let n=Ct(di(t),e.template),s=!ui(n)&&r.basePath?Ct(r.basePath,e.template):n;if(r.expandingPaths.has(s))throw new Error(`Circular template reference detected: ${s} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${s}`);r.referencedPaths.add(s);let o;try{o=pi(s,"utf-8")}catch(u){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${u.message}`)}let i=_n(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let a=i.params||[],c=e.params||{};for(let u of a)if(!(u in c))throw new Error(`Template ${e.template} requires param "${u}" but it was not provided. Required params: [${a.join(", ")}]`);let l=i.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let d=fi(l);for(let[w,b]of Object.entries(c))d=d.split(`<<${w}>>`).join(String(b));l=_n(d)}let p={expandingPaths:new Set([...r.expandingPaths,s]),depth:r.depth+1,referencedPaths:r.referencedPaths},f=me(l,s,p),g={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:f};return Object.keys(c).length>0&&(g.template_params=c),g}function mi(e,t,r){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=me(n.statements,t,r)),Array.isArray(n.THEN)&&(n.THEN=me(n.THEN,t,r)),Array.isArray(n.ELSE)&&(n.ELSE=me(n.ELSE,t,r)),Array.isArray(n.DO)&&(n.DO=me(n.DO,t,r)),n}function Ft(e,t,r){let n=li(e),s=n?.name,o=n?.tags,i=n?.use;if(n&&(n.name!==void 0||n.tags!==void 0||n.use!==void 0)&&(delete n.name,delete n.tags,delete n.use),n?.suite){if(n.goal||n.statements)throw new Ut('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return wi(n,s,o,i,t,r)}return yi(n,s,o,i,t,r)}function yi(e,t,r,n,s,o){let i=e?.beforeEach,a=e?.afterEach,c=In(e?.parameters),l=e?.timeout,p=e?.skip,f=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 Ut(`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 u=[];if(s&&e&&typeof e=="object"){let b=$n(e,s,o);e=b.doc,u=b.referencedTemplatePaths}let d=An(e),w=Y(d);return s&&(Pe(w.statements??[],s,"main"),w.teardown&&Pe(w.teardown,s,"teardown")),{testFlow:w,name:t,tags:r,use:n,beforeEach:i,afterEach:a,parameters:c,timeout:l,skip:p,fail:f,only:h,slow:g,referencedTemplatePaths:u}}function wi(e,t,r,n,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,c=i.afterAll,l=i.beforeEach,p=i.afterEach,f=[],h=i.tests.map(d=>{if(!d.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(d.statements)||d.statements.length===0)throw new Error(`Suite test "${d.name}" must have a non-empty "statements" array.`);let w={goal:d.name,statements:d.statements};d.teardown&&(w.teardown=d.teardown);let b=[],y=w;if(s&&typeof w=="object"){let k=$n(w,s,o);y=k.doc,b=k.referencedTemplatePaths,f.push(...b)}let m=An(y),S=Y(m),P=In(d.parameters);return{testFlow:S,name:d.name,tags:Array.isArray(d.tags)?d.tags:void 0,parameters:P,timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}}),g=i.base_url,u=g?{...n,baseURL:g}:n;return{suite:{beforeAll:a,afterAll:c,beforeEach:l,afterEach:p,tests:h},name:t,tags:r,use:u,referencedTemplatePaths:f}}function In(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Pe(e,t,r){for(let n=0;n<e.length;n++){let s=e[n],o=`${r}.${n}`,i=s.description||"";if(s.uid=bi(t,o,i),s.type===ne.STEP)Pe(s.statements,t,o);else if(s.type===ne.IF_ELSE){let a=s;Pe(a.then,t,`${o}.then`),a.else&&Pe(a.else,t,`${o}.else`)}else s.type===ne.WHILE_LOOP&&Pe(s.body,t,`${o}.body`)}}function bi(e,t,r){let n=ci("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function Mn(e,t){let r;try{r=vi(e,"utf-8")}catch(n){return{valid:!1,errors:[`Failed to read file: ${n.message}`],warnings:[]}}return xi(r,e,t)}function xi(e,t,r){let n=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=n||s?null:Mt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,c=[];try{let l=r?.parsed??Ft(e,t,r?.basePath);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore,yamlDir:Tn(t),projectRoot:r?.projectRoot??r?.basePath},f=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?i=oi(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):i=si(l.testFlow,{...p,testName:l.name,tags:l.tags,use:f,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let h=i.split(`
220
+ `}async function ko(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function _t(e){let{yamlFilePath:t,configPath:r,tempSuffix:n="",headed:s}=e,o=X.dirname(r),i=await ko(16174),a;if(!fe.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let S=xo(fe.readFileSync(t,"utf-8"));S?.use&&typeof S.use=="object"&&!Array.isArray(S.use)&&(a=S.use),S?.base_url&&!a?.baseURL&&(a={...a,baseURL:S.base_url})}catch(S){console.error("[debugger] Could not parse YAML for `use` block:",S)}let c=X.dirname(X.resolve(t)),l=n?`-${n}`:"",p=X.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=To(t,i,a,o);fe.writeFileSync(p,d);let h=Rr(p),u=["playwright","test",X.relative(o,p).replace(/\\/g,"/"),...s?["--headed"]:[]],f=_o("npx",u,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",OMNITERM_BROWSER_REGISTRY_URL:""}});f.stdout?.on("data",S=>{process.stderr.write(S)}),f.stderr?.on("data",S=>{process.stderr.write(S)});let w=()=>{f.killed||f.kill("SIGTERM")};process.on("SIGTERM",w),process.on("SIGINT",w),process.on("exit",w),f.on("close",S=>{process.removeListener("SIGTERM",w),process.removeListener("SIGINT",w),process.removeListener("exit",w),h(),S!==0&&S!==null&&console.error(`[debugger] Playwright process exited with code ${S}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let b=["127.0.0.1","::1"];async function y(S){try{let P=S.includes(":")?`[${S}]`:S,k=await fetch(`http://${P}:${i}/api/test-flow`);if(k.ok){try{await k.text()}catch{}return!0}}catch{}return!1}let m=null;for(let S=0;S<180;S++){if(f.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${f.exitCode} before sandbox was ready`);for(let P of b)if(await y(P)){m=P;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${i}`);break}if(S===179)throw w(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(P=>setTimeout(P,1e3))}if(!m)throw w(),h(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:m,pid:f.pid??0,cleanup:async()=>{w(),h()}}}var xt=x(()=>{"use strict"});var Cr=x(()=>{"use strict"});var Nr=x(()=>{"use strict"});var Dr=x(()=>{"use strict"});var Tt,ze,Ye=x(()=>{"use strict";Tt=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},ze=e=>{let t=[],r=n=>{for(let s of n){t.push(s);let o=Tt(s);for(let i of o)r(i.statements)}};return r(e),t}});function kt(){return{version:"1.0",entries:{}}}var jr=x(()=>{"use strict";Ye()});import{v4 as dc}from"uuid";var Ur=x(()=>{"use strict"});import{z as v}from"zod";var Fr,Pt,Hr,xe,Br,Wr,Gr,V,Kr,Je,Et,At=x(()=>{"use strict";Fr=v.enum(["JS_CODE","AI_MODE"]),Pt=v.object({type:Fr,expression:v.string()}),Hr=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),xe=v.object({uid:v.string(),type:Hr,comment:v.string().optional()}),Br=v.object({action_data:v.object({action_name:v.string(),kwargs:v.record(v.any()).optional(),args:v.array(v.any()).optional()}),action_description:v.string().optional(),url:v.string().optional(),xpath:v.string().nullable().optional(),locator:v.string().nullable().optional(),css_selector:v.string().nullable().optional(),unique_selector:v.string().nullable().optional(),element_index:v.number().nullable().optional(),frame_path:v.array(v.any()).optional(),artifacts:v.record(v.any()).optional(),feedback:v.string().optional(),original_browser_use_action:v.any().optional()}).passthrough(),Wr=xe.extend({type:v.literal("DRAFT"),description:v.string()}),Gr=xe.extend({type:v.literal("ACTION"),description:v.string(),action_entity:Br.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),V=v.lazy(()=>v.union([Wr,Gr,xe.extend({type:v.literal("STEP"),description:v.string().optional().default(""),statements:v.array(V),reference_id:v.number().optional(),template_path:v.string().optional(),template_params:v.record(v.string()).optional()}),xe.extend({type:v.literal("IF_ELSE"),description:v.string().optional(),condition:Pt,then:v.array(V),else:v.array(V).optional()}),xe.extend({type:v.literal("WHILE_LOOP"),description:v.string().optional(),condition:Pt,body:v.array(V),timeout_ms:v.number().optional()})])),Kr=v.object({name:v.string(),statements:v.array(V),teardown:v.array(V).optional(),skip:v.union([v.boolean(),v.string()]).optional(),timeout:v.number().optional(),fail:v.union([v.boolean(),v.string()]).optional(),only:v.boolean().optional(),slow:v.boolean().optional()}),Je=v.object({tests:v.array(Kr).min(1),beforeAll:v.array(V).optional(),afterAll:v.array(V).optional(),beforeEach:v.array(V).optional(),afterEach:v.array(V).optional()}),Et=v.object({comment:v.string().optional(),version:v.string().optional(),goal:v.string().optional(),url:v.string().optional(),baseURL:v.string().optional(),final_feedback:v.string().optional(),completed:v.boolean().optional(),success:v.boolean().optional(),statements:v.array(V).optional(),teardown:v.array(V).optional(),last_modified_at:v.string().optional(),testGroup:Je.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 wc,parse as Jr,parseAllDocuments as bc,parseDocument as Po,Document as Xr,isMap as ke,isSeq as z}from"yaml";import{v4 as re}from"uuid";function Xe(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(q)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(q)),r}function It(e,t){if(e.testGroup)return Zr(e,t);let r=Xe(e,t),n=new Xr(r),s=n.contents?.get("tags",!0);return z(s)&&(s.flow=!0),e.comment&&(n.commentBefore=e.comment),Vr(n,e.statements??[]),e.teardown&&Vr(n,e.teardown,"teardown"),n.toString(qr)}function Vr(e,t,r="statements"){let n=e.contents;if(!n||!ke(n))return;let s=n.get(r,!0);z(s)&&Le(s,t)}function Le(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=t[r],s=e.items[r];if(r>0&&(s.spaceBefore=!0),n.comment&&(r===0?e.commentBefore=n.comment:s.commentBefore=n.comment),ke(s)){let o=s;if(n.type==="STEP"){let i=o.get("statements",!0);z(i)&&Le(i,n.statements)}else if(n.type==="IF_ELSE"){let i=o.get("THEN",!0);z(i)&&Le(i,n.then);let a=o.get("ELSE",!0);z(a)&&n.else&&Le(a,n.else)}else if(n.type==="WHILE_LOOP"){let i=o.get("DO",!0);z(i)&&Le(i,n.body)}}}}function Zr(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let n={};t?.test_case_id!==void 0&&(n.test_case_id=t.test_case_id),t?.name&&(n.name=t.name),t?.tags&&t.tags.length>0&&(n.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(n.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(n.settings=t.settings);let s={};e.baseURL&&(s.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(s.beforeAll=r.beforeAll.map(q)),r.beforeEach&&r.beforeEach.length>0&&(s.beforeEach=r.beforeEach.map(q)),r.afterEach&&r.afterEach.length>0&&(s.afterEach=r.afterEach.map(q)),r.afterAll&&r.afterAll.length>0&&(s.afterAll=r.afterAll.map(q)),s.tests=r.tests.map(a=>{let c={name:a.name};return a.skip!==void 0&&(c.skip=a.skip),a.timeout!==void 0&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only!==void 0&&(c.only=a.only),a.slow!==void 0&&(c.slow=a.slow),c.statements=a.statements.map(q),a.teardown&&a.teardown.length>0&&(c.teardown=a.teardown.map(q)),c}),n.suite=s;let o=new Xr(n),i=o.contents?.get("tags",!0);return z(i)&&(i.flow=!0),o.toString(qr)}function q(e){switch(e.type){case"DRAFT":return Eo(e);case"ACTION":return Ao(e);case"STEP":return $o(e);case"IF_ELSE":return Io(e);case"WHILE_LOOP":return Mo(e)}}function Eo(e){return{intent:e.description}}function Ao(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&a.trim()&&e.description)return{description:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:a}}if(!e.action_entity)return{intent:e.description};let n=e.action_entity.action_data??e.action_entity.action;if(!n)return{intent:e.description};let s={intent:e.description,action:n.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),n.kwargs&&Object.keys(n.kwargs).length>0)for(let[a,c]of Object.entries(n.kwargs))a!=="uid"&&(a==="statement"&&(t==="ai_action"||t==="ai_step")||(s[a]=c));return n.args&&n.args.length>0&&(s.args=n.args),s}function $o(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(q)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Io(e){let t={IF:Qr(e.condition),THEN:e.then.map(q)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(q)),t}function Mo(e){let t={WHILE:Qr(e.condition),DO:e.body.map(q)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Qr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ce(e){try{let t=Jr(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(n=>typeof n=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function $t(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map($t);let t=e,r=Object.keys(t);if(r.length===1){let s=r[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let n={};for(let[s,o]of Object.entries(t))n[s]=$t(o);return n}function Y(e){if(e.length>zr)throw new Error(`YAML input too large (${e.length} bytes, max ${zr})`);let t=$t(Jr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Oo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:Z(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=Z(t.teardown));let n=Et.safeParse(r);if(!n.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(n.error.errors)}`);let s=n.data;return qe(e,s),s}function Oo(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:Z(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=Z(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=Z(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=Z(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=Z(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=Z(t.afterEach));let o=Je.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 Z(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Ro)}function Ro(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 Lo(t);if("WHILE"in t)return Co(t);if("STEP"in t)return No(t);if("VERIFY"in t){let r=t.VERIFY,n={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(n.code=t.js),{uid:re(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:n}}}}if("URL"in t){let r=t.URL,n=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return n&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:re(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:re(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:n}}}}}if("WAIT"in t){let r=t.WAIT,n=typeof t.seconds=="number"?t.seconds:3;return{uid:re(),type:"ACTION",description:typeof r=="string"?r:`Wait ${n}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${n}s`,action_data:{action_name:"wait",kwargs:{seconds:n}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');let n=typeof t.description=="string"?t.description:"Code block";return{uid:re(),type:"ACTION",description:n,action_entity:{action_description:n,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",n=t.js;return{uid:re(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...n}=t;return Yr({...n,action:"function",functionName:r})}if("action"in t)return Yr(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:re(),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 Lo(e){let t=en(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let n={uid:re(),type:"IF_ELSE",condition:t,then:Z(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(n.else=Z(e.ELSE)),n}function Co(e){let t=en(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let n={uid:re(),type:"WHILE_LOOP",condition:t,body:Z(r)};return typeof e.timeout_ms=="number"&&(n.timeout_ms=e.timeout_ms),n}function No(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:re(),type:"STEP",description:t,statements:Z(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let n=e.template_params,s={};for(let[o,i]of Object.entries(n))s[o]=String(i);r.template_params=s}return r}function Yr(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",n=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[l,p]of Object.entries(e))Do.has(l)||(i[l]=p);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js),(t==="ai_action"||t==="ai_step")&&(i.statement===void 0&&(i.statement=r),o&&(i.use_pure_vision=!0));let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};n&&(a.locator=n),s&&(a.xpath=s);let c={uid:re(),type:"ACTION",description:r,action_entity:a};return o&&(c.use_pure_vision=!0),c}function qe(e,t){let r;try{r=Po(e)}catch{return}let n=r.contents;if(!n||!ke(n))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=n.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let s=n,o=s.get("statements",!0);z(o)&&t.statements&&Te(o,t.statements);let i=s.get("teardown",!0);z(i)&&t.teardown&&Te(i,t.teardown)}function Te(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=e.items[r];n.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=n.commentBefore);let s=t[r];if(s.type==="STEP"&&ke(n)){let o=n.get("statements",!0);z(o)&&Te(o,s.statements)}else if(s.type==="IF_ELSE"&&ke(n)){let o=n.get("THEN",!0);z(o)&&Te(o,s.then);let i=n.get("ELSE",!0);z(i)&&s.else&&Te(i,s.else)}else if(s.type==="WHILE_LOOP"&&ke(n)){let o=n.get("DO",!0);z(o)&&Te(o,s.body)}}}var qr,zr,Do,Ze=x(()=>{"use strict";At();qr={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};zr=1024*1024;Do=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as Tc,stringify as kc}from"yaml";var tn=x(()=>{"use strict";Ze()});function an(e){let t=0,r=0;for(let n of e)if(n.type==="DRAFT")r++;else if(n.type==="ACTION"){let s=n.action_entity?.action_data?.action_name??"";on.has(s)||t++}return{action:t,draft:r}}function Uo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function sn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Fo(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,n=t.join(`
221
+ `).match(/\bawait\b/g),s=n?n.length:0;return r<=rn&&s<=nn?null:`${r} non-blank line(s), ${s} await(s) \u2014 limits are ${rn} lines and ${nn} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function Mt(e,t){let r=t?.coverageThreshold??jo,n=[],s=[],o;try{o=Y(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||n.push('Missing required field: "goal"'),o.statements?.length||n.push('Missing required field: "statements"');let i=[...ze(o.statements??[]),...o.teardown?ze(o.teardown):[]],{action:a,draft:c}=an(i),l="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 u=g.description||h.uid;n.push(`Unresolved cloud template reference on statement "${u}" (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 u=h,f=u.action_entity?.action_data?.action_name??"";if(f==="js_code"||f==="js_action"||f==="verify"||f==="ai_assert"){let w=u.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let b=Uo(w);if(b){let y=u.description||f;n.push(`Invalid JS in "${y}": ${b}. ${l}`)}else if(f==="verify"){let y=Fo(w);if(y){let m=u.description||f;n.push(`JS cache for "${m}" is too complex: ${y}`)}}}}}if(h.type==="IF_ELSE"){let u=h;if(u.condition.type==="JS_CODE"){let f=sn(u.condition.expression);f&&n.push(`Invalid JS in IF condition "${u.condition.expression}": ${f}. ${l}`)}}if(h.type==="WHILE_LOOP"){let u=h;if(u.condition.type==="JS_CODE"){let f=sn(u.condition.expression);f&&n.push(`Invalid JS in WHILE condition "${u.condition.expression}": ${f}. ${l}`)}}}let p=a+c,d=p>0?Math.round(a/p*100):0;return p>0&&d/100<r&&s.push(`Low action coverage: ${a}/${p} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:n.length===0,errors:n,warnings:s,stats:{total:p,action:a,draft:c,coverage:d}}}var jo,rn,nn,on,cn=x(()=>{"use strict";Ye();Ze();jo=.5,rn=5,nn=3,on=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var ne,Ot,ln=x(()=>{"use strict";ne=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(ne||{}),Ot=18e4});var Qe=x(()=>{"use strict"});var pn=x(()=>{"use strict";Qe()});var un=x(()=>{"use strict";Qe()});var dn,Wo,fn,hn,Ne,Go,Ko,gn=x(()=>{"use strict";dn=112,Wo=1080-dn,fn={"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"}},hn={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"]},Ne=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),hn[e].map(n=>fn[n]).filter(n=>n.defaultBrowserType&&r.includes(n.defaultBrowserType))},Go={desktop:{label:"Desktop",type:"desktop",devices:Ne("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Ne("mobile")}},Ko={desktop:{label:"Desktop",type:"desktop",devices:Ne("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Ne("mobile",!0)}}});var mn=x(()=>{"use strict"});var yn=x(()=>{"use strict"});var wn=x(()=>{"use strict"});var bn=x(()=>{"use strict"});var ge=x(()=>{"use strict";Cr();Nr();Dr();jr();Ur();tn();cn();Ye();Ze();ln();At();pn();un();Qe();gn();mn();yn();wn();bn()});import{stringify as Vo}from"yaml";import Rt from"node:path";import{createHash as pi}from"crypto";import{parse as ui,stringify as $n}from"yaml";import{readFileSync as di,existsSync as fi}from"fs";import{resolve as Ct,dirname as hi}from"path";import{parse as xn,stringify as gi}from"yaml";import{readFileSync as _i,writeFileSync as xi,mkdirSync as Ti}from"fs";import{dirname as kn}from"path";function ae(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function C(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function zo(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Yo(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 tt(e){let t=zo(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let n=Yo(e);if(n){let s=JSON.stringify(n);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:Jo.includes(t)}function qo(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!Xo.includes(t)}function _(e,t){B.set(e,t)}function Zo(e){return B.get(e)}function be(e,t,r=[]){let n=[...r];return t.locator?n.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&n.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(t.frame_path)}`),n.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...n.map(s=>` ${s},`),"});"]}function Dt(e,t){let r=t.action_data?.kwargs||{},n=[];return typeof r.relative_x=="number"&&typeof r.relative_y=="number"?(n.push(`action_data: { kwargs: { relative_x: ${r.relative_x}, relative_y: ${r.relative_y} } }`),t.locator?n.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&n.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&n.push(`frame_path: ${JSON.stringify(t.frame_path)}`)):typeof r.x=="number"&&typeof r.y=="number"&&n.push(`action_data: { kwargs: { x: ${r.x}, y: ${r.y} } }`),[`await agent.execAction("${e}", page, {`,...n.map(s=>` ${s},`),"});"]}function Sn(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):Array.isArray(e.parameterValues)?e.parameterValues.map(String):[];if(r.length===0)return`await ${t}()`;let n=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=r.map(i=>n.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 ce(e,t,r,n="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${n}.${o}`,c=Qo(i,t,a,r);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function Qo(e,t,r,n){let s=" ".repeat(t);switch(e.type){case"DRAFT":return ei(e,t,r,n);case"ACTION":return ti(e,t,r,n);case"STEP":return ri(e,t,r,n);case"IF_ELSE":return ni(e,t,r,n);case"WHILE_LOOP":return si(e,t,r,n);default:return[`${s}// Unknown statement type: ${e.type}`]}}function ei(e,t,r,n){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${r}: Skipping - no description`];if(n.noAgent)return[`${s}// ${r}: ${C(o)}`,`${s}// DRAFT: ${C(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${r}: ${C(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}, '${r}');`]}function ti(e,t,r,n){let s=" ".repeat(t),o=e.description,i=e.uid,c=n.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${s}// ${r}: Skipping - no description`];if(n.noAgent)return[`${s}// ${r}: ${C(o)}`,`${s}// DRAFT: ${C(o)} (requires agent - skipped in hook)`];let m=JSON.stringify(o),S=!!e.use_pure_vision;return[`${s}// ${r}: ${C(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, ${m}, '${r}', ${S});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let p=l.action_data?.action_name||"",d=l.action_description||"",h=Zo(p);if(!h)return[`${s}// ${r}: Unknown action: ${p}`,`${s}throw new Error(${JSON.stringify(`Unknown action: ${p}`)});`];let g={imports:n.imports,yamlDir:n.yamlDir,projectRoot:n.projectRoot},u=h(l,r,g);if(n.noAgent){if(vn(l))return[`${s}// ${r}: ${C(d)}`,`${s}// AI action: ${C(d)} (requires agent - skipped in hook)`];let m=oi(l,p,s,r);return m||[`${s}// ${r}: ${C(d)}`,...u.map(S=>`${s}${S}`)]}if(vn(l))return[`${s}// ${r}: ${C(d)}`,`${s}page = agent.agentServices.validatePage(page);`,...u.map(m=>`${s}${m}`)];let f=JSON.stringify(d),w=u.map(m=>`${s} ${m}`),b=qo(l),y=i?`'${i}'`:"undefined";return[`${s}// ${r}: ${C(d)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...w,`${s}}, ${f}, '${r}', ${y}, ${b});`]}function ri(e,t,r,n){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${C(e.description)}`);let i=ce(e.statements,t,n,r);return o.push(...i),o}function ni(e,t,r,n){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${C(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let i=ce(e.then,t+1,n,`${r}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=ce(e.else,t+1,n,`${r}.else`);o.push(...a)}return o.push(`${s}}`),o}function si(e,t,r,n){let s=" ".repeat(t),o=[];o.push(`${s}// ${r}: Loop`);let i=e.timeout_ms??Ot,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${i};`),o.push(`${s}const ${l}_check = () => {`),o.push(`${s} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${s} throw new Error('${c}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${C(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${s}while (${l}_check() && await agent.evaluate(page, ${d}, "${r}")) {`)}let p=ce(e.body,t+1,n,`${r}.body`);return o.push(...p),o.push(`${s}}`),o}function oi(e,t,r,n){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??Nt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${r}// ${n}: ${C(s)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${n}: ${C(s)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${n}: ${C(s)}`,`${r}await page.goForward();`];case"input_text":{let a=o.text||"",c=tt(e);return c?[`${r}// ${n}: ${C(s)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=tt(e);return c?[`${r}// ${n}: ${C(s)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function ii(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...En()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore,yamlDir:t?.yamlDir,projectRoot:t?.projectRoot};t?.beforeEach&&t.beforeEach.length>0&&(r.push(..._n("beforeEach",t.beforeEach,o)),r.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",c=Lt(t?.tags);for(let l of t.parameters)r.push(...nt(e,`${c}${ae(a)} [${ae(l.name)}]`,o,0,i,l.name,l.values)),r.push("")}else{let a=t?.testName||e.goal||"Generated test",c=Lt(t?.tags);r.push(...nt(e,`${c}${ae(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(..._n("afterEach",t.afterEach,o))),An(r,s),r.join(`
222
+ `)}function ai(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...En()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore,yamlDir:t?.yamlDir,projectRoot:t?.projectRoot},i=t?.testName||"Test Suite",a=Lt(t?.tags);r.push(`test.describe.serial('${a}${ae(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...et("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...et("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],d=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let h of p.parameters)r.push(...nt(p.testFlow,`${ae(p.name)} [${ae(h.name)}]`,o,1,d,h.name,h.values)),r.push("");else r.push(...nt(p.testFlow,ae(p.name),o,1,d)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...et("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...et("afterAll",e.afterAll,o,1)),r.push("});"),An(r,s),r.join(`
223
+ `)}function Lt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function rt(e){let t=new Set;function r(n){for(let s of n)switch(s.type){case ne.ACTION:{let i=s.action_entity?.action_data?.kwargs,a=[...Array.isArray(i?.args)?i.args:[],...Array.isArray(i?.parameterValues)?i.parameterValues:[]];for(let c of a)typeof c=="string"&&ci.includes(c)&&t.add(c);break}case ne.STEP:r(s.statements);break;case ne.IF_ELSE:{let o=s;r(o.then),o.else&&r(o.else);break}case ne.WHILE_LOOP:r(s.body);break}}return r(e),t}function li(e){let t=rt(e.statements??[]);if(e.teardown)for(let r of rt(e.teardown))t.add(r);return t}function jt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function nt(e,t,r,n=0,s,o,i){let a=" ".repeat(n),c=[],l=li(e),p=jt(l),d=s?.only?"test.only":"test",h=o?`, { tag: '@${ae(o)}' }`:"";if(c.push(`${a}${d}('${t}'${h}, async (${p}) => {`),s?.skip===!0?c.push(`${a} test.skip();`):typeof s?.skip=="string"&&c.push(`${a} test.skip(true, '${ae(s.skip)}');`),s?.fail===!0?c.push(`${a} test.fail();`):typeof s?.fail=="string"&&c.push(`${a} test.fail(true, '${ae(s.fail)}');`),s?.slow&&c.push(`${a} test.slow();`),s?.timeout&&c.push(`${a} test.setTimeout(${s.timeout});`),i){for(let[f,w]of Object.entries(i))c.push(`${a} agent.agentServices.saveVariable(${JSON.stringify(f)}, ${JSON.stringify(w)});`);c.push("")}let g=e.teardown&&e.teardown.length>0,u=n+1;if(g){if(c.push(`${a} try {`),e.statements&&e.statements.length>0){c.push(`${a} // Test steps`);let w=ce(e.statements,u+1,r);c.push(...w)}c.push(`${a} } finally {`),c.push(`${a} // Teardown`);let f=ce(e.teardown,u+1,r,"teardown");c.push(...f),c.push(`${a} }`)}else if(e.statements&&e.statements.length>0){c.push(`${a} // Test steps`);let f=ce(e.statements,u,r);c.push(...f)}return c.push(`${a}});`),c}function _n(e,t,r){let n=[],s=Pn(t),o=rt(s),i=jt(o);return n.push(`test.${e}(async (${i}) => {`),n.push(...ce(s,1,r,e)),n.push("});"),n}function et(e,t,r,n){let s=" ".repeat(n),o=[],i=Pn(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,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(...ce(i,n+1,c,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let c=rt(i),l=jt(c);o.push(`${s}test.${e}(async (${l}) => {`),o.push(...ce(i,n+1,r,e)),o.push(`${s}});`)}return o}function Pn(e){let r=Vo({goal:"_hook",statements:e});return Y(r).statements??[]}function En(){return["import { test, expect } from 'shiplightai/fixture';"]}function An(e,t){if(t.size>0){let r=0;for(let s=0;s<e.length;s++)e[s].startsWith("import ")&&(r=s+1);let n=Array.from(t);e.splice(r,0,...n)}}function In(e,t,r){let n={expandingPaths:new Set([Ct(t)]),depth:0,referencedPaths:new Set,basePath:r},s={...e};Array.isArray(s.statements)&&(s.statements=me(s.statements,t,n)),Array.isArray(s.teardown)&&(s.teardown=me(s.teardown,t,n));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=me(s[o],t,n));return{doc:s,referencedTemplatePaths:Array.from(n.referencedPaths)}}function me(e,t,r){let n=[];for(let s of e)if(mi(s)){let o=yi(s,t,r);n.push(o)}else n.push(wi(s,t,r));return n}function mi(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function yi(e,t,r){if(r.depth>=Tn)throw new Error(`Template expansion exceeded maximum depth of ${Tn}. Check for deeply nested or circular template references.`);let n=Ct(hi(t),e.template),s=!fi(n)&&r.basePath?Ct(r.basePath,e.template):n;if(r.expandingPaths.has(s))throw new Error(`Circular template reference detected: ${s} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${s}`);r.referencedPaths.add(s);let o;try{o=di(s,"utf-8")}catch(u){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${u.message}`)}let i=xn(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let a=i.params||[],c=e.params||{};for(let u of a)if(!(u in c))throw new Error(`Template ${e.template} requires param "${u}" but it was not provided. Required params: [${a.join(", ")}]`);let l=i.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let f=gi(l);for(let[w,b]of Object.entries(c))f=f.split(`<<${w}>>`).join(String(b));l=xn(f)}let p={expandingPaths:new Set([...r.expandingPaths,s]),depth:r.depth+1,referencedPaths:r.referencedPaths},d=me(l,s,p),g={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(c).length>0&&(g.template_params=c),g}function wi(e,t,r){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=me(n.statements,t,r)),Array.isArray(n.THEN)&&(n.THEN=me(n.THEN,t,r)),Array.isArray(n.ELSE)&&(n.ELSE=me(n.ELSE,t,r)),Array.isArray(n.DO)&&(n.DO=me(n.DO,t,r)),n}function Ft(e,t,r){let n=ui(e),s=n?.name,o=n?.tags,i=n?.use;if(n&&(n.name!==void 0||n.tags!==void 0||n.use!==void 0)&&(delete n.name,delete n.tags,delete n.use),n?.suite){if(n.goal||n.statements)throw new Ut('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return vi(n,s,o,i,t,r)}return bi(n,s,o,i,t,r)}function bi(e,t,r,n,s,o){let i=e?.beforeEach,a=e?.afterEach,c=Mn(e?.parameters),l=e?.timeout,p=e?.skip,d=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 Ut(`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 u=[];if(s&&e&&typeof e=="object"){let b=In(e,s,o);e=b.doc,u=b.referencedTemplatePaths}let f=$n(e),w=Y(f);return s&&(Pe(w.statements??[],s,"main"),w.teardown&&Pe(w.teardown,s,"teardown")),{testFlow:w,name:t,tags:r,use:n,beforeEach:i,afterEach:a,parameters:c,timeout:l,skip:p,fail:d,only:h,slow:g,referencedTemplatePaths:u}}function vi(e,t,r,n,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,c=i.afterAll,l=i.beforeEach,p=i.afterEach,d=[],h=i.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let w={goal:f.name,statements:f.statements};f.teardown&&(w.teardown=f.teardown);let b=[],y=w;if(s&&typeof w=="object"){let k=In(w,s,o);y=k.doc,b=k.referencedTemplatePaths,d.push(...b)}let m=$n(y),S=Y(m),P=Mn(f.parameters);return{testFlow:S,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:P,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),g=i.base_url,u=g?{...n,baseURL:g}:n;return{suite:{beforeAll:a,afterAll:c,beforeEach:l,afterEach:p,tests:h},name:t,tags:r,use:u,referencedTemplatePaths:d}}function Mn(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Pe(e,t,r){for(let n=0;n<e.length;n++){let s=e[n],o=`${r}.${n}`,i=s.description||"";if(s.uid=Si(t,o,i),s.type===ne.STEP)Pe(s.statements,t,o);else if(s.type===ne.IF_ELSE){let a=s;Pe(a.then,t,`${o}.then`),a.else&&Pe(a.else,t,`${o}.else`)}else s.type===ne.WHILE_LOOP&&Pe(s.body,t,`${o}.body`)}}function Si(e,t,r){let n=pi("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function On(e,t){let r;try{r=_i(e,"utf-8")}catch(n){return{valid:!1,errors:[`Failed to read file: ${n.message}`],warnings:[]}}return ki(r,e,t)}function ki(e,t,r){let n=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=n||s?null:Mt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,c=[];try{let l=r?.parsed??Ft(e,t,r?.basePath);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore,yamlDir:kn(t),projectRoot:r?.projectRoot??r?.basePath},d=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?i=ai(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):i=ii(l.testFlow,{...p,testName:l.name,tags:l.tags,use:d,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let h=i.split(`
219
224
  `).filter(g=>!g.startsWith("import ")).join(`
220
- `);new Function(h),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),_i(Tn(a),{recursive:!0}),Si(a,i)}catch(l){let p=l instanceof Ut?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var Nt,zo,Yo,H,ii,xn,Ut,Ht=x(()=>{"use strict";ge();ge();ge();ge();Nt=5e3;zo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],Yo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];H=new Map;_("click",e=>{let t=tt(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Nt;return[`await ${t}.click({ timeout: ${r} });`]});_("click_element",H.get("click"));_("click_element_by_index",H.get("click"));_("double_click",e=>be("double_click",e));_("double_click_on_element",H.get("double_click"));_("right_click",e=>be("right_click",e));_("right_click_on_element",H.get("right_click"));_("hover",e=>be("hover",e));_("hover_element_by_index",H.get("hover"));_("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return be("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});_("fill",H.get("input_text"));_("clear_input",e=>be("clear_input",e));_("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});_("send_keys",H.get("press"));_("send_keys_on_element",e=>{let t=tt(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let n=e.action_data?.kwargs?.timeout_ms??Nt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${n} });`]});_("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return be("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});_("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)})');`]});_("scroll_down",H.get("scroll"));_("scroll_up",H.get("scroll"));_("scroll_element",H.get("scroll"));_("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});_("scroll_on_element",e=>be("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));_("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)} } },`,"});"]});_("open_tab",H.get("go_to_url"));_("go_back",()=>['await agent.execAction("go_back", page, {});']);_("reload_page",()=>['await agent.execAction("reload_page", page, {});']);_("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);_("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);_("verify",(e,t)=>{let r=e.action_data?.kwargs,n=typeof r?.code=="string",s=n?r?.statement||e.action_description:e.action_description||r?.statement;if(n&&s){let i=r.code.split(`
225
+ `);new Function(h),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Ti(kn(a),{recursive:!0}),xi(a,i)}catch(l){let p=l instanceof Ut?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var Nt,Jo,Xo,B,ci,Tn,Ut,Ht=x(()=>{"use strict";ge();ge();ge();ge();Nt=5e3;Jo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],Xo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];B=new Map;_("click",e=>{let t=tt(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Nt;return[`await ${t}.click({ timeout: ${r} });`]});_("click_element",B.get("click"));_("click_element_by_index",B.get("click"));_("double_click",e=>be("double_click",e));_("double_click_on_element",B.get("double_click"));_("right_click",e=>be("right_click",e));_("right_click_on_element",B.get("right_click"));_("hover",e=>be("hover",e));_("hover_element_by_index",B.get("hover"));_("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return be("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});_("fill",B.get("input_text"));_("clear_input",e=>be("clear_input",e));_("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});_("send_keys",B.get("press"));_("send_keys_on_element",e=>{let t=tt(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let n=e.action_data?.kwargs?.timeout_ms??Nt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${n} });`]});_("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return be("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});_("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)})');`]});_("scroll_down",B.get("scroll"));_("scroll_up",B.get("scroll"));_("scroll_element",B.get("scroll"));_("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});_("scroll_on_element",e=>be("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));_("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)} } },`,"});"]});_("open_tab",B.get("go_to_url"));_("go_back",()=>['await agent.execAction("go_back", page, {});']);_("reload_page",()=>['await agent.execAction("reload_page", page, {});']);_("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);_("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);_("verify",(e,t)=>{let r=e.action_data?.kwargs,n=typeof r?.code=="string",s=n?r?.statement||e.action_description:e.action_description||r?.statement;if(n&&s){let i=r.code.split(`
221
226
  `),a=JSON.stringify(s);return["{ const _t = Date.now(); try {",...i.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${a}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${a}\`);`,` await agent.assert(page, ${a}, ${JSON.stringify(t||"")});`,"} }"]}return n?r.code.split(`
222
- `):s?[`await agent.assert(page, ${JSON.stringify(s)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});_("ai_assert",H.get("verify"));_("assert",H.get("verify"));_("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let n=JSON.stringify(r),s=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${n}, '${t||""}', ${s});`]});_("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});_("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,n=e.action_data?.kwargs?.variable_name;if(!r||!n)return["// Skipping ai_extract: missing element_description or variable_name"];let s=JSON.stringify(r),o=JSON.stringify(n);return[`await agent.extract(page, ${s}, ${o}, '${t||""}');`]});_("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,n=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${n}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});_("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});_("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],n=t.split(`
223
- `);for(let s of n)r.push(` ${s}`);return r.push("}"),r});_("function",(e,t,r)=>{let n=e.action_data?.kwargs||{},s=n.functionName;if(s&&s.includes("#")){let[i,a]=s.split("#");if(i&&a){let c=i.replace(/\.(ts|js|mjs)$/,"");if(!c.startsWith(".")&&!Rt.isAbsolute(c)&&r?.projectRoot&&r?.yamlDir){let h=Rt.resolve(r.projectRoot,c),g=Rt.relative(r.yamlDir,h).replace(/\\/g,"/");c=g.startsWith("..")?g:"./"+g}let l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let p={...n,functionName:a},f=vn(p);return f?[f.endsWith(";")?f:`${f};`]:["// Skipping function: invalid export pattern"]}}let o=vn(n);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});_("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)} } },`,"});"]});_("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],n={};return t.paths?n.paths=t.paths:t.path&&(n.path=t.path),t.use_file_input&&(n.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(n)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(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} } },`,"});"]);_("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} } },`,"});"]);_("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} } },`,"});"]});_("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(n=>` ${n},`),"});"]});_("done",()=>["// Done - no action needed"]);_("click_by_coordinates",e=>jt("click_by_coordinates",e));_("right_click_by_coordinates",e=>jt("right_click_by_coordinates",e));_("double_click_by_coordinates",e=>jt("double_click_by_coordinates",e));_("drag_drop",e=>{let t=e.action_data?.kwargs||{},r={};typeof t.relative_x=="number"&&(r.relative_x=t.relative_x),typeof t.relative_y=="number"&&(r.relative_y=t.relative_y),typeof t.delta_x=="number"&&(r.delta_x=t.delta_x),typeof t.delta_y=="number"&&(r.delta_y=t.delta_y),typeof t.coord_source_x=="number"&&(r.coord_source_x=t.coord_source_x),typeof t.coord_source_y=="number"&&(r.coord_source_y=t.coord_source_y),typeof t.coord_target_x=="number"&&(r.coord_target_x=t.coord_target_x),typeof t.coord_target_y=="number"&&(r.coord_target_y=t.coord_target_y);let n=[`action_data: { kwargs: ${JSON.stringify(r)} }`];return 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("drag_drop", page, {',...n.map(s=>` ${s},`),"});"]});_("get_dropdown_options",e=>{let t=[];return e.xpath&&t.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&t.push(`frame_path: ${JSON.stringify(e.frame_path)}`),t.length===0?['await agent.execAction("get_dropdown_options", page, {});']:['await agent.execAction("get_dropdown_options", page, {',...t.map(r=>` ${r},`),"});"]});_("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});_("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
224
- `):["// Skipping js_action: missing code"]});ii=["testContext","request"];xn=5;Ut=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as Ti}from"express";import*as E from"fs/promises";import*as T from"path";import{stringify as Ue,parse as Ee}from"yaml";function je(e){return e.map(t=>{if(t.type==="ACTION"&&t.action_entity){let r=t.action_entity,n=r.action_data??r.action,s=n?.kwargs?Object.fromEntries(Object.entries(n.kwargs).filter(([i])=>i!=="element_index")):n?.kwargs,o={...r};if(delete o.xpath,n&&s!==void 0){let i={...n,kwargs:s};r.action_data?o.action_data=i:o.action=i}return{...t,action_entity:o}}return t.type==="STEP"?{...t,statements:je(t.statements)}:t.type==="IF_ELSE"?{...t,then:je(t.then),...t.else?{else:je(t.else)}:{}}:t.type==="WHILE_LOOP"?{...t,body:je(t.body)}:t})}function ve(e){if(!e||e.length===0)return[];let r=Ue({goal:"_hook",statements:e});return Y(r).statements??[]}async function On(e){try{let t=await E.readdir(e,{withFileTypes:!0});for(let r of t)if(!(r.name==="node_modules"||r.name.startsWith("."))&&(r.isFile()&&r.name.endsWith(".test.yaml")||r.isDirectory()&&await On(T.join(e,r.name))))return!0}catch{}return!1}function st(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function ki(e){let t=new Map,r;try{r=await E.readdir(e)}catch(n){if(n.code==="ENOENT")return t;throw n}for(let n of r.filter(s=>s.endsWith(".yaml")))try{let s=T.join(e,n),i=Ee(await E.readFile(s,"utf-8"))?.name??T.basename(n,".yaml");t.set(st(i),`templates/${n}`)}catch{}return t}function Bt(e,t){if(Array.isArray(e))return e.map(o=>Bt(o,t));if(!e||typeof e!="object")return e;let r=e,n={};for(let[o,i]of Object.entries(r))n[o]=Bt(i,t);let s=typeof r.reference_id=="number"?r.reference_id:void 0;if(s!==void 0&&typeof n.template_path!="string"){let o=t.get(s);o&&(n.template_path=o,delete n.reference_id)}return n}function ot(e){return Xe({statements:e}).statements}async function De(e,t){for(let r of e)if(r.type==="STEP"){if(r.template_path&&r.statements.length>0){let n=T.join(t,r.template_path);try{let s=Ee(await E.readFile(n,"utf-8"))??{};s.statements=ot(r.statements),await E.writeFile(n,Ue(s),"utf-8")}catch{}}await De(r.statements,t)}else r.type==="IF_ELSE"?(await De(r.then,t),r.else&&await De(r.else,t)):r.type==="WHILE_LOOP"&&await De(r.body,t)}function Rn(e){let{initialDir:t,initialFile:r,projectRoot:n,onFileSelected:s}=e,o=Ti();function i(){return T.join(n??t,"fixtures")}o.get("/api/files",async(c,l)=>{try{let p=typeof c.query.dir=="string"?c.query.dir:t,f=T.resolve(p),h=await E.readdir(f,{withFileTypes:!0}),g=[];for(let d of h)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let w=T.join(f,d.name);await On(w)&&g.push({name:d.name,type:"directory",path:w})}else d.isFile()&&d.name.endsWith(".test.yaml")&&g.push({name:d.name,type:"file",path:T.join(f,d.name)});g.sort((d,w)=>d.type!==w.type?d.type==="directory"?-1:1:d.name.localeCompare(w.name));let u=T.dirname(f);l.json({dir:f,parent:u!==f?u:null,entries:g,initialFile:r??null,projectRoot:n??t})}catch(p){console.error("[debugger] Error listing files:",p),l.status(500).json({error:p.message})}});function a(c){if(typeof c=="string"&&c){let l=T.resolve(c);return l.endsWith(".test.yaml")?{filePath:l}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return o.get("/api/test-flow",async(c,l)=>{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let f=p.filePath;try{s?.(f);let h=await E.readFile(f,"utf-8"),g=await E.stat(f),u=Ce(h),d=Ft(h,f,n);if(d.suite){let w=d.suite,b={tests:w.tests.map(m=>({name:m.name,statements:m.testFlow.statements??[],teardown:m.testFlow.teardown,skip:m.skip,timeout:m.timeout,fail:m.fail,only:m.only,slow:m.slow})),beforeAll:ve(w.beforeAll),afterAll:ve(w.afterAll),beforeEach:ve(w.beforeEach),afterEach:ve(w.afterEach)},y={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:b};l.json({isSuite:!0,testFlow:y,metadata:u,name:d.name,tags:d.tags,use:d.use,filePath:f,fileName:T.basename(f),lastModified:g.mtimeMs})}else{let w=d.testFlow;qe(h,w),l.json({isSuite:!1,testFlow:w,metadata:u,name:d.name,tags:d.tags,use:d.use,parameters:d.parameters,filePath:f,fileName:T.basename(f),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return l.status(404).json({error:`File not found: ${f}`});console.error("[debugger] Error loading test flow:",h),l.status(500).json({error:h.message})}}),o.put("/api/test-flow",async(c,l)=>{try{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let f=p.filePath,{testFlow:h,metadata:g}=c.body;if(!h)return l.status(400).json({error:"testFlow is required"});let u=T.join(n??t,"templates"),d=await ki(u),w=Bt(h,d),b=je(w.statements??[]),y={...w,statements:b},m=n??t,S=[y.statements??[],y.teardown??[],...y.testGroup?.tests?.flatMap(F=>[F.statements,...F.teardown?[F.teardown]:[]])??[],...y.testGroup?.beforeEach?[y.testGroup.beforeEach]:[],...y.testGroup?.afterEach?[y.testGroup.afterEach]:[]];for(let F of S)await De(F,m);let P=It(y,g),k=f+".tmp";await E.writeFile(k,P,"utf-8"),await E.rename(k,f);let K=await E.stat(f);l.json({success:!0,lastModified:K.mtimeMs})}catch(p){console.error("[debugger] Error saving test flow:",p),l.status(500).json({error:p.message})}}),o.get("/api/fixtures",async(c,l)=>{try{let p=i();try{let h=(await E.readdir(p,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:h,dir:p})}catch(f){if(f.code==="ENOENT")l.json({files:[],dir:p});else throw f}}catch(p){console.error("[debugger] Error listing fixtures:",p),l.status(500).json({error:p.message})}}),o.post("/api/fixtures",async(c,l)=>{try{let p=i();await E.mkdir(p,{recursive:!0});let{name:f,content:h}=c.body;if(!f||!h)return l.status(400).json({error:"name and content are required"});let g=T.basename(f);if(!g)return l.status(400).json({error:"Invalid file name"});let u=T.join(p,g);await E.writeFile(u,Buffer.from(h,"base64")),l.json({fileName:g})}catch(p){console.error("[debugger] Error saving fixture:",p),l.status(500).json({error:p.message})}}),o.get("/api/functions",async(c,l)=>{let p=T.join(n??t,"helpers"),f;try{f=await E.readdir(p)}catch(d){return d.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading helpers dir:",d),l.status(500).json({error:d.message}))}let h=f.filter(d=>/\.(ts|js|mjs)$/.test(d)),g=[],u=1;for(let d of h){let w=await E.readFile(T.join(p,d),"utf-8"),b=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,y;for(;(y=b.exec(w))!==null;){let[,m,S]=y,P=S.replace(/\s*:\s*[^,)]+/g,"");g.push({id:u++,name:`helpers/${d}#${m}`,description:"",status:"Active",code:`async function ${m}${P} {}`})}}l.json(g)}),o.get("/api/reusable-steps",async(c,l)=>{let p=T.join(n??t,"templates"),f;try{f=await E.readdir(p)}catch(u){return u.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading templates dir:",u),l.status(500).json({error:u.message}))}let h=f.filter(u=>u.endsWith(".yaml")).sort(),g=[];for(let u of h)try{let d=await E.readFile(T.join(p,u),"utf-8"),w=Ee(d);if(!w||typeof w!="object")continue;let b=w.name??T.basename(u,".yaml"),y=w.description??"",m=ve(w.statements);g.push({id:st(b),organizationId:"local",name:b,description:y,statements:m})}catch(d){console.error(`[debugger] Error parsing template ${u}:`,d)}l.json(g)}),o.post("/api/reusable-steps/exists/:name",async(c,l)=>{let p=T.join(n??t,"templates"),f=decodeURIComponent(c.params.name).toLowerCase(),h;try{h=await E.readdir(p)}catch(g){return g.code==="ENOENT"?l.json(!1):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=await E.readFile(T.join(p,g),"utf-8");if((Ee(u)?.name??T.basename(g,".yaml")).toLowerCase()===f)return l.json(!0)}catch{}l.json(!1)}),o.post("/api/reusable-steps",async(c,l)=>{let p=T.join(n??t,"templates"),{reusableStep:f}=c.body??{};if(!f?.name||!Array.isArray(f.statements))return l.status(400).json({error:"reusableStep.name and reusableStep.statements are required"});await E.mkdir(p,{recursive:!0});let h=f.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=T.join(p,`${h}.yaml`),u={name:f.name,statements:ot(f.statements)};f.description&&(u.description=f.description),await E.writeFile(g,Ue(u),"utf-8"),l.json({id:st(f.name),organizationId:"local",name:f.name,description:f.description??"",statements:f.statements})}),o.put("/api/reusable-steps/:id/update",async(c,l)=>{let p=parseInt(c.params.id,10),f=T.join(n??t,"templates"),h;try{h=await E.readdir(f)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=T.join(f,g),d=Ee(await E.readFile(u,"utf-8"))??{},w=d.name??T.basename(g,".yaml");if(st(w)!==p)continue;let b=c.body??{};return b.description!==void 0&&(d.description=b.description),b.statements!==void 0&&(d.statements=ot(b.statements)),await E.writeFile(u,Ue(d),"utf-8"),l.json({id:p,organizationId:"local",...d,statements:b.statements!==void 0?b.statements:ve(d.statements)})}catch{}l.status(404).json({error:`Template with id ${p} not found`})}),o.put("/api/templates/:name",async(c,l)=>{let p=T.join(n??t,"templates"),f=decodeURIComponent(c.params.name),h;try{h=await E.readdir(p)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=T.join(p,g),d=Ee(await E.readFile(u,"utf-8"))??{};if((d.name??T.basename(g,".yaml"))!==f)continue;let b=c.body??{};return b.name!==void 0&&(d.name=b.name),b.description!==void 0&&(d.description=b.description),b.statements!==void 0&&(d.statements=ot(b.statements)),await E.writeFile(u,Ue(d),"utf-8"),l.json({organizationId:"local",...d,statements:b.statements!==void 0?b.statements:ve(d.statements)})}catch{}l.status(404).json({error:`Template "${f}" not found`})}),o.get("/api/test-results",async(c,l)=>{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let f=n??t,h=T.basename(p.filePath,".test.yaml"),g=T.join(f,".shiplight","artifacts",h),u=[],d=null;try{let w=await E.readdir(g,{withFileTypes:!0});for(let b of w)if(b.isDirectory()){let y=T.join(g,b.name),m=await E.readdir(y);for(let S of m.filter(P=>/\.(png|jpe?g|webp)$/i.test(P))){let P=T.relative(g,T.join(y,S)),k=b.name.match(/^(.+?)_(before|after)$/),K=k?k[1]:b.name,F=k?k[2]:"screenshot";u.push({url:`/api/report-assets/${h}/${P}`,label:F,stepId:K})}}else/\.(webm|mp4)$/i.test(b.name)&&(d||(d=`/api/report-assets/${h}/${b.name}`));u.sort((b,y)=>b.stepId!==y.stepId?(b.stepId??"").localeCompare(y.stepId??""):b.label.localeCompare(y.label))}catch{}if(u.length===0&&!d)return l.status(404).json({error:"No test results found"});l.json({videoPath:d,screenshots:u})}),o}var Ln=x(()=>{"use strict";ge();Ht()});import*as Dn from"http";import*as Un from"net";import*as le from"path";import*as Fn from"fs";import{randomUUID as Pi}from"crypto";function Nn(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let n=[];e.on("data",s=>n.push(s)),e.on("end",()=>t(Buffer.concat(n))),e.on("error",r)})}function jn(e,t,r,n,s,o){return new Promise(i=>{let a=Array.isArray(o)?o[0]:o,c=Dn.request({hostname:r,port:n,path:e.originalUrl,method:e.method,headers:{"content-type":a||"application/json","content-length":String(s.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",p=>t.write(p)),l.on("end",()=>{t.end(),i()}),l.on("error",()=>{t.writableEnded||t.end(),i()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),i()}),c.end(s)})}function Wt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var Cn,it,Hn=x(()=>{"use strict";xt();Cn=1e4,it=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??_t,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}resetToIdle(t,r){t.session.innerPort=0,t.session.innerHost="127.0.0.1",t.session.pid=0,t.session.status="idle",t.session.exitInfo=r,t.cleanup=async()=>{},t.readyPromise=Promise.resolve(),this.notifyStateChange(t.session)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}openSession(t){let r=le.resolve(t),n=this.byYamlPath.get(r);if(n){let l=this.sessions.get(n);if(l&&l.session.status!=="ended")return{...l.session};this.byYamlPath.delete(r)}if(!Fn.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!Re(le.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Pi().slice(0,8)}`,i=new Date().toISOString(),a={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:i,status:"idle",exitInfo:null},c={session:a,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,c),this.byYamlPath.set(r,o),this.notifyStateChange(a),this.log(`[manager] session ${o} created (idle) for ${le.basename(r)}`),{...a}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let n=r.session.yamlPath,s=Re(le.dirname(n));if(!s)throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);r.session.status="starting",this.notifyStateChange(r.session);let o=(async()=>{let i=Cn+18e4,a,c=new Promise((p,f)=>{a=setTimeout(()=>f(new Error(`Timed out after ${i/1e3}s waiting for inner playwright to register on a port.`)),i)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),c])}finally{a&&clearTimeout(a)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup,r.livenessTimer=this.startLivenessProbe(t,p=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,p)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${le.basename(n)})`)})();r.readyPromise=o;try{await o}catch(i){throw this.resetToIdle(r,i.message),i}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(n){this.log(`[manager] stopSandbox ${t} cleanup error: ${n.message}`)}this.resetToIdle(r,null)}}startLivenessProbe(t,r){let o=0,i=setInterval(()=>{let a=this.sessions.get(t);if(!a||a.session.status==="ended"||a.session.status==="idle"){clearInterval(i);return}let c=!1;try{process.kill(a.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?o=0:(o+=1,o>=3&&(clearInterval(i),r("process-exited")))},3e3);return i.unref(),i}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(n){this.log(`[manager] closeSession ${t} cleanup error: ${n.message}`)}}}async restartInner(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to restart`);if(r.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let n=r.session.yamlPath,s=Re(le.dirname(n));if(!s)throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);r.restartInProgress=!0;try{r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(l){this.log(`[manager] restartInner ${t} old cleanup error: ${l.message}`)}r.session.status="starting",r.session.innerPort=0,r.session.pid=0,this.notifyStateChange(r.session);let o=Cn+18e4,i,a=new Promise((l,p)=>{i=setTimeout(()=>p(new Error(`Timed out after ${o/1e3}s waiting for inner playwright to respawn.`)),o)}),c;try{c=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),a])}catch(l){throw this.resetToIdle(r,l.message),l}finally{i&&clearTimeout(i)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup,r.livenessTimer=this.startLivenessProbe(t,l=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,l)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} restarted on ${r.session.innerHost}:${r.session.innerPort} (yaml=${le.basename(n)})`)}finally{r.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:n}=r;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 c=`${s.method} ${s.path}`;if(c==="POST /api/int-runner/create-session"){let p=await Nn(s),f={};if(p.length)try{f=JSON.parse(p.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}f.testFilePath=a.session.yamlPath,await jn(s,o,a.session.innerHost,a.session.innerPort,Buffer.from(JSON.stringify(f),"utf-8"),"application/json");return}if(c==="POST /api/int-runner/terminate-session"){try{await this.stopSandbox(t),o.json({status:"success",details:"Sandbox stopped"})}catch(p){o.status(500).json({status:"error",message:p.message})}return}if(c==="POST /api/int-runner/liveview-url"){let p=n?.(s)??"";o.json({liveviewUrl:p,browserWsUrl:""});return}let l=await Nn(s);await jn(s,o,a.session.innerHost,a.session.innerPort,l,s.headers["content-type"])}}wsUpgradeFor(t){return async(r,n,s,o)=>{let i=this.sessions.get(t);if(!i){Wt(n,`HTTP/1.1 404 Not Found\r
227
+ `):s?[`await agent.assert(page, ${JSON.stringify(s)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});_("ai_assert",B.get("verify"));_("assert",B.get("verify"));_("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let n=JSON.stringify(r),s=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${n}, '${t||""}', ${s});`]});_("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});_("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,n=e.action_data?.kwargs?.variable_name;if(!r||!n)return["// Skipping ai_extract: missing element_description or variable_name"];let s=JSON.stringify(r),o=JSON.stringify(n);return[`await agent.extract(page, ${s}, ${o}, '${t||""}');`]});_("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,n=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${n}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});_("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});_("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],n=t.split(`
228
+ `);for(let s of n)r.push(` ${s}`);return r.push("}"),r});_("function",(e,t,r)=>{let n=e.action_data?.kwargs||{},s=n.functionName;if(s&&s.includes("#")){let[i,a]=s.split("#");if(i&&a){let c=i.replace(/\.(ts|js|mjs)$/,"");if(!c.startsWith(".")&&!Rt.isAbsolute(c)&&r?.projectRoot&&r?.yamlDir){let h=Rt.resolve(r.projectRoot,c),g=Rt.relative(r.yamlDir,h).replace(/\\/g,"/");c=g.startsWith("..")?g:"./"+g}let l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let p={...n,functionName:a},d=Sn(p);return d?[d.endsWith(";")?d:`${d};`]:["// Skipping function: invalid export pattern"]}}let o=Sn(n);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});_("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)} } },`,"});"]});_("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],n={};return t.paths?n.paths=t.paths:t.path&&(n.path=t.path),t.use_file_input&&(n.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(n)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(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} } },`,"});"]);_("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} } },`,"});"]);_("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} } },`,"});"]});_("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(n=>` ${n},`),"});"]});_("done",()=>["// Done - no action needed"]);_("click_by_coordinates",e=>Dt("click_by_coordinates",e));_("right_click_by_coordinates",e=>Dt("right_click_by_coordinates",e));_("double_click_by_coordinates",e=>Dt("double_click_by_coordinates",e));_("drag_drop",e=>{let t=e.action_data?.kwargs||{},r={};typeof t.relative_x=="number"&&(r.relative_x=t.relative_x),typeof t.relative_y=="number"&&(r.relative_y=t.relative_y),typeof t.delta_x=="number"&&(r.delta_x=t.delta_x),typeof t.delta_y=="number"&&(r.delta_y=t.delta_y),typeof t.coord_source_x=="number"&&(r.coord_source_x=t.coord_source_x),typeof t.coord_source_y=="number"&&(r.coord_source_y=t.coord_source_y),typeof t.coord_target_x=="number"&&(r.coord_target_x=t.coord_target_x),typeof t.coord_target_y=="number"&&(r.coord_target_y=t.coord_target_y);let n=[`action_data: { kwargs: ${JSON.stringify(r)} }`];return 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("drag_drop", page, {',...n.map(s=>` ${s},`),"});"]});_("get_dropdown_options",e=>{let t=[];return e.xpath&&t.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&t.push(`frame_path: ${JSON.stringify(e.frame_path)}`),t.length===0?['await agent.execAction("get_dropdown_options", page, {});']:['await agent.execAction("get_dropdown_options", page, {',...t.map(r=>` ${r},`),"});"]});_("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});_("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
229
+ `):["// Skipping js_action: missing code"]});ci=["testContext","request"];Tn=5;Ut=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as Pi}from"express";import*as E from"fs/promises";import*as T from"path";import{stringify as Ue,parse as Ee}from"yaml";function De(e){return e.map(t=>{if(t.type==="ACTION"&&t.action_entity){let r=t.action_entity,n=r.action_data??r.action,s=n?.kwargs?Object.fromEntries(Object.entries(n.kwargs).filter(([i])=>i!=="element_index")):n?.kwargs,o={...r};if(delete o.xpath,n&&s!==void 0){let i={...n,kwargs:s};r.action_data?o.action_data=i:o.action=i}return{...t,action_entity:o}}return t.type==="STEP"?{...t,statements:De(t.statements)}:t.type==="IF_ELSE"?{...t,then:De(t.then),...t.else?{else:De(t.else)}:{}}:t.type==="WHILE_LOOP"?{...t,body:De(t.body)}:t})}function ve(e){if(!e||e.length===0)return[];let r=Ue({goal:"_hook",statements:e});return Y(r).statements??[]}async function Rn(e){try{let t=await E.readdir(e,{withFileTypes:!0});for(let r of t)if(!(r.name==="node_modules"||r.name.startsWith("."))&&(r.isFile()&&r.name.endsWith(".test.yaml")||r.isDirectory()&&await Rn(T.join(e,r.name))))return!0}catch{}return!1}function st(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function Ei(e){let t=new Map,r;try{r=await E.readdir(e)}catch(n){if(n.code==="ENOENT")return t;throw n}for(let n of r.filter(s=>s.endsWith(".yaml")))try{let s=T.join(e,n),i=Ee(await E.readFile(s,"utf-8"))?.name??T.basename(n,".yaml");t.set(st(i),`templates/${n}`)}catch{}return t}function Bt(e,t){if(Array.isArray(e))return e.map(o=>Bt(o,t));if(!e||typeof e!="object")return e;let r=e,n={};for(let[o,i]of Object.entries(r))n[o]=Bt(i,t);let s=typeof r.reference_id=="number"?r.reference_id:void 0;if(s!==void 0&&typeof n.template_path!="string"){let o=t.get(s);o&&(n.template_path=o,delete n.reference_id)}return n}function ot(e){return Xe({statements:e}).statements}async function je(e,t){for(let r of e)if(r.type==="STEP"){if(r.template_path&&r.statements.length>0){let n=T.join(t,r.template_path);try{let s=Ee(await E.readFile(n,"utf-8"))??{};s.statements=ot(r.statements),await E.writeFile(n,Ue(s),"utf-8")}catch{}}await je(r.statements,t)}else r.type==="IF_ELSE"?(await je(r.then,t),r.else&&await je(r.else,t)):r.type==="WHILE_LOOP"&&await je(r.body,t)}function Ln(e){let{initialDir:t,initialFile:r,projectRoot:n,onFileSelected:s}=e,o=Pi();function i(){return T.join(n??t,"fixtures")}o.get("/api/files",async(c,l)=>{try{let p=typeof c.query.dir=="string"?c.query.dir:t,d=T.resolve(p),h=await E.readdir(d,{withFileTypes:!0}),g=[];for(let f of h)if(f.name!=="node_modules"&&!f.name.startsWith("."))if(f.isDirectory()){let w=T.join(d,f.name);await Rn(w)&&g.push({name:f.name,type:"directory",path:w})}else f.isFile()&&f.name.endsWith(".test.yaml")&&g.push({name:f.name,type:"file",path:T.join(d,f.name)});g.sort((f,w)=>f.type!==w.type?f.type==="directory"?-1:1:f.name.localeCompare(w.name));let u=T.dirname(d);l.json({dir:d,parent:u!==d?u:null,entries:g,initialFile:r??null,projectRoot:n??t})}catch(p){console.error("[debugger] Error listing files:",p),l.status(500).json({error:p.message})}});function a(c){if(typeof c=="string"&&c){let l=T.resolve(c);return l.endsWith(".test.yaml")?{filePath:l}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return o.get("/api/test-flow",async(c,l)=>{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let d=p.filePath;try{s?.(d);let h=await E.readFile(d,"utf-8"),g=await E.stat(d),u=Ce(h),f=Ft(h,d,n);if(f.suite){let w=f.suite,b={tests:w.tests.map(m=>({name:m.name,statements:m.testFlow.statements??[],teardown:m.testFlow.teardown,skip:m.skip,timeout:m.timeout,fail:m.fail,only:m.only,slow:m.slow})),beforeAll:ve(w.beforeAll),afterAll:ve(w.afterAll),beforeEach:ve(w.beforeEach),afterEach:ve(w.afterEach)},y={version:"1.3.0",baseURL:f.use?.baseURL,testGroup:b};l.json({isSuite:!0,testFlow:y,metadata:u,name:f.name,tags:f.tags,use:f.use,filePath:d,fileName:T.basename(d),lastModified:g.mtimeMs})}else{let w=f.testFlow;qe(h,w),l.json({isSuite:!1,testFlow:w,metadata:u,name:f.name,tags:f.tags,use:f.use,parameters:f.parameters,filePath:d,fileName:T.basename(d),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return l.status(404).json({error:`File not found: ${d}`});console.error("[debugger] Error loading test flow:",h),l.status(500).json({error:h.message})}}),o.put("/api/test-flow",async(c,l)=>{try{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let d=p.filePath,{testFlow:h,metadata:g}=c.body;if(!h)return l.status(400).json({error:"testFlow is required"});let u=T.join(n??t,"templates"),f=await Ei(u),w=Bt(h,f),b=De(w.statements??[]),y={...w,statements:b},m=n??t,S=[y.statements??[],y.teardown??[],...y.testGroup?.tests?.flatMap(H=>[H.statements,...H.teardown?[H.teardown]:[]])??[],...y.testGroup?.beforeEach?[y.testGroup.beforeEach]:[],...y.testGroup?.afterEach?[y.testGroup.afterEach]:[]];for(let H of S)await je(H,m);let P=It(y,g),k=d+".tmp";await E.writeFile(k,P,"utf-8"),await E.rename(k,d);let K=await E.stat(d);l.json({success:!0,lastModified:K.mtimeMs})}catch(p){console.error("[debugger] Error saving test flow:",p),l.status(500).json({error:p.message})}}),o.get("/api/fixtures",async(c,l)=>{try{let p=i();try{let h=(await E.readdir(p,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:h,dir:p})}catch(d){if(d.code==="ENOENT")l.json({files:[],dir:p});else throw d}}catch(p){console.error("[debugger] Error listing fixtures:",p),l.status(500).json({error:p.message})}}),o.post("/api/fixtures",async(c,l)=>{try{let p=i();await E.mkdir(p,{recursive:!0});let{name:d,content:h}=c.body;if(!d||!h)return l.status(400).json({error:"name and content are required"});let g=T.basename(d);if(!g)return l.status(400).json({error:"Invalid file name"});let u=T.join(p,g);await E.writeFile(u,Buffer.from(h,"base64")),l.json({fileName:g})}catch(p){console.error("[debugger] Error saving fixture:",p),l.status(500).json({error:p.message})}}),o.get("/api/functions",async(c,l)=>{let p=T.join(n??t,"helpers"),d;try{d=await E.readdir(p)}catch(f){return f.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading helpers dir:",f),l.status(500).json({error:f.message}))}let h=d.filter(f=>/\.(ts|js|mjs)$/.test(f)),g=[],u=1;for(let f of h){let w=await E.readFile(T.join(p,f),"utf-8"),b=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,y;for(;(y=b.exec(w))!==null;){let[,m,S]=y,P=S.replace(/\s*:\s*[^,)]+/g,"");g.push({id:u++,name:`helpers/${f}#${m}`,description:"",status:"Active",code:`async function ${m}${P} {}`})}}l.json(g)}),o.get("/api/reusable-steps",async(c,l)=>{let p=T.join(n??t,"templates"),d;try{d=await E.readdir(p)}catch(u){return u.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading templates dir:",u),l.status(500).json({error:u.message}))}let h=d.filter(u=>u.endsWith(".yaml")).sort(),g=[];for(let u of h)try{let f=await E.readFile(T.join(p,u),"utf-8"),w=Ee(f);if(!w||typeof w!="object")continue;let b=w.name??T.basename(u,".yaml"),y=w.description??"",m=ve(w.statements);g.push({id:st(b),organizationId:"local",name:b,description:y,statements:m})}catch(f){console.error(`[debugger] Error parsing template ${u}:`,f)}l.json(g)}),o.post("/api/reusable-steps/exists/:name",async(c,l)=>{let p=T.join(n??t,"templates"),d=decodeURIComponent(c.params.name).toLowerCase(),h;try{h=await E.readdir(p)}catch(g){return g.code==="ENOENT"?l.json(!1):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=await E.readFile(T.join(p,g),"utf-8");if((Ee(u)?.name??T.basename(g,".yaml")).toLowerCase()===d)return l.json(!0)}catch{}l.json(!1)}),o.post("/api/reusable-steps",async(c,l)=>{let p=T.join(n??t,"templates"),{reusableStep:d}=c.body??{};if(!d?.name||!Array.isArray(d.statements))return l.status(400).json({error:"reusableStep.name and reusableStep.statements are required"});await E.mkdir(p,{recursive:!0});let h=d.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=T.join(p,`${h}.yaml`),u={name:d.name,statements:ot(d.statements)};d.description&&(u.description=d.description),await E.writeFile(g,Ue(u),"utf-8"),l.json({id:st(d.name),organizationId:"local",name:d.name,description:d.description??"",statements:d.statements})}),o.put("/api/reusable-steps/:id/update",async(c,l)=>{let p=parseInt(c.params.id,10),d=T.join(n??t,"templates"),h;try{h=await E.readdir(d)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=T.join(d,g),f=Ee(await E.readFile(u,"utf-8"))??{},w=f.name??T.basename(g,".yaml");if(st(w)!==p)continue;let b=c.body??{};return b.description!==void 0&&(f.description=b.description),b.statements!==void 0&&(f.statements=ot(b.statements)),await E.writeFile(u,Ue(f),"utf-8"),l.json({id:p,organizationId:"local",...f,statements:b.statements!==void 0?b.statements:ve(f.statements)})}catch{}l.status(404).json({error:`Template with id ${p} not found`})}),o.put("/api/templates/:name",async(c,l)=>{let p=T.join(n??t,"templates"),d=decodeURIComponent(c.params.name),h;try{h=await E.readdir(p)}catch(g){return g.code==="ENOENT"?l.status(404).json({error:"Templates directory not found"}):l.status(500).json({error:g.message})}for(let g of h.filter(u=>u.endsWith(".yaml")))try{let u=T.join(p,g),f=Ee(await E.readFile(u,"utf-8"))??{};if((f.name??T.basename(g,".yaml"))!==d)continue;let b=c.body??{};return b.name!==void 0&&(f.name=b.name),b.description!==void 0&&(f.description=b.description),b.statements!==void 0&&(f.statements=ot(b.statements)),await E.writeFile(u,Ue(f),"utf-8"),l.json({organizationId:"local",...f,statements:b.statements!==void 0?b.statements:ve(f.statements)})}catch{}l.status(404).json({error:`Template "${d}" not found`})}),o.get("/api/test-results",async(c,l)=>{let p=a(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let d=n??t,h=T.basename(p.filePath,".test.yaml"),g=T.join(d,".shiplight","artifacts",h),u=[],f=null;try{let w=await E.readdir(g,{withFileTypes:!0});for(let b of w)if(b.isDirectory()){let y=T.join(g,b.name),m=await E.readdir(y);for(let S of m.filter(P=>/\.(png|jpe?g|webp)$/i.test(P))){let P=T.relative(g,T.join(y,S)),k=b.name.match(/^(.+?)_(before|after)$/),K=k?k[1]:b.name,H=k?k[2]:"screenshot";u.push({url:`/api/report-assets/${h}/${P}`,label:H,stepId:K})}}else/\.(webm|mp4)$/i.test(b.name)&&(f||(f=`/api/report-assets/${h}/${b.name}`));u.sort((b,y)=>b.stepId!==y.stepId?(b.stepId??"").localeCompare(y.stepId??""):b.label.localeCompare(y.label))}catch{}if(u.length===0&&!f)return l.status(404).json({error:"No test results found"});l.json({videoPath:f,screenshots:u})}),o}var Cn=x(()=>{"use strict";ge();Ht()});import*as Un from"http";import*as Fn from"net";import*as le from"path";import*as Hn from"fs";import{randomUUID as Ai}from"crypto";function Dn(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let n=[];e.on("data",s=>n.push(s)),e.on("end",()=>t(Buffer.concat(n))),e.on("error",r)})}function jn(e,t,r,n,s,o){return new Promise(i=>{let a=Array.isArray(o)?o[0]:o,c=Un.request({hostname:r,port:n,path:e.originalUrl,method:e.method,headers:{"content-type":a||"application/json","content-length":String(s.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",p=>t.write(p)),l.on("end",()=>{t.end(),i()}),l.on("error",()=>{t.writableEnded||t.end(),i()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),i()}),c.end(s)})}function Wt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var Nn,it,Bn=x(()=>{"use strict";xt();Nn=1e4,it=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??_t,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}resetToIdle(t,r){t.session.innerPort=0,t.session.innerHost="127.0.0.1",t.session.pid=0,t.session.status="idle",t.session.exitInfo=r,t.cleanup=async()=>{},t.readyPromise=Promise.resolve(),this.notifyStateChange(t.session)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}openSession(t){let r=le.resolve(t),n=this.byYamlPath.get(r);if(n){let l=this.sessions.get(n);if(l&&l.session.status!=="ended")return{...l.session};this.byYamlPath.delete(r)}if(!Hn.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!Re(le.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${Ai().slice(0,8)}`,i=new Date().toISOString(),a={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:i,status:"idle",exitInfo:null},c={session:a,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,c),this.byYamlPath.set(r,o),this.notifyStateChange(a),this.log(`[manager] session ${o} created (idle) for ${le.basename(r)}`),{...a}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let n=r.session.yamlPath,s=Re(le.dirname(n));if(!s)throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);r.session.status="starting",this.notifyStateChange(r.session);let o=(async()=>{let i=Nn+18e4,a,c=new Promise((p,d)=>{a=setTimeout(()=>d(new Error(`Timed out after ${i/1e3}s waiting for inner playwright to register on a port.`)),i)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),c])}finally{a&&clearTimeout(a)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup,r.livenessTimer=this.startLivenessProbe(t,p=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,p)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${le.basename(n)})`)})();r.readyPromise=o;try{await o}catch(i){throw this.resetToIdle(r,i.message),i}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(n){this.log(`[manager] stopSandbox ${t} cleanup error: ${n.message}`)}this.resetToIdle(r,null)}}startLivenessProbe(t,r){let o=0,i=setInterval(()=>{let a=this.sessions.get(t);if(!a||a.session.status==="ended"||a.session.status==="idle"){clearInterval(i);return}let c=!1;try{process.kill(a.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?o=0:(o+=1,o>=3&&(clearInterval(i),r("process-exited")))},3e3);return i.unref(),i}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(n){this.log(`[manager] closeSession ${t} cleanup error: ${n.message}`)}}}async restartInner(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to restart`);if(r.restartInProgress)throw new Error(`Restart already in progress for session ${t}`);let n=r.session.yamlPath,s=Re(le.dirname(n));if(!s)throw new Error(`No Playwright config found for ${n} (searched parents for playwright.config.{ts,js,mjs}).`);r.restartInProgress=!0;try{r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(l){this.log(`[manager] restartInner ${t} old cleanup error: ${l.message}`)}r.session.status="starting",r.session.innerPort=0,r.session.pid=0,this.notifyStateChange(r.session);let o=Nn+18e4,i,a=new Promise((l,p)=>{i=setTimeout(()=>p(new Error(`Timed out after ${o/1e3}s waiting for inner playwright to respawn.`)),o)}),c;try{c=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),a])}catch(l){throw this.resetToIdle(r,l.message),l}finally{i&&clearTimeout(i)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup,r.livenessTimer=this.startLivenessProbe(t,l=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,l)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} restarted on ${r.session.innerHost}:${r.session.innerPort} (yaml=${le.basename(n)})`)}finally{r.restartInProgress=!1}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:n}=r;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 c=`${s.method} ${s.path}`;if(c==="POST /api/int-runner/create-session"){let p=await Dn(s),d={};if(p.length)try{d=JSON.parse(p.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}d.testFilePath=a.session.yamlPath,await jn(s,o,a.session.innerHost,a.session.innerPort,Buffer.from(JSON.stringify(d),"utf-8"),"application/json");return}if(c==="POST /api/int-runner/terminate-session"){try{await this.stopSandbox(t),o.json({status:"success",details:"Sandbox stopped"})}catch(p){o.status(500).json({status:"error",message:p.message})}return}if(c==="POST /api/int-runner/liveview-url"){let p=n?.(s)??"";o.json({liveviewUrl:p,browserWsUrl:""});return}let l=await Dn(s);await jn(s,o,a.session.innerHost,a.session.innerPort,l,s.headers["content-type"])}}wsUpgradeFor(t){return async(r,n,s,o)=>{let i=this.sessions.get(t);if(!i){Wt(n,`HTTP/1.1 404 Not Found\r
225
230
  \r
226
231
  `);return}if(i.session.status==="ended"){Wt(n,`HTTP/1.1 410 Gone\r
227
232
  \r
228
- `);return}try{let a=i.session.innerHost.includes(":")?`[${i.session.innerHost}]`:i.session.innerHost,c=await fetch(`http://${a}:${i.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),p=new URL(l.replace(/^ws/,"http")),f=parseInt(p.port||"80",10),h=p.hostname,g=p.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let u=Un.createConnection(f,h);u.on("connect",()=>{let d=`GET ${g} HTTP/1.1\r
229
- Host: ${h}:${f}\r
230
- `;for(let[w,b]of Object.entries(r.headers)){let y=w.toLowerCase();y!=="host"&&y!=="origin"&&(d+=`${w}: ${Array.isArray(b)?b.join(", "):b}\r
231
- `)}d+=`\r
232
- `,u.write(d),s.length&&u.write(s),u.pipe(n),n.pipe(u)}),u.on("error",()=>{(!("destroyed"in n)||!n.destroyed)&&n.destroy()}),n.on("error",()=>{u.destroyed||u.destroy()}),n.on("close",()=>{u.destroyed||u.destroy()})}catch(a){this.log(`[manager] WS upgrade for ${t} failed: ${a.message}`),Wt(n,`HTTP/1.1 502 Bad Gateway\r
233
+ `);return}try{let a=i.session.innerHost.includes(":")?`[${i.session.innerHost}]`:i.session.innerHost,c=await fetch(`http://${a}:${i.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),p=new URL(l.replace(/^ws/,"http")),d=parseInt(p.port||"80",10),h=p.hostname,g=p.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let u=Fn.createConnection(d,h);u.on("connect",()=>{let f=`GET ${g} HTTP/1.1\r
234
+ Host: ${h}:${d}\r
235
+ `;for(let[w,b]of Object.entries(r.headers)){let y=w.toLowerCase();y!=="host"&&y!=="origin"&&(f+=`${w}: ${Array.isArray(b)?b.join(", "):b}\r
236
+ `)}f+=`\r
237
+ `,u.write(f),s.length&&u.write(s),u.pipe(n),n.pipe(u)}),u.on("error",()=>{(!("destroyed"in n)||!n.destroyed)&&n.destroy()}),n.on("error",()=>{u.destroyed||u.destroy()}),n.on("close",()=>{u.destroyed||u.destroy()})}catch(a){this.log(`[manager] WS upgrade for ${t} failed: ${a.message}`),Wt(n,`HTTP/1.1 502 Bad Gateway\r
233
238
  \r
234
- `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});function Bn(e){return e.startsWith(Ei)}function Se(e,t){let r=t?.trim();return r?Ii(r):Bn(e)?Ai:$i}function Ii(e){return e.endsWith("/")?e.slice(0,-1):e}var Ei,Ai,$i,Gt=x(()=>{"use strict";Ei="shp_",Ai="https://nova-api.shiplight.ai",$i="https://api.shiplight.ai"});import{Router as Mi}from"express";function Wn(){let e=Mi();return e.get("/api/email-forwarding/addresses",async(t,r)=>{let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.error("[email-forwarding] SHIPLIGHT_API_TOKEN not set, skipping"),r.json({addresses:[],configured:!1});return}let s=Se(n,process.env.SHIPLIGHT_API_URL),o=`${s}/email-forwarding/addresses`;try{let i=await fetch(o,{headers:{Authorization:`Bearer ${n}`},signal:AbortSignal.timeout(1e4)});if(!i.ok){let l=(await i.json().catch(()=>({}))).error??`${i.status} ${i.statusText}`;console.error(`[email-forwarding] upstream ${i.status}: ${l}`),r.status(i.status).json({error:l});return}let a=await i.json();r.json({...a,configured:!0})}catch(i){let a=i instanceof Error?i.message:String(i);console.error(`[email-forwarding] fetch error: ${a}`),r.status(502).json({error:`Failed to reach Shiplight API (${s}): ${a}`})}}),e}var Gn=x(()=>{"use strict";Gt()});import{Router as Oi}from"express";import Kt from"express";import*as Ae from"fs";import*as pe from"path";function Ri(e){return e?pe.isAbsolute(e)?pe.normalize(e):pe.resolve(e):null}function Li(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function Ci(e){let t=pe.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Kn(e){let{manager:t,staticDir:r,resolveYamlPath:n=Ri,liveviewUrlBuilder:s=Li,fallbackRouter:o,artifactsDir:i}=e,a=Oi(),c=i?Kt.static(i):null;a.post("/api/debugger/sessions",Kt.json(),async(p,f)=>{let h=p.body?.yamlPath;if(typeof h!="string"||!h){f.status(400).json({error:"yamlPath is required"});return}let g=n(h);if(!g){f.status(403).json({error:"Path outside allowed roots"});return}if(!Ci(g)){f.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!Ae.existsSync(g)){f.status(404).json({error:"File not found"});return}let u=t.listSessions().find(d=>d.yamlPath===g&&d.status!=="ended");try{let d=t.openSession(g);f.status(u?200:201).json({sessionId:d.sessionId,yamlPath:d.yamlPath,startedAt:d.startedAt,status:d.status})}catch(d){f.status(500).json({error:d.message})}}),a.get("/api/debugger/sessions",(p,f)=>{f.setHeader("Cache-Control","no-store"),f.json({sessions:t.listSessions().map(h=>({sessionId:h.sessionId,yamlPath:h.yamlPath,startedAt:h.startedAt,status:h.status}))})}),a.delete("/api/debugger/sessions/:sessionId",async(p,f)=>{let h=p.params.sessionId,g=!!t.getSession(h);await t.closeSession(h),f.json({deleted:!0,alreadyGone:!g})}),Ae.existsSync(r)?a.use("/debugger/static",Kt.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),a.get("/debugger/:sessionId/",(p,f)=>{let h=p.params.sessionId;if(!t.getSession(h)){f.status(404).send("Debugger session not found");return}let u=pe.join(r,"index.html");if(!Ae.existsSync(u)){f.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let w=Ae.readFileSync(u,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');f.type("html").send(w)}),a.use("/debugger/:sessionId",Wn());let l=async(p,f,h)=>{let g=p.params.sessionId;if(g==="static")return h();let u=t.getSession(g);if(!u){f.status(404).json({status:"error",message:"Session not found"});return}let d=p.originalUrl,w=`/debugger/${g}`;if(d.startsWith(w)){let y=d.slice(w.length)||"/";p.url=y,Object.defineProperty(p,"originalUrl",{value:y,configurable:!0})}if(c&&p.path.startsWith("/api/report-assets/")){p.url=p.url.replace(/^\/api\/report-assets/,""),c(p,f,h);return}if(u.status==="idle"||u.status==="starting"){let y=`${p.method} ${p.path}`;if(y==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(m){f.status(500).json({status:"error",message:m.message});return}else if(y==="POST /api/int-runner/liveview-url"){f.json({liveviewUrl:"",browserWsUrl:""});return}else if(p.path.startsWith("/api/int-runner/")){f.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(p,f,h);return}else{f.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>s(g,p)})(p,f,h)};return a.use("/debugger/:sessionId",l),a}function Vn(e,t,r,n){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
239
+ `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});function Wn(e){return e.startsWith($i)}function Se(e,t){let r=t?.trim();return r?Oi(r):Wn(e)?Ii:Mi}function Oi(e){return e.endsWith("/")?e.slice(0,-1):e}var $i,Ii,Mi,Gt=x(()=>{"use strict";$i="shp_",Ii="https://nova-api.shiplight.ai",Mi="https://api.shiplight.ai"});import{Router as Ri}from"express";function Gn(){let e=Ri();return e.get("/api/email-forwarding/addresses",async(t,r)=>{let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.error("[email-forwarding] SHIPLIGHT_API_TOKEN not set, skipping"),r.json({addresses:[],configured:!1});return}let s=Se(n,process.env.SHIPLIGHT_API_URL),o=`${s}/email-forwarding/addresses`;try{let i=await fetch(o,{headers:{Authorization:`Bearer ${n}`},signal:AbortSignal.timeout(1e4)});if(!i.ok){let l=(await i.json().catch(()=>({}))).error??`${i.status} ${i.statusText}`;console.error(`[email-forwarding] upstream ${i.status}: ${l}`),r.status(i.status).json({error:l});return}let a=await i.json();r.json({...a,configured:!0})}catch(i){let a=i instanceof Error?i.message:String(i);console.error(`[email-forwarding] fetch error: ${a}`),r.status(502).json({error:`Failed to reach Shiplight API (${s}): ${a}`})}}),e}var Kn=x(()=>{"use strict";Gt()});import{Router as Li}from"express";import Kt from"express";import*as Ae from"fs";import*as pe from"path";function Ci(e){return e?pe.isAbsolute(e)?pe.normalize(e):pe.resolve(e):null}function Ni(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function Di(e){let t=pe.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Vn(e){let{manager:t,staticDir:r,resolveYamlPath:n=Ci,liveviewUrlBuilder:s=Ni,fallbackRouter:o,artifactsDir:i}=e,a=Li(),c=i?Kt.static(i):null;a.post("/api/debugger/sessions",Kt.json(),async(p,d)=>{let h=p.body?.yamlPath;if(typeof h!="string"||!h){d.status(400).json({error:"yamlPath is required"});return}let g=n(h);if(!g){d.status(403).json({error:"Path outside allowed roots"});return}if(!Di(g)){d.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!Ae.existsSync(g)){d.status(404).json({error:"File not found"});return}let u=t.listSessions().find(f=>f.yamlPath===g&&f.status!=="ended");try{let f=t.openSession(g);d.status(u?200:201).json({sessionId:f.sessionId,yamlPath:f.yamlPath,startedAt:f.startedAt,status:f.status})}catch(f){d.status(500).json({error:f.message})}}),a.get("/api/debugger/sessions",(p,d)=>{d.setHeader("Cache-Control","no-store"),d.json({sessions:t.listSessions().map(h=>({sessionId:h.sessionId,yamlPath:h.yamlPath,startedAt:h.startedAt,status:h.status}))})}),a.delete("/api/debugger/sessions/:sessionId",async(p,d)=>{let h=p.params.sessionId,g=!!t.getSession(h);await t.closeSession(h),d.json({deleted:!0,alreadyGone:!g})}),Ae.existsSync(r)?a.use("/debugger/static",Kt.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),a.get("/debugger/:sessionId/",(p,d)=>{let h=p.params.sessionId;if(!t.getSession(h)){d.status(404).send("Debugger session not found");return}let u=pe.join(r,"index.html");if(!Ae.existsSync(u)){d.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let w=Ae.readFileSync(u,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');d.type("html").send(w)}),a.use("/debugger/:sessionId",Gn());let l=async(p,d,h)=>{let g=p.params.sessionId;if(g==="static")return h();let u=t.getSession(g);if(!u){d.status(404).json({status:"error",message:"Session not found"});return}let f=p.originalUrl,w=`/debugger/${g}`;if(f.startsWith(w)){let y=f.slice(w.length)||"/";p.url=y,Object.defineProperty(p,"originalUrl",{value:y,configurable:!0})}if(c&&p.path.startsWith("/api/report-assets/")){p.url=p.url.replace(/^\/api\/report-assets/,""),c(p,d,h);return}if(u.status==="idle"||u.status==="starting"){let y=`${p.method} ${p.path}`;if(y==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(m){d.status(500).json({status:"error",message:m.message});return}else if(y==="POST /api/int-runner/liveview-url"){d.json({liveviewUrl:"",browserWsUrl:""});return}else if(p.path.startsWith("/api/int-runner/")){d.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(p,d,h);return}else{d.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>s(g,p)})(p,d,h)};return a.use("/debugger/:sessionId",l),a}function zn(e,t,r,n){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
235
240
  \r
236
- `),r.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,r,n,a||void 0)}var zn=x(()=>{"use strict";Gn()});var Jn={};ue(Jn,{startDebuggerServer:()=>Di});import Fe from"express";import*as G from"path";import{createRequire as Ni}from"node:module";async function Di(e){let{initialDir:t,initialFile:r,projectRoot:n,port:s,headed:o=!1}=e,i=new it({headed:o}),a=Fe();a.use((b,y,m)=>{if(y.setHeader("Access-Control-Allow-Origin","*"),y.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),y.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),b.method==="OPTIONS")return y.sendStatus(204);m()}),a.use(Fe.json({limit:"10mb"}));let c=Rn({initialDir:t,initialFile:r,projectRoot:n});a.use(c);let l=G.join(n??t,".shiplight","artifacts");a.use("/api/report-assets",Fe.static(l));let p=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,f=p.includes(G.sep+"src"+G.sep),h=f?G.resolve(p,"../../dist/static"):G.join(p,"static"),g=f?G.resolve(p,"../../dist/static-embedded"):G.join(p,"static-embedded");a.use("/devtools",Fe.static(Yn)),a.use(Fe.static(h)),a.use(Kn({manager:i,staticDir:g,fallbackRouter:c,artifactsDir:l})),a.get("*path",(b,y,m)=>{if(b.path.startsWith("/api/"))return m();y.sendFile(G.join(h,"index.html"),S=>{S&&y.send(Ui(t,s))})});let u=await new Promise((b,y)=>{let m=a.listen(s,"localhost",()=>{b(m)});m.on("error",y)});u.on("upgrade",(b,y,m)=>{if((b.url??"").startsWith("/ws/debugger/")){Vn(i,b,y,m);return}y.write(`HTTP/1.1 404 Not Found\r
241
+ `),r.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,r,n,a||void 0)}var Yn=x(()=>{"use strict";Kn()});var Xn={};ue(Xn,{startDebuggerServer:()=>Fi});import Fe from"express";import*as G from"path";import{createRequire as ji}from"node:module";async function Fi(e){let{initialDir:t,initialFile:r,projectRoot:n,port:s,headed:o=!1}=e,i=new it({headed:o}),a=Fe();a.use((b,y,m)=>{if(y.setHeader("Access-Control-Allow-Origin","*"),y.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),y.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),b.method==="OPTIONS")return y.sendStatus(204);m()}),a.use(Fe.json({limit:"10mb"}));let c=Ln({initialDir:t,initialFile:r,projectRoot:n});a.use(c);let l=G.join(n??t,".shiplight","artifacts");a.use("/api/report-assets",Fe.static(l));let p=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,d=p.includes(G.sep+"src"+G.sep),h=d?G.resolve(p,"../../dist/static"):G.join(p,"static"),g=d?G.resolve(p,"../../dist/static-embedded"):G.join(p,"static-embedded");a.use("/devtools",Fe.static(Jn)),a.use(Fe.static(h)),a.use(Vn({manager:i,staticDir:g,fallbackRouter:c,artifactsDir:l})),a.get("*path",(b,y,m)=>{if(b.path.startsWith("/api/"))return m();y.sendFile(G.join(h,"index.html"),S=>{S&&y.send(Hi(t,s))})});let u=await new Promise((b,y)=>{let m=a.listen(s,"localhost",()=>{b(m)});m.on("error",y)});u.on("upgrade",(b,y,m)=>{if((b.url??"").startsWith("/ws/debugger/")){zn(i,b,y,m);return}y.write(`HTTP/1.1 404 Not Found\r
237
242
  \r
238
- `),y.destroy()});let d=`http://localhost:${s}`,w=r?`${d}/?open=${encodeURIComponent(r)}`:d;return console.error(`[debugger] Server running at ${d}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:w,close:async()=>{await i.shutdown(),await new Promise((b,y)=>{u.close(m=>m?y(m):b())})}}}function Ui(e,t){return`<!DOCTYPE html>
243
+ `),y.destroy()});let f=`http://localhost:${s}`,w=r?`${f}/?open=${encodeURIComponent(r)}`:f;return console.error(`[debugger] Server running at ${f}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:w,close:async()=>{await i.shutdown(),await new Promise((b,y)=>{u.close(m=>m?y(m):b())})}}}function Hi(e,t){return`<!DOCTYPE html>
239
244
  <html>
240
245
  <head>
241
246
  <meta charset="utf-8">
@@ -253,27 +258,27 @@ Host: ${h}:${f}\r
253
258
  <p>Directory: <code>${e}</code></p>
254
259
  <p>Server: localhost:${t}</p>
255
260
  </body>
256
- </html>`}var ji,Yn,Xn=x(()=>{"use strict";Ln();Hn();zn();ji=Ni(import.meta.url);try{Yn=G.join(G.dirname(ji.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 Zn={};ue(Zn,{startDebugger:()=>Hi});import*as se from"fs";import*as ye from"path";function at(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function Hi(e){let t,r=$e,n=!1,s,o=!1,i=!0,a=!1;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),n=!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"?(Bi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let c,l;if(t){let y=ye.resolve(t);se.existsSync(y)&&se.statSync(y).isDirectory()?c=y:(c=ye.dirname(y),l=y)}else c=process.cwd();if(o&&l&&!se.existsSync(l)){let y=ye.dirname(l);se.existsSync(y)||se.mkdirSync(y,{recursive:!0}),se.writeFileSync(l,Fi(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${l}`)}l&&!se.existsSync(l)&&(console.error(`Error: File not found: ${l}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:p}=await Promise.resolve().then(()=>(xt(),Rr)),f=p(c),h=f?ye.dirname(f):c;(await import("dotenv")).config({path:ye.join(h,".env"),override:!0}),f||(console.error("Error: No Playwright config found in "+c),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:u}=await Promise.resolve().then(()=>(Xn(),Jn));if(console.log(l?`Starting debugger for: ${l}`:`Starting debugger in: ${c}`),f&&console.log(`Using Playwright config: ${f}`),!n){let y=await Ir($e,qn);y===null&&(console.error(at(`Error: No available port found in range ${$e}-${$e+qn-1}.`)),console.error(at("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),y!==$e&&console.log(`Port ${$e} is in use; using port ${y} instead.`),r=y}let d;try{d=await u({initialDir:c,initialFile:l,projectRoot:h,port:r,headed:a})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(at(`Error: Port ${r} is already in use.`)),console.error(at(n?"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(kr(r,c),console.log(`Debugger running at: ${d.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!i)try{let{default:y}=await import("open");await y(d.url)}catch{}let w=!1,b=async()=>{w&&(console.log(`
261
+ </html>`}var Ui,Jn,qn=x(()=>{"use strict";Cn();Bn();Yn();Ui=ji(import.meta.url);try{Jn=G.join(G.dirname(Ui.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 Qn={};ue(Qn,{startDebugger:()=>Wi});import*as se from"fs";import*as ye from"path";function at(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function Wi(e){let t,r=$e,n=!1,s,o=!1,i=!0,a=!1;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),n=!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"?(Gi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let c,l;if(t){let y=ye.resolve(t);se.existsSync(y)&&se.statSync(y).isDirectory()?c=y:(c=ye.dirname(y),l=y)}else c=process.cwd();if(o&&l&&!se.existsSync(l)){let y=ye.dirname(l);se.existsSync(y)||se.mkdirSync(y,{recursive:!0}),se.writeFileSync(l,Bi(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${l}`)}l&&!se.existsSync(l)&&(console.error(`Error: File not found: ${l}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:p}=await Promise.resolve().then(()=>(xt(),Lr)),d=p(c),h=d?ye.dirname(d):c;(await import("dotenv")).config({path:ye.join(h,".env"),override:!0}),d||(console.error("Error: No Playwright config found in "+c),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:u}=await Promise.resolve().then(()=>(qn(),Xn));if(console.log(l?`Starting debugger for: ${l}`:`Starting debugger in: ${c}`),d&&console.log(`Using Playwright config: ${d}`),!n){let y=await Mr($e,Zn);y===null&&(console.error(at(`Error: No available port found in range ${$e}-${$e+Zn-1}.`)),console.error(at("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),y!==$e&&console.log(`Port ${$e} is in use; using port ${y} instead.`),r=y}let f;try{f=await u({initialDir:c,initialFile:l,projectRoot:h,port:r,headed:a})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(at(`Error: Port ${r} is already in use.`)),console.error(at(n?"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(Pr(r,c),console.log(`Debugger running at: ${f.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!i)try{let{default:y}=await import("open");await y(f.url)}catch{}let w=!1,b=async()=>{w&&(console.log(`
257
262
  Force exiting...`),process.exit(1)),w=!0,console.log(`
258
- Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Pr(r,c),await d.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",b),process.on("SIGTERM",b)}function Bi(){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 $e,qn,Fi,Qn=x(()=>{"use strict";Er();Mr();$e=6174,qn=10;Fi=e=>`goal: New test
263
+ Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Er(r,c),await f.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",b),process.on("SIGTERM",b)}function Gi(){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 $e,Zn,Bi,es=x(()=>{"use strict";Ar();Or();$e=6174,Zn=10;Bi=e=>`goal: New test
259
264
  base_url: ${e}
260
265
  statements:
261
266
  - URL: /
262
- `});import xu from"dotenv";function Gi(){return globalThis[Wi]}function ct(){let e=Gi();return e===void 0?process.env:e}var Wi,Vt=x(()=>{"use strict";Wi="__shiplightDotenvCache__"});var zt=x(()=>{"use strict";Gt()});var Yt={};ue(Yt,{lookupActionStores:()=>zi,updateActionStores:()=>Yi});function es(){let e=ct(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Se(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function zi(e){let t=es();if(!t||e.length===0)return new Map;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Ki),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:r.signal});if(clearTimeout(n),!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,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Yi(e){let t=es();if(!t||e.size===0)return 0;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Vi),s={};for(let[a,c]of e)s[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:s}),signal:r.signal});return clearTimeout(n),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var Ki,Vi,Jt=x(()=>{"use strict";zt();Vt();Ki=2e3,Vi=5e3});import*as J from"fs";import*as He from"path";import{globSync as Ji}from"glob";function ts(e){let t=ct().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new qt:new Xt(e)}function Be(e){return e.replace(/\//g,"__")+".json"}function qi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Xi,Xt,qt,rs=x(()=>{"use strict";ge();Vt();Xi=".shiplight/action-cache";Xt=class{constructor(t){this.cwd=t;this.cacheDir=He.join(t,Xi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!J.existsSync(this.cacheDir))return r;for(let n of t){let s=He.join(this.cacheDir,Be(n));try{if(J.existsSync(s)){let o=J.readFileSync(s,"utf-8");r.set(n,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;J.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[n,s]of t)try{let o=He.join(this.cacheDir,Be(n)),i=kt();if(J.existsSync(o))try{i=JSON.parse(J.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};J.writeFileSync(o,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!J.existsSync(this.cacheDir))return;let t=Ji("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,n=0;for(let s of t)try{let o=J.readFileSync(He.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=qi(s);r.set(a,i),n+=Object.keys(i.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},qt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Jt(),Yt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Jt(),Yt));return r(t)}loadAll(){}}});var ls={};ue(ls,{buildPlaywrightSpawnOptions:()=>ss,buildTestPathsFromGitInfo:()=>is,cloudKeyToCwdRelPath:()=>rr,extractVarOverrideArgs:()=>cs,loadVarsFile:()=>er,parseVarsArg:()=>Qt,rewriteTestYamlArg:()=>as,runTests:()=>ea});import{spawn as Zi,execFileSync as ns}from"child_process";import*as B from"fs";import*as C from"path";import{pathToFileURL as Qi}from"url";import{globSync as Zt}from"glob";async function ea(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Shiplight options:"),console.log(" --vars KEY=VAL[,KEY=VAL...] Override runtime {{VAR}} values (repeatable)."),console.log(" Comma separates pairs, so values cannot"),console.log(" contain commas \u2014 use --vars-file for those."),console.log(" In the space-separated form, the value cannot"),console.log(" start with `--` (it is treated as the next"),console.log(" flag); use --vars=KEY=--value instead."),console.log(' --vars-file <path> JSON file of overrides ({ KEY: "value", ... }).'),console.log(" Values must be strings \u2014 quote numbers."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),console.log(" shiplight test --vars SAUCE_USER=standard_user"),console.log(" shiplight test --vars-file ./local-vars.json"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(h=>B.existsSync(C.join(t,h)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
263
- `));let s=e.includes("--magic"),o=e.filter(h=>h!=="--magic"),i={};try{let{extracted:h,remaining:g}=cs(o,t);o=g,i=h}catch(h){console.error(`[shiplight] ${h.message}`),process.exit(2)}let a=o.map(h=>h.endsWith(".test.yaml")?as(h,t):h),c=ts(t);await ra(t,c),Me&&process.stdout.write(`shiplightai v${Me}
264
- `);let l={...process.env};s&&(l.SHIPLIGHT_MAGIC="1"),Object.keys(i).length>0&&(l.SHIPLIGHT_VARS_OVERRIDE=JSON.stringify(i));let p=Zi("npx",["playwright","test",...a],ss(t,l)),f=await new Promise(h=>{p.on("close",g=>h(g??1))});await na(t,c),process.exit(f)}function ss(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function tr(){try{return ns("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function ta(){try{let e=ns("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function os(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let n=ta(),s=tr();return is(e,t,n,s)}function is(e,t,r,n){return{testPaths:e.map(o=>{let i=n?C.relative(n,C.resolve(t,o)):o;return`${r}${i}`}),branchPrefix:r}}function rr(e,t,r,n){let s=t&&e.startsWith(t)?e.slice(t.length):e;return r?C.relative(n,C.resolve(r,s)):s}async function ra(e,t){try{let r=Zt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:n,branchPrefix:s}=os(r,e,t.isCloud),o=await t.lookup(n);if(o.size===0)return;let i=C.join(e,".shiplight","action-cache");B.mkdirSync(i,{recursive:!0});let a=tr(),c=0;for(let[l,p]of o){let f=t.isCloud?rr(l,s,a,e):l,h=C.join(i,Be(f));B.writeFileSync(h,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function na(e,t){try{let r=C.join(e,"test-results");if(!B.existsSync(r))return;let n=Zt("**/new-action-entities.json",{cwd:r});if(n.length===0)return;let s={};for(let f of n)try{let h=B.readFileSync(C.join(r,f),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(s,g.entries)}catch{}if(Object.keys(s).length===0)return;let o=Zt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=os(o,e,t.isCloud),c=new Map;for(let f=0;f<o.length;f++){let h=o[f],g=i[f],u=C.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!B.existsSync(u))continue;let d=B.readFileSync(u,"utf-8"),w={};for(let[b,y]of Object.entries(s))d.includes(b)&&(w[b]=y);if(Object.keys(w).length>0){let b=t.isCloud?rr(g,a,tr(),e):g,y=C.join(e,".shiplight","action-cache",Be(b)),m={};if(B.existsSync(y))try{m=JSON.parse(B.readFileSync(y,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...m,...w}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((f,h)=>f+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}function as(e,t){let r=e.replace(/\.test\.yaml$/,".yaml.spec.ts");return process.platform==="win32"&&sa.test(e)?Qi(C.resolve(t,r)).href:r}function Qt(e){let t={};for(let r of e.split(",")){let n=r.trim();if(n==="")continue;let s=n.indexOf("=");if(s<=0)throw new Error(`Invalid --vars entry "${n}": expected KEY=VALUE`);let o=n.slice(0,s).trim();t[o]=n.slice(s+1)}return t}function er(e,t){let r=C.isAbsolute(e)?e:C.join(t,e);if(!B.existsSync(r))throw new Error(`--vars-file not found: ${r}`);let n;try{n=JSON.parse(B.readFileSync(r,"utf-8"))}catch(o){throw new Error(`--vars-file ${r} is not valid JSON: ${o.message}`)}if(n===null||typeof n!="object"||Array.isArray(n))throw new Error(`--vars-file ${r} must be a JSON object`);let s={};for(let[o,i]of Object.entries(n)){if(typeof i!="string")throw new Error(`--vars-file ${r}: key "${o}" must be a string (got ${typeof i})`);s[o]=i}return s}function cs(e,t){let r={},n={},s=[];for(let o=0;o<e.length;o++){let i=e[o];if(i==="--vars"||i==="--vars-file"){let a=e[o+1];if(a===void 0||a.startsWith("--"))throw new Error(`${i} requires a value`);i==="--vars"?Object.assign(n,Qt(a)):Object.assign(r,er(a,t)),o+=1;continue}if(i.startsWith("--vars=")){Object.assign(n,Qt(i.slice(7)));continue}if(i.startsWith("--vars-file=")){Object.assign(r,er(i.slice(12),t));continue}s.push(i)}return{extracted:{...r,...n},remaining:s}}var sa,ps=x(()=>{"use strict";rs();Ge();sa=/[^\x00-\x7F]/});import us from"node:path";function R(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function pt(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function oa(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function ia(e,t){let r=e||"?",n=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${n}`:`failed after ${r} ${n}`}function lt(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 aa(e){return`<span class="badge badge-${e}">${e}</span>`}function ca(e){if(!e.code)return"";let t=e.code.split(`
265
- `);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,c=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${R(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(n=>`<span class="code-line code-line-body">${R(n)}</span>`).join("")}</pre></div>`}function la(e){let t=e.duration!=null?`<span class="step-duration">${pt(e.duration)}</span>`:"",r=lt(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${R(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":ca(e),i="";e.error&&(i=`<div class="step-error"><pre>${R(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${R(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
267
+ `});import ku from"dotenv";function Vi(){return globalThis[Ki]}function ct(){let e=Vi();return e===void 0?process.env:e}var Ki,Vt=x(()=>{"use strict";Ki="__shiplightDotenvCache__"});var zt=x(()=>{"use strict";Gt()});var Yt={};ue(Yt,{lookupActionStores:()=>Ji,updateActionStores:()=>Xi});function ts(){let e=ct(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Se(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function Ji(e){let t=ts();if(!t||e.length===0)return new Map;try{let r=new AbortController,n=setTimeout(()=>r.abort(),zi),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:r.signal});if(clearTimeout(n),!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,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Xi(e){let t=ts();if(!t||e.size===0)return 0;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Yi),s={};for(let[a,c]of e)s[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:s}),signal:r.signal});return clearTimeout(n),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var zi,Yi,Jt=x(()=>{"use strict";zt();Vt();zi=2e3,Yi=5e3});import*as J from"fs";import*as He from"path";import{globSync as qi}from"glob";function rs(e){let t=ct().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new qt:new Xt(e)}function Be(e){return e.replace(/\//g,"__")+".json"}function Qi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Zi,Xt,qt,ns=x(()=>{"use strict";ge();Vt();Zi=".shiplight/action-cache";Xt=class{constructor(t){this.cwd=t;this.cacheDir=He.join(t,Zi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!J.existsSync(this.cacheDir))return r;for(let n of t){let s=He.join(this.cacheDir,Be(n));try{if(J.existsSync(s)){let o=J.readFileSync(s,"utf-8");r.set(n,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;J.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[n,s]of t)try{let o=He.join(this.cacheDir,Be(n)),i=kt();if(J.existsSync(o))try{i=JSON.parse(J.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};J.writeFileSync(o,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!J.existsSync(this.cacheDir))return;let t=qi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,n=0;for(let s of t)try{let o=J.readFileSync(He.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=Qi(s);r.set(a,i),n+=Object.keys(i.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},qt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Jt(),Yt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Jt(),Yt));return r(t)}loadAll(){}}});var us={};ue(us,{buildPlaywrightSpawnOptions:()=>os,buildTestPathsFromGitInfo:()=>cs,cloudKeyToCwdRelPath:()=>nr,extractConfigArg:()=>tr,extractVarOverrideArgs:()=>ps,loadVarsFile:()=>er,parseVarsArg:()=>Qt,resolveProjectRootDir:()=>is,rewriteTestYamlArg:()=>ls,runTests:()=>ra});import{spawn as ea,execFileSync as ss}from"child_process";import*as D from"fs";import*as R from"path";import{pathToFileURL as ta}from"url";import{globSync as Zt}from"glob";async function ra(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Shiplight options:"),console.log(" --vars KEY=VAL[,KEY=VAL...] Override runtime {{VAR}} values (repeatable)."),console.log(" Comma separates pairs, so values cannot"),console.log(" contain commas \u2014 use --vars-file for those."),console.log(" In the space-separated form, the value cannot"),console.log(" start with `--` (it is treated as the next"),console.log(" flag); use --vars=KEY=--value instead."),console.log(' --vars-file <path> JSON file of overrides ({ KEY: "value", ... }).'),console.log(" Values must be strings \u2014 quote numbers."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),console.log(" shiplight test --vars SAUCE_USER=standard_user"),console.log(" shiplight test --vars-file ./local-vars.json"),process.exit(0));let t=process.cwd(),r=is(e,t);tr(e)===void 0&&(["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(g=>D.existsSync(R.join(t,g)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
268
+ `)));let n=e.includes("--magic"),s=e.filter(d=>d!=="--magic"),o={};try{let{extracted:d,remaining:h}=ps(s,t);s=h,o=d}catch(d){console.error(`[shiplight] ${d.message}`),process.exit(2)}let i=s.map(d=>d.endsWith(".test.yaml")?ls(d,t):d),a=rs(r);await sa(r,a),Me&&process.stdout.write(`shiplightai v${Me}
269
+ `);let c={...process.env};n&&(c.SHIPLIGHT_MAGIC="1"),Object.keys(o).length>0&&(c.SHIPLIGHT_VARS_OVERRIDE=JSON.stringify(o)),c.SHIPLIGHT_PROJECT_ROOT=r;let l=ea("npx",["playwright","test",...i],os(t,c)),p=await new Promise(d=>{l.on("close",h=>d(h??1))});await oa(r,a),process.exit(p)}function os(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function tr(e){for(let t=0;t<e.length;t++){let r=e[t];if(r==="-c"||r==="--config")return e[t+1];if(r.startsWith("--config="))return r.slice(9);if(r.startsWith("-c="))return r.slice(3);if(r.startsWith("-c")&&r.length>2)return r.slice(2)}}function is(e,t){let r=tr(e),n=r?R.resolve(t,r):t;try{return D.statSync(n).isDirectory()?n:R.dirname(n)}catch{return t}}function rr(){try{return ss("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function na(){try{let e=ss("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function as(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let n=na(),s=rr();return cs(e,t,n,s)}function cs(e,t,r,n){return{testPaths:e.map(o=>{let i=n?R.relative(n,R.resolve(t,o)):o;return`${r}${i}`}),branchPrefix:r}}function nr(e,t,r,n){let s=t&&e.startsWith(t)?e.slice(t.length):e;return r?R.relative(n,R.resolve(r,s)):s}async function sa(e,t){try{let r=Zt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:n,branchPrefix:s}=as(r,e,t.isCloud),o=await t.lookup(n);if(o.size===0)return;let i=R.join(e,".shiplight","action-cache");D.mkdirSync(i,{recursive:!0});let a=rr(),c=0;for(let[l,p]of o){let d=t.isCloud?nr(l,s,a,e):l,h=R.join(i,Be(d));D.writeFileSync(h,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function oa(e,t){try{let r=R.join(e,"test-results");if(!D.existsSync(r))return;let n=Zt("**/new-action-entities.json",{cwd:r});if(n.length===0)return;let s={};for(let d of n)try{let h=D.readFileSync(R.join(r,d),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(s,g.entries)}catch{}if(Object.keys(s).length===0)return;let o=Zt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=as(o,e,t.isCloud),c=new Map;for(let d=0;d<o.length;d++){let h=o[d],g=i[d],u=R.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!D.existsSync(u))continue;let f=D.readFileSync(u,"utf-8"),w={};for(let[b,y]of Object.entries(s))f.includes(b)&&(w[b]=y);if(Object.keys(w).length>0){let b=t.isCloud?nr(g,a,rr(),e):g,y=R.join(e,".shiplight","action-cache",Be(b)),m={};if(D.existsSync(y))try{m=JSON.parse(D.readFileSync(y,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...m,...w}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((d,h)=>d+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}function ls(e,t){let r=e.replace(/\.test\.yaml$/,".yaml.spec.ts");return process.platform==="win32"&&ia.test(e)?ta(R.resolve(t,r)).href:r}function Qt(e){let t={};for(let r of e.split(",")){let n=r.trim();if(n==="")continue;let s=n.indexOf("=");if(s<=0)throw new Error(`Invalid --vars entry "${n}": expected KEY=VALUE`);let o=n.slice(0,s).trim();t[o]=n.slice(s+1)}return t}function er(e,t){let r=R.isAbsolute(e)?e:R.join(t,e);if(!D.existsSync(r))throw new Error(`--vars-file not found: ${r}`);let n;try{n=JSON.parse(D.readFileSync(r,"utf-8"))}catch(o){throw new Error(`--vars-file ${r} is not valid JSON: ${o.message}`)}if(n===null||typeof n!="object"||Array.isArray(n))throw new Error(`--vars-file ${r} must be a JSON object`);let s={};for(let[o,i]of Object.entries(n)){if(typeof i!="string")throw new Error(`--vars-file ${r}: key "${o}" must be a string (got ${typeof i})`);s[o]=i}return s}function ps(e,t){let r={},n={},s=[];for(let o=0;o<e.length;o++){let i=e[o];if(i==="--vars"||i==="--vars-file"){let a=e[o+1];if(a===void 0||a.startsWith("--"))throw new Error(`${i} requires a value`);i==="--vars"?Object.assign(n,Qt(a)):Object.assign(r,er(a,t)),o+=1;continue}if(i.startsWith("--vars=")){Object.assign(n,Qt(i.slice(7)));continue}if(i.startsWith("--vars-file=")){Object.assign(r,er(i.slice(12),t));continue}s.push(i)}return{extracted:{...r,...n},remaining:s}}var ia,ds=x(()=>{"use strict";ns();Ge();ia=/[^\x00-\x7F]/});import fs from"node:path";function L(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function pt(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function aa(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function ca(e,t){let r=e||"?",n=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${n}`:`failed after ${r} ${n}`}function lt(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 la(e){return`<span class="badge badge-${e}">${e}</span>`}function pa(e){if(!e.code)return"";let t=e.code.split(`
270
+ `);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,c=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${L(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(n=>`<span class="code-line code-line-body">${L(n)}</span>`).join("")}</pre></div>`}function ua(e){let t=e.duration!=null?`<span class="step-duration">${pt(e.duration)}</span>`:"",r=lt(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${L(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":pa(e),i="";e.error&&(i=`<div class="step-error"><pre>${L(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${L(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
266
271
  <details class="step-details step step-${e.status}"${c}>
267
272
  <summary class="step-header">
268
273
  ${r}
269
- <span class="step-id">${R(e.stepId)}</span>
270
- <span class="step-description-collapsed">${R(e.description)}</span>
274
+ <span class="step-id">${L(e.stepId)}</span>
275
+ <span class="step-description-collapsed">${L(e.description)}</span>
271
276
  ${t}
272
277
  </summary>
273
278
  <div class="step-expanded">
274
279
  ${s}
275
280
  ${o}
276
- <div class="step-description-full">${R(e.description)}</div>
281
+ <div class="step-description-full">${L(e.description)}</div>
277
282
  ${a}
278
283
  ${i}
279
284
  </div>
@@ -281,32 +286,32 @@ statements:
281
286
  <div class="step step-${e.status}">
282
287
  <div class="step-header">
283
288
  ${r}
284
- <span class="step-id">${R(e.stepId)}</span>
285
- <span class="step-description">${R(e.description)}</span>
289
+ <span class="step-id">${L(e.stepId)}</span>
290
+ <span class="step-description">${L(e.description)}</span>
286
291
  ${t}
287
292
  </div>
288
- </div>`}function pa(e,t,r,n){let s=[];e&&s.push(`
293
+ </div>`}function da(e,t,r,n){let s=[];e&&s.push(`
289
294
  <details class="artifact-section">
290
295
  <summary class="artifact-summary">Video</summary>
291
296
  <div class="artifact-content">
292
297
  <div class="video-container">
293
298
  <video controls preload="metadata" class="artifact-video">
294
- <source src="${R(e)}" type="video/webm" />
299
+ <source src="${L(e)}" type="video/webm" />
295
300
  </video>
296
301
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
297
302
  </div>
298
303
  </div>
299
- </details>`);let o=r.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=R(JSON.stringify(o)),a=o.map((c,l)=>`
304
+ </details>`);let o=r.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=L(JSON.stringify(o)),a=o.map((c,l)=>`
300
305
  <div class="screenshot-thumb" onclick="openGalleryAt(this, ${l})" data-gallery="${i}">
301
- <img src="${R(c.src)}" alt="${R(c.stepId)}" />
302
- <span class="thumb-label">${R(c.stepId)}</span>
306
+ <img src="${L(c.src)}" alt="${L(c.stepId)}" />
307
+ <span class="thumb-label">${L(c.stepId)}</span>
303
308
  </div>`).join("");s.push(`
304
309
  <details class="artifact-section">
305
310
  <summary class="artifact-summary">Screenshots (${o.length})</summary>
306
311
  <div class="artifact-content">
307
312
  <div class="screenshot-grid">${a}</div>
308
313
  </div>
309
- </details>`)}if(t){let i=R(t);s.push(`
314
+ </details>`)}if(t){let i=L(t);s.push(`
310
315
  <details class="artifact-section">
311
316
  <summary class="artifact-summary">Trace</summary>
312
317
  <div class="artifact-content">
@@ -315,40 +320,40 @@ statements:
315
320
  <button class="copy-btn" onclick="copyTraceCmd('${n}')" title="Copy command">Copy</button>
316
321
  </div>
317
322
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
318
- <p class="trace-hint"><a href="${R(t)}" class="attachment-link" download>Download trace.zip</a></p>
323
+ <p class="trace-hint"><a href="${L(t)}" class="attachment-link" download>Download trace.zip</a></p>
319
324
  </div>
320
- </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function ds(e,t,r,n,s){let o=e.map(la).join(`
321
- `),i="";t&&!e.some(c=>c.error)&&(i=`<div class="test-error"><pre>${R(t)}</pre></div>`);let a=pa(r,n,e,s);return`
325
+ </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function hs(e,t,r,n,s){let o=e.map(ua).join(`
326
+ `),i="";t&&!e.some(c=>c.error)&&(i=`<div class="test-error"><pre>${L(t)}</pre></div>`);let a=da(r,n,e,s);return`
322
327
  ${i}
323
328
  <div class="steps-list">
324
329
  ${o||'<div class="no-steps">No YAML step details available</div>'}
325
330
  </div>
326
- ${a}`}function ua(e,t){let r=e.flaky?"flaky":e.status,n=lt(r),s;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,a=e.attempts.length-1,c=e.attempts.map((h,g)=>{let u=g===a,d=h.status==="passed"?"passed":"failed",w=`Attempt ${h.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${d}"
331
+ ${a}`}function fa(e,t){let r=e.flaky?"flaky":e.status,n=lt(r),s;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,a=e.attempts.length-1,c=e.attempts.map((h,g)=>{let u=g===a,f=h.status==="passed"?"passed":"failed",w=`Attempt ${h.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${f}"
327
332
  onclick="switchAttemptTab('${o}', ${g})"
328
- data-tab-index="${g}">${lt(d)} ${w} <span class="attempt-tab-badge badge-${d}">${h.status}</span></button>`}).join(""),l=e.attempts.map((h,g)=>{let u=g===a,d=ds(h.steps,h.error,h.videoPath,h.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${g}">
333
+ data-tab-index="${g}">${lt(f)} ${w} <span class="attempt-tab-badge badge-${f}">${h.status}</span></button>`}).join(""),l=e.attempts.map((h,g)=>{let u=g===a,f=hs(h.steps,h.error,h.videoPath,h.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${g}">
329
334
  <div class="attempt-meta">
330
335
  ${lt(h.status==="passed"?"passed":"failed")}
331
336
  <span class="attempt-meta-text">Attempt ${h.attemptNumber} &mdash; ${h.status} in ${pt(h.duration)}</span>
332
337
  </div>
333
- ${d}
334
- </div>`}).join(""),p=e.flaky?`Flaky &mdash; ${oa(e.retries)}`:`Retried &mdash; ${ia(e.retries,e.status)}`;s=`
338
+ ${f}
339
+ </div>`}).join(""),p=e.flaky?`Flaky &mdash; ${aa(e.retries)}`:`Retried &mdash; ${ca(e.retries,e.status)}`;s=`
335
340
  <div class="${e.flaky?"flaky-note":"retried-note"}">${p}</div>
336
341
  <div class="attempt-tabs" id="${o}">
337
342
  <div class="attempt-tab-bar">${c}</div>
338
343
  ${l}
339
- </div>`}else s=ds(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
344
+ </div>`}else s=hs(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
340
345
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
341
346
  <summary class="test-summary test-${r}">
342
347
  ${n}
343
- <span class="test-title">${R(e.title)}</span>
344
- <span class="test-file">${R(e.file)}</span>
345
- ${aa(r)}
348
+ <span class="test-title">${L(e.title)}</span>
349
+ <span class="test-file">${L(e.file)}</span>
350
+ ${la(r)}
346
351
  <span class="test-duration">${pt(e.duration)}</span>
347
352
  </summary>
348
353
  <div class="test-body">
349
354
  ${s}
350
355
  </div>
351
- </details>`}function fs(e,t){return!e||!t||us.isAbsolute(e)?e:us.join(t,e)}function da(e,t){return t?{...e,tracePath:fs(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:fs(r.tracePath,t)}))}:e}function nr(e){let t=e.tests.filter(l=>l.flaky).length,r=e.tests.filter(l=>!l.flaky&&l.retries!=null&&l.retries>0).length,n=e.tests.filter(l=>l.status==="passed"&&!l.flaky).length,s=e.tests.filter(l=>l.status==="failed"||l.status==="timedOut").length,o=e.tests.filter(l=>l.status==="skipped").length,i=e.tests.length,c=e.tests.map(l=>da(l,e.outputDir)).map((l,p)=>ua(l,p)).join(`
356
+ </details>`}function gs(e,t){return!e||!t||fs.isAbsolute(e)?e:fs.join(t,e)}function ha(e,t){return t?{...e,tracePath:gs(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:gs(r.tracePath,t)}))}:e}function sr(e){let t=e.tests.filter(l=>l.flaky).length,r=e.tests.filter(l=>!l.flaky&&l.retries!=null&&l.retries>0).length,n=e.tests.filter(l=>l.status==="passed"&&!l.flaky).length,s=e.tests.filter(l=>l.status==="failed"||l.status==="timedOut").length,o=e.tests.filter(l=>l.status==="skipped").length,i=e.tests.length,c=e.tests.map(l=>ha(l,e.outputDir)).map((l,p)=>fa(l,p)).join(`
352
357
  `);return`<!DOCTYPE html>
353
358
  <html lang="en">
354
359
  <head>
@@ -1015,7 +1020,7 @@ statements:
1015
1020
  ${c}
1016
1021
  </div>
1017
1022
  <div class="footer">
1018
- Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${R(e.shiplightVersion)}`:""}
1023
+ Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${L(e.shiplightVersion)}`:""}
1019
1024
  </div>
1020
1025
  </div>
1021
1026
 
@@ -1142,9 +1147,9 @@ statements:
1142
1147
  });
1143
1148
  </script>
1144
1149
  </body>
1145
- </html>`}var hs=x(()=>{"use strict"});import*as W from"fs";import*as Ie from"path";import{execFileSync as fa}from"child_process";import{createHash as ha}from"crypto";import{Agent as ga}from"http";import{Agent as ma}from"https";import We from"axios";function gs(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function he(...e){try{return fa("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function ya(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=W.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function wa(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",n=process.env.GITHUB_RUN_ID??"",s=process.env.GITHUB_EVENT_NAME??"",i=ya().pull_request,a=i?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],f=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,h=process.env.SHIPLIGHT_PR_TITLE??i?.title,g=he("log","-1","--pretty=%s"),u=he("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:f,gitRepo:r,commitMessage:g,authorEmail:u,prNumber:p,prTitle:h,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:n,ciBuildUrl:n&&r?`${t}/${r}/actions/runs/${n}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`: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??"",r=process.env.CI_COMMIT_SHA??"",n=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:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:s,prNumber:n,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:n&&t?`${t}/-/merge_requests/${n}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",n=process.env.CIRCLE_PROJECT_REPONAME??"",s=r&&n?`${r}/${n}`: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,c=a?gs(a):void 0,l=he("log","-1","--pretty=%s"),p=he("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:l,authorEmail:p,prNumber:i,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=he("rev-parse","HEAD"),r=he("rev-parse","--abbrev-ref","HEAD"),n=he("log","-1","--pretty=%s"),s=he("log","-1","--pretty=%ae"),o=he("remote","get-url","origin"),i=o?gs(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:n,authorEmail:s,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function ba(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 sr(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function va(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 Sa(e,t){let r=new Map;for(let n of t){n.screenshotS3Uris={};let s=r.get(n.testCaseName);s?s.push(n):r.set(n.testCaseName,[n])}return e.map(n=>r.get(n.title)?.shift())}function Q(e,t){return Ie.isAbsolute(t)?t:Ie.join(e,t)}function ms(e,t){let r={};for(let n of e){let s={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};n.type&&(s.type=n.type),n.code&&(s.code=n.code),n.startTime&&(s.startTime=n.startTime),n.autoHealed&&(s.autoHealed=n.autoHealed),n.healedAction&&(s.healedAction=n.healedAction),n.dismissedModalActions?.length&&(s.dismissedModalActions=n.dismissedModalActions),n.contextBefore&&(s.contextBefore=n.contextBefore),n.contextAfter&&(s.contextAfter=n.contextAfter),r[n.stepId]=s}return r}function _a(e,t,r,n){let s=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let i=o.attemptNumber-1,a=t[i]??{},c={outcome:sr(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ms(o.steps,a),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[i],traceS3Uri:n[i],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),s.push(c)}else s.push({outcome:sr(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ms(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:n[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:sr(e.status),flaky:e.flaky??!1,segments:s}}async function or(e,t,r){let n=new Array(e.length),s=0;async function o(){for(;s<e.length;){let i=s++;n[i]=await r(e[i],i)}}return await Promise.all(Array.from({length:Math.min(t,e.length)},o)),n}function xa(e){let t=0,r=[];function n(){let s=r.shift();s?s():t-=1}return async function(o){t>=e?await new Promise(i=>{r.push(i)}):t+=1;try{return await o()}finally{n()}}}function Ta(e,t){if(!e)return{value:t,fromEnv:!1};let r=Number(e);return!Number.isInteger(r)||r<1?{value:t,fromEnv:!1}:{value:r,fromEnv:!0}}function ka(){let e=Ta(process.env.SHIPLIGHT_CLOUD_MAX_PUT_UPLOADS,6);return{maxConcurrentPutUploads:e.value,maxConcurrentPutUploadsFromEnv:e.fromEnv,screenshotUrlRequestWorkers:8,testUploadWorkers:5,perTestAssetWorkers:10}}function Pa(e){let t={keepAlive:!0,keepAliveMsecs:1e3,maxSockets:e};return{httpAgent:new ga(t),httpsAgent:new ma(t)}}function Ea(e,t){let r=t.maxConcurrentPutUploadsFromEnv?"env:SHIPLIGHT_CLOUD_MAX_PUT_UPLOADS":"default";return`[reporter] [config] Cloud upload settings for ${e} test(s): presignedPutCap=${t.maxConcurrentPutUploads} (${r}), screenshotUrlWorkers=${t.screenshotUrlRequestWorkers}, testUploadWorkers=${t.testUploadWorkers}, perTestAssetWorkers=${t.perTestAssetWorkers}`}async function Aa(e,t=3){for(let r=0;;r++)try{return await e()}catch(n){let s=n.code;if(!(s==="ECONNRESET"||s==="ETIMEDOUT"||s==="EPIPE"||s==="EAI_AGAIN"||s==="EADDRNOTAVAIL"||s==="EAGAIN")||r>=t)throw n;let i=Math.min(1e3*2**r,8e3);console.warn(`[reporter] Upload failed (${s}), retrying in ${i}ms (attempt ${r+1}/${t})...`),await new Promise(a=>setTimeout(a,i))}}function cr(e){return ha("md5").update(e).digest("base64")}function ir(e){return cr(W.readFileSync(e))}function $a(e,t){let r=xa(e.maxConcurrentPutUploads);return(n,s,o)=>r(()=>We.put(n,s,{...o,...t}))}async function ys(e,t,r,n,s){await Aa(()=>s(e,t,{headers:{"Content-Type":r,"Content-MD5":n}}))}async function ar(e,t,r){let n=W.readFileSync(t),s=Ie.extname(t).toLowerCase(),i={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await ys(e,n,i,cr(n),r)}async function ws(e,t,r,n){let s=Se(n,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${n}`,"Content-Type":"application/json"},i=ka(),a=Pa(Math.max(i.maxConcurrentPutUploads,i.screenshotUrlRequestWorkers,i.testUploadWorkers)),c=$a(i,a),l={headers:o,...a},p=wa(),f=e.tests.map(m=>{let S={testCaseName:m.title,testCaseBaseName:m.baseTitle,suiteName:m.suiteName,file:m.file,tags:m.tags,suiteTags:m.suiteTags,baseUrl:m.baseUrl,skip:m.skip,slow:m.slow,timeout:m.timeout,parameterSetName:m.parameterSetName,flaky:m.flaky,retries:m.retries};if(m.videoPath){let P=Q(t,m.videoPath);W.existsSync(P)&&(S.videoMd5=ir(P))}if(m.tracePath){let P=Q(t,m.tracePath);W.existsSync(P)&&(S.traceMd5=ir(P))}return S}),h=e.tests.length;console.log(`[reporter] Uploading ${h} test result(s) to Shiplight cloud...`),console.log(Ea(h,i)),console.log("[reporter] [1/4] Creating run record...");let u=(await We.post(`${s}/v1/local-runs`,{trigger:p.ciProvider,startTime:r,metadata:p,tests:f},l)).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Sa(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await or(e.tests,i.screenshotUrlRequestWorkers,async(m,S)=>{let P=d[S];if(!P)return;let k=[];if(m.attempts&&m.attempts.length>1)for(let I of m.attempts){let oe=I.attemptNumber-1,we=I.attemptNumber===m.attempts.length;for(let j of I.steps)j.screenshot&&k.push({key:`attempt-${oe}.${j.stepId}`,filePath:Q(t,j.screenshot)});if(!we){if(I.videoPath){let j=Q(t,I.videoPath);W.existsSync(j)&&k.push({key:`attempt-${oe}.__video__`,filePath:j})}if(I.tracePath){let j=Q(t,I.tracePath);W.existsSync(j)&&k.push({key:`attempt-${oe}.__trace__`,filePath:j})}}}else for(let I of m.steps)I.screenshot&&k.push({key:I.stepId,filePath:Q(t,I.screenshot)});if(!k.length)return;let K=k.map(I=>I.key),F={};for(let{key:I,filePath:oe}of k)W.existsSync(oe)&&(F[I]=ir(oe));try{let I=await We.post(`${s}/v1/local-runs/${u.testRunId}/results/${P.testCaseResultId}/screenshot-urls`,{stepIds:K,md5s:F},l);P.uploadUrls.screenshots=I.data.screenshots,P.screenshotS3Uris=I.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${K.length} screenshot URL(s) for "${m.title}"`)}catch(I){console.warn(`[reporter] Failed to get screenshot URLs for "${m.title}":`,I)}}),console.log("[reporter] [3/4] Uploading assets...");let w=(await or(e.tests,i.testUploadWorkers,async(m,S)=>{let P=d[S];if(!P){console.warn(`[reporter] No result slot found for test "${m.title}", skipping.`);return}let k=P.uploadUrls,K={},F={},I={},oe=0,we=[];if(m.attempts&&m.attempts.length>1)for(let A of m.attempts){let D=A.attemptNumber-1,Fs=A.attemptNumber===m.attempts.length;for(let ie of A.steps)ie.screenshot&&we.push({key:`attempt-${D}.${ie.stepId}`,filePath:Q(t,ie.screenshot),attemptIdx:D,assetType:"screenshot",originalStepId:ie.stepId});if(!Fs){if(A.videoPath){let ie=Q(t,A.videoPath);W.existsSync(ie)&&we.push({key:`attempt-${D}.__video__`,filePath:ie,attemptIdx:D,assetType:"video"})}if(A.tracePath){let ie=Q(t,A.tracePath);W.existsSync(ie)&&we.push({key:`attempt-${D}.__trace__`,filePath:ie,attemptIdx:D,assetType:"trace"})}}}else for(let A of m.steps)A.screenshot&&we.push({key:A.stepId,filePath:Q(t,A.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:A.stepId});await or(we,i.perTestAssetWorkers,async A=>{if(k.screenshots?.[A.key]&&W.existsSync(A.filePath))try{await ar(k.screenshots[A.key],A.filePath,c);let D=P.screenshotS3Uris[A.key];switch(A.assetType){case"screenshot":K[A.attemptIdx]||(K[A.attemptIdx]={}),K[A.attemptIdx][A.originalStepId]=D;break;case"video":F[A.attemptIdx]=D;break;case"trace":I[A.attemptIdx]=D;break}oe++}catch(D){console.warn(`[reporter] Asset upload failed for ${A.key}:`,D)}}),oe>0&&console.log(`[reporter] [3/4] Uploaded ${oe} asset(s) for "${m.title}"`);let j;if(m.videoPath&&k.video){let A=Q(t,m.videoPath);if(W.existsSync(A)){console.log(`[reporter] [3/4] Uploading video for "${m.title}"...`);try{await ar(k.video,A,c),j=P.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${m.title}"`)}catch(D){console.warn("[reporter] Video upload failed:",D)}}}let _e;if(m.tracePath&&k.trace){let A=Q(t,m.tracePath);if(W.existsSync(A)){console.log(`[reporter] [3/4] Uploading trace for "${m.title}"...`);try{await ar(k.trace,A,c),_e=P.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${m.title}"`)}catch(D){console.warn("[reporter] Trace upload failed:",D)}}}if(m.attempts&&m.attempts.length>1){let A=m.attempts.length-1;j&&(F[A]=j),_e&&(I[A]=_e)}else j&&(F[0]=j),_e&&(I[0]=_e);console.log(`[reporter] [3/4] Uploading report for "${m.title}"...`);let js=_a(m,K,F,I),lr=Buffer.from(JSON.stringify(js)),pr=cr(lr),ur=await We.post(`${s}/v1/local-runs/${u.testRunId}/results/${P.testCaseResultId}/report-url`,{md5:pr},l),Ds=ur.data.reportUrl,Us=ur.data.reportS3Uri;return await ys(Ds,lr,"application/json",pr,c),console.log(`[reporter] [3/4] Report uploaded for "${m.title}"`),{testCaseResultId:P.testCaseResultId,result:ba(m.status),durationMs:m.duration,startTime:m.startTime,endTime:m.endTime,error:m.error,reportS3Uri:Us,videoS3Uri:j,traceS3Uri:_e,metadata:{suiteName:m.suiteName,file:m.file,...m.retries!=null&&{retries:m.retries},...m.flaky&&{flaky:!0}}}})).filter(m=>!!m);console.log("[reporter] [4/4] Finalising run...");let b=va(e.tests);console.log(`[reporter] [4/4] Overall status: ${b}`);let y=await We.put(`${s}/v1/local-runs/${u.testRunId}/complete`,{status:b,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:w},l);console.log(`
1146
- Shiplight cloud report: ${Ia(y.data.reportUrl,s)}`)}function Ia(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${Ma(t)}${r}`}function Ma(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 bs=x(()=>{"use strict";zt()});var ks={};ue(ks,{buildGitHubSummary:()=>Ts,isReportToCloudEnabled:()=>_s,runReport:()=>Oa});import*as M from"fs";import*as $ from"path";async function Oa(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),n=e.includes("--github-summary");!r&&n&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Ca(e,t,n):await La(e,t)}function Ra(){try{return M.realpathSync("shiplight-report/latest")}catch{return"shiplight-report"}}async function La(e,t){let r=e.find(a=>!a.startsWith("--"))||Ra(),n=$.isAbsolute(r)?r:$.join(process.cwd(),r),s=$.join(n,"report-data.json");M.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(M.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=$.join(n,"index.html");if(M.writeFileSync(i,nr({...o,outputDir:n}),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await xs(o,n),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Ca(e,t,r){let n=$.join(process.cwd(),"shiplight-report"),s=e.findIndex(g=>g==="-o"||g==="--output");if(s!==-1&&e[s+1]){let g=e[s+1];n=$.isAbsolute(g)?g:$.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 u=e[g];if(o.has(u)){g++;continue}if(i.has(u))continue;let d=$.isAbsolute(u)?u:$.join(process.cwd(),u);a.push(d)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;M.mkdirSync($.join(n,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let u=a[g],d=`shard-${g}`,w=$.join(u,"report-data.json");if(!M.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let b;try{b=JSON.parse(M.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${d}: ${b.tests.length} tests from ${u}`),l+=b.totalDuration||0;let y=$.join(u,"screenshots");M.existsSync(y)&&Ss(y,$.join(n,"screenshots",d));let m=$.join(n,d);for(let S of b.tests){let P=[S,...S.attempts||[]];for(let k of P)Na(k.steps,d),k.videoPath&&vs(u,k.videoPath,m)&&(k.videoPath=`${d}/${k.videoPath}`),k.tracePath&&vs(u,k.tracePath,m)&&(k.tracePath=`${d}/${k.tracePath}`);c.push(S)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let f={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:Me};M.writeFileSync($.join(n,"report-data.json"),JSON.stringify(f,null,2),"utf-8");let h=$.join(n,"index.html");if(M.writeFileSync(h,nr({...f,outputDir:n}),"utf-8"),console.log(`
1147
- Merged ${c.length} tests from ${p} shards into: ${h}`),await xs(f,n),r&&ja(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function vs(e,t,r){let n=$.resolve(e,t);return n.startsWith($.resolve(e)+$.sep)?M.existsSync(n)?(M.mkdirSync(r,{recursive:!0}),M.copyFileSync(n,$.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Ss(e,t){M.mkdirSync(t,{recursive:!0});for(let r of M.readdirSync(e,{withFileTypes:!0})){let n=$.join(e,r.name),s=$.join(t,r.name);r.isDirectory()?Ss(n,s):M.copyFileSync(n,s)}}function Na(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function _s(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function xs(e,t){if(!_s())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let n=e.tests.map(o=>o.startTime).filter(o=>!!o),s=n.length>0?n.sort()[0]:e.timestamp??new Date().toISOString();try{await ws(e,t,s,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ut(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=$.join("tests",$.basename(t));return{name:e.title||$.basename(t),yamlPath:r}}function Ts(e){let t=e.filter(c=>!c.file.includes("auth.setup")),r=t.filter(c=>c.flaky),n=t.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0),s=t.filter(c=>c.status==="passed"&&!c.flaky),o=t.filter(c=>c.status!=="passed"),i=t.length,a=`## Test Results
1150
+ </html>`}var ms=x(()=>{"use strict"});import*as W from"fs";import*as Ie from"path";import{execFileSync as ga}from"child_process";import{createHash as ma}from"crypto";import{Agent as ya}from"http";import{Agent as wa}from"https";import We from"axios";function ys(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function he(...e){try{return ga("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function ba(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=W.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function va(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",n=process.env.GITHUB_RUN_ID??"",s=process.env.GITHUB_EVENT_NAME??"",i=ba().pull_request,a=i?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,h=process.env.SHIPLIGHT_PR_TITLE??i?.title,g=he("log","-1","--pretty=%s"),u=he("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:d,gitRepo:r,commitMessage:g,authorEmail:u,prNumber:p,prTitle:h,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:n,ciBuildUrl:n&&r?`${t}/${r}/actions/runs/${n}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`: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??"",r=process.env.CI_COMMIT_SHA??"",n=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:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:s,prNumber:n,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:n&&t?`${t}/-/merge_requests/${n}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",n=process.env.CIRCLE_PROJECT_REPONAME??"",s=r&&n?`${r}/${n}`: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,c=a?ys(a):void 0,l=he("log","-1","--pretty=%s"),p=he("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:l,authorEmail:p,prNumber:i,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=he("rev-parse","HEAD"),r=he("rev-parse","--abbrev-ref","HEAD"),n=he("log","-1","--pretty=%s"),s=he("log","-1","--pretty=%ae"),o=he("remote","get-url","origin"),i=o?ys(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:n,authorEmail:s,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function Sa(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 or(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function _a(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 xa(e,t){let r=new Map;for(let n of t){n.screenshotS3Uris={};let s=r.get(n.testCaseName);s?s.push(n):r.set(n.testCaseName,[n])}return e.map(n=>r.get(n.title)?.shift())}function Q(e,t){return Ie.isAbsolute(t)?t:Ie.join(e,t)}function ws(e,t){let r={};for(let n of e){let s={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};n.type&&(s.type=n.type),n.code&&(s.code=n.code),n.startTime&&(s.startTime=n.startTime),n.autoHealed&&(s.autoHealed=n.autoHealed),n.healedAction&&(s.healedAction=n.healedAction),n.dismissedModalActions?.length&&(s.dismissedModalActions=n.dismissedModalActions),n.contextBefore&&(s.contextBefore=n.contextBefore),n.contextAfter&&(s.contextAfter=n.contextAfter),r[n.stepId]=s}return r}function Ta(e,t,r,n){let s=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let i=o.attemptNumber-1,a=t[i]??{},c={outcome:or(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ws(o.steps,a),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[i],traceS3Uri:n[i],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),s.push(c)}else s.push({outcome:or(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ws(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:n[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:or(e.status),flaky:e.flaky??!1,segments:s}}async function ir(e,t,r){let n=new Array(e.length),s=0;async function o(){for(;s<e.length;){let i=s++;n[i]=await r(e[i],i)}}return await Promise.all(Array.from({length:Math.min(t,e.length)},o)),n}function ka(e){let t=0,r=[];function n(){let s=r.shift();s?s():t-=1}return async function(o){t>=e?await new Promise(i=>{r.push(i)}):t+=1;try{return await o()}finally{n()}}}function Pa(e,t){if(!e)return{value:t,fromEnv:!1};let r=Number(e);return!Number.isInteger(r)||r<1?{value:t,fromEnv:!1}:{value:r,fromEnv:!0}}function Ea(){let e=Pa(process.env.SHIPLIGHT_CLOUD_MAX_PUT_UPLOADS,6);return{maxConcurrentPutUploads:e.value,maxConcurrentPutUploadsFromEnv:e.fromEnv,screenshotUrlRequestWorkers:8,testUploadWorkers:5,perTestAssetWorkers:10}}function Aa(e){let t={keepAlive:!0,keepAliveMsecs:1e3,maxSockets:e};return{httpAgent:new ya(t),httpsAgent:new wa(t)}}function $a(e,t){let r=t.maxConcurrentPutUploadsFromEnv?"env:SHIPLIGHT_CLOUD_MAX_PUT_UPLOADS":"default";return`[reporter] [config] Cloud upload settings for ${e} test(s): presignedPutCap=${t.maxConcurrentPutUploads} (${r}), screenshotUrlWorkers=${t.screenshotUrlRequestWorkers}, testUploadWorkers=${t.testUploadWorkers}, perTestAssetWorkers=${t.perTestAssetWorkers}`}async function Ia(e,t=3){for(let r=0;;r++)try{return await e()}catch(n){let s=n.code;if(!(s==="ECONNRESET"||s==="ETIMEDOUT"||s==="EPIPE"||s==="EAI_AGAIN"||s==="EADDRNOTAVAIL"||s==="EAGAIN")||r>=t)throw n;let i=Math.min(1e3*2**r,8e3);console.warn(`[reporter] Upload failed (${s}), retrying in ${i}ms (attempt ${r+1}/${t})...`),await new Promise(a=>setTimeout(a,i))}}function lr(e){return ma("md5").update(e).digest("base64")}function ar(e){return lr(W.readFileSync(e))}function Ma(e,t){let r=ka(e.maxConcurrentPutUploads);return(n,s,o)=>r(()=>We.put(n,s,{...o,...t}))}async function bs(e,t,r,n,s){await Ia(()=>s(e,t,{headers:{"Content-Type":r,"Content-MD5":n}}))}async function cr(e,t,r){let n=W.readFileSync(t),s=Ie.extname(t).toLowerCase(),i={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await bs(e,n,i,lr(n),r)}async function vs(e,t,r,n){let s=Se(n,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${n}`,"Content-Type":"application/json"},i=Ea(),a=Aa(Math.max(i.maxConcurrentPutUploads,i.screenshotUrlRequestWorkers,i.testUploadWorkers)),c=Ma(i,a),l={headers:o,...a},p=va(),d=e.tests.map(m=>{let S={testCaseName:m.title,testCaseBaseName:m.baseTitle,suiteName:m.suiteName,file:m.file,tags:m.tags,suiteTags:m.suiteTags,baseUrl:m.baseUrl,skip:m.skip,slow:m.slow,timeout:m.timeout,parameterSetName:m.parameterSetName,flaky:m.flaky,retries:m.retries};if(m.videoPath){let P=Q(t,m.videoPath);W.existsSync(P)&&(S.videoMd5=ar(P))}if(m.tracePath){let P=Q(t,m.tracePath);W.existsSync(P)&&(S.traceMd5=ar(P))}return S}),h=e.tests.length;console.log(`[reporter] Uploading ${h} test result(s) to Shiplight cloud...`),console.log($a(h,i)),console.log("[reporter] [1/4] Creating run record...");let u=(await We.post(`${s}/v1/local-runs`,{trigger:p.ciProvider,startTime:r,metadata:p,tests:d},l)).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let f=xa(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await ir(e.tests,i.screenshotUrlRequestWorkers,async(m,S)=>{let P=f[S];if(!P)return;let k=[];if(m.attempts&&m.attempts.length>1)for(let I of m.attempts){let oe=I.attemptNumber-1,we=I.attemptNumber===m.attempts.length;for(let j of I.steps)j.screenshot&&k.push({key:`attempt-${oe}.${j.stepId}`,filePath:Q(t,j.screenshot)});if(!we){if(I.videoPath){let j=Q(t,I.videoPath);W.existsSync(j)&&k.push({key:`attempt-${oe}.__video__`,filePath:j})}if(I.tracePath){let j=Q(t,I.tracePath);W.existsSync(j)&&k.push({key:`attempt-${oe}.__trace__`,filePath:j})}}}else for(let I of m.steps)I.screenshot&&k.push({key:I.stepId,filePath:Q(t,I.screenshot)});if(!k.length)return;let K=k.map(I=>I.key),H={};for(let{key:I,filePath:oe}of k)W.existsSync(oe)&&(H[I]=ar(oe));try{let I=await We.post(`${s}/v1/local-runs/${u.testRunId}/results/${P.testCaseResultId}/screenshot-urls`,{stepIds:K,md5s:H},l);P.uploadUrls.screenshots=I.data.screenshots,P.screenshotS3Uris=I.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${K.length} screenshot URL(s) for "${m.title}"`)}catch(I){console.warn(`[reporter] Failed to get screenshot URLs for "${m.title}":`,I)}}),console.log("[reporter] [3/4] Uploading assets...");let w=(await ir(e.tests,i.testUploadWorkers,async(m,S)=>{let P=f[S];if(!P){console.warn(`[reporter] No result slot found for test "${m.title}", skipping.`);return}let k=P.uploadUrls,K={},H={},I={},oe=0,we=[];if(m.attempts&&m.attempts.length>1)for(let A of m.attempts){let U=A.attemptNumber-1,Bs=A.attemptNumber===m.attempts.length;for(let ie of A.steps)ie.screenshot&&we.push({key:`attempt-${U}.${ie.stepId}`,filePath:Q(t,ie.screenshot),attemptIdx:U,assetType:"screenshot",originalStepId:ie.stepId});if(!Bs){if(A.videoPath){let ie=Q(t,A.videoPath);W.existsSync(ie)&&we.push({key:`attempt-${U}.__video__`,filePath:ie,attemptIdx:U,assetType:"video"})}if(A.tracePath){let ie=Q(t,A.tracePath);W.existsSync(ie)&&we.push({key:`attempt-${U}.__trace__`,filePath:ie,attemptIdx:U,assetType:"trace"})}}}else for(let A of m.steps)A.screenshot&&we.push({key:A.stepId,filePath:Q(t,A.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:A.stepId});await ir(we,i.perTestAssetWorkers,async A=>{if(k.screenshots?.[A.key]&&W.existsSync(A.filePath))try{await cr(k.screenshots[A.key],A.filePath,c);let U=P.screenshotS3Uris[A.key];switch(A.assetType){case"screenshot":K[A.attemptIdx]||(K[A.attemptIdx]={}),K[A.attemptIdx][A.originalStepId]=U;break;case"video":H[A.attemptIdx]=U;break;case"trace":I[A.attemptIdx]=U;break}oe++}catch(U){console.warn(`[reporter] Asset upload failed for ${A.key}:`,U)}}),oe>0&&console.log(`[reporter] [3/4] Uploaded ${oe} asset(s) for "${m.title}"`);let j;if(m.videoPath&&k.video){let A=Q(t,m.videoPath);if(W.existsSync(A)){console.log(`[reporter] [3/4] Uploading video for "${m.title}"...`);try{await cr(k.video,A,c),j=P.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${m.title}"`)}catch(U){console.warn("[reporter] Video upload failed:",U)}}}let _e;if(m.tracePath&&k.trace){let A=Q(t,m.tracePath);if(W.existsSync(A)){console.log(`[reporter] [3/4] Uploading trace for "${m.title}"...`);try{await cr(k.trace,A,c),_e=P.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${m.title}"`)}catch(U){console.warn("[reporter] Trace upload failed:",U)}}}if(m.attempts&&m.attempts.length>1){let A=m.attempts.length-1;j&&(H[A]=j),_e&&(I[A]=_e)}else j&&(H[0]=j),_e&&(I[0]=_e);console.log(`[reporter] [3/4] Uploading report for "${m.title}"...`);let Us=Ta(m,K,H,I),pr=Buffer.from(JSON.stringify(Us)),ur=lr(pr),dr=await We.post(`${s}/v1/local-runs/${u.testRunId}/results/${P.testCaseResultId}/report-url`,{md5:ur},l),Fs=dr.data.reportUrl,Hs=dr.data.reportS3Uri;return await bs(Fs,pr,"application/json",ur,c),console.log(`[reporter] [3/4] Report uploaded for "${m.title}"`),{testCaseResultId:P.testCaseResultId,result:Sa(m.status),durationMs:m.duration,startTime:m.startTime,endTime:m.endTime,error:m.error,reportS3Uri:Hs,videoS3Uri:j,traceS3Uri:_e,metadata:{suiteName:m.suiteName,file:m.file,...m.retries!=null&&{retries:m.retries},...m.flaky&&{flaky:!0}}}})).filter(m=>!!m);console.log("[reporter] [4/4] Finalising run...");let b=_a(e.tests);console.log(`[reporter] [4/4] Overall status: ${b}`);let y=await We.put(`${s}/v1/local-runs/${u.testRunId}/complete`,{status:b,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:w},l);console.log(`
1151
+ Shiplight cloud report: ${Oa(y.data.reportUrl,s)}`)}function Oa(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${Ra(t)}${r}`}function Ra(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 Ss=x(()=>{"use strict";zt()});var Es={};ue(Es,{buildGitHubSummary:()=>Ps,isReportToCloudEnabled:()=>Ts,runReport:()=>La});import*as M from"fs";import*as $ from"path";async function La(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),n=e.includes("--github-summary");!r&&n&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Da(e,t,n):await Na(e,t)}function Ca(){try{return M.realpathSync("shiplight-report/latest")}catch{return"shiplight-report"}}async function Na(e,t){let r=e.find(a=>!a.startsWith("--"))||Ca(),n=$.isAbsolute(r)?r:$.join(process.cwd(),r),s=$.join(n,"report-data.json");M.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(M.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=$.join(n,"index.html");if(M.writeFileSync(i,sr({...o,outputDir:n}),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await ks(o,n),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Da(e,t,r){let n=$.join(process.cwd(),"shiplight-report"),s=e.findIndex(g=>g==="-o"||g==="--output");if(s!==-1&&e[s+1]){let g=e[s+1];n=$.isAbsolute(g)?g:$.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 u=e[g];if(o.has(u)){g++;continue}if(i.has(u))continue;let f=$.isAbsolute(u)?u:$.join(process.cwd(),u);a.push(f)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;M.mkdirSync($.join(n,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let u=a[g],f=`shard-${g}`,w=$.join(u,"report-data.json");if(!M.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let b;try{b=JSON.parse(M.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${f}: ${b.tests.length} tests from ${u}`),l+=b.totalDuration||0;let y=$.join(u,"screenshots");M.existsSync(y)&&xs(y,$.join(n,"screenshots",f));let m=$.join(n,f);for(let S of b.tests){let P=[S,...S.attempts||[]];for(let k of P)ja(k.steps,f),k.videoPath&&_s(u,k.videoPath,m)&&(k.videoPath=`${f}/${k.videoPath}`),k.tracePath&&_s(u,k.tracePath,m)&&(k.tracePath=`${f}/${k.tracePath}`);c.push(S)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:Me};M.writeFileSync($.join(n,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let h=$.join(n,"index.html");if(M.writeFileSync(h,sr({...d,outputDir:n}),"utf-8"),console.log(`
1152
+ Merged ${c.length} tests from ${p} shards into: ${h}`),await ks(d,n),r&&Ua(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function _s(e,t,r){let n=$.resolve(e,t);return n.startsWith($.resolve(e)+$.sep)?M.existsSync(n)?(M.mkdirSync(r,{recursive:!0}),M.copyFileSync(n,$.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function xs(e,t){M.mkdirSync(t,{recursive:!0});for(let r of M.readdirSync(e,{withFileTypes:!0})){let n=$.join(e,r.name),s=$.join(t,r.name);r.isDirectory()?xs(n,s):M.copyFileSync(n,s)}}function ja(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function Ts(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function ks(e,t){if(!Ts())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let n=e.tests.map(o=>o.startTime).filter(o=>!!o),s=n.length>0?n.sort()[0]:e.timestamp??new Date().toISOString();try{await vs(e,t,s,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ut(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=$.join("tests",$.basename(t));return{name:e.title||$.basename(t),yamlPath:r}}function Ps(e){let t=e.filter(c=>!c.file.includes("auth.setup")),r=t.filter(c=>c.flaky),n=t.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0),s=t.filter(c=>c.status==="passed"&&!c.flaky),o=t.filter(c=>c.status!=="passed"),i=t.length,a=`## Test Results
1148
1153
 
1149
1154
  `;if(o.length===0&&r.length===0?a+=`\u2705 All ${i} tests passed
1150
1155
 
@@ -1173,9 +1178,9 @@ Merged ${c.length} tests from ${p} shards into: ${h}`),await xs(f,n),r&&ja(c),t)
1173
1178
  `;for(let l of c)a+=`- ${l}
1174
1179
  `;a+=`
1175
1180
  </details>
1176
- `}return a}function ja(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}M.appendFileSync(t,Ts(e)),console.log("GitHub step summary written.")}var Ps=x(()=>{"use strict";hs();bs();Ge()});var Es,As=x(()=>{"use strict";Es="0.1.84"});var $s={};ue($s,{runTranspile:()=>Ua});import*as dt from"path";import{glob as Da}from"glob";async function Ua(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),n=await Da(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});n.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of n.sort()){let c=dt.resolve(r,a),l=Mn(c,{version:Es,basePath:r});if(!l.valid){s++,console.log(`
1177
- \u2717 ${a}`);for(let f of l.errors)console.log(` ERROR: ${f}`);continue}i++;let p=dt.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let f of l.warnings)console.log(` WARNING: ${f}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
1178
- ${n.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var Is=x(()=>{"use strict";Ht();As()});var Rs={};ue(Rs,{runInspect:()=>Fa});import*as ft from"fs";import*as Ms from"path";async function Fa(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),n=e.find(i=>!i.startsWith("--"));n||(console.error("Error: no file specified"),process.exit(1));let s=Ms.resolve(process.cwd(),n);ft.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=ft.readFileSync(s,"utf-8");try{let i=Ce(o),a=Y(o);if(r)Ha(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${n}: ${i.message}`),process.exit(1)}}function Ha(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let n of r.tests){let s=n.skip?` [SKIP${typeof n.skip=="string"?`: ${n.skip}`:""}]`:"";console.log(` - ${n.name}: ${n.statements.length} statements${n.teardown?`, ${n.teardown.length} teardown`:""}${s}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Os(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Os(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let n=Os(r.statements??[]);t.drafts+=n.drafts,t.actions+=n.actions,t.steps+=n.steps}return t}var Ls=x(()=>{"use strict";ge()});var Cs=Bs((td,Ba)=>{Ba.exports={name:"shiplightai",version:"0.1.84",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test $(find src -name '*.test.ts' ! -name '*.e2e.test.ts')","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts","test:browser":"tsx --test $(find src -name '*.e2e.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:*",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"}});Ge();import Wa from"dotenv";Wa.config();dr();fr();function Ns(){console.log(`
1181
+ `}return a}function Ua(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}M.appendFileSync(t,Ps(e)),console.log("GitHub step summary written.")}var As=x(()=>{"use strict";ms();Ss();Ge()});var $s,Is=x(()=>{"use strict";$s="0.1.85"});var Ms={};ue(Ms,{runTranspile:()=>Ha});import*as dt from"path";import{glob as Fa}from"glob";async function Ha(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),n=await Fa(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});n.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of n.sort()){let c=dt.resolve(r,a),l=On(c,{version:$s,basePath:r});if(!l.valid){s++,console.log(`
1182
+ \u2717 ${a}`);for(let d of l.errors)console.log(` ERROR: ${d}`);continue}i++;let p=dt.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let d of l.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
1183
+ ${n.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var Os=x(()=>{"use strict";Ht();Is()});var Cs={};ue(Cs,{runInspect:()=>Ba});import*as ft from"fs";import*as Rs from"path";async function Ba(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),n=e.find(i=>!i.startsWith("--"));n||(console.error("Error: no file specified"),process.exit(1));let s=Rs.resolve(process.cwd(),n);ft.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=ft.readFileSync(s,"utf-8");try{let i=Ce(o),a=Y(o);if(r)Wa(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${n}: ${i.message}`),process.exit(1)}}function Wa(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let n of r.tests){let s=n.skip?` [SKIP${typeof n.skip=="string"?`: ${n.skip}`:""}]`:"";console.log(` - ${n.name}: ${n.statements.length} statements${n.teardown?`, ${n.teardown.length} teardown`:""}${s}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Ls(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Ls(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let n=Ls(r.statements??[]);t.drafts+=n.drafts,t.actions+=n.actions,t.steps+=n.steps}return t}var Ns=x(()=>{"use strict";ge()});var Ds=Gs((nd,Ga)=>{Ga.exports={name:"shiplightai",version:"0.1.85",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test $(find src -name '*.test.ts' ! -name '*.e2e.test.ts')","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts","test:browser":"tsx --test $(find src -name '*.e2e.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:*",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"}});Ge();import Ka from"dotenv";Ka.config();fr();hr();function js(){console.log(`
1179
1184
  Usage: shiplight <command> [options]
1180
1185
 
1181
1186
  Commands:
@@ -1197,5 +1202,5 @@ Examples:
1197
1202
  shiplight transpile
1198
1203
  shiplight transpile "tests/**/*.test.yaml"
1199
1204
  shiplight debug tests/login.test.yaml
1200
- `)}var ht=process.argv[2];switch(ht){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(Sr(),vr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Qn(),Zn));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(ps(),ls));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(Ps(),ks));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(Is(),$s));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(Ls(),Rs));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=Cs().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":Ns();break;default:ht&&console.error(`Unknown command: ${ht}
1201
- `),Ns(),process.exit(ht?1:0)}
1205
+ `)}var ht=process.argv[2];switch(ht){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(_r(),Sr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(es(),Qn));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(ds(),us));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(As(),Es));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(Os(),Ms));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(Ns(),Cs));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=Ds().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":js();break;default:ht&&console.error(`Unknown command: ${ht}
1206
+ `),js(),process.exit(ht?1:0)}