shiplightai 0.1.75 → 0.1.76

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 $s=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var Is=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ce=(e,t)=>{for(var r in t)$s(e,r,{get:t[r],enumerable:!0})};import*as D from"fs";import*as q from"path";import*as pt from"os";import{execFileSync as Ms}from"child_process";function dt(){return q.join(pt.homedir(),".shiplight","version-check.json")}function Cs(){return q.join(pt.homedir(),".shiplight","npm-prefix.json")}function Ns(e=Cs()){try{let r=D.readFileSync(e,"utf-8"),n=JSON.parse(r);if(typeof n.prefix=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<Rs)return n.prefix}catch{}let t;try{t=Ms("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{D.mkdirSync(q.dirname(e),{recursive:!0}),D.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function or(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Ns()),n=e.warn??(a=>console.warn(a)),s=e.error??(a=>console.error(a)),o=e.exit??(a=>process.exit(a));try{if(!t)return;let a=r();if(!a)return;let i,c;try{i=D.realpathSync(t),c=D.realpathSync(a)}catch{return}if(i.startsWith(c+q.sep)){s(`
4
+ var Is=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var Ms=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),ce=(e,t)=>{for(var r in t)Is(e,r,{get:t[r],enumerable:!0})};import*as j from"fs";import*as q from"path";import*as ut from"os";import{execFileSync as Os}from"child_process";function ft(){return q.join(ut.homedir(),".shiplight","version-check.json")}function Ns(){return q.join(ut.homedir(),".shiplight","npm-prefix.json")}function Ds(e=Ns()){try{let r=j.readFileSync(e,"utf-8"),n=JSON.parse(r);if(typeof n.prefix=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<Cs)return n.prefix}catch{}let t;try{t=Os("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{j.mkdirSync(q.dirname(e),{recursive:!0}),j.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function ir(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Ds()),n=e.warn??(a=>console.warn(a)),s=e.error??(a=>console.error(a)),o=e.exit??(a=>process.exit(a));try{if(!t)return;let a=r();if(!a)return;let i,c;try{i=j.realpathSync(t),c=j.realpathSync(a)}catch{return}if(i.startsWith(c+q.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=q.join(c,"lib","node_modules","shiplightai"),d=q.join(c,"node_modules","shiplightai"),f=D.existsSync(l)?l:D.existsSync(d)?d:null;f&&n(`
13
+ `),o(1);return}let l=q.join(c,"lib","node_modules","shiplightai"),d=q.join(c,"node_modules","shiplightai"),f=j.existsSync(l)?l:j.existsSync(d)?d:null;f&&n(`
14
14
  \x1B[33m\u26A0 A global shiplightai install was detected at ${f}.
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=dt()){try{let t=D.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<Ls)return r}catch{}return null}function Ds(e,t=dt()){try{D.mkdirSync(q.dirname(t),{recursive:!0}),D.writeFileSync(t,JSON.stringify(e))}catch{}}async function Fs(){try{let e=await fetch(Os,{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 Us(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),[a,i]=r(t),c=n(s),l=n(a),d=Math.max(c.length,l.length);for(let f=0;f<d;f++){let h=c[f]??0,g=l[f]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!i)}async function ir(e={}){let t=e.runningVersion??lt,r=e.cwd??process.cwd(),n=e.cacheFile??dt(),s=e.fetchLatest??Fs,o=e.env??process.env,a=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!D.existsSync(q.join(r,"package-lock.json")))return;let i=null,c=js(n);if(c)i=c.latest;else{try{i=await s()}catch{i=null}i&&Ds({latest:i,fetchedAt:Date.now()},n)}i&&Us(t,i)&&a(`
17
+ `)}catch{}}function js(e=ft()){try{let t=j.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<Rs)return r}catch{}return null}function Fs(e,t=ft()){try{j.mkdirSync(q.dirname(t),{recursive:!0}),j.writeFileSync(t,JSON.stringify(e))}catch{}}async function Us(){try{let e=await fetch(Ls,{headers:{Accept:"application/json"},signal:AbortSignal.timeout(3e3)});if(!e.ok)return null;let t=await e.json();return typeof t.version=="string"?t.version:null}catch{return null}}function Hs(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),[a,i]=r(t),c=n(s),l=n(a),d=Math.max(c.length,l.length);for(let f=0;f<d;f++){let h=c[f]??0,g=l[f]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!i)}async function ar(e={}){let t=e.runningVersion??dt,r=e.cwd??process.cwd(),n=e.cacheFile??ft(),s=e.fetchLatest??Us,o=e.env??process.env,a=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!j.existsSync(q.join(r,"package-lock.json")))return;let i=null,c=js(n);if(c)i=c.latest;else{try{i=await s()}catch{i=null}i&&Fs({latest:i,fetchedAt:Date.now()},n)}i&&Hs(t,i)&&a(`
18
18
  \x1B[33m\u26A0 shiplightai ${i} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var lt,Pe,Os,Ls,Rs,je=_(()=>{"use strict";lt="0.1.75",Pe=lt!=="dev"?lt:void 0,Os="https://registry.npmjs.org/shiplightai/latest",Ls=3600*1e3,Rs=10080*60*1e3});import*as C from"fs";import*as M from"path";import{builtinModules as Hs}from"node:module";function Ee(e){let t;try{t=C.lstatSync(e)}catch(r){return r.code==="ENOENT"?"absent":"inaccessible"}if(t.isSymbolicLink())try{return C.statSync(e).isDirectory()?"directory":"non-directory"}catch(r){return r.code==="ENOENT"?"broken-symlink":"inaccessible"}return t.isDirectory()?"directory":"non-directory"}function cr(e){let t=Ee(e);return t==="directory"||t==="non-directory"}function ft(e,t){try{return C.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 Xs(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?Js.has(n)?{name:t,original:e}:n.length>Ys?{name:t,original:e}:{name:n,original:e}:{name:t,original:e}}function qs(e){let t=Ee(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 Qs(e){let t=M.join(e,"package.json"),r;try{r=C.statSync(t)}catch(a){return a.code==="ENOENT"?"absent":"unknown"}if(!r.isFile()||r.size>Zs)return"unknown";let n;try{n=C.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 ht(e){return e.some(t=>!lr.has(t))}function pr(e,t){let r=M.relative(e,t);return r===""?!0:!(r===".."||r.startsWith(".."+M.sep)||M.isAbsolute(r))}function to(e,t,r,n){let s=Ee(e);if(s==="absent")return;let o=null;try{o=C.lstatSync(e)}catch{}let a=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=a?"a symlink that resolves to a directory":"a directory",l=a?"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(!a||n==="skip")return;let i=ft(e,`leaf symlink at ${r}`);if(!pr(t,i))throw new Error(`Cannot scaffold ${r}: ${e} is a symlink that resolves to ${i}, which is outside the project root ${t}. Writes through this link would escape the project.`)}function gt(e){let t=M.resolve(e.projectPath),r=Xs(e.projectName??M.basename(t)),n=r.name;qs(t);let s=[];if(Ee(t)==="directory")try{s=C.readdirSync(t)}catch{}C.mkdirSync(t,{recursive:!0});let o=ft(t,"project root"),a=ar.split(`
21
- `).map(u=>u.trim()).filter(u=>u.length>0&&!u.startsWith("#")),i=eo.map(u=>({rel:u,abs:M.join(t,u)})).filter(u=>cr(u.abs)),c=Qs(t),l=i.length===0&&c==="commonjs",d=[];if(d.push({kind:"agent_merge",relPath:"package.json",content:Bs.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 dt,Pe,Ls,Rs,Cs,Fe=_(()=>{"use strict";dt="0.1.76",Pe=dt!=="dev"?dt:void 0,Ls="https://registry.npmjs.org/shiplightai/latest",Rs=3600*1e3,Cs=10080*60*1e3});import*as C from"fs";import*as O from"path";import{builtinModules as Bs}from"node:module";function Ae(e){let t;try{t=C.lstatSync(e)}catch(r){return r.code==="ENOENT"?"absent":"inaccessible"}if(t.isSymbolicLink())try{return C.statSync(e).isDirectory()?"directory":"non-directory"}catch(r){return r.code==="ENOENT"?"broken-symlink":"inaccessible"}return t.isDirectory()?"directory":"non-directory"}function lr(e){let t=Ae(e);return t==="directory"||t==="non-directory"}function gt(e,t){try{return C.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 qs(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?Ys.has(n)?{name:t,original:e}:n.length>Xs?{name:t,original:e}:{name:n,original:e}:{name:t,original:e}}function Zs(e){let t=Ae(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 eo(e){let t=O.join(e,"package.json"),r;try{r=C.statSync(t)}catch(a){return a.code==="ENOENT"?"absent":"unknown"}if(!r.isFile()||r.size>Qs)return"unknown";let n;try{n=C.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 mt(e){return e.some(t=>!pr.has(t))}function dr(e,t){let r=O.relative(e,t);return r===""?!0:!(r===".."||r.startsWith(".."+O.sep)||O.isAbsolute(r))}function ro(e,t,r,n){let s=Ae(e);if(s==="absent")return;let o=null;try{o=C.lstatSync(e)}catch{}let a=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=a?"a symlink that resolves to a directory":"a directory",l=a?"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(!a||n==="skip")return;let i=gt(e,`leaf symlink at ${r}`);if(!dr(t,i))throw new Error(`Cannot scaffold ${r}: ${e} is a symlink that resolves to ${i}, which is outside the project root ${t}. Writes through this link would escape the project.`)}function yt(e){let t=O.resolve(e.projectPath),r=qs(e.projectName??O.basename(t)),n=r.name;Zs(t);let s=[];if(Ae(t)==="directory")try{s=C.readdirSync(t)}catch{}C.mkdirSync(t,{recursive:!0});let o=gt(t,"project root"),a=cr.split(`
21
+ `).map(u=>u.trim()).filter(u=>u.length>0&&!u.startsWith("#")),i=to.map(u=>({rel:u,abs:O.join(t,u)})).filter(u=>lr(u.abs)),c=eo(t),l=i.length===0&&c==="commonjs",d=[];if(d.push({kind:"agent_merge",relPath:"package.json",content:Gs.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.`}}),i.length>0)for(let u of i)d.push({kind:"agent_merge",relPath:u.rel,content:ut,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?d.push({kind:"agent_merge",relPath:"playwright.config.ts",content:ut,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.'}}):d.push({kind:"agent_merge",relPath:"playwright.config.ts",content:ut,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."}});d.push({kind:"agent_merge",relPath:".gitignore",content:ar,merge:{strategy:"append_missing_lines",linesToEnsure:[...a],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."}}),d.push({kind:"agent_merge",relPath:".env.example",content:Gs,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."}}),d.push({kind:"agent_merge",relPath:".mcp.json",content:Ws,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."}}),d.push({kind:"skip",relPath:"auth/example.login.ts",content:Ks}),d.push({kind:"skip",relPath:"environments/example.env.yaml",content:Vs}),d.push({kind:"skip",relPath:"tests/example.test.yaml",content:zs});for(let u of d){let p=M.join(t,u.relPath),w=M.dirname(p);for(;;){let y=M.relative(t,w);if(y===""||y===".."||y.startsWith(".."+M.sep)||M.isAbsolute(y))break;let m=Ee(w);if(m!=="absent"&&m!=="directory"){let x=` (project root: ${o}). If multiple plans share this ancestor, the same issue blocks them all.`;throw new Error(m==="broken-symlink"?`Cannot scaffold ${u.relPath}: parent path ${w} is a broken symlink.${x}`:m==="inaccessible"?`Cannot scaffold ${u.relPath}: cannot stat parent path ${w} (permission denied or other I/O error).${x}`:`Cannot scaffold ${u.relPath}: parent path ${w} exists and is not a directory.${x}`)}if(m==="directory"){let x=ft(w,`parent of ${u.relPath}`);if(!pr(o,x))throw new Error(`Cannot scaffold ${u.relPath}: parent path ${w} resolves to ${x}, which is outside the project root ${o}.`)}let b=M.dirname(w);if(b===w)break;w=b}to(p,o,u.relPath,u.kind)}let f=[],h=[],g=[];for(let u of d){let p=M.join(t,u.relPath),w=cr(p),y=u.kind==="agent_merge"&&u.forceMerge===!0;if(!w&&!y){C.mkdirSync(M.dirname(p),{recursive:!0}),C.writeFileSync(p,u.content),f.push(u.relPath);continue}if(u.kind==="skip"){h.push(u.relPath);continue}let m={path:u.relPath,absPath:p,mergeStrategy:u.merge.strategy,template:u.content,instructions:u.merge.instructions,humanSummary:u.merge.humanSummary};u.merge.mergeKey!==void 0&&(m.mergeKey=u.merge.mergeKey),u.merge.linesToEnsure!==void 0&&(m.linesToEnsure=u.merge.linesToEnsure),g.push(m)}return{projectPath:t,projectName:n,originalProjectName:r.original,filesCreated:f,filesSkipped:h,filesNeedingAgentMerge:g,preExistingTopLevelEntries:s}}var Bs,ut,ar,Gs,Ws,Ks,Vs,zs,Js,Ys,Zs,eo,lr,dr=_(()=>{"use strict";Bs=`{
27
+ Use Read + Edit to perform a minimal in-place merge \u2014 do not rewrite the whole file.`}}),i.length>0)for(let u of i)d.push({kind:"agent_merge",relPath:u.rel,content:ht,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?d.push({kind:"agent_merge",relPath:"playwright.config.ts",content:ht,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.'}}):d.push({kind:"agent_merge",relPath:"playwright.config.ts",content:ht,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."}});d.push({kind:"agent_merge",relPath:".gitignore",content:cr,merge:{strategy:"append_missing_lines",linesToEnsure:[...a],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."}}),d.push({kind:"agent_merge",relPath:".env.example",content:Ws,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."}}),d.push({kind:"agent_merge",relPath:".mcp.json",content:Ks,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."}}),d.push({kind:"skip",relPath:"auth/example.login.ts",content:Vs}),d.push({kind:"skip",relPath:"environments/example.env.yaml",content:zs}),d.push({kind:"skip",relPath:"tests/example.test.yaml",content:Js});for(let u of d){let p=O.join(t,u.relPath),w=O.dirname(p);for(;;){let y=O.relative(t,w);if(y===""||y===".."||y.startsWith(".."+O.sep)||O.isAbsolute(y))break;let m=Ae(w);if(m!=="absent"&&m!=="directory"){let x=` (project root: ${o}). If multiple plans share this ancestor, the same issue blocks them all.`;throw new Error(m==="broken-symlink"?`Cannot scaffold ${u.relPath}: parent path ${w} is a broken symlink.${x}`:m==="inaccessible"?`Cannot scaffold ${u.relPath}: cannot stat parent path ${w} (permission denied or other I/O error).${x}`:`Cannot scaffold ${u.relPath}: parent path ${w} exists and is not a directory.${x}`)}if(m==="directory"){let x=gt(w,`parent of ${u.relPath}`);if(!dr(o,x))throw new Error(`Cannot scaffold ${u.relPath}: parent path ${w} resolves to ${x}, which is outside the project root ${o}.`)}let b=O.dirname(w);if(b===w)break;w=b}ro(p,o,u.relPath,u.kind)}let f=[],h=[],g=[];for(let u of d){let p=O.join(t,u.relPath),w=lr(p),y=u.kind==="agent_merge"&&u.forceMerge===!0;if(!w&&!y){C.mkdirSync(O.dirname(p),{recursive:!0}),C.writeFileSync(p,u.content),f.push(u.relPath);continue}if(u.kind==="skip"){h.push(u.relPath);continue}let m={path:u.relPath,absPath:p,mergeStrategy:u.merge.strategy,template:u.content,instructions:u.merge.instructions,humanSummary:u.merge.humanSummary};u.merge.mergeKey!==void 0&&(m.mergeKey=u.merge.mergeKey),u.merge.linesToEnsure!==void 0&&(m.linesToEnsure=u.merge.linesToEnsure),g.push(m)}return{projectPath:t,projectName:n,originalProjectName:r.original,filesCreated:f,filesSkipped:h,filesNeedingAgentMerge:g,preExistingTopLevelEntries:s}}var Gs,ht,cr,Ws,Ks,Vs,zs,Js,Ys,Xs,Qs,to,pr,ur=_(()=>{"use strict";Gs=`{
28
28
  "name": "{{name}}",
29
29
  "type": "module",
30
30
  "engines": {
@@ -38,7 +38,7 @@ Use Read + Edit to perform a minimal in-place merge \u2014 do not rewrite the wh
38
38
  "shiplightai": "latest"
39
39
  }
40
40
  }
41
- `,ut=`import { defineConfig, shiplightConfig } from 'shiplightai';
41
+ `,ht=`import { defineConfig, shiplightConfig } from 'shiplightai';
42
42
 
43
43
  export default defineConfig({
44
44
  ...shiplightConfig(),
@@ -56,13 +56,13 @@ export default defineConfig({
56
56
  trace: 'on',
57
57
  },
58
58
  });
59
- `,ar=`node_modules/
59
+ `,cr=`node_modules/
60
60
  test-results/
61
61
  shiplight-report/
62
62
  .shiplight/
63
63
  .env
64
64
  *.yaml.spec.ts
65
- .auth`,Gs=`# Shiplight API token
65
+ .auth`,Ws=`# Shiplight API token
66
66
  # Get yours at https://nova.shiplight.ai/api-tokens
67
67
  # If set, no AI provider API key required.
68
68
  # SHIPLIGHT_API_TOKEN=
@@ -74,7 +74,7 @@ shiplight-report/
74
74
 
75
75
  # Optional: override the default AI model.
76
76
  # WEB_AGENT_MODEL=claude-sonnet-4-6
77
- `,Ws=`{
77
+ `,Ks=`{
78
78
  "mcpServers": {
79
79
  "shiplight": {
80
80
  "command": "npx",
@@ -87,7 +87,7 @@ shiplight-report/
87
87
  }
88
88
  }
89
89
  }
90
- }`,Ks=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
90
+ }`,Vs=`// This is an example auth fixture. Copy and adapt it for your actual login flow.
91
91
  import path from 'path';
92
92
  import fs from 'fs/promises';
93
93
  import { chromium } from '@playwright/test';
@@ -130,13 +130,13 @@ export async function login(args: Record<string, unknown> = {}): Promise<string>
130
130
 
131
131
  return sessionPath;
132
132
  }
133
- `,Vs=`# This is an example of environment. Don't use it.
133
+ `,zs=`# This is an example of environment. Don't use it.
134
134
  name: example
135
135
  url: https://staging.example.com
136
136
  accounts:
137
137
  - username: test-user-1@example.com
138
138
  password: EXAMPLE_TEST_USER_1_PASSWORD
139
- 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,zs=`goal: Verify the Shiplight homepage links to the quick-start docs
139
+ 2fa_secret: EXAMPLE_TEST_USER_1_2FA_SECRET`,Js=`goal: Verify the Shiplight homepage links to the quick-start docs
140
140
  base_url: https://www.shiplight.ai
141
141
  statements:
142
142
  - URL: /
@@ -150,7 +150,7 @@ statements:
150
150
 
151
151
  - VERIFY: The browser lands on the docs site
152
152
  js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
153
- `;Js=new Set(["node_modules","favicon.ico","node","npm","js","test","sea",...Hs]),Ys=214;Zs=1048576;eo=["playwright.config.ts","playwright.config.js","playwright.config.mjs","playwright.config.cjs"],lr=new Set([".git",".gitkeep",".DS_Store","Thumbs.db",".idea",".vscode",".husky","node_modules",".next",".turbo",".cache",".pnpm-store"])});var ur=_(()=>{"use strict";dr()});var fr={};ce(fr,{runCreate:()=>no});import*as Fe from"path";function ro(e){return/^[A-Za-z0-9_./~+-]+$/.test(e)?e:`'${e.replace(/'/g,"'\\''")}'`}function De(){console.log(`
153
+ `;Ys=new Set(["node_modules","favicon.ico","node","npm","js","test","sea",...Bs]),Xs=214;Qs=1048576;to=["playwright.config.ts","playwright.config.js","playwright.config.mjs","playwright.config.cjs"],pr=new Set([".git",".gitkeep",".DS_Store","Thumbs.db",".idea",".vscode",".husky","node_modules",".next",".turbo",".cache",".pnpm-store"])});var fr=_(()=>{"use strict";ur()});var hr={};ce(hr,{runCreate:()=>so});import*as He from"path";function no(e){return/^[A-Za-z0-9_./~+-]+$/.test(e)?e:`'${e.replace(/'/g,"'\\''")}'`}function Ue(){console.log(`
154
154
  Usage: shiplight create <path> [options]
155
155
 
156
156
  Scaffold a Shiplight test project at <path>. Writes package.json,
@@ -167,11 +167,11 @@ Examples:
167
167
  shiplight create ./my-tests
168
168
  shiplight create ./my-tests --name acme-e2e
169
169
  shiplight create . # scaffold into the current repo
170
- `)}async function no(e){let t,r;for(let u=0;u<e.length;u++){let p=e[u];if(p==="--help"||p==="-h"){De();return}else p==="--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.
170
+ `)}async function so(e){let t,r;for(let u=0;u<e.length;u++){let p=e[u];if(p==="--help"||p==="-h"){Ue();return}else p==="--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
171
  npm package names must be lowercase.
172
172
  Use only lowercase letters, digits, hyphens, underscores, and dots;
173
- must start with a lowercase letter or digit.`),process.exit(1))):p.startsWith("--")?(console.error(`Error: Unknown option: ${p}`),De(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${p}`),De(),process.exit(1)):t=p}t||(console.error(`Error: missing required <path> argument
174
- `),De(),process.exit(1));let n=Fe.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=gt({projectPath:n,projectName:r})}catch(u){let p=u instanceof Error?u.message:String(u);console.error(`Error: ${p}`),process.exit(1)}let a=ht(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(`
173
+ must start with a lowercase letter or digit.`),process.exit(1))):p.startsWith("--")?(console.error(`Error: Unknown option: ${p}`),Ue(),process.exit(1)):t?(console.error(`Error: Unexpected argument: ${p}`),Ue(),process.exit(1)):t=p}t||(console.error(`Error: missing required <path> argument
174
+ `),Ue(),process.exit(1));let n=He.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=yt({projectPath:n,projectName:r})}catch(u){let p=u instanceof Error?u.message:String(u);console.error(`Error: ${p}`),process.exit(1)}let a=mt(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
175
  ${a?"Added Shiplight files to existing project at":"Scaffolded Shiplight test project at"} ${o.projectPath}
176
176
  `),console.log(` Name: ${o.projectName}`),o.filesCreated.length>0){console.log(`
177
177
  Files created:`);for(let u of o.filesCreated)console.log(` + ${u}`)}if(o.filesSkipped.length>0){console.log(`
@@ -179,7 +179,7 @@ ${a?"Added Shiplight files to existing project at":"Scaffolded Shiplight test pr
179
179
  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
180
  Open each file and apply the change yourself, or let your coding
181
181
  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"),d=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&&d?"# 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=ro(Fe.relative(process.cwd(),o.projectPath)||"."),g=o.filesNeedingAgentMerge.length>0?`
182
+ content + per-file merge_strategy in its response.`)}let c=o.preExistingTopLevelEntries.includes(".env"),l=o.filesNeedingAgentMerge.some(u=>u.path===".env.example"),d=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&&d?"# 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=no(He.relative(process.cwd(),o.projectPath)||"."),g=o.filesNeedingAgentMerge.length>0?`
183
183
  # resolve the merge-required files listed above first`:"";console.log(`
184
184
  Next steps:
185
185
 
@@ -188,7 +188,7 @@ Next steps:
188
188
  npm install
189
189
  npx playwright install chromium
190
190
  npx shiplight test
191
- `)}var hr=_(()=>{"use strict";ur()});import*as Z from"fs";import*as le from"path";function gr(e){let t=le.resolve(e);for(;;){if(Z.existsSync(le.join(t,"package.json"))||Z.existsSync(le.join(t,".shiplight")))return t;let r=le.dirname(t);if(r===t)break;t=r}return null}function mr(e){return le.join(e,so)}function yr(e,t){return le.join(mr(e),`${oo}${t}.json`)}function wr(e,t){let r=gr(t);if(!r)return;let n=mr(r);Z.mkdirSync(n,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};Z.writeFileSync(yr(r,e),JSON.stringify(s),"utf-8")}function br(e,t){let r=gr(t);if(r)try{Z.unlinkSync(yr(r,e))}catch{}}var so,oo,vr=_(()=>{"use strict";so=".shiplight/run",oo="debug-"});import*as _r from"node:net";function Sr(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 io(e){return await Sr(e,"127.0.0.1")?Sr(e,"::1"):!1}async function xr(e,t){for(let r=e;r<e+t;r++)if(await io(r))return r;return null}var Tr=_(()=>{"use strict"});var Pr={};ce(Pr,{findPlaywrightConfig:()=>Ae,makeIdempotentFileCleaner:()=>kr,spawnPlaywrightProcess:()=>mt});import*as pe from"fs";import*as ne from"path";import{spawn as ao}from"child_process";import{parse as co}from"yaml";function kr(e){let t=!1;return()=>{if(!t){t=!0;try{pe.unlinkSync(e)}catch{}}}}function Ae(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=ne.resolve(e);for(;;){for(let s of t){let o=ne.join(r,s);if(pe.existsSync(o))return o}let n=ne.dirname(r);if(n===r)break;r=n}return null}function lo(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
191
+ `)}var gr=_(()=>{"use strict";fr()});import*as Z from"fs";import*as le from"path";function mr(e){let t=le.resolve(e);for(;;){if(Z.existsSync(le.join(t,"package.json"))||Z.existsSync(le.join(t,".shiplight")))return t;let r=le.dirname(t);if(r===t)break;t=r}return null}function yr(e){return le.join(e,oo)}function wr(e,t){return le.join(yr(e),`${io}${t}.json`)}function br(e,t){let r=mr(t);if(!r)return;let n=yr(r);Z.mkdirSync(n,{recursive:!0});let s={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};Z.writeFileSync(wr(r,e),JSON.stringify(s),"utf-8")}function vr(e,t){let r=mr(t);if(r)try{Z.unlinkSync(wr(r,e))}catch{}}var oo,io,Sr=_(()=>{"use strict";oo=".shiplight/run",io="debug-"});import*as xr from"node:net";function _r(e,t){return new Promise(r=>{let n=xr.createServer();n.once("error",s=>{r(s.code!=="EADDRINUSE")}),n.once("listening",()=>{n.close(()=>r(!0))}),n.listen(e,t)})}async function ao(e){return await _r(e,"127.0.0.1")?_r(e,"::1"):!1}async function Tr(e,t){for(let r=e;r<e+t;r++)if(await ao(r))return r;return null}var kr=_(()=>{"use strict"});var Pr={};ce(Pr,{findPlaywrightConfig:()=>$e,makeIdempotentFileCleaner:()=>Er,spawnPlaywrightProcess:()=>wt});import*as pe from"fs";import*as ne from"path";import{spawn as co}from"child_process";import{parse as lo}from"yaml";function Er(e){let t=!1;return()=>{if(!t){t=!0;try{pe.unlinkSync(e)}catch{}}}}function $e(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=ne.resolve(e);for(;;){for(let s of t){let o=ne.join(r,s);if(pe.existsSync(o))return o}let n=ne.dirname(r);if(n===r)break;r=n}return null}function po(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
192
  import { test } from 'shiplightai/fixture';
193
193
  ${`
194
194
  test.use(${JSON.stringify(s)});
@@ -212,30 +212,30 @@ test('__shiplight_debug__', async ({ page, agent }) => {
212
212
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
213
213
  await new Promise(() => {});
214
214
  });
215
- `}async function po(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 mt(e){let{yamlFilePath:t,configPath:r,tempSuffix:n="",headed:s}=e,o=ne.dirname(r),a=await po(16174),i;if(!pe.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let b=co(pe.readFileSync(t,"utf-8"));b?.use&&typeof b.use=="object"&&!Array.isArray(b.use)&&(i=b.use),b?.base_url&&!i?.baseURL&&(i={...i,baseURL:b.base_url})}catch(b){console.error("[debugger] Could not parse YAML for `use` block:",b)}let c=ne.dirname(ne.resolve(t)),l=n?`-${n}`:"",d=ne.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),f=lo(t,a,i,o);pe.writeFileSync(d,f);let h=kr(d),g=["playwright","test",d,...s?["--headed"]:[]],u=ao("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});u.stdout?.on("data",b=>{process.stderr.write(b)}),u.stderr?.on("data",b=>{process.stderr.write(b)});let p=()=>{u.killed||u.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),u.on("close",b=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),h(),b!==0&&b!==null&&console.error(`[debugger] Playwright process exited with code ${b}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let w=["127.0.0.1","::1"];async function y(b){try{let x=b.includes(":")?`[${b}]`:b,T=await fetch(`http://${x}:${a}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let m=null;for(let b=0;b<180;b++){if(u.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${u.exitCode} before sandbox was ready`);for(let x of w)if(await y(x)){m=x;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${a}`);break}if(b===179)throw p(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(x=>setTimeout(x,1e3))}if(!m)throw p(),h(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:m,pid:u.pid??0,cleanup:async()=>{p(),h()}}}var yt=_(()=>{"use strict"});var Er=_(()=>{"use strict"});var Ar=_(()=>{"use strict"});var $r=_(()=>{"use strict"});import{v4 as ja}from"uuid";var Ir=_(()=>{"use strict"});import{z as v}from"zod";var Mr,wt,Or,we,Lr,Rr,Cr,G,Nr,Ue,bt,vt=_(()=>{"use strict";Mr=v.enum(["JS_CODE","AI_MODE"]),wt=v.object({type:Mr,expression:v.string()}),Or=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=v.object({uid:v.string(),type:Or,comment:v.string().optional()}),Lr=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(),Rr=we.extend({type:v.literal("DRAFT"),description:v.string()}),Cr=we.extend({type:v.literal("ACTION"),description:v.string(),action_entity:Lr.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),G=v.lazy(()=>v.union([Rr,Cr,we.extend({type:v.literal("STEP"),description:v.string().optional().default(""),statements:v.array(G),reference_id:v.number().optional(),template_path:v.string().optional(),template_params:v.record(v.string()).optional()}),we.extend({type:v.literal("IF_ELSE"),description:v.string().optional(),condition:wt,then:v.array(G),else:v.array(G).optional()}),we.extend({type:v.literal("WHILE_LOOP"),description:v.string().optional(),condition:wt,body:v.array(G),timeout_ms:v.number().optional()})])),Nr=v.object({name:v.string(),statements:v.array(G),teardown:v.array(G).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()}),Ue=v.object({tests:v.array(Nr).min(1),beforeAll:v.array(G).optional(),afterAll:v.array(G).optional(),beforeEach:v.array(G).optional(),afterEach:v.array(G).optional()}),bt=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(G).optional(),teardown:v.array(G).optional(),last_modified_at:v.string().optional(),testGroup:Ue.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as Ga,parse as Ur,parseAllDocuments as Wa,parseDocument as uo,Document as Hr,isMap as ve,isSeq as W}from"yaml";import{v4 as Q}from"uuid";function He(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(z)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(z)),r}function _t(e,t){if(e.testGroup)return Gr(e,t);let r=He(e,t),n=new Hr(r),s=n.contents?.get("tags",!0);return W(s)&&(s.flow=!0),e.comment&&(n.commentBefore=e.comment),jr(n,e.statements??[]),e.teardown&&jr(n,e.teardown,"teardown"),n.toString(Br)}function jr(e,t,r="statements"){let n=e.contents;if(!n||!ve(n))return;let s=n.get(r,!0);W(s)&&$e(s,t)}function $e(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),ve(s)){let o=s;if(n.type==="STEP"){let a=o.get("statements",!0);W(a)&&$e(a,n.statements)}else if(n.type==="IF_ELSE"){let a=o.get("THEN",!0);W(a)&&$e(a,n.then);let i=o.get("ELSE",!0);W(i)&&n.else&&$e(i,n.else)}else if(n.type==="WHILE_LOOP"){let a=o.get("DO",!0);W(a)&&$e(a,n.body)}}}}function Gr(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(z)),r.beforeEach&&r.beforeEach.length>0&&(s.beforeEach=r.beforeEach.map(z)),r.afterEach&&r.afterEach.length>0&&(s.afterEach=r.afterEach.map(z)),r.afterAll&&r.afterAll.length>0&&(s.afterAll=r.afterAll.map(z)),s.tests=r.tests.map(i=>{let c={name:i.name};return i.skip!==void 0&&(c.skip=i.skip),i.timeout!==void 0&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only!==void 0&&(c.only=i.only),i.slow!==void 0&&(c.slow=i.slow),c.statements=i.statements.map(z),i.teardown&&i.teardown.length>0&&(c.teardown=i.teardown.map(z)),c}),n.suite=s;let o=new Hr(n),a=o.contents?.get("tags",!0);return W(a)&&(a.flow=!0),o.toString(Br)}function z(e){switch(e.type){case"DRAFT":return fo(e);case"ACTION":return ho(e);case"STEP":return go(e);case"IF_ELSE":return mo(e);case"WHILE_LOOP":return yo(e)}}function fo(e){return{intent:e.description}}function ho(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:i,js:c}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:i};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let i=r?.code;if(typeof i=="string"&&i.trim()&&e.description)return{description:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let c={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let i=r?.seconds,l={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(l.seconds=i),l}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:i}}if(!e.action_entity)return{intent:e.description};let 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 a=e.action_entity.xpath;if(a&&(s.xpath=a),e.use_pure_vision&&(s.use_pure_vision=!0),n.kwargs&&Object.keys(n.kwargs).length>0)for(let[i,c]of Object.entries(n.kwargs))i!=="uid"&&(i==="statement"&&(t==="ai_action"||t==="ai_step")||(s[i]=c));return n.args&&n.args.length>0&&(s.args=n.args),s}function go(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(z)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function mo(e){let t={IF:Wr(e.condition),THEN:e.then.map(z)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(z)),t}function yo(e){let t={WHILE:Wr(e.condition),DO:e.body.map(z)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Wr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ie(e){try{let t=Ur(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 St(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(St);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]=St(o);return n}function K(e){if(e.length>Dr)throw new Error(`YAML input too large (${e.length} bytes, max ${Dr})`);let t=St(Ur(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return wo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:J(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=J(t.teardown));let n=bt.safeParse(r);if(!n.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(n.error.errors)}`);let s=n.data;return Be(e,s),s}function wo(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(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let c={name:i.name,statements:J(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(c.teardown=J(i.teardown)),i.skip!==void 0&&(c.skip=i.skip),typeof i.timeout=="number"&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only===!0&&(c.only=!0),i.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=J(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=J(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=J(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=J(t.afterEach));let o=Ue.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 J(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(bo)}function bo(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 vo(t);if("WHILE"in t)return So(t);if("STEP"in t)return _o(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:Q(),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:Q(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:Q(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:n}}}}}if("WAIT"in t){let r=t.WAIT,n=typeof t.seconds=="number"?t.seconds:3;return{uid:Q(),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:Q(),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:Q(),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 Fr({...n,action:"function",functionName:r})}if("action"in t)return Fr(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:Q(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Kr(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 vo(e){let t=Kr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let n={uid:Q(),type:"IF_ELSE",condition:t,then:J(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(n.else=J(e.ELSE)),n}function So(e){let t=Kr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let n={uid:Q(),type:"WHILE_LOOP",condition:t,body:J(r)};return typeof e.timeout_ms=="number"&&(n.timeout_ms=e.timeout_ms),n}function _o(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:Q(),type:"STEP",description:t,statements:J(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,a]of Object.entries(n))s[o]=String(a);r.template_params=s}return r}function Fr(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,a={};for(let[l,d]of Object.entries(e))xo.has(l)||(a[l]=d);t==="verify"&&typeof a.js=="string"&&(a.code=a.js,delete a.js),(t==="ai_action"||t==="ai_step")&&a.statement===void 0&&(a.statement=r);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};n&&(i.locator=n),s&&(i.xpath=s);let c={uid:Q(),type:"ACTION",description:r,action_entity:i};return o&&(c.use_pure_vision=!0),c}function Be(e,t){let r;try{r=uo(e)}catch{return}let n=r.contents;if(!n||!ve(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);W(o)&&t.statements&&be(o,t.statements);let a=s.get("teardown",!0);W(a)&&t.teardown&&be(a,t.teardown)}function be(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=e.items[r];n.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=n.commentBefore);let s=t[r];if(s.type==="STEP"&&ve(n)){let o=n.get("statements",!0);W(o)&&be(o,s.statements)}else if(s.type==="IF_ELSE"&&ve(n)){let o=n.get("THEN",!0);W(o)&&be(o,s.then);let a=n.get("ELSE",!0);W(a)&&s.else&&be(a,s.else)}else if(s.type==="WHILE_LOOP"&&ve(n)){let o=n.get("DO",!0);W(o)&&be(o,s.body)}}}var Br,Dr,xo,Ge=_(()=>{"use strict";vt();Br={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Dr=1024*1024;xo=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as Ya,stringify as Xa}from"yaml";var Vr=_(()=>{"use strict";Ge()});var xt,We,Ke=_(()=>{"use strict";xt=e=>{let t=[];switch(e.type){case"STEP":e.statements&&t.push({key:"statements",statements:e.statements});break;case"IF_ELSE":e.then&&t.push({key:"then",statements:e.then}),e.else&&t.push({key:"else",statements:e.else});break;case"WHILE_LOOP":e.body&&t.push({key:"body",statements:e.body});break}return t},We=e=>{let t=[],r=n=>{for(let s of n){t.push(s);let o=xt(s);for(let a of o)r(a.statements)}};return r(e),t}});function qr(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??"";Xr.has(s)||t++}return{action:t,draft:r}}function ko(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Yr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Po(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<=zr&&s<=Jr?null:`${r} non-blank line(s), ${s} await(s) \u2014 limits are ${zr} lines and ${Jr} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function Tt(e,t){let r=t?.coverageThreshold??To,n=[],s=[],o;try{o=K(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 a=[...We(o.statements??[]),...o.teardown?We(o.teardown):[]],{action:i,draft:c}=qr(a),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 a){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,p=u.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let w=u.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let y=ko(w);if(y){let m=u.description||p;n.push(`Invalid JS in "${m}": ${y}. ${l}`)}else if(p==="verify"){let m=Po(w);if(m){let b=u.description||p;n.push(`JS cache for "${b}" is too complex: ${m}`)}}}}}if(h.type==="IF_ELSE"){let u=h;if(u.condition.type==="JS_CODE"){let p=Yr(u.condition.expression);p&&n.push(`Invalid JS in IF condition "${u.condition.expression}": ${p}. ${l}`)}}if(h.type==="WHILE_LOOP"){let u=h;if(u.condition.type==="JS_CODE"){let p=Yr(u.condition.expression);p&&n.push(`Invalid JS in WHILE condition "${u.condition.expression}": ${p}. ${l}`)}}}let d=i+c,f=d>0?Math.round(i/d*100):0;return d>0&&f/100<r&&s.push(`Low action coverage: ${i}/${d} 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:d,action:i,draft:c,coverage:f}}}var To,zr,Jr,Xr,Zr=_(()=>{"use strict";Ge();Ke();To=.5,zr=5,Jr=3,Xr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Ve=_(()=>{"use strict"});var Qr=_(()=>{"use strict";Ve()});var en,Ao,tn,rn,Me,$o,Io,nn=_(()=>{"use strict";en=112,Ao=1080-en,tn={"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"}},rn={desktop:["Desktop Chrome","Desktop Chrome Medium Resolution","Desktop Chrome (Branded)","Desktop Chrome Medium Resolution (Branded)","Desktop Edge (Branded)","Desktop Edge Medium Resolution (Branded)","Desktop Safari"],mobile:["iPhone 15 Pro Max","iPhone 15 Pro","iPhone 15 Plus","iPhone 15","iPhone 14 Pro Max","iPhone 14 Pro","iPhone 14 Plus","iPhone 14","iPhone 13 Pro Max","iPhone 13 Pro","iPhone 13","iPhone 13 Mini","iPhone 12 Pro Max","iPhone 12 Pro","iPhone 12","iPhone 12 Mini","iPhone 11 Pro Max","iPhone 11 Pro","iPhone 11","iPhone XR","iPhone X","iPhone SE (3rd gen)","iPhone SE","iPhone 8 Plus","iPhone 8","iPhone 7 Plus","iPhone 7","iPhone 6 Plus","iPhone 6","Galaxy S24","Galaxy A55","Galaxy S9+","Galaxy S8","Galaxy S5","Galaxy Note 3","Galaxy Note II","Galaxy S III","Pixel 7","Pixel 5","Pixel 4a (5G)","Pixel 4","Pixel 3","Pixel 2 XL","Pixel 2","Nexus 6P","Nexus 6","Nexus 5X","Nexus 5","Nexus 4","Moto G4","LG Optimus L70","Microsoft Lumia 950","Microsoft Lumia 550","Nokia Lumia 520","Nokia N9","BlackBerry Z30"]},Me=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),rn[e].map(n=>tn[n]).filter(n=>n.defaultBrowserType&&r.includes(n.defaultBrowserType))},$o={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile")}},Io={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile",!0)}}});var sn=_(()=>{"use strict"});function kt(){return{version:"1.0",entries:{}}}var on=_(()=>{"use strict";Ke()});var ee,Pt,an=_(()=>{"use strict";ee=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(ee||{}),Pt=18e4});var cn=_(()=>{"use strict"});var ln=_(()=>{"use strict"});var pn=_(()=>{"use strict"});var dn=_(()=>{"use strict";Ve()});var ue=_(()=>{"use strict";Er();Ar();$r();Ir();Vr();Zr();Ge();vt();Qr();nn();sn();on();Ke();an();cn();ln();pn();dn();Ve()});import{stringify as Oo}from"yaml";import{createHash as Xo}from"crypto";import{parse as qo,stringify as vn}from"yaml";import{readFileSync as Zo,existsSync as Qo}from"fs";import{resolve as At,dirname as ei}from"path";import{parse as gn,stringify as ti}from"yaml";import{readFileSync as ci,writeFileSync as li,mkdirSync as pi}from"fs";import{dirname as di}from"path";function se(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 Lo(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Ro(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 Je(e){let t=Lo(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let n=Ro(e);if(n){let s=JSON.stringify(n);return`${t}.locator(${s}).first()`}return null}function un(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Co.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:!No.includes(t)}function S(e,t){F.set(e,t)}function Do(e){return F.get(e)}function ge(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 It(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 fn(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let n=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=r.map(a=>n.includes(a)||s.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${o.join(", ")})`}function oe(e,t,r,n="main"){let s=[];for(let o=0;o<e.length;o++){let a=e[o],i=`${n}.${o}`,c=Fo(a,t,i,r);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function Fo(e,t,r,n){let s=" ".repeat(t);switch(e.type){case"DRAFT":return Uo(e,t,r,n);case"ACTION":return Ho(e,t,r,n);case"STEP":return Bo(e,t,r,n);case"IF_ELSE":return Go(e,t,r,n);case"WHILE_LOOP":return Wo(e,t,r,n);default:return[`${s}// Unknown statement type: ${e.type}`]}}function Uo(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 a=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, ${a}, '${r}');`]}function Ho(e,t,r,n){let s=" ".repeat(t),o=e.description,a=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 b=JSON.stringify(o),x=!!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, ${b}, '${r}', ${x});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let d=l.action_data?.action_name||"",f=l.action_description||"",h=Do(d);if(!h)return[`${s}// ${r}: Unknown action: ${d}`,`${s}throw new Error(${JSON.stringify(`Unknown action: ${d}`)});`];let g={imports:n.imports},u=h(l,r,g);if(n.noAgent){if(un(l))return[`${s}// ${r}: ${L(f)}`,`${s}// AI action: ${L(f)} (requires agent - skipped in hook)`];let b=Ko(l,d,s,r);return b||[`${s}// ${r}: ${L(f)}`,...u.map(x=>`${s}${x}`)]}if(un(l))return[`${s}// ${r}: ${L(f)}`,`${s}page = agent.agentServices.validatePage(page);`,...u.map(b=>`${s}${b}`)];let p=JSON.stringify(f),w=u.map(b=>`${s} ${b}`),y=jo(l),m=a?`'${a}'`:"undefined";return[`${s}// ${r}: ${L(f)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...w,`${s}}, ${p}, '${r}', ${m}, ${y});`]}function Bo(e,t,r,n){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${L(e.description)}`);let a=oe(e.statements,t,n,r);return o.push(...a),o}function Go(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 i=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=oe(e.then,t+1,n,`${r}.then`);if(o.push(...a),e.else&&e.else.length>0){o.push(`${s}} else {`);let i=oe(e.else,t+1,n,`${r}.else`);o.push(...i)}return o.push(`${s}}`),o}function Wo(e,t,r,n){let s=" ".repeat(t),o=[];o.push(`${s}// ${r}: Loop`);let a=e.timeout_ms??Pt,i=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${a};`),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 d=oe(e.body,t+1,n,`${r}.body`);return o.push(...d),o.push(`${s}}`),o}function Ko(e,t,r,n){let s=e.action_description||"",o=e.action_data?.kwargs||{},a=o.timeout_ms??$t;switch(t){case"go_to_url":case"open_tab":{let i=o.url||"";return[`${r}// ${n}: ${L(s)}`,`${r}await page.goto(${JSON.stringify(i)}, { 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 i=o.text||"",c=Je(e);return c?[`${r}// ${n}: ${L(s)}`,`${r}await ${c}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=o.text||o.label||"",c=Je(e);return c?[`${r}// ${n}: ${L(s)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function Vo(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...wn()),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};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...hn("beforeEach",t.beforeEach,o)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let i=t?.testName||e.goal||"Generated test",c=Et(t?.tags);for(let l of t.parameters)r.push(...Xe(e,`${c}${se(i)} [${se(l.name)}]`,o,0,a,l.name,l.values)),r.push("")}else{let i=t?.testName||e.goal||"Generated test",c=Et(t?.tags);r.push(...Xe(e,`${c}${se(i)}`,o,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...hn("afterEach",t.afterEach,o))),bn(r,s),r.join(`
217
- `)}function zo(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...wn()),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},a=t?.testName||"Test Suite",i=Et(t?.tags);r.push(`test.describe.serial('${i}${se(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...ze("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...ze("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let d=e.tests[l],f=d.timeout||d.skip!==void 0||d.fail!==void 0||d.only||d.slow?{timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}:void 0;if(d.parameters&&d.parameters.length>0)for(let h of d.parameters)r.push(...Xe(d.testFlow,`${se(d.name)} [${se(h.name)}]`,o,1,f,h.name,h.values)),r.push("");else r.push(...Xe(d.testFlow,se(d.name),o,1,f)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...ze("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...ze("afterAll",e.afterAll,o,1)),r.push("});"),bn(r,s),r.join(`
218
- `)}function Et(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Ye(e){let t=new Set;function r(n){for(let s of n)switch(s.type){case ee.ACTION:{let a=s.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&Jo.includes(i)&&t.add(i);break}case ee.STEP:r(s.statements);break;case ee.IF_ELSE:{let o=s;r(o.then),o.else&&r(o.else);break}case ee.WHILE_LOOP:r(s.body);break}}return r(e),t}function Yo(e){let t=Ye(e.statements??[]);if(e.teardown)for(let r of Ye(e.teardown))t.add(r);return t}function Mt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Xe(e,t,r,n=0,s,o,a){let i=" ".repeat(n),c=[],l=Yo(e),d=Mt(l),f=s?.only?"test.only":"test",h=o?`, { tag: '@${se(o)}' }`:"";if(c.push(`${i}${f}('${t}'${h}, async (${d}) => {`),s?.skip===!0?c.push(`${i} test.skip();`):typeof s?.skip=="string"&&c.push(`${i} test.skip(true, '${se(s.skip)}');`),s?.fail===!0?c.push(`${i} test.fail();`):typeof s?.fail=="string"&&c.push(`${i} test.fail(true, '${se(s.fail)}');`),s?.slow&&c.push(`${i} test.slow();`),s?.timeout&&c.push(`${i} test.setTimeout(${s.timeout});`),a){for(let[p,w]of Object.entries(a))c.push(`${i} agent.agentServices.saveVariable(${JSON.stringify(p)}, ${JSON.stringify(w)});`);c.push("")}let g=e.teardown&&e.teardown.length>0,u=n+1;if(g){if(c.push(`${i} try {`),e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let w=oe(e.statements,u+1,r);c.push(...w)}c.push(`${i} } finally {`),c.push(`${i} // Teardown`);let p=oe(e.teardown,u+1,r,"teardown");c.push(...p),c.push(`${i} }`)}else if(e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let p=oe(e.statements,u,r);c.push(...p)}return c.push(`${i}});`),c}function hn(e,t,r){let n=[],s=yn(t),o=Ye(s),a=Mt(o);return n.push(`test.${e}(async (${a}) => {`),n.push(...oe(s,1,r,e)),n.push("});"),n}function ze(e,t,r,n){let s=" ".repeat(n),o=[],a=yn(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(...oe(a,n+1,c,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let c=Ye(a),l=Mt(c);o.push(`${s}test.${e}(async (${l}) => {`),o.push(...oe(a,n+1,r,e)),o.push(`${s}});`)}return o}function yn(e){let r=Oo({goal:"_hook",statements:e});return K(r).statements??[]}function wn(){return["import { test, expect } from 'shiplightai/fixture';"]}function bn(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 Sn(e,t,r){let n={expandingPaths:new Set([At(t)]),depth:0,referencedPaths:new Set,basePath:r},s={...e};Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,n)),Array.isArray(s.teardown)&&(s.teardown=fe(s.teardown,t,n));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=fe(s[o],t,n));return{doc:s,referencedTemplatePaths:Array.from(n.referencedPaths)}}function fe(e,t,r){let n=[];for(let s of e)if(ri(s)){let o=ni(s,t,r);n.push(o)}else n.push(si(s,t,r));return n}function ri(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function ni(e,t,r){if(r.depth>=mn)throw new Error(`Template expansion exceeded maximum depth of ${mn}. Check for deeply nested or circular template references.`);let n=At(ei(t),e.template),s=!Qo(n)&&r.basePath?At(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=Zo(s,"utf-8")}catch(u){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${u.message}`)}let a=gn(o);if(!a||typeof a!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let i=a.params||[],c=e.params||{};for(let u of i)if(!(u in c))throw new Error(`Template ${e.template} requires param "${u}" but it was not provided. Required params: [${i.join(", ")}]`);let l=a.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let p=ti(l);for(let[w,y]of Object.entries(c))p=p.split(`<<${w}>>`).join(String(y));l=gn(p)}let d={expandingPaths:new Set([...r.expandingPaths,s]),depth:r.depth+1,referencedPaths:r.referencedPaths},f=fe(l,s,d),g={STEP:a.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 si(e,t,r){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,r)),Array.isArray(n.THEN)&&(n.THEN=fe(n.THEN,t,r)),Array.isArray(n.ELSE)&&(n.ELSE=fe(n.ELSE,t,r)),Array.isArray(n.DO)&&(n.DO=fe(n.DO,t,r)),n}function Lt(e,t,r){let n=qo(e),s=n?.name,o=n?.tags,a=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 Ot('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ii(n,s,o,a,t,r)}return oi(n,s,o,a,t,r)}function oi(e,t,r,n,s,o){let a=e?.beforeEach,i=e?.afterEach,c=_n(e?.parameters),l=e?.timeout,d=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 Ot(`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 y=Sn(e,s,o);e=y.doc,u=y.referencedTemplatePaths}let p=vn(e),w=K(p);return s&&(Se(w.statements??[],s,"main"),w.teardown&&Se(w.teardown,s,"teardown")),{testFlow:w,name:t,tags:r,use:n,beforeEach:a,afterEach:i,parameters:c,timeout:l,skip:d,fail:f,only:h,slow:g,referencedTemplatePaths:u}}function ii(e,t,r,n,s,o){let a=e.suite;if(!Array.isArray(a.tests)||a.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let i=a.beforeAll,c=a.afterAll,l=a.beforeEach,d=a.afterEach,f=[],h=a.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let w={goal:p.name,statements:p.statements};p.teardown&&(w.teardown=p.teardown);let y=[],m=w;if(s&&typeof w=="object"){let $=Sn(w,s,o);m=$.doc,y=$.referencedTemplatePaths,f.push(...y)}let b=vn(m),x=K(b),T=_n(p.parameters);return{testFlow:x,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:T,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=a.base_url,u=g?{...n,baseURL:g}:n;return{suite:{beforeAll:i,afterAll:c,beforeEach:l,afterEach:d,tests:h},name:t,tags:r,use:u,referencedTemplatePaths:f}}function _n(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Se(e,t,r){for(let n=0;n<e.length;n++){let s=e[n],o=`${r}.${n}`,a=s.description||"";if(s.uid=ai(t,o,a),s.type===ee.STEP)Se(s.statements,t,o);else if(s.type===ee.IF_ELSE){let i=s;Se(i.then,t,`${o}.then`),i.else&&Se(i.else,t,`${o}.else`)}else s.type===ee.WHILE_LOOP&&Se(s.body,t,`${o}.body`)}}function ai(e,t,r){let n=Xo("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 xn(e,t){let r;try{r=ci(e,"utf-8")}catch(n){return{valid:!1,errors:[`Failed to read file: ${n.message}`],warnings:[]}}return ui(r,e,t)}function ui(e,t,r){let n=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=n||s?null:Tt(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let a,i,c=[];try{let l=r?.parsed??Lt(e,t,r?.basePath);c=l.referencedTemplatePaths;let d={version:r?.version,actionEntityStore:r?.actionEntityStore},f=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?a=zo(l.suite,{...d,testName:l.name,tags:l.tags,use:l.use}):a=Vo(l.testFlow,{...d,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=a.split(`
215
+ `}async function uo(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 wt(e){let{yamlFilePath:t,configPath:r,tempSuffix:n="",headed:s}=e,o=ne.dirname(r),a=await uo(16174),i;if(!pe.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let b=lo(pe.readFileSync(t,"utf-8"));b?.use&&typeof b.use=="object"&&!Array.isArray(b.use)&&(i=b.use),b?.base_url&&!i?.baseURL&&(i={...i,baseURL:b.base_url})}catch(b){console.error("[debugger] Could not parse YAML for `use` block:",b)}let c=ne.dirname(ne.resolve(t)),l=n?`-${n}`:"",d=ne.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),f=po(t,a,i,o);pe.writeFileSync(d,f);let h=Er(d),g=["playwright","test",d,...s?["--headed"]:[]],u=co("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});u.stdout?.on("data",b=>{process.stderr.write(b)}),u.stderr?.on("data",b=>{process.stderr.write(b)});let p=()=>{u.killed||u.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),u.on("close",b=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),h(),b!==0&&b!==null&&console.error(`[debugger] Playwright process exited with code ${b}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let w=["127.0.0.1","::1"];async function y(b){try{let x=b.includes(":")?`[${b}]`:b,k=await fetch(`http://${x}:${a}/api/test-flow`);if(k.ok){try{await k.text()}catch{}return!0}}catch{}return!1}let m=null;for(let b=0;b<180;b++){if(u.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${u.exitCode} before sandbox was ready`);for(let x of w)if(await y(x)){m=x;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${a}`);break}if(b===179)throw p(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(x=>setTimeout(x,1e3))}if(!m)throw p(),h(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:m,pid:u.pid??0,cleanup:async()=>{p(),h()}}}var bt=_(()=>{"use strict"});var Ar=_(()=>{"use strict"});var $r=_(()=>{"use strict"});var Ir=_(()=>{"use strict"});var Mr=_(()=>{"use strict"});var Be=_(()=>{"use strict"});var Or=_(()=>{"use strict";Be()});var Lr,ho,Rr,Cr,Ie,go,mo,Nr=_(()=>{"use strict";Lr=112,ho=1080-Lr,Rr={"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"}},Cr={desktop:["Desktop Chrome","Desktop Chrome Medium Resolution","Desktop Chrome (Branded)","Desktop Chrome Medium Resolution (Branded)","Desktop Edge (Branded)","Desktop Edge Medium Resolution (Branded)","Desktop Safari"],mobile:["iPhone 15 Pro Max","iPhone 15 Pro","iPhone 15 Plus","iPhone 15","iPhone 14 Pro Max","iPhone 14 Pro","iPhone 14 Plus","iPhone 14","iPhone 13 Pro Max","iPhone 13 Pro","iPhone 13","iPhone 13 Mini","iPhone 12 Pro Max","iPhone 12 Pro","iPhone 12","iPhone 12 Mini","iPhone 11 Pro Max","iPhone 11 Pro","iPhone 11","iPhone XR","iPhone X","iPhone SE (3rd gen)","iPhone SE","iPhone 8 Plus","iPhone 8","iPhone 7 Plus","iPhone 7","iPhone 6 Plus","iPhone 6","Galaxy S24","Galaxy A55","Galaxy S9+","Galaxy S8","Galaxy S5","Galaxy Note 3","Galaxy Note II","Galaxy S III","Pixel 7","Pixel 5","Pixel 4a (5G)","Pixel 4","Pixel 3","Pixel 2 XL","Pixel 2","Nexus 6P","Nexus 6","Nexus 5X","Nexus 5","Nexus 4","Moto G4","LG Optimus L70","Microsoft Lumia 950","Microsoft Lumia 550","Nokia Lumia 520","Nokia N9","BlackBerry Z30"]},Ie=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Cr[e].map(n=>Rr[n]).filter(n=>n.defaultBrowserType&&r.includes(n.defaultBrowserType))},go={desktop:{label:"Desktop",type:"desktop",devices:Ie("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Ie("mobile")}},mo={desktop:{label:"Desktop",type:"desktop",devices:Ie("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Ie("mobile",!0)}}});import{v4 as Ga}from"uuid";var Dr=_(()=>{"use strict"});import{z as v}from"zod";var jr,vt,Fr,we,Ur,Hr,Br,W,Gr,Ge,St,_t=_(()=>{"use strict";jr=v.enum(["JS_CODE","AI_MODE"]),vt=v.object({type:jr,expression:v.string()}),Fr=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=v.object({uid:v.string(),type:Fr,comment:v.string().optional()}),Ur=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(),Hr=we.extend({type:v.literal("DRAFT"),description:v.string()}),Br=we.extend({type:v.literal("ACTION"),description:v.string(),action_entity:Ur.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),W=v.lazy(()=>v.union([Hr,Br,we.extend({type:v.literal("STEP"),description:v.string().optional().default(""),statements:v.array(W),reference_id:v.number().optional(),template_path:v.string().optional(),template_params:v.record(v.string()).optional()}),we.extend({type:v.literal("IF_ELSE"),description:v.string().optional(),condition:vt,then:v.array(W),else:v.array(W).optional()}),we.extend({type:v.literal("WHILE_LOOP"),description:v.string().optional(),condition:vt,body:v.array(W),timeout_ms:v.number().optional()})])),Gr=v.object({name:v.string(),statements:v.array(W),teardown:v.array(W).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()}),Ge=v.object({tests:v.array(Gr).min(1),beforeAll:v.array(W).optional(),afterAll:v.array(W).optional(),beforeEach:v.array(W).optional(),afterEach:v.array(W).optional()}),St=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(W).optional(),teardown:v.array(W).optional(),last_modified_at:v.string().optional(),testGroup:Ge.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 Ya,parse as zr,parseAllDocuments as Xa,parseDocument as yo,Document as Jr,isMap as ve,isSeq as K}from"yaml";import{v4 as Q}from"uuid";function We(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(J)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(J)),r}function Tt(e,t){if(e.testGroup)return Xr(e,t);let r=We(e,t),n=new Jr(r),s=n.contents?.get("tags",!0);return K(s)&&(s.flow=!0),e.comment&&(n.commentBefore=e.comment),Wr(n,e.statements??[]),e.teardown&&Wr(n,e.teardown,"teardown"),n.toString(Yr)}function Wr(e,t,r="statements"){let n=e.contents;if(!n||!ve(n))return;let s=n.get(r,!0);K(s)&&Me(s,t)}function Me(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),ve(s)){let o=s;if(n.type==="STEP"){let a=o.get("statements",!0);K(a)&&Me(a,n.statements)}else if(n.type==="IF_ELSE"){let a=o.get("THEN",!0);K(a)&&Me(a,n.then);let i=o.get("ELSE",!0);K(i)&&n.else&&Me(i,n.else)}else if(n.type==="WHILE_LOOP"){let a=o.get("DO",!0);K(a)&&Me(a,n.body)}}}}function Xr(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(J)),r.beforeEach&&r.beforeEach.length>0&&(s.beforeEach=r.beforeEach.map(J)),r.afterEach&&r.afterEach.length>0&&(s.afterEach=r.afterEach.map(J)),r.afterAll&&r.afterAll.length>0&&(s.afterAll=r.afterAll.map(J)),s.tests=r.tests.map(i=>{let c={name:i.name};return i.skip!==void 0&&(c.skip=i.skip),i.timeout!==void 0&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only!==void 0&&(c.only=i.only),i.slow!==void 0&&(c.slow=i.slow),c.statements=i.statements.map(J),i.teardown&&i.teardown.length>0&&(c.teardown=i.teardown.map(J)),c}),n.suite=s;let o=new Jr(n),a=o.contents?.get("tags",!0);return K(a)&&(a.flow=!0),o.toString(Yr)}function J(e){switch(e.type){case"DRAFT":return wo(e);case"ACTION":return bo(e);case"STEP":return vo(e);case"IF_ELSE":return So(e);case"WHILE_LOOP":return _o(e)}}function wo(e){return{intent:e.description}}function bo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:i,js:c}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:i};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let i=r?.code;if(typeof i=="string"&&i.trim()&&e.description)return{description:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let c={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let i=r?.seconds,l={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(l.seconds=i),l}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:i}}if(!e.action_entity)return{intent:e.description};let 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 a=e.action_entity.xpath;if(a&&(s.xpath=a),e.use_pure_vision&&(s.use_pure_vision=!0),n.kwargs&&Object.keys(n.kwargs).length>0)for(let[i,c]of Object.entries(n.kwargs))i!=="uid"&&(i==="statement"&&(t==="ai_action"||t==="ai_step")||(s[i]=c));return n.args&&n.args.length>0&&(s.args=n.args),s}function vo(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(J)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function So(e){let t={IF:qr(e.condition),THEN:e.then.map(J)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(J)),t}function _o(e){let t={WHILE:qr(e.condition),DO:e.body.map(J)};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 Oe(e){try{let t=zr(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 xt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(xt);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]=xt(o);return n}function V(e){if(e.length>Kr)throw new Error(`YAML input too large (${e.length} bytes, max ${Kr})`);let t=xt(zr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return xo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:Y(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=Y(t.teardown));let n=St.safeParse(r);if(!n.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(n.error.errors)}`);let s=n.data;return Ke(e,s),s}function xo(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(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let c={name:i.name,statements:Y(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(c.teardown=Y(i.teardown)),i.skip!==void 0&&(c.skip=i.skip),typeof i.timeout=="number"&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only===!0&&(c.only=!0),i.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=Y(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=Y(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=Y(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=Y(t.afterEach));let o=Ge.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 Y(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(To)}function To(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 ko(t);if("WHILE"in t)return Eo(t);if("STEP"in t)return Po(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:Q(),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:Q(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:Q(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:n}}}}}if("WAIT"in t){let r=t.WAIT,n=typeof t.seconds=="number"?t.seconds:3;return{uid:Q(),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:Q(),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:Q(),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 Vr({...n,action:"function",functionName:r})}if("action"in t)return Vr(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:Q(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Zr(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 ko(e){let t=Zr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let n={uid:Q(),type:"IF_ELSE",condition:t,then:Y(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(n.else=Y(e.ELSE)),n}function Eo(e){let t=Zr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let n={uid:Q(),type:"WHILE_LOOP",condition:t,body:Y(r)};return typeof e.timeout_ms=="number"&&(n.timeout_ms=e.timeout_ms),n}function Po(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:Q(),type:"STEP",description:t,statements:Y(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,a]of Object.entries(n))s[o]=String(a);r.template_params=s}return r}function Vr(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,a={};for(let[l,d]of Object.entries(e))Ao.has(l)||(a[l]=d);t==="verify"&&typeof a.js=="string"&&(a.code=a.js,delete a.js),(t==="ai_action"||t==="ai_step")&&a.statement===void 0&&(a.statement=r);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};n&&(i.locator=n),s&&(i.xpath=s);let c={uid:Q(),type:"ACTION",description:r,action_entity:i};return o&&(c.use_pure_vision=!0),c}function Ke(e,t){let r;try{r=yo(e)}catch{return}let n=r.contents;if(!n||!ve(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);K(o)&&t.statements&&be(o,t.statements);let a=s.get("teardown",!0);K(a)&&t.teardown&&be(a,t.teardown)}function be(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let n=e.items[r];n.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=n.commentBefore);let s=t[r];if(s.type==="STEP"&&ve(n)){let o=n.get("statements",!0);K(o)&&be(o,s.statements)}else if(s.type==="IF_ELSE"&&ve(n)){let o=n.get("THEN",!0);K(o)&&be(o,s.then);let a=n.get("ELSE",!0);K(a)&&s.else&&be(a,s.else)}else if(s.type==="WHILE_LOOP"&&ve(n)){let o=n.get("DO",!0);K(o)&&be(o,s.body)}}}var Yr,Kr,Ao,Ve=_(()=>{"use strict";_t();Yr={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Kr=1024*1024;Ao=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as tc,stringify as rc}from"yaml";var Qr=_(()=>{"use strict";Ve()});var kt,ze,Je=_(()=>{"use strict";kt=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=kt(s);for(let a of o)r(a.statements)}};return r(e),t}});function sn(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??"";nn.has(s)||t++}return{action:t,draft:r}}function Io(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function rn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Mo(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<=en&&s<=tn?null:`${r} non-blank line(s), ${s} await(s) \u2014 limits are ${en} lines and ${tn} 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 Et(e,t){let r=t?.coverageThreshold??$o,n=[],s=[],o;try{o=V(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 a=[...ze(o.statements??[]),...o.teardown?ze(o.teardown):[]],{action:i,draft:c}=sn(a),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 a){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,p=u.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let w=u.action_entity?.action_data?.kwargs?.code;if(typeof w=="string"){let y=Io(w);if(y){let m=u.description||p;n.push(`Invalid JS in "${m}": ${y}. ${l}`)}else if(p==="verify"){let m=Mo(w);if(m){let b=u.description||p;n.push(`JS cache for "${b}" is too complex: ${m}`)}}}}}if(h.type==="IF_ELSE"){let u=h;if(u.condition.type==="JS_CODE"){let p=rn(u.condition.expression);p&&n.push(`Invalid JS in IF condition "${u.condition.expression}": ${p}. ${l}`)}}if(h.type==="WHILE_LOOP"){let u=h;if(u.condition.type==="JS_CODE"){let p=rn(u.condition.expression);p&&n.push(`Invalid JS in WHILE condition "${u.condition.expression}": ${p}. ${l}`)}}}let d=i+c,f=d>0?Math.round(i/d*100):0;return d>0&&f/100<r&&s.push(`Low action coverage: ${i}/${d} 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:d,action:i,draft:c,coverage:f}}}var $o,en,tn,nn,on=_(()=>{"use strict";Ve();Je();$o=.5,en=5,tn=3,nn=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});function Pt(){return{version:"1.0",entries:{}}}var an=_(()=>{"use strict";Je()});var ee,At,cn=_(()=>{"use strict";ee=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(ee||{}),At=18e4});var ln=_(()=>{"use strict"});var pn=_(()=>{"use strict"});var dn=_(()=>{"use strict"});var un=_(()=>{"use strict";Be()});var ue=_(()=>{"use strict";Ar();$r();Ir();Mr();Or();Nr();Dr();Qr();on();Ve();_t();an();Je();cn();ln();pn();dn();un();Be()});import{stringify as Lo}from"yaml";import{createHash as qo}from"crypto";import{parse as Zo,stringify as Sn}from"yaml";import{readFileSync as Qo,existsSync as ei}from"fs";import{resolve as It,dirname as ti}from"path";import{parse as mn,stringify as ri}from"yaml";import{readFileSync as li,writeFileSync as pi,mkdirSync as di}from"fs";import{dirname as ui}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function R(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Ro(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Co(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 Xe(e){let t=Ro(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let n=Co(e);if(n){let s=JSON.stringify(n);return`${t}.locator(${s}).first()`}return null}function fn(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:No.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:!Do.includes(t)}function S(e,t){F.set(e,t)}function Fo(e){return F.get(e)}function ge(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 Ot(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 hn(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let n=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=r.map(a=>n.includes(a)||s.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${o.join(", ")})`}function oe(e,t,r,n="main"){let s=[];for(let o=0;o<e.length;o++){let a=e[o],i=`${n}.${o}`,c=Uo(a,t,i,r);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function Uo(e,t,r,n){let s=" ".repeat(t);switch(e.type){case"DRAFT":return Ho(e,t,r,n);case"ACTION":return Bo(e,t,r,n);case"STEP":return Go(e,t,r,n);case"IF_ELSE":return Wo(e,t,r,n);case"WHILE_LOOP":return Ko(e,t,r,n);default:return[`${s}// Unknown statement type: ${e.type}`]}}function Ho(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}: ${R(o)}`,`${s}// DRAFT: ${R(o)} (requires agent - skipped in hook)`];let a=JSON.stringify(o);return[`${s}// ${r}: ${R(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, ${a}, '${r}');`]}function Bo(e,t,r,n){let s=" ".repeat(t),o=e.description,a=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}: ${R(o)}`,`${s}// DRAFT: ${R(o)} (requires agent - skipped in hook)`];let b=JSON.stringify(o),x=!!e.use_pure_vision;return[`${s}// ${r}: ${R(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, ${b}, '${r}', ${x});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let d=l.action_data?.action_name||"",f=l.action_description||"",h=Fo(d);if(!h)return[`${s}// ${r}: Unknown action: ${d}`,`${s}throw new Error(${JSON.stringify(`Unknown action: ${d}`)});`];let g={imports:n.imports},u=h(l,r,g);if(n.noAgent){if(fn(l))return[`${s}// ${r}: ${R(f)}`,`${s}// AI action: ${R(f)} (requires agent - skipped in hook)`];let b=Vo(l,d,s,r);return b||[`${s}// ${r}: ${R(f)}`,...u.map(x=>`${s}${x}`)]}if(fn(l))return[`${s}// ${r}: ${R(f)}`,`${s}page = agent.agentServices.validatePage(page);`,...u.map(b=>`${s}${b}`)];let p=JSON.stringify(f),w=u.map(b=>`${s} ${b}`),y=jo(l),m=a?`'${a}'`:"undefined";return[`${s}// ${r}: ${R(f)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...w,`${s}}, ${p}, '${r}', ${m}, ${y});`]}function Go(e,t,r,n){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${R(e.description)}`);let a=oe(e.statements,t,n,r);return o.push(...a),o}function Wo(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: ${R(e.condition.expression)}`);let i=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=oe(e.then,t+1,n,`${r}.then`);if(o.push(...a),e.else&&e.else.length>0){o.push(`${s}} else {`);let i=oe(e.else,t+1,n,`${r}.else`);o.push(...i)}return o.push(`${s}}`),o}function Ko(e,t,r,n){let s=" ".repeat(t),o=[];o.push(`${s}// ${r}: Loop`);let a=e.timeout_ms??At,i=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${a};`),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: ${R(e.condition.expression)}`);let f=JSON.stringify(e.condition.expression);o.push(`${s}while (${l}_check() && await agent.evaluate(page, ${f}, "${r}")) {`)}let d=oe(e.body,t+1,n,`${r}.body`);return o.push(...d),o.push(`${s}}`),o}function Vo(e,t,r,n){let s=e.action_description||"",o=e.action_data?.kwargs||{},a=o.timeout_ms??Mt;switch(t){case"go_to_url":case"open_tab":{let i=o.url||"";return[`${r}// ${n}: ${R(s)}`,`${r}await page.goto(${JSON.stringify(i)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${n}: ${R(s)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${n}: ${R(s)}`,`${r}await page.goForward();`];case"input_text":{let i=o.text||"",c=Xe(e);return c?[`${r}// ${n}: ${R(s)}`,`${r}await ${c}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=o.text||o.label||"",c=Xe(e);return c?[`${r}// ${n}: ${R(s)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function zo(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...bn()),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};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...gn("beforeEach",t.beforeEach,o)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let i=t?.testName||e.goal||"Generated test",c=$t(t?.tags);for(let l of t.parameters)r.push(...Ze(e,`${c}${se(i)} [${se(l.name)}]`,o,0,a,l.name,l.values)),r.push("")}else{let i=t?.testName||e.goal||"Generated test",c=$t(t?.tags);r.push(...Ze(e,`${c}${se(i)}`,o,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...gn("afterEach",t.afterEach,o))),vn(r,s),r.join(`
217
+ `)}function Jo(e,t){let r=[],n=t?.version||"unknown";r.push(`// @generated by shiplightai v${n}`),r.push(...bn()),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},a=t?.testName||"Test Suite",i=$t(t?.tags);r.push(`test.describe.serial('${i}${se(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ye("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ye("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let d=e.tests[l],f=d.timeout||d.skip!==void 0||d.fail!==void 0||d.only||d.slow?{timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}:void 0;if(d.parameters&&d.parameters.length>0)for(let h of d.parameters)r.push(...Ze(d.testFlow,`${se(d.name)} [${se(h.name)}]`,o,1,f,h.name,h.values)),r.push("");else r.push(...Ze(d.testFlow,se(d.name),o,1,f)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ye("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ye("afterAll",e.afterAll,o,1)),r.push("});"),vn(r,s),r.join(`
218
+ `)}function $t(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function qe(e){let t=new Set;function r(n){for(let s of n)switch(s.type){case ee.ACTION:{let a=s.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&Yo.includes(i)&&t.add(i);break}case ee.STEP:r(s.statements);break;case ee.IF_ELSE:{let o=s;r(o.then),o.else&&r(o.else);break}case ee.WHILE_LOOP:r(s.body);break}}return r(e),t}function Xo(e){let t=qe(e.statements??[]);if(e.teardown)for(let r of qe(e.teardown))t.add(r);return t}function Lt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ze(e,t,r,n=0,s,o,a){let i=" ".repeat(n),c=[],l=Xo(e),d=Lt(l),f=s?.only?"test.only":"test",h=o?`, { tag: '@${se(o)}' }`:"";if(c.push(`${i}${f}('${t}'${h}, async (${d}) => {`),s?.skip===!0?c.push(`${i} test.skip();`):typeof s?.skip=="string"&&c.push(`${i} test.skip(true, '${se(s.skip)}');`),s?.fail===!0?c.push(`${i} test.fail();`):typeof s?.fail=="string"&&c.push(`${i} test.fail(true, '${se(s.fail)}');`),s?.slow&&c.push(`${i} test.slow();`),s?.timeout&&c.push(`${i} test.setTimeout(${s.timeout});`),a){for(let[p,w]of Object.entries(a))c.push(`${i} agent.agentServices.saveVariable(${JSON.stringify(p)}, ${JSON.stringify(w)});`);c.push("")}let g=e.teardown&&e.teardown.length>0,u=n+1;if(g){if(c.push(`${i} try {`),e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let w=oe(e.statements,u+1,r);c.push(...w)}c.push(`${i} } finally {`),c.push(`${i} // Teardown`);let p=oe(e.teardown,u+1,r,"teardown");c.push(...p),c.push(`${i} }`)}else if(e.statements&&e.statements.length>0){c.push(`${i} // Test steps`);let p=oe(e.statements,u,r);c.push(...p)}return c.push(`${i}});`),c}function gn(e,t,r){let n=[],s=wn(t),o=qe(s),a=Lt(o);return n.push(`test.${e}(async (${a}) => {`),n.push(...oe(s,1,r,e)),n.push("});"),n}function Ye(e,t,r,n){let s=" ".repeat(n),o=[],a=wn(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(...oe(a,n+1,c,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let c=qe(a),l=Lt(c);o.push(`${s}test.${e}(async (${l}) => {`),o.push(...oe(a,n+1,r,e)),o.push(`${s}});`)}return o}function wn(e){let r=Lo({goal:"_hook",statements:e});return V(r).statements??[]}function bn(){return["import { test, expect } from 'shiplightai/fixture';"]}function vn(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([It(t)]),depth:0,referencedPaths:new Set,basePath:r},s={...e};Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,n)),Array.isArray(s.teardown)&&(s.teardown=fe(s.teardown,t,n));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=fe(s[o],t,n));return{doc:s,referencedTemplatePaths:Array.from(n.referencedPaths)}}function fe(e,t,r){let n=[];for(let s of e)if(ni(s)){let o=si(s,t,r);n.push(o)}else n.push(oi(s,t,r));return n}function ni(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function si(e,t,r){if(r.depth>=yn)throw new Error(`Template expansion exceeded maximum depth of ${yn}. Check for deeply nested or circular template references.`);let n=It(ti(t),e.template),s=!ei(n)&&r.basePath?It(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=Qo(s,"utf-8")}catch(u){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${u.message}`)}let a=mn(o);if(!a||typeof a!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let i=a.params||[],c=e.params||{};for(let u of i)if(!(u in c))throw new Error(`Template ${e.template} requires param "${u}" but it was not provided. Required params: [${i.join(", ")}]`);let l=a.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let p=ri(l);for(let[w,y]of Object.entries(c))p=p.split(`<<${w}>>`).join(String(y));l=mn(p)}let d={expandingPaths:new Set([...r.expandingPaths,s]),depth:r.depth+1,referencedPaths:r.referencedPaths},f=fe(l,s,d),g={STEP:a.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 oi(e,t,r){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,r)),Array.isArray(n.THEN)&&(n.THEN=fe(n.THEN,t,r)),Array.isArray(n.ELSE)&&(n.ELSE=fe(n.ELSE,t,r)),Array.isArray(n.DO)&&(n.DO=fe(n.DO,t,r)),n}function Ct(e,t,r){let n=Zo(e),s=n?.name,o=n?.tags,a=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 Rt('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ai(n,s,o,a,t,r)}return ii(n,s,o,a,t,r)}function ii(e,t,r,n,s,o){let a=e?.beforeEach,i=e?.afterEach,c=xn(e?.parameters),l=e?.timeout,d=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 Rt(`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 y=_n(e,s,o);e=y.doc,u=y.referencedTemplatePaths}let p=Sn(e),w=V(p);return s&&(Se(w.statements??[],s,"main"),w.teardown&&Se(w.teardown,s,"teardown")),{testFlow:w,name:t,tags:r,use:n,beforeEach:a,afterEach:i,parameters:c,timeout:l,skip:d,fail:f,only:h,slow:g,referencedTemplatePaths:u}}function ai(e,t,r,n,s,o){let a=e.suite;if(!Array.isArray(a.tests)||a.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let i=a.beforeAll,c=a.afterAll,l=a.beforeEach,d=a.afterEach,f=[],h=a.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let w={goal:p.name,statements:p.statements};p.teardown&&(w.teardown=p.teardown);let y=[],m=w;if(s&&typeof w=="object"){let A=_n(w,s,o);m=A.doc,y=A.referencedTemplatePaths,f.push(...y)}let b=Sn(m),x=V(b),k=xn(p.parameters);return{testFlow:x,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:k,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=a.base_url,u=g?{...n,baseURL:g}:n;return{suite:{beforeAll:i,afterAll:c,beforeEach:l,afterEach:d,tests:h},name:t,tags:r,use:u,referencedTemplatePaths:f}}function xn(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Se(e,t,r){for(let n=0;n<e.length;n++){let s=e[n],o=`${r}.${n}`,a=s.description||"";if(s.uid=ci(t,o,a),s.type===ee.STEP)Se(s.statements,t,o);else if(s.type===ee.IF_ELSE){let i=s;Se(i.then,t,`${o}.then`),i.else&&Se(i.else,t,`${o}.else`)}else s.type===ee.WHILE_LOOP&&Se(s.body,t,`${o}.body`)}}function ci(e,t,r){let n=qo("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 Tn(e,t){let r;try{r=li(e,"utf-8")}catch(n){return{valid:!1,errors:[`Failed to read file: ${n.message}`],warnings:[]}}return fi(r,e,t)}function fi(e,t,r){let n=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=n||s?null:Et(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let a,i,c=[];try{let l=r?.parsed??Ct(e,t,r?.basePath);c=l.referencedTemplatePaths;let d={version:r?.version,actionEntityStore:r?.actionEntityStore},f=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?a=Jo(l.suite,{...d,testName:l.name,tags:l.tags,use:l.use}):a=zo(l.testFlow,{...d,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=a.split(`
219
219
  `).filter(g=>!g.startsWith("import ")).join(`
220
- `);new Function(h),i=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),pi(di(i),{recursive:!0}),li(i,a)}catch(l){let d=l instanceof Ot?"":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}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:i,referencedTemplatePaths:c}}var $t,Co,No,F,Jo,mn,Ot,Rt=_(()=>{"use strict";ue();ue();ue();ue();$t=5e3;Co=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],No=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];F=new Map;S("click",e=>{let t=Je(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??$t;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",F.get("click"));S("click_element_by_index",F.get("click"));S("double_click",e=>ge("double_click",e));S("double_click_on_element",F.get("double_click"));S("right_click",e=>ge("right_click",e));S("right_click_on_element",F.get("right_click"));S("hover",e=>ge("hover",e));S("hover_element_by_index",F.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",F.get("input_text"));S("clear_input",e=>ge("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",F.get("press"));S("send_keys_on_element",e=>{let t=Je(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??$t;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${n} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",F.get("scroll"));S("scroll_up",F.get("scroll"));S("scroll_element",F.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>ge("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",F.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,n=typeof r?.code=="string",s=n?r?.statement||e.action_description:e.action_description||r?.statement;if(n&&s){let a=r.code.split(`
220
+ `);new Function(h),i=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),di(ui(i),{recursive:!0}),pi(i,a)}catch(l){let d=l instanceof Rt?"":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}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:i,referencedTemplatePaths:c}}var Mt,No,Do,F,Yo,yn,Rt,Nt=_(()=>{"use strict";ue();ue();ue();ue();Mt=5e3;No=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],Do=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];F=new Map;S("click",e=>{let t=Xe(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Mt;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",F.get("click"));S("click_element_by_index",F.get("click"));S("double_click",e=>ge("double_click",e));S("double_click_on_element",F.get("double_click"));S("right_click",e=>ge("right_click",e));S("right_click_on_element",F.get("right_click"));S("hover",e=>ge("hover",e));S("hover_element_by_index",F.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",F.get("input_text"));S("clear_input",e=>ge("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",F.get("press"));S("send_keys_on_element",e=>{let t=Xe(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??Mt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${n} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",F.get("scroll"));S("scroll_up",F.get("scroll"));S("scroll_element",F.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>ge("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",F.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,n=typeof r?.code=="string",s=n?r?.statement||e.action_description:e.action_description||r?.statement;if(n&&s){let a=r.code.split(`
221
221
  `),i=JSON.stringify(s);return["{ const _t = Date.now(); try {",...a.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${i}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${i}\`);`,` await agent.assert(page, ${i}, ${JSON.stringify(t||"")});`,"} }"]}return n?r.code.split(`
222
222
  `):s?[`await agent.assert(page, ${JSON.stringify(s)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",F.get("verify"));S("assert",F.get("verify"));S("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});`]});S("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});S("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,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||""}');`]});S("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"]});S("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});S("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],n=t.split(`
223
- `);for(let s of n)r.push(` ${s}`);return r.push("}"),r});S("function",(e,t,r)=>{let n=e.action_data?.kwargs||{},s=n.functionName;if(s&&s.includes("#")){let[a,i]=s.split("#");if(a&&i){let c=a.replace(/\.(ts|js|mjs)$/,""),l=`import { ${i} } from '${c}';`;r?.imports?.add(l);let d={...n,functionName:i},f=fn(d);return f?[f.endsWith(";")?f:`${f};`]:["// Skipping function: invalid export pattern"]}}let o=fn(n);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});S("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});S("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],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},`),"});"]});S("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);S("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);S("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});S("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(n=>` ${n},`),"});"]});S("done",()=>["// Done - no action needed"]);S("click_by_coordinates",e=>It("click_by_coordinates",e));S("right_click_by_coordinates",e=>It("right_click_by_coordinates",e));S("double_click_by_coordinates",e=>It("double_click_by_coordinates",e));S("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},`),"});"]});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},`),"});"]});S("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
224
- `):["// Skipping js_action: missing code"]});Jo=["testContext","request"];mn=5;Ot=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as fi}from"express";import*as E from"fs/promises";import*as k from"path";import{stringify as qe,parse as Oe}from"yaml";function Le(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(([a])=>a!=="element_index")):n?.kwargs,o={...r};if(delete o.xpath,n&&s!==void 0){let a={...n,kwargs:s};r.action_data?o.action_data=a:o.action=a}return{...t,action_entity:o}}return t.type==="STEP"?{...t,statements:Le(t.statements)}:t.type==="IF_ELSE"?{...t,then:Le(t.then),...t.else?{else:Le(t.else)}:{}}:t.type==="WHILE_LOOP"?{...t,body:Le(t.body)}:t})}function me(e){if(!e||e.length===0)return[];let r=qe({goal:"_hook",statements:e});return K(r).statements??[]}async function Tn(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 Tn(k.join(e,r.name))))return!0}catch{}return!1}function Ze(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function hi(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=k.join(e,n),a=Oe(await E.readFile(s,"utf-8"))?.name??k.basename(n,".yaml");t.set(Ze(a),`templates/${n}`)}catch{}return t}function Nt(e,t){if(Array.isArray(e))return e.map(o=>Nt(o,t));if(!e||typeof e!="object")return e;let r=e,n={};for(let[o,a]of Object.entries(r))n[o]=Nt(a,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 Ct(e){return He({statements:e}).statements}function kn(e){let{initialDir:t,initialFile:r,projectRoot:n,onFileSelected:s}=e,o=fi();function a(){return k.join(n??t,"fixtures")}o.get("/api/files",async(c,l)=>{try{let d=typeof c.query.dir=="string"?c.query.dir:t,f=k.resolve(d),h=await E.readdir(f,{withFileTypes:!0}),g=[];for(let p of h)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let w=k.join(f,p.name);await Tn(w)&&g.push({name:p.name,type:"directory",path:w})}else p.isFile()&&p.name.endsWith(".test.yaml")&&g.push({name:p.name,type:"file",path:k.join(f,p.name)});g.sort((p,w)=>p.type!==w.type?p.type==="directory"?-1:1:p.name.localeCompare(w.name));let u=k.dirname(f);l.json({dir:f,parent:u!==f?u:null,entries:g,initialFile:r??null,projectRoot:n??t})}catch(d){console.error("[debugger] Error listing files:",d),l.status(500).json({error:d.message})}});function i(c){if(typeof c=="string"&&c){let l=k.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 d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=d.filePath;try{s?.(f);let h=await E.readFile(f,"utf-8"),g=await E.stat(f),u=Ie(h),p=Lt(h,f,n);if(p.suite){let w=p.suite,y={tests:w.tests.map(b=>({name:b.name,statements:b.testFlow.statements??[],teardown:b.testFlow.teardown,skip:b.skip,timeout:b.timeout,fail:b.fail,only:b.only,slow:b.slow})),beforeAll:me(w.beforeAll),afterAll:me(w.afterAll),beforeEach:me(w.beforeEach),afterEach:me(w.afterEach)},m={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:y};l.json({isSuite:!0,testFlow:m,metadata:u,name:p.name,tags:p.tags,use:p.use,filePath:f,fileName:k.basename(f),lastModified:g.mtimeMs})}else{let w=p.testFlow;Be(h,w),l.json({isSuite:!1,testFlow:w,metadata:u,name:p.name,tags:p.tags,use:p.use,parameters:p.parameters,filePath:f,fileName:k.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 d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=d.filePath,{testFlow:h,metadata:g}=c.body;if(!h)return l.status(400).json({error:"testFlow is required"});let u=k.join(n??t,"templates"),p=await hi(u),w=Nt(h,p),y=Le(w.statements??[]),m={...w,statements:y},b=_t(m,g),x=f+".tmp";await E.writeFile(x,b,"utf-8"),await E.rename(x,f);let T=await E.stat(f);l.json({success:!0,lastModified:T.mtimeMs})}catch(d){console.error("[debugger] Error saving test flow:",d),l.status(500).json({error:d.message})}}),o.get("/api/fixtures",async(c,l)=>{try{let d=a();try{let h=(await E.readdir(d,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:h,dir:d})}catch(f){if(f.code==="ENOENT")l.json({files:[],dir:d});else throw f}}catch(d){console.error("[debugger] Error listing fixtures:",d),l.status(500).json({error:d.message})}}),o.post("/api/fixtures",async(c,l)=>{try{let d=a();await E.mkdir(d,{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=k.basename(f);if(!g)return l.status(400).json({error:"Invalid file name"});let u=k.join(d,g);await E.writeFile(u,Buffer.from(h,"base64")),l.json({fileName:g})}catch(d){console.error("[debugger] Error saving fixture:",d),l.status(500).json({error:d.message})}}),o.get("/api/functions",async(c,l)=>{let d=k.join(n??t,"helpers"),f;try{f=await E.readdir(d)}catch(p){return p.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading helpers dir:",p),l.status(500).json({error:p.message}))}let h=f.filter(p=>/\.(ts|js|mjs)$/.test(p)),g=[],u=1;for(let p of h){let w=await E.readFile(k.join(d,p),"utf-8"),y=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,m;for(;(m=y.exec(w))!==null;){let[,b,x]=m,T=x.replace(/\s*:\s*[^,)]+/g,"");g.push({id:u++,name:`helpers/${p}#${b}`,description:"",status:"Active",code:`async function ${b}${T} {}`})}}l.json(g)}),o.get("/api/reusable-steps",async(c,l)=>{let d=k.join(n??t,"templates"),f;try{f=await E.readdir(d)}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 p=await E.readFile(k.join(d,u),"utf-8"),w=Oe(p);if(!w||typeof w!="object")continue;let y=w.name??k.basename(u,".yaml"),m=w.description??"",b=me(w.statements);g.push({id:Ze(y),organizationId:"local",name:y,description:m,statements:b})}catch(p){console.error(`[debugger] Error parsing template ${u}:`,p)}l.json(g)}),o.post("/api/reusable-steps/exists/:name",async(c,l)=>{let d=k.join(n??t,"templates"),f=decodeURIComponent(c.params.name).toLowerCase(),h;try{h=await E.readdir(d)}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(k.join(d,g),"utf-8");if((Oe(u)?.name??k.basename(g,".yaml")).toLowerCase()===f)return l.json(!0)}catch{}l.json(!1)}),o.post("/api/reusable-steps",async(c,l)=>{let d=k.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(d,{recursive:!0});let h=f.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=k.join(d,`${h}.yaml`),u={name:f.name,statements:Ct(f.statements)};f.description&&(u.description=f.description),await E.writeFile(g,qe(u),"utf-8"),l.json({id:Ze(f.name),organizationId:"local",name:f.name,description:f.description??"",statements:f.statements})}),o.put("/api/reusable-steps/:id/update",async(c,l)=>{let d=parseInt(c.params.id,10),f=k.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=k.join(f,g),p=Oe(await E.readFile(u,"utf-8"))??{},w=p.name??k.basename(g,".yaml");if(Ze(w)!==d)continue;let y=c.body??{};return y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=Ct(y.statements)),await E.writeFile(u,qe(p),"utf-8"),l.json({id:d,organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template with id ${d} not found`})}),o.put("/api/templates/:name",async(c,l)=>{let d=k.join(n??t,"templates"),f=decodeURIComponent(c.params.name),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=k.join(d,g),p=Oe(await E.readFile(u,"utf-8"))??{};if((p.name??k.basename(g,".yaml"))!==f)continue;let y=c.body??{};return y.name!==void 0&&(p.name=y.name),y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=Ct(y.statements)),await E.writeFile(u,qe(p),"utf-8"),l.json({organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template "${f}" not found`})}),o.get("/api/test-results",async(c,l)=>{let d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=n??t,h=k.basename(d.filePath,".test.yaml"),g=k.join(f,".shiplight","artifacts",h),u=[],p=null;try{let w=await E.readdir(g,{withFileTypes:!0});for(let y of w)if(y.isDirectory()){let m=k.join(g,y.name),b=await E.readdir(m);for(let x of b.filter(T=>/\.(png|jpe?g|webp)$/i.test(T))){let T=k.relative(g,k.join(m,x)),$=y.name.match(/^(.+?)_(before|after)$/),X=$?$[1]:y.name,R=$?$[2]:"screenshot";u.push({url:`/api/report-assets/${h}/${T}`,label:R,stepId:X})}}else/\.(webm|mp4)$/i.test(y.name)&&(p||(p=`/api/report-assets/${h}/${y.name}`));u.sort((y,m)=>y.stepId!==m.stepId?(y.stepId??"").localeCompare(m.stepId??""):y.label.localeCompare(m.label))}catch{}if(u.length===0&&!p)return l.status(404).json({error:"No test results found"});l.json({videoPath:p,screenshots:u})}),o}var Pn=_(()=>{"use strict";ue();Rt()});import*as In from"http";import*as Mn from"net";import*as ie from"path";import*as On from"fs";import{randomUUID as gi}from"crypto";function An(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 $n(e,t,r,n,s,o){return new Promise(a=>{let i=Array.isArray(o)?o[0]:o,c=In.request({hostname:r,port:n,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(s.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",d=>t.write(d)),l.on("end",()=>{t.end(),a()}),l.on("error",()=>{t.writableEnded||t.end(),a()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),a()}),c.end(s)})}function jt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var En,Qe,Ln=_(()=>{"use strict";yt();En=1e4,Qe=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??mt,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=ie.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(!On.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!Ae(ie.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${gi().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"idle",exitInfo:null},c={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,c),this.byYamlPath.set(r,o),this.notifyStateChange(i),this.log(`[manager] session ${o} created (idle) for ${ie.basename(r)}`),{...i}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let n=r.session.yamlPath,s=Ae(ie.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 a=En+18e4,i,c=new Promise((d,f)=>{i=setTimeout(()=>f(new Error(`Timed out after ${a/1e3}s waiting for inner playwright to register on a port.`)),a)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),c])}finally{i&&clearTimeout(i)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup,r.livenessTimer=this.startLivenessProbe(t,d=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,d)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ie.basename(n)})`)})();r.readyPromise=o;try{await o}catch(a){throw this.resetToIdle(r,a.message),a}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(n){this.log(`[manager] stopSandbox ${t} cleanup error: ${n.message}`)}this.resetToIdle(r,null)}}startLivenessProbe(t,r){let o=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"||i.session.status==="idle"){clearInterval(a);return}let c=!1;try{process.kill(i.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?o=0:(o+=1,o>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(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=Ae(ie.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=En+18e4,a,i=new Promise((l,d)=>{a=setTimeout(()=>d(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}),i])}catch(l){throw this.resetToIdle(r,l.message),l}finally{a&&clearTimeout(a)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup,r.livenessTimer=this.startLivenessProbe(t,l=>{(r.session.status==="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=${ie.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,a)=>{let i=this.sessions.get(t);if(!i){o.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}if(i.session.status==="idle"){o.status(503).json({status:"error",message:"Sandbox not started"});return}let c=`${s.method} ${s.path}`;if(c==="POST /api/int-runner/create-session"){let d=await An(s),f={};if(d.length)try{f=JSON.parse(d.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}f.testFilePath=i.session.yamlPath,await $n(s,o,i.session.innerHost,i.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(d){o.status(500).json({status:"error",message:d.message})}return}if(c==="POST /api/int-runner/liveview-url"){let d=n?.(s)??"";o.json({liveviewUrl:d,browserWsUrl:""});return}let l=await An(s);await $n(s,o,i.session.innerHost,i.session.innerPort,l,s.headers["content-type"])}}wsUpgradeFor(t){return async(r,n,s,o)=>{let a=this.sessions.get(t);if(!a){jt(n,`HTTP/1.1 404 Not Found\r
223
+ `);for(let s of n)r.push(` ${s}`);return r.push("}"),r});S("function",(e,t,r)=>{let n=e.action_data?.kwargs||{},s=n.functionName;if(s&&s.includes("#")){let[a,i]=s.split("#");if(a&&i){let c=a.replace(/\.(ts|js|mjs)$/,""),l=`import { ${i} } from '${c}';`;r?.imports?.add(l);let d={...n,functionName:i},f=hn(d);return f?[f.endsWith(";")?f:`${f};`]:["// Skipping function: invalid export pattern"]}}let o=hn(n);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});S("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});S("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],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},`),"});"]});S("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);S("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);S("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});S("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(n=>` ${n},`),"});"]});S("done",()=>["// Done - no action needed"]);S("click_by_coordinates",e=>Ot("click_by_coordinates",e));S("right_click_by_coordinates",e=>Ot("right_click_by_coordinates",e));S("double_click_by_coordinates",e=>Ot("double_click_by_coordinates",e));S("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},`),"});"]});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},`),"});"]});S("extract_email_content",e=>{let t=e.action_data?.kwargs||{};return['await agent.execAction("extract_email_content", page, {',` action_data: { kwargs: ${JSON.stringify(t)} },`,"});"]});S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
224
+ `):["// Skipping js_action: missing code"]});Yo=["testContext","request"];yn=5;Rt=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as hi}from"express";import*as E from"fs/promises";import*as T from"path";import{stringify as Ce,parse as _e}from"yaml";function Le(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(([a])=>a!=="element_index")):n?.kwargs,o={...r};if(delete o.xpath,n&&s!==void 0){let a={...n,kwargs:s};r.action_data?o.action_data=a:o.action=a}return{...t,action_entity:o}}return t.type==="STEP"?{...t,statements:Le(t.statements)}:t.type==="IF_ELSE"?{...t,then:Le(t.then),...t.else?{else:Le(t.else)}:{}}:t.type==="WHILE_LOOP"?{...t,body:Le(t.body)}:t})}function me(e){if(!e||e.length===0)return[];let r=Ce({goal:"_hook",statements:e});return V(r).statements??[]}async function kn(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 kn(T.join(e,r.name))))return!0}catch{}return!1}function Qe(e){let t=2166136261;for(let r=0;r<e.length;r++)t^=e.charCodeAt(r),t=Math.imul(t,16777619)>>>0;return t%2147483647+1}async function gi(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),a=_e(await E.readFile(s,"utf-8"))?.name??T.basename(n,".yaml");t.set(Qe(a),`templates/${n}`)}catch{}return t}function Dt(e,t){if(Array.isArray(e))return e.map(o=>Dt(o,t));if(!e||typeof e!="object")return e;let r=e,n={};for(let[o,a]of Object.entries(r))n[o]=Dt(a,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 et(e){return We({statements:e}).statements}async function Re(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=_e(await E.readFile(n,"utf-8"))??{};s.statements=et(r.statements),await E.writeFile(n,Ce(s),"utf-8")}catch{}}await Re(r.statements,t)}else r.type==="IF_ELSE"?(await Re(r.then,t),r.else&&await Re(r.else,t)):r.type==="WHILE_LOOP"&&await Re(r.body,t)}function En(e){let{initialDir:t,initialFile:r,projectRoot:n,onFileSelected:s}=e,o=hi();function a(){return T.join(n??t,"fixtures")}o.get("/api/files",async(c,l)=>{try{let d=typeof c.query.dir=="string"?c.query.dir:t,f=T.resolve(d),h=await E.readdir(f,{withFileTypes:!0}),g=[];for(let p of h)if(p.name!=="node_modules"&&!p.name.startsWith("."))if(p.isDirectory()){let w=T.join(f,p.name);await kn(w)&&g.push({name:p.name,type:"directory",path:w})}else p.isFile()&&p.name.endsWith(".test.yaml")&&g.push({name:p.name,type:"file",path:T.join(f,p.name)});g.sort((p,w)=>p.type!==w.type?p.type==="directory"?-1:1:p.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(d){console.error("[debugger] Error listing files:",d),l.status(500).json({error:d.message})}});function i(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 d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=d.filePath;try{s?.(f);let h=await E.readFile(f,"utf-8"),g=await E.stat(f),u=Oe(h),p=Ct(h,f,n);if(p.suite){let w=p.suite,y={tests:w.tests.map(b=>({name:b.name,statements:b.testFlow.statements??[],teardown:b.testFlow.teardown,skip:b.skip,timeout:b.timeout,fail:b.fail,only:b.only,slow:b.slow})),beforeAll:me(w.beforeAll),afterAll:me(w.afterAll),beforeEach:me(w.beforeEach),afterEach:me(w.afterEach)},m={version:"1.3.0",baseURL:p.use?.baseURL,testGroup:y};l.json({isSuite:!0,testFlow:m,metadata:u,name:p.name,tags:p.tags,use:p.use,filePath:f,fileName:T.basename(f),lastModified:g.mtimeMs})}else{let w=p.testFlow;Ke(h,w),l.json({isSuite:!1,testFlow:w,metadata:u,name:p.name,tags:p.tags,use:p.use,parameters:p.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 d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=d.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"),p=await gi(u),w=Dt(h,p),y=Le(w.statements??[]),m={...w,statements:y},b=n??t,x=[m.statements??[],m.teardown??[],...m.testGroup?.tests?.flatMap(M=>[M.statements,...M.teardown?[M.teardown]:[]])??[],...m.testGroup?.beforeEach?[m.testGroup.beforeEach]:[],...m.testGroup?.afterEach?[m.testGroup.afterEach]:[]];for(let M of x)await Re(M,b);let k=Tt(m,g),A=f+".tmp";await E.writeFile(A,k,"utf-8"),await E.rename(A,f);let G=await E.stat(f);l.json({success:!0,lastModified:G.mtimeMs})}catch(d){console.error("[debugger] Error saving test flow:",d),l.status(500).json({error:d.message})}}),o.get("/api/fixtures",async(c,l)=>{try{let d=a();try{let h=(await E.readdir(d,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:h,dir:d})}catch(f){if(f.code==="ENOENT")l.json({files:[],dir:d});else throw f}}catch(d){console.error("[debugger] Error listing fixtures:",d),l.status(500).json({error:d.message})}}),o.post("/api/fixtures",async(c,l)=>{try{let d=a();await E.mkdir(d,{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(d,g);await E.writeFile(u,Buffer.from(h,"base64")),l.json({fileName:g})}catch(d){console.error("[debugger] Error saving fixture:",d),l.status(500).json({error:d.message})}}),o.get("/api/functions",async(c,l)=>{let d=T.join(n??t,"helpers"),f;try{f=await E.readdir(d)}catch(p){return p.code==="ENOENT"?l.json([]):(console.error("[debugger] Error reading helpers dir:",p),l.status(500).json({error:p.message}))}let h=f.filter(p=>/\.(ts|js|mjs)$/.test(p)),g=[],u=1;for(let p of h){let w=await E.readFile(T.join(d,p),"utf-8"),y=/export\s+async\s+function\s+(\w+)\s*(\([^)]*\))/g,m;for(;(m=y.exec(w))!==null;){let[,b,x]=m,k=x.replace(/\s*:\s*[^,)]+/g,"");g.push({id:u++,name:`helpers/${p}#${b}`,description:"",status:"Active",code:`async function ${b}${k} {}`})}}l.json(g)}),o.get("/api/reusable-steps",async(c,l)=>{let d=T.join(n??t,"templates"),f;try{f=await E.readdir(d)}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 p=await E.readFile(T.join(d,u),"utf-8"),w=_e(p);if(!w||typeof w!="object")continue;let y=w.name??T.basename(u,".yaml"),m=w.description??"",b=me(w.statements);g.push({id:Qe(y),organizationId:"local",name:y,description:m,statements:b})}catch(p){console.error(`[debugger] Error parsing template ${u}:`,p)}l.json(g)}),o.post("/api/reusable-steps/exists/:name",async(c,l)=>{let d=T.join(n??t,"templates"),f=decodeURIComponent(c.params.name).toLowerCase(),h;try{h=await E.readdir(d)}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(d,g),"utf-8");if((_e(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 d=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(d,{recursive:!0});let h=f.name.replace(/[/\\:*?"<>|]/g,"-").trim(),g=T.join(d,`${h}.yaml`),u={name:f.name,statements:et(f.statements)};f.description&&(u.description=f.description),await E.writeFile(g,Ce(u),"utf-8"),l.json({id:Qe(f.name),organizationId:"local",name:f.name,description:f.description??"",statements:f.statements})}),o.put("/api/reusable-steps/:id/update",async(c,l)=>{let d=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),p=_e(await E.readFile(u,"utf-8"))??{},w=p.name??T.basename(g,".yaml");if(Qe(w)!==d)continue;let y=c.body??{};return y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=et(y.statements)),await E.writeFile(u,Ce(p),"utf-8"),l.json({id:d,organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template with id ${d} not found`})}),o.put("/api/templates/:name",async(c,l)=>{let d=T.join(n??t,"templates"),f=decodeURIComponent(c.params.name),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),p=_e(await E.readFile(u,"utf-8"))??{};if((p.name??T.basename(g,".yaml"))!==f)continue;let y=c.body??{};return y.name!==void 0&&(p.name=y.name),y.description!==void 0&&(p.description=y.description),y.statements!==void 0&&(p.statements=et(y.statements)),await E.writeFile(u,Ce(p),"utf-8"),l.json({organizationId:"local",...p,statements:y.statements!==void 0?y.statements:me(p.statements)})}catch{}l.status(404).json({error:`Template "${f}" not found`})}),o.get("/api/test-results",async(c,l)=>{let d=i(c.query.file);if("error"in d)return l.status(400).json({error:d.error});let f=n??t,h=T.basename(d.filePath,".test.yaml"),g=T.join(f,".shiplight","artifacts",h),u=[],p=null;try{let w=await E.readdir(g,{withFileTypes:!0});for(let y of w)if(y.isDirectory()){let m=T.join(g,y.name),b=await E.readdir(m);for(let x of b.filter(k=>/\.(png|jpe?g|webp)$/i.test(k))){let k=T.relative(g,T.join(m,x)),A=y.name.match(/^(.+?)_(before|after)$/),G=A?A[1]:y.name,M=A?A[2]:"screenshot";u.push({url:`/api/report-assets/${h}/${k}`,label:M,stepId:G})}}else/\.(webm|mp4)$/i.test(y.name)&&(p||(p=`/api/report-assets/${h}/${y.name}`));u.sort((y,m)=>y.stepId!==m.stepId?(y.stepId??"").localeCompare(m.stepId??""):y.label.localeCompare(m.label))}catch{}if(u.length===0&&!p)return l.status(404).json({error:"No test results found"});l.json({videoPath:p,screenshots:u})}),o}var Pn=_(()=>{"use strict";ue();Nt()});import*as Mn from"http";import*as On from"net";import*as ie from"path";import*as Ln from"fs";import{randomUUID as mi}from"crypto";function $n(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 In(e,t,r,n,s,o){return new Promise(a=>{let i=Array.isArray(o)?o[0]:o,c=Mn.request({hostname:r,port:n,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(s.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",d=>t.write(d)),l.on("end",()=>{t.end(),a()}),l.on("error",()=>{t.writableEnded||t.end(),a()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),a()}),c.end(s)})}function jt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var An,tt,Rn=_(()=>{"use strict";bt();An=1e4,tt=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??wt,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=ie.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(!Ln.existsSync(r))throw new Error(`YAML file not found: ${r}`);if(!$e(ie.dirname(r)))throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let o=`dbg-${mi().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:o,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"idle",exitInfo:null},c={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};return this.sessions.set(o,c),this.byYamlPath.set(r,o),this.notifyStateChange(i),this.log(`[manager] session ${o} created (idle) for ${ie.basename(r)}`),{...i}}async startSandbox(t){let r=this.sessions.get(t);if(!r)throw new Error(`No session ${t} to start sandbox for`);if(r.session.status==="running"||r.session.status==="starting"){r.readyPromise&&await r.readyPromise;return}if(r.session.status==="ended")throw new Error(`Session ${t} has ended`);let n=r.session.yamlPath,s=$e(ie.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 a=An+18e4,i,c=new Promise((d,f)=>{i=setTimeout(()=>f(new Error(`Timed out after ${a/1e3}s waiting for inner playwright to register on a port.`)),a)}),l;try{l=await Promise.race([this.spawner({yamlFilePath:n,configPath:s,tempSuffix:t,headed:this.headed}),c])}finally{i&&clearTimeout(i)}r.session.innerPort=l.port,r.session.innerHost=l.host,r.session.pid=l.pid,r.session.status="running",r.cleanup=l.cleanup,r.livenessTimer=this.startLivenessProbe(t,d=>{(r.session.status==="running"||r.session.status==="starting")&&this.resetToIdle(r,d)}),this.notifyStateChange(r.session),this.log(`[manager] session ${t} running on ${r.session.innerHost}:${r.session.innerPort} (yaml=${ie.basename(n)})`)})();r.readyPromise=o;try{await o}catch(a){throw this.resetToIdle(r,a.message),a}}async stopSandbox(t){let r=this.sessions.get(t);if(r&&r.session.status!=="idle"&&r.session.status!=="ended"){this.log(`[manager] stopSandbox ${t}`),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0);try{await r.cleanup()}catch(n){this.log(`[manager] stopSandbox ${t} cleanup error: ${n.message}`)}this.resetToIdle(r,null)}}startLivenessProbe(t,r){let o=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"||i.session.status==="idle"){clearInterval(a);return}let c=!1;try{process.kill(i.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?o=0:(o+=1,o>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.log(`[manager] closeSession ${t} (status=${r.session.status})`),this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(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=$e(ie.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=An+18e4,a,i=new Promise((l,d)=>{a=setTimeout(()=>d(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}),i])}catch(l){throw this.resetToIdle(r,l.message),l}finally{a&&clearTimeout(a)}r.session.innerPort=c.port,r.session.innerHost=c.host,r.session.pid=c.pid,r.session.status="running",r.cleanup=c.cleanup,r.livenessTimer=this.startLivenessProbe(t,l=>{(r.session.status==="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=${ie.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,a)=>{let i=this.sessions.get(t);if(!i){o.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){o.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}if(i.session.status==="idle"){o.status(503).json({status:"error",message:"Sandbox not started"});return}let c=`${s.method} ${s.path}`;if(c==="POST /api/int-runner/create-session"){let d=await $n(s),f={};if(d.length)try{f=JSON.parse(d.toString("utf-8"))}catch{o.status(400).json({status:"error",message:"Invalid JSON body"});return}f.testFilePath=i.session.yamlPath,await In(s,o,i.session.innerHost,i.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(d){o.status(500).json({status:"error",message:d.message})}return}if(c==="POST /api/int-runner/liveview-url"){let d=n?.(s)??"";o.json({liveviewUrl:d,browserWsUrl:""});return}let l=await $n(s);await In(s,o,i.session.innerHost,i.session.innerPort,l,s.headers["content-type"])}}wsUpgradeFor(t){return async(r,n,s,o)=>{let a=this.sessions.get(t);if(!a){jt(n,`HTTP/1.1 404 Not Found\r
225
225
  \r
