shiplightai 0.1.53 → 0.1.54

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 Rc(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...Ho()),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=Fo(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("});"),Wo(i,a),i.join(`
4365
4365
  `)}function Yn(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var Fc=["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"&&Fc.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 Hc(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=Hc(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 Lo(e,t,i){let n=[],a=Ro(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=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(...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 Ro(e){let i=(0,No.stringify)({goal:"_hook",statements:e});return pe(i).statements??[]}function Fo(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 Ho(){return["import { test, expect } from 'shiplightai/fixture';"]}function Wo(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 Do=5;function Go(e,t){let i={expandingPaths:new Set([(0,xt.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(Wc(a)){let o=Uc(a,t,i);n.push(...o)}else n.push(Bc(a,t,i));return n}function Wc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Uc(e,t,i){if(i.depth>=Do)throw new Error(`Template expansion exceeded maximum depth of ${Do}. Check for deeply nested or circular template references.`);let n=(0,xt.resolve)((0,xt.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,Bo.readFileSync)(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=(0,bt.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,bt.stringify)(l);for(let[p,g]of Object.entries(s))h=h.split(`<<${p}>>`).join(String(g));l=(0,bt.parse)(h)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return Re(l,n,c)}function Bc(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,yt.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 jc(i,n,a,o,t)}return Gc(i,n,a,o,t)}function Gc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=jo(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=Go(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=(0,yt.stringify)(e),x=pe(f);return a&&(Ye(x.statements??[],a,"main"),x.teardown&&Ye(x.teardown,a,"teardown")),{testFlow:x,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 jc(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 x={goal:f.name,statements:f.statements};f.teardown&&(x.teardown=f.teardown);let b=[],m=x;if(a&&typeof x=="object"){let A=Go(x,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=(0,yt.stringify)(m),S=pe(y),E=jo(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 jo(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=Kc(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 Kc(e,t,i){let n=(0,Uo.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 zo(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=Rc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Nc(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,vt.mkdirSync)((0,Ko.dirname)(s),{recursive:!0}),(0,vt.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.53";function Vo(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var zc=`// @generated by shiplightai v${ea}`;function Vc(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,vt.mkdirSync)((0,Ko.dirname)(s),{recursive:!0}),(0,vt.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.54";function Vo(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var zc=`// @generated by shiplightai v${ea}`;function Vc(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4368
4368
  `,1)[0]!==zc)return!1;let n=Vo(e);for(let a of t)if(Vo(a)>n)return!1;return!0}function Xc(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 Yo(e){let t=Xc(e.cwd),i=t??(0,Xo.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)&&Vc(r,[o,...l.referencedTemplatePaths]))continue;let p=zo(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),_t=J(require("path"),1),Zo=require("glob");function Qo(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new aa:new na(e)}var qc=".shiplight/action-cache";function qo(e){return e.replace(/\//g,"__")+".json"}function Zc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var na=class{constructor(t){this.cwd=t;this.cacheDir=_t.join(t,qc)}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=_t.join(this.cacheDir,qo(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=_t.join(this.cacheDir,qo(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,Zo.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(_t.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=Zc(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 nr(e={}){e.dotenv!==!1&&Qc(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=Qo(t).loadAll();return Yo({cwd:t,actionEntityStores:n}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function ar(e,...t){return(0,ir.defineConfig)(e,...t)}function Qc(e){let t=[],i=Be.resolve(e),n=Be.resolve(process.cwd());for(;;){let a=Be.join(i,".env");if(er.existsSync(a)&&t.push(a),i===n)break;let o=Be.dirname(i);if(o===i)break;i=o}for(let a of t)tr.default.config({path:a})}F();F();Ut();tt();var Ya=require("zod"),Ju=Ya.z.object({instruction:Ya.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 xl(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:Ju,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 Ht(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 vl=require("ai"),_l=require("html-to-text");async function qu(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||{},x=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&&!x.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:x,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 x=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(x.ok){let b=await x.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,_l.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 ${x.status}`)}catch(x){u.warn(`Failed to parse JSON response: ${x}`)}}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 x=await f.text();u.info(`Fallback: Raw email length: ${x.length}`);let b=x.split(`
@@ -873,7 +873,7 @@
873
873
  });
874
874
  </script>
875
875
  </body>
876
- </html>`}var Pe="0.1.53",Oe=Pe!=="dev"?Pe:void 0;var ao=3600*1e3,so=10080*60*1e3;var wt={before:0,main:1,teardown:2,after:3};function Ie(e){let t=e.split(".")[0];return wt[t]??1}function Le(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function vt(e){return[...e].sort(([t],[r])=>{let i=Ie(t),o=Ie(r);if(i!==o)return i-o;let n=Le(t),a=Le(r);for(let c=0;c<Math.max(n.length,a.length);c++){let d=n[c]??-1,p=a[c]??-1;if(d!==p)return d-p}return 0})}function _t(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function St(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function ee(e,t="main",r,i,o){r===void 0&&(r=!_t(e).has("test.step")),o||(o=new Map);let n=[];for(let a of e){if(a.category==="fixture"||a.category==="test.attach")continue;if(a.category==="hook"){let d=St(a.title,t);n.push(...ee(a.steps,d,r,i,o));continue}if(a.category==="test.step"||r&&(a.category==="expect"||a.category==="pw:api")){let d=o.get(t)??0;o.set(t,d+1);let p=`${t}.${d}`,g={stepId:p,description:a.title,status:a.error?"failure":a.duration===-1?"skipped":"success",duration:a.duration>=0?a.duration:void 0};a.error&&(g.error=a.error.message??a.error.stack),i&&a.location&&i.set(p,a.location),n.push(g),a.steps.length>0&&n.push(...ee(a.steps,p,r,i,o))}}return n}var te=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let p of this.collected){let g=p.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(p)}let i=[];for(let[,p]of r.entries()){let g=p[0].test.location.file,y=[],l,u,h;for(let b=0;b<p.length;b++){let{test:A,result:E}=p[b],M=await this.buildReportTest(A,E,g);l=M,u||(u=M.startTime),h=M.endTime,y.push({attemptNumber:b+1,status:E.status,duration:E.duration,steps:M.steps,error:M.error,videoPath:M.videoPath,tracePath:M.tracePath})}let m=y[y.length-1],{test:k}=p[p.length-1],v={title:k.title,baseTitle:l?.baseTitle,file:_.relative(process.cwd(),g),status:m.status,duration:m.duration,steps:m.steps,error:m.error,videoPath:m.videoPath,tracePath:m.tracePath,actionStepsMap:l?.actionStepsMap,tags:l?.tags,baseUrl:l?.baseUrl,skip:l?.skip,slow:l?.slow,timeout:l?.timeout,parameterSetName:l?.parameterSetName,startTime:u,endTime:h,suiteName:l?.suiteName},w=y.some(b=>b.status==="failed"||b.status==="timedOut");y.length>1&&w&&m.status==="passed"&&(v.flaky=!0,v.retries=y.length-1,v.attempts=y),i.push(v)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},n=_.isAbsolute(this.outputFolder)?this.outputFolder:_.join(process.cwd(),this.outputFolder);T.mkdirSync(n,{recursive:!0});let a=_.join(n,"screenshots");for(let p=0;p<o.tests.length;p++){let g=o.tests[p],y=g.attempts&&g.attempts.length>0,l=[{obj:g,prefix:y?`test-${p}-attempt-0`:`test-${p}`,screenshotSubDir:`test-${p}`}];if(g.attempts)for(let u=0;u<g.attempts.length;u++)l.push({obj:g.attempts[u],prefix:`test-${p}-attempt-${u+1}`,screenshotSubDir:`test-${p}/attempt-${u}`});for(let{obj:u,prefix:h,screenshotSubDir:m}of l){let k=_.join(a,m),v=!1;for(let w of u.steps)if(w.screenshot&&_.isAbsolute(w.screenshot))try{v||(T.mkdirSync(k,{recursive:!0}),v=!0);let b=`${w.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(w.screenshot,_.join(k,b)),w.screenshot=`screenshots/${m}/${b}`}catch(b){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,b)}if(u.videoPath&&_.isAbsolute(u.videoPath)){let w=_.extname(u.videoPath)||".webm",b=`${h}-video${w}`;try{T.copyFileSync(u.videoPath,_.join(n,b)),u.videoPath=b}catch{u.videoPath=void 0}}if(u.tracePath&&_.isAbsolute(u.tracePath)){let w=_.extname(u.tracePath)||".zip",b=`${h}-trace${w}`;try{T.copyFileSync(u.tracePath,_.join(n,b)),u.tracePath=b}catch{u.tracePath=void 0}}}}let c=_.join(n,"report-data.json");T.writeFileSync(c,JSON.stringify(o,null,2),"utf-8");let d=_.join(n,"index.html");if(T.writeFileSync(d,Me(o),"utf-8"),console.log(`
876
+ </html>`}var Pe="0.1.54",Oe=Pe!=="dev"?Pe:void 0;var ao=3600*1e3,so=10080*60*1e3;var wt={before:0,main:1,teardown:2,after:3};function Ie(e){let t=e.split(".")[0];return wt[t]??1}function Le(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function vt(e){return[...e].sort(([t],[r])=>{let i=Ie(t),o=Ie(r);if(i!==o)return i-o;let n=Le(t),a=Le(r);for(let c=0;c<Math.max(n.length,a.length);c++){let d=n[c]??-1,p=a[c]??-1;if(d!==p)return d-p}return 0})}function _t(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function St(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function ee(e,t="main",r,i,o){r===void 0&&(r=!_t(e).has("test.step")),o||(o=new Map);let n=[];for(let a of e){if(a.category==="fixture"||a.category==="test.attach")continue;if(a.category==="hook"){let d=St(a.title,t);n.push(...ee(a.steps,d,r,i,o));continue}if(a.category==="test.step"||r&&(a.category==="expect"||a.category==="pw:api")){let d=o.get(t)??0;o.set(t,d+1);let p=`${t}.${d}`,g={stepId:p,description:a.title,status:a.error?"failure":a.duration===-1?"skipped":"success",duration:a.duration>=0?a.duration:void 0};a.error&&(g.error=a.error.message??a.error.stack),i&&a.location&&i.set(p,a.location),n.push(g),a.steps.length>0&&n.push(...ee(a.steps,p,r,i,o))}}return n}var te=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let p of this.collected){let g=p.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(p)}let i=[];for(let[,p]of r.entries()){let g=p[0].test.location.file,y=[],l,u,h;for(let b=0;b<p.length;b++){let{test:A,result:E}=p[b],M=await this.buildReportTest(A,E,g);l=M,u||(u=M.startTime),h=M.endTime,y.push({attemptNumber:b+1,status:E.status,duration:E.duration,steps:M.steps,error:M.error,videoPath:M.videoPath,tracePath:M.tracePath})}let m=y[y.length-1],{test:k}=p[p.length-1],v={title:k.title,baseTitle:l?.baseTitle,file:_.relative(process.cwd(),g),status:m.status,duration:m.duration,steps:m.steps,error:m.error,videoPath:m.videoPath,tracePath:m.tracePath,actionStepsMap:l?.actionStepsMap,tags:l?.tags,baseUrl:l?.baseUrl,skip:l?.skip,slow:l?.slow,timeout:l?.timeout,parameterSetName:l?.parameterSetName,startTime:u,endTime:h,suiteName:l?.suiteName},w=y.some(b=>b.status==="failed"||b.status==="timedOut");y.length>1&&w&&m.status==="passed"&&(v.flaky=!0,v.retries=y.length-1,v.attempts=y),i.push(v)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},n=_.isAbsolute(this.outputFolder)?this.outputFolder:_.join(process.cwd(),this.outputFolder);T.mkdirSync(n,{recursive:!0});let a=_.join(n,"screenshots");for(let p=0;p<o.tests.length;p++){let g=o.tests[p],y=g.attempts&&g.attempts.length>0,l=[{obj:g,prefix:y?`test-${p}-attempt-0`:`test-${p}`,screenshotSubDir:`test-${p}`}];if(g.attempts)for(let u=0;u<g.attempts.length;u++)l.push({obj:g.attempts[u],prefix:`test-${p}-attempt-${u+1}`,screenshotSubDir:`test-${p}/attempt-${u}`});for(let{obj:u,prefix:h,screenshotSubDir:m}of l){let k=_.join(a,m),v=!1;for(let w of u.steps)if(w.screenshot&&_.isAbsolute(w.screenshot))try{v||(T.mkdirSync(k,{recursive:!0}),v=!0);let b=`${w.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(w.screenshot,_.join(k,b)),w.screenshot=`screenshots/${m}/${b}`}catch(b){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,b)}if(u.videoPath&&_.isAbsolute(u.videoPath)){let w=_.extname(u.videoPath)||".webm",b=`${h}-video${w}`;try{T.copyFileSync(u.videoPath,_.join(n,b)),u.videoPath=b}catch{u.videoPath=void 0}}if(u.tracePath&&_.isAbsolute(u.tracePath)){let w=_.extname(u.tracePath)||".zip",b=`${h}-trace${w}`;try{T.copyFileSync(u.tracePath,_.join(n,b)),u.tracePath=b}catch{u.tracePath=void 0}}}}let c=_.join(n,"report-data.json");T.writeFileSync(c,JSON.stringify(o,null,2),"utf-8");let d=_.join(n,"index.html");if(T.writeFileSync(d,Me(o),"utf-8"),console.log(`
877
877
  Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let p=(await import("open")).default;await p(d)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:_.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(l=>l.message||l.stack||String(l)).join(`
878
878
 
879
879
  `)),r.stdout.length>0&&(o.stdout=r.stdout.map(l=>typeof l=="string"?l:l.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(l=>typeof l=="string"?l:l.toString()).join(""));for(let l of r.attachments)l.name==="video"&&l.path&&(o.videoPath=l.path),l.name==="trace"&&l.path&&(o.tracePath=l.path);let n=r.attachments.find(l=>l.name==="shiplight-results"),a=null;if(n)try{if(n.body)a=JSON.parse(n.body.toString("utf-8"));else if(n.path){let l=T.readFileSync(n.path,"utf-8");a=JSON.parse(l)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},p=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),g=p?p[1]:t.title,y=p?p[2]:void 0;if(y&&(o.parameterSetName=y),T.existsSync(c))try{let l=T.readFileSync(c,"utf-8"),u=Te(l,c);if(u.suite){let h=u.suite.tests.find(m=>m.name===g);h&&(d=z(h.testFlow),h.tags?.length&&(o.tags=h.tags),h.skip!==void 0&&(o.skip=h.skip),h.slow&&(o.slow=h.slow),h.timeout!==void 0&&(o.timeout=h.timeout),o.baseTitle=h.name||h.testFlow?.goal),o.suiteName=u.name,u.tags?.length&&(o.suiteTags=u.tags),u.use?.baseURL&&(o.baseUrl=u.use.baseURL)}else u.testFlow&&(d=z(u.testFlow),o.baseTitle=u.name||u.testFlow?.goal,u.tags?.length&&(o.tags=u.tags),u.use?.baseURL&&(o.baseUrl=u.use.baseURL))}catch{}if(a||Object.keys(d).length>0){let l=new Set([...Object.keys(d),...Object.keys(a||{})]),u=Array.from(l).map(m=>[m,null]),h=vt(u);for(let[m]of h){let k=d[m],v=a?.[m],w=k?.description;if(!w||w==="Action"||w==="Draft"){let A=k?.action_entity;w=A?.action_description||A?.action_data?.kwargs?.description||v?.description||m}let b={stepId:m,description:w,status:v?.status||"pending",duration:v?.duration};if(v?.message){let A=typeof v.message=="string"?v.message:JSON.stringify(v.message,null,2);v.status==="failure"?b.error=A:b.message=A}if(v?.screenshot){let A=v.screenshot,E=n?.path?_.dirname(n.path):"",M=_.isAbsolute(A)?A:_.join(E,A);T.existsSync(M)&&(b.screenshot=M)}v?.code&&(b.code=v.code),o.steps.push(b)}}if(a===null&&Object.keys(d).length===0&&!i.endsWith(".yaml.spec.ts")){let l=new Map;if(o.steps=ee(r.steps,"main",void 0,l),o.steps.length>0){let u=new Map;o.actionStepsMap=Object.fromEntries(o.steps.map(h=>{let m=l.get(h.stepId),k;if(m?.file){if(!u.has(m.file))try{u.set(m.file,T.readFileSync(m.file,"utf-8").split(`
package/dist/cli.js CHANGED
@@ -14,10 +14,10 @@ Install it as a project dependency instead:
14
14
  \x1B[33m\u26A0 A global shiplightai install was detected at ${u}.
15
15
  Global installs don't auto-update and can shadow the project-local CLI on PATH.
16
16
  Please remove it: npm uninstall -g shiplightai\x1B[0m
17
- `)}catch{}}function Jo(e=Ze()){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<Ko)return r}catch{}return null}function Xo(e,t=Ze()){try{M.mkdirSync(W.dirname(t),{recursive:!0}),M.writeFileSync(t,JSON.stringify(e))}catch{}}async function qo(){try{let e=await fetch(Wo,{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 Zo(e,t){let r=u=>{let h=u.indexOf("-");return h===-1?[u,!1]:[u.slice(0,h),!0]},o=u=>u.split(".").map(h=>parseInt(h,10)||0),[n,s]=r(e),[a,i]=r(t),c=o(n),l=o(a),p=Math.max(c.length,l.length);for(let u=0;u<p;u++){let h=c[u]??0,g=l[u]??0;if(h<g)return!0;if(h>g)return!1}return!!(s&&!i)}async function Ct(e={}){let t=e.runningVersion??Xe,r=e.cwd??process.cwd(),o=e.cacheFile??Ze(),n=e.fetchLatest??qo,s=e.env??process.env,a=e.warn??(l=>console.warn(l));if(s.CI||t==="dev"||!M.existsSync(W.join(r,"package-lock.json")))return;let i=null,c=Jo(o);if(c)i=c.latest;else{try{i=await n()}catch{i=null}i&&Xo({latest:i,fetchedAt:Date.now()},o)}i&&Zo(t,i)&&a(`
17
+ `)}catch{}}function Jo(e=Ze()){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<Ko)return r}catch{}return null}function Xo(e,t=Ze()){try{M.mkdirSync(W.dirname(t),{recursive:!0}),M.writeFileSync(t,JSON.stringify(e))}catch{}}async function qo(){try{let e=await fetch(Wo,{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 Zo(e,t){let r=u=>{let h=u.indexOf("-");return h===-1?[u,!1]:[u.slice(0,h),!0]},o=u=>u.split(".").map(h=>parseInt(h,10)||0),[n,s]=r(e),[a,i]=r(t),c=o(n),l=o(a),p=Math.max(c.length,l.length);for(let u=0;u<p;u++){let h=c[u]??0,m=l[u]??0;if(h<m)return!0;if(h>m)return!1}return!!(s&&!i)}async function Ct(e={}){let t=e.runningVersion??Xe,r=e.cwd??process.cwd(),o=e.cacheFile??Ze(),n=e.fetchLatest??qo,s=e.env??process.env,a=e.warn??(l=>console.warn(l));if(s.CI||t==="dev"||!M.existsSync(W.join(r,"package-lock.json")))return;let i=null,c=Jo(o);if(c)i=c.latest;else{try{i=await n()}catch{i=null}i&&Xo({latest:i,fetchedAt:Date.now()},o)}i&&Zo(t,i)&&a(`
18
18
  \x1B[33m\u26A0 shiplightai ${i} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var Xe,be,Wo,Ko,Vo,Pe=x(()=>{"use strict";Xe="0.1.53",be=Xe!=="dev"?Xe:void 0,Wo="https://registry.npmjs.org/shiplightai/latest",Ko=3600*1e3,Vo=10080*60*1e3});import*as D from"fs";import*as K from"path";function Qe(e){let{projectPath:t}=e,r=e.projectName??K.basename(K.resolve(t));if(D.existsSync(t)&&D.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);D.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"}};D.writeFileSync(K.join(t,"package.json"),JSON.stringify(o,null,2)+`
20
+ `)}var Xe,be,Wo,Ko,Vo,Pe=x(()=>{"use strict";Xe="0.1.54",be=Xe!=="dev"?Xe:void 0,Wo="https://registry.npmjs.org/shiplightai/latest",Ko=3600*1e3,Vo=10080*60*1e3});import*as D from"fs";import*as K from"path";function Qe(e){let{projectPath:t}=e,r=e.projectName??K.basename(K.resolve(t));if(D.existsSync(t)&&D.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);D.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"}};D.writeFileSync(K.join(t,"package.json"),JSON.stringify(o,null,2)+`
21
21
  `),D.writeFileSync(K.join(t,"playwright.config.ts"),`import { defineConfig, shiplightConfig } from 'shiplightai';
22
22
 
23
23
  export default defineConfig({
@@ -104,31 +104,31 @@ test('__shiplight_debug__', async ({ page, agent }) => {
104
104
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
105
105
  await new Promise(() => {});
106
106
  });
107
- `}async function sn(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let s=t();s.once("error",()=>n(!1)),s.once("listening",()=>{s.close(()=>n(!0))}),s.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function tt(e){let{yamlFilePath:t,configPath:r,tempSuffix:o="",headed:n}=e,s=Z.dirname(r),a=await sn(16174),i;if(!ne.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let b=on(ne.readFileSync(t,"utf-8"));b?.use&&typeof b.use=="object"&&!Array.isArray(b.use)&&(i=b.use),b?.base_url&&!i?.baseURL&&(i={...i,baseURL:b.base_url}),b?.settings?.auto_dismiss_modal!==void 0&&(i={...i,autoDismissModal:!!b.settings.auto_dismiss_modal})}catch(b){console.error("[debugger] Could not parse YAML for `use` block:",b)}let c=Z.dirname(Z.resolve(t)),l=o?`-${o}`:"",p=Z.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),u=nn(t,a,i,s);ne.writeFileSync(p,u);let h=()=>{try{ne.unlinkSync(p)}catch{}},g=["playwright","test",p,...n?["--headed"]:[]],m=rn("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:s,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});m.stdout?.on("data",b=>{process.stderr.write(b)}),m.stderr?.on("data",b=>{process.stderr.write(b)});let d=()=>{m.killed||m.kill("SIGTERM")};process.on("SIGTERM",d),process.on("SIGINT",d),process.on("exit",d),m.on("close",b=>{process.removeListener("SIGTERM",d),process.removeListener("SIGINT",d),process.removeListener("exit",d),h(),b!==0&&b!==null&&console.error(`[debugger] Playwright process exited with code ${b}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=["127.0.0.1","::1"];async function w(b){try{let _=b.includes(":")?`[${b}]`:b,k=await fetch(`http://${_}:${a}/api/test-flow`);if(k.ok){try{await k.text()}catch{}return!0}}catch{}return!1}let S=null;for(let b=0;b<180;b++){if(m.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${m.exitCode} before sandbox was ready`);for(let _ of f)if(await w(_)){S=_;break}if(S){console.error(`[debugger] Playwright sandbox ready on ${S}:${a}`);break}if(b===179)throw d(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(_=>setTimeout(_,1e3))}if(!S)throw d(),h(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:S,pid:m.pid??0,cleanup:async()=>{d(),h()}}}var rt=x(()=>{"use strict"});import{z as y}from"zod";var Vt,ot,zt,de,Yt,Jt,Xt,F,qt,Me,nt,st=x(()=>{"use strict";Vt=y.enum(["JS_CODE","AI_MODE"]),ot=y.object({type:Vt,expression:y.string()}),zt=y.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),de=y.object({uid:y.string(),type:zt,comment:y.string().optional()}),Yt=y.object({action_data:y.object({action_name:y.string(),kwargs:y.record(y.any()).optional(),args:y.array(y.any()).optional()}),action_description:y.string().optional(),url:y.string().optional(),xpath:y.string().nullable().optional(),locator:y.string().nullable().optional(),css_selector:y.string().nullable().optional(),unique_selector:y.string().nullable().optional(),element_index:y.number().nullable().optional(),frame_path:y.array(y.any()).optional(),artifacts:y.record(y.any()).optional(),feedback:y.string().optional(),original_browser_use_action:y.any().optional()}).passthrough(),Jt=de.extend({type:y.literal("DRAFT"),description:y.string()}),Xt=de.extend({type:y.literal("ACTION"),description:y.string(),action_entity:Yt.optional(),locator:y.string().optional(),use_pure_vision:y.boolean().optional()}),F=y.lazy(()=>y.union([Jt,Xt,de.extend({type:y.literal("STEP"),description:y.string().optional().default(""),statements:y.array(F),reference_id:y.number().optional()}),de.extend({type:y.literal("IF_ELSE"),description:y.string().optional(),condition:ot,then:y.array(F),else:y.array(F).optional()}),de.extend({type:y.literal("WHILE_LOOP"),description:y.string().optional(),condition:ot,body:y.array(F),timeout_ms:y.number().optional()})])),qt=y.object({name:y.string(),statements:y.array(F),teardown:y.array(F).optional(),skip:y.union([y.boolean(),y.string()]).optional(),timeout:y.number().optional(),fail:y.union([y.boolean(),y.string()]).optional(),only:y.boolean().optional(),slow:y.boolean().optional()}),Me=y.object({tests:y.array(qt).min(1),beforeAll:y.array(F).optional(),afterAll:y.array(F).optional(),beforeEach:y.array(F).optional(),afterEach:y.array(F).optional()}),nt=y.object({comment:y.string().optional(),version:y.string().optional(),goal:y.string().optional(),url:y.string().optional(),baseURL:y.string().optional(),final_feedback:y.string().optional(),completed:y.boolean().optional(),success:y.boolean().optional(),statements:y.array(F).optional(),teardown:y.array(F).optional(),last_modified_at:y.string().optional(),testGroup:Me.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 an,parse as tr,parseAllDocuments as yi,parseDocument as cn,Document as ln,isMap as fe,isSeq as Y}from"yaml";import{v4 as z}from"uuid";function rr(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(j)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(j)),r}function ve(e,t){if(e.testGroup)return nr(e,t);let r=rr(e,t),o=new ln(r);return e.comment&&(o.commentBefore=e.comment),Zt(o,e.statements??[]),e.teardown&&Zt(o,e.teardown,"teardown"),o.toString(or)}function Zt(e,t,r="statements"){let o=e.contents;if(!o||!fe(o))return;let n=o.get(r,!0);Y(n)&&Se(n,t)}function Se(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:n.commentBefore=o.comment),fe(n)){let s=n;if(o.type==="STEP"){let a=s.get("statements",!0);Y(a)&&Se(a,o.statements)}else if(o.type==="IF_ELSE"){let a=s.get("THEN",!0);Y(a)&&Se(a,o.then);let i=s.get("ELSE",!0);Y(i)&&o.else&&Se(i,o.else)}else if(o.type==="WHILE_LOOP"){let a=s.get("DO",!0);Y(a)&&Se(a,o.body)}}}}function nr(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 n={};return e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(j)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(j)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(j)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(j)),n.tests=r.tests.map(s=>{let a={name:s.name};return s.skip!==void 0&&(a.skip=s.skip),s.timeout!==void 0&&(a.timeout=s.timeout),s.fail!==void 0&&(a.fail=s.fail),s.only!==void 0&&(a.only=s.only),s.slow!==void 0&&(a.slow=s.slow),a.statements=s.statements.map(j),s.teardown&&s.teardown.length>0&&(a.teardown=s.teardown.map(j)),a}),o.suite=n,an(o,or)}function j(e){switch(e.type){case"DRAFT":return pn(e);case"ACTION":return un(e);case"STEP":return dn(e);case"IF_ELSE":return hn(e);case"WHILE_LOOP":return fn(e)}}function pn(e){return{intent:e.description}}function un(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:i,js:c}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:i};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let i=r?.code;if(typeof i=="string"&&e.description)return{intent:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let c={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let i=r?.seconds,l={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(l.seconds=i),l}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:i}}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 n={intent:e.description,action:o.action_name},s=e.locator??e.action_entity.locator;s&&(n.locator=s);let a=e.action_entity.xpath;if(a&&(n.xpath=a),e.use_pure_vision&&(n.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[i,c]of Object.entries(o.kwargs))n[i]=c;return o.args&&o.args.length>0&&(n.args=o.args),n}function dn(e){let t={STEP:e.description,statements:e.statements.map(j)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function hn(e){let t={IF:sr(e.condition),THEN:e.then.map(j)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(j)),t}function fn(e){let t={WHILE:sr(e.condition),DO:e.body.map(j)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function sr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function xe(e){try{let t=tr(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 it(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(it);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let o={};for(let[n,s]of Object.entries(t))o[n]=it(s);return o}function R(e){if(e.length>Qt)throw new Error(`YAML input too large (${e.length} bytes, max ${Qt})`);let t=it(tr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return gn(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=nt.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let n=o.data;return Ie(e,n),n}function gn(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 n={tests:r.map(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let c={name:i.name,statements:G(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(c.teardown=G(i.teardown)),i.skip!==void 0&&(c.skip=i.skip),typeof i.timeout=="number"&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only===!0&&(c.only=!0),i.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=G(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=G(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=G(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=G(t.afterEach));let s=Me.safeParse(n);if(!s.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(s.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:s.data}}function G(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(mn)}function mn(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 yn(t);if("WHILE"in t)return wn(t);if("STEP"in t)return bn(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:z(),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,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,s={url:typeof r=="string"?r:String(r)};return o&&(s.new_tab=!0),n!==void 0&&(s.timeout_seconds=n),{uid:z(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:s}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:z(),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:z(),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:z(),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:z(),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 er({...o,action:"function",functionName:r})}if("action"in t)return er(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:z(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function ir(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 yn(e){let t=ir(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:z(),type:"IF_ELSE",condition:t,then:G(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=G(e.ELSE)),o}function wn(e){let t=ir(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:z(),type:"WHILE_LOOP",condition:t,body:G(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function bn(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:z(),type:"STEP",description:t,statements:G(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function er(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,n=typeof e.xpath=="string"?e.xpath:void 0,s=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[l,p]of Object.entries(e))Sn.has(l)||(a[l]=p);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};o&&(i.locator=o),n&&(i.xpath=n);let c={uid:z(),type:"ACTION",description:r,action_entity:i};return s&&(c.use_pure_vision=!0),c}function Ie(e,t){let r;try{r=cn(e)}catch{return}let o=r.contents;if(!o||!fe(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 n=o,s=n.get("statements",!0);Y(s)&&t.statements&&he(s,t.statements);let a=n.get("teardown",!0);Y(a)&&t.teardown&&he(a,t.teardown)}function he(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 n=t[r];if(n.type==="STEP"&&fe(o)){let s=o.get("statements",!0);Y(s)&&he(s,n.statements)}else if(n.type==="IF_ELSE"&&fe(o)){let s=o.get("THEN",!0);Y(s)&&he(s,n.then);let a=o.get("ELSE",!0);Y(a)&&n.else&&he(a,n.else)}else if(n.type==="WHILE_LOOP"&&fe(o)){let s=o.get("DO",!0);Y(s)&&he(s,n.body)}}}var or,Qt,Sn,Oe=x(()=>{"use strict";st();or={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Qt=1024*1024;Sn=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as xi,stringify as _i}from"yaml";var ar=x(()=>{"use strict";Oe()});var at,Le,Ce=x(()=>{"use strict";at=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},Le=e=>{let t=[],r=o=>{for(let n of o){t.push(n);let s=at(n);for(let a of s)r(a.statements)}};return r(e),t}});function pr(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let n=o.action_entity?.action_data?.action_name??"";lr.has(n)||t++}return{action:t,draft:r}}function xn(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function cr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function ct(e,t){let r=t?.coverageThreshold??vn,o=[],n=[],s;try{s=R(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}s.goal||o.push('Missing required field: "goal"'),s.statements?.length||o.push('Missing required field: "statements"');let a=[...Le(s.statements??[]),...s.teardown?Le(s.teardown):[]],{action:i,draft:c}=pr(a),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let h of a){if(h.type==="ACTION"){let g=h,m=g.action_entity?.action_data?.action_name??"";if(m==="js_code"||m==="js_action"||m==="verify"||m==="ai_assert"){let d=g.action_entity?.action_data?.kwargs?.code;if(typeof d=="string"){let f=xn(d);if(f){let w=g.description||m;o.push(`Invalid JS in "${w}": ${f}. ${l}`)}}}}if(h.type==="IF_ELSE"){let g=h;if(g.condition.type==="JS_CODE"){let m=cr(g.condition.expression);m&&o.push(`Invalid JS in IF condition "${g.condition.expression}": ${m}. ${l}`)}}if(h.type==="WHILE_LOOP"){let g=h;if(g.condition.type==="JS_CODE"){let m=cr(g.condition.expression);m&&o.push(`Invalid JS in WHILE condition "${g.condition.expression}": ${m}. ${l}`)}}}let p=i+c,u=p>0?Math.round(i/p*100):0;return p>0&&u/100<r&&n.push(`Low action coverage: ${i}/${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:n,stats:{total:p,action:i,draft:c,coverage:u}}}var vn,lr,ur=x(()=>{"use strict";Oe();Ce();vn=.5,lr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});function lt(){return{version:"1.0",entries:{}}}var dr=x(()=>{"use strict";Ce()});import{v4 as Oi}from"uuid";var hr=x(()=>{"use strict"});var J,pt,fr=x(()=>{"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||{}),pt=18e4});var Re=x(()=>{"use strict"});var gr=x(()=>{"use strict";Re()});var mr,Tn,yr,wr,_e,kn,Pn,br=x(()=>{"use strict";mr=112,Tn=1080-mr,yr={"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"}},wr={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"]},_e=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),wr[e].map(o=>yr[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},kn={desktop:{label:"Desktop",type:"desktop",devices:_e("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:_e("mobile")}},Pn={desktop:{label:"Desktop",type:"desktop",devices:_e("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:_e("mobile",!0)}}});var Sr=x(()=>{"use strict"});var vr=x(()=>{"use strict"});var xr=x(()=>{"use strict"});var _r=x(()=>{"use strict"});var Tr=x(()=>{"use strict"});var kr=x(()=>{"use strict"});var Pr=x(()=>{"use strict"});var Er=x(()=>{"use strict";Re()});var ae=x(()=>{"use strict";ar();ur();Oe();st();dr();hr();Ce();fr();gr();br();Sr();vr();xr();_r();Tr();kr();Pr();Er();Re()});import{stringify as An}from"yaml";import{createHash as Vn}from"crypto";import{parse as zn,stringify as Dr}from"yaml";import{readFileSync as Yn}from"fs";import{resolve as Fr,dirname as Jn}from"path";import{parse as Ir,stringify as Xn}from"yaml";import{readFileSync as os,writeFileSync as ns,mkdirSync as ss}from"fs";import{dirname as is}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function A(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function $n(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Mn(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 De(e){let t=$n(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=Mn(e);if(o){let n=JSON.stringify(o);return`${t}.locator(${n}).first()`}return null}function Ar(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:In.includes(t)}function Ln(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!On.includes(t)}function v(e,t){L.set(e,t)}function Cn(e){return L.get(e)}function pe(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(n=>` ${n},`),"});"]}function $r(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"],n=["undefined","null","true","false"],s=r.map(a=>o.includes(a)||n.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${s.join(", ")})`}function Q(e,t,r,o="main"){let n=[];for(let s=0;s<e.length;s++){let a=e[s],i=`${o}.${s}`,c=Rn(a,t,i,r);c.length>0&&(n.push(...c),s<e.length-1&&n.push(""))}return n}function Rn(e,t,r,o){let n=" ".repeat(t);switch(e.type){case"DRAFT":return Nn(e,t,r,o);case"ACTION":return Dn(e,t,r,o);case"STEP":return Fn(e,t,r,o);case"IF_ELSE":return Un(e,t,r,o);case"WHILE_LOOP":return Bn(e,t,r,o);default:return[`${n}// Unknown statement type: ${e.type}`]}}function Nn(e,t,r,o){let n=" ".repeat(t),s=e.description?.trim()||"";if(!s)return[`${n}// ${r}: Skipping - no description`];if(o.noAgent)return[`${n}// ${r}: ${A(s)}`,`${n}// DRAFT: ${A(s)} (requires agent - skipped in hook)`];let a=JSON.stringify(s);return[`${n}// ${r}: ${A(s)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${a}, '${r}');`]}function Dn(e,t,r,o){let n=" ".repeat(t),s=e.description,a=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!s)return[`${n}// ${r}: Skipping - no description`];if(o.noAgent)return[`${n}// ${r}: ${A(s)}`,`${n}// DRAFT: ${A(s)} (requires agent - skipped in hook)`];let b=JSON.stringify(s),_=!!e.use_pure_vision;return[`${n}// ${r}: ${A(s)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${b}, '${r}', ${_});`]}let l=e.locator?{...c,locator:e.locator}:c;s&&s!==l.action_description&&(l={...l,action_description:s});let p=l.action_data?.action_name||"",u=l.action_description||"",h=Cn(p);if(!h)return[`${n}// ${r}: Unknown action: ${p}`];let g={imports:o.imports},m=h(l,r,g);if(o.noAgent){if(Ar(l))return[`${n}// ${r}: ${A(u)}`,`${n}// AI action: ${A(u)} (requires agent - skipped in hook)`];let b=Hn(l,p,n,r);return b||[`${n}// ${r}: ${A(u)}`,...m.map(_=>`${n}${_}`)]}if(Ar(l))return[`${n}// ${r}: ${A(u)}`,`${n}page = agent.agentServices.validatePage(page);`,...m.map(b=>`${n}${b}`)];let d=JSON.stringify(u),f=m.map(b=>`${n} ${b}`),w=Ln(l),S=a?`'${a}'`:"undefined";return[`${n}// ${r}: ${A(u)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...f,`${n}}, ${d}, '${r}', ${S}, ${w});`]}function Fn(e,t,r,o){let n=" ".repeat(t),s=[];e.description&&e.description.trim()&&s.push(`${n}// Step: ${A(e.description)}`);let a=Q(e.statements,t,o,r);return s.push(...a),s}function Un(e,t,r,o){let n=" ".repeat(t),s=[];if(s.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")s.push(`${n}if (${e.condition.expression}) {`);else{s.push(`${n}// AI Condition: ${A(e.condition.expression)}`);let i=JSON.stringify(e.condition.expression);s.push(`${n}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=Q(e.then,t+1,o,`${r}.then`);if(s.push(...a),e.else&&e.else.length>0){s.push(`${n}} else {`);let i=Q(e.else,t+1,o,`${r}.else`);s.push(...i)}return s.push(`${n}}`),s}function Bn(e,t,r,o){let n=" ".repeat(t),s=[];s.push(`${n}// ${r}: Loop`);let a=e.timeout_ms??pt,i=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,l=`loop_${r.replace(/\./g,"_")}`;if(s.push(`${n}const ${l}_start = Date.now();`),s.push(`${n}const ${l}_timeout = ${a};`),s.push(`${n}const ${l}_check = () => {`),s.push(`${n} if (Date.now() - ${l}_start > ${l}_timeout) {`),s.push(`${n} throw new Error('${c}');`),s.push(`${n} }`),s.push(`${n} return true;`),s.push(`${n}};`),e.condition.type==="JS_CODE")s.push(`${n}while (${l}_check() && (${e.condition.expression})) {`);else{s.push(`${n}// AI Loop Condition: ${A(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);s.push(`${n}while (${l}_check() && await agent.evaluate(page, ${u}, "${r}")) {`)}let p=Q(e.body,t+1,o,`${r}.body`);return s.push(...p),s.push(`${n}}`),s}function Hn(e,t,r,o){let n=e.action_description||"",s=e.action_data?.kwargs||{},a=s.timeout_ms??dt;switch(t){case"go_to_url":case"open_tab":{let i=s.url||"";return[`${r}// ${o}: ${A(n)}`,`${r}await page.goto(${JSON.stringify(i)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${A(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${A(n)}`,`${r}await page.goForward();`];case"input_text":{let i=s.text||"",c=De(e);return c?[`${r}// ${o}: ${A(n)}`,`${r}await ${c}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=s.text||s.label||"",c=De(e);return c?[`${r}// ${o}: ${A(n)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function jn(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Rr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,s={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...Mr("beforeEach",t.beforeEach,s)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let i=t?.testName||e.goal||"Generated test",c=ut(t?.tags);for(let l of t.parameters){let p=Cr(e,l.values);r.push(...Ue(p,`${c}${se(i)} [${se(l.name)}]`,s,0,a)),r.push("")}}else{let i=t?.testName||e.goal||"Generated test",c=ut(t?.tags);r.push(...Ue(e,`${c}${se(i)}`,s,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...Mr("afterEach",t.afterEach,s))),Nr(r,n),r.join(`
108
- `)}function Gn(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Rr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,s={imports:n,actionEntityStore:t?.actionEntityStore},a=t?.testName||"Test Suite",i=ut(t?.tags);r.push(`test.describe.serial('${i}${se(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ne("beforeAll",e.beforeAll,s,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ne("beforeEach",e.beforeEach,s,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 h of p.parameters){let g=Cr(p.testFlow,h.values);r.push(...Ue(g,`${se(p.name)} [${se(h.name)}]`,s,1,u)),r.push("")}else r.push(...Ue(p.testFlow,se(p.name),s,1,u)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ne("afterEach",e.afterEach,s,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ne("afterAll",e.afterAll,s,1)),r.push("});"),Nr(r,n),r.join(`
109
- `)}function ut(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Fe(e){let t=new Set;function r(o){for(let n of o)switch(n.type){case J.ACTION:{let a=n.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&Wn.includes(i)&&t.add(i);break}case J.STEP:r(n.statements);break;case J.IF_ELSE:{let s=n;r(s.then),s.else&&r(s.else);break}case J.WHILE_LOOP:r(n.body);break}}return r(e),t}function Kn(e){let t=Fe(e.statements??[]);if(e.teardown)for(let r of Fe(e.teardown))t.add(r);return t}function ht(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ue(e,t,r,o=0,n){let s=" ".repeat(o),a=[],i=Kn(e),c=ht(i),l=n?.only?"test.only":"test";a.push(`${s}${l}('${t}', async (${c}) => {`),n?.skip===!0?a.push(`${s} test.skip();`):typeof n?.skip=="string"&&a.push(`${s} test.skip(true, '${se(n.skip)}');`),n?.fail===!0?a.push(`${s} test.fail();`):typeof n?.fail=="string"&&a.push(`${s} test.fail(true, '${se(n.fail)}');`),n?.slow&&a.push(`${s} test.slow();`),n?.timeout&&a.push(`${s} test.setTimeout(${n.timeout});`);let p=e.teardown&&e.teardown.length>0,u=o+1;if(p){if(a.push(`${s} try {`),e.statements&&e.statements.length>0){a.push(`${s} // Test steps`);let g=Q(e.statements,u+1,r);a.push(...g)}a.push(`${s} } finally {`),a.push(`${s} // Teardown`);let h=Q(e.teardown,u+1,r,"teardown");a.push(...h),a.push(`${s} }`)}else if(e.statements&&e.statements.length>0){a.push(`${s} // Test steps`);let h=Q(e.statements,u,r);a.push(...h)}return a.push(`${s}});`),a}function Mr(e,t,r){let o=[],n=Lr(t),s=Fe(n),a=ht(s);return o.push(`test.${e}(async (${a}) => {`),o.push(...Q(n,1,r,e)),o.push("});"),o}function Ne(e,t,r,o){let n=" ".repeat(o),s=[],a=Lr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};s.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),s.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),s.push(...Q(a,o+1,c,e)),s.push(`${n} await page.close();`),s.push(`${n}});`)}else{let c=Fe(a),l=ht(c);s.push(`${n}test.${e}(async (${l}) => {`),s.push(...Q(a,o+1,r,e)),s.push(`${n}});`)}return s}function Lr(e){let r=An({goal:"_hook",statements:e});return R(r).statements??[]}function Cr(e,t){let r=ve(e);for(let[o,n]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(n));return R(r)}function Rr(){return["import { test, expect } from 'shiplightai/fixture';"]}function Nr(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let o=Array.from(t);e.splice(r,0,...o)}}function Ur(e,t){let r={expandingPaths:new Set([Fr(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=ce(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=ce(o.teardown,t,r));for(let n of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[n])&&(o[n]=ce(o[n],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ce(e,t,r){let o=[];for(let n of e)if(qn(n)){let s=Zn(n,t,r);o.push(...s)}else o.push(Qn(n,t,r));return o}function qn(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Zn(e,t,r){if(r.depth>=Or)throw new Error(`Template expansion exceeded maximum depth of ${Or}. Check for deeply nested or circular template references.`);let o=Fr(Jn(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 n;try{n=Yn(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let s=Ir(n);if(!s||typeof s!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let a=s.params||[],i=e.params||{};for(let p of a)if(!(p in i))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${a.join(", ")}]`);let c=s.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(i).length>0){let u=Xn(c);for(let[h,g]of Object.entries(i))u=u.split(`<<${h}>>`).join(String(g));c=Ir(u)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return ce(c,o,l)}function Qn(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=ce(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=ce(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=ce(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=ce(o.DO,t,r)),o}function gt(e,t){let r=zn(e),o=r?.name,n=r?.tags,s=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 ft('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ts(r,o,n,s,t)}return es(r,o,n,s,t)}function es(e,t,r,o,n){let s=e?.beforeEach,a=e?.afterEach,i=Br(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,u=e?.only,h=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 ft(`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(n&&e&&typeof e=="object"){let f=Ur(e,n);e=f.doc,g=f.referencedTemplatePaths}let m=Dr(e),d=R(m);return n&&(ge(d.statements??[],n,"main"),d.teardown&&ge(d.teardown,n,"teardown")),{testFlow:d,name:t,tags:r,use:o,beforeEach:s,afterEach:a,parameters:i,timeout:c,skip:l,fail:p,only:u,slow:h,referencedTemplatePaths:g}}function ts(e,t,r,o,n){let s=e.suite;if(!Array.isArray(s.tests)||s.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=s.beforeAll,i=s.afterAll,c=s.beforeEach,l=s.afterEach,p=[],u=s.tests.map(m=>{if(!m.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(m.statements)||m.statements.length===0)throw new Error(`Suite test "${m.name}" must have a non-empty "statements" array.`);let d={goal:m.name,statements:m.statements};m.teardown&&(d.teardown=m.teardown);let f=[],w=d;if(n&&typeof d=="object"){let k=Ur(d,n);w=k.doc,f=k.referencedTemplatePaths,p.push(...f)}let S=Dr(w),b=R(S),_=Br(m.parameters);return{testFlow:b,name:m.name,tags:Array.isArray(m.tags)?m.tags:void 0,parameters:_,timeout:m.timeout,skip:m.skip,fail:m.fail,only:m.only,slow:m.slow}}),h=s.base_url,g=h?{...o,baseURL:h}:o;return{suite:{beforeAll:a,afterAll:i,beforeEach:c,afterEach:l,tests:u},name:t,tags:r,use:g,referencedTemplatePaths:p}}function Br(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 ge(e,t,r){for(let o=0;o<e.length;o++){let n=e[o],s=`${r}.${o}`,a=n.description||"";if(n.uid=rs(t,s,a),n.type===J.STEP)ge(n.statements,t,s);else if(n.type===J.IF_ELSE){let i=n;ge(i.then,t,`${s}.then`),i.else&&ge(i.else,t,`${s}.else`)}else n.type===J.WHILE_LOOP&&ge(n.body,t,`${s}.body`)}}function rs(e,t,r){let o=Vn("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 Hr(e,t){let r;try{r=os(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return as(r,e,t)}function as(e,t,r){let o=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),s=o||n?null:ct(e);if(s&&!s.valid)return{valid:!1,errors:s.errors,warnings:[],stats:s.stats};let a,i,c=[];try{let l=r?.parsed??gt(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?a=Gn(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):a=jn(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 h=a.split(`
110
- `).filter(g=>!g.startsWith("import ")).join(`
107
+ `}async function sn(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let s=t();s.once("error",()=>n(!1)),s.once("listening",()=>{s.close(()=>n(!0))}),s.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function tt(e){let{yamlFilePath:t,configPath:r,tempSuffix:o="",headed:n}=e,s=Z.dirname(r),a=await sn(16174),i;if(!ne.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let w=on(ne.readFileSync(t,"utf-8"));w?.use&&typeof w.use=="object"&&!Array.isArray(w.use)&&(i=w.use),w?.base_url&&!i?.baseURL&&(i={...i,baseURL:w.base_url}),w?.settings?.auto_dismiss_modal!==void 0&&(i={...i,autoDismissModal:!!w.settings.auto_dismiss_modal})}catch(w){console.error("[debugger] Could not parse YAML for `use` block:",w)}let c=Z.dirname(Z.resolve(t)),l=o?`-${o}`:"",p=Z.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),u=nn(t,a,i,s);ne.writeFileSync(p,u);let h=()=>{try{ne.unlinkSync(p)}catch{}},m=["playwright","test",p,...n?["--headed"]:[]],g=rn("npx",m,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:s,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});g.stdout?.on("data",w=>{process.stderr.write(w)}),g.stderr?.on("data",w=>{process.stderr.write(w)});let d=()=>{g.killed||g.kill("SIGTERM")};process.on("SIGTERM",d),process.on("SIGINT",d),process.on("exit",d),g.on("close",w=>{process.removeListener("SIGTERM",d),process.removeListener("SIGINT",d),process.removeListener("exit",d),h(),w!==0&&w!==null&&console.error(`[debugger] Playwright process exited with code ${w}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let S=["127.0.0.1","::1"];async function f(w){try{let _=w.includes(":")?`[${w}]`:w,T=await fetch(`http://${_}:${a}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let b=null;for(let w=0;w<180;w++){if(g.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${g.exitCode} before sandbox was ready`);for(let _ of S)if(await f(_)){b=_;break}if(b){console.error(`[debugger] Playwright sandbox ready on ${b}:${a}`);break}if(w===179)throw d(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(_=>setTimeout(_,1e3))}if(!b)throw d(),h(),new Error("Sandbox poll finished without a reachable host");return{port:a,host:b,pid:g.pid??0,cleanup:async()=>{d(),h()}}}var rt=x(()=>{"use strict"});import{z as y}from"zod";var Vt,ot,zt,de,Yt,Jt,Xt,F,qt,Me,nt,st=x(()=>{"use strict";Vt=y.enum(["JS_CODE","AI_MODE"]),ot=y.object({type:Vt,expression:y.string()}),zt=y.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),de=y.object({uid:y.string(),type:zt,comment:y.string().optional()}),Yt=y.object({action_data:y.object({action_name:y.string(),kwargs:y.record(y.any()).optional(),args:y.array(y.any()).optional()}),action_description:y.string().optional(),url:y.string().optional(),xpath:y.string().nullable().optional(),locator:y.string().nullable().optional(),css_selector:y.string().nullable().optional(),unique_selector:y.string().nullable().optional(),element_index:y.number().nullable().optional(),frame_path:y.array(y.any()).optional(),artifacts:y.record(y.any()).optional(),feedback:y.string().optional(),original_browser_use_action:y.any().optional()}).passthrough(),Jt=de.extend({type:y.literal("DRAFT"),description:y.string()}),Xt=de.extend({type:y.literal("ACTION"),description:y.string(),action_entity:Yt.optional(),locator:y.string().optional(),use_pure_vision:y.boolean().optional()}),F=y.lazy(()=>y.union([Jt,Xt,de.extend({type:y.literal("STEP"),description:y.string().optional().default(""),statements:y.array(F),reference_id:y.number().optional()}),de.extend({type:y.literal("IF_ELSE"),description:y.string().optional(),condition:ot,then:y.array(F),else:y.array(F).optional()}),de.extend({type:y.literal("WHILE_LOOP"),description:y.string().optional(),condition:ot,body:y.array(F),timeout_ms:y.number().optional()})])),qt=y.object({name:y.string(),statements:y.array(F),teardown:y.array(F).optional(),skip:y.union([y.boolean(),y.string()]).optional(),timeout:y.number().optional(),fail:y.union([y.boolean(),y.string()]).optional(),only:y.boolean().optional(),slow:y.boolean().optional()}),Me=y.object({tests:y.array(qt).min(1),beforeAll:y.array(F).optional(),afterAll:y.array(F).optional(),beforeEach:y.array(F).optional(),afterEach:y.array(F).optional()}),nt=y.object({comment:y.string().optional(),version:y.string().optional(),goal:y.string().optional(),url:y.string().optional(),baseURL:y.string().optional(),final_feedback:y.string().optional(),completed:y.boolean().optional(),success:y.boolean().optional(),statements:y.array(F).optional(),teardown:y.array(F).optional(),last_modified_at:y.string().optional(),testGroup:Me.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 an,parse as tr,parseAllDocuments as yi,parseDocument as cn,Document as ln,isMap as fe,isSeq as Y}from"yaml";import{v4 as z}from"uuid";function rr(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(j)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(j)),r}function ve(e,t){if(e.testGroup)return nr(e,t);let r=rr(e,t),o=new ln(r);return e.comment&&(o.commentBefore=e.comment),Zt(o,e.statements??[]),e.teardown&&Zt(o,e.teardown,"teardown"),o.toString(or)}function Zt(e,t,r="statements"){let o=e.contents;if(!o||!fe(o))return;let n=o.get(r,!0);Y(n)&&Se(n,t)}function Se(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:n.commentBefore=o.comment),fe(n)){let s=n;if(o.type==="STEP"){let a=s.get("statements",!0);Y(a)&&Se(a,o.statements)}else if(o.type==="IF_ELSE"){let a=s.get("THEN",!0);Y(a)&&Se(a,o.then);let i=s.get("ELSE",!0);Y(i)&&o.else&&Se(i,o.else)}else if(o.type==="WHILE_LOOP"){let a=s.get("DO",!0);Y(a)&&Se(a,o.body)}}}}function nr(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 n={};return e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(j)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(j)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(j)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(j)),n.tests=r.tests.map(s=>{let a={name:s.name};return s.skip!==void 0&&(a.skip=s.skip),s.timeout!==void 0&&(a.timeout=s.timeout),s.fail!==void 0&&(a.fail=s.fail),s.only!==void 0&&(a.only=s.only),s.slow!==void 0&&(a.slow=s.slow),a.statements=s.statements.map(j),s.teardown&&s.teardown.length>0&&(a.teardown=s.teardown.map(j)),a}),o.suite=n,an(o,or)}function j(e){switch(e.type){case"DRAFT":return pn(e);case"ACTION":return un(e);case"STEP":return dn(e);case"IF_ELSE":return hn(e);case"WHILE_LOOP":return fn(e)}}function pn(e){return{intent:e.description}}function un(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let i=r?.statement;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:i,js:c}:{VERIFY:i}}}if(t==="go_to_url"){let i=r?.url;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:i};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let i=r?.code;if(typeof i=="string"&&e.description)return{intent:e.description,js:i}}if(t==="ai_wait_until"){let i=r?.condition;if(typeof i=="string"){let c={WAIT_UNTIL:i};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let i=r?.seconds,l={WAIT:e.description||`Wait ${i}s`};return typeof i=="number"&&(l.seconds=i),l}if(t==="js_code"){let i=r?.code;if(typeof i=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:i}}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 n={intent:e.description,action:o.action_name},s=e.locator??e.action_entity.locator;s&&(n.locator=s);let a=e.action_entity.xpath;if(a&&(n.xpath=a),e.use_pure_vision&&(n.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[i,c]of Object.entries(o.kwargs))n[i]=c;return o.args&&o.args.length>0&&(n.args=o.args),n}function dn(e){let t={STEP:e.description,statements:e.statements.map(j)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function hn(e){let t={IF:sr(e.condition),THEN:e.then.map(j)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(j)),t}function fn(e){let t={WHILE:sr(e.condition),DO:e.body.map(j)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function sr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function xe(e){try{let t=tr(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 it(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(it);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let o={};for(let[n,s]of Object.entries(t))o[n]=it(s);return o}function R(e){if(e.length>Qt)throw new Error(`YAML input too large (${e.length} bytes, max ${Qt})`);let t=it(tr(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return gn(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=nt.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let n=o.data;return Ie(e,n),n}function gn(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 n={tests:r.map(i=>{if(!i.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(i.statements)||i.statements.length===0)throw new Error(`Suite test "${i.name}" must have a non-empty "statements" array`);let c={name:i.name,statements:G(i.statements)};return Array.isArray(i.teardown)&&i.teardown.length>0&&(c.teardown=G(i.teardown)),i.skip!==void 0&&(c.skip=i.skip),typeof i.timeout=="number"&&(c.timeout=i.timeout),i.fail!==void 0&&(c.fail=i.fail),i.only===!0&&(c.only=!0),i.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=G(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=G(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=G(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=G(t.afterEach));let s=Me.safeParse(n);if(!s.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(s.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:s.data}}function G(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(mn)}function mn(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 yn(t);if("WHILE"in t)return wn(t);if("STEP"in t)return bn(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:z(),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,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,s={url:typeof r=="string"?r:String(r)};return o&&(s.new_tab=!0),n!==void 0&&(s.timeout_seconds=n),{uid:z(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:s}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:z(),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:z(),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:z(),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:z(),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 er({...o,action:"function",functionName:r})}if("action"in t)return er(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:z(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function ir(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 yn(e){let t=ir(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:z(),type:"IF_ELSE",condition:t,then:G(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=G(e.ELSE)),o}function wn(e){let t=ir(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:z(),type:"WHILE_LOOP",condition:t,body:G(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function bn(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:z(),type:"STEP",description:t,statements:G(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function er(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,n=typeof e.xpath=="string"?e.xpath:void 0,s=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[l,p]of Object.entries(e))Sn.has(l)||(a[l]=p);let i={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};o&&(i.locator=o),n&&(i.xpath=n);let c={uid:z(),type:"ACTION",description:r,action_entity:i};return s&&(c.use_pure_vision=!0),c}function Ie(e,t){let r;try{r=cn(e)}catch{return}let o=r.contents;if(!o||!fe(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 n=o,s=n.get("statements",!0);Y(s)&&t.statements&&he(s,t.statements);let a=n.get("teardown",!0);Y(a)&&t.teardown&&he(a,t.teardown)}function he(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 n=t[r];if(n.type==="STEP"&&fe(o)){let s=o.get("statements",!0);Y(s)&&he(s,n.statements)}else if(n.type==="IF_ELSE"&&fe(o)){let s=o.get("THEN",!0);Y(s)&&he(s,n.then);let a=o.get("ELSE",!0);Y(a)&&n.else&&he(a,n.else)}else if(n.type==="WHILE_LOOP"&&fe(o)){let s=o.get("DO",!0);Y(s)&&he(s,n.body)}}}var or,Qt,Sn,Oe=x(()=>{"use strict";st();or={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Qt=1024*1024;Sn=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as xi,stringify as _i}from"yaml";var ar=x(()=>{"use strict";Oe()});var at,Le,Ce=x(()=>{"use strict";at=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},Le=e=>{let t=[],r=o=>{for(let n of o){t.push(n);let s=at(n);for(let a of s)r(a.statements)}};return r(e),t}});function pr(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let n=o.action_entity?.action_data?.action_name??"";lr.has(n)||t++}return{action:t,draft:r}}function xn(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function cr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function ct(e,t){let r=t?.coverageThreshold??vn,o=[],n=[],s;try{s=R(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}s.goal||o.push('Missing required field: "goal"'),s.statements?.length||o.push('Missing required field: "statements"');let a=[...Le(s.statements??[]),...s.teardown?Le(s.teardown):[]],{action:i,draft:c}=pr(a),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let h of a){if(h.type==="ACTION"){let m=h,g=m.action_entity?.action_data?.action_name??"";if(g==="js_code"||g==="js_action"||g==="verify"||g==="ai_assert"){let d=m.action_entity?.action_data?.kwargs?.code;if(typeof d=="string"){let S=xn(d);if(S){let f=m.description||g;o.push(`Invalid JS in "${f}": ${S}. ${l}`)}}}}if(h.type==="IF_ELSE"){let m=h;if(m.condition.type==="JS_CODE"){let g=cr(m.condition.expression);g&&o.push(`Invalid JS in IF condition "${m.condition.expression}": ${g}. ${l}`)}}if(h.type==="WHILE_LOOP"){let m=h;if(m.condition.type==="JS_CODE"){let g=cr(m.condition.expression);g&&o.push(`Invalid JS in WHILE condition "${m.condition.expression}": ${g}. ${l}`)}}}let p=i+c,u=p>0?Math.round(i/p*100):0;return p>0&&u/100<r&&n.push(`Low action coverage: ${i}/${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:n,stats:{total:p,action:i,draft:c,coverage:u}}}var vn,lr,ur=x(()=>{"use strict";Oe();Ce();vn=.5,lr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});function lt(){return{version:"1.0",entries:{}}}var dr=x(()=>{"use strict";Ce()});import{v4 as Oi}from"uuid";var hr=x(()=>{"use strict"});var J,pt,fr=x(()=>{"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||{}),pt=18e4});var Re=x(()=>{"use strict"});var gr=x(()=>{"use strict";Re()});var mr,Tn,yr,wr,_e,kn,Pn,br=x(()=>{"use strict";mr=112,Tn=1080-mr,yr={"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"}},wr={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"]},_e=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),wr[e].map(o=>yr[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},kn={desktop:{label:"Desktop",type:"desktop",devices:_e("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:_e("mobile")}},Pn={desktop:{label:"Desktop",type:"desktop",devices:_e("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:_e("mobile",!0)}}});var Sr=x(()=>{"use strict"});var vr=x(()=>{"use strict"});var xr=x(()=>{"use strict"});var _r=x(()=>{"use strict"});var Tr=x(()=>{"use strict"});var kr=x(()=>{"use strict"});var Pr=x(()=>{"use strict"});var Er=x(()=>{"use strict";Re()});var ae=x(()=>{"use strict";ar();ur();Oe();st();dr();hr();Ce();fr();gr();br();Sr();vr();xr();_r();Tr();kr();Pr();Er();Re()});import{stringify as An}from"yaml";import{createHash as Vn}from"crypto";import{parse as zn,stringify as Dr}from"yaml";import{readFileSync as Yn}from"fs";import{resolve as Fr,dirname as Jn}from"path";import{parse as Ir,stringify as Xn}from"yaml";import{readFileSync as os,writeFileSync as ns,mkdirSync as ss}from"fs";import{dirname as is}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function A(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function $n(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Mn(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 De(e){let t=$n(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=Mn(e);if(o){let n=JSON.stringify(o);return`${t}.locator(${n}).first()`}return null}function Ar(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:In.includes(t)}function Ln(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!On.includes(t)}function v(e,t){L.set(e,t)}function Cn(e){return L.get(e)}function pe(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(n=>` ${n},`),"});"]}function $r(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"],n=["undefined","null","true","false"],s=r.map(a=>o.includes(a)||n.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${s.join(", ")})`}function Q(e,t,r,o="main"){let n=[];for(let s=0;s<e.length;s++){let a=e[s],i=`${o}.${s}`,c=Rn(a,t,i,r);c.length>0&&(n.push(...c),s<e.length-1&&n.push(""))}return n}function Rn(e,t,r,o){let n=" ".repeat(t);switch(e.type){case"DRAFT":return Nn(e,t,r,o);case"ACTION":return Dn(e,t,r,o);case"STEP":return Fn(e,t,r,o);case"IF_ELSE":return Un(e,t,r,o);case"WHILE_LOOP":return Bn(e,t,r,o);default:return[`${n}// Unknown statement type: ${e.type}`]}}function Nn(e,t,r,o){let n=" ".repeat(t),s=e.description?.trim()||"";if(!s)return[`${n}// ${r}: Skipping - no description`];if(o.noAgent)return[`${n}// ${r}: ${A(s)}`,`${n}// DRAFT: ${A(s)} (requires agent - skipped in hook)`];let a=JSON.stringify(s);return[`${n}// ${r}: ${A(s)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${a}, '${r}');`]}function Dn(e,t,r,o){let n=" ".repeat(t),s=e.description,a=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!s)return[`${n}// ${r}: Skipping - no description`];if(o.noAgent)return[`${n}// ${r}: ${A(s)}`,`${n}// DRAFT: ${A(s)} (requires agent - skipped in hook)`];let w=JSON.stringify(s),_=!!e.use_pure_vision;return[`${n}// ${r}: ${A(s)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${w}, '${r}', ${_});`]}let l=e.locator?{...c,locator:e.locator}:c;s&&s!==l.action_description&&(l={...l,action_description:s});let p=l.action_data?.action_name||"",u=l.action_description||"",h=Cn(p);if(!h)return[`${n}// ${r}: Unknown action: ${p}`];let m={imports:o.imports},g=h(l,r,m);if(o.noAgent){if(Ar(l))return[`${n}// ${r}: ${A(u)}`,`${n}// AI action: ${A(u)} (requires agent - skipped in hook)`];let w=Hn(l,p,n,r);return w||[`${n}// ${r}: ${A(u)}`,...g.map(_=>`${n}${_}`)]}if(Ar(l))return[`${n}// ${r}: ${A(u)}`,`${n}page = agent.agentServices.validatePage(page);`,...g.map(w=>`${n}${w}`)];let d=JSON.stringify(u),S=g.map(w=>`${n} ${w}`),f=Ln(l),b=a?`'${a}'`:"undefined";return[`${n}// ${r}: ${A(u)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...S,`${n}}, ${d}, '${r}', ${b}, ${f});`]}function Fn(e,t,r,o){let n=" ".repeat(t),s=[];e.description&&e.description.trim()&&s.push(`${n}// Step: ${A(e.description)}`);let a=Q(e.statements,t,o,r);return s.push(...a),s}function Un(e,t,r,o){let n=" ".repeat(t),s=[];if(s.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")s.push(`${n}if (${e.condition.expression}) {`);else{s.push(`${n}// AI Condition: ${A(e.condition.expression)}`);let i=JSON.stringify(e.condition.expression);s.push(`${n}if (await agent.evaluate(page, ${i}, "${r}")) {`)}let a=Q(e.then,t+1,o,`${r}.then`);if(s.push(...a),e.else&&e.else.length>0){s.push(`${n}} else {`);let i=Q(e.else,t+1,o,`${r}.else`);s.push(...i)}return s.push(`${n}}`),s}function Bn(e,t,r,o){let n=" ".repeat(t),s=[];s.push(`${n}// ${r}: Loop`);let a=e.timeout_ms??pt,i=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${i}s`:`While loop exceeded default timeout of ${i}s`,l=`loop_${r.replace(/\./g,"_")}`;if(s.push(`${n}const ${l}_start = Date.now();`),s.push(`${n}const ${l}_timeout = ${a};`),s.push(`${n}const ${l}_check = () => {`),s.push(`${n} if (Date.now() - ${l}_start > ${l}_timeout) {`),s.push(`${n} throw new Error('${c}');`),s.push(`${n} }`),s.push(`${n} return true;`),s.push(`${n}};`),e.condition.type==="JS_CODE")s.push(`${n}while (${l}_check() && (${e.condition.expression})) {`);else{s.push(`${n}// AI Loop Condition: ${A(e.condition.expression)}`);let u=JSON.stringify(e.condition.expression);s.push(`${n}while (${l}_check() && await agent.evaluate(page, ${u}, "${r}")) {`)}let p=Q(e.body,t+1,o,`${r}.body`);return s.push(...p),s.push(`${n}}`),s}function Hn(e,t,r,o){let n=e.action_description||"",s=e.action_data?.kwargs||{},a=s.timeout_ms??dt;switch(t){case"go_to_url":case"open_tab":{let i=s.url||"";return[`${r}// ${o}: ${A(n)}`,`${r}await page.goto(${JSON.stringify(i)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${A(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${A(n)}`,`${r}await page.goForward();`];case"input_text":{let i=s.text||"",c=De(e);return c?[`${r}// ${o}: ${A(n)}`,`${r}await ${c}.fill(${JSON.stringify(i)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let i=s.text||s.label||"",c=De(e);return c?[`${r}// ${o}: ${A(n)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(i)} }, { timeout: ${a} });`]:null}default:return null}}function jn(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Rr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,s={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...Mr("beforeEach",t.beforeEach,s)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let i=t?.testName||e.goal||"Generated test",c=ut(t?.tags);for(let l of t.parameters){let p=Cr(e,l.values);r.push(...Ue(p,`${c}${se(i)} [${se(l.name)}]`,s,0,a)),r.push("")}}else{let i=t?.testName||e.goal||"Generated test",c=ut(t?.tags);r.push(...Ue(e,`${c}${se(i)}`,s,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...Mr("afterEach",t.afterEach,s))),Nr(r,n),r.join(`
108
+ `)}function Gn(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...Rr()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,s={imports:n,actionEntityStore:t?.actionEntityStore},a=t?.testName||"Test Suite",i=ut(t?.tags);r.push(`test.describe.serial('${i}${se(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ne("beforeAll",e.beforeAll,s,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ne("beforeEach",e.beforeEach,s,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 h of p.parameters){let m=Cr(p.testFlow,h.values);r.push(...Ue(m,`${se(p.name)} [${se(h.name)}]`,s,1,u)),r.push("")}else r.push(...Ue(p.testFlow,se(p.name),s,1,u)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ne("afterEach",e.afterEach,s,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ne("afterAll",e.afterAll,s,1)),r.push("});"),Nr(r,n),r.join(`
109
+ `)}function ut(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Fe(e){let t=new Set;function r(o){for(let n of o)switch(n.type){case J.ACTION:{let a=n.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let i of a.args)typeof i=="string"&&Wn.includes(i)&&t.add(i);break}case J.STEP:r(n.statements);break;case J.IF_ELSE:{let s=n;r(s.then),s.else&&r(s.else);break}case J.WHILE_LOOP:r(n.body);break}}return r(e),t}function Kn(e){let t=Fe(e.statements??[]);if(e.teardown)for(let r of Fe(e.teardown))t.add(r);return t}function ht(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ue(e,t,r,o=0,n){let s=" ".repeat(o),a=[],i=Kn(e),c=ht(i),l=n?.only?"test.only":"test";a.push(`${s}${l}('${t}', async (${c}) => {`),n?.skip===!0?a.push(`${s} test.skip();`):typeof n?.skip=="string"&&a.push(`${s} test.skip(true, '${se(n.skip)}');`),n?.fail===!0?a.push(`${s} test.fail();`):typeof n?.fail=="string"&&a.push(`${s} test.fail(true, '${se(n.fail)}');`),n?.slow&&a.push(`${s} test.slow();`),n?.timeout&&a.push(`${s} test.setTimeout(${n.timeout});`);let p=e.teardown&&e.teardown.length>0,u=o+1;if(p){if(a.push(`${s} try {`),e.statements&&e.statements.length>0){a.push(`${s} // Test steps`);let m=Q(e.statements,u+1,r);a.push(...m)}a.push(`${s} } finally {`),a.push(`${s} // Teardown`);let h=Q(e.teardown,u+1,r,"teardown");a.push(...h),a.push(`${s} }`)}else if(e.statements&&e.statements.length>0){a.push(`${s} // Test steps`);let h=Q(e.statements,u,r);a.push(...h)}return a.push(`${s}});`),a}function Mr(e,t,r){let o=[],n=Lr(t),s=Fe(n),a=ht(s);return o.push(`test.${e}(async (${a}) => {`),o.push(...Q(n,1,r,e)),o.push("});"),o}function Ne(e,t,r,o){let n=" ".repeat(o),s=[],a=Lr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};s.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),s.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),s.push(...Q(a,o+1,c,e)),s.push(`${n} await page.close();`),s.push(`${n}});`)}else{let c=Fe(a),l=ht(c);s.push(`${n}test.${e}(async (${l}) => {`),s.push(...Q(a,o+1,r,e)),s.push(`${n}});`)}return s}function Lr(e){let r=An({goal:"_hook",statements:e});return R(r).statements??[]}function Cr(e,t){let r=ve(e);for(let[o,n]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(n));return R(r)}function Rr(){return["import { test, expect } from 'shiplightai/fixture';"]}function Nr(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let o=Array.from(t);e.splice(r,0,...o)}}function Ur(e,t){let r={expandingPaths:new Set([Fr(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=ce(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=ce(o.teardown,t,r));for(let n of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[n])&&(o[n]=ce(o[n],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ce(e,t,r){let o=[];for(let n of e)if(qn(n)){let s=Zn(n,t,r);o.push(...s)}else o.push(Qn(n,t,r));return o}function qn(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Zn(e,t,r){if(r.depth>=Or)throw new Error(`Template expansion exceeded maximum depth of ${Or}. Check for deeply nested or circular template references.`);let o=Fr(Jn(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 n;try{n=Yn(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let s=Ir(n);if(!s||typeof s!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let a=s.params||[],i=e.params||{};for(let p of a)if(!(p in i))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${a.join(", ")}]`);let c=s.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(i).length>0){let u=Xn(c);for(let[h,m]of Object.entries(i))u=u.split(`<<${h}>>`).join(String(m));c=Ir(u)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return ce(c,o,l)}function Qn(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=ce(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=ce(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=ce(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=ce(o.DO,t,r)),o}function gt(e,t){let r=zn(e),o=r?.name,n=r?.tags,s=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 ft('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ts(r,o,n,s,t)}return es(r,o,n,s,t)}function es(e,t,r,o,n){let s=e?.beforeEach,a=e?.afterEach,i=Br(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,u=e?.only,h=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 ft(`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 m=[];if(n&&e&&typeof e=="object"){let S=Ur(e,n);e=S.doc,m=S.referencedTemplatePaths}let g=Dr(e),d=R(g);return n&&(ge(d.statements??[],n,"main"),d.teardown&&ge(d.teardown,n,"teardown")),{testFlow:d,name:t,tags:r,use:o,beforeEach:s,afterEach:a,parameters:i,timeout:c,skip:l,fail:p,only:u,slow:h,referencedTemplatePaths:m}}function ts(e,t,r,o,n){let s=e.suite;if(!Array.isArray(s.tests)||s.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=s.beforeAll,i=s.afterAll,c=s.beforeEach,l=s.afterEach,p=[],u=s.tests.map(g=>{if(!g.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(g.statements)||g.statements.length===0)throw new Error(`Suite test "${g.name}" must have a non-empty "statements" array.`);let d={goal:g.name,statements:g.statements};g.teardown&&(d.teardown=g.teardown);let S=[],f=d;if(n&&typeof d=="object"){let T=Ur(d,n);f=T.doc,S=T.referencedTemplatePaths,p.push(...S)}let b=Dr(f),w=R(b),_=Br(g.parameters);return{testFlow:w,name:g.name,tags:Array.isArray(g.tags)?g.tags:void 0,parameters:_,timeout:g.timeout,skip:g.skip,fail:g.fail,only:g.only,slow:g.slow}}),h=s.base_url,m=h?{...o,baseURL:h}:o;return{suite:{beforeAll:a,afterAll:i,beforeEach:c,afterEach:l,tests:u},name:t,tags:r,use:m,referencedTemplatePaths:p}}function Br(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 ge(e,t,r){for(let o=0;o<e.length;o++){let n=e[o],s=`${r}.${o}`,a=n.description||"";if(n.uid=rs(t,s,a),n.type===J.STEP)ge(n.statements,t,s);else if(n.type===J.IF_ELSE){let i=n;ge(i.then,t,`${s}.then`),i.else&&ge(i.else,t,`${s}.else`)}else n.type===J.WHILE_LOOP&&ge(n.body,t,`${s}.body`)}}function rs(e,t,r){let o=Vn("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 Hr(e,t){let r;try{r=os(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return as(r,e,t)}function as(e,t,r){let o=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),s=o||n?null:ct(e);if(s&&!s.valid)return{valid:!1,errors:s.errors,warnings:[],stats:s.stats};let a,i,c=[];try{let l=r?.parsed??gt(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?a=Gn(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):a=jn(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 h=a.split(`
110
+ `).filter(m=>!m.startsWith("import ")).join(`
111
111
  `);new Function(h),i=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),ss(is(i),{recursive:!0}),ns(i,a)}catch(l){let p=l instanceof ft?"":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:s?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:s?.warnings??[],stats:s?.stats??{total:0,action:0,draft:0,coverage:0},specFile:i,referencedTemplatePaths:c}}var dt,In,On,L,Wn,Or,ft,mt=x(()=>{"use strict";ae();ae();ae();ae();dt=5e3;In=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],On=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];L=new Map;v("click",e=>{let t=De(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??dt;return[`await ${t}.click({ timeout: ${r} });`]});v("click_element",L.get("click"));v("click_element_by_index",L.get("click"));v("double_click",e=>pe("double_click",e));v("double_click_on_element",L.get("double_click"));v("right_click",e=>pe("right_click",e));v("right_click_on_element",L.get("right_click"));v("hover",e=>pe("hover",e));v("hover_element_by_index",L.get("hover"));v("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return pe("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("fill",L.get("input_text"));v("clear_input",e=>pe("clear_input",e));v("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});v("send_keys",L.get("press"));v("send_keys_on_element",e=>{let t=De(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??dt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});v("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return pe("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("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)})');`]});v("scroll_down",L.get("scroll"));v("scroll_up",L.get("scroll"));v("scroll_element",L.get("scroll"));v("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});v("scroll_on_element",e=>pe("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));v("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)} } },`,"});"]});v("open_tab",L.get("go_to_url"));v("go_back",()=>['await agent.execAction("go_back", page, {});']);v("reload_page",()=>['await agent.execAction("reload_page", page, {});']);v("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);v("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);v("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",n=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&n){let a=r.code.split(`
112
112
  `),i=JSON.stringify(n);return["{ const _t = Date.now(); try {",...a.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${i}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${i}\`);`,` await agent.assert(page, ${i}, ${JSON.stringify(t||"")});`,"} }"]}return o?r.code.split(`
113
113
  `):n?[`await agent.assert(page, ${JSON.stringify(n)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});v("ai_assert",L.get("verify"));v("assert",L.get("verify"));v("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let o=JSON.stringify(r),n=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${o}, '${t||""}', ${n});`]});v("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"]});v("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 n=JSON.stringify(r),s=JSON.stringify(o);return[`await agent.extract(page, ${n}, ${s}, '${t||""}');`]});v("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"]});v("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)} } },`,"});"]});v("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],o=t.split(`
114
114
  `);for(let n of o)r.push(` ${n}`);return r.push("}"),r});v("function",(e,t,r)=>{let o=e.action_data?.kwargs||{},n=o.functionName;if(n&&n.includes("#")){let[a,i]=n.split("#");if(a&&i){let c=a.replace(/\.(ts|js|mjs)$/,""),l=`import { ${i} } from '${c}';`;r?.imports?.add(l);let p={...o,functionName:i},u=$r(p);return u?[u.endsWith(";")?u:`${u};`]:["// Skipping function: invalid export pattern"]}}let s=$r(o);return s?[s.endsWith(";")?s:`${s};`]:["// Skipping function: missing functionName"]});v("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)} } },`,"});"]});v("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(n=>` ${n},`),"});"]});v("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} } },`,"});"]);v("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} } },`,"});"]);v("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} } },`,"});"]});v("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},`),"});"]});v("done",()=>["// Done - no action needed"]);v("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
115
- `):["// Skipping js_action: missing code"]});Wn=["testContext","request"];Or=5;ft=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as cs}from"express";import*as I from"fs/promises";import*as C from"path";import{stringify as ls}from"yaml";function Be(e){if(!e||e.length===0)return[];let r=ls({goal:"_hook",statements:e});return R(r).statements??[]}async function jr(e){try{let t=await I.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 jr(C.join(e,r.name))))return!0}catch{}return!1}function Gr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:n}=e,s=cs();function a(){return C.join(o??t,"fixtures")}s.get("/api/files",async(c,l)=>{try{let p=typeof c.query.dir=="string"?c.query.dir:t,u=C.resolve(p),h=await I.readdir(u,{withFileTypes:!0}),g=[];for(let d of h)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let f=C.join(u,d.name);await jr(f)&&g.push({name:d.name,type:"directory",path:f})}else d.isFile()&&d.name.endsWith(".test.yaml")&&g.push({name:d.name,type:"file",path:C.join(u,d.name)});g.sort((d,f)=>d.type!==f.type?d.type==="directory"?-1:1:d.name.localeCompare(f.name));let m=C.dirname(u);l.json({dir:u,parent:m!==u?m:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(p){console.error("[debugger] Error listing files:",p),l.status(500).json({error:p.message})}});function i(c){if(typeof c=="string"&&c){let l=C.resolve(c);return l.endsWith(".test.yaml")?{filePath:l}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return s.get("/api/test-flow",async(c,l)=>{let p=i(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let u=p.filePath;try{n?.(u);let h=await I.readFile(u,"utf-8"),g=await I.stat(u),m=xe(h),d=gt(h,u);if(d.suite){let f=d.suite,w={tests:f.tests.map(b=>({name:b.name,statements:b.testFlow.statements??[],teardown:b.testFlow.teardown,skip:b.skip,timeout:b.timeout,fail:b.fail,only:b.only,slow:b.slow})),beforeAll:Be(f.beforeAll),afterAll:Be(f.afterAll),beforeEach:Be(f.beforeEach),afterEach:Be(f.afterEach)},S={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:w};l.json({isSuite:!0,testFlow:S,metadata:m,name:d.name,tags:d.tags,use:d.use,filePath:u,fileName:C.basename(u),lastModified:g.mtimeMs})}else{let f=d.testFlow;Ie(h,f),l.json({isSuite:!1,testFlow:f,metadata:m,name:d.name,tags:d.tags,use:d.use,filePath:u,fileName:C.basename(u),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return l.status(404).json({error:`File not found: ${u}`});console.error("[debugger] Error loading test flow:",h),l.status(500).json({error:h.message})}}),s.put("/api/test-flow",async(c,l)=>{try{let p=i(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let u=p.filePath,{testFlow:h,metadata:g}=c.body;if(!h)return l.status(400).json({error:"testFlow is required"});let m=ve(h,g),d=u+".tmp";await I.writeFile(d,m,"utf-8"),await I.rename(d,u);let f=await I.stat(u);l.json({success:!0,lastModified:f.mtimeMs})}catch(p){console.error("[debugger] Error saving test flow:",p),l.status(500).json({error:p.message})}}),s.get("/api/fixtures",async(c,l)=>{try{let p=a();try{let h=(await I.readdir(p,{withFileTypes:!0})).filter(g=>g.isFile()).map(g=>g.name).sort();l.json({files:h,dir:p})}catch(u){if(u.code==="ENOENT")l.json({files:[],dir:p});else throw u}}catch(p){console.error("[debugger] Error listing fixtures:",p),l.status(500).json({error:p.message})}}),s.post("/api/fixtures",async(c,l)=>{try{let p=a();await I.mkdir(p,{recursive:!0});let{name:u,content:h}=c.body;if(!u||!h)return l.status(400).json({error:"name and content are required"});let g=C.basename(u);if(!g)return l.status(400).json({error:"Invalid file name"});let m=C.join(p,g);await I.writeFile(m,Buffer.from(h,"base64")),l.json({fileName:g})}catch(p){console.error("[debugger] Error saving fixture:",p),l.status(500).json({error:p.message})}}),s}var Wr=x(()=>{"use strict";ae();mt()});import*as zr from"http";import*as Yr from"net";import*as me from"path";import*as Jr from"fs";import{randomUUID as ps}from"crypto";function Kr(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let o=[];e.on("data",n=>o.push(n)),e.on("end",()=>t(Buffer.concat(o))),e.on("error",r)})}function Vr(e,t,r,o,n,s){return new Promise(a=>{let i=Array.isArray(s)?s[0]:s,c=zr.request({hostname:r,port:o,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(n.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",p=>t.write(p)),l.on("end",()=>{t.end(),a()}),l.on("error",()=>{t.writableEnded||t.end(),a()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),a()}),c.end(n)})}function yt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var us,He,Xr=x(()=>{"use strict";rt();us=1e4,He=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??tt,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}async openSession(t){let r=me.resolve(t),o=this.byYamlPath.get(r);if(o){let l=this.sessions.get(o);if(l&&l.session.status!=="ended")return{...l.session};this.byYamlPath.delete(r)}if(!Jr.existsSync(r))throw new Error(`YAML file not found: ${r}`);let n=et(me.dirname(r));if(!n)throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let s=`dbg-${ps().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:s,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"starting",exitInfo:null},c={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};this.sessions.set(s,c),this.byYamlPath.set(r,s),this.notifyStateChange(i);try{let l=us+18e4,p,u=new Promise((m,d)=>{p=setTimeout(()=>d(new Error(`Timed out after ${l/1e3}s waiting for inner playwright to register on a port.`)),l)}),h;try{h=await Promise.race([this.spawner({yamlFilePath:r,configPath:n,tempSuffix:s,headed:this.headed}),u])}finally{p&&clearTimeout(p)}i.innerPort=h.port,i.innerHost=h.host,i.pid=h.pid,i.status="running",c.cleanup=h.cleanup;let g=m=>{i.status!=="ended"&&(i.status="ended",i.exitInfo=m,this.notifyStateChange(i))};return c.livenessTimer=this.startLivenessProbe(s,g),this.notifyStateChange(i),this.log(`[manager] session ${s} running on ${i.innerHost}:${i.innerPort} (yaml=${me.basename(r)})`),{...i}}catch(l){throw this.sessions.delete(s),this.byYamlPath.get(r)===s&&this.byYamlPath.delete(r),i.status="ended",i.exitInfo=l.message,this.notifyStateChange(i),l}}startLivenessProbe(t,r){let s=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"){clearInterval(a);return}let c=!1;try{process.kill(i.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?s=0:(s+=1,s>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(o){this.log(`[manager] closeSession ${t} cleanup error: ${o.message}`)}}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:o}=r;return async(n,s,a)=>{let i=this.sessions.get(t);if(!i){s.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){s.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}let c=`${n.method} ${n.path}`;if(c==="POST /api/int-runner/create-session"){let p=await Kr(n),u={};if(p.length)try{u=JSON.parse(p.toString("utf-8"))}catch{s.status(400).json({status:"error",message:"Invalid JSON body"});return}u.testFilePath=i.session.yamlPath,await Vr(n,s,i.session.innerHost,i.session.innerPort,Buffer.from(JSON.stringify(u),"utf-8"),"application/json");return}if(c==="POST /api/int-runner/terminate-session"){await this.closeSession(t),s.json({status:"success",details:"Session terminated"});return}if(c==="POST /api/int-runner/liveview-url"){let p=o?.(n)??"";s.json({liveviewUrl:p,browserWsUrl:""});return}let l=await Kr(n);await Vr(n,s,i.session.innerHost,i.session.innerPort,l,n.headers["content-type"])}}wsUpgradeFor(t){return async(r,o,n,s)=>{let a=this.sessions.get(t);if(!a){yt(o,`HTTP/1.1 404 Not Found\r
115
+ `):["// Skipping js_action: missing code"]});Wn=["testContext","request"];Or=5;ft=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as cs}from"express";import*as I from"fs/promises";import*as C from"path";import{stringify as ls}from"yaml";function Be(e){if(!e||e.length===0)return[];let r=ls({goal:"_hook",statements:e});return R(r).statements??[]}async function jr(e){try{let t=await I.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 jr(C.join(e,r.name))))return!0}catch{}return!1}function Gr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:n}=e,s=cs();function a(){return C.join(o??t,"fixtures")}s.get("/api/files",async(c,l)=>{try{let p=typeof c.query.dir=="string"?c.query.dir:t,u=C.resolve(p),h=await I.readdir(u,{withFileTypes:!0}),m=[];for(let d of h)if(d.name!=="node_modules"&&!d.name.startsWith("."))if(d.isDirectory()){let S=C.join(u,d.name);await jr(S)&&m.push({name:d.name,type:"directory",path:S})}else d.isFile()&&d.name.endsWith(".test.yaml")&&m.push({name:d.name,type:"file",path:C.join(u,d.name)});m.sort((d,S)=>d.type!==S.type?d.type==="directory"?-1:1:d.name.localeCompare(S.name));let g=C.dirname(u);l.json({dir:u,parent:g!==u?g:null,entries:m,initialFile:r??null,projectRoot:o??t})}catch(p){console.error("[debugger] Error listing files:",p),l.status(500).json({error:p.message})}});function i(c){if(typeof c=="string"&&c){let l=C.resolve(c);return l.endsWith(".test.yaml")?{filePath:l}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return s.get("/api/test-flow",async(c,l)=>{let p=i(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let u=p.filePath;try{n?.(u);let h=await I.readFile(u,"utf-8"),m=await I.stat(u),g=xe(h),d=gt(h,u);if(d.suite){let S=d.suite,f={tests:S.tests.map(w=>({name:w.name,statements:w.testFlow.statements??[],teardown:w.testFlow.teardown,skip:w.skip,timeout:w.timeout,fail:w.fail,only:w.only,slow:w.slow})),beforeAll:Be(S.beforeAll),afterAll:Be(S.afterAll),beforeEach:Be(S.beforeEach),afterEach:Be(S.afterEach)},b={version:"1.3.0",baseURL:d.use?.baseURL,testGroup:f};l.json({isSuite:!0,testFlow:b,metadata:g,name:d.name,tags:d.tags,use:d.use,filePath:u,fileName:C.basename(u),lastModified:m.mtimeMs})}else{let S=d.testFlow;Ie(h,S),l.json({isSuite:!1,testFlow:S,metadata:g,name:d.name,tags:d.tags,use:d.use,filePath:u,fileName:C.basename(u),lastModified:m.mtimeMs})}}catch(h){if(h.code==="ENOENT")return l.status(404).json({error:`File not found: ${u}`});console.error("[debugger] Error loading test flow:",h),l.status(500).json({error:h.message})}}),s.put("/api/test-flow",async(c,l)=>{try{let p=i(c.query.file);if("error"in p)return l.status(400).json({error:p.error});let u=p.filePath,{testFlow:h,metadata:m}=c.body;if(!h)return l.status(400).json({error:"testFlow is required"});let g=ve(h,m),d=u+".tmp";await I.writeFile(d,g,"utf-8"),await I.rename(d,u);let S=await I.stat(u);l.json({success:!0,lastModified:S.mtimeMs})}catch(p){console.error("[debugger] Error saving test flow:",p),l.status(500).json({error:p.message})}}),s.get("/api/fixtures",async(c,l)=>{try{let p=a();try{let h=(await I.readdir(p,{withFileTypes:!0})).filter(m=>m.isFile()).map(m=>m.name).sort();l.json({files:h,dir:p})}catch(u){if(u.code==="ENOENT")l.json({files:[],dir:p});else throw u}}catch(p){console.error("[debugger] Error listing fixtures:",p),l.status(500).json({error:p.message})}}),s.post("/api/fixtures",async(c,l)=>{try{let p=a();await I.mkdir(p,{recursive:!0});let{name:u,content:h}=c.body;if(!u||!h)return l.status(400).json({error:"name and content are required"});let m=C.basename(u);if(!m)return l.status(400).json({error:"Invalid file name"});let g=C.join(p,m);await I.writeFile(g,Buffer.from(h,"base64")),l.json({fileName:m})}catch(p){console.error("[debugger] Error saving fixture:",p),l.status(500).json({error:p.message})}}),s}var Wr=x(()=>{"use strict";ae();mt()});import*as zr from"http";import*as Yr from"net";import*as me from"path";import*as Jr from"fs";import{randomUUID as ps}from"crypto";function Kr(e){return new Promise((t,r)=>{if(e.body&&typeof e.body=="object")try{t(Buffer.from(JSON.stringify(e.body),"utf-8"));return}catch{}if(e.readableEnded){t(Buffer.alloc(0));return}let o=[];e.on("data",n=>o.push(n)),e.on("end",()=>t(Buffer.concat(o))),e.on("error",r)})}function Vr(e,t,r,o,n,s){return new Promise(a=>{let i=Array.isArray(s)?s[0]:s,c=zr.request({hostname:r,port:o,path:e.originalUrl,method:e.method,headers:{"content-type":i||"application/json","content-length":String(n.length)},timeout:3e5},l=>{t.writeHead(l.statusCode??502,l.headers),l.on("data",p=>t.write(p)),l.on("end",()=>{t.end(),a()}),l.on("error",()=>{t.writableEnded||t.end(),a()})});c.on("error",l=>{t.headersSent?t.writableEnded||t.end():t.status(502).json({status:"error",message:"Inner server unavailable: "+l.message}),a()}),c.end(n)})}function yt(e,t){try{(!("destroyed"in e)||!e.destroyed)&&(e.write(t),e.destroy())}catch{}}var us,He,Xr=x(()=>{"use strict";rt();us=1e4,He=class{sessions=new Map;byYamlPath=new Map;options;spawner;headed;constructor(t={}){this.options=t,this.spawner=t.spawner??tt,this.headed=t.headed??!1}log(t){this.options.onLog?this.options.onLog(t):console.error(t)}notifyStateChange(t){try{this.options.onSessionStateChange?.({...t})}catch(r){this.log(`[manager] onSessionStateChange listener threw: ${r.message}`)}}async openSession(t){let r=me.resolve(t),o=this.byYamlPath.get(r);if(o){let l=this.sessions.get(o);if(l&&l.session.status!=="ended")return{...l.session};this.byYamlPath.delete(r)}if(!Jr.existsSync(r))throw new Error(`YAML file not found: ${r}`);let n=et(me.dirname(r));if(!n)throw new Error(`No Playwright config found for ${r} (searched parents for playwright.config.{ts,js,mjs}).`);let s=`dbg-${ps().slice(0,8)}`,a=new Date().toISOString(),i={sessionId:s,yamlPath:r,innerPort:0,innerHost:"127.0.0.1",pid:0,startedAt:a,status:"starting",exitInfo:null},c={session:i,cleanup:async()=>{},readyPromise:Promise.resolve()};this.sessions.set(s,c),this.byYamlPath.set(r,s),this.notifyStateChange(i);try{let l=us+18e4,p,u=new Promise((g,d)=>{p=setTimeout(()=>d(new Error(`Timed out after ${l/1e3}s waiting for inner playwright to register on a port.`)),l)}),h;try{h=await Promise.race([this.spawner({yamlFilePath:r,configPath:n,tempSuffix:s,headed:this.headed}),u])}finally{p&&clearTimeout(p)}i.innerPort=h.port,i.innerHost=h.host,i.pid=h.pid,i.status="running",c.cleanup=h.cleanup;let m=g=>{i.status!=="ended"&&(i.status="ended",i.exitInfo=g,this.notifyStateChange(i))};return c.livenessTimer=this.startLivenessProbe(s,m),this.notifyStateChange(i),this.log(`[manager] session ${s} running on ${i.innerHost}:${i.innerPort} (yaml=${me.basename(r)})`),{...i}}catch(l){throw this.sessions.delete(s),this.byYamlPath.get(r)===s&&this.byYamlPath.delete(r),i.status="ended",i.exitInfo=l.message,this.notifyStateChange(i),l}}startLivenessProbe(t,r){let s=0,a=setInterval(()=>{let i=this.sessions.get(t);if(!i||i.session.status==="ended"){clearInterval(a);return}let c=!1;try{process.kill(i.session.pid,0),c=!0}catch(l){l?.code!=="ESRCH"&&(c=!0)}c?s=0:(s+=1,s>=3&&(clearInterval(a),r("process-exited")))},3e3);return a.unref(),a}async closeSession(t){let r=this.sessions.get(t);if(r){this.sessions.delete(t),this.byYamlPath.get(r.session.yamlPath)===t&&this.byYamlPath.delete(r.session.yamlPath),r.livenessTimer&&(clearInterval(r.livenessTimer),r.livenessTimer=void 0),r.session.status!=="ended"&&(r.session.status="ended",r.session.exitInfo="SIGTERM",this.notifyStateChange(r.session));try{await r.cleanup()}catch(o){this.log(`[manager] closeSession ${t} cleanup error: ${o.message}`)}}}listSessions(){return Array.from(this.sessions.values()).map(t=>({...t.session}))}getSession(t){let r=this.sessions.get(t);return r?{...r.session}:void 0}httpProxyFor(t,r={}){let{liveviewUrlBuilder:o}=r;return async(n,s,a)=>{let i=this.sessions.get(t);if(!i){s.status(404).json({status:"error",message:"Session not found"});return}if(i.session.status==="ended"){s.status(410).json({status:"error",message:"Session has ended",exitInfo:i.session.exitInfo});return}let c=`${n.method} ${n.path}`;if(c==="POST /api/int-runner/create-session"){let p=await Kr(n),u={};if(p.length)try{u=JSON.parse(p.toString("utf-8"))}catch{s.status(400).json({status:"error",message:"Invalid JSON body"});return}u.testFilePath=i.session.yamlPath,await Vr(n,s,i.session.innerHost,i.session.innerPort,Buffer.from(JSON.stringify(u),"utf-8"),"application/json");return}if(c==="POST /api/int-runner/terminate-session"){await this.closeSession(t),s.json({status:"success",details:"Session terminated"});return}if(c==="POST /api/int-runner/liveview-url"){let p=o?.(n)??"";s.json({liveviewUrl:p,browserWsUrl:""});return}let l=await Kr(n);await Vr(n,s,i.session.innerHost,i.session.innerPort,l,n.headers["content-type"])}}wsUpgradeFor(t){return async(r,o,n,s)=>{let a=this.sessions.get(t);if(!a){yt(o,`HTTP/1.1 404 Not Found\r
116
116
  \r
117
117
  `);return}if(a.session.status==="ended"){yt(o,`HTTP/1.1 410 Gone\r
118
118
  \r
119
- `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,c=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),p=new URL(l.replace(/^ws/,"http")),u=parseInt(p.port||"80",10),h=p.hostname,g=p.pathname;s&&s.startsWith("/page/")&&(g=`/devtools${s}`);let m=Yr.createConnection(u,h);m.on("connect",()=>{let d=`GET ${g} HTTP/1.1\r
119
+ `);return}try{let i=a.session.innerHost.includes(":")?`[${a.session.innerHost}]`:a.session.innerHost,c=await fetch(`http://${i}:${a.session.innerPort}/api/browser-cdp`);if(!c.ok)throw new Error(`Inner /api/browser-cdp returned ${c.status}`);let{cdpUrl:l}=await c.json(),p=new URL(l.replace(/^ws/,"http")),u=parseInt(p.port||"80",10),h=p.hostname,m=p.pathname;s&&s.startsWith("/page/")&&(m=`/devtools${s}`);let g=Yr.createConnection(u,h);g.on("connect",()=>{let d=`GET ${m} HTTP/1.1\r
120
120
  Host: ${h}:${u}\r
121
- `;for(let[f,w]of Object.entries(r.headers)){let S=f.toLowerCase();S!=="host"&&S!=="origin"&&(d+=`${f}: ${Array.isArray(w)?w.join(", "):w}\r
121
+ `;for(let[S,f]of Object.entries(r.headers)){let b=S.toLowerCase();b!=="host"&&b!=="origin"&&(d+=`${S}: ${Array.isArray(f)?f.join(", "):f}\r
122
122
  `)}d+=`\r
123
- `,m.write(d),n.length&&m.write(n),m.pipe(o),o.pipe(m)}),m.on("error",()=>{(!("destroyed"in o)||!o.destroyed)&&o.destroy()}),o.on("error",()=>{m.destroyed||m.destroy()}),o.on("close",()=>{m.destroyed||m.destroy()})}catch(i){this.log(`[manager] WS upgrade for ${t} failed: ${i.message}`),yt(o,`HTTP/1.1 502 Bad Gateway\r
123
+ `,g.write(d),n.length&&g.write(n),g.pipe(o),o.pipe(g)}),g.on("error",()=>{(!("destroyed"in o)||!o.destroyed)&&o.destroy()}),o.on("error",()=>{g.destroyed||g.destroy()}),o.on("close",()=>{g.destroyed||g.destroy()})}catch(i){this.log(`[manager] WS upgrade for ${t} failed: ${i.message}`),yt(o,`HTTP/1.1 502 Bad Gateway\r
124
124
  \r
125
- `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as ds}from"express";import qr from"express";import*as ye from"fs";import*as ee from"path";function hs(e){return e?ee.isAbsolute(e)?ee.normalize(e):ee.resolve(e):null}function fs(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function gs(e){let t=ee.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Zr(e){let{manager:t,staticDir:r,resolveYamlPath:o=hs,liveviewUrlBuilder:n=fs}=e,s=ds();s.post("/api/debugger/sessions",qr.json(),async(i,c)=>{let l=i.body?.yamlPath;if(typeof l!="string"||!l){c.status(400).json({error:"yamlPath is required"});return}let p=o(l);if(!p){c.status(403).json({error:"Path outside allowed roots"});return}if(!gs(p)){c.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!ye.existsSync(p)){c.status(404).json({error:"File not found"});return}let u=t.listSessions().find(h=>h.yamlPath===p&&h.status!=="ended");try{let h=await t.openSession(p);c.status(u?200:201).json({sessionId:h.sessionId,yamlPath:h.yamlPath,startedAt:h.startedAt,status:h.status})}catch(h){c.status(500).json({error:h.message})}}),s.get("/api/debugger/sessions",(i,c)=>{c.setHeader("Cache-Control","no-store"),c.json({sessions:t.listSessions().map(l=>({sessionId:l.sessionId,yamlPath:l.yamlPath,startedAt:l.startedAt,status:l.status}))})}),s.delete("/api/debugger/sessions/:sessionId",async(i,c)=>{let l=i.params.sessionId,p=!!t.getSession(l);await t.closeSession(l),c.json({deleted:!0,alreadyGone:!p})}),ye.existsSync(r)?s.use("/debugger/static",qr.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),s.get("/debugger/:sessionId/",(i,c)=>{let l=i.params.sessionId;if(!t.getSession(l)){c.status(404).send("Debugger session not found");return}let u=ee.join(r,"index.html");if(!ye.existsSync(u)){c.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let g=ye.readFileSync(u,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');c.type("html").send(g)});let a=(i,c,l)=>{let p=i.params.sessionId,u=i.originalUrl,h=`/debugger/${p}`;if(u.startsWith(h)){let m=u.slice(h.length)||"/";i.url=m,Object.defineProperty(i,"originalUrl",{value:m,configurable:!0})}t.httpProxyFor(p,{liveviewUrlBuilder:()=>n(p,i)})(i,c,l)};return s.use("/debugger/:sessionId",a),s}function Qr(e,t,r,o){let s=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!s){r.write(`HTTP/1.1 404 Not Found\r
125
+ `)}}}async shutdown(){let t=Array.from(this.sessions.keys());await Promise.allSettled(t.map(r=>this.closeSession(r)))}}});import{Router as ds}from"express";import qr from"express";import*as ye from"fs";import*as ee from"path";function hs(e){return e?ee.isAbsolute(e)?ee.normalize(e):ee.resolve(e):null}function fs(e,t){let r=t.headers.host??"127.0.0.1";return`${(t.headers["x-forwarded-proto"]??"ws")==="https"?"wss":"ws"}://${r}/ws/debugger/${e}/cdp-browser`}function gs(e){let t=ee.basename(e);return t.endsWith(".test.yaml")||t.endsWith(".test.yml")}function Zr(e){let{manager:t,staticDir:r,resolveYamlPath:o=hs,liveviewUrlBuilder:n=fs}=e,s=ds();s.post("/api/debugger/sessions",qr.json(),async(i,c)=>{let l=i.body?.yamlPath;if(typeof l!="string"||!l){c.status(400).json({error:"yamlPath is required"});return}let p=o(l);if(!p){c.status(403).json({error:"Path outside allowed roots"});return}if(!gs(p)){c.status(400).json({error:"Not a Shiplight test file (expected *.test.yaml or *.test.yml)"});return}if(!ye.existsSync(p)){c.status(404).json({error:"File not found"});return}let u=t.listSessions().find(h=>h.yamlPath===p&&h.status!=="ended");try{let h=await t.openSession(p);c.status(u?200:201).json({sessionId:h.sessionId,yamlPath:h.yamlPath,startedAt:h.startedAt,status:h.status})}catch(h){c.status(500).json({error:h.message})}}),s.get("/api/debugger/sessions",(i,c)=>{c.setHeader("Cache-Control","no-store"),c.json({sessions:t.listSessions().map(l=>({sessionId:l.sessionId,yamlPath:l.yamlPath,startedAt:l.startedAt,status:l.status}))})}),s.delete("/api/debugger/sessions/:sessionId",async(i,c)=>{let l=i.params.sessionId,p=!!t.getSession(l);await t.closeSession(l),c.json({deleted:!0,alreadyGone:!p})}),ye.existsSync(r)?s.use("/debugger/static",qr.static(r)):console.error(`[debugger] WARNING: debugger static dir missing at ${r} \u2014 iframe routes will 404`),s.get("/debugger/:sessionId/",(i,c)=>{let l=i.params.sessionId;if(!t.getSession(l)){c.status(404).send("Debugger session not found");return}let u=ee.join(r,"index.html");if(!ye.existsSync(u)){c.status(500).send(`Debugger SPA bundle missing at ${u}`);return}let m=ye.readFileSync(u,"utf-8").replace(/(src|href)="\/assets\//g,'$1="/debugger/static/assets/').replace(/(src|href)="\/index\.html/g,'$1="/debugger/static/index.html');c.type("html").send(m)});let a=(i,c,l)=>{let p=i.params.sessionId,u=i.originalUrl,h=`/debugger/${p}`;if(u.startsWith(h)){let g=u.slice(h.length)||"/";i.url=g,Object.defineProperty(i,"originalUrl",{value:g,configurable:!0})}t.httpProxyFor(p,{liveviewUrlBuilder:()=>n(p,i)})(i,c,l)};return s.use("/debugger/:sessionId",a),s}function Qr(e,t,r,o){let s=(t.url??"").match(/^\/ws\/debugger\/([^/]+)(.*)$/);if(!s){r.write(`HTTP/1.1 404 Not Found\r
126
126
  \r
127
- `),r.destroy();return}let a=s[1],i=s[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,o,i||void 0)}var eo=x(()=>{"use strict"});var ro={};re(ro,{startDebuggerServer:()=>ws});import je from"express";import*as U from"path";import{createRequire as ms}from"node:module";async function ws(e){let{initialDir:t,initialFile:r,projectRoot:o,port:n}=e,s=r,a=null,i=new He,c=f=>{if(f!==s&&(s=f,a)){let w=i.getSession(a);(!w||w.yamlPath!==f)&&(i.closeSession(a).catch(()=>{}),a=null)}};async function l(f){if(a){let S=i.getSession(a);if(S&&S.status!=="ended"&&S.yamlPath===f)return a;await i.closeSession(a),a=null}console.error("[debugger] Starting Playwright sandbox for "+U.basename(f)+"...");let w=await i.openSession(f);return a=w.sessionId,w.sessionId}let p=je();p.use((f,w,S)=>{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"),f.method==="OPTIONS")return w.sendStatus(204);S()});let u=async(f,w,S)=>{try{if(!s){w.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});return}let b=await l(s);f.url=f.originalUrl,i.httpProxyFor(b,{liveviewUrlBuilder:()=>`ws://localhost:${n}/ws/cdp-browser`})(f,w,S)}catch(b){let _=b.message;console.error("[debugger] Failed to ensure session: "+_),w.headersSent||w.status(503).json({status:"error",message:"Failed to start sandbox: "+_})}};p.use("/api/int-runner",u),p.use("/api/browser-cdp",u),p.use(je.json({limit:"10mb"})),p.use(Gr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:c}));let h=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=h.includes(U.sep+"src"+U.sep)?U.resolve(h,"../../dist/static"):U.join(h,"static");p.use("/devtools",je.static(to)),p.use(je.static(g)),p.use(Zr({manager:i,staticDir:g})),p.get("*path",(f,w,S)=>{if(f.path.startsWith("/api/"))return S();w.sendFile(U.join(g,"index.html"),b=>{b&&w.send(bs(t,n))})});let m=await new Promise((f,w)=>{let S=p.listen(n,"localhost",()=>{f(S)});S.on("error",w)});m.on("upgrade",(f,w,S)=>{let b=f.url??"";if(b.startsWith("/ws/debugger/")){Qr(i,f,w,S);return}if(b.startsWith("/ws/cdp-browser")){if(!a){w.write(`HTTP/1.1 503 Service Unavailable\r
127
+ `),r.destroy();return}let a=s[1],i=s[2]||"";i.startsWith("/cdp-browser/page/")?i=i.slice(12):(i==="/cdp-browser"||i==="/cdp-browser/")&&(i=""),e.wsUpgradeFor(a)(t,r,o,i||void 0)}var eo=x(()=>{"use strict"});var ro={};re(ro,{startDebuggerServer:()=>ws});import je from"express";import*as U from"path";import{createRequire as ms}from"node:module";async function ws(e){let{initialDir:t,initialFile:r,projectRoot:o,port:n,headed:s=!1}=e,a=r,i=null,c=new He({headed:s}),l=f=>{if(f!==a&&(a=f,i)){let b=c.getSession(i);(!b||b.yamlPath!==f)&&(c.closeSession(i).catch(()=>{}),i=null)}};async function p(f){if(i){let w=c.getSession(i);if(w&&w.status!=="ended"&&w.yamlPath===f)return i;await c.closeSession(i),i=null}console.error("[debugger] Starting Playwright sandbox for "+U.basename(f)+"...");let b=await c.openSession(f);return i=b.sessionId,b.sessionId}let u=je();u.use((f,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"),f.method==="OPTIONS")return b.sendStatus(204);w()});let h=async(f,b,w)=>{try{if(!a){b.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});return}let _=await p(a);f.url=f.originalUrl,c.httpProxyFor(_,{liveviewUrlBuilder:()=>`ws://localhost:${n}/ws/cdp-browser`})(f,b,w)}catch(_){let T=_.message;console.error("[debugger] Failed to ensure session: "+T),b.headersSent||b.status(503).json({status:"error",message:"Failed to start sandbox: "+T})}};u.use("/api/int-runner",h),u.use("/api/browser-cdp",h),u.use(je.json({limit:"10mb"})),u.use(Gr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:l}));let m=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=m.includes(U.sep+"src"+U.sep)?U.resolve(m,"../../dist/static"):U.join(m,"static");u.use("/devtools",je.static(to)),u.use(je.static(g)),u.use(Zr({manager:c,staticDir:g})),u.get("*path",(f,b,w)=>{if(f.path.startsWith("/api/"))return w();b.sendFile(U.join(g,"index.html"),_=>{_&&b.send(bs(t,n))})});let d=await new Promise((f,b)=>{let w=u.listen(n,"localhost",()=>{f(w)});w.on("error",b)});d.on("upgrade",(f,b,w)=>{let _=f.url??"";if(_.startsWith("/ws/debugger/")){Qr(c,f,b,w);return}if(_.startsWith("/ws/cdp-browser")){if(!i){b.write(`HTTP/1.1 503 Service Unavailable\r
128
128
  \r
129
- `),w.destroy();return}let _=b.slice(15);i.wsUpgradeFor(a)(f,w,S,_||void 0);return}w.write(`HTTP/1.1 404 Not Found\r
129
+ `),b.destroy();return}let T=_.slice(15);c.wsUpgradeFor(i)(f,b,w,T||void 0);return}b.write(`HTTP/1.1 404 Not Found\r
130
130
  \r
131
- `),w.destroy()});let d=`http://localhost:${n}`;return console.error(`[debugger] Server running at ${d}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:d,close:async()=>{await i.shutdown(),await new Promise((f,w)=>{m.close(S=>S?w(S):f())})}}}function bs(e,t){return`<!DOCTYPE html>
131
+ `),b.destroy()});let S=`http://localhost:${n}`;return console.error(`[debugger] Server running at ${S}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:S,close:async()=>{await c.shutdown(),await new Promise((f,b)=>{d.close(w=>w?b(w):f())})}}}function bs(e,t){return`<!DOCTYPE html>
132
132
  <html>
133
133
  <head>
134
134
  <meta charset="utf-8">
@@ -146,15 +146,15 @@ Host: ${h}:${u}\r
146
146
  <p>Directory: <code>${e}</code></p>
147
147
  <p>Server: localhost:${t}</p>
148
148
  </body>
149
- </html>`}var ys,to,oo=x(()=>{"use strict";Wr();Xr();eo();ys=ms(import.meta.url);try{to=U.join(U.dirname(ys.resolve("@shiplightai/devtools-assets/package.json")),"dist")}catch{console.error("[debugger] Required peer package @shiplightai/devtools-assets is not installed. Reinstall shiplightai (it pulls the assets package as a dependency)."),process.exit(1)}});var no={};re(no,{startDebugger:()=>vs});import*as X from"fs";import*as le from"path";async function vs(e){let t,r=6174,o,n=!1,s=!0;for(let f=0;f<e.length;f++)e[f]==="--port"&&e[f+1]?(r=parseInt(e[f+1],10),f++):e[f]==="--url"&&e[f+1]?(o=e[f+1],f++):e[f]==="--new"?n=!0:e[f]==="--open"?s=!1:e[f]==="--no-open"?s=!0:e[f]==="--help"||e[f]==="-h"?(xs(),process.exit(0)):e[f].startsWith("--")||(t=e[f]);let a,i;if(t){let f=le.resolve(t);X.existsSync(f)&&X.statSync(f).isDirectory()?a=f:(a=le.dirname(f),i=f)}else a=process.cwd();if(n&&i&&!X.existsSync(i)){let f=le.dirname(i);X.existsSync(f)||X.mkdirSync(f,{recursive:!0}),X.writeFileSync(i,Ss(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${i}`)}i&&!X.existsSync(i)&&(console.error(`Error: File not found: ${i}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(rt(),Kt)),l=c(a),p=l?le.dirname(l):a;(await import("dotenv")).config({path:le.join(p,".env"),override:!0}),l||(console.error("Error: No Playwright config found in "+a),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:h}=await Promise.resolve().then(()=>(oo(),ro));console.log(i?`Starting debugger for: ${i}`:`Starting debugger in: ${a}`),l&&console.log(`Using Playwright config: ${l}`);let g;try{g=await h({initialDir:a,initialFile:i,projectRoot:p,port:r})}catch(f){throw f?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),f}if(jt(r,a),console.log(`Debugger running at: ${g.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!s)try{let{default:f}=await import("open");await f(g.url)}catch{}let m=!1,d=async()=>{m&&(console.log(`
150
- Force exiting...`),process.exit(1)),m=!0,console.log(`
151
- Shutting down...`);let f=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Gt(r,a),await g.close()}catch{}clearTimeout(f),process.exit(0)};process.on("SIGINT",d),process.on("SIGTERM",d)}function xs(){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 Ss,so=x(()=>{"use strict";Wt();Ss=e=>`goal: New test
149
+ </html>`}var ys,to,oo=x(()=>{"use strict";Wr();Xr();eo();ys=ms(import.meta.url);try{to=U.join(U.dirname(ys.resolve("@shiplightai/devtools-assets/package.json")),"dist")}catch{console.error("[debugger] Required peer package @shiplightai/devtools-assets is not installed. Reinstall shiplightai (it pulls the assets package as a dependency)."),process.exit(1)}});var no={};re(no,{startDebugger:()=>vs});import*as X from"fs";import*as le from"path";async function vs(e){let t,r=6174,o,n=!1,s=!0,a=!1;for(let f=0;f<e.length;f++)e[f]==="--port"&&e[f+1]?(r=parseInt(e[f+1],10),f++):e[f]==="--url"&&e[f+1]?(o=e[f+1],f++):e[f]==="--new"?n=!0:e[f]==="--open"?s=!1:e[f]==="--no-open"?s=!0:e[f]==="--headed"?a=!0:e[f]==="--help"||e[f]==="-h"?(xs(),process.exit(0)):e[f].startsWith("--")||(t=e[f]);let i,c;if(t){let f=le.resolve(t);X.existsSync(f)&&X.statSync(f).isDirectory()?i=f:(i=le.dirname(f),c=f)}else i=process.cwd();if(n&&c&&!X.existsSync(c)){let f=le.dirname(c);X.existsSync(f)||X.mkdirSync(f,{recursive:!0}),X.writeFileSync(c,Ss(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${c}`)}c&&!X.existsSync(c)&&(console.error(`Error: File not found: ${c}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:l}=await Promise.resolve().then(()=>(rt(),Kt)),p=l(i),u=p?le.dirname(p):i;(await import("dotenv")).config({path:le.join(u,".env"),override:!0}),p||(console.error("Error: No Playwright config found in "+i),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let{startDebuggerServer:m}=await Promise.resolve().then(()=>(oo(),ro));console.log(c?`Starting debugger for: ${c}`:`Starting debugger in: ${i}`),p&&console.log(`Using Playwright config: ${p}`);let g;try{g=await m({initialDir:i,initialFile:c,projectRoot:u,port:r,headed:a})}catch(f){throw f?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),f}if(jt(r,i),console.log(`Debugger running at: ${g.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!s)try{let{default:f}=await import("open");await f(g.url)}catch{}let d=!1,S=async()=>{d&&(console.log(`
150
+ Force exiting...`),process.exit(1)),d=!0,console.log(`
151
+ Shutting down...`);let f=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{Gt(r,i),await g.close()}catch{}clearTimeout(f),process.exit(0)};process.on("SIGINT",S),process.on("SIGTERM",S)}function xs(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" --headed Force a visible Chromium window (overrides use.headless)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var Ss,so=x(()=>{"use strict";Wt();Ss=e=>`goal: New test
152
152
  statements:
153
153
  - URL: ${e}
154
154
  - "TODO: Add your first step"
155
155
  `});var wt={};re(wt,{lookupActionStores:()=>_s,updateActionStores:()=>Ts});function io(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function _s(e){let t=io();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),n=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),!n.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${n.status}`),new Map;let s=await n.json(),a=new Map;for(let[i,c]of Object.entries(s.stores??{}))a.set(i,c);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function Ts(e){let t=io();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),n={};for(let[i,c]of e)n[i]=c;let s=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:n}),signal:r.signal});return clearTimeout(o),s.ok?(await s.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${s.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var bt=x(()=>{"use strict"});import*as B from"fs";import*as Te from"path";import{globSync as ks}from"glob";function ao(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new vt:new St(e)}function ke(e){return e.replace(/\//g,"__")+".json"}function Es(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Ps,St,vt,co=x(()=>{"use strict";ae();Ps=".shiplight/action-cache";St=class{constructor(t){this.cwd=t;this.cacheDir=Te.join(t,Ps)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!B.existsSync(this.cacheDir))return r;for(let o of t){let n=Te.join(this.cacheDir,ke(o));try{if(B.existsSync(n)){let s=B.readFileSync(n,"utf-8");r.set(o,JSON.parse(s))}}catch{}}return r}async update(t){if(t.size===0)return 0;B.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,n]of t)try{let s=Te.join(this.cacheDir,ke(o)),a=lt();if(B.existsSync(s))try{a=JSON.parse(B.readFileSync(s,"utf-8"))}catch{}let i={...a,entries:{...a.entries,...n.entries}};B.writeFileSync(s,JSON.stringify(i,null,2)),r++}catch{}return r}loadAll(){if(!B.existsSync(this.cacheDir))return;let t=ks("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let n of t)try{let s=B.readFileSync(Te.join(this.cacheDir,n),"utf-8"),a=JSON.parse(s),i=Es(n);r.set(i,a),o+=Object.keys(a.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}},vt=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(bt(),wt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(bt(),wt));return r(t)}loadAll(){}}});var ho={};re(ho,{buildTestPathsFromGitInfo:()=>uo,cloudKeyToCwdRelPath:()=>Tt,runTests:()=>$s});import{spawn as As,execFileSync as lo}from"child_process";import*as H from"fs";import*as N from"path";import{globSync as xt}from"glob";async function $s(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=>H.existsSync(N.join(t,u)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
156
156
  `));let n=e.includes("--magic"),a=e.filter(u=>u!=="--magic").map(u=>u.endsWith(".test.yaml")?u.replace(/\.test\.yaml$/,".yaml.spec.ts"):u),i=ao(t);await Is(t,i),be&&process.stdout.write(`shiplightai v${be}
157
- `);let c={...process.env};n&&(c.SHIPLIGHT_MAGIC="1");let l=As("npx",["playwright","test",...a],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(u=>{l.on("close",h=>u(h??1))});await Os(t,i),process.exit(p)}function _t(){try{return lo("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Ms(){try{let e=lo("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function po(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=Ms(),n=_t();return uo(e,t,o,n)}function uo(e,t,r,o){return{testPaths:e.map(s=>{let a=o?N.relative(o,N.resolve(t,s)):s;return`${r}${a}`}),branchPrefix:r}}function Tt(e,t,r,o){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?N.relative(o,N.resolve(r,n)):n}async function Is(e,t){try{let r=xt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:n}=po(r,e,t.isCloud),s=await t.lookup(o);if(s.size===0)return;let a=N.join(e,".shiplight","action-cache");H.mkdirSync(a,{recursive:!0});let i=_t(),c=0;for(let[l,p]of s){let u=t.isCloud?Tt(l,n,i,e):l,h=N.join(a,ke(u));H.writeFileSync(h,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function Os(e,t){try{let r=N.join(e,"test-results");if(!H.existsSync(r))return;let o=xt("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let n={};for(let u of o)try{let h=H.readFileSync(N.join(r,u),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(n,g.entries)}catch{}if(Object.keys(n).length===0)return;let s=xt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=po(s,e,t.isCloud),c=new Map;for(let u=0;u<s.length;u++){let h=s[u],g=a[u],m=N.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!H.existsSync(m))continue;let d=H.readFileSync(m,"utf-8"),f={};for(let[w,S]of Object.entries(n))d.includes(w)&&(f[w]=S);if(Object.keys(f).length>0){let w=t.isCloud?Tt(g,i,_t(),e):g,S=N.join(e,".shiplight","action-cache",ke(w)),b={};if(H.existsSync(S))try{b=JSON.parse(H.readFileSync(S,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...b,...f}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((u,h)=>u+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var fo=x(()=>{"use strict";co();Pe()});function E(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function We(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 go(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Ge(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 Ls(e){return`<span class="badge badge-${e}">${e}</span>`}function Cs(e){if(!e.code)return"";let t=e.code.split(`
157
+ `);let c={...process.env};n&&(c.SHIPLIGHT_MAGIC="1");let l=As("npx",["playwright","test",...a],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(u=>{l.on("close",h=>u(h??1))});await Os(t,i),process.exit(p)}function _t(){try{return lo("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Ms(){try{let e=lo("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function po(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=Ms(),n=_t();return uo(e,t,o,n)}function uo(e,t,r,o){return{testPaths:e.map(s=>{let a=o?N.relative(o,N.resolve(t,s)):s;return`${r}${a}`}),branchPrefix:r}}function Tt(e,t,r,o){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?N.relative(o,N.resolve(r,n)):n}async function Is(e,t){try{let r=xt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:n}=po(r,e,t.isCloud),s=await t.lookup(o);if(s.size===0)return;let a=N.join(e,".shiplight","action-cache");H.mkdirSync(a,{recursive:!0});let i=_t(),c=0;for(let[l,p]of s){let u=t.isCloud?Tt(l,n,i,e):l,h=N.join(a,ke(u));H.writeFileSync(h,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function Os(e,t){try{let r=N.join(e,"test-results");if(!H.existsSync(r))return;let o=xt("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let n={};for(let u of o)try{let h=H.readFileSync(N.join(r,u),"utf-8"),m=JSON.parse(h);m?.entries&&Object.assign(n,m.entries)}catch{}if(Object.keys(n).length===0)return;let s=xt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:i}=po(s,e,t.isCloud),c=new Map;for(let u=0;u<s.length;u++){let h=s[u],m=a[u],g=N.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!H.existsSync(g))continue;let d=H.readFileSync(g,"utf-8"),S={};for(let[f,b]of Object.entries(n))d.includes(f)&&(S[f]=b);if(Object.keys(S).length>0){let f=t.isCloud?Tt(m,i,_t(),e):m,b=N.join(e,".shiplight","action-cache",ke(f)),w={};if(H.existsSync(b))try{w=JSON.parse(H.readFileSync(b,"utf-8")).entries}catch{}c.set(m,{version:"1.0",entries:{...w,...S}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((u,h)=>u+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var fo=x(()=>{"use strict";co();Pe()});function E(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function We(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 go(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Ge(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 Ls(e){return`<span class="badge badge-${e}">${e}</span>`}function Cs(e){if(!e.code)return"";let t=e.code.split(`
158
158
  `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((n,s)=>{let a=e.codeStartLine+s,i=a===e.codeLine,c=String(a).padStart(4);return`<span class="code-line${i?" code-line-active":""}">${c} \u2502 ${E(n)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(o=>`<span class="code-line code-line-body">${E(o)}</span>`).join("")}</pre></div>`}function Rs(e){let t=e.duration!=null?`<span class="step-duration">${We(e.duration)}</span>`:"",r=Ge(e.status);if(e.screenshot||e.message||e.error||e.code){let n="";e.screenshot&&(n=`<img src="${E(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let s=e.screenshot?"":Cs(e),a="";e.error&&(a=`<div class="step-error"><pre>${E(e.error)}</pre></div>`);let i="";e.message&&!e.error&&(i=`<div class="step-message">${E(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
159
159
  <details class="step-details step step-${e.status}"${c}>
160
160
  <summary class="step-header">
@@ -216,9 +216,9 @@ statements:
216
216
  <div class="steps-list">
217
217
  ${s||'<div class="no-steps">No YAML step details available</div>'}
218
218
  </div>
219
- ${i}`}function Ds(e,t){let r=e.flaky?"flaky":e.status,o=Ge(r),n;if(e.flaky&&e.attempts&&e.attempts.length>1){let s=`tabs-${t}`,a=e.attempts.length,i=e.attempts.map((l,p)=>{let u=p===a-1,h=l.status==="passed"?"passed":"failed",g=`Attempt ${l.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${h}"
219
+ ${i}`}function Ds(e,t){let r=e.flaky?"flaky":e.status,o=Ge(r),n;if(e.flaky&&e.attempts&&e.attempts.length>1){let s=`tabs-${t}`,a=e.attempts.length,i=e.attempts.map((l,p)=>{let u=p===a-1,h=l.status==="passed"?"passed":"failed",m=`Attempt ${l.attemptNumber}`;return`<button class="attempt-tab ${u?"active":""} attempt-tab-${h}"
220
220
  onclick="switchAttemptTab('${s}', ${p})"
221
- data-tab-index="${p}">${Ge(h)} ${g} <span class="attempt-tab-badge badge-${h}">${l.status}</span></button>`}).join(""),c=e.attempts.map((l,p)=>{let u=p===a-1,h=mo(l.steps,l.error,l.videoPath,l.tracePath,`${t}-attempt-${p}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${p}">
221
+ data-tab-index="${p}">${Ge(h)} ${m} <span class="attempt-tab-badge badge-${h}">${l.status}</span></button>`}).join(""),c=e.attempts.map((l,p)=>{let u=p===a-1,h=mo(l.steps,l.error,l.videoPath,l.tracePath,`${t}-attempt-${p}`);return`<div class="attempt-panel ${u?"active":""}" data-panel-index="${p}">
222
222
  <div class="attempt-meta">
223
223
  ${Ge(l.status==="passed"?"passed":"failed")}
224
224
  <span class="attempt-meta-text">Attempt ${l.attemptNumber} &mdash; ${l.status} in ${We(l.duration)}</span>
@@ -1025,9 +1025,9 @@ statements:
1025
1025
  });
1026
1026
  </script>
1027
1027
  </body>
1028
- </html>`}var yo=x(()=>{"use strict"});import*as q from"fs";import*as te from"path";import{execFileSync as Fs}from"child_process";import{createHash as Us}from"crypto";import we from"axios";function wo(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 Fs("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Bs(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=q.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Hs(){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??"",n=process.env.GITHUB_EVENT_NAME??"",a=Bs().pull_request,i=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??i??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,h=process.env.SHIPLIGHT_PR_TITLE??a?.title,g=ie("log","-1","--pretty=%s"),m=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:u,gitRepo:r,commitMessage:g,authorEmail:m,prNumber:p,prTitle:h,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:n,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,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,s=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:s,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,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??"",n=r&&o?`${r}/${o}`:void 0,s=process.env.CIRCLE_PULL_REQUEST,a=process.env.CIRCLE_PR_NUMBER??s?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,c=i?wo(i):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:n,commitMessage:l,authorEmail:p,prNumber:a,prUrl:s,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"),n=ie("log","-1","--pretty=%ae"),s=ie("remote","get-url","origin"),a=s?wo(s):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:n,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function js(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 bo(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Gs(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 Ws(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let n=r.get(o.testCaseName);n?n.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Ke(e,t){return te.isAbsolute(t)?t:te.join(e,t)}function Ks(e,t,r,o){let n={};for(let s of e.steps)n[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return{schemaVersion:2,result:bo(e.status),flaky:!1,segments:[{outcome:bo(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:n,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function At(e){return Us("md5").update(e).digest("base64")}function Pt(e){return At(q.readFileSync(e))}async function Et(e,t){let r=q.readFileSync(t),o=te.extname(t).toLowerCase(),s={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await we.put(e,r,{headers:{"Content-Type":s,"Content-MD5":At(r)}})}async function So(e,t,r,o){let n=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",s={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},a=Hs(),i=e.tests.map(d=>{let f={testCaseName:d.title,testCaseBaseName:d.baseTitle,suiteName:d.suiteName,file:d.file,tags:d.tags,suiteTags:d.suiteTags,baseUrl:d.baseUrl,skip:d.skip,slow:d.slow,timeout:d.timeout,parameterSetName:d.parameterSetName,flaky:d.flaky,retries:d.retries};if(d.videoPath){let w=Ke(t,d.videoPath);q.existsSync(w)&&(f.videoMd5=Pt(w))}if(d.tracePath){let w=Ke(t,d.tracePath);q.existsSync(w)&&(f.traceMd5=Pt(w))}return f}),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 we.post(`${n}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:s})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let u=Ws(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(d,f)=>{let w=u[f];if(!w)return;let S=d.steps.filter(k=>k.screenshot);if(!S.length)return;let b=S.map(k=>k.stepId),_={};for(let k of S){let $=te.isAbsolute(k.screenshot)?k.screenshot:te.join(t,k.screenshot);q.existsSync($)&&(_[k.stepId]=Pt($))}try{let k=await we.post(`${n}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:b,md5s:_},{headers:s});w.uploadUrls.screenshots=k.data.screenshots,w.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${b.length} screenshot URL(s) for "${d.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${d.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let h=(await Promise.all(e.tests.map(async(d,f)=>{let w=u[f];if(!w){console.warn(`[reporter] No result slot found for test "${d.title}", skipping.`);return}let S=w.uploadUrls,b={},_=0;await Promise.all(d.steps.map(async O=>{if(O.screenshot&&S.screenshots?.[O.stepId]){let ue=te.isAbsolute(O.screenshot)?O.screenshot:te.join(t,O.screenshot);if(q.existsSync(ue))try{await Et(S.screenshots[O.stepId],ue),b[O.stepId]=w.screenshotS3Uris[O.stepId],_++}catch(Bo){console.warn(`[reporter] Screenshot upload failed for step ${O.stepId}:`,Bo)}}})),_>0&&console.log(`[reporter] [3/4] Uploaded ${_} screenshot(s) for "${d.title}"`);let k;if(d.videoPath&&S.video){let O=Ke(t,d.videoPath);if(q.existsSync(O)){console.log(`[reporter] [3/4] Uploading video for "${d.title}"...`);try{await Et(S.video,O),k=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${d.title}"`)}catch(ue){console.warn("[reporter] Video upload failed:",ue)}}}let $;if(d.tracePath&&S.trace){let O=Ke(t,d.tracePath);if(q.existsSync(O)){console.log(`[reporter] [3/4] Uploading trace for "${d.title}"...`);try{await Et(S.trace,O),$=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${d.title}"`)}catch(ue){console.warn("[reporter] Trace upload failed:",ue)}}}console.log(`[reporter] [3/4] Uploading report for "${d.title}"...`);let Je=Ks(d,b,k,$),Mt=Buffer.from(JSON.stringify(Je)),It=At(Mt),Ot=await we.post(`${n}/v1/local-runs/${p.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:It},{headers:s}),Fo=Ot.data.reportUrl,Uo=Ot.data.reportS3Uri;return await we.put(Fo,Mt,{headers:{"Content-Type":"application/json","Content-MD5":It}}),console.log(`[reporter] [3/4] Report uploaded for "${d.title}"`),{testCaseResultId:w.testCaseResultId,result:js(d.status),durationMs:d.duration,startTime:d.startTime,endTime:d.endTime,error:d.error,reportS3Uri:Uo,videoS3Uri:k,traceS3Uri:$,metadata:{suiteName:d.suiteName,file:d.file}}}))).filter(d=>!!d);console.log("[reporter] [4/4] Finalising run...");let g=Gs(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let m=await we.put(`${n}/v1/local-runs/${p.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:s});console.log(`
1029
- Shiplight cloud report: ${m.data.reportUrl}`)}var vo=x(()=>{"use strict"});var Po={};re(Po,{buildGitHubSummary:()=>ko,runReport:()=>Vs});import*as P from"fs";import*as T from"path";async function Vs(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 Ys(e,t,o):await zs(e,t)}async function zs(e,t){let r=e.find(i=>!i.startsWith("--"))||"shiplight-report",o=T.isAbsolute(r)?r:T.join(process.cwd(),r),n=T.join(o,"report-data.json");P.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let s;try{s=JSON.parse(P.readFileSync(n,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${n}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=T.join(o,"index.html");if(P.writeFileSync(a,kt(s),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await To(s,o),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function Ys(e,t,r){let o=T.join(process.cwd(),"shiplight-report"),n=e.findIndex(g=>g==="-o"||g==="--output");if(n!==-1&&e[n+1]){let g=e[n+1];o=T.isAbsolute(g)?g:T.join(process.cwd(),g)}let s=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let g=0;g<e.length;g++){let m=e[g];if(s.has(m)){g++;continue}if(a.has(m))continue;let d=T.isAbsolute(m)?m:T.join(process.cwd(),m);i.push(d)}i.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;P.mkdirSync(T.join(o,"screenshots"),{recursive:!0});for(let g=0;g<i.length;g++){let m=i[g],d=`shard-${g}`,f=T.join(m,"report-data.json");if(!P.existsSync(f)){console.warn(`Warning: No report-data.json found in ${m}, skipping.`);continue}let w;try{w=JSON.parse(P.readFileSync(f,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${f}, skipping.`);continue}console.log(`Merging ${d}: ${w.tests.length} tests from ${m}`),l+=w.totalDuration||0;let S=T.join(m,"screenshots");P.existsSync(S)&&_o(S,T.join(o,"screenshots",d));let b=T.join(o,d);for(let _ of w.tests){let k=[_,..._.attempts||[]];for(let $ of k)Js($.steps,d),$.videoPath&&xo(m,$.videoPath,b)&&($.videoPath=`${d}/${$.videoPath}`),$.tracePath&&xo(m,$.tracePath,b)&&($.tracePath=`${d}/${$.tracePath}`);c.push(_)}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(),shiplightVersion:be};P.writeFileSync(T.join(o,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let h=T.join(o,"index.html");if(P.writeFileSync(h,kt(u),"utf-8"),console.log(`
1030
- Merged ${c.length} tests from ${p} shards into: ${h}`),await To(u,o),r&&Xs(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function xo(e,t,r){let o=T.resolve(e,t);return o.startsWith(T.resolve(e)+T.sep)?P.existsSync(o)?(P.mkdirSync(r,{recursive:!0}),P.copyFileSync(o,T.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function _o(e,t){P.mkdirSync(t,{recursive:!0});for(let r of P.readdirSync(e,{withFileTypes:!0})){let o=T.join(e,r.name),n=T.join(t,r.name);r.isDirectory()?_o(o,n):P.copyFileSync(o,n)}}function Js(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function To(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN;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(s=>s.startTime).filter(s=>!!s),n=o.length>0?o.sort()[0]:e.timestamp??new Date().toISOString();try{await So(e,t,n,r)}catch(s){console.warn("[report] Cloud upload failed:",s)}}function $t(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=T.join("tests",T.basename(t));return{name:e.title||T.basename(t),yamlPath:r}}function ko(e){let t=e.filter(i=>!i.file.includes("auth.setup")),r=t.filter(i=>i.flaky),o=t.filter(i=>i.status==="passed"&&!i.flaky),n=t.filter(i=>i.status!=="passed"),s=t.length,a=`## Test Results
1028
+ </html>`}var yo=x(()=>{"use strict"});import*as q from"fs";import*as te from"path";import{execFileSync as Fs}from"child_process";import{createHash as Us}from"crypto";import we from"axios";function wo(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 Fs("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Bs(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=q.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Hs(){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??"",n=process.env.GITHUB_EVENT_NAME??"",a=Bs().pull_request,i=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??i??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,h=process.env.SHIPLIGHT_PR_TITLE??a?.title,m=ie("log","-1","--pretty=%s"),g=ie("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:u,gitRepo:r,commitMessage:m,authorEmail:g,prNumber:p,prTitle:h,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:n,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,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,s=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:s,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,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??"",n=r&&o?`${r}/${o}`:void 0,s=process.env.CIRCLE_PULL_REQUEST,a=process.env.CIRCLE_PR_NUMBER??s?.match(/\/pull\/(\d+)$/)?.[1],i=process.env.CIRCLE_REPOSITORY_URL,c=i?wo(i):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:n,commitMessage:l,authorEmail:p,prNumber:a,prUrl:s,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"),n=ie("log","-1","--pretty=%ae"),s=ie("remote","get-url","origin"),a=s?wo(s):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:n,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function js(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 bo(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Gs(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 Ws(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let n=r.get(o.testCaseName);n?n.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Ke(e,t){return te.isAbsolute(t)?t:te.join(e,t)}function Ks(e,t,r,o){let n={};for(let s of e.steps)n[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return{schemaVersion:2,result:bo(e.status),flaky:!1,segments:[{outcome:bo(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:n,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function At(e){return Us("md5").update(e).digest("base64")}function Pt(e){return At(q.readFileSync(e))}async function Et(e,t){let r=q.readFileSync(t),o=te.extname(t).toLowerCase(),s={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await we.put(e,r,{headers:{"Content-Type":s,"Content-MD5":At(r)}})}async function So(e,t,r,o){let n=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",s={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},a=Hs(),i=e.tests.map(d=>{let S={testCaseName:d.title,testCaseBaseName:d.baseTitle,suiteName:d.suiteName,file:d.file,tags:d.tags,suiteTags:d.suiteTags,baseUrl:d.baseUrl,skip:d.skip,slow:d.slow,timeout:d.timeout,parameterSetName:d.parameterSetName,flaky:d.flaky,retries:d.retries};if(d.videoPath){let f=Ke(t,d.videoPath);q.existsSync(f)&&(S.videoMd5=Pt(f))}if(d.tracePath){let f=Ke(t,d.tracePath);q.existsSync(f)&&(S.traceMd5=Pt(f))}return S}),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 we.post(`${n}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:i},{headers:s})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let u=Ws(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(d,S)=>{let f=u[S];if(!f)return;let b=d.steps.filter(T=>T.screenshot);if(!b.length)return;let w=b.map(T=>T.stepId),_={};for(let T of b){let $=te.isAbsolute(T.screenshot)?T.screenshot:te.join(t,T.screenshot);q.existsSync($)&&(_[T.stepId]=Pt($))}try{let T=await we.post(`${n}/v1/local-runs/${p.testRunId}/results/${f.testCaseResultId}/screenshot-urls`,{stepIds:w,md5s:_},{headers:s});f.uploadUrls.screenshots=T.data.screenshots,f.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${w.length} screenshot URL(s) for "${d.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${d.title}":`,T)}})),console.log("[reporter] [3/4] Uploading assets...");let h=(await Promise.all(e.tests.map(async(d,S)=>{let f=u[S];if(!f){console.warn(`[reporter] No result slot found for test "${d.title}", skipping.`);return}let b=f.uploadUrls,w={},_=0;await Promise.all(d.steps.map(async O=>{if(O.screenshot&&b.screenshots?.[O.stepId]){let ue=te.isAbsolute(O.screenshot)?O.screenshot:te.join(t,O.screenshot);if(q.existsSync(ue))try{await Et(b.screenshots[O.stepId],ue),w[O.stepId]=f.screenshotS3Uris[O.stepId],_++}catch(Bo){console.warn(`[reporter] Screenshot upload failed for step ${O.stepId}:`,Bo)}}})),_>0&&console.log(`[reporter] [3/4] Uploaded ${_} screenshot(s) for "${d.title}"`);let T;if(d.videoPath&&b.video){let O=Ke(t,d.videoPath);if(q.existsSync(O)){console.log(`[reporter] [3/4] Uploading video for "${d.title}"...`);try{await Et(b.video,O),T=f.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${d.title}"`)}catch(ue){console.warn("[reporter] Video upload failed:",ue)}}}let $;if(d.tracePath&&b.trace){let O=Ke(t,d.tracePath);if(q.existsSync(O)){console.log(`[reporter] [3/4] Uploading trace for "${d.title}"...`);try{await Et(b.trace,O),$=f.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${d.title}"`)}catch(ue){console.warn("[reporter] Trace upload failed:",ue)}}}console.log(`[reporter] [3/4] Uploading report for "${d.title}"...`);let Je=Ks(d,w,T,$),Mt=Buffer.from(JSON.stringify(Je)),It=At(Mt),Ot=await we.post(`${n}/v1/local-runs/${p.testRunId}/results/${f.testCaseResultId}/report-url`,{md5:It},{headers:s}),Fo=Ot.data.reportUrl,Uo=Ot.data.reportS3Uri;return await we.put(Fo,Mt,{headers:{"Content-Type":"application/json","Content-MD5":It}}),console.log(`[reporter] [3/4] Report uploaded for "${d.title}"`),{testCaseResultId:f.testCaseResultId,result:js(d.status),durationMs:d.duration,startTime:d.startTime,endTime:d.endTime,error:d.error,reportS3Uri:Uo,videoS3Uri:T,traceS3Uri:$,metadata:{suiteName:d.suiteName,file:d.file}}}))).filter(d=>!!d);console.log("[reporter] [4/4] Finalising run...");let m=Gs(e.tests);console.log(`[reporter] [4/4] Overall status: ${m}`);let g=await we.put(`${n}/v1/local-runs/${p.testRunId}/complete`,{status:m,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:s});console.log(`
1029
+ Shiplight cloud report: ${g.data.reportUrl}`)}var vo=x(()=>{"use strict"});var Po={};re(Po,{buildGitHubSummary:()=>ko,runReport:()=>Vs});import*as P from"fs";import*as k from"path";async function Vs(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 Ys(e,t,o):await zs(e,t)}async function zs(e,t){let r=e.find(i=>!i.startsWith("--"))||"shiplight-report",o=k.isAbsolute(r)?r:k.join(process.cwd(),r),n=k.join(o,"report-data.json");P.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let s;try{s=JSON.parse(P.readFileSync(n,"utf-8"))}catch(i){console.error(`Error: Failed to parse ${n}`),console.error(i instanceof Error?i.message:String(i)),process.exit(1)}let a=k.join(o,"index.html");if(P.writeFileSync(a,kt(s),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await To(s,o),t)try{let i=(await import("open")).default;await i(a)}catch{}}async function Ys(e,t,r){let o=k.join(process.cwd(),"shiplight-report"),n=e.findIndex(m=>m==="-o"||m==="--output");if(n!==-1&&e[n+1]){let m=e[n+1];o=k.isAbsolute(m)?m:k.join(process.cwd(),m)}let s=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),i=[];for(let m=0;m<e.length;m++){let g=e[m];if(s.has(g)){m++;continue}if(a.has(g))continue;let d=k.isAbsolute(g)?g:k.join(process.cwd(),g);i.push(d)}i.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;P.mkdirSync(k.join(o,"screenshots"),{recursive:!0});for(let m=0;m<i.length;m++){let g=i[m],d=`shard-${m}`,S=k.join(g,"report-data.json");if(!P.existsSync(S)){console.warn(`Warning: No report-data.json found in ${g}, skipping.`);continue}let f;try{f=JSON.parse(P.readFileSync(S,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${S}, skipping.`);continue}console.log(`Merging ${d}: ${f.tests.length} tests from ${g}`),l+=f.totalDuration||0;let b=k.join(g,"screenshots");P.existsSync(b)&&_o(b,k.join(o,"screenshots",d));let w=k.join(o,d);for(let _ of f.tests){let T=[_,..._.attempts||[]];for(let $ of T)Js($.steps,d),$.videoPath&&xo(g,$.videoPath,w)&&($.videoPath=`${d}/${$.videoPath}`),$.tracePath&&xo(g,$.tracePath,w)&&($.tracePath=`${d}/${$.tracePath}`);c.push(_)}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(),shiplightVersion:be};P.writeFileSync(k.join(o,"report-data.json"),JSON.stringify(u,null,2),"utf-8");let h=k.join(o,"index.html");if(P.writeFileSync(h,kt(u),"utf-8"),console.log(`
1030
+ Merged ${c.length} tests from ${p} shards into: ${h}`),await To(u,o),r&&Xs(c),t)try{let m=(await import("open")).default;await m(h)}catch{}}function xo(e,t,r){let o=k.resolve(e,t);return o.startsWith(k.resolve(e)+k.sep)?P.existsSync(o)?(P.mkdirSync(r,{recursive:!0}),P.copyFileSync(o,k.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function _o(e,t){P.mkdirSync(t,{recursive:!0});for(let r of P.readdirSync(e,{withFileTypes:!0})){let o=k.join(e,r.name),n=k.join(t,r.name);r.isDirectory()?_o(o,n):P.copyFileSync(o,n)}}function Js(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function To(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN;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(s=>s.startTime).filter(s=>!!s),n=o.length>0?o.sort()[0]:e.timestamp??new Date().toISOString();try{await So(e,t,n,r)}catch(s){console.warn("[report] Cloud upload failed:",s)}}function $t(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=k.join("tests",k.basename(t));return{name:e.title||k.basename(t),yamlPath:r}}function ko(e){let t=e.filter(i=>!i.file.includes("auth.setup")),r=t.filter(i=>i.flaky),o=t.filter(i=>i.status==="passed"&&!i.flaky),n=t.filter(i=>i.status!=="passed"),s=t.length,a=`## Test Results
1031
1031
 
1032
1032
  `;if(n.length===0&&r.length===0?a+=`\u2705 All ${s} tests passed
1033
1033
 
@@ -1052,9 +1052,9 @@ Merged ${c.length} tests from ${p} shards into: ${h}`),await To(u,o),r&&Xs(c),t)
1052
1052
  `;for(let c of i)a+=`- ${c}
1053
1053
  `;a+=`
1054
1054
  </details>
1055
- `}return a}function Xs(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}P.appendFileSync(t,ko(e)),console.log("GitHub step summary written.")}var Eo=x(()=>{"use strict";yo();vo();Pe()});var Ao,$o=x(()=>{"use strict";Ao="0.1.53"});var Mo={};re(Mo,{runTranspile:()=>Zs});import*as Ve from"path";import{glob as qs}from"glob";async function Zs(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 qs(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,s=0,a=0;for(let i of o.sort()){let c=Ve.resolve(r,i),l=Hr(c,{version:Ao});if(!l.valid){n++,console.log(`
1055
+ `}return a}function Xs(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}P.appendFileSync(t,ko(e)),console.log("GitHub step summary written.")}var Eo=x(()=>{"use strict";yo();vo();Pe()});var Ao,$o=x(()=>{"use strict";Ao="0.1.54"});var Mo={};re(Mo,{runTranspile:()=>Zs});import*as Ve from"path";import{glob as qs}from"glob";async function Zs(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 qs(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,s=0,a=0;for(let i of o.sort()){let c=Ve.resolve(r,i),l=Hr(c,{version:Ao});if(!l.valid){n++,console.log(`
1056
1056
  \u2717 ${i}`);for(let u of l.errors)console.log(` ERROR: ${u}`);continue}a++;let p=Ve.basename(l.specFile);if(l.warnings.length>0){s++,console.log(`\u26A0 ${i} \u2192 ${p}`);for(let u of l.warnings)console.log(` WARNING: ${u}`)}else console.log(`\u2713 ${i} \u2192 ${p}`)}console.log(`
1057
- ${o.length} file(s): ${a} transpiled, ${n} error(s), ${s} warning(s)`),process.exit(n>0?1:0)}var Io=x(()=>{"use strict";mt();$o()});var Co={};re(Co,{runInspect:()=>Qs});import*as ze from"fs";import*as Oo from"path";async function Qs(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(a=>!a.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let n=Oo.resolve(process.cwd(),o);ze.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let s=ze.readFileSync(n,"utf-8");try{let a=xe(s),i=R(s);if(r)ei(i,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${o}: ${a.message}`),process.exit(1)}}function ei(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 n=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${n}`)}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=Lo(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Lo(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=Lo(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var Ro=x(()=>{"use strict";ae()});var No=jo((Kl,ti)=>{ti.exports={name:"shiplightai",version:"0.1.53",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./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","dev:run":"node --import tsx/esm src/cli.ts",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.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"}});Pe();import ri from"dotenv";ri.config();Lt();Ct();function Do(){console.log(`
1057
+ ${o.length} file(s): ${a} transpiled, ${n} error(s), ${s} warning(s)`),process.exit(n>0?1:0)}var Io=x(()=>{"use strict";mt();$o()});var Co={};re(Co,{runInspect:()=>Qs});import*as ze from"fs";import*as Oo from"path";async function Qs(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(a=>!a.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let n=Oo.resolve(process.cwd(),o);ze.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let s=ze.readFileSync(n,"utf-8");try{let a=xe(s),i=R(s);if(r)ei(i,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:i};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${o}: ${a.message}`),process.exit(1)}}function ei(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 n=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${n}`)}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=Lo(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Lo(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=Lo(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var Ro=x(()=>{"use strict";ae()});var No=jo((Kl,ti)=>{ti.exports={name:"shiplightai",version:"0.1.54",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./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","dev:run":"node --import tsx/esm src/cli.ts",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.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"}});Pe();import ri from"dotenv";ri.config();Lt();Ct();function Do(){console.log(`
1058
1058
  Usage: shiplight <command> [options]
1059
1059
 
1060
1060
  Commands:
package/dist/index.js CHANGED
@@ -4366,7 +4366,7 @@ ${p.join(`
4366
4366
  `)}function tc(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...so()),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=ro(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("});"),lo(i,a),i.join(`
4367
4367
  `)}function $n(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var ic=["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"&&ic.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 nc(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=nc(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 io(e,t,i){let n=[],a=oo(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=oo(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 oo(e){let i=Wl({goal:"_hook",statements:e});return me(i).statements??[]}function ro(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 so(){return["import { test, expect } from 'shiplightai/fixture';"]}function lo(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 ao=5;function ho(e,t){let i={expandingPaths:new Set([uo(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(cc(a)){let o=dc(a,t,i);n.push(...o)}else n.push(uc(a,t,i));return n}function cc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function dc(e,t,i){if(i.depth>=ao)throw new Error(`Template expansion exceeded maximum depth of ${ao}. Check for deeply nested or circular template references.`);let n=uo(sc(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=rc(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=no(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=lc(l);for(let[p,g]of Object.entries(s))h=h.split(`<<${p}>>`).join(String(g));l=no(h)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return De(l,n,c)}function uc(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=oc(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 mc(i,n,a,o,t)}return hc(i,n,a,o,t)}function hc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=mo(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=ho(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=co(e),x=me(f);return a&&(Ve(x.statements??[],a,"main"),x.teardown&&Ve(x.teardown,a,"teardown")),{testFlow:x,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 mc(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 x={goal:f.name,statements:f.statements};f.teardown&&(x.teardown=f.teardown);let b=[],m=x;if(a&&typeof x=="object"){let A=ho(x,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=co(m),S=me(y),E=mo(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 mo(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=pc(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 pc(e,t,i){let n=ac("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 po(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=tc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=ec(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"),fc(wc(s),{recursive:!0}),gc(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.53";function go(e){try{return bc(e).mtimeMs}catch{return 0}}var vc=`// @generated by shiplightai v${On}`;function _c(e,t){if(!wo(e)||fo(e,"utf-8").split(`
4369
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),fc(wc(s),{recursive:!0}),gc(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.54";function go(e){try{return bc(e).mtimeMs}catch{return 0}}var vc=`// @generated by shiplightai v${On}`;function _c(e,t){if(!wo(e)||fo(e,"utf-8").split(`
4370
4370
  `,1)[0]!==vc)return!1;let n=go(e);for(let a of t)if(go(a)>n)return!1;return!0}function kc(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);wo(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function bo(e){let t=kc(e.cwd),i=t??xc("**/*.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=fo(o,"utf-8");try{let l=Pn(s,o),c=yc(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&_c(r,[o,...l.referencedTemplatePaths]))continue;let p=po(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 pt from"path";import{globSync as Tc}from"glob";function vo(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new Fn:new Rn(e)}var Ac=".shiplight/action-cache";function xo(e){return e.replace(/\//g,"__")+".json"}function $c(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Rn=class{constructor(t){this.cwd=t;this.cacheDir=pt.join(t,Ac)}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=pt.join(this.cacheDir,xo(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=pt.join(this.cacheDir,xo(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=Tc("*.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(pt.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=$c(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 Cc(e={}){e.dotenv!==!1&&Oc(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=vo(t).loadAll();return bo({cwd:t,actionEntityStores:n}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function Pc(e,...t){return Ic(e,...t)}function Oc(e){let t=[],i=We.resolve(e),n=We.resolve(process.cwd());for(;;){let a=We.join(i,".env");if(_o.existsSync(a)&&t.push(a),i===n)break;let o=We.dirname(i);if(o===i)break;i=o}for(let a of t)Mc.config({path:a})}R();R();Ct();Ze();import{z as Rs}from"zod";var qu=Rs.object({instruction:Rs.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 Fs(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 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 Mt(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 Zu}from"ai";import{convert as Qu}from"html-to-text";async function eh(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||{},x=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&&!x.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:x,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 x=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(x.ok){let b=await x.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=Qu(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 ${x.status}`)}catch(x){u.warn(`Failed to parse JSON response: ${x}`)}}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 x=await f.text();u.info(`Fallback: Raw email length: ${x.length}`);let b=x.split(`
package/dist/reporter.js CHANGED
@@ -875,7 +875,7 @@ import*as k from"fs";import*as _ from"path";import{z as s}from"zod";var Z=s.enum
875
875
  });
876
876
  </script>
877
877
  </body>
878
- </html>`}var Te="0.1.53",Ae=Te!=="dev"?Te:void 0;var go=3600*1e3,yo=10080*60*1e3;var ut={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return ut[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function pt(e){return[...e].sort(([t],[r])=>{let i=$e(t),o=$e(r);if(i!==o)return i-o;let n=Ee(t),a=Ee(r);for(let c=0;c<Math.max(n.length,a.length);c++){let d=n[c]??-1,p=a[c]??-1;if(d!==p)return d-p}return 0})}function ht(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function ft(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function X(e,t="main",r,i,o){r===void 0&&(r=!ht(e).has("test.step")),o||(o=new Map);let n=[];for(let a of e){if(a.category==="fixture"||a.category==="test.attach")continue;if(a.category==="hook"){let d=ft(a.title,t);n.push(...X(a.steps,d,r,i,o));continue}if(a.category==="test.step"||r&&(a.category==="expect"||a.category==="pw:api")){let d=o.get(t)??0;o.set(t,d+1);let p=`${t}.${d}`,g={stepId:p,description:a.title,status:a.error?"failure":a.duration===-1?"skipped":"success",duration:a.duration>=0?a.duration:void 0};a.error&&(g.error=a.error.message??a.error.stack),i&&a.location&&i.set(p,a.location),n.push(g),a.steps.length>0&&n.push(...X(a.steps,p,r,i,o))}}return n}var q=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let p of this.collected){let g=p.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(p)}let i=[];for(let[,p]of r.entries()){let g=p[0].test.location.file,y=[],l,u,h;for(let b=0;b<p.length;b++){let{test:T,result:$}=p[b],E=await this.buildReportTest(T,$,g);l=E,u||(u=E.startTime),h=E.endTime,y.push({attemptNumber:b+1,status:$.status,duration:$.duration,steps:E.steps,error:E.error,videoPath:E.videoPath,tracePath:E.tracePath})}let m=y[y.length-1],{test:x}=p[p.length-1],v={title:x.title,baseTitle:l?.baseTitle,file:_.relative(process.cwd(),g),status:m.status,duration:m.duration,steps:m.steps,error:m.error,videoPath:m.videoPath,tracePath:m.tracePath,actionStepsMap:l?.actionStepsMap,tags:l?.tags,baseUrl:l?.baseUrl,skip:l?.skip,slow:l?.slow,timeout:l?.timeout,parameterSetName:l?.parameterSetName,startTime:u,endTime:h,suiteName:l?.suiteName},w=y.some(b=>b.status==="failed"||b.status==="timedOut");y.length>1&&w&&m.status==="passed"&&(v.flaky=!0,v.retries=y.length-1,v.attempts=y),i.push(v)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},n=_.isAbsolute(this.outputFolder)?this.outputFolder:_.join(process.cwd(),this.outputFolder);k.mkdirSync(n,{recursive:!0});let a=_.join(n,"screenshots");for(let p=0;p<o.tests.length;p++){let g=o.tests[p],y=g.attempts&&g.attempts.length>0,l=[{obj:g,prefix:y?`test-${p}-attempt-0`:`test-${p}`,screenshotSubDir:`test-${p}`}];if(g.attempts)for(let u=0;u<g.attempts.length;u++)l.push({obj:g.attempts[u],prefix:`test-${p}-attempt-${u+1}`,screenshotSubDir:`test-${p}/attempt-${u}`});for(let{obj:u,prefix:h,screenshotSubDir:m}of l){let x=_.join(a,m),v=!1;for(let w of u.steps)if(w.screenshot&&_.isAbsolute(w.screenshot))try{v||(k.mkdirSync(x,{recursive:!0}),v=!0);let b=`${w.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(w.screenshot,_.join(x,b)),w.screenshot=`screenshots/${m}/${b}`}catch(b){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,b)}if(u.videoPath&&_.isAbsolute(u.videoPath)){let w=_.extname(u.videoPath)||".webm",b=`${h}-video${w}`;try{k.copyFileSync(u.videoPath,_.join(n,b)),u.videoPath=b}catch{u.videoPath=void 0}}if(u.tracePath&&_.isAbsolute(u.tracePath)){let w=_.extname(u.tracePath)||".zip",b=`${h}-trace${w}`;try{k.copyFileSync(u.tracePath,_.join(n,b)),u.tracePath=b}catch{u.tracePath=void 0}}}}let c=_.join(n,"report-data.json");k.writeFileSync(c,JSON.stringify(o,null,2),"utf-8");let d=_.join(n,"index.html");if(k.writeFileSync(d,ke(o),"utf-8"),console.log(`
878
+ </html>`}var Te="0.1.54",Ae=Te!=="dev"?Te:void 0;var go=3600*1e3,yo=10080*60*1e3;var ut={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return ut[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function pt(e){return[...e].sort(([t],[r])=>{let i=$e(t),o=$e(r);if(i!==o)return i-o;let n=Ee(t),a=Ee(r);for(let c=0;c<Math.max(n.length,a.length);c++){let d=n[c]??-1,p=a[c]??-1;if(d!==p)return d-p}return 0})}function ht(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function ft(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function X(e,t="main",r,i,o){r===void 0&&(r=!ht(e).has("test.step")),o||(o=new Map);let n=[];for(let a of e){if(a.category==="fixture"||a.category==="test.attach")continue;if(a.category==="hook"){let d=ft(a.title,t);n.push(...X(a.steps,d,r,i,o));continue}if(a.category==="test.step"||r&&(a.category==="expect"||a.category==="pw:api")){let d=o.get(t)??0;o.set(t,d+1);let p=`${t}.${d}`,g={stepId:p,description:a.title,status:a.error?"failure":a.duration===-1?"skipped":"success",duration:a.duration>=0?a.duration:void 0};a.error&&(g.error=a.error.message??a.error.stack),i&&a.location&&i.set(p,a.location),n.push(g),a.steps.length>0&&n.push(...X(a.steps,p,r,i,o))}}return n}var q=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let p of this.collected){let g=p.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(p)}let i=[];for(let[,p]of r.entries()){let g=p[0].test.location.file,y=[],l,u,h;for(let b=0;b<p.length;b++){let{test:T,result:$}=p[b],E=await this.buildReportTest(T,$,g);l=E,u||(u=E.startTime),h=E.endTime,y.push({attemptNumber:b+1,status:$.status,duration:$.duration,steps:E.steps,error:E.error,videoPath:E.videoPath,tracePath:E.tracePath})}let m=y[y.length-1],{test:x}=p[p.length-1],v={title:x.title,baseTitle:l?.baseTitle,file:_.relative(process.cwd(),g),status:m.status,duration:m.duration,steps:m.steps,error:m.error,videoPath:m.videoPath,tracePath:m.tracePath,actionStepsMap:l?.actionStepsMap,tags:l?.tags,baseUrl:l?.baseUrl,skip:l?.skip,slow:l?.slow,timeout:l?.timeout,parameterSetName:l?.parameterSetName,startTime:u,endTime:h,suiteName:l?.suiteName},w=y.some(b=>b.status==="failed"||b.status==="timedOut");y.length>1&&w&&m.status==="passed"&&(v.flaky=!0,v.retries=y.length-1,v.attempts=y),i.push(v)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},n=_.isAbsolute(this.outputFolder)?this.outputFolder:_.join(process.cwd(),this.outputFolder);k.mkdirSync(n,{recursive:!0});let a=_.join(n,"screenshots");for(let p=0;p<o.tests.length;p++){let g=o.tests[p],y=g.attempts&&g.attempts.length>0,l=[{obj:g,prefix:y?`test-${p}-attempt-0`:`test-${p}`,screenshotSubDir:`test-${p}`}];if(g.attempts)for(let u=0;u<g.attempts.length;u++)l.push({obj:g.attempts[u],prefix:`test-${p}-attempt-${u+1}`,screenshotSubDir:`test-${p}/attempt-${u}`});for(let{obj:u,prefix:h,screenshotSubDir:m}of l){let x=_.join(a,m),v=!1;for(let w of u.steps)if(w.screenshot&&_.isAbsolute(w.screenshot))try{v||(k.mkdirSync(x,{recursive:!0}),v=!0);let b=`${w.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(w.screenshot,_.join(x,b)),w.screenshot=`screenshots/${m}/${b}`}catch(b){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,b)}if(u.videoPath&&_.isAbsolute(u.videoPath)){let w=_.extname(u.videoPath)||".webm",b=`${h}-video${w}`;try{k.copyFileSync(u.videoPath,_.join(n,b)),u.videoPath=b}catch{u.videoPath=void 0}}if(u.tracePath&&_.isAbsolute(u.tracePath)){let w=_.extname(u.tracePath)||".zip",b=`${h}-trace${w}`;try{k.copyFileSync(u.tracePath,_.join(n,b)),u.tracePath=b}catch{u.tracePath=void 0}}}}let c=_.join(n,"report-data.json");k.writeFileSync(c,JSON.stringify(o,null,2),"utf-8");let d=_.join(n,"index.html");if(k.writeFileSync(d,ke(o),"utf-8"),console.log(`
879
879
  Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let p=(await import("open")).default;await p(d)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:_.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(l=>l.message||l.stack||String(l)).join(`
880
880
 
881
881
  `)),r.stdout.length>0&&(o.stdout=r.stdout.map(l=>typeof l=="string"?l:l.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(l=>typeof l=="string"?l:l.toString()).join(""));for(let l of r.attachments)l.name==="video"&&l.path&&(o.videoPath=l.path),l.name==="trace"&&l.path&&(o.tracePath=l.path);let n=r.attachments.find(l=>l.name==="shiplight-results"),a=null;if(n)try{if(n.body)a=JSON.parse(n.body.toString("utf-8"));else if(n.path){let l=k.readFileSync(n.path,"utf-8");a=JSON.parse(l)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},p=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),g=p?p[1]:t.title,y=p?p[2]:void 0;if(y&&(o.parameterSetName=y),k.existsSync(c))try{let l=k.readFileSync(c,"utf-8"),u=ve(l,c);if(u.suite){let h=u.suite.tests.find(m=>m.name===g);h&&(d=K(h.testFlow),h.tags?.length&&(o.tags=h.tags),h.skip!==void 0&&(o.skip=h.skip),h.slow&&(o.slow=h.slow),h.timeout!==void 0&&(o.timeout=h.timeout),o.baseTitle=h.name||h.testFlow?.goal),o.suiteName=u.name,u.tags?.length&&(o.suiteTags=u.tags),u.use?.baseURL&&(o.baseUrl=u.use.baseURL)}else u.testFlow&&(d=K(u.testFlow),o.baseTitle=u.name||u.testFlow?.goal,u.tags?.length&&(o.tags=u.tags),u.use?.baseURL&&(o.baseUrl=u.use.baseURL))}catch{}if(a||Object.keys(d).length>0){let l=new Set([...Object.keys(d),...Object.keys(a||{})]),u=Array.from(l).map(m=>[m,null]),h=pt(u);for(let[m]of h){let x=d[m],v=a?.[m],w=x?.description;if(!w||w==="Action"||w==="Draft"){let T=x?.action_entity;w=T?.action_description||T?.action_data?.kwargs?.description||v?.description||m}let b={stepId:m,description:w,status:v?.status||"pending",duration:v?.duration};if(v?.message){let T=typeof v.message=="string"?v.message:JSON.stringify(v.message,null,2);v.status==="failure"?b.error=T:b.message=T}if(v?.screenshot){let T=v.screenshot,$=n?.path?_.dirname(n.path):"",E=_.isAbsolute(T)?T:_.join($,T);k.existsSync(E)&&(b.screenshot=E)}v?.code&&(b.code=v.code),o.steps.push(b)}}if(a===null&&Object.keys(d).length===0&&!i.endsWith(".yaml.spec.ts")){let l=new Map;if(o.steps=X(r.steps,"main",void 0,l),o.steps.length>0){let u=new Map;o.actionStepsMap=Object.fromEntries(o.steps.map(h=>{let m=l.get(h.stepId),x;if(m?.file){if(!u.has(m.file))try{u.set(m.file,k.readFileSync(m.file,"utf-8").split(`
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "shiplightai",
3
- "version": "0.1.53",
3
+ "version": "0.1.54",
4
4
  "type": "module",
5
5
  "description": "Shiplight CLI for running and debugging .test.yaml files",
6
6
  "main": "dist/index.js",
@@ -90,12 +90,12 @@
90
90
  "@types/node": "^24.0.0",
91
91
  "tsup": "^8.3.5",
92
92
  "typescript": "5.5.4",
93
- "mcp-tools": "1.0.0",
94
93
  "sdk-core": "0.1.0",
95
- "sdk-internal": "0.1.1",
96
- "shiplight-tools": "1.0.0",
97
94
  "shiplight-types": "0.1.0",
98
- "@loggia/common": "1.0.0"
95
+ "mcp-tools": "1.0.0",
96
+ "@loggia/common": "1.0.0",
97
+ "shiplight-tools": "1.0.0",
98
+ "sdk-internal": "0.1.1"
99
99
  },
100
100
  "peerDependencies": {
101
101
  "@playwright/test": "1.58.2"