shiplightai 0.1.76 → 0.1.78
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/cjs/debugger-pw.cjs +80 -80
- package/dist/cjs/fixture.cjs +88 -88
- package/dist/cjs/index.cjs +75 -75
- package/dist/cjs/reporter.cjs +2 -2
- package/dist/cli.js +6 -6
- package/dist/debugger-pw.js +80 -80
- package/dist/fixture.d.ts +12 -7
- package/dist/fixture.js +88 -88
- package/dist/index.js +75 -75
- package/dist/reporter.js +2 -2
- package/package.json +4 -4
package/dist/cli.js
CHANGED
|
@@ -17,7 +17,7 @@ Install it as a project dependency instead:
|
|
|
17
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 dt,Pe,Ls,Rs,Cs,Fe=_(()=>{"use strict";dt="0.1.
|
|
20
|
+
`)}var dt,Pe,Ls,Rs,Cs,Fe=_(()=>{"use strict";dt="0.1.78",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
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.
|
|
@@ -212,8 +212,8 @@ 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 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(`
|
|
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"});import{v4 as ja}from"uuid";var Mr=_(()=>{"use strict"});import{z as v}from"zod";var Or,vt,Lr,we,Rr,Cr,Nr,W,Dr,Be,St,_t=_(()=>{"use strict";Or=v.enum(["JS_CODE","AI_MODE"]),vt=v.object({type:Or,expression:v.string()}),Lr=v.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=v.object({uid:v.string(),type:Lr,comment:v.string().optional()}),Rr=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(),Cr=we.extend({type:v.literal("DRAFT"),description:v.string()}),Nr=we.extend({type:v.literal("ACTION"),description:v.string(),action_entity:Rr.optional(),locator:v.string().optional(),use_pure_vision:v.boolean().optional()}),W=v.lazy(()=>v.union([Cr,Nr,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()})])),Dr=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()}),Be=v.object({tests:v.array(Dr).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:Be.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as Wa,parse as Hr,parseAllDocuments as Ka,parseDocument as fo,Document as Br,isMap as ve,isSeq as K}from"yaml";import{v4 as Q}from"uuid";function Ge(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 Wr(e,t);let r=Ge(e,t),n=new Br(r),s=n.contents?.get("tags",!0);return K(s)&&(s.flow=!0),e.comment&&(n.commentBefore=e.comment),jr(n,e.statements??[]),e.teardown&&jr(n,e.teardown,"teardown"),n.toString(Gr)}function jr(e,t,r="statements"){let n=e.contents;if(!n||!ve(n))return;let s=n.get(r,!0);K(s)&&Ie(s,t)}function Ie(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)&&Ie(a,n.statements)}else if(n.type==="IF_ELSE"){let a=o.get("THEN",!0);K(a)&&Ie(a,n.then);let i=o.get("ELSE",!0);K(i)&&n.else&&Ie(i,n.else)}else if(n.type==="WHILE_LOOP"){let a=o.get("DO",!0);K(a)&&Ie(a,n.body)}}}}function Wr(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 Br(n),a=o.contents?.get("tags",!0);return K(a)&&(a.flow=!0),o.toString(Gr)}function J(e){switch(e.type){case"DRAFT":return ho(e);case"ACTION":return go(e);case"STEP":return mo(e);case"IF_ELSE":return yo(e);case"WHILE_LOOP":return wo(e)}}function ho(e){return{intent:e.description}}function go(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 mo(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 yo(e){let t={IF:Kr(e.condition),THEN:e.then.map(J)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(J)),t}function wo(e){let t={WHILE:Kr(e.condition),DO:e.body.map(J)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Kr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Me(e){try{let t=Hr(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>Fr)throw new Error(`YAML input too large (${e.length} bytes, max ${Fr})`);let t=xt(Hr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return bo(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 We(e,s),s}function bo(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=Be.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(vo)}function vo(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 So(t);if("WHILE"in t)return _o(t);if("STEP"in t)return xo(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 Ur({...n,action:"function",functionName:r})}if("action"in t)return Ur(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 Vr(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 So(e){let t=Vr(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 _o(e){let t=Vr(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 xo(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 Ur(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))To.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 We(e,t){let r;try{r=fo(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 Gr,Fr,To,Ke=_(()=>{"use strict";_t();Gr={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Fr=1024*1024;To=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as Xa,stringify as qa}from"yaml";var zr=_(()=>{"use strict";Ke()});var kt,Ve,ze=_(()=>{"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},Ve=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 Zr(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??"";qr.has(s)||t++}return{action:t,draft:r}}function Eo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Xr(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<=Jr&&s<=Yr?null:`${r} non-blank line(s), ${s} await(s) \u2014 limits are ${Jr} lines and ${Yr} 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??ko,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=[...Ve(o.statements??[]),...o.teardown?Ve(o.teardown):[]],{action:i,draft:c}=Zr(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=Eo(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=Xr(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=Xr(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 ko,Jr,Yr,qr,Qr=_(()=>{"use strict";Ke();ze();ko=.5,Jr=5,Yr=3,qr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Je=_(()=>{"use strict"});var en=_(()=>{"use strict";Je()});var tn,$o,rn,nn,Oe,Io,Mo,sn=_(()=>{"use strict";tn=112,$o=1080-tn,rn={"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"}},nn={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"]},Oe=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),nn[e].map(n=>rn[n]).filter(n=>n.defaultBrowserType&&r.includes(n.defaultBrowserType))},Io={desktop:{label:"Desktop",type:"desktop",devices:Oe("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Oe("mobile")}},Mo={desktop:{label:"Desktop",type:"desktop",devices:Oe("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Oe("mobile",!0)}}});var on=_(()=>{"use strict"});function Pt(){return{version:"1.0",entries:{}}}var an=_(()=>{"use strict";ze()});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";Je()});var ue=_(()=>{"use strict";Ar();$r();Ir();Mr();zr();Qr();Ke();_t();en();sn();on();an();ze();cn();ln();pn();dn();un();Je()});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
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
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(`
|
|
@@ -221,7 +221,7 @@ test('__shiplight_debug__', async ({ page, agent }) => {
|
|
|
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
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
|
|
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 Ge({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=Me(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;We(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
|
|
@@ -1173,9 +1173,9 @@ Merged ${c.length} tests from ${d} shards into: ${h}`),await us(f,n),r&&ha(c),t)
|
|
|
1173
1173
|
`;for(let l of c)i+=`- ${l}
|
|
1174
1174
|
`;i+=`
|
|
1175
1175
|
</details>
|
|
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.
|
|
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.78"});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
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=
|
|
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=Me(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.78",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:
|