226
226
  `);return}if(a.session.status==="ended"){jt(n,`HTTP/1.1 410 Gone\r
227
227
  \r
228
- `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,c=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),d=new URL(l.replace(/^ws/,"http")),f=parseInt(d.port||"80",10),h=d.hostname,g=d.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let u=Mn.createConnection(f,h);u.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
228
+ `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,c=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),d=new URL(l.replace(/^ws/,"http")),f=parseInt(d.port||"80",10),h=d.hostname,g=d.pathname;o&&o.startsWith("/page/")&&(g=`/devtools${o}`);let u=On.createConnection(f,h);u.on("connect",()=>{let p=`GET ${g} HTTP/1.1\r
229
229
  Host: ${h}:${f}\r
230
230
  `;for(let[w,y]of Object.entries(r.headers)){let m=w.toLowerCase();m!=="host"&&m!=="origin"&&(p+=`${w}: ${Array.isArray(y)?y.join(", "):y}\r
231
231
  `)}p+=`\r
232
232
  `,u.write(p),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(i){this.log(`[manager] WS upgrade for ${t} failed: ${i.message}`),jt(n,`HTTP/1.1 502 Bad Gateway\r
233
233
  \r
234
- `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as mi}from"express";import Dt from"express";import*as _e from"fs";import*as ae from"path";function yi(e){return e?ae.isAbsolute(e)?ae.normalize(e):ae.resolve(e):null}function wi(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 bi(e){let t=ae.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Rn(e){let{manager:t,staticDir:r,resolveYamlPath:n=yi,liveviewUrlBuilder:s=wi,fallbackRouter:o,artifactsDir:a}=e,i=mi(),c=a?Dt.static(a):null;i.post("/api/debugger/sessions",Dt.json(),async(d,f)=>{let h=d.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(!bi(g)){f.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!_e.existsSync(g)){f.status(404).json({error:"File not found"});return}let u=t.listSessions().find(p=>p.yamlPath===g&&p.status!=="ended");try{let p=t.openSession(g);f.status(u?200:201).json({sessionId:p.sessionId,yamlPath:p.yamlPath,startedAt:p.startedAt,status:p.status})}catch(p){f.status(500).json({error:p.message})}}),i.get("/api/debugger/sessions",(d,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}))})}),i.delete("/api/debugger/sessions/:sessionId",async(d,f)=>{let h=d.params.sessionId,g=!!t.getSession(h);await t.closeSession(h),f.json({deleted:!0,alreadyGone:!g})}),_e.existsSync(r)?i.use("/debugger/static",Dt.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),i.get("/debugger/:sessionId/",(d,f)=>{let h=d.params.sessionId;if(!t.getSession(h)){f.status(404).send("Debugger session not found");return}let u=ae.join(r,"index.html");if(!_e.existsSync(u)){f.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let w=_e.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)});let l=async(d,f,h)=>{let g=d.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 p=d.originalUrl,w=`/debugger/${g}`;if(p.startsWith(w)){let m=p.slice(w.length)||"/";d.url=m,Object.defineProperty(d,"originalUrl",{value:m,configurable:!0})}if(c&&d.path.startsWith("/api/report-assets/")){d.url=d.url.replace(/^\/api\/report-assets/,""),c(d,f,h);return}if(u.status==="idle"||u.status==="starting"){let m=`${d.method} ${d.path}`;if(m==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(b){f.status(500).json({status:"error",message:b.message});return}else if(m==="POST /api/int-runner/liveview-url"){f.json({liveviewUrl:"",browserWsUrl:""});return}else if(d.path.startsWith("/api/int-runner/")){f.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(d,f,h);return}else{f.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>s(g,d)})(d,f,h)};return i.use("/debugger/:sessionId",l),i}function Cn(e,t,r,n){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
234
+ `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as yi}from"express";import Ft from"express";import*as xe from"fs";import*as ae from"path";function wi(e){return e?ae.isAbsolute(e)?ae.normalize(e):ae.resolve(e):null}function bi(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 vi(e){let t=ae.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Cn(e){let{manager:t,staticDir:r,resolveYamlPath:n=wi,liveviewUrlBuilder:s=bi,fallbackRouter:o,artifactsDir:a}=e,i=yi(),c=a?Ft.static(a):null;i.post("/api/debugger/sessions",Ft.json(),async(d,f)=>{let h=d.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(!vi(g)){f.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!xe.existsSync(g)){f.status(404).json({error:"File not found"});return}let u=t.listSessions().find(p=>p.yamlPath===g&&p.status!=="ended");try{let p=t.openSession(g);f.status(u?200:201).json({sessionId:p.sessionId,yamlPath:p.yamlPath,startedAt:p.startedAt,status:p.status})}catch(p){f.status(500).json({error:p.message})}}),i.get("/api/debugger/sessions",(d,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}))})}),i.delete("/api/debugger/sessions/:sessionId",async(d,f)=>{let h=d.params.sessionId,g=!!t.getSession(h);await t.closeSession(h),f.json({deleted:!0,alreadyGone:!g})}),xe.existsSync(r)?i.use("/debugger/static",Ft.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),i.get("/debugger/:sessionId/",(d,f)=>{let h=d.params.sessionId;if(!t.getSession(h)){f.status(404).send("Debugger session not found");return}let u=ae.join(r,"index.html");if(!xe.existsSync(u)){f.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let w=xe.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)});let l=async(d,f,h)=>{let g=d.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 p=d.originalUrl,w=`/debugger/${g}`;if(p.startsWith(w)){let m=p.slice(w.length)||"/";d.url=m,Object.defineProperty(d,"originalUrl",{value:m,configurable:!0})}if(c&&d.path.startsWith("/api/report-assets/")){d.url=d.url.replace(/^\/api\/report-assets/,""),c(d,f,h);return}if(u.status==="idle"||u.status==="starting"){let m=`${d.method} ${d.path}`;if(m==="POST /api/int-runner/create-session")try{await t.startSandbox(g)}catch(b){f.status(500).json({status:"error",message:b.message});return}else if(m==="POST /api/int-runner/liveview-url"){f.json({liveviewUrl:"",browserWsUrl:""});return}else if(d.path.startsWith("/api/int-runner/")){f.status(503).json({status:"error",message:"Sandbox not started"});return}else if(o){o(d,f,h);return}else{f.status(503).json({status:"error",message:"Sandbox not started"});return}}t.httpProxyFor(g,{liveviewUrlBuilder:()=>s(g,d)})(d,f,h)};return i.use("/debugger/:sessionId",l),i}function Nn(e,t,r,n){let o=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!o){r.write(`HTTP/1.1 404 Not Found\r
235
235
  \r
236
- `),r.destroy();return}let a=o[1],i=o[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,n,i||void 0)}var Nn=_(()=>{"use strict"});var Dn={};ce(Dn,{startDebuggerServer:()=>_i});import Re from"express";import*as B from"path";import{createRequire as vi}from"node:module";async function _i(e){let{initialDir:t,initialFile:r,projectRoot:n,port:s,headed:o=!1}=e,a=new Qe({headed:o}),i=Re();i.use((y,m,b)=>{if(m.setHeader("Access-Control-Allow-Origin","*"),m.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),m.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),y.method==="OPTIONS")return m.sendStatus(204);b()}),i.use(Re.json({limit:"10mb"}));let c=kn({initialDir:t,initialFile:r,projectRoot:n});i.use(c);let l=B.join(n??t,".shiplight","artifacts");i.use("/api/report-assets",Re.static(l));let d=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,f=d.includes(B.sep+"src"+B.sep),h=f?B.resolve(d,"../../dist/static"):B.join(d,"static"),g=f?B.resolve(d,"../../dist/static-embedded"):B.join(d,"static-embedded");i.use("/devtools",Re.static(jn)),i.use(Re.static(h)),i.use(Rn({manager:a,staticDir:g,fallbackRouter:c,artifactsDir:l})),i.get("*path",(y,m,b)=>{if(y.path.startsWith("/api/"))return b();m.sendFile(B.join(h,"index.html"),x=>{x&&m.send(xi(t,s))})});let u=await new Promise((y,m)=>{let b=i.listen(s,"localhost",()=>{y(b)});b.on("error",m)});u.on("upgrade",(y,m,b)=>{if((y.url??"").startsWith("/ws/debugger/")){Cn(a,y,m,b);return}m.write(`HTTP/1.1 404 Not Found\r
236
+ `),r.destroy();return}let a=o[1],i=o[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,n,i||void 0)}var Dn=_(()=>{"use strict"});var Fn={};ce(Fn,{startDebuggerServer:()=>xi});import Ne from"express";import*as B from"path";import{createRequire as Si}from"node:module";async function xi(e){let{initialDir:t,initialFile:r,projectRoot:n,port:s,headed:o=!1}=e,a=new tt({headed:o}),i=Ne();i.use((y,m,b)=>{if(m.setHeader("Access-Control-Allow-Origin","*"),m.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),m.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),y.method==="OPTIONS")return m.sendStatus(204);b()}),i.use(Ne.json({limit:"10mb"}));let c=En({initialDir:t,initialFile:r,projectRoot:n});i.use(c);let l=B.join(n??t,".shiplight","artifacts");i.use("/api/report-assets",Ne.static(l));let d=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,f=d.includes(B.sep+"src"+B.sep),h=f?B.resolve(d,"../../dist/static"):B.join(d,"static"),g=f?B.resolve(d,"../../dist/static-embedded"):B.join(d,"static-embedded");i.use("/devtools",Ne.static(jn)),i.use(Ne.static(h)),i.use(Cn({manager:a,staticDir:g,fallbackRouter:c,artifactsDir:l})),i.get("*path",(y,m,b)=>{if(y.path.startsWith("/api/"))return b();m.sendFile(B.join(h,"index.html"),x=>{x&&m.send(Ti(t,s))})});let u=await new Promise((y,m)=>{let b=i.listen(s,"localhost",()=>{y(b)});b.on("error",m)});u.on("upgrade",(y,m,b)=>{if((y.url??"").startsWith("/ws/debugger/")){Nn(a,y,m,b);return}m.write(`HTTP/1.1 404 Not Found\r
237
237
  \r
238
- `),m.destroy()});let p=`http://localhost:${s}`,w=r?`${p}/?open=${encodeURIComponent(r)}`:p;return console.error(`[debugger] Server running at ${p}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:w,close:async()=>{await a.shutdown(),await new Promise((y,m)=>{u.close(b=>b?m(b):y())})}}}function xi(e,t){return`<!DOCTYPE html>
238
+ `),m.destroy()});let p=`http://localhost:${s}`,w=r?`${p}/?open=${encodeURIComponent(r)}`:p;return console.error(`[debugger] Server running at ${p}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:w,close:async()=>{await a.shutdown(),await new Promise((y,m)=>{u.close(b=>b?m(b):y())})}}}function Ti(e,t){return`<!DOCTYPE html>
239
239
  <html>
240
240
  <head>
241
241
  <meta charset="utf-8">
@@ -253,27 +253,27 @@ Host: ${h}:${f}\r
253
253
  <p>Directory: <code>${e}</code></p>
254
254
  <p>Server: localhost:${t}</p>
255
255
  </body>
256
- </html>`}var Si,jn,Fn=_(()=>{"use strict";Pn();Ln();Nn();Si=vi(import.meta.url);try{jn=B.join(B.dirname(Si.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 Hn={};ce(Hn,{startDebugger:()=>ki});import*as te from"fs";import*as he from"path";function et(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function ki(e){let t,r=xe,n=!1,s,o=!1,a=!0,i=!1;for(let m=0;m<e.length;m++)e[m]==="--port"&&e[m+1]?(r=parseInt(e[m+1],10),n=!0,m++):e[m]==="--url"&&e[m+1]?(s=e[m+1],m++):e[m]==="--new"?o=!0:e[m]==="--open"?a=!1:e[m]==="--no-open"?a=!0:e[m]==="--headed"?i=!0:e[m]==="--help"||e[m]==="-h"?(Pi(),process.exit(0)):e[m].startsWith("--")||(t=e[m]);let c,l;if(t){let m=he.resolve(t);te.existsSync(m)&&te.statSync(m).isDirectory()?c=m:(c=he.dirname(m),l=m)}else c=process.cwd();if(o&&l&&!te.existsSync(l)){let m=he.dirname(l);te.existsSync(m)||te.mkdirSync(m,{recursive:!0}),te.writeFileSync(l,Ti(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${l}`)}l&&!te.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:d}=await Promise.resolve().then(()=>(yt(),Pr)),f=d(c),h=f?he.dirname(f):c;(await import("dotenv")).config({path:he.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(()=>(Fn(),Dn));if(console.log(l?`Starting debugger for: ${l}`:`Starting debugger in: ${c}`),f&&console.log(`Using Playwright config: ${f}`),!n){let m=await xr(xe,Un);m===null&&(console.error(et(`Error: No available port found in range ${xe}-${xe+Un-1}.`)),console.error(et("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),m!==xe&&console.log(`Port ${xe} is in use; using port ${m} instead.`),r=m}let p;try{p=await u({initialDir:c,initialFile:l,projectRoot:h,port:r,headed:i})}catch(m){throw m?.code==="EADDRINUSE"&&(console.error(et(`Error: Port ${r} is already in use.`)),console.error(et(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)),m}if(wr(r,c),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!a)try{let{default:m}=await import("open");await m(p.url)}catch{}let w=!1,y=async()=>{w&&(console.log(`
256
+ </html>`}var _i,jn,Un=_(()=>{"use strict";Pn();Rn();Dn();_i=Si(import.meta.url);try{jn=B.join(B.dirname(_i.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 Bn={};ce(Bn,{startDebugger:()=>Ei});import*as te from"fs";import*as he from"path";function rt(e){return process.stderr.isTTY?`\x1B[31m${e}\x1B[0m`:e}async function Ei(e){let t,r=Te,n=!1,s,o=!1,a=!0,i=!1;for(let m=0;m<e.length;m++)e[m]==="--port"&&e[m+1]?(r=parseInt(e[m+1],10),n=!0,m++):e[m]==="--url"&&e[m+1]?(s=e[m+1],m++):e[m]==="--new"?o=!0:e[m]==="--open"?a=!1:e[m]==="--no-open"?a=!0:e[m]==="--headed"?i=!0:e[m]==="--help"||e[m]==="-h"?(Pi(),process.exit(0)):e[m].startsWith("--")||(t=e[m]);let c,l;if(t){let m=he.resolve(t);te.existsSync(m)&&te.statSync(m).isDirectory()?c=m:(c=he.dirname(m),l=m)}else c=process.cwd();if(o&&l&&!te.existsSync(l)){let m=he.dirname(l);te.existsSync(m)||te.mkdirSync(m,{recursive:!0}),te.writeFileSync(l,ki(s||"https://example.com"),"utf-8"),console.log(`Created new test file: ${l}`)}l&&!te.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:d}=await Promise.resolve().then(()=>(bt(),Pr)),f=d(c),h=f?he.dirname(f):c;(await import("dotenv")).config({path:he.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(()=>(Un(),Fn));if(console.log(l?`Starting debugger for: ${l}`:`Starting debugger in: ${c}`),f&&console.log(`Using Playwright config: ${f}`),!n){let m=await Tr(Te,Hn);m===null&&(console.error(rt(`Error: No available port found in range ${Te}-${Te+Hn-1}.`)),console.error(rt("Close some debugger sessions, or pass --port <number> to pick a specific port.")),process.exit(1)),m!==Te&&console.log(`Port ${Te} is in use; using port ${m} instead.`),r=m}let p;try{p=await u({initialDir:c,initialFile:l,projectRoot:h,port:r,headed:i})}catch(m){throw m?.code==="EADDRINUSE"&&(console.error(rt(`Error: Port ${r} is already in use.`)),console.error(rt(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)),m}if(br(r,c),console.log(`Debugger running at: ${p.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!a)try{let{default:m}=await import("open");await m(p.url)}catch{}let w=!1,y=async()=>{w&&(console.log(`
257
257
  Force exiting...`),process.exit(1)),w=!0,console.log(`
258
- Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{br(r,c),await p.close()}catch{}clearTimeout(m),process.exit(0)};process.on("SIGINT",y),process.on("SIGTERM",y)}function Pi(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" --headed Force a visible Chromium window (overrides use.headless)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var xe,Un,Ti,Bn=_(()=>{"use strict";vr();Tr();xe=6174,Un=10;Ti=e=>`goal: New test
258
+ Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{vr(r,c),await p.close()}catch{}clearTimeout(m),process.exit(0)};process.on("SIGINT",y),process.on("SIGTERM",y)}function Pi(){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 Te,Hn,ki,Gn=_(()=>{"use strict";Sr();kr();Te=6174,Hn=10;ki=e=>`goal: New test
259
259
  base_url: ${e}
260
260
  statements:
261
261
  - URL: /
262
- `});import zp from"dotenv";function Ai(){return globalThis[Ei]}function tt(){let e=Ai();return e===void 0?process.env:e}var Ei,Ft=_(()=>{"use strict";Ei="__shiplightDotenvCache__"});function rt(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith($i)?Ii:Mi}var $i,Ii,Mi,Ut=_(()=>{"use strict";$i="shp_",Ii="https://nova-api.shiplight.ai",Mi="https://api.shiplight.ai"});var Ht={};ce(Ht,{lookupActionStores:()=>Ri,updateActionStores:()=>Ci});function Gn(){let e=tt(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:rt(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function Ri(e){let t=Gn();if(!t||e.length===0)return new Map;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Oi),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(),a=new Map;for(let[i,c]of Object.entries(o.stores??{}))a.set(i,c);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Ci(e){let t=Gn();if(!t||e.size===0)return 0;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Li),s={};for(let[i,c]of e)s[i]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores: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 Oi,Li,Bt=_(()=>{"use strict";Ut();Ft();Oi=2e3,Li=5e3});import*as V from"fs";import*as Ce from"path";import{globSync as Ni}from"glob";function Wn(e){let t=tt().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Wt:new Gt(e)}function Ne(e){return e.replace(/\//g,"__")+".json"}function Di(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ji,Gt,Wt,Kn=_(()=>{"use strict";ue();Ft();ji=".shiplight/action-cache";Gt=class{constructor(t){this.cwd=t;this.cacheDir=Ce.join(t,ji)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!V.existsSync(this.cacheDir))return r;for(let n of t){let s=Ce.join(this.cacheDir,Ne(n));try{if(V.existsSync(s)){let o=V.readFileSync(s,"utf-8");r.set(n,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;V.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[n,s]of t)try{let o=Ce.join(this.cacheDir,Ne(n)),a=kt();if(V.existsSync(o))try{a=JSON.parse(V.readFileSync(o,"utf-8"))}catch{}let i={...a,entries:{...a.entries,...s.entries}};V.writeFileSync(o,JSON.stringify(i,null,2)),r++}catch{}return r}loadAll(){if(!V.existsSync(this.cacheDir))return;let t=Ni("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,n=0;for(let s of t)try{let o=V.readFileSync(Ce.join(this.cacheDir,s),"utf-8"),a=JSON.parse(o),i=Di(s);r.set(i,a),n+=Object.keys(a.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}},Wt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Bt(),Ht));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Bt(),Ht));return r(t)}loadAll(){}}});var qn={};ce(qn,{buildPlaywrightSpawnOptions:()=>zn,buildTestPathsFromGitInfo:()=>Yn,cloudKeyToCwdRelPath:()=>Yt,extractVarOverrideArgs:()=>Xn,loadVarsFile:()=>zt,parseVarsArg:()=>Vt,runTests:()=>Ui});import{spawn as Fi,execFileSync as Vn}from"child_process";import*as U from"fs";import*as N from"path";import{globSync as Kt}from"glob";async function Ui(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=>U.existsSync(N.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"),a={};try{let{extracted:h,remaining:g}=Xn(o,t);o=g,a=h}catch(h){console.error(`[shiplight] ${h.message}`),process.exit(2)}let i=o.map(h=>h.endsWith(".test.yaml")?h.replace(/\.test\.yaml$/,".yaml.spec.ts"):h),c=Wn(t);await Bi(t,c),Pe&&process.stdout.write(`shiplightai v${Pe}
264
- `);let l={...process.env};s&&(l.SHIPLIGHT_MAGIC="1"),Object.keys(a).length>0&&(l.SHIPLIGHT_VARS_OVERRIDE=JSON.stringify(a));let d=Fi("npx",["playwright","test",...i],zn(t,l)),f=await new Promise(h=>{d.on("close",g=>h(g??1))});await Gi(t,c),process.exit(f)}function zn(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Jt(){try{return Vn("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Hi(){try{let e=Vn("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Jn(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let n=Hi(),s=Jt();return Yn(e,t,n,s)}function Yn(e,t,r,n){return{testPaths:e.map(o=>{let a=n?N.relative(n,N.resolve(t,o)):o;return`${r}${a}`}),branchPrefix:r}}function Yt(e,t,r,n){let s=t&&e.startsWith(t)?e.slice(t.length):e;return r?N.relative(n,N.resolve(r,s)):s}async function Bi(e,t){try{let r=Kt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:n,branchPrefix:s}=Jn(r,e,t.isCloud),o=await t.lookup(n);if(o.size===0)return;let a=N.join(e,".shiplight","action-cache");U.mkdirSync(a,{recursive:!0});let i=Jt(),c=0;for(let[l,d]of o){let f=t.isCloud?Yt(l,s,i,e):l,h=N.join(a,Ne(f));U.writeFileSync(h,JSON.stringify(d,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 Gi(e,t){try{let r=N.join(e,"test-results");if(!U.existsSync(r))return;let n=Kt("**/new-action-entities.json",{cwd:r});if(n.length===0)return;let s={};for(let f of n)try{let h=U.readFileSync(N.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=Kt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=Jn(o,e,t.isCloud),c=new Map;for(let f=0;f<o.length;f++){let h=o[f],g=a[f],u=N.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!U.existsSync(u))continue;let p=U.readFileSync(u,"utf-8"),w={};for(let[y,m]of Object.entries(s))p.includes(y)&&(w[y]=m);if(Object.keys(w).length>0){let y=t.isCloud?Yt(g,i,Jt(),e):g,m=N.join(e,".shiplight","action-cache",Ne(y)),b={};if(U.existsSync(m))try{b=JSON.parse(U.readFileSync(m,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...b,...w}})}}if(c.size===0)return;let l=await t.update(c),d=Array.from(c.values()).reduce((f,h)=>f+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${d} action entit${d!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}function Vt(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 zt(e,t){let r=N.isAbsolute(e)?e:N.join(t,e);if(!U.existsSync(r))throw new Error(`--vars-file not found: ${r}`);let n;try{n=JSON.parse(U.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,a]of Object.entries(n)){if(typeof a!="string")throw new Error(`--vars-file ${r}: key "${o}" must be a string (got ${typeof a})`);s[o]=a}return s}function Xn(e,t){let r={},n={},s=[];for(let o=0;o<e.length;o++){let a=e[o];if(a==="--vars"||a==="--vars-file"){let i=e[o+1];if(i===void 0||i.startsWith("--"))throw new Error(`${a} requires a value`);a==="--vars"?Object.assign(n,Vt(i)):Object.assign(r,zt(i,t)),o+=1;continue}if(a.startsWith("--vars=")){Object.assign(n,Vt(a.slice(7)));continue}if(a.startsWith("--vars-file=")){Object.assign(r,zt(a.slice(12),t));continue}s.push(a)}return{extracted:{...r,...n},remaining:s}}var Zn=_(()=>{"use strict";Kn();je()});import Qn from"node:path";function O(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function st(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function Wi(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Ki(e,t){let r=e||"?",n=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${n}`:`failed after ${r} ${n}`}function nt(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function Vi(e){return`<span class="badge badge-${e}">${e}</span>`}function zi(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 a=e.codeStartLine+o,i=a===e.codeLine,c=String(a).padStart(4);return`<span class="code-line${i?" code-line-active":""}">${c} \u2502 ${O(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(n=>`<span class="code-line code-line-body">${O(n)}</span>`).join("")}</pre></div>`}function Ji(e){let t=e.duration!=null?`<span class="step-duration">${st(e.duration)}</span>`:"",r=nt(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${O(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":zi(e),a="";e.error&&(a=`<div class="step-error"><pre>${O(e.error)}</pre></div>`);let i="";e.message&&!e.error&&(i=`<div class="step-message">${O(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
262
+ `});import Jp from"dotenv";function $i(){return globalThis[Ai]}function nt(){let e=$i();return e===void 0?process.env:e}var Ai,Ut=_(()=>{"use strict";Ai="__shiplightDotenvCache__"});function st(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith(Ii)?Mi:Oi}var Ii,Mi,Oi,Ht=_(()=>{"use strict";Ii="shp_",Mi="https://nova-api.shiplight.ai",Oi="https://api.shiplight.ai"});var Bt={};ce(Bt,{lookupActionStores:()=>Ci,updateActionStores:()=>Ni});function Wn(){let e=nt(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:st(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function Ci(e){let t=Wn();if(!t||e.length===0)return new Map;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Li),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(),a=new Map;for(let[i,c]of Object.entries(o.stores??{}))a.set(i,c);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Ni(e){let t=Wn();if(!t||e.size===0)return 0;try{let r=new AbortController,n=setTimeout(()=>r.abort(),Ri),s={};for(let[i,c]of e)s[i]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores: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 Li,Ri,Gt=_(()=>{"use strict";Ht();Ut();Li=2e3,Ri=5e3});import*as z from"fs";import*as De from"path";import{globSync as Di}from"glob";function Kn(e){let t=nt().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Kt:new Wt(e)}function je(e){return e.replace(/\//g,"__")+".json"}function Fi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ji,Wt,Kt,Vn=_(()=>{"use strict";ue();Ut();ji=".shiplight/action-cache";Wt=class{constructor(t){this.cwd=t;this.cacheDir=De.join(t,ji)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!z.existsSync(this.cacheDir))return r;for(let n of t){let s=De.join(this.cacheDir,je(n));try{if(z.existsSync(s)){let o=z.readFileSync(s,"utf-8");r.set(n,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;z.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[n,s]of t)try{let o=De.join(this.cacheDir,je(n)),a=Pt();if(z.existsSync(o))try{a=JSON.parse(z.readFileSync(o,"utf-8"))}catch{}let i={...a,entries:{...a.entries,...s.entries}};z.writeFileSync(o,JSON.stringify(i,null,2)),r++}catch{}return r}loadAll(){if(!z.existsSync(this.cacheDir))return;let t=Di("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,n=0;for(let s of t)try{let o=z.readFileSync(De.join(this.cacheDir,s),"utf-8"),a=JSON.parse(o),i=Fi(s);r.set(i,a),n+=Object.keys(a.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}},Kt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Gt(),Bt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Gt(),Bt));return r(t)}loadAll(){}}});var Zn={};ce(Zn,{buildPlaywrightSpawnOptions:()=>Jn,buildTestPathsFromGitInfo:()=>Xn,cloudKeyToCwdRelPath:()=>Xt,extractVarOverrideArgs:()=>qn,loadVarsFile:()=>Jt,parseVarsArg:()=>zt,runTests:()=>Hi});import{spawn as Ui,execFileSync as zn}from"child_process";import*as U from"fs";import*as N from"path";import{globSync as Vt}from"glob";async function Hi(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=>U.existsSync(N.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"),a={};try{let{extracted:h,remaining:g}=qn(o,t);o=g,a=h}catch(h){console.error(`[shiplight] ${h.message}`),process.exit(2)}let i=o.map(h=>h.endsWith(".test.yaml")?h.replace(/\.test\.yaml$/,".yaml.spec.ts"):h),c=Kn(t);await Gi(t,c),Pe&&process.stdout.write(`shiplightai v${Pe}
264
+ `);let l={...process.env};s&&(l.SHIPLIGHT_MAGIC="1"),Object.keys(a).length>0&&(l.SHIPLIGHT_VARS_OVERRIDE=JSON.stringify(a));let d=Ui("npx",["playwright","test",...i],Jn(t,l)),f=await new Promise(h=>{d.on("close",g=>h(g??1))});await Wi(t,c),process.exit(f)}function Jn(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Yt(){try{return zn("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Bi(){try{let e=zn("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Yn(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let n=Bi(),s=Yt();return Xn(e,t,n,s)}function Xn(e,t,r,n){return{testPaths:e.map(o=>{let a=n?N.relative(n,N.resolve(t,o)):o;return`${r}${a}`}),branchPrefix:r}}function Xt(e,t,r,n){let s=t&&e.startsWith(t)?e.slice(t.length):e;return r?N.relative(n,N.resolve(r,s)):s}async function Gi(e,t){try{let r=Vt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:n,branchPrefix:s}=Yn(r,e,t.isCloud),o=await t.lookup(n);if(o.size===0)return;let a=N.join(e,".shiplight","action-cache");U.mkdirSync(a,{recursive:!0});let i=Yt(),c=0;for(let[l,d]of o){let f=t.isCloud?Xt(l,s,i,e):l,h=N.join(a,je(f));U.writeFileSync(h,JSON.stringify(d,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 Wi(e,t){try{let r=N.join(e,"test-results");if(!U.existsSync(r))return;let n=Vt("**/new-action-entities.json",{cwd:r});if(n.length===0)return;let s={};for(let f of n)try{let h=U.readFileSync(N.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=Vt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=Yn(o,e,t.isCloud),c=new Map;for(let f=0;f<o.length;f++){let h=o[f],g=a[f],u=N.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!U.existsSync(u))continue;let p=U.readFileSync(u,"utf-8"),w={};for(let[y,m]of Object.entries(s))p.includes(y)&&(w[y]=m);if(Object.keys(w).length>0){let y=t.isCloud?Xt(g,i,Yt(),e):g,m=N.join(e,".shiplight","action-cache",je(y)),b={};if(U.existsSync(m))try{b=JSON.parse(U.readFileSync(m,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...b,...w}})}}if(c.size===0)return;let l=await t.update(c),d=Array.from(c.values()).reduce((f,h)=>f+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${d} action entit${d!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}function zt(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 Jt(e,t){let r=N.isAbsolute(e)?e:N.join(t,e);if(!U.existsSync(r))throw new Error(`--vars-file not found: ${r}`);let n;try{n=JSON.parse(U.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,a]of Object.entries(n)){if(typeof a!="string")throw new Error(`--vars-file ${r}: key "${o}" must be a string (got ${typeof a})`);s[o]=a}return s}function qn(e,t){let r={},n={},s=[];for(let o=0;o<e.length;o++){let a=e[o];if(a==="--vars"||a==="--vars-file"){let i=e[o+1];if(i===void 0||i.startsWith("--"))throw new Error(`${a} requires a value`);a==="--vars"?Object.assign(n,zt(i)):Object.assign(r,Jt(i,t)),o+=1;continue}if(a.startsWith("--vars=")){Object.assign(n,zt(a.slice(7)));continue}if(a.startsWith("--vars-file=")){Object.assign(r,Jt(a.slice(12),t));continue}s.push(a)}return{extracted:{...r,...n},remaining:s}}var Qn=_(()=>{"use strict";Vn();Fe()});import es from"node:path";function L(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function it(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function Ki(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Vi(e,t){let r=e||"?",n=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${n}`:`failed after ${r} ${n}`}function ot(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 zi(e){return`<span class="badge badge-${e}">${e}</span>`}function Ji(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 a=e.codeStartLine+o,i=a===e.codeLine,c=String(a).padStart(4);return`<span class="code-line${i?" 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 Yi(e){let t=e.duration!=null?`<span class="step-duration">${it(e.duration)}</span>`:"",r=ot(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?"":Ji(e),a="";e.error&&(a=`<div class="step-error"><pre>${L(e.error)}</pre></div>`);let i="";e.message&&!e.error&&(i=`<div class="step-message">${L(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
266
266
  <details class="step-details step step-${e.status}"${c}>
267
267
  <summary class="step-header">
268
268
  ${r}
269
- <span class="step-id">${O(e.stepId)}</span>
270
- <span class="step-description-collapsed">${O(e.description)}</span>
269
+ <span class="step-id">${L(e.stepId)}</span>
270
+ <span class="step-description-collapsed">${L(e.description)}</span>
271
271
  ${t}
272
272
  </summary>
273
273
  <div class="step-expanded">
274
274
  ${s}
275
275
  ${o}
276
- <div class="step-description-full">${O(e.description)}</div>
276
+ <div class="step-description-full">${L(e.description)}</div>
277
277
  ${i}
278
278
  ${a}
279
279
  </div>
@@ -281,32 +281,32 @@ statements:
281
281
  <div class="step step-${e.status}">
282
282
  <div class="step-header">
283
283
  ${r}
284
- <span class="step-id">${O(e.stepId)}</span>
285
- <span class="step-description">${O(e.description)}</span>
284
+ <span class="step-id">${L(e.stepId)}</span>
285
+ <span class="step-description">${L(e.description)}</span>
286
286
  ${t}
287
287
  </div>
288
- </div>`}function Yi(e,t,r,n){let s=[];e&&s.push(`
288
+ </div>`}function Xi(e,t,r,n){let s=[];e&&s.push(`
289
289
  <details class="artifact-section">
290
290
  <summary class="artifact-summary">Video</summary>
291
291
  <div class="artifact-content">
292
292
  <div class="video-container">
293
293
  <video controls preload="metadata" class="artifact-video">
294
- <source src="${O(e)}" type="video/webm" />
294
+ <source src="${L(e)}" type="video/webm" />
295
295
  </video>
296
296
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
297
297
  </div>
298
298
  </div>
299
- </details>`);let o=r.filter(a=>a.screenshot).map(a=>({src:a.screenshot,stepId:a.stepId,description:a.description,status:a.status,message:a.message||a.error||""}));if(o.length>0){let a=O(JSON.stringify(o)),i=o.map((c,l)=>`
299
+ </details>`);let o=r.filter(a=>a.screenshot).map(a=>({src:a.screenshot,stepId:a.stepId,description:a.description,status:a.status,message:a.message||a.error||""}));if(o.length>0){let a=L(JSON.stringify(o)),i=o.map((c,l)=>`
300
300
  <div class="screenshot-thumb" onclick="openGalleryAt(this, ${l})" data-gallery="${a}">
301
- <img src="${O(c.src)}" alt="${O(c.stepId)}" />
302
- <span class="thumb-label">${O(c.stepId)}</span>
301
+ <img src="${L(c.src)}" alt="${L(c.stepId)}" />
302
+ <span class="thumb-label">${L(c.stepId)}</span>
303
303
  </div>`).join("");s.push(`
304
304
  <details class="artifact-section">
305
305
  <summary class="artifact-summary">Screenshots (${o.length})</summary>
306
306
  <div class="artifact-content">
307
307
  <div class="screenshot-grid">${i}</div>
308
308
  </div>
309
- </details>`)}if(t){let a=O(t);s.push(`
309
+ </details>`)}if(t){let a=L(t);s.push(`
310
310
  <details class="artifact-section">
311
311
  <summary class="artifact-summary">Trace</summary>
312
312
  <div class="artifact-content">
@@ -315,40 +315,40 @@ statements:
315
315
  <button class="copy-btn" onclick="copyTraceCmd('${n}')" title="Copy command">Copy</button>
316
316
  </div>
317
317
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
318
- <p class="trace-hint"><a href="${O(t)}" class="attachment-link" download>Download trace.zip</a></p>
318
+ <p class="trace-hint"><a href="${L(t)}" class="attachment-link" download>Download trace.zip</a></p>
319
319
  </div>
320
- </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function es(e,t,r,n,s){let o=e.map(Ji).join(`
321
- `),a="";t&&!e.some(c=>c.error)&&(a=`<div class="test-error"><pre>${O(t)}</pre></div>`);let i=Yi(r,n,e,s);return`
320
+ </details>`)}return s.length===0?"":`<div class="test-artifacts">${s.join("")}</div>`}function ts(e,t,r,n,s){let o=e.map(Yi).join(`
321
+ `),a="";t&&!e.some(c=>c.error)&&(a=`<div class="test-error"><pre>${L(t)}</pre></div>`);let i=Xi(r,n,e,s);return`
322
322
  ${a}
323
323
  <div class="steps-list">
324
324
  ${o||'<div class="no-steps">No YAML step details available</div>'}
325
325
  </div>
326
- ${i}`}function Xi(e,t){let r=e.flaky?"flaky":e.status,n=nt(r),s;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length-1,c=e.attempts.map((h,g)=>{let u=g===i,p=h.status==="passed"?"passed":"failed",w=`Attempt ${h.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${p}"
326
+ ${i}`}function qi(e,t){let r=e.flaky?"flaky":e.status,n=ot(r),s;if(e.attempts&&e.attempts.length>1){let o=`tabs-${t}`,i=e.attempts.length-1,c=e.attempts.map((h,g)=>{let u=g===i,p=h.status==="passed"?"passed":"failed",w=`Attempt ${h.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${p}"
327
327
  onclick="switchAttemptTab('${o}', ${g})"
328
- data-tab-index="${g}">${nt(p)} ${w} <span class="attempt-tab-badge badge-${p}">${h.status}</span></button>`}).join(""),l=e.attempts.map((h,g)=>{let u=g===i,p=es(h.steps,h.error,h.videoPath,h.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${g}">
328
+ data-tab-index="${g}">${ot(p)} ${w} <span class="attempt-tab-badge badge-${p}">${h.status}</span></button>`}).join(""),l=e.attempts.map((h,g)=>{let u=g===i,p=ts(h.steps,h.error,h.videoPath,h.tracePath,`${t}-attempt-${g}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${g}">
329
329
  <div class="attempt-meta">
330
- ${nt(h.status==="passed"?"passed":"failed")}
331
- <span class="attempt-meta-text">Attempt ${h.attemptNumber} &mdash; ${h.status} in ${st(h.duration)}</span>
330
+ ${ot(h.status==="passed"?"passed":"failed")}
331
+ <span class="attempt-meta-text">Attempt ${h.attemptNumber} &mdash; ${h.status} in ${it(h.duration)}</span>
332
332
  </div>
333
333
  ${p}
334
- </div>`}).join(""),d=e.flaky?`Flaky &mdash; ${Wi(e.retries)}`:`Retried &mdash; ${Ki(e.retries,e.status)}`;s=`
334
+ </div>`}).join(""),d=e.flaky?`Flaky &mdash; ${Ki(e.retries)}`:`Retried &mdash; ${Vi(e.retries,e.status)}`;s=`
335
335
  <div class="${e.flaky?"flaky-note":"retried-note"}">${d}</div>
336
336
  <div class="attempt-tabs" id="${o}">
337
337
  <div class="attempt-tab-bar">${c}</div>
338
338
  ${l}
339
- </div>`}else s=es(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
339
+ </div>`}else s=ts(e.steps,e.error,e.videoPath,e.tracePath,String(t));return`
340
340
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
341
341
  <summary class="test-summary test-${r}">
342
342
  ${n}
343
- <span class="test-title">${O(e.title)}</span>
344
- <span class="test-file">${O(e.file)}</span>
345
- ${Vi(r)}
346
- <span class="test-duration">${st(e.duration)}</span>
343
+ <span class="test-title">${L(e.title)}</span>
344
+ <span class="test-file">${L(e.file)}</span>
345
+ ${zi(r)}
346
+ <span class="test-duration">${it(e.duration)}</span>
347
347
  </summary>
348
348
  <div class="test-body">
349
349
  ${s}
350
350
  </div>
351
- </details>`}function ts(e,t){return!e||!t||Qn.isAbsolute(e)?e:Qn.join(t,e)}function qi(e,t){return t?{...e,tracePath:ts(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:ts(r.tracePath,t)}))}:e}function Xt(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,a=e.tests.length,c=e.tests.map(l=>qi(l,e.outputDir)).map((l,d)=>Xi(l,d)).join(`
351
+ </details>`}function rs(e,t){return!e||!t||es.isAbsolute(e)?e:es.join(t,e)}function Zi(e,t){return t?{...e,tracePath:rs(e.tracePath,t),attempts:e.attempts?.map(r=>({...r,tracePath:rs(r.tracePath,t)}))}:e}function qt(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,a=e.tests.length,c=e.tests.map(l=>Zi(l,e.outputDir)).map((l,d)=>qi(l,d)).join(`
352
352
  `);return`<!DOCTYPE html>
353
353
  <html lang="en">
354
354
  <head>
@@ -998,7 +998,7 @@ statements:
998
998
  ${r>0?`<span class="summary-stat retried">${r} retried</span>`:""}
999
999
  ${s>0?`<span class="summary-stat failed">${s} failed</span>`:""}
1000
1000
  ${o>0?`<span class="summary-stat skipped">${o} skipped</span>`:""}
1001
- <span class="summary-stat">${st(e.totalDuration)}</span>
1001
+ <span class="summary-stat">${it(e.totalDuration)}</span>
1002
1002
  ${e.timestamp?`<span class="summary-stat" style="margin-left:auto;color:var(--color-text-secondary)">${new Date(e.timestamp).toLocaleString()}</span>`:""}
1003
1003
  </div>
1004
1004
  </div>${e.cacheSummary?`
@@ -1015,7 +1015,7 @@ statements:
1015
1015
  ${c}
1016
1016
  </div>
1017
1017
  <div class="footer">
1018
- Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${O(e.shiplightVersion)}`:""}
1018
+ Generated by Shiplight Reporter${e.shiplightVersion?` \xB7 shiplightai v${L(e.shiplightVersion)}`:""}
1019
1019
  </div>
1020
1020
  </div>
1021
1021
 
@@ -1142,9 +1142,9 @@ statements:
1142
1142
  });
1143
1143
  </script>
1144
1144
  </body>
1145
- </html>`}var rs=_(()=>{"use strict"});import*as H from"fs";import*as ke from"path";import{execFileSync as Zi}from"child_process";import{createHash as Qi}from"crypto";import Te from"axios";function ns(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 de(...e){try{return Zi("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function ea(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=H.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function ta(){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??"",a=ea().pull_request,i=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??i??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",d=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??a?.title,g=de("log","-1","--pretty=%s"),u=de("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:f,gitRepo:r,commitMessage:g,authorEmail:u,prNumber:d,prTitle:h,prUrl:d&&r?`${t}/${r}/pull/${d}`: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,a=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,c=i?ns(i):void 0,l=de("log","-1","--pretty=%s"),d=de("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:l,authorEmail:d,prNumber:a,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=de("rev-parse","HEAD"),r=de("rev-parse","--abbrev-ref","HEAD"),n=de("log","-1","--pretty=%s"),s=de("log","-1","--pretty=%ae"),o=de("remote","get-url","origin"),a=o?ns(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:n,authorEmail:s,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function ra(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 qt(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function na(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 Y(e,t){return ke.isAbsolute(t)?t:ke.join(e,t)}function ss(e,t){let r={};for(let n of e)r[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return r}function oa(e,t,r,n){let s=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let a=o.attemptNumber-1,i=t[a]??{},c={outcome:qt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ss(o.steps,i),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[a],traceS3Uri:n[a],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),s.push(c)}else s.push({outcome:qt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:ss(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:n[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:qt(e.status),flaky:e.flaky??!1,segments:s}}async function Zt(e,t,r){let n=new Array(e.length),s=0;async function o(){for(;s<e.length;){let a=s++;n[a]=await r(e[a],a)}}return await Promise.all(Array.from({length:Math.min(t,e.length)},o)),n}async function os(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")||r>=t)throw n;let a=Math.min(1e3*2**r,8e3);console.warn(`[reporter] Upload failed (${s}), retrying in ${a}ms (attempt ${r+1}/${t})...`),await new Promise(i=>setTimeout(i,a))}}function tr(e){return Qi("md5").update(e).digest("base64")}function Qt(e){return tr(H.readFileSync(e))}async function er(e,t){let r=H.readFileSync(t),n=ke.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[n]??"application/octet-stream";await os(()=>Te.put(e,r,{headers:{"Content-Type":o,"Content-MD5":tr(r)}}))}async function is(e,t,r,n){let s=rt(n,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${n}`,"Content-Type":"application/json"},a=ta(),i=e.tests.map(p=>{let w={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let y=Y(t,p.videoPath);H.existsSync(y)&&(w.videoMd5=Qt(y))}if(p.tracePath){let y=Y(t,p.tracePath);H.existsSync(y)&&(w.traceMd5=Qt(y))}return w}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let d=(await Te.post(`${s}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${d.testRunId})`);let f=sa(e.tests,d.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Zt(e.tests,8,async(p,w)=>{let y=f[w];if(!y)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let T of p.attempts){let $=T.attemptNumber-1,X=T.attemptNumber===p.attempts.length;for(let R of T.steps)R.screenshot&&m.push({key:`attempt-${$}.${R.stepId}`,filePath:Y(t,R.screenshot)});if(!X){if(T.videoPath){let R=Y(t,T.videoPath);H.existsSync(R)&&m.push({key:`attempt-${$}.__video__`,filePath:R})}if(T.tracePath){let R=Y(t,T.tracePath);H.existsSync(R)&&m.push({key:`attempt-${$}.__trace__`,filePath:R})}}}else for(let T of p.steps)T.screenshot&&m.push({key:T.stepId,filePath:Y(t,T.screenshot)});if(!m.length)return;let b=m.map(T=>T.key),x={};for(let{key:T,filePath:$}of m)H.existsSync($)&&(x[T]=Qt($));try{let T=await Te.post(`${s}/v1/local-runs/${d.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:b,md5s:x},{headers:o});y.uploadUrls.screenshots=T.data.screenshots,y.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${b.length} screenshot URL(s) for "${p.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,T)}}),console.log("[reporter] [3/4] Uploading assets...");let h=(await Zt(e.tests,5,async(p,w)=>{let y=f[w];if(!y){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=y.uploadUrls,b={},x={},T={},$=0,X=[];if(p.attempts&&p.attempts.length>1)for(let P of p.attempts){let j=P.attemptNumber-1,As=P.attemptNumber===p.attempts.length;for(let re of P.steps)re.screenshot&&X.push({key:`attempt-${j}.${re.stepId}`,filePath:Y(t,re.screenshot),attemptIdx:j,assetType:"screenshot",originalStepId:re.stepId});if(!As){if(P.videoPath){let re=Y(t,P.videoPath);H.existsSync(re)&&X.push({key:`attempt-${j}.__video__`,filePath:re,attemptIdx:j,assetType:"video"})}if(P.tracePath){let re=Y(t,P.tracePath);H.existsSync(re)&&X.push({key:`attempt-${j}.__trace__`,filePath:re,attemptIdx:j,assetType:"trace"})}}}else for(let P of p.steps)P.screenshot&&X.push({key:P.stepId,filePath:Y(t,P.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:P.stepId});await Zt(X,10,async P=>{if(m.screenshots?.[P.key]&&H.existsSync(P.filePath))try{await er(m.screenshots[P.key],P.filePath);let j=y.screenshotS3Uris[P.key];switch(P.assetType){case"screenshot":b[P.attemptIdx]||(b[P.attemptIdx]={}),b[P.attemptIdx][P.originalStepId]=j;break;case"video":x[P.attemptIdx]=j;break;case"trace":T[P.attemptIdx]=j;break}$++}catch(j){console.warn(`[reporter] Asset upload failed for ${P.key}:`,j)}}),$>0&&console.log(`[reporter] [3/4] Uploaded ${$} asset(s) for "${p.title}"`);let R;if(p.videoPath&&m.video){let P=Y(t,p.videoPath);if(H.existsSync(P)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await er(m.video,P),R=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(j){console.warn("[reporter] Video upload failed:",j)}}}let ye;if(p.tracePath&&m.trace){let P=Y(t,p.tracePath);if(H.existsSync(P)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await er(m.trace,P),ye=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(j){console.warn("[reporter] Trace upload failed:",j)}}}if(p.attempts&&p.attempts.length>1){let P=p.attempts.length-1;R&&(x[P]=R),ye&&(T[P]=ye)}else R&&(x[0]=R),ye&&(T[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let ks=oa(p,b,x,T),rr=Buffer.from(JSON.stringify(ks)),nr=tr(rr),sr=await Te.post(`${s}/v1/local-runs/${d.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:nr},{headers:o}),Ps=sr.data.reportUrl,Es=sr.data.reportS3Uri;return await os(()=>Te.put(Ps,rr,{headers:{"Content-Type":"application/json","Content-MD5":nr}})),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:y.testCaseResultId,result:ra(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:Es,videoS3Uri:R,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}})).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=na(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let u=await Te.put(`${s}/v1/local-runs/${d.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:o});console.log(`
1146
- Shiplight cloud report: ${ia(u.data.reportUrl,s)}`)}function ia(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${aa(t)}${r}`}function aa(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 as=_(()=>{"use strict";Ut()});var fs={};ce(fs,{buildGitHubSummary:()=>us,isReportToCloudEnabled:()=>ps,runReport:()=>ca});import*as I from"fs";import*as A from"path";async function ca(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 pa(e,t)}function la(){try{return I.realpathSync("shiplight-report/latest")}catch{return"shiplight-report"}}async function pa(e,t){let r=e.find(i=>!i.startsWith("--"))||la(),n=A.isAbsolute(r)?r:A.join(process.cwd(),r),s=A.join(n,"report-data.json");I.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(I.readFileSync(s,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${s}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=A.join(n,"index.html");if(I.writeFileSync(a,Xt({...o,outputDir:n}),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await ds(o,n),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function da(e,t,r){let n=A.join(process.cwd(),"shiplight-report"),s=e.findIndex(g=>g==="-o"||g==="--output");if(s!==-1&&e[s+1]){let g=e[s+1];n=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let g=0;g<e.length;g++){let u=e[g];if(o.has(u)){g++;continue}if(a.has(u))continue;let p=A.isAbsolute(u)?u:A.join(process.cwd(),u);i.push(p)}i.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,d=0;I.mkdirSync(A.join(n,"screenshots"),{recursive:!0});for(let g=0;g<i.length;g++){let u=i[g],p=`shard-${g}`,w=A.join(u,"report-data.json");if(!I.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let y;try{y=JSON.parse(I.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${p}: ${y.tests.length} tests from ${u}`),l+=y.totalDuration||0;let m=A.join(u,"screenshots");I.existsSync(m)&&ls(m,A.join(n,"screenshots",p));let b=A.join(n,p);for(let x of y.tests){let T=[x,...x.attempts||[]];for(let $ of T)ua($.steps,p),$.videoPath&&cs(u,$.videoPath,b)&&($.videoPath=`${p}/${$.videoPath}`),$.tracePath&&cs(u,$.tracePath,b)&&($.tracePath=`${p}/${$.tracePath}`);c.push(x)}d++}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:Pe};I.writeFileSync(A.join(n,"report-data.json"),JSON.stringify(f,null,2),"utf-8");let h=A.join(n,"index.html");if(I.writeFileSync(h,Xt({...f,outputDir:n}),"utf-8"),console.log(`
1147
- Merged ${c.length} tests from ${d} shards into: ${h}`),await ds(f,n),r&&fa(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function cs(e,t,r){let n=A.resolve(e,t);return n.startsWith(A.resolve(e)+A.sep)?I.existsSync(n)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(n,A.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function ls(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let n=A.join(e,r.name),s=A.join(t,r.name);r.isDirectory()?ls(n,s):I.copyFileSync(n,s)}}function ua(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function ps(){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 ds(e,t){if(!ps())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 is(e,t,s,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ot(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:r}}function us(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"),a=t.length,i=`## Test Results
1145
+ </html>`}var ns=_(()=>{"use strict"});import*as H from"fs";import*as Ee from"path";import{execFileSync as Qi}from"child_process";import{createHash as ea}from"crypto";import ke from"axios";function ss(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 de(...e){try{return Qi("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function ta(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=H.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function ra(){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??"",a=ta().pull_request,i=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??i??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",d=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??a?.title,g=de("log","-1","--pretty=%s"),u=de("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:f,gitRepo:r,commitMessage:g,authorEmail:u,prNumber:d,prTitle:h,prUrl:d&&r?`${t}/${r}/pull/${d}`: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,a=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,c=i?ss(i):void 0,l=de("log","-1","--pretty=%s"),d=de("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:l,authorEmail:d,prNumber:a,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=de("rev-parse","HEAD"),r=de("rev-parse","--abbrev-ref","HEAD"),n=de("log","-1","--pretty=%s"),s=de("log","-1","--pretty=%ae"),o=de("remote","get-url","origin"),a=o?ss(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:n,authorEmail:s,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function na(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 Zt(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function sa(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 oa(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 X(e,t){return Ee.isAbsolute(t)?t:Ee.join(e,t)}function os(e,t){let r={};for(let n of e)r[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return r}function ia(e,t,r,n){let s=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let a=o.attemptNumber-1,i=t[a]??{},c={outcome:Zt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:os(o.steps,i),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[a],traceS3Uri:n[a],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),s.push(c)}else s.push({outcome:Zt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:os(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:n[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:Zt(e.status),flaky:e.flaky??!1,segments:s}}async function Qt(e,t,r){let n=new Array(e.length),s=0;async function o(){for(;s<e.length;){let a=s++;n[a]=await r(e[a],a)}}return await Promise.all(Array.from({length:Math.min(t,e.length)},o)),n}async function is(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")||r>=t)throw n;let a=Math.min(1e3*2**r,8e3);console.warn(`[reporter] Upload failed (${s}), retrying in ${a}ms (attempt ${r+1}/${t})...`),await new Promise(i=>setTimeout(i,a))}}function rr(e){return ea("md5").update(e).digest("base64")}function er(e){return rr(H.readFileSync(e))}async function tr(e,t){let r=H.readFileSync(t),n=Ee.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[n]??"application/octet-stream";await is(()=>ke.put(e,r,{headers:{"Content-Type":o,"Content-MD5":rr(r)}}))}async function as(e,t,r,n){let s=st(n,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${n}`,"Content-Type":"application/json"},a=ra(),i=e.tests.map(p=>{let w={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let y=X(t,p.videoPath);H.existsSync(y)&&(w.videoMd5=er(y))}if(p.tracePath){let y=X(t,p.tracePath);H.existsSync(y)&&(w.traceMd5=er(y))}return w}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let d=(await ke.post(`${s}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${d.testRunId})`);let f=oa(e.tests,d.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Qt(e.tests,8,async(p,w)=>{let y=f[w];if(!y)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let k of p.attempts){let A=k.attemptNumber-1,G=k.attemptNumber===p.attempts.length;for(let M of k.steps)M.screenshot&&m.push({key:`attempt-${A}.${M.stepId}`,filePath:X(t,M.screenshot)});if(!G){if(k.videoPath){let M=X(t,k.videoPath);H.existsSync(M)&&m.push({key:`attempt-${A}.__video__`,filePath:M})}if(k.tracePath){let M=X(t,k.tracePath);H.existsSync(M)&&m.push({key:`attempt-${A}.__trace__`,filePath:M})}}}else for(let k of p.steps)k.screenshot&&m.push({key:k.stepId,filePath:X(t,k.screenshot)});if(!m.length)return;let b=m.map(k=>k.key),x={};for(let{key:k,filePath:A}of m)H.existsSync(A)&&(x[k]=er(A));try{let k=await ke.post(`${s}/v1/local-runs/${d.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:b,md5s:x},{headers:o});y.uploadUrls.screenshots=k.data.screenshots,y.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${b.length} screenshot URL(s) for "${p.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,k)}}),console.log("[reporter] [3/4] Uploading assets...");let h=(await Qt(e.tests,5,async(p,w)=>{let y=f[w];if(!y){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=y.uploadUrls,b={},x={},k={},A=0,G=[];if(p.attempts&&p.attempts.length>1)for(let P of p.attempts){let D=P.attemptNumber-1,$s=P.attemptNumber===p.attempts.length;for(let re of P.steps)re.screenshot&&G.push({key:`attempt-${D}.${re.stepId}`,filePath:X(t,re.screenshot),attemptIdx:D,assetType:"screenshot",originalStepId:re.stepId});if(!$s){if(P.videoPath){let re=X(t,P.videoPath);H.existsSync(re)&&G.push({key:`attempt-${D}.__video__`,filePath:re,attemptIdx:D,assetType:"video"})}if(P.tracePath){let re=X(t,P.tracePath);H.existsSync(re)&&G.push({key:`attempt-${D}.__trace__`,filePath:re,attemptIdx:D,assetType:"trace"})}}}else for(let P of p.steps)P.screenshot&&G.push({key:P.stepId,filePath:X(t,P.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:P.stepId});await Qt(G,10,async P=>{if(m.screenshots?.[P.key]&&H.existsSync(P.filePath))try{await tr(m.screenshots[P.key],P.filePath);let D=y.screenshotS3Uris[P.key];switch(P.assetType){case"screenshot":b[P.attemptIdx]||(b[P.attemptIdx]={}),b[P.attemptIdx][P.originalStepId]=D;break;case"video":x[P.attemptIdx]=D;break;case"trace":k[P.attemptIdx]=D;break}A++}catch(D){console.warn(`[reporter] Asset upload failed for ${P.key}:`,D)}}),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} asset(s) for "${p.title}"`);let M;if(p.videoPath&&m.video){let P=X(t,p.videoPath);if(H.existsSync(P)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await tr(m.video,P),M=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(D){console.warn("[reporter] Video upload failed:",D)}}}let ye;if(p.tracePath&&m.trace){let P=X(t,p.tracePath);if(H.existsSync(P)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await tr(m.trace,P),ye=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(D){console.warn("[reporter] Trace upload failed:",D)}}}if(p.attempts&&p.attempts.length>1){let P=p.attempts.length-1;M&&(x[P]=M),ye&&(k[P]=ye)}else M&&(x[0]=M),ye&&(k[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let Es=ia(p,b,x,k),nr=Buffer.from(JSON.stringify(Es)),sr=rr(nr),or=await ke.post(`${s}/v1/local-runs/${d.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:sr},{headers:o}),Ps=or.data.reportUrl,As=or.data.reportS3Uri;return await is(()=>ke.put(Ps,nr,{headers:{"Content-Type":"application/json","Content-MD5":sr}})),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:y.testCaseResultId,result:na(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:As,videoS3Uri:M,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}})).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=sa(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let u=await ke.put(`${s}/v1/local-runs/${d.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:o});console.log(`
1146
+ Shiplight cloud report: ${aa(u.data.reportUrl,s)}`)}function aa(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${ca(t)}${r}`}function ca(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 cs=_(()=>{"use strict";Ht()});var hs={};ce(hs,{buildGitHubSummary:()=>fs,isReportToCloudEnabled:()=>ds,runReport:()=>la});import*as I 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 ua(e,t,n):await da(e,t)}function pa(){try{return I.realpathSync("shiplight-report/latest")}catch{return"shiplight-report"}}async function da(e,t){let r=e.find(i=>!i.startsWith("--"))||pa(),n=$.isAbsolute(r)?r:$.join(process.cwd(),r),s=$.join(n,"report-data.json");I.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(I.readFileSync(s,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${s}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=$.join(n,"index.html");if(I.writeFileSync(a,qt({...o,outputDir:n}),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await us(o,n),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function ua(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"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let g=0;g<e.length;g++){let u=e[g];if(o.has(u)){g++;continue}if(a.has(u))continue;let p=$.isAbsolute(u)?u:$.join(process.cwd(),u);i.push(p)}i.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,d=0;I.mkdirSync($.join(n,"screenshots"),{recursive:!0});for(let g=0;g<i.length;g++){let u=i[g],p=`shard-${g}`,w=$.join(u,"report-data.json");if(!I.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let y;try{y=JSON.parse(I.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${p}: ${y.tests.length} tests from ${u}`),l+=y.totalDuration||0;let m=$.join(u,"screenshots");I.existsSync(m)&&ps(m,$.join(n,"screenshots",p));let b=$.join(n,p);for(let x of y.tests){let k=[x,...x.attempts||[]];for(let A of k)fa(A.steps,p),A.videoPath&&ls(u,A.videoPath,b)&&(A.videoPath=`${p}/${A.videoPath}`),A.tracePath&&ls(u,A.tracePath,b)&&(A.tracePath=`${p}/${A.tracePath}`);c.push(x)}d++}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:Pe};I.writeFileSync($.join(n,"report-data.json"),JSON.stringify(f,null,2),"utf-8");let h=$.join(n,"index.html");if(I.writeFileSync(h,qt({...f,outputDir:n}),"utf-8"),console.log(`
1147
+ Merged ${c.length} tests from ${d} shards into: ${h}`),await us(f,n),r&&ha(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function ls(e,t,r){let n=$.resolve(e,t);return n.startsWith($.resolve(e)+$.sep)?I.existsSync(n)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(n,$.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function ps(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let n=$.join(e,r.name),s=$.join(t,r.name);r.isDirectory()?ps(n,s):I.copyFileSync(n,s)}}function fa(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function ds(){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 us(e,t){if(!ds())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 as(e,t,s,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function at(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 fs(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"),a=t.length,i=`## Test Results
1148
1148
 
1149
1149
  `;if(o.length===0&&r.length===0?i+=`\u2705 All ${a} tests passed
1150
1150
 
@@ -1152,7 +1152,7 @@ Merged ${c.length} tests from ${d} shards into: ${h}`),await ds(f,n),r&&fa(c),t)
1152
1152
 
1153
1153
  `:i+=`\u274C ${o.length} failed, \u26A0\uFE0F ${r.length} flaky, \u2705 ${s.length} passed / ${a} total
1154
1154
 
1155
- `,o.length>0){let c=[...new Set(o.map(l=>ot(l).yamlPath))];i+=`### Failed
1155
+ `,o.length>0){let c=[...new Set(o.map(l=>at(l).yamlPath))];i+=`### Failed
1156
1156
 
1157
1157
  `;for(let l of c)i+=`- \`npx shiplight test ${l}\`
1158
1158
  `;i+=`
@@ -1160,22 +1160,22 @@ Merged ${c.length} tests from ${d} shards into: ${h}`),await ds(f,n),r&&fa(c),t)
1160
1160
 
1161
1161
  `,i+="```sh\n",i+=`npx shiplight test ${c.map(l=>`"${l}"`).join(` \\
1162
1162
  `)}
1163
- `,i+="```\n\n"}if(r.length>0){let c=[...new Set(r.map(l=>ot(l).yamlPath))];i+=`### Flaky (${c.length})
1163
+ `,i+="```\n\n"}if(r.length>0){let c=[...new Set(r.map(l=>at(l).yamlPath))];i+=`### Flaky (${c.length})
1164
1164
 
1165
1165
  `;for(let l of c)i+=`- \`${l}\`
1166
1166
  `;i+=`
1167
- `}if(n.length>0){let c=[...new Set(n.map(l=>ot(l).yamlPath))];i+=`### Retried (${c.length})
1167
+ `}if(n.length>0){let c=[...new Set(n.map(l=>at(l).yamlPath))];i+=`### Retried (${c.length})
1168
1168
 
1169
1169
  `;for(let l of c)i+=`- \`${l}\`
1170
1170
  `;i+=`
1171
- `}if(s.length>0){let c=[...new Set(s.map(l=>ot(l).yamlPath))];i+=`<details><summary>Passed (${c.length})</summary>
1171
+ `}if(s.length>0){let c=[...new Set(s.map(l=>at(l).yamlPath))];i+=`<details><summary>Passed (${c.length})</summary>
1172
1172
 
1173
1173
  `;for(let l of c)i+=`- ${l}
1174
1174
  `;i+=`
1175
1175
  </details>
1176
- `}return i}function fa(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,us(e)),console.log("GitHub step summary written.")}var hs=_(()=>{"use strict";rs();as();je()});var gs,ms=_(()=>{"use strict";gs="0.1.75"});var ys={};ce(ys,{runTranspile:()=>ga});import*as it from"path";import{glob as ha}from"glob";async function ga(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 ha(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,a=0;for(let i of n.sort()){let c=it.resolve(r,i),l=xn(c,{version:gs,basePath:r});if(!l.valid){s++,console.log(`
1177
- \u2717 ${i}`);for(let f of l.errors)console.log(` ERROR: ${f}`);continue}a++;let d=it.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${i} \u2192 ${d}`);for(let f of l.warnings)console.log(` WARNING: ${f}`)}else console.log(`\u2713 ${i} \u2192 ${d}`)}console.log(`
1178
- ${n.length} file(s): ${a} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var ws=_(()=>{"use strict";Rt();ms()});var Ss={};ce(Ss,{runInspect:()=>ma});import*as at from"fs";import*as bs from"path";async function ma(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(a=>!a.startsWith("--"));n||(console.error("Error: no file specified"),process.exit(1));let s=bs.resolve(process.cwd(),n);at.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=at.readFileSync(s,"utf-8");try{let a=Ie(o),i=K(o);if(r)ya(i,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${n}: ${a.message}`),process.exit(1)}}function ya(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=vs(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function vs(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=vs(r.statements??[]);t.drafts+=n.drafts,t.actions+=n.actions,t.steps+=n.steps}return t}var _s=_(()=>{"use strict";ue()});var xs=Is((Sd,wa)=>{wa.exports={name:"shiplightai",version:"0.1.75",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",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});je();import ba from"dotenv";ba.config();or();ir();function Ts(){console.log(`
1176
+ `}return i}function ha(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,fs(e)),console.log("GitHub step summary written.")}var gs=_(()=>{"use strict";ns();cs();Fe()});var ms,ys=_(()=>{"use strict";ms="0.1.76"});var ws={};ce(ws,{runTranspile:()=>ma});import*as ct from"path";import{glob as ga}from"glob";async function ma(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 ga(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,a=0;for(let i of n.sort()){let c=ct.resolve(r,i),l=Tn(c,{version:ms,basePath:r});if(!l.valid){s++,console.log(`
1177
+ \u2717 ${i}`);for(let f of l.errors)console.log(` ERROR: ${f}`);continue}a++;let d=ct.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${i} \u2192 ${d}`);for(let f of l.warnings)console.log(` WARNING: ${f}`)}else console.log(`\u2713 ${i} \u2192 ${d}`)}console.log(`
1178
+ ${n.length} file(s): ${a} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var bs=_(()=>{"use strict";Nt();ys()});var _s={};ce(_s,{runInspect:()=>ya});import*as lt from"fs";import*as vs from"path";async function ya(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(a=>!a.startsWith("--"));n||(console.error("Error: no file specified"),process.exit(1));let s=vs.resolve(process.cwd(),n);lt.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=lt.readFileSync(s,"utf-8");try{let a=Oe(o),i=V(o);if(r)wa(i,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${n}: ${a.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=Ss(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Ss(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=Ss(r.statements??[]);t.drafts+=n.drafts,t.actions+=n.actions,t.steps+=n.steps}return t}var xs=_(()=>{"use strict";ue()});var Ts=Ms((_d,ba)=>{ba.exports={name:"shiplightai",version:"0.1.76",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",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Fe();import va from"dotenv";va.config();ir();ar();function ks(){console.log(`
1179
1179
  Usage: shiplight <command> [options]
1180
1180
 
1181
1181
  Commands:
@@ -1197,5 +1197,5 @@ Examples:
1197
1197
  shiplight transpile
1198
1198
  shiplight transpile "tests/**/*.test.yaml"
1199
1199
  shiplight debug tests/login.test.yaml
1200
- `)}var ct=process.argv[2];switch(ct){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(hr(),fr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Bn(),Hn));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Zn(),qn));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(hs(),fs));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(ws(),ys));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(_s(),Ss));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=xs().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":Ts();break;default:ct&&console.error(`Unknown command: ${ct}
1201
- `),Ts(),process.exit(ct?1:0)}
1200
+ `)}var pt=process.argv[2];switch(pt){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>(gr(),hr));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Gn(),Bn));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Qn(),Zn));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(gs(),hs));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(bs(),ws));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(xs(),_s));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=Ts().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":ks();break;default:pt&&console.error(`Unknown command: ${pt}
1201
+ `),ks(),process.exit(pt?1:0)}