shiplightai 0.1.48 → 0.1.49

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.
@@ -4364,7 +4364,7 @@ ${p.join(`
4364
4364
  `)}function Fc(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...Wo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let a=new Set,o={imports:a,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=Yn(t?.tags);i.push(`test.describe.serial('${s}${Se(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...$i("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...$i("beforeEach",e.beforeEach,o,1)),i.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],h=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 p of d.parameters){let g=Ho(d.testFlow,p.values);i.push(...Ci(g,`${Se(d.name)} [${Se(p.name)}]`,o,1,h)),i.push("")}else i.push(...Ci(d.testFlow,Se(d.name),o,1,h)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&i.push("")}return e.afterEach&&e.afterEach.length>0&&(i.push(...$i("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...$i("afterAll",e.afterAll,o,1)),i.push("});"),Uo(i,a),i.join(`
4365
4365
  `)}function Yn(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var Hc=["testContext","request"];function Ii(e){let t=new Set;function i(n){for(let a of n)switch(a.type){case me.ACTION:{let r=a.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&Hc.includes(s)&&t.add(s);break}case me.STEP:i(a.statements);break;case me.IF_ELSE:{let o=a;i(o.then),o.else&&i(o.else);break}case me.WHILE_LOOP:i(a.body);break}}return i(e),t}function Wc(e){let t=Ii(e.statements??[]);if(e.teardown)for(let i of Ii(e.teardown))t.add(i);return t}function qn(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ci(e,t,i,n=0,a){let o=" ".repeat(n),r=[],s=Wc(e),l=qn(s),c=a?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),a?.skip===!0?r.push(`${o} test.skip();`):typeof a?.skip=="string"&&r.push(`${o} test.skip(true, '${Se(a.skip)}');`),a?.fail===!0?r.push(`${o} test.fail();`):typeof a?.fail=="string"&&r.push(`${o} test.fail(true, '${Se(a.fail)}');`),a?.slow&&r.push(`${o} test.slow();`),a?.timeout&&r.push(`${o} test.setTimeout(${a.timeout});`);let d=e.teardown&&e.teardown.length>0,h=n+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=ge(e.statements,h+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=ge(e.teardown,h+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=ge(e.statements,h,i);r.push(...p)}return r.push(`${o}});`),r}function Do(e,t,i){let n=[],a=Fo(t),o=Ii(a),r=qn(o);return n.push(`test.${e}(async (${r}) => {`),n.push(...ge(a,1,i,e)),n.push("});"),n}function $i(e,t,i,n){let a=" ".repeat(n),o=[],r=Fo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${a}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${a} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...ge(r,n+1,l,e)),o.push(`${a} await page.close();`),o.push(`${a}});`)}else{let l=Ii(r),c=qn(l);o.push(`${a}test.${e}(async (${c}) => {`),o.push(...ge(r,n+1,i,e)),o.push(`${a}});`)}return o}function Fo(e){let i=(0,Ro.stringify)({goal:"_hook",statements:e});return pe(i).statements??[]}function Ho(e,t){let i=jn(e);for(let[n,a]of Object.entries(t))i=i.split(`<<${n}>>`).join(String(a));return pe(i)}function Wo(){return["import { test, expect } from 'shiplightai/fixture';"]}function Uo(e,t){if(t.size>0){let i=0;for(let a=0;a<e.length;a++)e[a].startsWith("import ")&&(i=a+1);let n=Array.from(t);e.splice(i,0,...n)}}var No=5;function jo(e,t){let i={expandingPaths:new Set([(0,yt.resolve)(t)]),depth:0,referencedPaths:new Set},n={...e};Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,i)),Array.isArray(n.teardown)&&(n.teardown=Re(n.teardown,t,i));for(let a of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[a])&&(n[a]=Re(n[a],t,i));return{doc:n,referencedTemplatePaths:Array.from(i.referencedPaths)}}function Re(e,t,i){let n=[];for(let a of e)if(Uc(a)){let o=Bc(a,t,i);n.push(...o)}else n.push(Gc(a,t,i));return n}function Uc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Bc(e,t,i){if(i.depth>=No)throw new Error(`Template expansion exceeded maximum depth of ${No}. Check for deeply nested or circular template references.`);let n=(0,yt.resolve)((0,yt.dirname)(t),e.template);if(i.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);i.referencedPaths.add(n);let a;try{a=(0,Go.readFileSync)(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=(0,wt.parse)(a);if(!o||typeof o!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let r=o.params||[],s=e.params||{};for(let d of r)if(!(d in s))throw new Error(`Template ${e.template} requires param "${d}" but it was not provided. Required params: [${r.join(", ")}]`);let l=o.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(s).length>0){let h=(0,wt.stringify)(l);for(let[p,g]of Object.entries(s))h=h.split(`<<${p}>>`).join(String(g));l=(0,wt.parse)(h)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return Re(l,n,c)}function Gc(e,t,i){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,i)),Array.isArray(n.THEN)&&(n.THEN=Re(n.THEN,t,i)),Array.isArray(n.ELSE)&&(n.ELSE=Re(n.ELSE,t,i)),Array.isArray(n.DO)&&(n.DO=Re(n.DO,t,i)),n}var Zn=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Qn(e,t){let i=(0,bt.parse)(e),n=i?.name,a=i?.tags,o=i?.use;if(i&&(i.name!==void 0||i.tags!==void 0||i.use!==void 0)&&(delete i.name,delete i.tags,delete i.use),i?.suite){if(i.goal||i.statements)throw new Zn('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Kc(i,n,a,o,t)}return jc(i,n,a,o,t)}function jc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=Ko(e?.parameters),l=e?.timeout,c=e?.skip,d=e?.fail,h=e?.only,p=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 Zn(`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 g=[];if(a&&e&&typeof e=="object"){let b=jo(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=(0,bt.stringify)(e),v=pe(f);return a&&(Ye(v.statements??[],a,"main"),v.teardown&&Ye(v.teardown,a,"teardown")),{testFlow:v,name:t,tags:i,use:n,beforeEach:o,afterEach:r,parameters:s,timeout:l,skip:c,fail:d,only:h,slow:p,referencedTemplatePaths:g}}function Kc(e,t,i,n,a){let o=e.suite;if(!Array.isArray(o.tests)||o.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let r=o.beforeAll,s=o.afterAll,l=o.beforeEach,c=o.afterEach,d=[],h=o.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let v={goal:f.name,statements:f.statements};f.teardown&&(v.teardown=f.teardown);let b=[],m=v;if(a&&typeof v=="object"){let A=jo(v,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=(0,bt.stringify)(m),S=pe(y),E=Ko(f.parameters);return{testFlow:S,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:E,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),p=o.base_url,g=p?{...n,baseURL:p}:n;return{suite:{beforeAll:r,afterAll:s,beforeEach:l,afterEach:c,tests:h},name:t,tags:i,use:g,referencedTemplatePaths:d}}function Ko(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Ye(e,t,i){for(let n=0;n<e.length;n++){let a=e[n],o=`${i}.${n}`,r=a.description||"";if(a.uid=zc(t,o,r),a.type===me.STEP)Ye(a.statements,t,o);else if(a.type===me.IF_ELSE){let s=a;Ye(s.then,t,`${o}.then`),s.else&&Ye(s.else,t,`${o}.else`)}else a.type===me.WHILE_LOOP&&Ye(a.body,t,`${o}.body`)}}function zc(e,t,i){let n=(0,Bo.createHash)("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function Vo(e,t,i){let n=/\btemplate:\s/.test(e),a=/^suite:/m.test(e),o=n||a?null:zn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Qn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},h=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=Fc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Rc(c.testFlow,{...d,testName:c.name,tags:c.tags,use:h,beforeEach:c.beforeEach,afterEach:c.afterEach,parameters:c.parameters,timeout:c.timeout,skip:c.skip,fail:c.fail,only:c.only,slow:c.slow});let p=r.split(`
4366
4366
  `).filter(g=>!g.startsWith("import ")).join(`
4367
- `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),(0,xt.mkdirSync)((0,zo.dirname)(s),{recursive:!0}),(0,xt.writeFileSync)(s,r)}catch(c){let d=c instanceof Zn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var ea="0.1.48";function Xo(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var Vc=`// @generated by shiplightai v${ea}`;function Xc(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4367
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),(0,xt.mkdirSync)((0,zo.dirname)(s),{recursive:!0}),(0,xt.writeFileSync)(s,r)}catch(c){let d=c instanceof Zn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var ea="0.1.49";function Xo(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var Vc=`// @generated by shiplightai v${ea}`;function Xc(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4368
4368
  `,1)[0]!==Vc)return!1;let n=Xo(e);for(let a of t)if(Xo(a)>n)return!1;return!0}function Yc(e){let t=process.argv.slice(2),i=[],n=(0,Je.resolve)(e);for(let a of t){if(a.startsWith("-"))continue;let o=a.endsWith(".yaml.spec.ts")?a.replace(/\.yaml\.spec\.ts$/,".test.yaml"):a;if(!o.endsWith(".test.yaml"))continue;let r=(0,Je.resolve)(e,o);(0,Fe.existsSync)(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function Jo(e){let t=Yc(e.cwd),i=t??(0,Yo.globSync)("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),n=[];for(let a of i){let o=(0,Je.resolve)(e.cwd,a),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=(0,Fe.readFileSync)(o,"utf-8");try{let l=Qn(s,o),c=(0,Je.relative)(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&Xc(r,[o,...l.referencedTemplatePaths]))continue;let p=Vo(s,o,{version:ea,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${a}:`,l),n.push({file:a,error:l})}}if(n.length>0){let a=`[shiplight] Transpilation failed for ${n.length} file(s):
4369
4369
  `+n.map(o=>` - ${o.file}`).join(`
4370
4370
  `);if(t)throw new Error(a);console.warn(a+" (skipped)")}}var re=J(require("fs"),1),vt=J(require("path"),1),Qo=require("glob");function er(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new aa:new na(e)}var Zc=".shiplight/action-cache";function Zo(e){return e.replace(/\//g,"__")+".json"}function Qc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var na=class{constructor(t){this.cwd=t;this.cacheDir=vt.join(t,Zc)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!re.existsSync(this.cacheDir))return i;for(let n of t){let a=vt.join(this.cacheDir,Zo(n));try{if(re.existsSync(a)){let o=re.readFileSync(a,"utf-8");i.set(n,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;re.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[n,a]of t)try{let o=vt.join(this.cacheDir,Zo(n)),r=Vn();if(re.existsSync(o))try{r=JSON.parse(re.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...a.entries}};re.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!re.existsSync(this.cacheDir))return;let t=(0,Qo.globSync)("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,n=0;for(let a of t)try{let o=re.readFileSync(vt.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=Qc(a);i.set(s,r),n+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},aa=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(ia(),ta));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(ia(),ta));return i(t)}loadAll(){}};function ar(e={}){e.dotenv!==!1&&ed(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=er(t).loadAll();return Jo({cwd:t,actionEntityStores:n}),e.apiKey&&(process.env.__SHIPLIGHT_API_KEY=e.apiKey),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never",reportToCloud:e.reportToCloud,apiKey:e.apiKey}]]}}function or(e,...t){return(0,nr.defineConfig)(e,...t)}function ed(e){let t=[],i=Be.resolve(e),n=Be.resolve(process.cwd());for(;;){let a=Be.join(i,".env");if(tr.existsSync(a)&&t.push(a),i===n)break;let o=Be.dirname(i);if(o===i)break;i=o}for(let a of t)ir.default.config({path:a})}F();F();Wt();tt();var Ja=require("zod"),qu=Ja.z.object({instruction:Ja.z.string().describe('The instruction of the operation to perform. Can only include one operation. Do not inlcude element indexes just describe the element, e.g. "select the text "Hello, world!" in "Hello, world!""')});function vl(e){e.register({name:"perform_accurate_operation",description:"Perform an operation that requires accurate interaction like dragging or interacting with a specific area of an element. Only use this action when neccecary.",schema:qu,usesElementIndex:!1,async execute(t,i){let{instruction:n}=t,a={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await et(n,a,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:n,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:n}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Ft(r,a);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}$e();F();var _l=require("ai"),kl=require("html-to-text");async function Zu(e,t,i){let{apiKey:n,domain:a}=e;if(!n||!a)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${a}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}u.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let h=(d.storage||{}).url;if(u.info(`message_url: ${h}`),!h){let f=(d.message||{}).headers||{},v=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!v.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:v,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${a}/messages/${g}`;try{let v=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(v.ok){let b=await v.json(),m=b.Subject||"",y=b.From||"",S=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${y}`),u.info(`to_addr: ${S}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let M=b["body-html"]||b["body-plain"]||"";if(M&&M.includes("<")&&(M=(0,kl.convert)(M)),u.info(`Body: ${M.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!M.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:y,to:S,date:E,body:M,message_id:A});continue}else u.warn(`Messages API returned ${v.status}`)}catch(v){u.warn(`Failed to parse JSON response: ${v}`)}}try{let f=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!f.ok){u.warn(`Could not fetch stored message: ${f.status}`);continue}let v=await f.text();u.info(`Fallback: Raw email length: ${v.length}`);let b=v.split(`
package/dist/cli.js CHANGED
@@ -1,8 +1,8 @@
1
1
  #!/usr/bin/env node
2
2
  import { createRequire as __cli_createRequire } from "module";
3
3
  const require = __cli_createRequire(import.meta.url);
4
- var To=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var ko=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),Q=(e,t)=>{for(var r in t)To(e,r,{get:t[r],enumerable:!0})};import*as ce from"fs";import*as ee from"path";function ze(e){let{projectPath:t}=e,r=e.projectName??ee.basename(ee.resolve(t));ce.mkdirSync(t,{recursive:!0});let o={name:r,type:"module",scripts:{test:"shiplight test","test:headed":"shiplight test --headed"},dependencies:{"@playwright/test":"^1.50.0",playwright:"^1.50.0",shiplightai:"latest",dotenv:"^16.4.7"}};ce.writeFileSync(ee.join(t,"package.json"),JSON.stringify(o,null,2)+`
5
- `),ce.writeFileSync(ee.join(t,"playwright.config.ts"),`import { defineConfig, shiplightConfig } from 'shiplightai';
4
+ var To=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var ko=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),te=(e,t)=>{for(var r in t)To(e,r,{get:t[r],enumerable:!0})};import*as R from"fs";import*as K from"path";function ze(e){let{projectPath:t}=e,r=e.projectName??K.basename(K.resolve(t));if(R.existsSync(t)&&R.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);R.mkdirSync(t,{recursive:!0});let o={name:r,type:"module",scripts:{test:"shiplight test","test:headed":"shiplight test --headed"},dependencies:{shiplightai:"latest",dotenv:"^16.4.7"}};R.writeFileSync(K.join(t,"package.json"),JSON.stringify(o,null,2)+`
5
+ `),R.writeFileSync(K.join(t,"playwright.config.ts"),`import { defineConfig, shiplightConfig } from 'shiplightai';
6
6
 
7
7
  export default defineConfig({
8
8
  ...shiplightConfig(),
@@ -21,8 +21,22 @@ export default defineConfig({
21
21
  },
22
22
  });
23
23
  `);let n=["node_modules/","test-results/","shiplight-report/",".shiplight/",".env","*.yaml.spec.ts",""].join(`
24
- `);ce.writeFileSync(ee.join(t,".gitignore"),n);let s=["# Shiplight API token (optional) \u2014 enables cloud sync tools","# Get yours at https://app.shiplight.ai/settings/api-tokens","# SHIPLIGHT_API_TOKEN=","","# AI provider API key (at least one required)","# Uncomment the one you want to use:","# GOOGLE_API_KEY= # \u2192 default model: gemini-3.1-flash-lite-preview","# ANTHROPIC_API_KEY= # \u2192 default model: claude-haiku-4-5","# OPENAI_API_KEY= # \u2192 default model: gpt-5.4-mini","","# Optional: override the default AI model","# Supports provider:model prefix for Azure, Bedrock, Vertex AI:","# WEB_AGENT_MODEL=claude-sonnet-4-6","# WEB_AGENT_MODEL=azure:gpt-4o","# WEB_AGENT_MODEL=bedrock:anthropic.claude-sonnet-4-6-v1","# WEB_AGENT_MODEL=vertex:gemini-2.5-pro","","# Optional: custom endpoint for OpenAI-compatible APIs","# OPENAI_BASE_URL=http://localhost:11434/v1","","# Optional: override starting URL for all tests","# PLAYWRIGHT_STARTING_URL=",""].join(`
25
- `);return ce.writeFileSync(ee.join(t,".env.example"),s),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example"]}}var At=_(()=>{"use strict"});var Et=_(()=>{"use strict";At()});var Pt={};Q(Pt,{runCreate:()=>Do});import*as xe from"path";import*as Te from"fs";function _e(){console.log(`
24
+ `);R.writeFileSync(K.join(t,".gitignore"),n);let s=["# Shiplight API token (optional) \u2014 enables cloud sync tools","# Get yours at https://app.shiplight.ai/settings/api-tokens","# SHIPLIGHT_API_TOKEN=","","# AI provider API key (at least one required).","# Uncomment the one you want to use \u2014 shiplightai auto-selects the","# default model for that provider (names below match the current","# defaults in packages/types/src/organization.ts):","# GOOGLE_API_KEY= # default model: gemini-3.1-flash-lite-preview","# ANTHROPIC_API_KEY= # default model: claude-haiku-4-5","# OPENAI_API_KEY= # default model: gpt-5.4-mini","","# Optional: override the default AI model.","# Supports provider:model prefix for Azure, Bedrock, Vertex AI.","# WEB_AGENT_MODEL=claude-sonnet-4-6","# WEB_AGENT_MODEL=azure:gpt-4o","# WEB_AGENT_MODEL=bedrock:anthropic.claude-sonnet-4-6-v1","# WEB_AGENT_MODEL=vertex:gemini-2.5-pro","","# Optional: custom endpoint for OpenAI-compatible APIs","# OPENAI_BASE_URL=http://localhost:11434/v1","","# Optional: override starting URL for all tests","# PLAYWRIGHT_STARTING_URL=",""].join(`
25
+ `);R.writeFileSync(K.join(t,".env.example"),s);let a=K.join(t,"tests");return R.mkdirSync(a,{recursive:!0}),R.writeFileSync(K.join(a,"example.test.yaml"),`goal: Verify the Shiplight homepage links to the quick-start docs
26
+ base_url: https://www.shiplight.ai
27
+ statements:
28
+ - URL: /
29
+
30
+ - VERIFY: The page title contains "Shiplight"
31
+ js: "await expect(page).toHaveTitle(/Shiplight/)"
32
+
33
+ - intent: Click the Install Plugin link in the navigation
34
+ action: click
35
+ locator: "getByRole('link', { name: 'Install Plugin' })"
36
+
37
+ - VERIFY: The browser lands on the docs site
38
+ js: "await expect(page).toHaveURL(/docs\\\\.shiplight\\\\.ai/)"
39
+ `),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example","tests/example.test.yaml"]}}var At=_(()=>{"use strict"});var Et=_(()=>{"use strict";At()});var Pt={};te(Pt,{runCreate:()=>Do});import*as xe from"path";import*as Te from"fs";function _e(){console.log(`
26
40
  Usage: shiplight create <path> [options]
27
41
 
28
42
  Scaffold a new Shiplight test project at <path>. Creates package.json,
@@ -50,7 +64,7 @@ Next steps:
50
64
  npm install
51
65
  npx playwright install chromium
52
66
  npx shiplight test
53
- `)}var $t=_(()=>{"use strict";Et()});import*as H from"fs";import*as te from"path";function Mt(e){let t=te.resolve(e);for(;;){if(H.existsSync(te.join(t,"package.json"))||H.existsSync(te.join(t,".shiplight")))return t;let r=te.dirname(t);if(r===t)break;t=r}return null}function It(e){return te.join(e,Fo)}function Ot(e,t){return te.join(It(e),`${Uo}${t}.json`)}function Lt(e,t){let r=Mt(t);if(!r)return;let o=It(r);H.mkdirSync(o,{recursive:!0});let i={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};H.writeFileSync(Ot(r,e),JSON.stringify(i),"utf-8")}function Ct(e,t){let r=Mt(t);if(r)try{H.unlinkSync(Ot(r,e))}catch{}}var Fo,Uo,Rt=_(()=>{"use strict";Fo=".shiplight/run",Uo="debug-"});var Ve={};Q(Ve,{findPlaywrightConfig:()=>Wo,spawnPlaywrightProcess:()=>Ho});import*as F from"fs";import*as X from"path";import{spawn as Bo}from"child_process";import{parse as Go}from"yaml";function Wo(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=X.resolve(e);for(;;){for(let i of t){let n=X.join(r,i);if(F.existsSync(n))return n}let o=X.dirname(r);if(o===r)break;r=o}return null}function jo(e,t,r,o){let i={...o},n=i.launchOptions?.args??[];return i.launchOptions={...i.launchOptions??{},args:[...n,`--remote-debugging-port=${r}`]},`// @generated by shiplightai \u2014 temporary debug test
67
+ `)}var $t=_(()=>{"use strict";Et()});import*as z from"fs";import*as re from"path";function Mt(e){let t=re.resolve(e);for(;;){if(z.existsSync(re.join(t,"package.json"))||z.existsSync(re.join(t,".shiplight")))return t;let r=re.dirname(t);if(r===t)break;t=r}return null}function It(e){return re.join(e,Fo)}function Ot(e,t){return re.join(It(e),`${Uo}${t}.json`)}function Lt(e,t){let r=Mt(t);if(!r)return;let o=It(r);z.mkdirSync(o,{recursive:!0});let i={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};z.writeFileSync(Ot(r,e),JSON.stringify(i),"utf-8")}function Ct(e,t){let r=Mt(t);if(r)try{z.unlinkSync(Ot(r,e))}catch{}}var Fo,Uo,Rt=_(()=>{"use strict";Fo=".shiplight/run",Uo="debug-"});var Ve={};te(Ve,{findPlaywrightConfig:()=>jo,spawnPlaywrightProcess:()=>Wo});import*as U from"fs";import*as Z from"path";import{spawn as Bo}from"child_process";import{parse as Go}from"yaml";function jo(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=Z.resolve(e);for(;;){for(let i of t){let n=Z.join(r,i);if(U.existsSync(n))return n}let o=Z.dirname(r);if(o===r)break;r=o}return null}function Ho(e,t,r,o){let i={...o},n=i.launchOptions?.args??[];return i.launchOptions={...i.launchOptions??{},args:[...n,`--remote-debugging-port=${r}`]},`// @generated by shiplightai \u2014 temporary debug test
54
68
  import { test } from 'shiplightai/fixture';
55
69
  ${`
56
70
  test.use(${JSON.stringify(i)});
@@ -74,15 +88,15 @@ test('__shiplight_debug__', async ({ page, agent }) => {
74
88
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
75
89
  await new Promise(() => {});
76
90
  });
77
- `}async function Nt(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function Ho(e){let{yamlFilePath:t,configPath:r}=e,o=X.dirname(r),i=await Nt(16174),n=await Nt(9222),s;if(!F.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let d=Go(F.readFileSync(t,"utf-8"));d?.use&&typeof d.use=="object"&&!Array.isArray(d.use)&&(s=d.use),d?.base_url&&!s?.baseURL&&(s={...s,baseURL:d.base_url}),d?.settings?.auto_dismiss_modal!==void 0&&(s={...s,autoDismissModal:!!d.settings.auto_dismiss_modal})}catch(d){console.error("[debugger] Could not parse YAML for `use` block:",d)}let a=X.dirname(X.resolve(t)),c=X.join(a,".__shiplight_debug__.yaml.spec.ts"),l=jo(t,i,n,s);F.writeFileSync(c,l);let u=Bo("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});u.stdout?.on("data",d=>{process.stderr.write(d)}),u.stderr?.on("data",d=>{process.stderr.write(d)});let g=()=>{u.killed||u.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),u.on("close",d=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{F.unlinkSync(c)}catch{}d!==0&&d!==null&&console.error(`[debugger] Playwright process exited with code ${d}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let d=0;d<180;d++){if(u.exitCode!==null){try{F.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${u.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(d===179){g();try{F.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(h=>setTimeout(h,1e3))}return{port:i,cleanup:async()=>{g();try{F.unlinkSync(c)}catch{}}}}var Ye=_(()=>{"use strict"});var Dt=_(()=>{"use strict"});var Ft=_(()=>{"use strict"});var Ut=_(()=>{"use strict"});import{v4 as Yn}from"uuid";var Bt=_(()=>{"use strict"});import{z as m}from"zod";var Gt,Je,Wt,ue,jt,Ht,Kt,R,zt,ke,Xe,qe=_(()=>{"use strict";Gt=m.enum(["JS_CODE","AI_MODE"]),Je=m.object({type:Gt,expression:m.string()}),Wt=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ue=m.object({uid:m.string(),type:Wt,comment:m.string().optional()}),jt=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Ht=ue.extend({type:m.literal("DRAFT"),description:m.string()}),Kt=ue.extend({type:m.literal("ACTION"),description:m.string(),action_entity:jt.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),R=m.lazy(()=>m.union([Ht,Kt,ue.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(R),reference_id:m.number().optional()}),ue.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Je,then:m.array(R),else:m.array(R).optional()}),ue.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Je,body:m.array(R),timeout_ms:m.number().optional()})])),zt=m.object({name:m.string(),statements:m.array(R),teardown:m.array(R).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),ke=m.object({tests:m.array(zt).min(1),beforeAll:m.array(R).optional(),afterAll:m.array(R).optional(),beforeEach:m.array(R).optional(),afterEach:m.array(R).optional()}),Xe=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(R).optional(),teardown:m.array(R).optional(),last_modified_at:m.string().optional(),testGroup:ke.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 Ko,parse as Xt,parseAllDocuments as es,parseDocument as zo,Document as Vo,isMap as he,isSeq as z}from"yaml";import{v4 as K}from"uuid";function qt(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(U)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(U)),r}function ye(e,t){if(e.testGroup)return Qt(e,t);let r=qt(e,t),o=new Vo(r);return e.comment&&(o.commentBefore=e.comment),Vt(o,e.statements??[]),e.teardown&&Vt(o,e.teardown,"teardown"),o.toString(Zt)}function Vt(e,t,r="statements"){let o=e.contents;if(!o||!he(o))return;let i=o.get(r,!0);z(i)&&me(i,t)}function me(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),he(i)){let n=i;if(o.type==="STEP"){let s=n.get("statements",!0);z(s)&&me(s,o.statements)}else if(o.type==="IF_ELSE"){let s=n.get("THEN",!0);z(s)&&me(s,o.then);let a=n.get("ELSE",!0);z(a)&&o.else&&me(a,o.else)}else if(o.type==="WHILE_LOOP"){let s=n.get("DO",!0);z(s)&&me(s,o.body)}}}}function Qt(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(U)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(U)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(U)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(U)),i.tests=r.tests.map(n=>{let s={name:n.name};return n.skip!==void 0&&(s.skip=n.skip),n.timeout!==void 0&&(s.timeout=n.timeout),n.fail!==void 0&&(s.fail=n.fail),n.only!==void 0&&(s.only=n.only),n.slow!==void 0&&(s.slow=n.slow),s.statements=n.statements.map(U),n.teardown&&n.teardown.length>0&&(s.teardown=n.teardown.map(U)),s}),o.suite=i,Ko(o,Zt)}function U(e){switch(e.type){case"DRAFT":return Yo(e);case"ACTION":return Jo(e);case"STEP":return Xo(e);case"IF_ELSE":return qo(e);case"WHILE_LOOP":return Zo(e)}}function Yo(e){return{intent:e.description}}function Jo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let s=e.action_entity.xpath;if(s&&(i.xpath=s),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[a,c]of Object.entries(o.kwargs))i[a]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function Xo(e){let t={STEP:e.description,statements:e.statements.map(U)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function qo(e){let t={IF:er(e.condition),THEN:e.then.map(U)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(U)),t}function Zo(e){let t={WHILE:er(e.condition),DO:e.body.map(U)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function er(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function be(e){try{let t=Xt(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()),r}catch{return{}}}function Ze(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(Ze);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=Ze(n);return o}function L(e){if(e.length>Yt)throw new Error(`YAML input too large (${e.length} bytes, max ${Yt})`);let t=Ze(Xt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Qo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:B(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=B(t.teardown));let o=Xe.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return Ae(e,i),i}function Qo(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 i={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:B(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=B(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=B(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=B(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=B(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=B(t.afterEach));let n=ke.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function B(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(ei)}function ei(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 ti(t);if("WHILE"in t)return ri(t);if("STEP"in t)return oi(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:K(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:K(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:K(),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:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:K(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}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.');return{uid:K(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:K(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Jt({...o,action:"function",functionName:r})}if("action"in t)return Jt(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:K(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function tr(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 ti(e){let t=tr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:K(),type:"IF_ELSE",condition:t,then:B(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=B(e.ELSE)),o}function ri(e){let t=tr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:K(),type:"WHILE_LOOP",condition:t,body:B(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function oi(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:K(),type:"STEP",description:t,statements:B(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Jt(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:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[l,p]of Object.entries(e))ii.has(l)||(s[l]=p);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};o&&(a.locator=o),i&&(a.xpath=i);let c={uid:K(),type:"ACTION",description:r,action_entity:a};return n&&(c.use_pure_vision=!0),c}function Ae(e,t){let r;try{r=zo(e)}catch{return}let o=r.contents;if(!o||!he(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);z(n)&&t.statements&&de(n,t.statements);let s=i.get("teardown",!0);z(s)&&t.teardown&&de(s,t.teardown)}function de(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 o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&he(o)){let n=o.get("statements",!0);z(n)&&de(n,i.statements)}else if(i.type==="IF_ELSE"&&he(o)){let n=o.get("THEN",!0);z(n)&&de(n,i.then);let s=o.get("ELSE",!0);z(s)&&i.else&&de(s,i.else)}else if(i.type==="WHILE_LOOP"&&he(o)){let n=o.get("DO",!0);z(n)&&de(n,i.body)}}}var Zt,Yt,ii,Ee=_(()=>{"use strict";qe();Zt={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Yt=1024*1024;ii=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as ns,stringify as ss}from"yaml";var rr=_(()=>{"use strict";Ee()});var Qe,Pe,$e=_(()=>{"use strict";Qe=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},Pe=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=Qe(i);for(let s of n)r(s.statements)}};return r(e),t}});function nr(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";ir.has(i)||t++}return{action:t,draft:r}}function si(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function or(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function et(e,t){let r=t?.coverageThreshold??ni,o=[],i=[],n;try{n=L(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let s=[...Pe(n.statements??[]),...n.teardown?Pe(n.teardown):[]],{action:a,draft:c}=nr(s),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of s){if(g.type==="ACTION"){let f=g,d=f.action_entity?.action_data?.action_name??"";if(d==="js_code"||d==="js_action"||d==="verify"||d==="ai_assert"){let h=f.action_entity?.action_data?.kwargs?.code;if(typeof h=="string"){let w=si(h);if(w){let b=f.description||d;o.push(`Invalid JS in "${b}": ${w}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${d}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${d}. ${l}`)}}}let p=a+c,u=p>0?Math.round(a/p*100):0;return p>0&&u/100<r&&i.push(`Low action coverage: ${a}/${p} statements (${u}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:a,draft:c,coverage:u}}}var ni,ir,sr=_(()=>{"use strict";Ee();$e();ni=.5,ir=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Me=_(()=>{"use strict"});var ar=_(()=>{"use strict";Me()});var cr,ci,lr,pr,we,li,pi,ur=_(()=>{"use strict";cr=112,ci=1080-cr,lr={"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"}},pr={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"]},we=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),pr[e].map(o=>lr[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},li={desktop:{label:"Desktop",type:"desktop",devices:we("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:we("mobile")}},pi={desktop:{label:"Desktop",type:"desktop",devices:we("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:we("mobile",!0)}}});var dr=_(()=>{"use strict"});function tt(){return{version:"1.0",entries:{}}}var hr=_(()=>{"use strict";$e()});var V,rt,fr=_(()=>{"use strict";V=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(V||{}),rt=18e4});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});var br=_(()=>{"use strict";Me()});var ne=_(()=>{"use strict";Dt();Ft();Ut();Bt();rr();sr();Ee();qe();ar();ur();dr();hr();$e();fr();gr();mr();yr();br();Me()});import{stringify as di}from"yaml";import{createHash as Mi}from"crypto";import{parse as Ii,stringify as Pr}from"yaml";import{readFileSync as Oi}from"fs";import{resolve as $r,dirname as Li}from"path";import{parse as _r,stringify as Ci}from"yaml";import{readFileSync as Gi,writeFileSync as Wi,mkdirSync as ji}from"fs";import{dirname as Hi}from"path";function re(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function E(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function hi(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function fi(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 Oe(e){let t=hi(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=fi(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function wr(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:gi.includes(t)}function yi(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!mi.includes(t)}function S(e,t){O.set(e,t)}function bi(e){return O.get(e)}function le(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function Sr(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 o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(s=>o.includes(s)||i.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}function q(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let s=e[n],a=`${o}.${n}`,c=wi(s,t,a,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function wi(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return Si(e,t,r,o);case"ACTION":return vi(e,t,r,o);case"STEP":return _i(e,t,r,o);case"IF_ELSE":return xi(e,t,r,o);case"WHILE_LOOP":return Ti(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function Si(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let s=JSON.stringify(n);return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${s}, '${r}');`]}function vi(e,t,r,o){let i=" ".repeat(t),n=e.description,s=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",u=l.action_description||"",g=bi(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},d=g(l,r,f);if(o.noAgent){if(wr(l))return[`${i}// ${r}: ${E(u)}`,`${i}// AI action: ${E(u)} (requires agent - skipped in hook)`];let y=ki(l,p,i,r);return y||[`${i}// ${r}: ${E(u)}`,...d.map(A=>`${i}${A}`)]}if(wr(l))return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,...d.map(y=>`${i}${y}`)];let h=JSON.stringify(u),w=d.map(y=>`${i} ${y}`),b=yi(l),v=s?`'${s}'`:"undefined";return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...w,`${i}}, ${h}, '${r}', ${v}, ${b});`]}function _i(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${E(e.description)}`);let s=q(e.statements,t,o,r);return n.push(...s),n}function xi(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${E(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let s=q(e.then,t+1,o,`${r}.then`);if(n.push(...s),e.else&&e.else.length>0){n.push(`${i}} else {`);let a=q(e.else,t+1,o,`${r}.else`);n.push(...a)}return n.push(`${i}}`),n}function Ti(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let s=e.timeout_ms??rt,a=s/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${s};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${E(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${u}, "${r}")) {`)}let p=q(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function ki(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},s=n.timeout_ms??it;switch(t){case"go_to_url":case"open_tab":{let a=n.url||"";return[`${r}// ${o}: ${E(i)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goForward();`];case"input_text":{let a=n.text||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${s} });`]:null}case"select_dropdown_option":{let a=n.text||n.label||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${s} });`]:null}default:return null}}function Ai(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...vr("beforeEach",t.beforeEach,n)),r.push(""));let s=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);for(let l of t.parameters){let p=kr(e,l.values);r.push(...Ce(p,`${c}${re(a)} [${re(l.name)}]`,n,0,s)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);r.push(...Ce(e,`${c}${re(a)}`,n,0,s))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...vr("afterEach",t.afterEach,n))),Er(r,i),r.join(`
78
- `)}function Ei(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore},s=t?.testName||"Test Suite",a=ot(t?.tags);r.push(`test.describe.serial('${a}${re(s)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ie("beforeAll",e.beforeAll,n,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ie("beforeEach",e.beforeEach,n,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],u=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let g of p.parameters){let f=kr(p.testFlow,g.values);r.push(...Ce(f,`${re(p.name)} [${re(g.name)}]`,n,1,u)),r.push("")}else r.push(...Ce(p.testFlow,re(p.name),n,1,u)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ie("afterEach",e.afterEach,n,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ie("afterAll",e.afterAll,n,1)),r.push("});"),Er(r,i),r.join(`
79
- `)}function ot(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Le(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case V.ACTION:{let s=i.action_entity?.action_data?.kwargs;if(s?.args&&Array.isArray(s.args))for(let a of s.args)typeof a=="string"&&Pi.includes(a)&&t.add(a);break}case V.STEP:r(i.statements);break;case V.IF_ELSE:{let n=i;r(n.then),n.else&&r(n.else);break}case V.WHILE_LOOP:r(i.body);break}}return r(e),t}function $i(e){let t=Le(e.statements??[]);if(e.teardown)for(let r of Le(e.teardown))t.add(r);return t}function nt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ce(e,t,r,o=0,i){let n=" ".repeat(o),s=[],a=$i(e),c=nt(a),l=i?.only?"test.only":"test";s.push(`${n}${l}('${t}', async (${c}) => {`),i?.skip===!0?s.push(`${n} test.skip();`):typeof i?.skip=="string"&&s.push(`${n} test.skip(true, '${re(i.skip)}');`),i?.fail===!0?s.push(`${n} test.fail();`):typeof i?.fail=="string"&&s.push(`${n} test.fail(true, '${re(i.fail)}');`),i?.slow&&s.push(`${n} test.slow();`),i?.timeout&&s.push(`${n} test.setTimeout(${i.timeout});`);let p=e.teardown&&e.teardown.length>0,u=o+1;if(p){if(s.push(`${n} try {`),e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let f=q(e.statements,u+1,r);s.push(...f)}s.push(`${n} } finally {`),s.push(`${n} // Teardown`);let g=q(e.teardown,u+1,r,"teardown");s.push(...g),s.push(`${n} }`)}else if(e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let g=q(e.statements,u,r);s.push(...g)}return s.push(`${n}});`),s}function vr(e,t,r){let o=[],i=Tr(t),n=Le(i),s=nt(n);return o.push(`test.${e}(async (${s}) => {`),o.push(...q(i,1,r,e)),o.push("});"),o}function Ie(e,t,r,o){let i=" ".repeat(o),n=[],s=Tr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};n.push(`${i}test.${e}(async ({ browser }, workerInfo) => {`),n.push(`${i} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),n.push(...q(s,o+1,c,e)),n.push(`${i} await page.close();`),n.push(`${i}});`)}else{let c=Le(s),l=nt(c);n.push(`${i}test.${e}(async (${l}) => {`),n.push(...q(s,o+1,r,e)),n.push(`${i}});`)}return n}function Tr(e){let r=di({goal:"_hook",statements:e});return L(r).statements??[]}function kr(e,t){let r=ye(e);for(let[o,i]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(i));return L(r)}function Ar(){return["import { test, expect } from 'shiplightai/fixture';"]}function Er(e,t){if(t.size>0){let r=0;for(let i=0;i<e.length;i++)e[i].startsWith("import ")&&(r=i+1);let o=Array.from(t);e.splice(r,0,...o)}}function Mr(e,t){let r={expandingPaths:new Set([$r(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=se(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=se(o.teardown,t,r));for(let i of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[i])&&(o[i]=se(o[i],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function se(e,t,r){let o=[];for(let i of e)if(Ri(i)){let n=Ni(i,t,r);o.push(...n)}else o.push(Di(i,t,r));return o}function Ri(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Ni(e,t,r){if(r.depth>=xr)throw new Error(`Template expansion exceeded maximum depth of ${xr}. Check for deeply nested or circular template references.`);let o=$r(Li(t),e.template);if(r.expandingPaths.has(o))throw new Error(`Circular template reference detected: ${o} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${o}`);r.referencedPaths.add(o);let i;try{i=Oi(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let n=_r(i);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let s=n.params||[],a=e.params||{};for(let p of s)if(!(p in a))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${s.join(", ")}]`);let c=n.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(a).length>0){let u=Ci(c);for(let[g,f]of Object.entries(a))u=u.split(`<<${g}>>`).join(String(f));c=_r(u)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return se(c,o,l)}function Di(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=se(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=se(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=se(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=se(o.DO,t,r)),o}function at(e,t){let r=Ii(e),o=r?.name,i=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new st('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Ui(r,o,i,n,t)}return Fi(r,o,i,n,t)}function Fi(e,t,r,o,i){let n=e?.beforeEach,s=e?.afterEach,a=Ir(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,u=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 st(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(i&&e&&typeof e=="object"){let w=Mr(e,i);e=w.doc,f=w.referencedTemplatePaths}let d=Pr(e),h=L(d);return i&&(fe(h.statements??[],i,"main"),h.teardown&&fe(h.teardown,i,"teardown")),{testFlow:h,name:t,tags:r,use:o,beforeEach:n,afterEach:s,parameters:a,timeout:c,skip:l,fail:p,only:u,slow:g,referencedTemplatePaths:f}}function Ui(e,t,r,o,i){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,a=n.afterAll,c=n.beforeEach,l=n.afterEach,p=[],u=n.tests.map(d=>{if(!d.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(d.statements)||d.statements.length===0)throw new Error(`Suite test "${d.name}" must have a non-empty "statements" array.`);let h={goal:d.name,statements:d.statements};d.teardown&&(h.teardown=d.teardown);let w=[],b=h;if(i&&typeof h=="object"){let k=Mr(h,i);b=k.doc,w=k.referencedTemplatePaths,p.push(...w)}let v=Pr(b),y=L(v),A=Ir(d.parameters);return{testFlow:y,name:d.name,tags:Array.isArray(d.tags)?d.tags:void 0,parameters:A,timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}}),g=n.base_url,f=g?{...o,baseURL:g}:o;return{suite:{beforeAll:s,afterAll:a,beforeEach:c,afterEach:l,tests:u},name:t,tags:r,use:f,referencedTemplatePaths:p}}function Ir(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 fe(e,t,r){for(let o=0;o<e.length;o++){let i=e[o],n=`${r}.${o}`,s=i.description||"";if(i.uid=Bi(t,n,s),i.type===V.STEP)fe(i.statements,t,n);else if(i.type===V.IF_ELSE){let a=i;fe(a.then,t,`${n}.then`),a.else&&fe(a.else,t,`${n}.else`)}else i.type===V.WHILE_LOOP&&fe(i.body,t,`${n}.body`)}}function Bi(e,t,r){let o=Mi("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${o.slice(0,8)}-${o.slice(8,12)}-${o.slice(12,16)}-${o.slice(16,20)}-${o.slice(20,32)}`}function Or(e,t){let r;try{r=Gi(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return Ki(r,e,t)}function Ki(e,t,r){let o=/\btemplate:\s/.test(e),i=/^suite:/m.test(e),n=o||i?null:et(e);if(n&&!n.valid)return{valid:!1,errors:n.errors,warnings:[],stats:n.stats};let s,a,c=[];try{let l=r?.parsed??at(e,t);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore},u=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?s=Ei(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):s=Ai(l.testFlow,{...p,testName:l.name,tags:l.tags,use:u,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 g=s.split(`
91
+ `}async function Nt(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function Wo(e){let{yamlFilePath:t,configPath:r}=e,o=Z.dirname(r),i=await Nt(16174),n=await Nt(9222),s;if(!U.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let d=Go(U.readFileSync(t,"utf-8"));d?.use&&typeof d.use=="object"&&!Array.isArray(d.use)&&(s=d.use),d?.base_url&&!s?.baseURL&&(s={...s,baseURL:d.base_url}),d?.settings?.auto_dismiss_modal!==void 0&&(s={...s,autoDismissModal:!!d.settings.auto_dismiss_modal})}catch(d){console.error("[debugger] Could not parse YAML for `use` block:",d)}let a=Z.dirname(Z.resolve(t)),c=Z.join(a,".__shiplight_debug__.yaml.spec.ts"),l=Ho(t,i,n,s);U.writeFileSync(c,l);let u=Bo("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});u.stdout?.on("data",d=>{process.stderr.write(d)}),u.stderr?.on("data",d=>{process.stderr.write(d)});let g=()=>{u.killed||u.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),u.on("close",d=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{U.unlinkSync(c)}catch{}d!==0&&d!==null&&console.error(`[debugger] Playwright process exited with code ${d}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let d=0;d<180;d++){if(u.exitCode!==null){try{U.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${u.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(d===179){g();try{U.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(h=>setTimeout(h,1e3))}return{port:i,cleanup:async()=>{g();try{U.unlinkSync(c)}catch{}}}}var Ye=_(()=>{"use strict"});var Dt=_(()=>{"use strict"});var Ft=_(()=>{"use strict"});var Ut=_(()=>{"use strict"});import{v4 as Yn}from"uuid";var Bt=_(()=>{"use strict"});import{z as m}from"zod";var Gt,Je,jt,ue,Ht,Wt,Kt,N,zt,ke,Xe,qe=_(()=>{"use strict";Gt=m.enum(["JS_CODE","AI_MODE"]),Je=m.object({type:Gt,expression:m.string()}),jt=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ue=m.object({uid:m.string(),type:jt,comment:m.string().optional()}),Ht=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Wt=ue.extend({type:m.literal("DRAFT"),description:m.string()}),Kt=ue.extend({type:m.literal("ACTION"),description:m.string(),action_entity:Ht.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),N=m.lazy(()=>m.union([Wt,Kt,ue.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(N),reference_id:m.number().optional()}),ue.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Je,then:m.array(N),else:m.array(N).optional()}),ue.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Je,body:m.array(N),timeout_ms:m.number().optional()})])),zt=m.object({name:m.string(),statements:m.array(N),teardown:m.array(N).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),ke=m.object({tests:m.array(zt).min(1),beforeAll:m.array(N).optional(),afterAll:m.array(N).optional(),beforeEach:m.array(N).optional(),afterEach:m.array(N).optional()}),Xe=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(N).optional(),teardown:m.array(N).optional(),last_modified_at:m.string().optional(),testGroup:ke.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 Ko,parse as Xt,parseAllDocuments as es,parseDocument as zo,Document as Vo,isMap as he,isSeq as Y}from"yaml";import{v4 as V}from"uuid";function qt(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(B)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(B)),r}function ye(e,t){if(e.testGroup)return Qt(e,t);let r=qt(e,t),o=new Vo(r);return e.comment&&(o.commentBefore=e.comment),Vt(o,e.statements??[]),e.teardown&&Vt(o,e.teardown,"teardown"),o.toString(Zt)}function Vt(e,t,r="statements"){let o=e.contents;if(!o||!he(o))return;let i=o.get(r,!0);Y(i)&&me(i,t)}function me(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),he(i)){let n=i;if(o.type==="STEP"){let s=n.get("statements",!0);Y(s)&&me(s,o.statements)}else if(o.type==="IF_ELSE"){let s=n.get("THEN",!0);Y(s)&&me(s,o.then);let a=n.get("ELSE",!0);Y(a)&&o.else&&me(a,o.else)}else if(o.type==="WHILE_LOOP"){let s=n.get("DO",!0);Y(s)&&me(s,o.body)}}}}function Qt(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(B)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(B)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(B)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(B)),i.tests=r.tests.map(n=>{let s={name:n.name};return n.skip!==void 0&&(s.skip=n.skip),n.timeout!==void 0&&(s.timeout=n.timeout),n.fail!==void 0&&(s.fail=n.fail),n.only!==void 0&&(s.only=n.only),n.slow!==void 0&&(s.slow=n.slow),s.statements=n.statements.map(B),n.teardown&&n.teardown.length>0&&(s.teardown=n.teardown.map(B)),s}),o.suite=i,Ko(o,Zt)}function B(e){switch(e.type){case"DRAFT":return Yo(e);case"ACTION":return Jo(e);case"STEP":return Xo(e);case"IF_ELSE":return qo(e);case"WHILE_LOOP":return Zo(e)}}function Yo(e){return{intent:e.description}}function Jo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let s=e.action_entity.xpath;if(s&&(i.xpath=s),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[a,c]of Object.entries(o.kwargs))i[a]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function Xo(e){let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function qo(e){let t={IF:er(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function Zo(e){let t={WHILE:er(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function er(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function we(e){try{let t=Xt(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()),r}catch{return{}}}function Ze(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(Ze);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=Ze(n);return o}function L(e){if(e.length>Yt)throw new Error(`YAML input too large (${e.length} bytes, max ${Yt})`);let t=Ze(Xt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Qo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:G(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=G(t.teardown));let o=Xe.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return Ae(e,i),i}function Qo(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 i={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:G(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=G(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=G(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=G(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=G(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=G(t.afterEach));let n=ke.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function G(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(ei)}function ei(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 ti(t);if("WHILE"in t)return ri(t);if("STEP"in t)return oi(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:V(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:V(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:V(),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:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:V(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}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.');return{uid:V(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:V(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Jt({...o,action:"function",functionName:r})}if("action"in t)return Jt(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:V(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function tr(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 ti(e){let t=tr(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:V(),type:"IF_ELSE",condition:t,then:G(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=G(e.ELSE)),o}function ri(e){let t=tr(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:V(),type:"WHILE_LOOP",condition:t,body:G(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function oi(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:V(),type:"STEP",description:t,statements:G(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Jt(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:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[l,p]of Object.entries(e))ii.has(l)||(s[l]=p);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};o&&(a.locator=o),i&&(a.xpath=i);let c={uid:V(),type:"ACTION",description:r,action_entity:a};return n&&(c.use_pure_vision=!0),c}function Ae(e,t){let r;try{r=zo(e)}catch{return}let o=r.contents;if(!o||!he(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);Y(n)&&t.statements&&de(n,t.statements);let s=i.get("teardown",!0);Y(s)&&t.teardown&&de(s,t.teardown)}function de(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 o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&he(o)){let n=o.get("statements",!0);Y(n)&&de(n,i.statements)}else if(i.type==="IF_ELSE"&&he(o)){let n=o.get("THEN",!0);Y(n)&&de(n,i.then);let s=o.get("ELSE",!0);Y(s)&&i.else&&de(s,i.else)}else if(i.type==="WHILE_LOOP"&&he(o)){let n=o.get("DO",!0);Y(n)&&de(n,i.body)}}}var Zt,Yt,ii,Ee=_(()=>{"use strict";qe();Zt={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Yt=1024*1024;ii=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as ns,stringify as ss}from"yaml";var rr=_(()=>{"use strict";Ee()});var Qe,Pe,$e=_(()=>{"use strict";Qe=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},Pe=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=Qe(i);for(let s of n)r(s.statements)}};return r(e),t}});function nr(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";ir.has(i)||t++}return{action:t,draft:r}}function si(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function or(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function et(e,t){let r=t?.coverageThreshold??ni,o=[],i=[],n;try{n=L(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let s=[...Pe(n.statements??[]),...n.teardown?Pe(n.teardown):[]],{action:a,draft:c}=nr(s),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of s){if(g.type==="ACTION"){let f=g,d=f.action_entity?.action_data?.action_name??"";if(d==="js_code"||d==="js_action"||d==="verify"||d==="ai_assert"){let h=f.action_entity?.action_data?.kwargs?.code;if(typeof h=="string"){let b=si(h);if(b){let w=f.description||d;o.push(`Invalid JS in "${w}": ${b}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${d}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let d=or(f.condition.expression);d&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${d}. ${l}`)}}}let p=a+c,u=p>0?Math.round(a/p*100):0;return p>0&&u/100<r&&i.push(`Low action coverage: ${a}/${p} statements (${u}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:a,draft:c,coverage:u}}}var ni,ir,sr=_(()=>{"use strict";Ee();$e();ni=.5,ir=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Me=_(()=>{"use strict"});var ar=_(()=>{"use strict";Me()});var cr,ci,lr,pr,be,li,pi,ur=_(()=>{"use strict";cr=112,ci=1080-cr,lr={"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"}},pr={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"]},be=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),pr[e].map(o=>lr[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},li={desktop:{label:"Desktop",type:"desktop",devices:be("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:be("mobile")}},pi={desktop:{label:"Desktop",type:"desktop",devices:be("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:be("mobile",!0)}}});var dr=_(()=>{"use strict"});function tt(){return{version:"1.0",entries:{}}}var hr=_(()=>{"use strict";$e()});var J,rt,fr=_(()=>{"use strict";J=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(J||{}),rt=18e4});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});var wr=_(()=>{"use strict";Me()});var se=_(()=>{"use strict";Dt();Ft();Ut();Bt();rr();sr();Ee();qe();ar();ur();dr();hr();$e();fr();gr();mr();yr();wr();Me()});import{stringify as di}from"yaml";import{createHash as Mi}from"crypto";import{parse as Ii,stringify as Pr}from"yaml";import{readFileSync as Oi}from"fs";import{resolve as $r,dirname as Li}from"path";import{parse as _r,stringify as Ci}from"yaml";import{readFileSync as Gi,writeFileSync as ji,mkdirSync as Hi}from"fs";import{dirname as Wi}from"path";function oe(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function E(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function hi(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function fi(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 Oe(e){let t=hi(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=fi(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function br(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:gi.includes(t)}function yi(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!mi.includes(t)}function S(e,t){O.set(e,t)}function wi(e){return O.get(e)}function le(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function Sr(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 o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(s=>o.includes(s)||i.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}function Q(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let s=e[n],a=`${o}.${n}`,c=bi(s,t,a,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function bi(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return Si(e,t,r,o);case"ACTION":return vi(e,t,r,o);case"STEP":return _i(e,t,r,o);case"IF_ELSE":return xi(e,t,r,o);case"WHILE_LOOP":return Ti(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function Si(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let s=JSON.stringify(n);return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${s}, '${r}');`]}function vi(e,t,r,o){let i=" ".repeat(t),n=e.description,s=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${E(n)}`,`${i}// DRAFT: ${E(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${E(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",u=l.action_description||"",g=wi(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},d=g(l,r,f);if(o.noAgent){if(br(l))return[`${i}// ${r}: ${E(u)}`,`${i}// AI action: ${E(u)} (requires agent - skipped in hook)`];let y=ki(l,p,i,r);return y||[`${i}// ${r}: ${E(u)}`,...d.map(A=>`${i}${A}`)]}if(br(l))return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,...d.map(y=>`${i}${y}`)];let h=JSON.stringify(u),b=d.map(y=>`${i} ${y}`),w=yi(l),v=s?`'${s}'`:"undefined";return[`${i}// ${r}: ${E(u)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...b,`${i}}, ${h}, '${r}', ${v}, ${w});`]}function _i(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${E(e.description)}`);let s=Q(e.statements,t,o,r);return n.push(...s),n}function xi(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${E(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let s=Q(e.then,t+1,o,`${r}.then`);if(n.push(...s),e.else&&e.else.length>0){n.push(`${i}} else {`);let a=Q(e.else,t+1,o,`${r}.else`);n.push(...a)}return n.push(`${i}}`),n}function Ti(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let s=e.timeout_ms??rt,a=s/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${s};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${E(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${u}, "${r}")) {`)}let p=Q(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function ki(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},s=n.timeout_ms??it;switch(t){case"go_to_url":case"open_tab":{let a=n.url||"";return[`${r}// ${o}: ${E(i)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${E(i)}`,`${r}await page.goForward();`];case"input_text":{let a=n.text||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${s} });`]:null}case"select_dropdown_option":{let a=n.text||n.label||"",c=Oe(e);return c?[`${r}// ${o}: ${E(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${s} });`]:null}default:return null}}function Ai(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...vr("beforeEach",t.beforeEach,n)),r.push(""));let s=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);for(let l of t.parameters){let p=kr(e,l.values);r.push(...Ce(p,`${c}${oe(a)} [${oe(l.name)}]`,n,0,s)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=ot(t?.tags);r.push(...Ce(e,`${c}${oe(a)}`,n,0,s))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...vr("afterEach",t.afterEach,n))),Er(r,i),r.join(`
92
+ `)}function Ei(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Ar()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore},s=t?.testName||"Test Suite",a=ot(t?.tags);r.push(`test.describe.serial('${a}${oe(s)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ie("beforeAll",e.beforeAll,n,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ie("beforeEach",e.beforeEach,n,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],u=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let g of p.parameters){let f=kr(p.testFlow,g.values);r.push(...Ce(f,`${oe(p.name)} [${oe(g.name)}]`,n,1,u)),r.push("")}else r.push(...Ce(p.testFlow,oe(p.name),n,1,u)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ie("afterEach",e.afterEach,n,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ie("afterAll",e.afterAll,n,1)),r.push("});"),Er(r,i),r.join(`
93
+ `)}function ot(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Le(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case J.ACTION:{let s=i.action_entity?.action_data?.kwargs;if(s?.args&&Array.isArray(s.args))for(let a of s.args)typeof a=="string"&&Pi.includes(a)&&t.add(a);break}case J.STEP:r(i.statements);break;case J.IF_ELSE:{let n=i;r(n.then),n.else&&r(n.else);break}case J.WHILE_LOOP:r(i.body);break}}return r(e),t}function $i(e){let t=Le(e.statements??[]);if(e.teardown)for(let r of Le(e.teardown))t.add(r);return t}function nt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ce(e,t,r,o=0,i){let n=" ".repeat(o),s=[],a=$i(e),c=nt(a),l=i?.only?"test.only":"test";s.push(`${n}${l}('${t}', async (${c}) => {`),i?.skip===!0?s.push(`${n} test.skip();`):typeof i?.skip=="string"&&s.push(`${n} test.skip(true, '${oe(i.skip)}');`),i?.fail===!0?s.push(`${n} test.fail();`):typeof i?.fail=="string"&&s.push(`${n} test.fail(true, '${oe(i.fail)}');`),i?.slow&&s.push(`${n} test.slow();`),i?.timeout&&s.push(`${n} test.setTimeout(${i.timeout});`);let p=e.teardown&&e.teardown.length>0,u=o+1;if(p){if(s.push(`${n} try {`),e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let f=Q(e.statements,u+1,r);s.push(...f)}s.push(`${n} } finally {`),s.push(`${n} // Teardown`);let g=Q(e.teardown,u+1,r,"teardown");s.push(...g),s.push(`${n} }`)}else if(e.statements&&e.statements.length>0){s.push(`${n} // Test steps`);let g=Q(e.statements,u,r);s.push(...g)}return s.push(`${n}});`),s}function vr(e,t,r){let o=[],i=Tr(t),n=Le(i),s=nt(n);return o.push(`test.${e}(async (${s}) => {`),o.push(...Q(i,1,r,e)),o.push("});"),o}function Ie(e,t,r,o){let i=" ".repeat(o),n=[],s=Tr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};n.push(`${i}test.${e}(async ({ browser }, workerInfo) => {`),n.push(`${i} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),n.push(...Q(s,o+1,c,e)),n.push(`${i} await page.close();`),n.push(`${i}});`)}else{let c=Le(s),l=nt(c);n.push(`${i}test.${e}(async (${l}) => {`),n.push(...Q(s,o+1,r,e)),n.push(`${i}});`)}return n}function Tr(e){let r=di({goal:"_hook",statements:e});return L(r).statements??[]}function kr(e,t){let r=ye(e);for(let[o,i]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(i));return L(r)}function Ar(){return["import { test, expect } from 'shiplightai/fixture';"]}function Er(e,t){if(t.size>0){let r=0;for(let i=0;i<e.length;i++)e[i].startsWith("import ")&&(r=i+1);let o=Array.from(t);e.splice(r,0,...o)}}function Mr(e,t){let r={expandingPaths:new Set([$r(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=ae(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=ae(o.teardown,t,r));for(let i of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[i])&&(o[i]=ae(o[i],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ae(e,t,r){let o=[];for(let i of e)if(Ri(i)){let n=Ni(i,t,r);o.push(...n)}else o.push(Di(i,t,r));return o}function Ri(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Ni(e,t,r){if(r.depth>=xr)throw new Error(`Template expansion exceeded maximum depth of ${xr}. Check for deeply nested or circular template references.`);let o=$r(Li(t),e.template);if(r.expandingPaths.has(o))throw new Error(`Circular template reference detected: ${o} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${o}`);r.referencedPaths.add(o);let i;try{i=Oi(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let n=_r(i);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let s=n.params||[],a=e.params||{};for(let p of s)if(!(p in a))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${s.join(", ")}]`);let c=n.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(a).length>0){let u=Ci(c);for(let[g,f]of Object.entries(a))u=u.split(`<<${g}>>`).join(String(f));c=_r(u)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return ae(c,o,l)}function Di(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=ae(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=ae(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=ae(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=ae(o.DO,t,r)),o}function at(e,t){let r=Ii(e),o=r?.name,i=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new st('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Ui(r,o,i,n,t)}return Fi(r,o,i,n,t)}function Fi(e,t,r,o,i){let n=e?.beforeEach,s=e?.afterEach,a=Ir(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,u=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 st(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(i&&e&&typeof e=="object"){let b=Mr(e,i);e=b.doc,f=b.referencedTemplatePaths}let d=Pr(e),h=L(d);return i&&(fe(h.statements??[],i,"main"),h.teardown&&fe(h.teardown,i,"teardown")),{testFlow:h,name:t,tags:r,use:o,beforeEach:n,afterEach:s,parameters:a,timeout:c,skip:l,fail:p,only:u,slow:g,referencedTemplatePaths:f}}function Ui(e,t,r,o,i){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,a=n.afterAll,c=n.beforeEach,l=n.afterEach,p=[],u=n.tests.map(d=>{if(!d.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(d.statements)||d.statements.length===0)throw new Error(`Suite test "${d.name}" must have a non-empty "statements" array.`);let h={goal:d.name,statements:d.statements};d.teardown&&(h.teardown=d.teardown);let b=[],w=h;if(i&&typeof h=="object"){let k=Mr(h,i);w=k.doc,b=k.referencedTemplatePaths,p.push(...b)}let v=Pr(w),y=L(v),A=Ir(d.parameters);return{testFlow:y,name:d.name,tags:Array.isArray(d.tags)?d.tags:void 0,parameters:A,timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}}),g=n.base_url,f=g?{...o,baseURL:g}:o;return{suite:{beforeAll:s,afterAll:a,beforeEach:c,afterEach:l,tests:u},name:t,tags:r,use:f,referencedTemplatePaths:p}}function Ir(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 fe(e,t,r){for(let o=0;o<e.length;o++){let i=e[o],n=`${r}.${o}`,s=i.description||"";if(i.uid=Bi(t,n,s),i.type===J.STEP)fe(i.statements,t,n);else if(i.type===J.IF_ELSE){let a=i;fe(a.then,t,`${n}.then`),a.else&&fe(a.else,t,`${n}.else`)}else i.type===J.WHILE_LOOP&&fe(i.body,t,`${n}.body`)}}function Bi(e,t,r){let o=Mi("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${o.slice(0,8)}-${o.slice(8,12)}-${o.slice(12,16)}-${o.slice(16,20)}-${o.slice(20,32)}`}function Or(e,t){let r;try{r=Gi(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return Ki(r,e,t)}function Ki(e,t,r){let o=/\btemplate:\s/.test(e),i=/^suite:/m.test(e),n=o||i?null:et(e);if(n&&!n.valid)return{valid:!1,errors:n.errors,warnings:[],stats:n.stats};let s,a,c=[];try{let l=r?.parsed??at(e,t);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore},u=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?s=Ei(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):s=Ai(l.testFlow,{...p,testName:l.name,tags:l.tags,use:u,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 g=s.split(`
80
94
  `).filter(f=>!f.startsWith("import ")).join(`
81
- `);new Function(g),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),ji(Hi(a),{recursive:!0}),Wi(a,s)}catch(l){let p=l instanceof st?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:n?.warnings??[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var it,gi,mi,O,Pi,xr,st,ct=_(()=>{"use strict";ne();ne();ne();ne();it=5e3;gi=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],mi=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];O=new Map;S("click",e=>{let t=Oe(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",O.get("click"));S("click_element_by_index",O.get("click"));S("double_click",e=>le("double_click",e));S("double_click_on_element",O.get("double_click"));S("right_click",e=>le("right_click",e));S("right_click_on_element",O.get("right_click"));S("hover",e=>le("hover",e));S("hover_element_by_index",O.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return le("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",O.get("input_text"));S("clear_input",e=>le("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",O.get("press"));S("send_keys_on_element",e=>{let t=Oe(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let o=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return le("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",O.get("scroll"));S("scroll_up",O.get("scroll"));S("scroll_element",O.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>le("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",O.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",i=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&i){let s=r.code.split(`
95
+ `);new Function(g),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Hi(Wi(a),{recursive:!0}),ji(a,s)}catch(l){let p=l instanceof st?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:n?.warnings??[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var it,gi,mi,O,Pi,xr,st,ct=_(()=>{"use strict";se();se();se();se();it=5e3;gi=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],mi=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];O=new Map;S("click",e=>{let t=Oe(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",O.get("click"));S("click_element_by_index",O.get("click"));S("double_click",e=>le("double_click",e));S("double_click_on_element",O.get("double_click"));S("right_click",e=>le("right_click",e));S("right_click_on_element",O.get("right_click"));S("hover",e=>le("hover",e));S("hover_element_by_index",O.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return le("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",O.get("input_text"));S("clear_input",e=>le("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",O.get("press"));S("send_keys_on_element",e=>{let t=Oe(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let o=e.action_data?.kwargs?.timeout_ms??it;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return le("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",O.get("scroll"));S("scroll_up",O.get("scroll"));S("scroll_element",O.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>le("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",O.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",i=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&i){let s=r.code.split(`
82
96
  `),a=JSON.stringify(i);return["{ const _t = Date.now(); try {",...s.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${a}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${a}\`);`,` await agent.assert(page, ${a}, ${JSON.stringify(t||"")});`,"} }"]}return o?r.code.split(`
83
97
  `):i?[`await agent.assert(page, ${JSON.stringify(i)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",O.get("verify"));S("assert",O.get("verify"));S("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let o=JSON.stringify(r),i=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${o}, '${t||""}', ${i});`]});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,o=e.action_data?.kwargs?.variable_name;if(!r||!o)return["// Skipping ai_extract: missing element_description or variable_name"];let i=JSON.stringify(r),n=JSON.stringify(o);return[`await agent.extract(page, ${i}, ${n}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,o=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${o}, '${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=["{"],o=t.split(`
84
98
  `);for(let i of o)r.push(` ${i}`);return r.push("}"),r});S("function",(e,t,r)=>{let o=e.action_data?.kwargs||{},i=o.functionName;if(i&&i.includes("#")){let[s,a]=i.split("#");if(s&&a){let c=s.replace(/\.(ts|js|mjs)$/,""),l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let p={...o,functionName:a},u=Sr(p);return u?[u.endsWith(";")?u:`${u};`]:["// Skipping function: invalid export pattern"]}}let n=Sr(o);return n?[n.endsWith(";")?n:`${n};`]:["// 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=[],o={};return t.paths?o.paths=t.paths:t.path&&(o.path=t.path),t.use_file_input&&(o.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(o)} }`),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(i=>` ${i},`),"});"]});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(o=>` ${o},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
85
- `):["// Skipping js_action: missing code"]});Pi=["testContext","request"];xr=5;st=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as zi}from"express";import*as G from"fs/promises";import*as W from"path";import{stringify as Vi}from"yaml";function Re(e){if(!e||e.length===0)return[];let r=Vi({goal:"_hook",statements:e});return L(r).statements??[]}async function Lr(e){try{let t=await G.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 Lr(W.join(e,r.name))))return!0}catch{}return!1}function Cr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:i}=e,n=zi();n.get("/api/files",async(a,c)=>{try{let l=typeof a.query.dir=="string"?a.query.dir:t,p=W.resolve(l),u=await G.readdir(p,{withFileTypes:!0}),g=[];for(let d of u)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let h=W.join(p,d.name);await Lr(h)&&g.push({name:d.name,type:"directory",path:h})}else d.isFile()&&d.name.endsWith(".test.yaml")&&g.push({name:d.name,type:"file",path:W.join(p,d.name)});g.sort((d,h)=>d.type!==h.type?d.type==="directory"?-1:1:d.name.localeCompare(h.name));let f=W.dirname(p);c.json({dir:p,parent:f!==p?f:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(l){console.error("[debugger] Error listing files:",l),c.status(500).json({error:l.message})}});function s(a){if(typeof a=="string"&&a){let c=W.resolve(a);return c.endsWith(".test.yaml")?{filePath:c}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return n.get("/api/test-flow",async(a,c)=>{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath;try{i?.(p);let u=await G.readFile(p,"utf-8"),g=await G.stat(p),f=be(u),d=at(u,p);if(d.suite){let h=d.suite,w={tests:h.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:Re(h.beforeAll),afterAll:Re(h.afterAll),beforeEach:Re(h.beforeEach),afterEach:Re(h.afterEach)},b={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:w};c.json({isSuite:!0,testFlow:b,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:W.basename(p),lastModified:g.mtimeMs})}else{let h=d.testFlow;Ae(u,h),c.json({isSuite:!1,testFlow:h,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:W.basename(p),lastModified:g.mtimeMs})}}catch(u){if(u.code==="ENOENT")return c.status(404).json({error:`File not found: ${p}`});console.error("[debugger] Error loading test flow:",u),c.status(500).json({error:u.message})}}),n.put("/api/test-flow",async(a,c)=>{try{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath,{testFlow:u,metadata:g}=a.body;if(!u)return c.status(400).json({error:"testFlow is required"});let f=ye(u,g),d=p+".tmp";await G.writeFile(d,f,"utf-8"),await G.rename(d,p);let h=await G.stat(p);c.json({success:!0,lastModified:h.mtimeMs})}catch(l){console.error("[debugger] Error saving test flow:",l),c.status(500).json({error:l.message})}}),n}var Rr=_(()=>{"use strict";ne();ct()});import*as Nr from"http";function Dr(e){let{getPort:t,ensureReady:r}=e,o=null,i=0,n=5e3,s=async(a,c,l)=>{let p=t();if(p===null){if(!o&&Date.now()-i<n)return c.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});o||(o=r().catch(f=>{throw i=Date.now(),o=null,f}));try{p=await o}catch(f){return c.status(503).json({status:"error",message:"Failed to start sandbox: "+f.message})}}let u=await new Promise((f,d)=>{let h=[];a.on("data",w=>h.push(w)),a.on("end",()=>f(Buffer.concat(h))),a.on("error",d),a.readableEnded&&f(Buffer.alloc(0))}),g=Nr.request({hostname:"localhost",port:p,path:a.originalUrl,method:a.method,headers:{"content-type":a.headers["content-type"]||"application/json","content-length":String(u.length)},timeout:3e5},f=>{c.writeHead(f.statusCode,f.headers),f.on("data",d=>{c.write(d)}),f.on("end",()=>{c.end()}),f.on("error",()=>{c.writableEnded||c.end()})});g.on("error",f=>{c.headersSent||c.status(502).json({status:"error",message:"Inner server unavailable: "+f.message})}),g.end(u)};return s.reset=()=>{o=null},s.getPort=()=>t(),s.ensureReady=async()=>{let a=t();return a!==null?a:(o||(o=r().catch(c=>{throw i=Date.now(),o=null,c})),o)},s}var Fr=_(()=>{"use strict"});var Ur={};Q(Ur,{startDebuggerServer:()=>Yi});import Ne from"express";import*as ae from"path";async function Yi(e){let{initialDir:t,initialFile:r,projectRoot:o,port:i}=e,n=r,s=h=>{n=h,e.onFileSelected?.(h)},a=null,c=null,l=Dr({getPort:()=>a,ensureReady:async()=>{console.error("[debugger] Initializing sandbox on first use...");let h=await e.innerServerFactory();return a=h.port,c=h.cleanup,h.port}}),p=Ne();p.use((h,w,b)=>{if(w.setHeader("Access-Control-Allow-Origin","*"),w.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),w.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),h.method==="OPTIONS")return w.sendStatus(204);b()}),p.post("/api/int-runner/create-session",Ne.json(),async(h,w)=>{n&&(h.body.testFilePath=n);try{let b=l.getPort();b||(b=await l.ensureReady());let v=await fetch(`http://localhost:${b}/api/int-runner/create-session`,{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify(h.body)}),y=await v.json();w.status(v.status).json(y)}catch(b){w.status(502).json({status:"error",message:b.message})}}),p.post("/api/int-runner/terminate-session",async(h,w)=>{let b=c;a=null,c=null,l.reset(),b&&(console.error("[debugger] Tearing down inner Playwright process for fresh restart..."),await b().catch(v=>console.error("[debugger] Cleanup error:",v.message))),w.json({status:"success",details:"Session terminated"})}),p.use("/api/int-runner",l),p.use("/api/browser-cdp",l),p.use(Ne.json({limit:"10mb"})),p.use(Cr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:s}));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=u.includes(ae.sep+"src"+ae.sep)?ae.resolve(u,"../../dist/static"):ae.join(u,"static");p.use(Ne.static(g)),p.get("*path",(h,w,b)=>{if(h.path.startsWith("/api/"))return b();w.sendFile(ae.join(g,"index.html"),v=>{v&&w.send(Ji(t,i))})});let f=await new Promise((h,w)=>{let b=p.listen(i,"localhost",()=>{h(b)});b.on("error",w)}),d=`http://localhost:${i}`;return console.error(`[debugger] Server running at ${d}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:d,close:async()=>{c&&await c(),await new Promise((h,w)=>{f.close(b=>b?w(b):h())})}}}function Ji(e,t){return`<!DOCTYPE html>
99
+ `):["// Skipping js_action: missing code"]});Pi=["testContext","request"];xr=5;st=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as zi}from"express";import*as j from"fs/promises";import*as H from"path";import{stringify as Vi}from"yaml";function Re(e){if(!e||e.length===0)return[];let r=Vi({goal:"_hook",statements:e});return L(r).statements??[]}async function Lr(e){try{let t=await j.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 Lr(H.join(e,r.name))))return!0}catch{}return!1}function Cr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:i}=e,n=zi();n.get("/api/files",async(a,c)=>{try{let l=typeof a.query.dir=="string"?a.query.dir:t,p=H.resolve(l),u=await j.readdir(p,{withFileTypes:!0}),g=[];for(let d of u)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let h=H.join(p,d.name);await Lr(h)&&g.push({name:d.name,type:"directory",path:h})}else d.isFile()&&d.name.endsWith(".test.yaml")&&g.push({name:d.name,type:"file",path:H.join(p,d.name)});g.sort((d,h)=>d.type!==h.type?d.type==="directory"?-1:1:d.name.localeCompare(h.name));let f=H.dirname(p);c.json({dir:p,parent:f!==p?f:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(l){console.error("[debugger] Error listing files:",l),c.status(500).json({error:l.message})}});function s(a){if(typeof a=="string"&&a){let c=H.resolve(a);return c.endsWith(".test.yaml")?{filePath:c}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return n.get("/api/test-flow",async(a,c)=>{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath;try{i?.(p);let u=await j.readFile(p,"utf-8"),g=await j.stat(p),f=we(u),d=at(u,p);if(d.suite){let h=d.suite,b={tests:h.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:Re(h.beforeAll),afterAll:Re(h.afterAll),beforeEach:Re(h.beforeEach),afterEach:Re(h.afterEach)},w={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:b};c.json({isSuite:!0,testFlow:w,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:H.basename(p),lastModified:g.mtimeMs})}else{let h=d.testFlow;Ae(u,h),c.json({isSuite:!1,testFlow:h,metadata:f,name:d.name,tags:d.tags,use:d.use,filePath:p,fileName:H.basename(p),lastModified:g.mtimeMs})}}catch(u){if(u.code==="ENOENT")return c.status(404).json({error:`File not found: ${p}`});console.error("[debugger] Error loading test flow:",u),c.status(500).json({error:u.message})}}),n.put("/api/test-flow",async(a,c)=>{try{let l=s(a.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath,{testFlow:u,metadata:g}=a.body;if(!u)return c.status(400).json({error:"testFlow is required"});let f=ye(u,g),d=p+".tmp";await j.writeFile(d,f,"utf-8"),await j.rename(d,p);let h=await j.stat(p);c.json({success:!0,lastModified:h.mtimeMs})}catch(l){console.error("[debugger] Error saving test flow:",l),c.status(500).json({error:l.message})}}),n}var Rr=_(()=>{"use strict";se();ct()});import*as Nr from"http";function Dr(e){let{getPort:t,ensureReady:r}=e,o=null,i=0,n=5e3,s=async(a,c,l)=>{let p=t();if(p===null){if(!o&&Date.now()-i<n)return c.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});o||(o=r().catch(f=>{throw i=Date.now(),o=null,f}));try{p=await o}catch(f){return c.status(503).json({status:"error",message:"Failed to start sandbox: "+f.message})}}let u=await new Promise((f,d)=>{let h=[];a.on("data",b=>h.push(b)),a.on("end",()=>f(Buffer.concat(h))),a.on("error",d),a.readableEnded&&f(Buffer.alloc(0))}),g=Nr.request({hostname:"localhost",port:p,path:a.originalUrl,method:a.method,headers:{"content-type":a.headers["content-type"]||"application/json","content-length":String(u.length)},timeout:3e5},f=>{c.writeHead(f.statusCode,f.headers),f.on("data",d=>{c.write(d)}),f.on("end",()=>{c.end()}),f.on("error",()=>{c.writableEnded||c.end()})});g.on("error",f=>{c.headersSent||c.status(502).json({status:"error",message:"Inner server unavailable: "+f.message})}),g.end(u)};return s.reset=()=>{o=null},s.getPort=()=>t(),s.ensureReady=async()=>{let a=t();return a!==null?a:(o||(o=r().catch(c=>{throw i=Date.now(),o=null,c})),o)},s}var Fr=_(()=>{"use strict"});var Ur={};te(Ur,{startDebuggerServer:()=>Yi});import Ne from"express";import*as ce from"path";async function Yi(e){let{initialDir:t,initialFile:r,projectRoot:o,port:i}=e,n=r,s=h=>{n=h,e.onFileSelected?.(h)},a=null,c=null,l=Dr({getPort:()=>a,ensureReady:async()=>{console.error("[debugger] Initializing sandbox on first use...");let h=await e.innerServerFactory();return a=h.port,c=h.cleanup,h.port}}),p=Ne();p.use((h,b,w)=>{if(b.setHeader("Access-Control-Allow-Origin","*"),b.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),b.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),h.method==="OPTIONS")return b.sendStatus(204);w()}),p.post("/api/int-runner/create-session",Ne.json(),async(h,b)=>{n&&(h.body.testFilePath=n);try{let w=l.getPort();w||(w=await l.ensureReady());let v=await fetch(`http://localhost:${w}/api/int-runner/create-session`,{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify(h.body)}),y=await v.json();b.status(v.status).json(y)}catch(w){b.status(502).json({status:"error",message:w.message})}}),p.post("/api/int-runner/terminate-session",async(h,b)=>{let w=c;a=null,c=null,l.reset(),w&&(console.error("[debugger] Tearing down inner Playwright process for fresh restart..."),await w().catch(v=>console.error("[debugger] Cleanup error:",v.message))),b.json({status:"success",details:"Session terminated"})}),p.use("/api/int-runner",l),p.use("/api/browser-cdp",l),p.use(Ne.json({limit:"10mb"})),p.use(Cr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:s}));let u=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=u.includes(ce.sep+"src"+ce.sep)?ce.resolve(u,"../../dist/static"):ce.join(u,"static");p.use(Ne.static(g)),p.get("*path",(h,b,w)=>{if(h.path.startsWith("/api/"))return w();b.sendFile(ce.join(g,"index.html"),v=>{v&&b.send(Ji(t,i))})});let f=await new Promise((h,b)=>{let w=p.listen(i,"localhost",()=>{h(w)});w.on("error",b)}),d=`http://localhost:${i}`;return console.error(`[debugger] Server running at ${d}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:d,close:async()=>{c&&await c(),await new Promise((h,b)=>{f.close(w=>w?b(w):h())})}}}function Ji(e,t){return`<!DOCTYPE html>
86
100
  <html>
87
101
  <head>
88
102
  <meta charset="utf-8">
@@ -100,14 +114,14 @@ test('__shiplight_debug__', async ({ page, agent }) => {
100
114
  <p>Directory: <code>${e}</code></p>
101
115
  <p>Server: localhost:${t}</p>
102
116
  </body>
103
- </html>`}var Br=_(()=>{"use strict";Rr();Fr()});var Gr={};Q(Gr,{startDebugger:()=>qi});import*as Y from"fs";import*as oe from"path";async function qi(e){let t,r=6174,o,i=!1,n=!0;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),y++):e[y]==="--url"&&e[y+1]?(o=e[y+1],y++):e[y]==="--new"?i=!0:e[y]==="--open"?n=!1:e[y]==="--no-open"?n=!0:e[y]==="--help"||e[y]==="-h"?(Zi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let s,a;if(t){let y=oe.resolve(t);Y.existsSync(y)&&Y.statSync(y).isDirectory()?s=y:(s=oe.dirname(y),a=y)}else s=process.cwd();if(i&&a&&!Y.existsSync(a)){let y=oe.dirname(a);Y.existsSync(y)||Y.mkdirSync(y,{recursive:!0}),Y.writeFileSync(a,Xi(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${a}`)}a&&!Y.existsSync(a)&&(console.error(`Error: File not found: ${a}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(Ye(),Ve)),l=c(s),p=l?oe.dirname(l):s;(await import("dotenv")).config({path:oe.join(p,".env"),override:!0});let g=a;l||(console.error("Error: No Playwright config found in "+s),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let f=async()=>{let{spawnPlaywrightProcess:y}=await Promise.resolve().then(()=>(Ye(),Ve));console.error("[debugger] Starting Playwright sandbox...");let A=g||oe.join(s,"_debug.test.yaml");return y({yamlFilePath:A,configPath:l})},d=y=>{g=y},{startDebuggerServer:h}=await Promise.resolve().then(()=>(Br(),Ur));console.log(a?`Starting debugger for: ${a}`:`Starting debugger in: ${s}`),l&&console.log(`Using Playwright config: ${l}`);let w;try{w=await h({initialDir:s,initialFile:a,projectRoot:p,port:r,innerServerFactory:f,onFileSelected:d})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),y}if(Lt(r,s),console.log(`Debugger running at: ${w.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!n)try{let{default:y}=await import("open");await y(w.url)}catch{}let b=!1,v=async()=>{b&&(console.log(`
104
- Force exiting...`),process.exit(1)),b=!0,console.log(`
105
- Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Ct(r,s),await w.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function Zi(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var Xi,Wr=_(()=>{"use strict";Rt();Xi=e=>`goal: New test
117
+ </html>`}var Br=_(()=>{"use strict";Rr();Fr()});var Gr={};te(Gr,{startDebugger:()=>qi});import*as X from"fs";import*as ie from"path";async function qi(e){let t,r=6174,o,i=!1,n=!0;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),y++):e[y]==="--url"&&e[y+1]?(o=e[y+1],y++):e[y]==="--new"?i=!0:e[y]==="--open"?n=!1:e[y]==="--no-open"?n=!0:e[y]==="--help"||e[y]==="-h"?(Zi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let s,a;if(t){let y=ie.resolve(t);X.existsSync(y)&&X.statSync(y).isDirectory()?s=y:(s=ie.dirname(y),a=y)}else s=process.cwd();if(i&&a&&!X.existsSync(a)){let y=ie.dirname(a);X.existsSync(y)||X.mkdirSync(y,{recursive:!0}),X.writeFileSync(a,Xi(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${a}`)}a&&!X.existsSync(a)&&(console.error(`Error: File not found: ${a}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(Ye(),Ve)),l=c(s),p=l?ie.dirname(l):s;(await import("dotenv")).config({path:ie.join(p,".env"),override:!0});let g=a;l||(console.error("Error: No Playwright config found in "+s),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let f=async()=>{let{spawnPlaywrightProcess:y}=await Promise.resolve().then(()=>(Ye(),Ve));console.error("[debugger] Starting Playwright sandbox...");let A=g||ie.join(s,"_debug.test.yaml");return y({yamlFilePath:A,configPath:l})},d=y=>{g=y},{startDebuggerServer:h}=await Promise.resolve().then(()=>(Br(),Ur));console.log(a?`Starting debugger for: ${a}`:`Starting debugger in: ${s}`),l&&console.log(`Using Playwright config: ${l}`);let b;try{b=await h({initialDir:s,initialFile:a,projectRoot:p,port:r,innerServerFactory:f,onFileSelected:d})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),y}if(Lt(r,s),console.log(`Debugger running at: ${b.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!n)try{let{default:y}=await import("open");await y(b.url)}catch{}let w=!1,v=async()=>{w&&(console.log(`
118
+ Force exiting...`),process.exit(1)),w=!0,console.log(`
119
+ Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Ct(r,s),await b.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function Zi(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var Xi,jr=_(()=>{"use strict";Rt();Xi=e=>`goal: New test
106
120
  statements:
107
121
  - URL: ${e}
108
122
  - "TODO: Add your first step"
109
- `});var lt={};Q(lt,{lookupActionStores:()=>Qi,updateActionStores:()=>en});function jr(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function Qi(e){let t=jr();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),i=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(o),!i.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${i.status}`),new Map;let n=await i.json(),s=new Map;for(let[a,c]of Object.entries(n.stores??{}))s.set(a,c);return s}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function en(e){let t=jr();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),i={};for(let[a,c]of e)i[a]=c;let n=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:i}),signal:r.signal});return clearTimeout(o),n.ok?(await n.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${n.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var pt=_(()=>{"use strict"});import*as N from"fs";import*as Se from"path";import{globSync as tn}from"glob";function Hr(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new dt:new ut(e)}function ve(e){return e.replace(/\//g,"__")+".json"}function on(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var rn,ut,dt,Kr=_(()=>{"use strict";ne();rn=".shiplight/action-cache";ut=class{constructor(t){this.cwd=t;this.cacheDir=Se.join(t,rn)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!N.existsSync(this.cacheDir))return r;for(let o of t){let i=Se.join(this.cacheDir,ve(o));try{if(N.existsSync(i)){let n=N.readFileSync(i,"utf-8");r.set(o,JSON.parse(n))}}catch{}}return r}async update(t){if(t.size===0)return 0;N.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,i]of t)try{let n=Se.join(this.cacheDir,ve(o)),s=tt();if(N.existsSync(n))try{s=JSON.parse(N.readFileSync(n,"utf-8"))}catch{}let a={...s,entries:{...s.entries,...i.entries}};N.writeFileSync(n,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!N.existsSync(this.cacheDir))return;let t=tn("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let i of t)try{let n=N.readFileSync(Se.join(this.cacheDir,i),"utf-8"),s=JSON.parse(n),a=on(i);r.set(a,s),o+=Object.keys(s.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${o} cached action entit${o===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},dt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}loadAll(){}}});var Jr={};Q(Jr,{buildTestPathsFromGitInfo:()=>Yr,cloudKeyToCwdRelPath:()=>gt,runTests:()=>sn});import{spawn as nn,execFileSync as zr}from"child_process";import*as D from"fs";import*as C from"path";import{globSync as ht}from"glob";async function sn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(u=>D.existsSync(C.join(t,u)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
110
- `));let i=e.includes("--magic"),s=e.filter(u=>u!=="--magic").map(u=>u.endsWith(".test.yaml")?u.replace(/\.test\.yaml$/,".yaml.spec.ts"):u),a=Hr(t);await cn(t,a);let c={...process.env};i&&(c.SHIPLIGHT_MAGIC="1");let l=nn("npx",["playwright","test",...s],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(u=>{l.on("close",g=>u(g??1))});await ln(t,a),process.exit(p)}function ft(){try{return zr("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function an(){try{let e=zr("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Vr(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=an(),i=ft();return Yr(e,t,o,i)}function Yr(e,t,r,o){return{testPaths:e.map(n=>{let s=o?C.relative(o,C.resolve(t,n)):n;return`${r}${s}`}),branchPrefix:r}}function gt(e,t,r,o){let i=t&&e.startsWith(t)?e.slice(t.length):e;return r?C.relative(o,C.resolve(r,i)):i}async function cn(e,t){try{let r=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:i}=Vr(r,e,t.isCloud),n=await t.lookup(o);if(n.size===0)return;let s=C.join(e,".shiplight","action-cache");D.mkdirSync(s,{recursive:!0});let a=ft(),c=0;for(let[l,p]of n){let u=t.isCloud?gt(l,i,a,e):l,g=C.join(s,ve(u));D.writeFileSync(g,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function ln(e,t){try{let r=C.join(e,"test-results");if(!D.existsSync(r))return;let o=ht("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let i={};for(let u of o)try{let g=D.readFileSync(C.join(r,u),"utf-8"),f=JSON.parse(g);f?.entries&&Object.assign(i,f.entries)}catch{}if(Object.keys(i).length===0)return;let n=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:s,branchPrefix:a}=Vr(n,e,t.isCloud),c=new Map;for(let u=0;u<n.length;u++){let g=n[u],f=s[u],d=C.join(e,g.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!D.existsSync(d))continue;let h=D.readFileSync(d,"utf-8"),w={};for(let[b,v]of Object.entries(i))h.includes(b)&&(w[b]=v);if(Object.keys(w).length>0){let b=t.isCloud?gt(f,a,ft(),e):f,v=C.join(e,".shiplight","action-cache",ve(b)),y={};if(D.existsSync(v))try{y=JSON.parse(D.readFileSync(v,"utf-8")).entries}catch{}c.set(f,{version:"1.0",entries:{...y,...w}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((u,g)=>u+Object.keys(g.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Xr=_(()=>{"use strict";Kr()});function P(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Fe(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function qr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function De(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function pn(e){return`<span class="badge badge-${e}">${e}</span>`}function un(e){if(!e.code)return"";let t=e.code.split(`
123
+ `});var lt={};te(lt,{lookupActionStores:()=>Qi,updateActionStores:()=>en});function Hr(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function Qi(e){let t=Hr();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),i=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(o),!i.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${i.status}`),new Map;let n=await i.json(),s=new Map;for(let[a,c]of Object.entries(n.stores??{}))s.set(a,c);return s}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function en(e){let t=Hr();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),i={};for(let[a,c]of e)i[a]=c;let n=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:i}),signal:r.signal});return clearTimeout(o),n.ok?(await n.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${n.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var pt=_(()=>{"use strict"});import*as D from"fs";import*as Se from"path";import{globSync as tn}from"glob";function Wr(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new dt:new ut(e)}function ve(e){return e.replace(/\//g,"__")+".json"}function on(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var rn,ut,dt,Kr=_(()=>{"use strict";se();rn=".shiplight/action-cache";ut=class{constructor(t){this.cwd=t;this.cacheDir=Se.join(t,rn)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!D.existsSync(this.cacheDir))return r;for(let o of t){let i=Se.join(this.cacheDir,ve(o));try{if(D.existsSync(i)){let n=D.readFileSync(i,"utf-8");r.set(o,JSON.parse(n))}}catch{}}return r}async update(t){if(t.size===0)return 0;D.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,i]of t)try{let n=Se.join(this.cacheDir,ve(o)),s=tt();if(D.existsSync(n))try{s=JSON.parse(D.readFileSync(n,"utf-8"))}catch{}let a={...s,entries:{...s.entries,...i.entries}};D.writeFileSync(n,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!D.existsSync(this.cacheDir))return;let t=tn("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let i of t)try{let n=D.readFileSync(Se.join(this.cacheDir,i),"utf-8"),s=JSON.parse(n),a=on(i);r.set(a,s),o+=Object.keys(s.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${o} cached action entit${o===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},dt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(pt(),lt));return r(t)}loadAll(){}}});var Jr={};te(Jr,{buildTestPathsFromGitInfo:()=>Yr,cloudKeyToCwdRelPath:()=>gt,runTests:()=>sn});import{spawn as nn,execFileSync as zr}from"child_process";import*as F from"fs";import*as C from"path";import{globSync as ht}from"glob";async function sn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(u=>F.existsSync(C.join(t,u)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
124
+ `));let i=e.includes("--magic"),s=e.filter(u=>u!=="--magic").map(u=>u.endsWith(".test.yaml")?u.replace(/\.test\.yaml$/,".yaml.spec.ts"):u),a=Wr(t);await cn(t,a);let c={...process.env};i&&(c.SHIPLIGHT_MAGIC="1");let l=nn("npx",["playwright","test",...s],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(u=>{l.on("close",g=>u(g??1))});await ln(t,a),process.exit(p)}function ft(){try{return zr("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function an(){try{let e=zr("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Vr(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=an(),i=ft();return Yr(e,t,o,i)}function Yr(e,t,r,o){return{testPaths:e.map(n=>{let s=o?C.relative(o,C.resolve(t,n)):n;return`${r}${s}`}),branchPrefix:r}}function gt(e,t,r,o){let i=t&&e.startsWith(t)?e.slice(t.length):e;return r?C.relative(o,C.resolve(r,i)):i}async function cn(e,t){try{let r=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:i}=Vr(r,e,t.isCloud),n=await t.lookup(o);if(n.size===0)return;let s=C.join(e,".shiplight","action-cache");F.mkdirSync(s,{recursive:!0});let a=ft(),c=0;for(let[l,p]of n){let u=t.isCloud?gt(l,i,a,e):l,g=C.join(s,ve(u));F.writeFileSync(g,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function ln(e,t){try{let r=C.join(e,"test-results");if(!F.existsSync(r))return;let o=ht("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let i={};for(let u of o)try{let g=F.readFileSync(C.join(r,u),"utf-8"),f=JSON.parse(g);f?.entries&&Object.assign(i,f.entries)}catch{}if(Object.keys(i).length===0)return;let n=ht("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:s,branchPrefix:a}=Vr(n,e,t.isCloud),c=new Map;for(let u=0;u<n.length;u++){let g=n[u],f=s[u],d=C.join(e,g.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!F.existsSync(d))continue;let h=F.readFileSync(d,"utf-8"),b={};for(let[w,v]of Object.entries(i))h.includes(w)&&(b[w]=v);if(Object.keys(b).length>0){let w=t.isCloud?gt(f,a,ft(),e):f,v=C.join(e,".shiplight","action-cache",ve(w)),y={};if(F.existsSync(v))try{y=JSON.parse(F.readFileSync(v,"utf-8")).entries}catch{}c.set(f,{version:"1.0",entries:{...y,...b}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((u,g)=>u+Object.keys(g.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Xr=_(()=>{"use strict";Kr()});function P(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Fe(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function qr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function De(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function pn(e){return`<span class="badge badge-${e}">${e}</span>`}function un(e){if(!e.code)return"";let t=e.code.split(`
111
125
  `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((i,n)=>{let s=e.codeStartLine+n,a=s===e.codeLine,c=String(s).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${P(i)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(o=>`<span class="code-line code-line-body">${P(o)}</span>`).join("")}</pre></div>`}function dn(e){let t=e.duration!=null?`<span class="step-duration">${Fe(e.duration)}</span>`:"",r=De(e.status);if(e.screenshot||e.message||e.error||e.code){let i="";e.screenshot&&(i=`<img src="${P(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n=e.screenshot?"":un(e),s="";e.error&&(s=`<div class="step-error"><pre>${P(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${P(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
112
126
  <details class="step-details step step-${e.status}"${c}>
113
127
  <summary class="step-header">
@@ -978,8 +992,8 @@ statements:
978
992
  });
979
993
  </script>
980
994
  </body>
981
- </html>`}var Qr=_(()=>{"use strict"});import*as J from"fs";import*as Z from"path";import{execFileSync as gn}from"child_process";import{createHash as mn}from"crypto";import ge from"axios";function eo(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function ie(...e){try{return gn("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function yn(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=J.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function bn(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",o=process.env.GITHUB_RUN_ID??"",i=process.env.GITHUB_EVENT_NAME??"",s=yn().pull_request,a=s?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],u=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,g=process.env.SHIPLIGHT_PR_TITLE??s?.title,f=ie("log","-1","--pretty=%s"),d=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:u,gitRepo:r,commitMessage:f,authorEmail:d,prNumber:p,prTitle:g,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:o,ciBuildUrl:o&&r?`${t}/${r}/actions/runs/${o}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:i,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",o=process.env.CI_MERGE_REQUEST_IID,i=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:i,prNumber:o,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:o&&t?`${t}/-/merge_requests/${o}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",o=process.env.CIRCLE_PROJECT_REPONAME??"",i=r&&o?`${r}/${o}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?eo(a):void 0,l=ie("log","-1","--pretty=%s"),p=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:i,commitMessage:l,authorEmail:p,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=ie("rev-parse","HEAD"),r=ie("rev-parse","--abbrev-ref","HEAD"),o=ie("log","-1","--pretty=%s"),i=ie("log","-1","--pretty=%ae"),n=ie("remote","get-url","origin"),s=n?eo(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:i,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function wn(e){switch(e){case"passed":return"Passed";case"failed":return"Failed";case"timedOut":return"TimedOut";case"skipped":return"Skipped";case"interrupted":return"Failed";default:return"Failed"}}function to(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Sn(e){return e.length===0||e.every(t=>t.status==="skipped")?"Skipped":e.some(t=>t.status==="failed"||t.status==="timedOut"||t.status==="interrupted")?"Failed":"Passed"}function vn(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let i=r.get(o.testCaseName);i?i.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Ue(e,t){return Z.isAbsolute(t)?t:Z.join(e,t)}function _n(e,t,r,o){let i={};for(let n of e.steps)i[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:to(e.status),flaky:!1,segments:[{outcome:to(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:i,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function wt(e){return mn("md5").update(e).digest("base64")}function yt(e){return wt(J.readFileSync(e))}async function bt(e,t){let r=J.readFileSync(t),o=Z.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await ge.put(e,r,{headers:{"Content-Type":n,"Content-MD5":wt(r)}})}async function ro(e,t,r,o){let i=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},s=bn(),a=e.tests.map(h=>{let w={testCaseName:h.title,testCaseBaseName:h.baseTitle,suiteName:h.suiteName,file:h.file,tags:h.tags,suiteTags:h.suiteTags,baseUrl:h.baseUrl,skip:h.skip,slow:h.slow,timeout:h.timeout,parameterSetName:h.parameterSetName,flaky:h.flaky,retries:h.retries};if(h.videoPath){let b=Ue(t,h.videoPath);J.existsSync(b)&&(w.videoMd5=yt(b))}if(h.tracePath){let b=Ue(t,h.tracePath);J.existsSync(b)&&(w.traceMd5=yt(b))}return w}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let p=(await ge.post(`${i}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:a},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let u=vn(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(h,w)=>{let b=u[w];if(!b)return;let v=h.steps.filter(k=>k.screenshot);if(!v.length)return;let y=v.map(k=>k.stepId),A={};for(let k of v){let $=Z.isAbsolute(k.screenshot)?k.screenshot:Z.join(t,k.screenshot);J.existsSync($)&&(A[k.stepId]=yt($))}try{let k=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:A},{headers:n});b.uploadUrls.screenshots=k.data.screenshots,b.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${h.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${h.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let g=(await Promise.all(e.tests.map(async(h,w)=>{let b=u[w];if(!b){console.warn(`[reporter] No result slot found for test "${h.title}", skipping.`);return}let v=b.uploadUrls,y={},A=0;await Promise.all(h.steps.map(async I=>{if(I.screenshot&&v.screenshots?.[I.stepId]){let pe=Z.isAbsolute(I.screenshot)?I.screenshot:Z.join(t,I.screenshot);if(J.existsSync(pe))try{await bt(v.screenshots[I.stepId],pe),y[I.stepId]=b.screenshotS3Uris[I.stepId],A++}catch(xo){console.warn(`[reporter] Screenshot upload failed for step ${I.stepId}:`,xo)}}})),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} screenshot(s) for "${h.title}"`);let k;if(h.videoPath&&v.video){let I=Ue(t,h.videoPath);if(J.existsSync(I)){console.log(`[reporter] [3/4] Uploading video for "${h.title}"...`);try{await bt(v.video,I),k=b.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Video upload failed:",pe)}}}let $;if(h.tracePath&&v.trace){let I=Ue(t,h.tracePath);if(J.existsSync(I)){console.log(`[reporter] [3/4] Uploading trace for "${h.title}"...`);try{await bt(v.trace,I),$=b.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Trace upload failed:",pe)}}}console.log(`[reporter] [3/4] Uploading report for "${h.title}"...`);let je=_n(h,y,k,$),vt=Buffer.from(JSON.stringify(je)),_t=wt(vt),xt=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/report-url`,{md5:_t},{headers:n}),vo=xt.data.reportUrl,_o=xt.data.reportS3Uri;return await ge.put(vo,vt,{headers:{"Content-Type":"application/json","Content-MD5":_t}}),console.log(`[reporter] [3/4] Report uploaded for "${h.title}"`),{testCaseResultId:b.testCaseResultId,result:wn(h.status),durationMs:h.duration,startTime:h.startTime,endTime:h.endTime,error:h.error,reportS3Uri:_o,videoS3Uri:k,traceS3Uri:$,metadata:{suiteName:h.suiteName,file:h.file}}}))).filter(h=>!!h);console.log("[reporter] [4/4] Finalising run...");let f=Sn(e.tests),d=await ge.put(`${i}/v1/local-runs/${p.testRunId}/complete`,{status:f,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:g},{headers:n});console.log(`
982
- Shiplight cloud report: ${d.data.reportUrl}`)}var oo=_(()=>{"use strict"});var co={};Q(co,{buildGitHubSummary:()=>ao,runReport:()=>xn});import*as T from"fs";import*as x from"path";async function xn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),o=e.includes("--github-summary");!r&&o&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await kn(e,t,o):await Tn(e,t)}async function Tn(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",o=x.isAbsolute(r)?r:x.join(process.cwd(),r),i=x.join(o,"report-data.json");T.existsSync(i)||(console.error(`Error: ${i} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let n;try{n=JSON.parse(T.readFileSync(i,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${i}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let s=x.join(o,"index.html");if(T.writeFileSync(s,mt(n),"utf-8"),console.log(`Shiplight report regenerated: ${s}`),await so(n,o),t)try{let a=(await import("open")).default;await a(s)}catch{}}async function kn(e,t,r){let o=x.join(process.cwd(),"shiplight-report"),i=e.findIndex(f=>f==="-o"||f==="--output");if(i!==-1&&e[i+1]){let f=e[i+1];o=x.isAbsolute(f)?f:x.join(process.cwd(),f)}let n=new Set(["-o","--output"]),s=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let f=0;f<e.length;f++){let d=e[f];if(n.has(d)){f++;continue}if(s.has(d))continue;let h=x.isAbsolute(d)?d:x.join(process.cwd(),d);a.push(h)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;T.mkdirSync(x.join(o,"screenshots"),{recursive:!0});for(let f=0;f<a.length;f++){let d=a[f],h=`shard-${f}`,w=x.join(d,"report-data.json");if(!T.existsSync(w)){console.warn(`Warning: No report-data.json found in ${d}, skipping.`);continue}let b;try{b=JSON.parse(T.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${h}: ${b.tests.length} tests from ${d}`),l+=b.totalDuration||0;let v=x.join(d,"screenshots");T.existsSync(v)&&no(v,x.join(o,"screenshots",h));let y=x.join(o,h);for(let A of b.tests){let k=[A,...A.attempts||[]];for(let $ of k)An($.steps,h),$.videoPath&&io(d,$.videoPath,y)&&($.videoPath=`${h}/${$.videoPath}`),$.tracePath&&io(d,$.tracePath,y)&&($.tracePath=`${h}/${$.tracePath}`);c.push(A)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let u={tests:c,totalDuration:l,timestamp:new Date().toISOString()};T.writeFileSync(x.join(o,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let g=x.join(o,"index.html");if(T.writeFileSync(g,mt(u),"utf-8"),console.log(`
995
+ </html>`}var Qr=_(()=>{"use strict"});import*as q from"fs";import*as ee from"path";import{execFileSync as gn}from"child_process";import{createHash as mn}from"crypto";import ge from"axios";function eo(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function ne(...e){try{return gn("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function yn(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=q.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function wn(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",r=process.env.GITHUB_REPOSITORY??"",o=process.env.GITHUB_RUN_ID??"",i=process.env.GITHUB_EVENT_NAME??"",s=yn().pull_request,a=s?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],u=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,g=process.env.SHIPLIGHT_PR_TITLE??s?.title,f=ne("log","-1","--pretty=%s"),d=ne("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:u,gitRepo:r,commitMessage:f,authorEmail:d,prNumber:p,prTitle:g,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:o,ciBuildUrl:o&&r?`${t}/${r}/actions/runs/${o}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:i,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",o=process.env.CI_MERGE_REQUEST_IID,i=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:i,prNumber:o,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:o&&t?`${t}/-/merge_requests/${o}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",o=process.env.CIRCLE_PROJECT_REPONAME??"",i=r&&o?`${r}/${o}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?eo(a):void 0,l=ne("log","-1","--pretty=%s"),p=ne("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:i,commitMessage:l,authorEmail:p,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=ne("rev-parse","HEAD"),r=ne("rev-parse","--abbrev-ref","HEAD"),o=ne("log","-1","--pretty=%s"),i=ne("log","-1","--pretty=%ae"),n=ne("remote","get-url","origin"),s=n?eo(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:i,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function bn(e){switch(e){case"passed":return"Passed";case"failed":return"Failed";case"timedOut":return"TimedOut";case"skipped":return"Skipped";case"interrupted":return"Failed";default:return"Failed"}}function to(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Sn(e){return e.length===0||e.every(t=>t.status==="skipped")?"Skipped":e.some(t=>t.status==="failed"||t.status==="timedOut"||t.status==="interrupted")?"Failed":"Passed"}function vn(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let i=r.get(o.testCaseName);i?i.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Ue(e,t){return ee.isAbsolute(t)?t:ee.join(e,t)}function _n(e,t,r,o){let i={};for(let n of e.steps)i[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:to(e.status),flaky:!1,segments:[{outcome:to(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:i,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function bt(e){return mn("md5").update(e).digest("base64")}function yt(e){return bt(q.readFileSync(e))}async function wt(e,t){let r=q.readFileSync(t),o=ee.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await ge.put(e,r,{headers:{"Content-Type":n,"Content-MD5":bt(r)}})}async function ro(e,t,r,o){let i=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},s=wn(),a=e.tests.map(h=>{let b={testCaseName:h.title,testCaseBaseName:h.baseTitle,suiteName:h.suiteName,file:h.file,tags:h.tags,suiteTags:h.suiteTags,baseUrl:h.baseUrl,skip:h.skip,slow:h.slow,timeout:h.timeout,parameterSetName:h.parameterSetName,flaky:h.flaky,retries:h.retries};if(h.videoPath){let w=Ue(t,h.videoPath);q.existsSync(w)&&(b.videoMd5=yt(w))}if(h.tracePath){let w=Ue(t,h.tracePath);q.existsSync(w)&&(b.traceMd5=yt(w))}return b}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let p=(await ge.post(`${i}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:a},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let u=vn(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(h,b)=>{let w=u[b];if(!w)return;let v=h.steps.filter(k=>k.screenshot);if(!v.length)return;let y=v.map(k=>k.stepId),A={};for(let k of v){let $=ee.isAbsolute(k.screenshot)?k.screenshot:ee.join(t,k.screenshot);q.existsSync($)&&(A[k.stepId]=yt($))}try{let k=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:A},{headers:n});w.uploadUrls.screenshots=k.data.screenshots,w.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${h.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${h.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let g=(await Promise.all(e.tests.map(async(h,b)=>{let w=u[b];if(!w){console.warn(`[reporter] No result slot found for test "${h.title}", skipping.`);return}let v=w.uploadUrls,y={},A=0;await Promise.all(h.steps.map(async I=>{if(I.screenshot&&v.screenshots?.[I.stepId]){let pe=ee.isAbsolute(I.screenshot)?I.screenshot:ee.join(t,I.screenshot);if(q.existsSync(pe))try{await wt(v.screenshots[I.stepId],pe),y[I.stepId]=w.screenshotS3Uris[I.stepId],A++}catch(xo){console.warn(`[reporter] Screenshot upload failed for step ${I.stepId}:`,xo)}}})),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} screenshot(s) for "${h.title}"`);let k;if(h.videoPath&&v.video){let I=Ue(t,h.videoPath);if(q.existsSync(I)){console.log(`[reporter] [3/4] Uploading video for "${h.title}"...`);try{await wt(v.video,I),k=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Video upload failed:",pe)}}}let $;if(h.tracePath&&v.trace){let I=Ue(t,h.tracePath);if(q.existsSync(I)){console.log(`[reporter] [3/4] Uploading trace for "${h.title}"...`);try{await wt(v.trace,I),$=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${h.title}"`)}catch(pe){console.warn("[reporter] Trace upload failed:",pe)}}}console.log(`[reporter] [3/4] Uploading report for "${h.title}"...`);let He=_n(h,y,k,$),vt=Buffer.from(JSON.stringify(He)),_t=bt(vt),xt=await ge.post(`${i}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:_t},{headers:n}),vo=xt.data.reportUrl,_o=xt.data.reportS3Uri;return await ge.put(vo,vt,{headers:{"Content-Type":"application/json","Content-MD5":_t}}),console.log(`[reporter] [3/4] Report uploaded for "${h.title}"`),{testCaseResultId:w.testCaseResultId,result:bn(h.status),durationMs:h.duration,startTime:h.startTime,endTime:h.endTime,error:h.error,reportS3Uri:_o,videoS3Uri:k,traceS3Uri:$,metadata:{suiteName:h.suiteName,file:h.file}}}))).filter(h=>!!h);console.log("[reporter] [4/4] Finalising run...");let f=Sn(e.tests),d=await ge.put(`${i}/v1/local-runs/${p.testRunId}/complete`,{status:f,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:g},{headers:n});console.log(`
996
+ Shiplight cloud report: ${d.data.reportUrl}`)}var oo=_(()=>{"use strict"});var co={};te(co,{buildGitHubSummary:()=>ao,runReport:()=>xn});import*as T from"fs";import*as x from"path";async function xn(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),o=e.includes("--github-summary");!r&&o&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await kn(e,t,o):await Tn(e,t)}async function Tn(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",o=x.isAbsolute(r)?r:x.join(process.cwd(),r),i=x.join(o,"report-data.json");T.existsSync(i)||(console.error(`Error: ${i} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let n;try{n=JSON.parse(T.readFileSync(i,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${i}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let s=x.join(o,"index.html");if(T.writeFileSync(s,mt(n),"utf-8"),console.log(`Shiplight report regenerated: ${s}`),await so(n,o),t)try{let a=(await import("open")).default;await a(s)}catch{}}async function kn(e,t,r){let o=x.join(process.cwd(),"shiplight-report"),i=e.findIndex(f=>f==="-o"||f==="--output");if(i!==-1&&e[i+1]){let f=e[i+1];o=x.isAbsolute(f)?f:x.join(process.cwd(),f)}let n=new Set(["-o","--output"]),s=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let f=0;f<e.length;f++){let d=e[f];if(n.has(d)){f++;continue}if(s.has(d))continue;let h=x.isAbsolute(d)?d:x.join(process.cwd(),d);a.push(h)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;T.mkdirSync(x.join(o,"screenshots"),{recursive:!0});for(let f=0;f<a.length;f++){let d=a[f],h=`shard-${f}`,b=x.join(d,"report-data.json");if(!T.existsSync(b)){console.warn(`Warning: No report-data.json found in ${d}, skipping.`);continue}let w;try{w=JSON.parse(T.readFileSync(b,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${b}, skipping.`);continue}console.log(`Merging ${h}: ${w.tests.length} tests from ${d}`),l+=w.totalDuration||0;let v=x.join(d,"screenshots");T.existsSync(v)&&no(v,x.join(o,"screenshots",h));let y=x.join(o,h);for(let A of w.tests){let k=[A,...A.attempts||[]];for(let $ of k)An($.steps,h),$.videoPath&&io(d,$.videoPath,y)&&($.videoPath=`${h}/${$.videoPath}`),$.tracePath&&io(d,$.tracePath,y)&&($.tracePath=`${h}/${$.tracePath}`);c.push(A)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let u={tests:c,totalDuration:l,timestamp:new Date().toISOString()};T.writeFileSync(x.join(o,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let g=x.join(o,"index.html");if(T.writeFileSync(g,mt(u),"utf-8"),console.log(`
983
997
  Merged ${c.length} tests from ${p} shards into: ${g}`),await so(u,o),r&&En(c),t)try{let f=(await import("open")).default;await f(g)}catch{}}function io(e,t,r){let o=x.resolve(e,t);return o.startsWith(x.resolve(e)+x.sep)?T.existsSync(o)?(T.mkdirSync(r,{recursive:!0}),T.copyFileSync(o,x.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function no(e,t){T.mkdirSync(t,{recursive:!0});for(let r of T.readdirSync(e,{withFileTypes:!0})){let o=x.join(e,r.name),i=x.join(t,r.name);r.isDirectory()?no(o,i):T.copyFileSync(o,i)}}function An(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function so(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN||process.env.__SHIPLIGHT_API_KEY;if(!r){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let o=e.tests.map(n=>n.startTime).filter(n=>!!n),i=o.length>0?o.sort()[0]:e.timestamp??new Date().toISOString();try{await ro(e,t,i,r)}catch(n){console.warn("[report] Cloud upload failed:",n)}}function St(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=x.join("tests",x.basename(t));return{name:e.title||x.basename(t),yamlPath:r}}function ao(e){let t=e.filter(a=>!a.file.includes("auth.setup")),r=t.filter(a=>a.flaky),o=t.filter(a=>a.status==="passed"&&!a.flaky),i=t.filter(a=>a.status!=="passed"),n=t.length,s=`## Test Results
984
998
 
985
999
  `;if(i.length===0&&r.length===0?s+=`\u2705 All ${n} tests passed
@@ -1005,9 +1019,9 @@ Merged ${c.length} tests from ${p} shards into: ${g}`),await so(u,o),r&&En(c),t)
1005
1019
  `;for(let c of a)s+=`- ${c}
1006
1020
  `;s+=`
1007
1021
  </details>
1008
- `}return s}function En(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}T.appendFileSync(t,ao(e)),console.log("GitHub step summary written.")}var lo=_(()=>{"use strict";Qr();oo()});var po,uo=_(()=>{"use strict";po="0.1.48"});var ho={};Q(ho,{runTranspile:()=>$n});import*as Be from"path";import{glob as Pn}from"glob";async function $n(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(),o=await Pn(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let i=0,n=0,s=0;for(let a of o.sort()){let c=Be.resolve(r,a),l=Or(c,{version:po});if(!l.valid){i++,console.log(`
1022
+ `}return s}function En(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}T.appendFileSync(t,ao(e)),console.log("GitHub step summary written.")}var lo=_(()=>{"use strict";Qr();oo()});var po,uo=_(()=>{"use strict";po="0.1.49"});var ho={};te(ho,{runTranspile:()=>$n});import*as Be from"path";import{glob as Pn}from"glob";async function $n(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(),o=await Pn(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let i=0,n=0,s=0;for(let a of o.sort()){let c=Be.resolve(r,a),l=Or(c,{version:po});if(!l.valid){i++,console.log(`
1009
1023
  \u2717 ${a}`);for(let u of l.errors)console.log(` ERROR: ${u}`);continue}s++;let p=Be.basename(l.specFile);if(l.warnings.length>0){n++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let u of l.warnings)console.log(` WARNING: ${u}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
1010
- ${o.length} file(s): ${s} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var fo=_(()=>{"use strict";ct();uo()});var yo={};Q(yo,{runInspect:()=>Mn});import*as Ge from"fs";import*as go from"path";async function Mn(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"),o=e.find(s=>!s.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let i=go.resolve(process.cwd(),o);Ge.existsSync(i)||(console.error(`Error: file not found: ${i}`),process.exit(1));let n=Ge.readFileSync(i,"utf-8");try{let s=be(n),a=L(n);if(r)In(a,s);else{let c={...s.test_case_id!==void 0?{test_case_id:s.test_case_id}:{},...s.name?{name:s.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(s){console.error(`Error parsing ${o}: ${s.message}`),process.exit(1)}}function In(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 o of r.tests){let i=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${i}`)}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=mo(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function mo(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 o=mo(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var bo=_(()=>{"use strict";ne()});var wo=ko((hl,On)=>{On.exports={name:"shiplightai",version:"0.1.48",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"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"}},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",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0",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.58.2","@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.58.2"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});import Ln from"dotenv";import*as M from"fs";import*as j from"path";import*as He from"os";import{execFileSync as Ao}from"child_process";var Eo="0.1.48",Po="https://registry.npmjs.org/shiplightai/latest",$o=3600*1e3,Mo=10080*60*1e3;function Ke(){return j.join(He.homedir(),".shiplight","version-check.json")}function Io(){return j.join(He.homedir(),".shiplight","npm-prefix.json")}function Oo(e=Io()){try{let r=M.readFileSync(e,"utf-8"),o=JSON.parse(r);if(typeof o.prefix=="string"&&typeof o.fetchedAt=="number"&&Date.now()-o.fetchedAt<Mo)return o.prefix}catch{}let t;try{t=Ao("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{M.mkdirSync(j.dirname(e),{recursive:!0}),M.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Tt(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Oo()),o=e.warn??(s=>console.warn(s)),i=e.error??(s=>console.error(s)),n=e.exit??(s=>process.exit(s));try{if(!t)return;let s=r();if(!s)return;let a,c;try{a=M.realpathSync(t),c=M.realpathSync(s)}catch{return}if(a.startsWith(c+j.sep)){i(`
1024
+ ${o.length} file(s): ${s} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var fo=_(()=>{"use strict";ct();uo()});var yo={};te(yo,{runInspect:()=>Mn});import*as Ge from"fs";import*as go from"path";async function Mn(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"),o=e.find(s=>!s.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let i=go.resolve(process.cwd(),o);Ge.existsSync(i)||(console.error(`Error: file not found: ${i}`),process.exit(1));let n=Ge.readFileSync(i,"utf-8");try{let s=we(n),a=L(n);if(r)In(a,s);else{let c={...s.test_case_id!==void 0?{test_case_id:s.test_case_id}:{},...s.name?{name:s.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(s){console.error(`Error parsing ${o}: ${s.message}`),process.exit(1)}}function In(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 o of r.tests){let i=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${i}`)}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=mo(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function mo(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 o=mo(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var wo=_(()=>{"use strict";se()});var bo=ko((hl,On)=>{On.exports={name:"shiplightai",version:"0.1.49",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"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"}},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",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0",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.58.2","@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.58.2"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});import Ln from"dotenv";import*as M from"fs";import*as W from"path";import*as We from"os";import{execFileSync as Ao}from"child_process";var Eo="0.1.49",Po="https://registry.npmjs.org/shiplightai/latest",$o=3600*1e3,Mo=10080*60*1e3;function Ke(){return W.join(We.homedir(),".shiplight","version-check.json")}function Io(){return W.join(We.homedir(),".shiplight","npm-prefix.json")}function Oo(e=Io()){try{let r=M.readFileSync(e,"utf-8"),o=JSON.parse(r);if(typeof o.prefix=="string"&&typeof o.fetchedAt=="number"&&Date.now()-o.fetchedAt<Mo)return o.prefix}catch{}let t;try{t=Ao("npm",["config","get","prefix"],{encoding:"utf-8",stdio:["ignore","pipe","ignore"]}).trim()}catch{return null}if(!t)return null;try{M.mkdirSync(W.dirname(e),{recursive:!0}),M.writeFileSync(e,JSON.stringify({prefix:t,fetchedAt:Date.now()}))}catch{}return t}function Tt(e={}){let t=e.scriptPath??process.argv[1],r=e.getPrefix??(()=>Oo()),o=e.warn??(s=>console.warn(s)),i=e.error??(s=>console.error(s)),n=e.exit??(s=>process.exit(s));try{if(!t)return;let s=r();if(!s)return;let a,c;try{a=M.realpathSync(t),c=M.realpathSync(s)}catch{return}if(a.startsWith(c+W.sep)){i(`
1011
1025
  shiplightai cannot be run from a global install.
1012
1026
  Global installs don't auto-update and cause version skew.
1013
1027
 
@@ -1016,11 +1030,11 @@ Install it as a project dependency instead:
1016
1030
  cd <your-project>
1017
1031
  npm i -D shiplightai
1018
1032
  npx shiplight <command>
1019
- `),n(1);return}let l=j.join(c,"lib","node_modules","shiplightai"),p=j.join(c,"node_modules","shiplightai"),u=M.existsSync(l)?l:M.existsSync(p)?p:null;u&&o(`
1033
+ `),n(1);return}let l=W.join(c,"lib","node_modules","shiplightai"),p=W.join(c,"node_modules","shiplightai"),u=M.existsSync(l)?l:M.existsSync(p)?p:null;u&&o(`
1020
1034
  \x1B[33m\u26A0 A global shiplightai install was detected at ${u}.
1021
1035
  Global installs don't auto-update and can shadow the project-local CLI on PATH.
1022
1036
  Please remove it: npm uninstall -g shiplightai\x1B[0m
1023
- `)}catch{}}function Lo(e=Ke()){try{let t=M.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<$o)return r}catch{}return null}function Co(e,t=Ke()){try{M.mkdirSync(j.dirname(t),{recursive:!0}),M.writeFileSync(t,JSON.stringify(e))}catch{}}async function Ro(){try{let e=await fetch(Po,{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 No(e,t){let r=u=>{let g=u.indexOf("-");return g===-1?[u,!1]:[u.slice(0,g),!0]},o=u=>u.split(".").map(g=>parseInt(g,10)||0),[i,n]=r(e),[s,a]=r(t),c=o(i),l=o(s),p=Math.max(c.length,l.length);for(let u=0;u<p;u++){let g=c[u]??0,f=l[u]??0;if(g<f)return!0;if(g>f)return!1}return!!(n&&!a)}async function kt(e={}){let t=e.runningVersion??Eo,r=e.cwd??process.cwd(),o=e.cacheFile??Ke(),i=e.fetchLatest??Ro,n=e.env??process.env,s=e.warn??(l=>console.warn(l));if(n.CI||t==="dev"||!M.existsSync(j.join(r,"package-lock.json")))return;let a=null,c=Lo(o);if(c)a=c.latest;else{try{a=await i()}catch{a=null}a&&Co({latest:a,fetchedAt:Date.now()},o)}a&&No(t,a)&&s(`
1037
+ `)}catch{}}function Lo(e=Ke()){try{let t=M.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<$o)return r}catch{}return null}function Co(e,t=Ke()){try{M.mkdirSync(W.dirname(t),{recursive:!0}),M.writeFileSync(t,JSON.stringify(e))}catch{}}async function Ro(){try{let e=await fetch(Po,{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 No(e,t){let r=u=>{let g=u.indexOf("-");return g===-1?[u,!1]:[u.slice(0,g),!0]},o=u=>u.split(".").map(g=>parseInt(g,10)||0),[i,n]=r(e),[s,a]=r(t),c=o(i),l=o(s),p=Math.max(c.length,l.length);for(let u=0;u<p;u++){let g=c[u]??0,f=l[u]??0;if(g<f)return!0;if(g>f)return!1}return!!(n&&!a)}async function kt(e={}){let t=e.runningVersion??Eo,r=e.cwd??process.cwd(),o=e.cacheFile??Ke(),i=e.fetchLatest??Ro,n=e.env??process.env,s=e.warn??(l=>console.warn(l));if(n.CI||t==="dev"||!M.existsSync(W.join(r,"package-lock.json")))return;let a=null,c=Lo(o);if(c)a=c.latest;else{try{a=await i()}catch{a=null}a&&Co({latest:a,fetchedAt:Date.now()},o)}a&&No(t,a)&&s(`
1024
1038
  \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
1025
1039
  Run: npm update shiplightai\x1B[0m
1026
1040
  `)}Ln.config();Tt();kt();function So(){console.log(`
@@ -1045,5 +1059,5 @@ Examples:
1045
1059
  shiplight transpile
1046
1060
  shiplight transpile "tests/**/*.test.yaml"
1047
1061
  shiplight debug tests/login.test.yaml
1048
- `)}var We=process.argv[2];switch(We){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>($t(),Pt));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(Wr(),Gr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Xr(),Jr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(lo(),co));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(fo(),ho));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(bo(),yo));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=wo().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":So();break;default:We&&console.error(`Unknown command: ${We}
1049
- `),So(),process.exit(We?1:0)}
1062
+ `)}var je=process.argv[2];switch(je){case"create":{let{runCreate:e}=await Promise.resolve().then(()=>($t(),Pt));await e(process.argv.slice(3));break}case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>(jr(),Gr));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Xr(),Jr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(lo(),co));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(fo(),ho));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(wo(),yo));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=bo().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":So();break;default:je&&console.error(`Unknown command: ${je}
1063
+ `),So(),process.exit(je?1:0)}
package/dist/index.js CHANGED
@@ -4366,7 +4366,7 @@ ${p.join(`
4366
4366
  `)}function ic(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...lo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let a=new Set,o={imports:a,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=$n(t?.tags);i.push(`test.describe.serial('${s}${_e(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...gi("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...gi("beforeEach",e.beforeEach,o,1)),i.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],h=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 p of d.parameters){let g=so(d.testFlow,p.values);i.push(...bi(g,`${_e(d.name)} [${_e(p.name)}]`,o,1,h)),i.push("")}else i.push(...bi(d.testFlow,_e(d.name),o,1,h)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&i.push("")}return e.afterEach&&e.afterEach.length>0&&(i.push(...gi("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...gi("afterAll",e.afterAll,o,1)),i.push("});"),co(i,a),i.join(`
4367
4367
  `)}function $n(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var nc=["testContext","request"];function wi(e){let t=new Set;function i(n){for(let a of n)switch(a.type){case ue.ACTION:{let r=a.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&nc.includes(s)&&t.add(s);break}case ue.STEP:i(a.statements);break;case ue.IF_ELSE:{let o=a;i(o.then),o.else&&i(o.else);break}case ue.WHILE_LOOP:i(a.body);break}}return i(e),t}function ac(e){let t=wi(e.statements??[]);if(e.teardown)for(let i of wi(e.teardown))t.add(i);return t}function In(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function bi(e,t,i,n=0,a){let o=" ".repeat(n),r=[],s=ac(e),l=In(s),c=a?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),a?.skip===!0?r.push(`${o} test.skip();`):typeof a?.skip=="string"&&r.push(`${o} test.skip(true, '${_e(a.skip)}');`),a?.fail===!0?r.push(`${o} test.fail();`):typeof a?.fail=="string"&&r.push(`${o} test.fail(true, '${_e(a.fail)}');`),a?.slow&&r.push(`${o} test.slow();`),a?.timeout&&r.push(`${o} test.setTimeout(${a.timeout});`);let d=e.teardown&&e.teardown.length>0,h=n+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=pe(e.statements,h+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=pe(e.teardown,h+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=pe(e.statements,h,i);r.push(...p)}return r.push(`${o}});`),r}function no(e,t,i){let n=[],a=ro(t),o=wi(a),r=In(o);return n.push(`test.${e}(async (${r}) => {`),n.push(...pe(a,1,i,e)),n.push("});"),n}function gi(e,t,i,n){let a=" ".repeat(n),o=[],r=ro(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${a}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${a} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...pe(r,n+1,l,e)),o.push(`${a} await page.close();`),o.push(`${a}});`)}else{let l=wi(r),c=In(l);o.push(`${a}test.${e}(async (${c}) => {`),o.push(...pe(r,n+1,i,e)),o.push(`${a}});`)}return o}function ro(e){let i=Ul({goal:"_hook",statements:e});return me(i).statements??[]}function so(e,t){let i=kn(e);for(let[n,a]of Object.entries(t))i=i.split(`<<${n}>>`).join(String(a));return me(i)}function lo(){return["import { test, expect } from 'shiplightai/fixture';"]}function co(e,t){if(t.size>0){let i=0;for(let a=0;a<e.length;a++)e[a].startsWith("import ")&&(i=a+1);let n=Array.from(t);e.splice(i,0,...n)}}var oo=5;function mo(e,t){let i={expandingPaths:new Set([ho(t)]),depth:0,referencedPaths:new Set},n={...e};Array.isArray(n.statements)&&(n.statements=De(n.statements,t,i)),Array.isArray(n.teardown)&&(n.teardown=De(n.teardown,t,i));for(let a of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[a])&&(n[a]=De(n[a],t,i));return{doc:n,referencedTemplatePaths:Array.from(i.referencedPaths)}}function De(e,t,i){let n=[];for(let a of e)if(dc(a)){let o=uc(a,t,i);n.push(...o)}else n.push(hc(a,t,i));return n}function dc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function uc(e,t,i){if(i.depth>=oo)throw new Error(`Template expansion exceeded maximum depth of ${oo}. Check for deeply nested or circular template references.`);let n=ho(lc(t),e.template);if(i.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);i.referencedPaths.add(n);let a;try{a=sc(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=ao(a);if(!o||typeof o!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let r=o.params||[],s=e.params||{};for(let d of r)if(!(d in s))throw new Error(`Template ${e.template} requires param "${d}" but it was not provided. Required params: [${r.join(", ")}]`);let l=o.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(s).length>0){let h=cc(l);for(let[p,g]of Object.entries(s))h=h.split(`<<${p}>>`).join(String(g));l=ao(h)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return De(l,n,c)}function hc(e,t,i){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=De(n.statements,t,i)),Array.isArray(n.THEN)&&(n.THEN=De(n.THEN,t,i)),Array.isArray(n.ELSE)&&(n.ELSE=De(n.ELSE,t,i)),Array.isArray(n.DO)&&(n.DO=De(n.DO,t,i)),n}var Cn=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Pn(e,t){let i=rc(e),n=i?.name,a=i?.tags,o=i?.use;if(i&&(i.name!==void 0||i.tags!==void 0||i.use!==void 0)&&(delete i.name,delete i.tags,delete i.use),i?.suite){if(i.goal||i.statements)throw new Cn('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return pc(i,n,a,o,t)}return mc(i,n,a,o,t)}function mc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=po(e?.parameters),l=e?.timeout,c=e?.skip,d=e?.fail,h=e?.only,p=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 Cn(`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 g=[];if(a&&e&&typeof e=="object"){let b=mo(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=uo(e),v=me(f);return a&&(Ve(v.statements??[],a,"main"),v.teardown&&Ve(v.teardown,a,"teardown")),{testFlow:v,name:t,tags:i,use:n,beforeEach:o,afterEach:r,parameters:s,timeout:l,skip:c,fail:d,only:h,slow:p,referencedTemplatePaths:g}}function pc(e,t,i,n,a){let o=e.suite;if(!Array.isArray(o.tests)||o.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let r=o.beforeAll,s=o.afterAll,l=o.beforeEach,c=o.afterEach,d=[],h=o.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let v={goal:f.name,statements:f.statements};f.teardown&&(v.teardown=f.teardown);let b=[],m=v;if(a&&typeof v=="object"){let A=mo(v,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=uo(m),S=me(y),E=po(f.parameters);return{testFlow:S,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:E,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),p=o.base_url,g=p?{...n,baseURL:p}:n;return{suite:{beforeAll:r,afterAll:s,beforeEach:l,afterEach:c,tests:h},name:t,tags:i,use:g,referencedTemplatePaths:d}}function po(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Ve(e,t,i){for(let n=0;n<e.length;n++){let a=e[n],o=`${i}.${n}`,r=a.description||"";if(a.uid=gc(t,o,r),a.type===ue.STEP)Ve(a.statements,t,o);else if(a.type===ue.IF_ELSE){let s=a;Ve(s.then,t,`${o}.then`),s.else&&Ve(s.else,t,`${o}.else`)}else a.type===ue.WHILE_LOOP&&Ve(a.body,t,`${o}.body`)}}function gc(e,t,i){let n=oc("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function go(e,t,i){let n=/\btemplate:\s/.test(e),a=/^suite:/m.test(e),o=n||a?null:En(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Pn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},h=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=ic(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=tc(c.testFlow,{...d,testName:c.name,tags:c.tags,use:h,beforeEach:c.beforeEach,afterEach:c.afterEach,parameters:c.parameters,timeout:c.timeout,skip:c.skip,fail:c.fail,only:c.only,slow:c.slow});let p=r.split(`
4368
4368
  `).filter(g=>!g.startsWith("import ")).join(`
4369
- `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),wc(bc(s),{recursive:!0}),fc(s,r)}catch(c){let d=c instanceof Cn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var On="0.1.48";function fo(e){try{return yc(e).mtimeMs}catch{return 0}}var _c=`// @generated by shiplightai v${On}`;function kc(e,t){if(!bo(e)||wo(e,"utf-8").split(`
4369
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),wc(bc(s),{recursive:!0}),fc(s,r)}catch(c){let d=c instanceof Cn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var On="0.1.49";function fo(e){try{return yc(e).mtimeMs}catch{return 0}}var _c=`// @generated by shiplightai v${On}`;function kc(e,t){if(!bo(e)||wo(e,"utf-8").split(`
4370
4370
  `,1)[0]!==_c)return!1;let n=fo(e);for(let a of t)if(fo(a)>n)return!1;return!0}function Sc(e){let t=process.argv.slice(2),i=[],n=Ln(e);for(let a of t){if(a.startsWith("-"))continue;let o=a.endsWith(".yaml.spec.ts")?a.replace(/\.yaml\.spec\.ts$/,".test.yaml"):a;if(!o.endsWith(".test.yaml"))continue;let r=Ln(e,o);bo(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function yo(e){let t=Sc(e.cwd),i=t??vc("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),n=[];for(let a of i){let o=Ln(e.cwd,a),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=wo(o,"utf-8");try{let l=Pn(s,o),c=xc(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&kc(r,[o,...l.referencedTemplatePaths]))continue;let p=go(s,o,{version:On,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${a}:`,l),n.push({file:a,error:l})}}if(n.length>0){let a=`[shiplight] Transpilation failed for ${n.length} file(s):
4371
4371
  `+n.map(o=>` - ${o.file}`).join(`
4372
4372
  `);if(t)throw new Error(a);console.warn(a+" (skipped)")}}import*as ae from"fs";import*as mt from"path";import{globSync as Ac}from"glob";function _o(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new Fn:new Rn(e)}var $c=".shiplight/action-cache";function vo(e){return e.replace(/\//g,"__")+".json"}function Mc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Rn=class{constructor(t){this.cwd=t;this.cacheDir=mt.join(t,$c)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!ae.existsSync(this.cacheDir))return i;for(let n of t){let a=mt.join(this.cacheDir,vo(n));try{if(ae.existsSync(a)){let o=ae.readFileSync(a,"utf-8");i.set(n,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;ae.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[n,a]of t)try{let o=mt.join(this.cacheDir,vo(n)),r=Tn();if(ae.existsSync(o))try{r=JSON.parse(ae.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...a.entries}};ae.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!ae.existsSync(this.cacheDir))return;let t=Ac("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,n=0;for(let a of t)try{let o=ae.readFileSync(mt.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=Mc(a);i.set(s,r),n+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},Fn=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(Nn(),Dn));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(Nn(),Dn));return i(t)}loadAll(){}};function Pc(e={}){e.dotenv!==!1&&Lc(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=_o(t).loadAll();return yo({cwd:t,actionEntityStores:n}),e.apiKey&&(process.env.__SHIPLIGHT_API_KEY=e.apiKey),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never",reportToCloud:e.reportToCloud,apiKey:e.apiKey}]]}}function Oc(e,...t){return Cc(e,...t)}function Lc(e){let t=[],i=We.resolve(e),n=We.resolve(process.cwd());for(;;){let a=We.join(i,".env");if(ko.existsSync(a)&&t.push(a),i===n)break;let o=We.dirname(i);if(o===i)break;i=o}for(let a of t)Ic.config({path:a})}R();R();It();Ze();import{z as Fs}from"zod";var Zu=Fs.object({instruction:Fs.string().describe('The instruction of the operation to perform. Can only include one operation. Do not inlcude element indexes just describe the element, e.g. "select the text "Hello, world!" in "Hello, world!""')});function Hs(e){e.register({name:"perform_accurate_operation",description:"Perform an operation that requires accurate interaction like dragging or interacting with a specific area of an element. Only use this action when neccecary.",schema:Zu,usesElementIndex:!1,async execute(t,i){let{instruction:n}=t,a={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await qe(n,a,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:n,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:n}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await $t(r,a);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}Te();R();import{generateText as Qu}from"ai";import{convert as eh}from"html-to-text";async function th(e,t,i){let{apiKey:n,domain:a}=e;if(!n||!a)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${a}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}u.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let h=(d.storage||{}).url;if(u.info(`message_url: ${h}`),!h){let f=(d.message||{}).headers||{},v=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!v.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:v,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${a}/messages/${g}`;try{let v=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(v.ok){let b=await v.json(),m=b.Subject||"",y=b.From||"",S=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${y}`),u.info(`to_addr: ${S}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let M=b["body-html"]||b["body-plain"]||"";if(M&&M.includes("<")&&(M=eh(M)),u.info(`Body: ${M.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!M.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:y,to:S,date:E,body:M,message_id:A});continue}else u.warn(`Messages API returned ${v.status}`)}catch(v){u.warn(`Failed to parse JSON response: ${v}`)}}try{let f=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!f.ok){u.warn(`Could not fetch stored message: ${f.status}`);continue}let v=await f.text();u.info(`Fallback: Raw email length: ${v.length}`);let b=v.split(`
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "shiplightai",
3
- "version": "0.1.48",
3
+ "version": "0.1.49",
4
4
  "type": "module",
5
5
  "description": "Shiplight CLI for running and debugging .test.yaml files",
6
6
  "main": "dist/index.js",
@@ -84,11 +84,11 @@
84
84
  "tsup": "^8.3.5",
85
85
  "typescript": "5.5.4",
86
86
  "mcp-tools": "1.0.0",
87
- "shiplight-tools": "1.0.0",
88
- "shiplight-types": "0.1.0",
89
- "@loggia/common": "1.0.0",
87
+ "sdk-core": "0.1.0",
90
88
  "sdk-internal": "0.1.1",
91
- "sdk-core": "0.1.0"
89
+ "shiplight-types": "0.1.0",
90
+ "shiplight-tools": "1.0.0",
91
+ "@loggia/common": "1.0.0"
92
92
  },
93
93
  "peerDependencies": {
94
94
  "@playwright/test": "1.58.2"