shiplightai 0.1.59 → 0.1.60

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 Jc(e,t){let i=[],a=t?.version||"unknown";i.push(`// @generated by shiplightai v${a}`),i.push(...Yo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=ea(t?.tags);i.push(`test.describe.serial('${s}${Se(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...Mi("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...Mi("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=Xo(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(...Mi("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...Mi("afterAll",e.afterAll,o,1)),i.push("});"),Jo(i,n),i.join(`
4365
4365
  `)}function ea(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var qc=["testContext","request"];function Pi(e){let t=new Set;function i(a){for(let n of a)switch(n.type){case me.ACTION:{let r=n.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&qc.includes(s)&&t.add(s);break}case me.STEP:i(n.statements);break;case me.IF_ELSE:{let o=n;i(o.then),o.else&&i(o.else);break}case me.WHILE_LOOP:i(n.body);break}}return i(e),t}function Zc(e){let t=Pi(e.statements??[]);if(e.teardown)for(let i of Pi(e.teardown))t.add(i);return t}function ia(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Ci(e,t,i,a=0,n){let o=" ".repeat(a),r=[],s=Zc(e),l=ia(s),c=n?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),n?.skip===!0?r.push(`${o} test.skip();`):typeof n?.skip=="string"&&r.push(`${o} test.skip(true, '${Se(n.skip)}');`),n?.fail===!0?r.push(`${o} test.fail();`):typeof n?.fail=="string"&&r.push(`${o} test.fail(true, '${Se(n.fail)}');`),n?.slow&&r.push(`${o} test.slow();`),n?.timeout&&r.push(`${o} test.setTimeout(${n.timeout});`);let d=e.teardown&&e.teardown.length>0,h=a+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 jo(e,t,i){let a=[],n=Vo(t),o=Pi(n),r=ia(o);return a.push(`test.${e}(async (${r}) => {`),a.push(...ge(n,1,i,e)),a.push("});"),a}function Mi(e,t,i,a){let n=" ".repeat(a),o=[],r=Vo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...ge(r,a+1,l,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let l=Pi(r),c=ia(l);o.push(`${n}test.${e}(async (${c}) => {`),o.push(...ge(r,a+1,i,e)),o.push(`${n}});`)}return o}function Vo(e){let i=(0,zo.stringify)({goal:"_hook",statements:e});return pe(i).statements??[]}function Xo(e,t){let i=Yn(e);for(let[a,n]of Object.entries(t))i=i.split(`<<${a}>>`).join(String(n));return pe(i)}function Yo(){return["import { test, expect } from 'shiplightai/fixture';"]}function Jo(e,t){if(t.size>0){let i=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(i=n+1);let a=Array.from(t);e.splice(i,0,...a)}}var Ko=5;function Zo(e,t,i){let a={expandingPaths:new Set([(0,qe.resolve)(t)]),depth:0,referencedPaths:new Set,basePath:i},n={...e};Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,a)),Array.isArray(n.teardown)&&(n.teardown=Re(n.teardown,t,a));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=Re(n[o],t,a));return{doc:n,referencedTemplatePaths:Array.from(a.referencedPaths)}}function Re(e,t,i){let a=[];for(let n of e)if(Qc(n)){let o=ed(n,t,i);a.push(o)}else a.push(td(n,t,i));return a}function Qc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function ed(e,t,i){if(i.depth>=Ko)throw new Error(`Template expansion exceeded maximum depth of ${Ko}. Check for deeply nested or circular template references.`);let a=(0,qe.resolve)((0,qe.dirname)(t),e.template),n=!(0,Oi.existsSync)(a)&&i.basePath?(0,qe.resolve)(i.basePath,e.template):a;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 o;try{o=(0,Oi.readFileSync)(n,"utf-8")}catch(w){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${w.message}`)}let r=(0,xt.parse)(o);if(!r||typeof r!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let s=r.params||[],l=e.params||{};for(let w of s)if(!(w in l))throw new Error(`Template ${e.template} requires param "${w}" but it was not provided. Required params: [${s.join(", ")}]`);let c=r.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let y=(0,xt.stringify)(c);for(let[b,m]of Object.entries(l))y=y.split(`<<${b}>>`).join(String(m));c=(0,xt.parse)(y)}let d={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths},h=Re(c,n,d),g={STEP:r.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:h};return Object.keys(l).length>0&&(g.template_params=l),g}function td(e,t,i){if(typeof e!="object"||e===null)return e;let a={...e};return Array.isArray(a.statements)&&(a.statements=Re(a.statements,t,i)),Array.isArray(a.THEN)&&(a.THEN=Re(a.THEN,t,i)),Array.isArray(a.ELSE)&&(a.ELSE=Re(a.ELSE,t,i)),Array.isArray(a.DO)&&(a.DO=Re(a.DO,t,i)),a}var na=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function aa(e,t,i){let a=(0,vt.parse)(e),n=a?.name,o=a?.tags,r=a?.use;if(a&&(a.name!==void 0||a.tags!==void 0||a.use!==void 0)&&(delete a.name,delete a.tags,delete a.use),a?.suite){if(a.goal||a.statements)throw new na('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return nd(a,n,o,r,t,i)}return id(a,n,o,r,t,i)}function id(e,t,i,a,n,o){let r=e?.beforeEach,s=e?.afterEach,l=Qo(e?.parameters),c=e?.timeout,d=e?.skip,h=e?.fail,p=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new na(`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 w=[];if(n&&e&&typeof e=="object"){let m=Zo(e,n,o);e=m.doc,w=m.referencedTemplatePaths}let y=(0,vt.stringify)(e),b=pe(y);return n&&(Je(b.statements??[],n,"main"),b.teardown&&Je(b.teardown,n,"teardown")),{testFlow:b,name:t,tags:i,use:a,beforeEach:r,afterEach:s,parameters:l,timeout:c,skip:d,fail:h,only:p,slow:g,referencedTemplatePaths:w}}function nd(e,t,i,a,n,o){let r=e.suite;if(!Array.isArray(r.tests)||r.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=r.beforeAll,l=r.afterAll,c=r.beforeEach,d=r.afterEach,h=[],p=r.tests.map(y=>{if(!y.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(y.statements)||y.statements.length===0)throw new Error(`Suite test "${y.name}" must have a non-empty "statements" array.`);let b={goal:y.name,statements:y.statements};y.teardown&&(b.teardown=y.teardown);let m=[],x=b;if(n&&typeof b=="object"){let $=Zo(b,n,o);x=$.doc,m=$.referencedTemplatePaths,h.push(...m)}let S=(0,vt.stringify)(x),E=pe(S),A=Qo(y.parameters);return{testFlow:E,name:y.name,tags:Array.isArray(y.tags)?y.tags:void 0,parameters:A,timeout:y.timeout,skip:y.skip,fail:y.fail,only:y.only,slow:y.slow}}),g=r.base_url,w=g?{...a,baseURL:g}:a;return{suite:{beforeAll:s,afterAll:l,beforeEach:c,afterEach:d,tests:p},name:t,tags:i,use:w,referencedTemplatePaths:h}}function Qo(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 Je(e,t,i){for(let a=0;a<e.length;a++){let n=e[a],o=`${i}.${a}`,r=n.description||"";if(n.uid=ad(t,o,r),n.type===me.STEP)Je(n.statements,t,o);else if(n.type===me.IF_ELSE){let s=n;Je(s.then,t,`${o}.then`),s.else&&Je(s.else,t,`${o}.else`)}else n.type===me.WHILE_LOOP&&Je(n.body,t,`${o}.body`)}}function ad(e,t,i){let a=(0,qo.createHash)("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${a.slice(0,8)}-${a.slice(8,12)}-${a.slice(12,16)}-${a.slice(16,20)}-${a.slice(20,32)}`}function tr(e,t,i){let a=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=a||n?null:qn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??aa(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=Jc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Yc(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,_t.mkdirSync)((0,er.dirname)(s),{recursive:!0}),(0,_t.writeFileSync)(s,r)}catch(c){let d=c instanceof na?"":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 oa="0.1.59";function ir(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var od=`// @generated by shiplightai v${oa}`;function rd(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,_t.mkdirSync)((0,er.dirname)(s),{recursive:!0}),(0,_t.writeFileSync)(s,r)}catch(c){let d=c instanceof na?"":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 oa="0.1.60";function ir(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var od=`// @generated by shiplightai v${oa}`;function rd(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4368
4368
  `,1)[0]!==od)return!1;let a=ir(e);for(let n of t)if(ir(n)>a)return!1;return!0}function sd(e){let t=process.argv.slice(2),i=[],a=(0,Ze.resolve)(e);for(let n of t){if(n.startsWith("-"))continue;let o=n.endsWith(".yaml.spec.ts")?n.replace(/\.yaml\.spec\.ts$/,".test.yaml"):n;if(!o.endsWith(".test.yaml"))continue;let r=(0,Ze.resolve)(e,o);(0,Fe.existsSync)(r)&&i.push(r.startsWith(a)?r.slice(a.length+1):o)}return i.length>0?i:null}function ar(e){let t=sd(e.cwd),i=t??(0,nr.globSync)("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),a=[];for(let n of i){let o=(0,Ze.resolve)(e.cwd,n),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=(0,Fe.readFileSync)(o,"utf-8");try{let l=aa(s,o,e.projectRoot??e.cwd),c=(0,Ze.relative)(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&rd(r,[o,...l.referencedTemplatePaths]))continue;let p=tr(s,o,{version:oa,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${n}:`,l),a.push({file:n,error:l})}}if(a.length>0){let n=`[shiplight] Transpilation failed for ${a.length} file(s):
4369
4369
  `+a.map(o=>` - ${o.file}`).join(`
4370
4370
  `);if(t)throw new Error(n);console.warn(n+" (skipped)")}}var re=X(require("fs"),1),kt=X(require("path"),1),hr=require("glob");Di();function mr(e){let t=Ni().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new da:new ca(e)}var wd=".shiplight/action-cache";function ur(e){return e.replace(/\//g,"__")+".json"}function bd(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ca=class{constructor(t){this.cwd=t;this.cacheDir=kt.join(t,wd)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!re.existsSync(this.cacheDir))return i;for(let a of t){let n=kt.join(this.cacheDir,ur(a));try{if(re.existsSync(n)){let o=re.readFileSync(n,"utf-8");i.set(a,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[a,n]of t)try{let o=kt.join(this.cacheDir,ur(a)),r=Zn();if(re.existsSync(o))try{r=JSON.parse(re.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...n.entries}};re.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!re.existsSync(this.cacheDir))return;let t=(0,hr.globSync)("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,a=0;for(let n of t)try{let o=re.readFileSync(kt.join(this.cacheDir,n),"utf-8"),r=JSON.parse(o),s=bd(n);i.set(s,r),a+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${a} cached action entit${a===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},da=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(la(),sa));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(la(),sa));return i(t)}loadAll(){}};Di();function fr(e={}){e.dotenv!==!1&&yd(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),a=mr(t).loadAll();return ar({cwd:t,projectRoot:process.cwd(),actionEntityStores:a}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function wr(e,...t){return(0,gr.defineConfig)(e,...t)}function yd(e){sr(e);for(let t of ra(e))pr.default.config({path:t})}F();F();Bt();nt();var oo=require("zod"),xh=oo.z.object({instruction:oo.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 Ol(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:xh,usesElementIndex:!1,async execute(t,i){let{instruction:a}=t,n={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await it(a,n,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:a,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:a}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Wt(r,n);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 Ll=require("ai"),Nl=require("html-to-text");async function vh(e,t,i){let{apiKey:a,domain:n}=e;if(!a||!n)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${n}/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:${a}`).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 w=(d.message||{}).headers||{},y=w.subject||"",b=w.from||"",m=w.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!y.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:y,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:w["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let w=`https://api.mailgun.net/v3/domains/${n}/messages/${g}`;try{let y=await fetch(w,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`,Accept:"application/json"}});if(y.ok){let b=await y.json(),m=b.Subject||"",x=b.From||"",S=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${x}`),u.info(`to_addr: ${S}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let $=b["body-html"]||b["body-plain"]||"";if($&&$.includes("<")&&($=(0,Nl.convert)($)),u.info(`Body: ${$.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!$.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:x,to:S,date:E,body:$,message_id:A});continue}else u.warn(`Messages API returned ${y.status}`)}catch(y){u.warn(`Failed to parse JSON response: ${y}`)}}try{let w=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`}});if(!w.ok){u.warn(`Could not fetch stored message: ${w.status}`);continue}let y=await w.text();u.info(`Fallback: Raw email length: ${y.length}`);let b=y.split(`
@@ -873,7 +873,7 @@
873
873
  });
874
874
  </script>
875
875
  </body>
876
- </html>`}var Pe="0.1.59",Oe=Pe!=="dev"?Pe:void 0;var sn=3600*1e3,cn=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 _t(e){return[...e].sort(([t],[r])=>{let i=Ie(t),n=Ie(r);if(i!==n)return i-n;let a=Le(t),o=Le(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function vt(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 te(e,t="main",r,i,n){r===void 0&&(r=!vt(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=St(o.title,t);a.push(...te(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,g={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(g.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(g),o.steps.length>0&&a.push(...te(o.steps,u,r,i,n))}}return a}var re=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 u of this.collected){let g=u.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(u)}let i=[];for(let[,u]of r.entries()){let g=u[0].test.location.file,y=[],d,p,m;for(let w=0;w<u.length;w++){let{test:A,result:P}=u[w],$=await this.buildReportTest(A,P,g);d=$,p||(p=$.startTime),m=$.endTime,y.push({attemptNumber:w+1,status:P.status,duration:P.duration,steps:$.steps,error:$.error,videoPath:$.videoPath,tracePath:$.tracePath})}let h=y[y.length-1],{test:S}=u[u.length-1],b={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),g),status:h.status,duration:h.duration,steps:h.steps,error:h.error,videoPath:h.videoPath,tracePath:h.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:p,endTime:m,suiteName:d?.suiteName},_=y.some(w=>w.status==="failed"||w.status==="timedOut");y.length>1&&_&&h.status==="passed"&&(b.flaky=!0,b.retries=y.length-1,b.attempts=y),i.push(b)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);T.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let g=n.tests[u],y=g.attempts&&g.attempts.length>0,d=[{obj:g,prefix:y?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(g.attempts)for(let p=0;p<g.attempts.length;p++)d.push({obj:g.attempts[p],prefix:`test-${u}-attempt-${p+1}`,screenshotSubDir:`test-${u}/attempt-${p}`});for(let{obj:p,prefix:m,screenshotSubDir:h}of d){let S=v.join(o,h),b=!1;for(let _ of p.steps)if(_.screenshot&&v.isAbsolute(_.screenshot))try{b||(T.mkdirSync(S,{recursive:!0}),b=!0);let w=`${_.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(_.screenshot,v.join(S,w)),_.screenshot=`screenshots/${h}/${w}`}catch(w){console.warn(`[reporter] Failed to copy screenshot for ${_.stepId}:`,w)}if(p.videoPath&&v.isAbsolute(p.videoPath)){let _=v.extname(p.videoPath)||".webm",w=`${m}-video${_}`;try{T.copyFileSync(p.videoPath,v.join(a,w)),p.videoPath=w}catch{p.videoPath=void 0}}if(p.tracePath&&v.isAbsolute(p.tracePath)){let _=v.extname(p.tracePath)||".zip",w=`${m}-trace${_}`;try{T.copyFileSync(p.tracePath,v.join(a,w)),p.tracePath=w}catch{p.tracePath=void 0}}}}let c=v.join(a,"report-data.json");T.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(T.writeFileSync(l,Me(n),"utf-8"),console.log(`
876
+ </html>`}var Pe="0.1.60",Oe=Pe!=="dev"?Pe:void 0;var sn=3600*1e3,cn=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 _t(e){return[...e].sort(([t],[r])=>{let i=Ie(t),n=Ie(r);if(i!==n)return i-n;let a=Le(t),o=Le(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function vt(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 te(e,t="main",r,i,n){r===void 0&&(r=!vt(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=St(o.title,t);a.push(...te(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,g={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(g.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(g),o.steps.length>0&&a.push(...te(o.steps,u,r,i,n))}}return a}var re=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 u of this.collected){let g=u.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(u)}let i=[];for(let[,u]of r.entries()){let g=u[0].test.location.file,y=[],d,p,m;for(let w=0;w<u.length;w++){let{test:A,result:P}=u[w],$=await this.buildReportTest(A,P,g);d=$,p||(p=$.startTime),m=$.endTime,y.push({attemptNumber:w+1,status:P.status,duration:P.duration,steps:$.steps,error:$.error,videoPath:$.videoPath,tracePath:$.tracePath})}let h=y[y.length-1],{test:S}=u[u.length-1],b={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),g),status:h.status,duration:h.duration,steps:h.steps,error:h.error,videoPath:h.videoPath,tracePath:h.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:p,endTime:m,suiteName:d?.suiteName},_=y.some(w=>w.status==="failed"||w.status==="timedOut");y.length>1&&_&&h.status==="passed"&&(b.flaky=!0,b.retries=y.length-1,b.attempts=y),i.push(b)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);T.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let g=n.tests[u],y=g.attempts&&g.attempts.length>0,d=[{obj:g,prefix:y?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(g.attempts)for(let p=0;p<g.attempts.length;p++)d.push({obj:g.attempts[p],prefix:`test-${u}-attempt-${p+1}`,screenshotSubDir:`test-${u}/attempt-${p}`});for(let{obj:p,prefix:m,screenshotSubDir:h}of d){let S=v.join(o,h),b=!1;for(let _ of p.steps)if(_.screenshot&&v.isAbsolute(_.screenshot))try{b||(T.mkdirSync(S,{recursive:!0}),b=!0);let w=`${_.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(_.screenshot,v.join(S,w)),_.screenshot=`screenshots/${h}/${w}`}catch(w){console.warn(`[reporter] Failed to copy screenshot for ${_.stepId}:`,w)}if(p.videoPath&&v.isAbsolute(p.videoPath)){let _=v.extname(p.videoPath)||".webm",w=`${m}-video${_}`;try{T.copyFileSync(p.videoPath,v.join(a,w)),p.videoPath=w}catch{p.videoPath=void 0}}if(p.tracePath&&v.isAbsolute(p.tracePath)){let _=v.extname(p.tracePath)||".zip",w=`${m}-trace${_}`;try{T.copyFileSync(p.tracePath,v.join(a,w)),p.tracePath=w}catch{p.tracePath=void 0}}}}let c=v.join(a,"report-data.json");T.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(T.writeFileSync(l,Me(n),"utf-8"),console.log(`
877
877
  Shiplight report written to: ${l}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(l)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let n={title:t.title,file:v.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&&(n.error=r.errors.map(d=>d.message||d.stack||String(d)).join(`
878
878
 
879
879
  `)),r.stdout.length>0&&(n.stdout=r.stdout.map(d=>typeof d=="string"?d:d.toString()).join("")),r.stderr.length>0&&(n.stderr=r.stderr.map(d=>typeof d=="string"?d:d.toString()).join(""));for(let d of r.attachments)d.name==="video"&&d.path&&(n.videoPath=d.path),d.name==="trace"&&d.path&&(n.tracePath=d.path);let a=r.attachments.find(d=>d.name==="shiplight-results"),o=null;if(a)try{if(a.body)o=JSON.parse(a.body.toString("utf-8"));else if(a.path){let d=T.readFileSync(a.path,"utf-8");o=JSON.parse(d)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),l={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),g=u?u[1]:t.title,y=u?u[2]:void 0;if(y&&(n.parameterSetName=y),T.existsSync(c))try{let d=T.readFileSync(c,"utf-8"),p=Te(d,c);if(p.suite){let m=p.suite.tests.find(h=>h.name===g);m&&(l=z(m.testFlow),m.tags?.length&&(n.tags=m.tags),m.skip!==void 0&&(n.skip=m.skip),m.slow&&(n.slow=m.slow),m.timeout!==void 0&&(n.timeout=m.timeout),n.baseTitle=m.name||m.testFlow?.goal),n.suiteName=p.name,p.tags?.length&&(n.suiteTags=p.tags),p.use?.baseURL&&(n.baseUrl=p.use.baseURL)}else p.testFlow&&(l=z(p.testFlow),n.baseTitle=p.name||p.testFlow?.goal,p.tags?.length&&(n.tags=p.tags),p.use?.baseURL&&(n.baseUrl=p.use.baseURL))}catch{}if(o||Object.keys(l).length>0){let d=new Set([...Object.keys(l),...Object.keys(o||{})]),p=Array.from(d).map(h=>[h,null]),m=_t(p);for(let[h]of m){let S=l[h],b=o?.[h],_=S?.description;if(!_||_==="Action"||_==="Draft"){let A=S?.action_entity;_=A?.action_description||A?.action_data?.kwargs?.description||b?.description||h}let w={stepId:h,description:_,status:b?.status||"pending",duration:b?.duration};if(b?.message){let A=typeof b.message=="string"?b.message:JSON.stringify(b.message,null,2);b.status==="failure"?w.error=A:w.message=A}if(b?.screenshot){let A=b.screenshot,P=a?.path?v.dirname(a.path):"",$=v.isAbsolute(A)?A:v.join(P,A);T.existsSync($)&&(w.screenshot=$)}b?.code&&(w.code=b.code),n.steps.push(w)}}if(o===null&&Object.keys(l).length===0&&!i.endsWith(".yaml.spec.ts")){let d=new Map;if(n.steps=te(r.steps,"main",void 0,d),n.steps.length>0){let p=new Map;n.actionStepsMap=Object.fromEntries(n.steps.map(m=>{let h=d.get(m.stepId),S;if(h?.file){if(!p.has(h.file))try{p.set(h.file,T.readFileSync(h.file,"utf-8").split(`
package/dist/cli.js CHANGED
@@ -17,7 +17,7 @@ Install it as a project dependency instead:
17
17
  `)}catch{}}function ss(e=st()){try{let t=O.readFileSync(e,"utf-8"),n=JSON.parse(t);if(typeof n.latest=="string"&&typeof n.fetchedAt=="number"&&Date.now()-n.fetchedAt<es)return n}catch{}return null}function os(e,t=st()){try{O.mkdirSync(W.dirname(t),{recursive:!0}),O.writeFileSync(t,JSON.stringify(e))}catch{}}async function is(){try{let e=await fetch(Qr,{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 as(e,t){let n=d=>{let f=d.indexOf("-");return f===-1?[d,!1]:[d.slice(0,f),!0]},r=d=>d.split(".").map(f=>parseInt(f,10)||0),[s,o]=n(e),[i,a]=n(t),c=r(s),l=r(i),p=Math.max(c.length,l.length);for(let d=0;d<p;d++){let f=c[d]??0,m=l[d]??0;if(f<m)return!0;if(f>m)return!1}return!!(o&&!a)}async function Kt(e={}){let t=e.runningVersion??nt,n=e.cwd??process.cwd(),r=e.cacheFile??st(),s=e.fetchLatest??is,o=e.env??process.env,i=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!O.existsSync(W.join(n,"package-lock.json")))return;let a=null,c=ss(r);if(c)a=c.latest;else{try{a=await s()}catch{a=null}a&&os({latest:a,fetchedAt:Date.now()},r)}a&&as(t,a)&&i(`
18
18
  \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var nt,ve,Qr,es,ts,Pe=_(()=>{"use strict";nt="0.1.59",ve=nt!=="dev"?nt:void 0,Qr="https://registry.npmjs.org/shiplightai/latest",es=3600*1e3,ts=10080*60*1e3});import*as te from"fs";import*as ie from"path";function ot(e){let{projectPath:t}=e,n=e.projectName??ie.basename(ie.resolve(t));if(te.existsSync(t)&&te.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);te.mkdirSync(t,{recursive:!0});let r=(s,o)=>{let i=ie.join(t,s);te.mkdirSync(ie.dirname(i),{recursive:!0}),te.writeFileSync(i,o)};return r("package.json",cs.replace("{{name}}",n)),r("playwright.config.ts",ls),r(".gitignore",ps),r(".env.example",ds),r(".mcp.json",us),r(".claude/CLAUDE.md",hs),r(".claude/auth.md",fs),r(".claude/creating-tests.md",gs),r(".claude/settings.local.json",ms),r(".claude/test-spec-template.md",ys),r(".claude/updating-tests.md",ws),r("auth/example.login.ts",bs),r("environments/example.env.yaml",vs),r("tests/example.test.yaml",Ss),{projectPath:t,projectName:n,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json",".claude/CLAUDE.md",".claude/auth.md",".claude/creating-tests.md",".claude/settings.local.json",".claude/test-spec-template.md",".claude/updating-tests.md","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var cs,ls,ps,ds,us,hs,fs,gs,ms,ys,ws,bs,vs,Ss,Vt=_(()=>{"use strict";cs=`{
20
+ `)}var nt,ve,Qr,es,ts,Pe=_(()=>{"use strict";nt="0.1.60",ve=nt!=="dev"?nt:void 0,Qr="https://registry.npmjs.org/shiplightai/latest",es=3600*1e3,ts=10080*60*1e3});import*as te from"fs";import*as ie from"path";function ot(e){let{projectPath:t}=e,n=e.projectName??ie.basename(ie.resolve(t));if(te.existsSync(t)&&te.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);te.mkdirSync(t,{recursive:!0});let r=(s,o)=>{let i=ie.join(t,s);te.mkdirSync(ie.dirname(i),{recursive:!0}),te.writeFileSync(i,o)};return r("package.json",cs.replace("{{name}}",n)),r("playwright.config.ts",ls),r(".gitignore",ps),r(".env.example",ds),r(".mcp.json",us),r(".claude/CLAUDE.md",hs),r(".claude/auth.md",fs),r(".claude/creating-tests.md",gs),r(".claude/settings.local.json",ms),r(".claude/test-spec-template.md",ys),r(".claude/updating-tests.md",ws),r("auth/example.login.ts",bs),r("environments/example.env.yaml",vs),r("tests/example.test.yaml",Ss),{projectPath:t,projectName:n,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json",".claude/CLAUDE.md",".claude/auth.md",".claude/creating-tests.md",".claude/settings.local.json",".claude/test-spec-template.md",".claude/updating-tests.md","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var cs,ls,ps,ds,us,hs,fs,gs,ms,ys,ws,bs,vs,Ss,Vt=_(()=>{"use strict";cs=`{
21
21
  "name": "{{name}}",
22
22
  "type": "module",
23
23
  "scripts": {
@@ -635,7 +635,7 @@ test('__shiplight_debug__', async ({ page, agent }) => {
635
635
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
636
636
  await new Promise(() => {});
637
637
  });
638
- `}async function Ps(e){let{createServer:t}=await import("net");for(let n=e;n<e+20;n++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(n,"127.0.0.1")}))return n;throw new Error(`No available port found in range ${e}-${e+19}`)}async function at(e){let{yamlFilePath:t,configPath:n,tempSuffix:r="",headed:s}=e,o=q.dirname(n),i=await Ps(16174),a;if(!re.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let y=As(re.readFileSync(t,"utf-8"));y?.use&&typeof y.use=="object"&&!Array.isArray(y.use)&&(a=y.use),y?.base_url&&!a?.baseURL&&(a={...a,baseURL:y.base_url}),y?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!y.settings.auto_dismiss_modal})}catch(y){console.error("[debugger] Could not parse YAML for `use` block:",y)}let c=q.dirname(q.resolve(t)),l=r?`-${r}`:"",p=q.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=Es(t,i,a,o);re.writeFileSync(p,d);let f=()=>{try{re.unlinkSync(p)}catch{}},m=["playwright","test",p,...s?["--headed"]:[]],g=ks("npx",m,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});g.stdout?.on("data",y=>{process.stderr.write(y)}),g.stderr?.on("data",y=>{process.stderr.write(y)});let u=()=>{g.killed||g.kill("SIGTERM")};process.on("SIGTERM",u),process.on("SIGINT",u),process.on("exit",u),g.on("close",y=>{process.removeListener("SIGTERM",u),process.removeListener("SIGINT",u),process.removeListener("exit",u),f(),y!==0&&y!==null&&console.error(`[debugger] Playwright process exited with code ${y}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let v=["127.0.0.1","::1"];async function h(y){try{let x=y.includes(":")?`[${y}]`:y,A=await fetch(`http://${x}:${i}/api/test-flow`);if(A.ok){try{await A.text()}catch{}return!0}}catch{}return!1}let b=null;for(let y=0;y<180;y++){if(g.exitCode!==null)throw f(),new Error(`Playwright process exited with code ${g.exitCode} before sandbox was ready`);for(let x of v)if(await h(x)){b=x;break}if(b){console.error(`[debugger] Playwright sandbox ready on ${b}:${i}`);break}if(y===179)throw u(),f(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(x=>setTimeout(x,1e3))}if(!b)throw u(),f(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:b,pid:g.pid??0,cleanup:async()=>{u(),f()}}}var ct=_(()=>{"use strict"});var rn=_(()=>{"use strict"});var sn=_(()=>{"use strict"});var on=_(()=>{"use strict"});import{v4 as Yi}from"uuid";var an=_(()=>{"use strict"});import{z as w}from"zod";var cn,lt,ln,he,pn,dn,un,N,hn,Oe,pt,dt=_(()=>{"use strict";cn=w.enum(["JS_CODE","AI_MODE"]),lt=w.object({type:cn,expression:w.string()}),ln=w.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),he=w.object({uid:w.string(),type:ln,comment:w.string().optional()}),pn=w.object({action_data:w.object({action_name:w.string(),kwargs:w.record(w.any()).optional(),args:w.array(w.any()).optional()}),action_description:w.string().optional(),url:w.string().optional(),xpath:w.string().nullable().optional(),locator:w.string().nullable().optional(),css_selector:w.string().nullable().optional(),unique_selector:w.string().nullable().optional(),element_index:w.number().nullable().optional(),frame_path:w.array(w.any()).optional(),artifacts:w.record(w.any()).optional(),feedback:w.string().optional(),original_browser_use_action:w.any().optional()}).passthrough(),dn=he.extend({type:w.literal("DRAFT"),description:w.string()}),un=he.extend({type:w.literal("ACTION"),description:w.string(),action_entity:pn.optional(),locator:w.string().optional(),use_pure_vision:w.boolean().optional()}),N=w.lazy(()=>w.union([dn,un,he.extend({type:w.literal("STEP"),description:w.string().optional().default(""),statements:w.array(N),reference_id:w.number().optional(),template_path:w.string().optional(),template_params:w.record(w.string()).optional()}),he.extend({type:w.literal("IF_ELSE"),description:w.string().optional(),condition:lt,then:w.array(N),else:w.array(N).optional()}),he.extend({type:w.literal("WHILE_LOOP"),description:w.string().optional(),condition:lt,body:w.array(N),timeout_ms:w.number().optional()})])),hn=w.object({name:w.string(),statements:w.array(N),teardown:w.array(N).optional(),skip:w.union([w.boolean(),w.string()]).optional(),timeout:w.number().optional(),fail:w.union([w.boolean(),w.string()]).optional(),only:w.boolean().optional(),slow:w.boolean().optional()}),Oe=w.object({tests:w.array(hn).min(1),beforeAll:w.array(N).optional(),afterAll:w.array(N).optional(),beforeEach:w.array(N).optional(),afterEach:w.array(N).optional()}),pt=w.object({comment:w.string().optional(),version:w.string().optional(),goal:w.string().optional(),url:w.string().optional(),baseURL:w.string().optional(),final_feedback:w.string().optional(),completed:w.boolean().optional(),success:w.boolean().optional(),statements:w.array(N).optional(),teardown:w.array(N).optional(),last_modified_at:w.string().optional(),testGroup:Oe.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 $s,parse as yn,parseAllDocuments as ea,parseDocument as Ms,Document as Is,isMap as ge,isSeq as V}from"yaml";import{v4 as K}from"uuid";function Le(e,t){let n={...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,...t?.timeout!==void 0?{timeout:t.timeout}:{},statements:(e.statements??[]).map(B)};return e.final_feedback&&(n.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(n.teardown=e.teardown.map(B)),n}function _e(e,t){if(e.testGroup)return bn(e,t);let n=Le(e,t),r=new Is(n);return e.comment&&(r.commentBefore=e.comment),fn(r,e.statements??[]),e.teardown&&fn(r,e.teardown,"teardown"),r.toString(wn)}function fn(e,t,n="statements"){let r=e.contents;if(!r||!ge(r))return;let s=r.get(n,!0);V(s)&&Se(s,t)}function Se(e,t){for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=t[n],s=e.items[n];if(n>0&&(s.spaceBefore=!0),r.comment&&(n===0?e.commentBefore=r.comment:s.commentBefore=r.comment),ge(s)){let o=s;if(r.type==="STEP"){let i=o.get("statements",!0);V(i)&&Se(i,r.statements)}else if(r.type==="IF_ELSE"){let i=o.get("THEN",!0);V(i)&&Se(i,r.then);let a=o.get("ELSE",!0);V(a)&&r.else&&Se(a,r.else)}else if(r.type==="WHILE_LOOP"){let i=o.get("DO",!0);V(i)&&Se(i,r.body)}}}}function bn(e,t){let n=e.testGroup;if(!n)throw new Error("suiteToYaml requires a TestFlow with testGroup");let r={};t?.test_case_id!==void 0&&(r.test_case_id=t.test_case_id),t?.name&&(r.name=t.name),t?.tags&&t.tags.length>0&&(r.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(r.use=t.use);let s={};return e.baseURL&&(s.base_url=e.baseURL),n.beforeAll&&n.beforeAll.length>0&&(s.beforeAll=n.beforeAll.map(B)),n.beforeEach&&n.beforeEach.length>0&&(s.beforeEach=n.beforeEach.map(B)),n.afterEach&&n.afterEach.length>0&&(s.afterEach=n.afterEach.map(B)),n.afterAll&&n.afterAll.length>0&&(s.afterAll=n.afterAll.map(B)),s.tests=n.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(B),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(B)),i}),r.suite=s,$s(r,wn)}function B(e){switch(e.type){case"DRAFT":return Os(e);case"ACTION":return Ls(e);case"STEP":return Rs(e);case"IF_ELSE":return Cs(e);case"WHILE_LOOP":return Ds(e)}}function Os(e){return{intent:e.description}}function Ls(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,n=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=n?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=n?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=n?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return n?.new_tab===!0&&(c.new_tab=!0),typeof n?.timeout_seconds=="number"&&(c.timeout_seconds=n.timeout_seconds),c}}if(t==="js_action"){let a=n?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=n?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof n?.timeout_seconds=="number"&&n.timeout_seconds!==60&&(c.timeout_seconds=n.timeout_seconds),c}}if(t==="wait"){let a=n?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=n?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let r=e.action_entity.action_data??e.action_entity.action;if(!r)return{intent:e.description};let s={intent:e.description,action:r.action_name},o=e.locator??e.action_entity.locator;o&&(s.locator=o);let i=e.action_entity.xpath;if(i&&(s.xpath=i),e.use_pure_vision&&(s.use_pure_vision=!0),r.kwargs&&Object.keys(r.kwargs).length>0)for(let[a,c]of Object.entries(r.kwargs))s[a]=c;return r.args&&r.args.length>0&&(s.args=r.args),s}function Rs(e){if(e.template_path){let n={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(n.params=e.template_params),n}let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Cs(e){let t={IF:vn(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function Ds(e){let t={WHILE:vn(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function vn(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function xe(e){try{let t=yn(e);if(!t||typeof t!="object")return{};let n={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(n.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(n.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(n.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(n.timeout=t.timeout),n}catch{return{}}}function ut(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(ut);let t=e,n=Object.keys(t);if(n.length===1){let s=n[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let r={};for(let[s,o]of Object.entries(t))r[s]=ut(o);return r}function C(e){if(e.length>gn)throw new Error(`YAML input too large (${e.length} bytes, max ${gn})`);let t=ut(yn(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Ns(t);let n={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:H(t.statements??[])};t.final_feedback&&(n.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(n.teardown=H(t.teardown));let r=pt.safeParse(n);if(!r.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(r.error.errors)}`);let s=r.data;return Re(e,s),s}function Ns(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let n=t.tests;if(!Array.isArray(n)||n.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:n.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:H(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=H(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=H(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=H(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=H(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=H(t.afterEach));let o=Oe.safeParse(s);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function H(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Fs)}function Fs(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 Us(t);if("WHILE"in t)return js(t);if("STEP"in t)return Bs(t);if("VERIFY"in t){let n=t.VERIFY,r={statement:typeof n=="string"?n:String(n)};return typeof t.js=="string"&&(r.code=t.js),{uid:K(),type:"ACTION",description:String(n),action_entity:{action_description:String(n),action_data:{action_name:"verify",kwargs:r}}}}if("URL"in t){let n=t.URL,r=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof n=="string"?n:String(n)};return r&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:K(),type:"ACTION",description:`Navigate to ${n}`,action_entity:{action_description:`Navigate to ${n}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let n=t.WAIT_UNTIL,r=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:K(),type:"ACTION",description:`Wait until: ${n}`,action_entity:{action_description:`Wait until: ${n}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof n=="string"?n:String(n),timeout_seconds:r}}}}}if("WAIT"in t){let n=t.WAIT,r=typeof t.seconds=="number"?t.seconds:3;return{uid:K(),type:"ACTION",description:typeof n=="string"?n:`Wait ${r}s`,action_entity:{action_description:typeof n=="string"?n:`Wait ${r}s`,action_data:{action_name:"wait",kwargs:{seconds:r}}}}}if("CODE"in t){let n=t.CODE;if(n==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:K(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)&&t.action!=="verify"){let n=t.js,r=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:K(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_action",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:n,...r}=t;return mn({...r,action:"function",functionName:n})}if("action"in t)return mn(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:K(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Sn(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 Us(e){let t=Sn(e.IF),n=e.THEN;if(!Array.isArray(n))throw new Error("IF_ELSE requires a THEN array");let r={uid:K(),type:"IF_ELSE",condition:t,then:H(n)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(r.else=H(e.ELSE)),r}function js(e){let t=Sn(e.WHILE),n=e.DO;if(!Array.isArray(n))throw new Error("WHILE_LOOP requires a DO array");let r={uid:K(),type:"WHILE_LOOP",condition:t,body:H(n)};return typeof e.timeout_ms=="number"&&(r.timeout_ms=e.timeout_ms),r}function Bs(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let n={uid:K(),type:"STEP",description:t,statements:H(e.statements)};if(typeof e.reference_id=="number"&&(n.reference_id=e.reference_id),typeof e.template_path=="string"&&(n.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let r=e.template_params,s={};for(let[o,i]of Object.entries(r))s[o]=String(i);n.template_params=s}return n}function mn(e){let t=typeof e.action=="string"?e.action:String(e.action),n=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",r=typeof e.locator=="string"?e.locator:void 0,s=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[l,p]of Object.entries(e))Hs.has(l)||(i[l]=p);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js);let a={action_description:n,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};r&&(a.locator=r),s&&(a.xpath=s);let c={uid:K(),type:"ACTION",description:n,action_entity:a};return o&&(c.use_pure_vision=!0),c}function Re(e,t){let n;try{n=Ms(e)}catch{return}let r=n.contents;if(!r||!ge(r))return;if(n.commentBefore)t.comment=n.commentBefore;else{let c=r.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let s=r,o=s.get("statements",!0);V(o)&&t.statements&&fe(o,t.statements);let i=s.get("teardown",!0);V(i)&&t.teardown&&fe(i,t.teardown)}function fe(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=e.items[n];r.commentBefore&&!(n===0&&e.commentBefore)&&(t[n].comment=r.commentBefore);let s=t[n];if(s.type==="STEP"&&ge(r)){let o=r.get("statements",!0);V(o)&&fe(o,s.statements)}else if(s.type==="IF_ELSE"&&ge(r)){let o=r.get("THEN",!0);V(o)&&fe(o,s.then);let i=r.get("ELSE",!0);V(i)&&s.else&&fe(i,s.else)}else if(s.type==="WHILE_LOOP"&&ge(r)){let o=r.get("DO",!0);V(o)&&fe(o,s.body)}}}var wn,gn,Hs,Ce=_(()=>{"use strict";dt();wn={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};gn=1024*1024;Hs=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as oa,stringify as ia}from"yaml";var _n=_(()=>{"use strict";Ce()});var ht,De,Ne=_(()=>{"use strict";ht=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},De=e=>{let t=[],n=r=>{for(let s of r){t.push(s);let o=ht(s);for(let i of o)n(i.statements)}};return n(e),t}});function kn(e){let t=0,n=0;for(let r of e)if(r.type==="DRAFT")n++;else if(r.type==="ACTION"){let s=r.action_entity?.action_data?.action_name??"";Tn.has(s)||t++}return{action:t,draft:n}}function Gs(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function xn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function ft(e,t){let n=t?.coverageThreshold??Ws,r=[],s=[],o;try{o=C(e)}catch(f){return{valid:!1,errors:[`Invalid YAML: ${f.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||r.push('Missing required field: "goal"'),o.statements?.length||r.push('Missing required field: "statements"');let i=[...De(o.statements??[]),...o.teardown?De(o.teardown):[]],{action:a,draft:c}=kn(i),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let f of i){let m=f;if(m.reference_id!==void 0){let g=m.description||f.uid;r.push(`Unresolved cloud template reference on statement "${g}" (reference_id: ${m.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(f.type==="ACTION"){let g=f,u=g.action_entity?.action_data?.action_name??"";if(u==="js_code"||u==="js_action"||u==="verify"||u==="ai_assert"){let v=g.action_entity?.action_data?.kwargs?.code;if(typeof v=="string"){let h=Gs(v);if(h){let b=g.description||u;r.push(`Invalid JS in "${b}": ${h}. ${l}`)}}}}if(f.type==="IF_ELSE"){let g=f;if(g.condition.type==="JS_CODE"){let u=xn(g.condition.expression);u&&r.push(`Invalid JS in IF condition "${g.condition.expression}": ${u}. ${l}`)}}if(f.type==="WHILE_LOOP"){let g=f;if(g.condition.type==="JS_CODE"){let u=xn(g.condition.expression);u&&r.push(`Invalid JS in WHILE condition "${g.condition.expression}": ${u}. ${l}`)}}}let p=a+c,d=p>0?Math.round(a/p*100):0;return p>0&&d/100<n&&s.push(`Low action coverage: ${a}/${p} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:r.length===0,errors:r,warnings:s,stats:{total:p,action:a,draft:c,coverage:d}}}var Ws,Tn,An=_(()=>{"use strict";Ce();Ne();Ws=.5,Tn=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Fe=_(()=>{"use strict"});var En=_(()=>{"use strict";Fe()});var Pn,Vs,$n,Mn,Te,zs,Ys,In=_(()=>{"use strict";Pn=112,Vs=1080-Pn,$n={"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"}},Mn={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"]},Te=(e,t=!1)=>{let n=["chromium"];return t&&n.push("webkit"),Mn[e].map(r=>$n[r]).filter(r=>r.defaultBrowserType&&n.includes(r.defaultBrowserType))},zs={desktop:{label:"Desktop",type:"desktop",devices:Te("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Te("mobile")}},Ys={desktop:{label:"Desktop",type:"desktop",devices:Te("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Te("mobile",!0)}}});var On=_(()=>{"use strict"});function gt(){return{version:"1.0",entries:{}}}var Ln=_(()=>{"use strict";Ne()});var z,mt,Rn=_(()=>{"use strict";z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(z||{}),mt=18e4});var Cn=_(()=>{"use strict"});var Dn=_(()=>{"use strict"});var Nn=_(()=>{"use strict"});var Fn=_(()=>{"use strict";Fe()});var ae=_(()=>{"use strict";rn();sn();on();an();_n();An();Ce();dt();En();In();On();Ln();Ne();Rn();Cn();Dn();Nn();Fn();Fe()});import{stringify as qs}from"yaml";import{createHash as go}from"crypto";import{parse as mo,stringify as Yn}from"yaml";import{readFileSync as yo,existsSync as wo}from"fs";import{resolve as wt,dirname as bo}from"path";import{parse as Hn,stringify as vo}from"yaml";import{readFileSync as Eo,writeFileSync as Po,mkdirSync as $o}from"fs";import{dirname as Mo}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function I(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Xs(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Zs(e){let t=e.xpath;return typeof t=="string"&&t.trim()?!t.startsWith("xpath=")&&!t.startsWith("/")&&!t.startsWith("//")?`xpath=//${t}`:t.startsWith("xpath=")?t:`xpath=${t}`:null}function je(e){let t=Xs(e),n=e.locator;if(typeof n=="string"&&n.trim())return n=n.trim(),n.endsWith("first()")?`${t}.${n}`:`${t}.${n}.first()`;let r=Zs(e);if(r){let s=JSON.stringify(r);return`${t}.locator(${s}).first()`}return null}function Un(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Qs.includes(t)}function to(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!eo.includes(t)}function S(e,t){R.set(e,t)}function no(e){return R.get(e)}function pe(e,t,n=[]){let r=[...n];return t.locator?r.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&r.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(t.frame_path)}`),r.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...r.map(s=>` ${s},`),"});"]}function jn(e){let t=e.functionName;if(!t)return null;let n=Array.isArray(e.args)?e.args.map(String):[];if(n.length===0)return`await ${t}()`;let r=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=n.map(i=>r.includes(i)||s.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function X(e,t,n,r="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${r}.${o}`,c=ro(i,t,a,n);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function ro(e,t,n,r){let s=" ".repeat(t);switch(e.type){case"DRAFT":return so(e,t,n,r);case"ACTION":return oo(e,t,n,r);case"STEP":return io(e,t,n,r);case"IF_ELSE":return ao(e,t,n,r);case"WHILE_LOOP":return co(e,t,n,r);default:return[`${s}// Unknown statement type: ${e.type}`]}}function so(e,t,n,r){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.run(page, ${i}, '${n}');`]}function oo(e,t,n,r){let s=" ".repeat(t),o=e.description,i=e.uid,c=r.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let y=JSON.stringify(o),x=!!e.use_pure_vision;return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.execute(page, ${y}, '${n}', ${x});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let p=l.action_data?.action_name||"",d=l.action_description||"",f=no(p);if(!f)return[`${s}// ${n}: Unknown action: ${p}`];let m={imports:r.imports},g=f(l,n,m);if(r.noAgent){if(Un(l))return[`${s}// ${n}: ${I(d)}`,`${s}// AI action: ${I(d)} (requires agent - skipped in hook)`];let y=lo(l,p,s,n);return y||[`${s}// ${n}: ${I(d)}`,...g.map(x=>`${s}${x}`)]}if(Un(l))return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,...g.map(y=>`${s}${y}`)];let u=JSON.stringify(d),v=g.map(y=>`${s} ${y}`),h=to(l),b=i?`'${i}'`:"undefined";return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...v,`${s}}, ${u}, '${n}', ${b}, ${h});`]}function io(e,t,n,r){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${I(e.description)}`);let i=X(e.statements,t,r,n);return o.push(...i),o}function ao(e,t,n,r){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${n}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${I(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${n}")) {`)}let i=X(e.then,t+1,r,`${n}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=X(e.else,t+1,r,`${n}.else`);o.push(...a)}return o.push(`${s}}`),o}function co(e,t,n,r){let s=" ".repeat(t),o=[];o.push(`${s}// ${n}: Loop`);let i=e.timeout_ms??mt,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${n.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${i};`),o.push(`${s}const ${l}_check = () => {`),o.push(`${s} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${s} throw new Error('${c}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${I(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${s}while (${l}_check() && await agent.evaluate(page, ${d}, "${n}")) {`)}let p=X(e.body,t+1,r,`${n}.body`);return o.push(...p),o.push(`${s}}`),o}function lo(e,t,n,r){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??bt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${n}// ${r}: ${I(s)}`,`${n}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goBack();`];case"go_forward":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goForward();`];case"input_text":{let a=o.text||"",c=je(e);return c?[`${n}// ${r}: ${I(s)}`,`${n}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=je(e);return c?[`${n}// ${r}: ${I(s)}`,`${n}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function po(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...Vn()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(n.push(...Bn("beforeEach",t.beforeEach,o)),n.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=yt(t?.tags);for(let l of t.parameters){let p=Kn(e,l.values);n.push(...He(p,`${c}${se(a)} [${se(l.name)}]`,o,0,i)),n.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=yt(t?.tags);n.push(...He(e,`${c}${se(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(n.push(""),n.push(...Bn("afterEach",t.afterEach,o))),zn(n,s),n.join(`
638
+ `}async function Ps(e){let{createServer:t}=await import("net");for(let n=e;n<e+20;n++)if(await new Promise(s=>{let o=t();o.once("error",()=>s(!1)),o.once("listening",()=>{o.close(()=>s(!0))}),o.listen(n,"127.0.0.1")}))return n;throw new Error(`No available port found in range ${e}-${e+19}`)}async function at(e){let{yamlFilePath:t,configPath:n,tempSuffix:r="",headed:s}=e,o=q.dirname(n),i=await Ps(16174),a;if(!re.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let y=As(re.readFileSync(t,"utf-8"));y?.use&&typeof y.use=="object"&&!Array.isArray(y.use)&&(a=y.use),y?.base_url&&!a?.baseURL&&(a={...a,baseURL:y.base_url}),y?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!y.settings.auto_dismiss_modal})}catch(y){console.error("[debugger] Could not parse YAML for `use` block:",y)}let c=q.dirname(q.resolve(t)),l=r?`-${r}`:"",p=q.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=Es(t,i,a,o);re.writeFileSync(p,d);let f=()=>{try{re.unlinkSync(p)}catch{}},m=["playwright","test",p,...s?["--headed"]:[]],g=ks("npx",m,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});g.stdout?.on("data",y=>{process.stderr.write(y)}),g.stderr?.on("data",y=>{process.stderr.write(y)});let u=()=>{g.killed||g.kill("SIGTERM")};process.on("SIGTERM",u),process.on("SIGINT",u),process.on("exit",u),g.on("close",y=>{process.removeListener("SIGTERM",u),process.removeListener("SIGINT",u),process.removeListener("exit",u),f(),y!==0&&y!==null&&console.error(`[debugger] Playwright process exited with code ${y}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let v=["127.0.0.1","::1"];async function h(y){try{let x=y.includes(":")?`[${y}]`:y,A=await fetch(`http://${x}:${i}/api/test-flow`);if(A.ok){try{await A.text()}catch{}return!0}}catch{}return!1}let b=null;for(let y=0;y<180;y++){if(g.exitCode!==null)throw f(),new Error(`Playwright process exited with code ${g.exitCode} before sandbox was ready`);for(let x of v)if(await h(x)){b=x;break}if(b){console.error(`[debugger] Playwright sandbox ready on ${b}:${i}`);break}if(y===179)throw u(),f(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(x=>setTimeout(x,1e3))}if(!b)throw u(),f(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:b,pid:g.pid??0,cleanup:async()=>{u(),f()}}}var ct=_(()=>{"use strict"});var rn=_(()=>{"use strict"});var sn=_(()=>{"use strict"});var on=_(()=>{"use strict"});import{v4 as qi}from"uuid";var an=_(()=>{"use strict"});import{z as w}from"zod";var cn,lt,ln,he,pn,dn,un,N,hn,Oe,pt,dt=_(()=>{"use strict";cn=w.enum(["JS_CODE","AI_MODE"]),lt=w.object({type:cn,expression:w.string()}),ln=w.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),he=w.object({uid:w.string(),type:ln,comment:w.string().optional()}),pn=w.object({action_data:w.object({action_name:w.string(),kwargs:w.record(w.any()).optional(),args:w.array(w.any()).optional()}),action_description:w.string().optional(),url:w.string().optional(),xpath:w.string().nullable().optional(),locator:w.string().nullable().optional(),css_selector:w.string().nullable().optional(),unique_selector:w.string().nullable().optional(),element_index:w.number().nullable().optional(),frame_path:w.array(w.any()).optional(),artifacts:w.record(w.any()).optional(),feedback:w.string().optional(),original_browser_use_action:w.any().optional()}).passthrough(),dn=he.extend({type:w.literal("DRAFT"),description:w.string()}),un=he.extend({type:w.literal("ACTION"),description:w.string(),action_entity:pn.optional(),locator:w.string().optional(),use_pure_vision:w.boolean().optional()}),N=w.lazy(()=>w.union([dn,un,he.extend({type:w.literal("STEP"),description:w.string().optional().default(""),statements:w.array(N),reference_id:w.number().optional(),template_path:w.string().optional(),template_params:w.record(w.string()).optional()}),he.extend({type:w.literal("IF_ELSE"),description:w.string().optional(),condition:lt,then:w.array(N),else:w.array(N).optional()}),he.extend({type:w.literal("WHILE_LOOP"),description:w.string().optional(),condition:lt,body:w.array(N),timeout_ms:w.number().optional()})])),hn=w.object({name:w.string(),statements:w.array(N),teardown:w.array(N).optional(),skip:w.union([w.boolean(),w.string()]).optional(),timeout:w.number().optional(),fail:w.union([w.boolean(),w.string()]).optional(),only:w.boolean().optional(),slow:w.boolean().optional()}),Oe=w.object({tests:w.array(hn).min(1),beforeAll:w.array(N).optional(),afterAll:w.array(N).optional(),beforeEach:w.array(N).optional(),afterEach:w.array(N).optional()}),pt=w.object({comment:w.string().optional(),version:w.string().optional(),goal:w.string().optional(),url:w.string().optional(),baseURL:w.string().optional(),final_feedback:w.string().optional(),completed:w.boolean().optional(),success:w.boolean().optional(),statements:w.array(N).optional(),teardown:w.array(N).optional(),last_modified_at:w.string().optional(),testGroup:Oe.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 $s,parse as yn,parseAllDocuments as na,parseDocument as Ms,Document as Is,isMap as ge,isSeq as V}from"yaml";import{v4 as K}from"uuid";function Le(e,t){let n={...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,...t?.timeout!==void 0?{timeout:t.timeout}:{},statements:(e.statements??[]).map(B)};return e.final_feedback&&(n.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(n.teardown=e.teardown.map(B)),n}function _e(e,t){if(e.testGroup)return bn(e,t);let n=Le(e,t),r=new Is(n);return e.comment&&(r.commentBefore=e.comment),fn(r,e.statements??[]),e.teardown&&fn(r,e.teardown,"teardown"),r.toString(wn)}function fn(e,t,n="statements"){let r=e.contents;if(!r||!ge(r))return;let s=r.get(n,!0);V(s)&&Se(s,t)}function Se(e,t){for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=t[n],s=e.items[n];if(n>0&&(s.spaceBefore=!0),r.comment&&(n===0?e.commentBefore=r.comment:s.commentBefore=r.comment),ge(s)){let o=s;if(r.type==="STEP"){let i=o.get("statements",!0);V(i)&&Se(i,r.statements)}else if(r.type==="IF_ELSE"){let i=o.get("THEN",!0);V(i)&&Se(i,r.then);let a=o.get("ELSE",!0);V(a)&&r.else&&Se(a,r.else)}else if(r.type==="WHILE_LOOP"){let i=o.get("DO",!0);V(i)&&Se(i,r.body)}}}}function bn(e,t){let n=e.testGroup;if(!n)throw new Error("suiteToYaml requires a TestFlow with testGroup");let r={};t?.test_case_id!==void 0&&(r.test_case_id=t.test_case_id),t?.name&&(r.name=t.name),t?.tags&&t.tags.length>0&&(r.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(r.use=t.use);let s={};return e.baseURL&&(s.base_url=e.baseURL),n.beforeAll&&n.beforeAll.length>0&&(s.beforeAll=n.beforeAll.map(B)),n.beforeEach&&n.beforeEach.length>0&&(s.beforeEach=n.beforeEach.map(B)),n.afterEach&&n.afterEach.length>0&&(s.afterEach=n.afterEach.map(B)),n.afterAll&&n.afterAll.length>0&&(s.afterAll=n.afterAll.map(B)),s.tests=n.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(B),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(B)),i}),r.suite=s,$s(r,wn)}function B(e){switch(e.type){case"DRAFT":return Os(e);case"ACTION":return Ls(e);case"STEP":return Rs(e);case"IF_ELSE":return Cs(e);case"WHILE_LOOP":return Ds(e)}}function Os(e){return{intent:e.description}}function Ls(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,n=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=n?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=n?.code;return typeof c=="string"?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=n?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return n?.new_tab===!0&&(c.new_tab=!0),typeof n?.timeout_seconds=="number"&&(c.timeout_seconds=n.timeout_seconds),c}}if(t==="js_action"){let a=n?.code;if(typeof a=="string"&&e.description)return{intent:e.description,js:a}}if(t==="ai_wait_until"){let a=n?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof n?.timeout_seconds=="number"&&n.timeout_seconds!==60&&(c.timeout_seconds=n.timeout_seconds),c}}if(t==="wait"){let a=n?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=n?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:a}}if(!e.action_entity)return{intent:e.description};let r=e.action_entity.action_data??e.action_entity.action;if(!r)return{intent:e.description};let s={intent:e.description,action:r.action_name},o=e.locator??e.action_entity.locator;o&&(s.locator=o);let i=e.action_entity.xpath;if(i&&(s.xpath=i),e.use_pure_vision&&(s.use_pure_vision=!0),r.kwargs&&Object.keys(r.kwargs).length>0)for(let[a,c]of Object.entries(r.kwargs))s[a]=c;return r.args&&r.args.length>0&&(s.args=r.args),s}function Rs(e){if(e.template_path){let n={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(n.params=e.template_params),n}let t={STEP:e.description,statements:e.statements.map(B)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Cs(e){let t={IF:vn(e.condition),THEN:e.then.map(B)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(B)),t}function Ds(e){let t={WHILE:vn(e.condition),DO:e.body.map(B)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function vn(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function xe(e){try{let t=yn(e);if(!t||typeof t!="object")return{};let n={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(n.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(n.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(n.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(n.timeout=t.timeout),n}catch{return{}}}function ut(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(ut);let t=e,n=Object.keys(t);if(n.length===1){let s=n[0];if(s.startsWith("{ ")&&s.endsWith(" }")&&t[s]===null)return`{{${s.slice(2,-2)}}}`}let r={};for(let[s,o]of Object.entries(t))r[s]=ut(o);return r}function C(e){if(e.length>gn)throw new Error(`YAML input too large (${e.length} bytes, max ${gn})`);let t=ut(yn(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Ns(t);let n={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:H(t.statements??[])};t.final_feedback&&(n.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(n.teardown=H(t.teardown));let r=pt.safeParse(n);if(!r.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(r.error.errors)}`);let s=r.data;return Re(e,s),s}function Ns(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let n=t.tests;if(!Array.isArray(n)||n.length===0)throw new Error('Suite must have a non-empty "tests" array');let s={tests:n.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:H(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=H(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(s.beforeAll=H(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(s.afterAll=H(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(s.beforeEach=H(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(s.afterEach=H(t.afterEach));let o=Oe.safeParse(s);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function H(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Fs)}function Fs(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 Us(t);if("WHILE"in t)return js(t);if("STEP"in t)return Bs(t);if("VERIFY"in t){let n=t.VERIFY,r={statement:typeof n=="string"?n:String(n)};return typeof t.js=="string"&&(r.code=t.js),{uid:K(),type:"ACTION",description:String(n),action_entity:{action_description:String(n),action_data:{action_name:"verify",kwargs:r}}}}if("URL"in t){let n=t.URL,r=t.new_tab===!0?!0:void 0,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof n=="string"?n:String(n)};return r&&(o.new_tab=!0),s!==void 0&&(o.timeout_seconds=s),{uid:K(),type:"ACTION",description:`Navigate to ${n}`,action_entity:{action_description:`Navigate to ${n}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let n=t.WAIT_UNTIL,r=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:K(),type:"ACTION",description:`Wait until: ${n}`,action_entity:{action_description:`Wait until: ${n}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof n=="string"?n:String(n),timeout_seconds:r}}}}}if("WAIT"in t){let n=t.WAIT,r=typeof t.seconds=="number"?t.seconds:3;return{uid:K(),type:"ACTION",description:typeof n=="string"?n:`Wait ${r}s`,action_entity:{action_description:typeof n=="string"?n:`Wait ${r}s`,action_data:{action_name:"wait",kwargs:{seconds:r}}}}}if("CODE"in t){let n=t.CODE;if(n==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:K(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)&&t.action!=="verify"){let n=t.js,r=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:K(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_action",kwargs:{code:typeof n=="string"?n:String(n)}}}}}if("call"in t&&typeof t.call=="string"){let{call:n,...r}=t;return mn({...r,action:"function",functionName:n})}if("action"in t)return mn(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:K(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Sn(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 Us(e){let t=Sn(e.IF),n=e.THEN;if(!Array.isArray(n))throw new Error("IF_ELSE requires a THEN array");let r={uid:K(),type:"IF_ELSE",condition:t,then:H(n)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(r.else=H(e.ELSE)),r}function js(e){let t=Sn(e.WHILE),n=e.DO;if(!Array.isArray(n))throw new Error("WHILE_LOOP requires a DO array");let r={uid:K(),type:"WHILE_LOOP",condition:t,body:H(n)};return typeof e.timeout_ms=="number"&&(r.timeout_ms=e.timeout_ms),r}function Bs(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let n={uid:K(),type:"STEP",description:t,statements:H(e.statements)};if(typeof e.reference_id=="number"&&(n.reference_id=e.reference_id),typeof e.template_path=="string"&&(n.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let r=e.template_params,s={};for(let[o,i]of Object.entries(r))s[o]=String(i);n.template_params=s}return n}function mn(e){let t=typeof e.action=="string"?e.action:String(e.action),n=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",r=typeof e.locator=="string"?e.locator:void 0,s=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[l,p]of Object.entries(e))Hs.has(l)||(i[l]=p);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js);let a={action_description:n,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};r&&(a.locator=r),s&&(a.xpath=s);let c={uid:K(),type:"ACTION",description:n,action_entity:a};return o&&(c.use_pure_vision=!0),c}function Re(e,t){let n;try{n=Ms(e)}catch{return}let r=n.contents;if(!r||!ge(r))return;if(n.commentBefore)t.comment=n.commentBefore;else{let c=r.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let s=r,o=s.get("statements",!0);V(o)&&t.statements&&fe(o,t.statements);let i=s.get("teardown",!0);V(i)&&t.teardown&&fe(i,t.teardown)}function fe(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let n=0;n<Math.min(e.items.length,t.length);n++){let r=e.items[n];r.commentBefore&&!(n===0&&e.commentBefore)&&(t[n].comment=r.commentBefore);let s=t[n];if(s.type==="STEP"&&ge(r)){let o=r.get("statements",!0);V(o)&&fe(o,s.statements)}else if(s.type==="IF_ELSE"&&ge(r)){let o=r.get("THEN",!0);V(o)&&fe(o,s.then);let i=r.get("ELSE",!0);V(i)&&s.else&&fe(i,s.else)}else if(s.type==="WHILE_LOOP"&&ge(r)){let o=r.get("DO",!0);V(o)&&fe(o,s.body)}}}var wn,gn,Hs,Ce=_(()=>{"use strict";dt();wn={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};gn=1024*1024;Hs=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as aa,stringify as ca}from"yaml";var _n=_(()=>{"use strict";Ce()});var ht,De,Ne=_(()=>{"use strict";ht=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},De=e=>{let t=[],n=r=>{for(let s of r){t.push(s);let o=ht(s);for(let i of o)n(i.statements)}};return n(e),t}});function kn(e){let t=0,n=0;for(let r of e)if(r.type==="DRAFT")n++;else if(r.type==="ACTION"){let s=r.action_entity?.action_data?.action_name??"";Tn.has(s)||t++}return{action:t,draft:n}}function Gs(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function xn(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function ft(e,t){let n=t?.coverageThreshold??Ws,r=[],s=[],o;try{o=C(e)}catch(f){return{valid:!1,errors:[`Invalid YAML: ${f.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||r.push('Missing required field: "goal"'),o.statements?.length||r.push('Missing required field: "statements"');let i=[...De(o.statements??[]),...o.teardown?De(o.teardown):[]],{action:a,draft:c}=kn(i),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let f of i){let m=f;if(m.reference_id!==void 0){let g=m.description||f.uid;r.push(`Unresolved cloud template reference on statement "${g}" (reference_id: ${m.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(f.type==="ACTION"){let g=f,u=g.action_entity?.action_data?.action_name??"";if(u==="js_code"||u==="js_action"||u==="verify"||u==="ai_assert"){let v=g.action_entity?.action_data?.kwargs?.code;if(typeof v=="string"){let h=Gs(v);if(h){let b=g.description||u;r.push(`Invalid JS in "${b}": ${h}. ${l}`)}}}}if(f.type==="IF_ELSE"){let g=f;if(g.condition.type==="JS_CODE"){let u=xn(g.condition.expression);u&&r.push(`Invalid JS in IF condition "${g.condition.expression}": ${u}. ${l}`)}}if(f.type==="WHILE_LOOP"){let g=f;if(g.condition.type==="JS_CODE"){let u=xn(g.condition.expression);u&&r.push(`Invalid JS in WHILE condition "${g.condition.expression}": ${u}. ${l}`)}}}let p=a+c,d=p>0?Math.round(a/p*100):0;return p>0&&d/100<n&&s.push(`Low action coverage: ${a}/${p} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:r.length===0,errors:r,warnings:s,stats:{total:p,action:a,draft:c,coverage:d}}}var Ws,Tn,An=_(()=>{"use strict";Ce();Ne();Ws=.5,Tn=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var Fe=_(()=>{"use strict"});var En=_(()=>{"use strict";Fe()});var Pn,Vs,$n,Mn,Te,zs,Ys,In=_(()=>{"use strict";Pn=112,Vs=1080-Pn,$n={"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"}},Mn={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"]},Te=(e,t=!1)=>{let n=["chromium"];return t&&n.push("webkit"),Mn[e].map(r=>$n[r]).filter(r=>r.defaultBrowserType&&n.includes(r.defaultBrowserType))},zs={desktop:{label:"Desktop",type:"desktop",devices:Te("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Te("mobile")}},Ys={desktop:{label:"Desktop",type:"desktop",devices:Te("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Te("mobile",!0)}}});var On=_(()=>{"use strict"});function gt(){return{version:"1.0",entries:{}}}var Ln=_(()=>{"use strict";Ne()});var z,mt,Rn=_(()=>{"use strict";z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(z||{}),mt=18e4});var Cn=_(()=>{"use strict"});var Dn=_(()=>{"use strict"});var Nn=_(()=>{"use strict"});var Fn=_(()=>{"use strict";Fe()});var ae=_(()=>{"use strict";rn();sn();on();an();_n();An();Ce();dt();En();In();On();Ln();Ne();Rn();Cn();Dn();Nn();Fn();Fe()});import{stringify as qs}from"yaml";import{createHash as go}from"crypto";import{parse as mo,stringify as Yn}from"yaml";import{readFileSync as yo,existsSync as wo}from"fs";import{resolve as wt,dirname as bo}from"path";import{parse as Hn,stringify as vo}from"yaml";import{readFileSync as Eo,writeFileSync as Po,mkdirSync as $o}from"fs";import{dirname as Mo}from"path";function se(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function I(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Xs(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Zs(e){let t=e.xpath;return typeof t=="string"&&t.trim()?!t.startsWith("xpath=")&&!t.startsWith("/")&&!t.startsWith("//")?`xpath=//${t}`:t.startsWith("xpath=")?t:`xpath=${t}`:null}function je(e){let t=Xs(e),n=e.locator;if(typeof n=="string"&&n.trim())return n=n.trim(),n.endsWith("first()")?`${t}.${n}`:`${t}.${n}.first()`;let r=Zs(e);if(r){let s=JSON.stringify(r);return`${t}.locator(${s}).first()`}return null}function Un(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Qs.includes(t)}function to(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!eo.includes(t)}function S(e,t){R.set(e,t)}function no(e){return R.get(e)}function pe(e,t,n=[]){let r=[...n];return t.locator?r.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&r.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(t.frame_path)}`),r.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...r.map(s=>` ${s},`),"});"]}function jn(e){let t=e.functionName;if(!t)return null;let n=Array.isArray(e.args)?e.args.map(String):[];if(n.length===0)return`await ${t}()`;let r=["page","testContext","request","agent"],s=["undefined","null","true","false"],o=n.map(i=>r.includes(i)||s.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function X(e,t,n,r="main"){let s=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${r}.${o}`,c=ro(i,t,a,n);c.length>0&&(s.push(...c),o<e.length-1&&s.push(""))}return s}function ro(e,t,n,r){let s=" ".repeat(t);switch(e.type){case"DRAFT":return so(e,t,n,r);case"ACTION":return oo(e,t,n,r);case"STEP":return io(e,t,n,r);case"IF_ELSE":return ao(e,t,n,r);case"WHILE_LOOP":return co(e,t,n,r);default:return[`${s}// Unknown statement type: ${e.type}`]}}function so(e,t,n,r){let s=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.run(page, ${i}, '${n}');`]}function oo(e,t,n,r){let s=" ".repeat(t),o=e.description,i=e.uid,c=r.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${s}// ${n}: Skipping - no description`];if(r.noAgent)return[`${s}// ${n}: ${I(o)}`,`${s}// DRAFT: ${I(o)} (requires agent - skipped in hook)`];let y=JSON.stringify(o),x=!!e.use_pure_vision;return[`${s}// ${n}: ${I(o)}`,`${s}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.execute(page, ${y}, '${n}', ${x});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let p=l.action_data?.action_name||"",d=l.action_description||"",f=no(p);if(!f)return[`${s}// ${n}: Unknown action: ${p}`];let m={imports:r.imports},g=f(l,n,m);if(r.noAgent){if(Un(l))return[`${s}// ${n}: ${I(d)}`,`${s}// AI action: ${I(d)} (requires agent - skipped in hook)`];let y=lo(l,p,s,n);return y||[`${s}// ${n}: ${I(d)}`,...g.map(x=>`${s}${x}`)]}if(Un(l))return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,...g.map(y=>`${s}${y}`)];let u=JSON.stringify(d),v=g.map(y=>`${s} ${y}`),h=to(l),b=i?`'${i}'`:"undefined";return[`${s}// ${n}: ${I(d)}`,`${s}page = agent.agentServices.validatePage(page);`,`${s}await agent.step(page, async () => {`,...v,`${s}}, ${u}, '${n}', ${b}, ${h});`]}function io(e,t,n,r){let s=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${s}// Step: ${I(e.description)}`);let i=X(e.statements,t,r,n);return o.push(...i),o}function ao(e,t,n,r){let s=" ".repeat(t),o=[];if(o.push(`${s}// ${n}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${s}if (${e.condition.expression}) {`);else{o.push(`${s}// AI Condition: ${I(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${s}if (await agent.evaluate(page, ${a}, "${n}")) {`)}let i=X(e.then,t+1,r,`${n}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${s}} else {`);let a=X(e.else,t+1,r,`${n}.else`);o.push(...a)}return o.push(`${s}}`),o}function co(e,t,n,r){let s=" ".repeat(t),o=[];o.push(`${s}// ${n}: Loop`);let i=e.timeout_ms??mt,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${n.replace(/\./g,"_")}`;if(o.push(`${s}const ${l}_start = Date.now();`),o.push(`${s}const ${l}_timeout = ${i};`),o.push(`${s}const ${l}_check = () => {`),o.push(`${s} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${s} throw new Error('${c}');`),o.push(`${s} }`),o.push(`${s} return true;`),o.push(`${s}};`),e.condition.type==="JS_CODE")o.push(`${s}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${s}// AI Loop Condition: ${I(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${s}while (${l}_check() && await agent.evaluate(page, ${d}, "${n}")) {`)}let p=X(e.body,t+1,r,`${n}.body`);return o.push(...p),o.push(`${s}}`),o}function lo(e,t,n,r){let s=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??bt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${n}// ${r}: ${I(s)}`,`${n}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goBack();`];case"go_forward":return[`${n}// ${r}: ${I(s)}`,`${n}await page.goForward();`];case"input_text":{let a=o.text||"",c=je(e);return c?[`${n}// ${r}: ${I(s)}`,`${n}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=je(e);return c?[`${n}// ${r}: ${I(s)}`,`${n}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function po(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...Vn()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(n.push(...Bn("beforeEach",t.beforeEach,o)),n.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=yt(t?.tags);for(let l of t.parameters){let p=Kn(e,l.values);n.push(...He(p,`${c}${se(a)} [${se(l.name)}]`,o,0,i)),n.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=yt(t?.tags);n.push(...He(e,`${c}${se(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(n.push(""),n.push(...Bn("afterEach",t.afterEach,o))),zn(n,s),n.join(`
639
639
  `)}function uo(e,t){let n=[],r=t?.version||"unknown";n.push(`// @generated by shiplightai v${r}`),n.push(...Vn()),n.push(""),t?.use&&Object.keys(t.use).length>0&&(n.push(`test.use(${JSON.stringify(t.use,null,2)});`),n.push(""));let s=new Set,o={imports:s,actionEntityStore:t?.actionEntityStore},i=t?.testName||"Test Suite",a=yt(t?.tags);n.push(`test.describe.serial('${a}${se(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(n.push(...Ue("beforeAll",e.beforeAll,o,1)),n.push("")),e.beforeEach&&e.beforeEach.length>0&&(n.push(...Ue("beforeEach",e.beforeEach,o,1)),n.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],d=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let f of p.parameters){let m=Kn(p.testFlow,f.values);n.push(...He(m,`${se(p.name)} [${se(f.name)}]`,o,1,d)),n.push("")}else n.push(...He(p.testFlow,se(p.name),o,1,d)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&n.push("")}return e.afterEach&&e.afterEach.length>0&&(n.push(...Ue("afterEach",e.afterEach,o,1)),n.push("")),e.afterAll&&e.afterAll.length>0&&n.push(...Ue("afterAll",e.afterAll,o,1)),n.push("});"),zn(n,s),n.join(`
640
640
  `)}function yt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Be(e){let t=new Set;function n(r){for(let s of r)switch(s.type){case z.ACTION:{let i=s.action_entity?.action_data?.kwargs;if(i?.args&&Array.isArray(i.args))for(let a of i.args)typeof a=="string"&&ho.includes(a)&&t.add(a);break}case z.STEP:n(s.statements);break;case z.IF_ELSE:{let o=s;n(o.then),o.else&&n(o.else);break}case z.WHILE_LOOP:n(s.body);break}}return n(e),t}function fo(e){let t=Be(e.statements??[]);if(e.teardown)for(let n of Be(e.teardown))t.add(n);return t}function vt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function He(e,t,n,r=0,s){let o=" ".repeat(r),i=[],a=fo(e),c=vt(a),l=s?.only?"test.only":"test";i.push(`${o}${l}('${t}', async (${c}) => {`),s?.skip===!0?i.push(`${o} test.skip();`):typeof s?.skip=="string"&&i.push(`${o} test.skip(true, '${se(s.skip)}');`),s?.fail===!0?i.push(`${o} test.fail();`):typeof s?.fail=="string"&&i.push(`${o} test.fail(true, '${se(s.fail)}');`),s?.slow&&i.push(`${o} test.slow();`),s?.timeout&&i.push(`${o} test.setTimeout(${s.timeout});`);let p=e.teardown&&e.teardown.length>0,d=r+1;if(p){if(i.push(`${o} try {`),e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let m=X(e.statements,d+1,n);i.push(...m)}i.push(`${o} } finally {`),i.push(`${o} // Teardown`);let f=X(e.teardown,d+1,n,"teardown");i.push(...f),i.push(`${o} }`)}else if(e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let f=X(e.statements,d,n);i.push(...f)}return i.push(`${o}});`),i}function Bn(e,t,n){let r=[],s=Gn(t),o=Be(s),i=vt(o);return r.push(`test.${e}(async (${i}) => {`),r.push(...X(s,1,n,e)),r.push("});"),r}function Ue(e,t,n,r){let s=" ".repeat(r),o=[],i=Gn(t);if(e==="beforeAll"||e==="afterAll"){let c={...n,noAgent:!0};o.push(`${s}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${s} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...X(i,r+1,c,e)),o.push(`${s} await page.close();`),o.push(`${s}});`)}else{let c=Be(i),l=vt(c);o.push(`${s}test.${e}(async (${l}) => {`),o.push(...X(i,r+1,n,e)),o.push(`${s}});`)}return o}function Gn(e){let n=qs({goal:"_hook",statements:e});return C(n).statements??[]}function Kn(e,t){let n=_e(e);for(let[r,s]of Object.entries(t))n=n.split(`<<${r}>>`).join(String(s));return C(n)}function Vn(){return["import { test, expect } from 'shiplightai/fixture';"]}function zn(e,t){if(t.size>0){let n=0;for(let s=0;s<e.length;s++)e[s].startsWith("import ")&&(n=s+1);let r=Array.from(t);e.splice(n,0,...r)}}function Jn(e,t,n){let r={expandingPaths:new Set([wt(t)]),depth:0,referencedPaths:new Set,basePath:n},s={...e};Array.isArray(s.statements)&&(s.statements=ce(s.statements,t,r)),Array.isArray(s.teardown)&&(s.teardown=ce(s.teardown,t,r));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(s[o])&&(s[o]=ce(s[o],t,r));return{doc:s,referencedTemplatePaths:Array.from(r.referencedPaths)}}function ce(e,t,n){let r=[];for(let s of e)if(So(s)){let o=_o(s,t,n);r.push(o)}else r.push(xo(s,t,n));return r}function So(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function _o(e,t,n){if(n.depth>=Wn)throw new Error(`Template expansion exceeded maximum depth of ${Wn}. Check for deeply nested or circular template references.`);let r=wt(bo(t),e.template),s=!wo(r)&&n.basePath?wt(n.basePath,e.template):r;if(n.expandingPaths.has(s))throw new Error(`Circular template reference detected: ${s} is already being expanded. Stack: ${Array.from(n.expandingPaths).join(" \u2192 ")} \u2192 ${s}`);n.referencedPaths.add(s);let o;try{o=yo(s,"utf-8")}catch(g){throw new Error(`Failed to read template file: ${s} (referenced from ${t}): ${g.message}`)}let i=Hn(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${s} \u2014 expected a YAML object`);let a=i.params||[],c=e.params||{};for(let g of a)if(!(g in c))throw new Error(`Template ${e.template} requires param "${g}" but it was not provided. Required params: [${a.join(", ")}]`);let l=i.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let u=vo(l);for(let[v,h]of Object.entries(c))u=u.split(`<<${v}>>`).join(String(h));l=Hn(u)}let p={expandingPaths:new Set([...n.expandingPaths,s]),depth:n.depth+1,referencedPaths:n.referencedPaths},d=ce(l,s,p),m={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(c).length>0&&(m.template_params=c),m}function xo(e,t,n){if(typeof e!="object"||e===null)return e;let r={...e};return Array.isArray(r.statements)&&(r.statements=ce(r.statements,t,n)),Array.isArray(r.THEN)&&(r.THEN=ce(r.THEN,t,n)),Array.isArray(r.ELSE)&&(r.ELSE=ce(r.ELSE,t,n)),Array.isArray(r.DO)&&(r.DO=ce(r.DO,t,n)),r}function _t(e,t,n){let r=mo(e),s=r?.name,o=r?.tags,i=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new St('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return ko(r,s,o,i,t,n)}return To(r,s,o,i,t,n)}function To(e,t,n,r,s,o){let i=e?.beforeEach,a=e?.afterEach,c=qn(e?.parameters),l=e?.timeout,p=e?.skip,d=e?.fail,f=e?.only,m=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new St(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let g=[];if(s&&e&&typeof e=="object"){let h=Jn(e,s,o);e=h.doc,g=h.referencedTemplatePaths}let u=Yn(e),v=C(u);return s&&(me(v.statements??[],s,"main"),v.teardown&&me(v.teardown,s,"teardown")),{testFlow:v,name:t,tags:n,use:r,beforeEach:i,afterEach:a,parameters:c,timeout:l,skip:p,fail:d,only:f,slow:m,referencedTemplatePaths:g}}function ko(e,t,n,r,s,o){let i=e.suite;if(!Array.isArray(i.tests)||i.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=i.beforeAll,c=i.afterAll,l=i.beforeEach,p=i.afterEach,d=[],f=i.tests.map(u=>{if(!u.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(u.statements)||u.statements.length===0)throw new Error(`Suite test "${u.name}" must have a non-empty "statements" array.`);let v={goal:u.name,statements:u.statements};u.teardown&&(v.teardown=u.teardown);let h=[],b=v;if(s&&typeof v=="object"){let M=Jn(v,s,o);b=M.doc,h=M.referencedTemplatePaths,d.push(...h)}let y=Yn(b),x=C(y),A=qn(u.parameters);return{testFlow:x,name:u.name,tags:Array.isArray(u.tags)?u.tags:void 0,parameters:A,timeout:u.timeout,skip:u.skip,fail:u.fail,only:u.only,slow:u.slow}}),m=i.base_url,g=m?{...r,baseURL:m}:r;return{suite:{beforeAll:a,afterAll:c,beforeEach:l,afterEach:p,tests:f},name:t,tags:n,use:g,referencedTemplatePaths:d}}function qn(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,n)=>{if(!t.name)throw new Error(`Parameter set at index ${n} 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 me(e,t,n){for(let r=0;r<e.length;r++){let s=e[r],o=`${n}.${r}`,i=s.description||"";if(s.uid=Ao(t,o,i),s.type===z.STEP)me(s.statements,t,o);else if(s.type===z.IF_ELSE){let a=s;me(a.then,t,`${o}.then`),a.else&&me(a.else,t,`${o}.else`)}else s.type===z.WHILE_LOOP&&me(s.body,t,`${o}.body`)}}function Ao(e,t,n){let r=go("sha256").update(`${e}:${t}:${n}`).digest("hex");return`${r.slice(0,8)}-${r.slice(8,12)}-${r.slice(12,16)}-${r.slice(16,20)}-${r.slice(20,32)}`}function Xn(e,t){let n;try{n=Eo(e,"utf-8")}catch(r){return{valid:!1,errors:[`Failed to read file: ${r.message}`],warnings:[]}}return Io(n,e,t)}function Io(e,t,n){let r=/\btemplate:\s/.test(e),s=/^suite:/m.test(e),o=r||s?null:ft(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,c=[];try{let l=n?.parsed??_t(e,t);c=l.referencedTemplatePaths;let p={version:n?.version,actionEntityStore:n?.actionEntityStore},d=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?i=uo(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):i=po(l.testFlow,{...p,testName:l.name,tags:l.tags,use:d,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let f=i.split(`
641
641
  `).filter(m=>!m.startsWith("import ")).join(`
@@ -683,7 +683,7 @@ Shutting down...`);let h=setTimeout(()=>{console.error("Cleanup timed out, force
683
683
  statements:
684
684
  - URL: ${e}
685
685
  - "TODO: Add your first step"
686
- `});import np from"dotenv";function Yo(){return globalThis[zo]}function ze(){let e=Yo();return e===void 0?process.env:e}var zo,Et=_(()=>{"use strict";zo="__shiplightDotenvCache__"});function Ye(e,t){let n=t?.trim();return n?n.endsWith("/")?n.slice(0,-1):n:e.startsWith(Jo)?qo:Xo}var Jo,qo,Xo,Pt=_(()=>{"use strict";Jo="shp_",qo="https://nova-api.shiplight.ai",Xo="https://api.shiplight.ai"});var $t={};ee($t,{lookupActionStores:()=>ei,updateActionStores:()=>ti});function mr(){let e=ze(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Ye(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function ei(e){let t=mr();if(!t||e.length===0)return new Map;try{let n=new AbortController,r=setTimeout(()=>n.abort(),Zo),s=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:n.signal});if(clearTimeout(r),!s.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${s.status}`),new Map;let o=await s.json(),i=new Map;for(let[a,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",n.message),new Map}}async function ti(e){let t=mr();if(!t||e.size===0)return 0;try{let n=new AbortController,r=setTimeout(()=>n.abort(),Qo),s={};for(let[a,c]of e)s[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:s}),signal:n.signal});return clearTimeout(r),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",n.message),0}}var Zo,Qo,Mt=_(()=>{"use strict";Pt();Et();Zo=2e3,Qo=5e3});import*as U from"fs";import*as Ae from"path";import{globSync as ni}from"glob";function yr(e){let t=ze().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ot:new It(e)}function Ee(e){return e.replace(/\//g,"__")+".json"}function si(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ri,It,Ot,wr=_(()=>{"use strict";ae();Et();ri=".shiplight/action-cache";It=class{constructor(t){this.cwd=t;this.cacheDir=Ae.join(t,ri)}isCloud=!1;cacheDir;async lookup(t){let n=new Map;if(t.length===0||!U.existsSync(this.cacheDir))return n;for(let r of t){let s=Ae.join(this.cacheDir,Ee(r));try{if(U.existsSync(s)){let o=U.readFileSync(s,"utf-8");n.set(r,JSON.parse(o))}}catch{}}return n}async update(t){if(t.size===0)return 0;U.mkdirSync(this.cacheDir,{recursive:!0});let n=0;for(let[r,s]of t)try{let o=Ae.join(this.cacheDir,Ee(r)),i=gt();if(U.existsSync(o))try{i=JSON.parse(U.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};U.writeFileSync(o,JSON.stringify(a,null,2)),n++}catch{}return n}loadAll(){if(!U.existsSync(this.cacheDir))return;let t=ni("*.json",{cwd:this.cacheDir});if(t.length===0)return;let n=new Map,r=0;for(let s of t)try{let o=U.readFileSync(Ae.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=si(s);n.set(a,i),r+=Object.keys(i.entries??{}).length}catch{}if(n.size!==0)return console.log(`[shiplight] Cache: loaded ${r} cached action entit${r===1?"y":"ies"} for ${n.size} test file${n.size!==1?"s":""}`),n}},Ot=class{isCloud=!0;async lookup(t){let{lookupActionStores:n}=await Promise.resolve().then(()=>(Mt(),$t));return n(t)}async update(t){let{updateActionStores:n}=await Promise.resolve().then(()=>(Mt(),$t));return n(t)}loadAll(){}}});var _r={};ee(_r,{buildTestPathsFromGitInfo:()=>Sr,cloudKeyToCwdRelPath:()=>Ct,runTests:()=>ii});import{spawn as oi,execFileSync as br}from"child_process";import*as j from"fs";import*as D from"path";import{globSync as Lt}from"glob";async function ii(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(d=>j.existsSync(D.join(t,d)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
686
+ `});import sp from"dotenv";function Yo(){return globalThis[zo]}function ze(){let e=Yo();return e===void 0?process.env:e}var zo,Et=_(()=>{"use strict";zo="__shiplightDotenvCache__"});function Ye(e,t){let n=t?.trim();return n?n.endsWith("/")?n.slice(0,-1):n:e.startsWith(Jo)?qo:Xo}var Jo,qo,Xo,Pt=_(()=>{"use strict";Jo="shp_",qo="https://nova-api.shiplight.ai",Xo="https://api.shiplight.ai"});var $t={};ee($t,{lookupActionStores:()=>ei,updateActionStores:()=>ti});function mr(){let e=ze(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:Ye(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function ei(e){let t=mr();if(!t||e.length===0)return new Map;try{let n=new AbortController,r=setTimeout(()=>n.abort(),Zo),s=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:n.signal});if(clearTimeout(r),!s.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${s.status}`),new Map;let o=await s.json(),i=new Map;for(let[a,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",n.message),new Map}}async function ti(e){let t=mr();if(!t||e.size===0)return 0;try{let n=new AbortController,r=setTimeout(()=>n.abort(),Qo),s={};for(let[a,c]of e)s[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:s}),signal:n.signal});return clearTimeout(r),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(n){return n instanceof Error&&n.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",n.message),0}}var Zo,Qo,Mt=_(()=>{"use strict";Pt();Et();Zo=2e3,Qo=5e3});import*as U from"fs";import*as Ae from"path";import{globSync as ni}from"glob";function yr(e){let t=ze().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ot:new It(e)}function Ee(e){return e.replace(/\//g,"__")+".json"}function si(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ri,It,Ot,wr=_(()=>{"use strict";ae();Et();ri=".shiplight/action-cache";It=class{constructor(t){this.cwd=t;this.cacheDir=Ae.join(t,ri)}isCloud=!1;cacheDir;async lookup(t){let n=new Map;if(t.length===0||!U.existsSync(this.cacheDir))return n;for(let r of t){let s=Ae.join(this.cacheDir,Ee(r));try{if(U.existsSync(s)){let o=U.readFileSync(s,"utf-8");n.set(r,JSON.parse(o))}}catch{}}return n}async update(t){if(t.size===0)return 0;U.mkdirSync(this.cacheDir,{recursive:!0});let n=0;for(let[r,s]of t)try{let o=Ae.join(this.cacheDir,Ee(r)),i=gt();if(U.existsSync(o))try{i=JSON.parse(U.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...s.entries}};U.writeFileSync(o,JSON.stringify(a,null,2)),n++}catch{}return n}loadAll(){if(!U.existsSync(this.cacheDir))return;let t=ni("*.json",{cwd:this.cacheDir});if(t.length===0)return;let n=new Map,r=0;for(let s of t)try{let o=U.readFileSync(Ae.join(this.cacheDir,s),"utf-8"),i=JSON.parse(o),a=si(s);n.set(a,i),r+=Object.keys(i.entries??{}).length}catch{}if(n.size!==0)return console.log(`[shiplight] Cache: loaded ${r} cached action entit${r===1?"y":"ies"} for ${n.size} test file${n.size!==1?"s":""}`),n}},Ot=class{isCloud=!0;async lookup(t){let{lookupActionStores:n}=await Promise.resolve().then(()=>(Mt(),$t));return n(t)}async update(t){let{updateActionStores:n}=await Promise.resolve().then(()=>(Mt(),$t));return n(t)}loadAll(){}}});var _r={};ee(_r,{buildTestPathsFromGitInfo:()=>Sr,cloudKeyToCwdRelPath:()=>Ct,runTests:()=>ii});import{spawn as oi,execFileSync as br}from"child_process";import*as j from"fs";import*as D from"path";import{globSync as Lt}from"glob";async function ii(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(d=>j.existsSync(D.join(t,d)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
687
687
  `));let s=e.includes("--magic"),i=e.filter(d=>d!=="--magic").map(d=>d.endsWith(".test.yaml")?d.replace(/\.test\.yaml$/,".yaml.spec.ts"):d),a=yr(t);await ci(t,a),ve&&process.stdout.write(`shiplightai v${ve}
688
688
  `);let c={...process.env};s&&(c.SHIPLIGHT_MAGIC="1");let l=oi("npx",["playwright","test",...i],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(d=>{l.on("close",f=>d(f??1))});await li(t,a),process.exit(p)}function Rt(){try{return br("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function ai(){try{let e=br("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function vr(e,t,n){if(!n)return{testPaths:[...e],branchPrefix:""};let r=ai(),s=Rt();return Sr(e,t,r,s)}function Sr(e,t,n,r){return{testPaths:e.map(o=>{let i=r?D.relative(r,D.resolve(t,o)):o;return`${n}${i}`}),branchPrefix:n}}function Ct(e,t,n,r){let s=t&&e.startsWith(t)?e.slice(t.length):e;return n?D.relative(r,D.resolve(n,s)):s}async function ci(e,t){try{let n=Lt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(n.length===0)return;let{testPaths:r,branchPrefix:s}=vr(n,e,t.isCloud),o=await t.lookup(r);if(o.size===0)return;let i=D.join(e,".shiplight","action-cache");j.mkdirSync(i,{recursive:!0});let a=Rt(),c=0;for(let[l,p]of o){let d=t.isCloud?Ct(l,s,a,e):l,f=D.join(i,Ee(d));j.writeFileSync(f,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(n){console.warn("[shiplight] Cache download failed:",n.message)}}async function li(e,t){try{let n=D.join(e,"test-results");if(!j.existsSync(n))return;let r=Lt("**/new-action-entities.json",{cwd:n});if(r.length===0)return;let s={};for(let d of r)try{let f=j.readFileSync(D.join(n,d),"utf-8"),m=JSON.parse(f);m?.entries&&Object.assign(s,m.entries)}catch{}if(Object.keys(s).length===0)return;let o=Lt("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=vr(o,e,t.isCloud),c=new Map;for(let d=0;d<o.length;d++){let f=o[d],m=i[d],g=D.join(e,f.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!j.existsSync(g))continue;let u=j.readFileSync(g,"utf-8"),v={};for(let[h,b]of Object.entries(s))u.includes(h)&&(v[h]=b);if(Object.keys(v).length>0){let h=t.isCloud?Ct(m,a,Rt(),e):m,b=D.join(e,".shiplight","action-cache",Ee(h)),y={};if(j.existsSync(b))try{y=JSON.parse(j.readFileSync(b,"utf-8")).entries}catch{}c.set(m,{version:"1.0",entries:{...y,...v}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((d,f)=>d+Object.keys(f.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(n){console.warn("[shiplight] Cache upload failed:",n.message)}}var xr=_(()=>{"use strict";wr();Pe()});function $(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function qe(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),n=(e%6e4/1e3).toFixed(0);return`${t}m ${n}s`}function Tr(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Je(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 pi(e){return`<span class="badge badge-${e}">${e}</span>`}function di(e){if(!e.code)return"";let t=e.code.split(`
689
689
  `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((s,o)=>{let i=e.codeStartLine+o,a=i===e.codeLine,c=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${$(s)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(r=>`<span class="code-line code-line-body">${$(r)}</span>`).join("")}</pre></div>`}function ui(e){let t=e.duration!=null?`<span class="step-duration">${qe(e.duration)}</span>`:"",n=Je(e.status);if(e.screenshot||e.message||e.error||e.code){let s="";e.screenshot&&(s=`<img src="${$(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":di(e),i="";e.error&&(i=`<div class="step-error"><pre>${$(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${$(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
@@ -1557,8 +1557,8 @@ statements:
1557
1557
  </script>
1558
1558
  </body>
1559
1559
  </html>`}var Ar=_(()=>{"use strict"});import*as J from"fs";import*as Q from"path";import{execFileSync as gi}from"child_process";import{createHash as mi}from"crypto";import be from"axios";function Er(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let n=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(n)return`https://${n[1]}/${n[2]}`}function oe(...e){try{return gi("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function yi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=J.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function wi(){let e={nodeVersion:process.version};if(process.env.GITHUB_ACTIONS){let t=process.env.GITHUB_SERVER_URL??"https://github.com",n=process.env.GITHUB_REPOSITORY??"",r=process.env.GITHUB_RUN_ID??"",s=process.env.GITHUB_EVENT_NAME??"",i=yi().pull_request,a=i?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",p=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,f=process.env.SHIPLIGHT_PR_TITLE??i?.title,m=oe("log","-1","--pretty=%s"),g=oe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:d,gitRepo:n,commitMessage:m,authorEmail:g,prNumber:p,prTitle:f,prUrl:p&&n?`${t}/${n}/pull/${p}`:void 0,ciBuildId:r,ciBuildUrl:r&&n?`${t}/${n}/actions/runs/${r}`:void 0,commitUrl:c&&n?`${t}/${n}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:s,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",n=process.env.CI_COMMIT_SHA??"",r=process.env.CI_MERGE_REQUEST_IID,s=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:n,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:s,prNumber:r,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:r&&t?`${t}/-/merge_requests/${r}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:n&&t?`${t}/commit/${n}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",n=process.env.CIRCLE_PROJECT_USERNAME??"",r=process.env.CIRCLE_PROJECT_REPONAME??"",s=n&&r?`${n}/${r}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,i=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?Er(a):void 0,l=oe("log","-1","--pretty=%s"),p=oe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:s,commitMessage:l,authorEmail:p,prNumber:i,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=oe("rev-parse","HEAD"),n=oe("rev-parse","--abbrev-ref","HEAD"),r=oe("log","-1","--pretty=%s"),s=oe("log","-1","--pretty=%ae"),o=oe("remote","get-url","origin"),i=o?Er(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:n,commitMessage:r,authorEmail:s,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function bi(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 Pr(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function vi(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 Si(e,t){let n=new Map;for(let r of t){r.screenshotS3Uris={};let s=n.get(r.testCaseName);s?s.push(r):n.set(r.testCaseName,[r])}return e.map(r=>n.get(r.title)?.shift())}function Xe(e,t){return Q.isAbsolute(t)?t:Q.join(e,t)}function _i(e,t,n,r){let s={};for(let o of e.steps)s[o.stepId]={description:o.description,status:o.status,duration:o.duration,message:o.error??o.message,screenshotS3Uri:t[o.stepId]};return{schemaVersion:2,result:Pr(e.status),flaky:!1,segments:[{outcome:Pr(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:s,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:n,traceS3Uri:r,actionStepsMap:e.actionStepsMap??{}}]}}function Ut(e){return mi("md5").update(e).digest("base64")}function Nt(e){return Ut(J.readFileSync(e))}async function Ft(e,t){let n=J.readFileSync(t),r=Q.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[r]??"application/octet-stream";await be.put(e,n,{headers:{"Content-Type":o,"Content-MD5":Ut(n)}})}async function $r(e,t,n,r){let s=Ye(r,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${r}`,"Content-Type":"application/json"},i=wi(),a=e.tests.map(u=>{let v={testCaseName:u.title,testCaseBaseName:u.baseTitle,suiteName:u.suiteName,file:u.file,tags:u.tags,suiteTags:u.suiteTags,baseUrl:u.baseUrl,skip:u.skip,slow:u.slow,timeout:u.timeout,parameterSetName:u.parameterSetName,flaky:u.flaky,retries:u.retries};if(u.videoPath){let h=Xe(t,u.videoPath);J.existsSync(h)&&(v.videoMd5=Nt(h))}if(u.tracePath){let h=Xe(t,u.tracePath);J.existsSync(h)&&(v.traceMd5=Nt(h))}return v}),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 be.post(`${s}/v1/local-runs`,{trigger:i.ciProvider,startTime:n,metadata:i,tests:a},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let d=Si(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(u,v)=>{let h=d[v];if(!h)return;let b=u.steps.filter(A=>A.screenshot);if(!b.length)return;let y=b.map(A=>A.stepId),x={};for(let A of b){let M=Q.isAbsolute(A.screenshot)?A.screenshot:Q.join(t,A.screenshot);J.existsSync(M)&&(x[A.stepId]=Nt(M))}try{let A=await be.post(`${s}/v1/local-runs/${p.testRunId}/results/${h.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:x},{headers:o});h.uploadUrls.screenshots=A.data.screenshots,h.screenshotS3Uris=A.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${u.title}"`)}catch(A){console.warn(`[reporter] Failed to get screenshot URLs for "${u.title}":`,A)}})),console.log("[reporter] [3/4] Uploading assets...");let f=(await Promise.all(e.tests.map(async(u,v)=>{let h=d[v];if(!h){console.warn(`[reporter] No result slot found for test "${u.title}", skipping.`);return}let b=h.uploadUrls,y={},x=0;await Promise.all(u.steps.map(async L=>{if(L.screenshot&&b.screenshots?.[L.stepId]){let ue=Q.isAbsolute(L.screenshot)?L.screenshot:Q.join(t,L.screenshot);if(J.existsSync(ue))try{await Ft(b.screenshots[L.stepId],ue),y[L.stepId]=h.screenshotS3Uris[L.stepId],x++}catch(Jr){console.warn(`[reporter] Screenshot upload failed for step ${L.stepId}:`,Jr)}}})),x>0&&console.log(`[reporter] [3/4] Uploaded ${x} screenshot(s) for "${u.title}"`);let A;if(u.videoPath&&b.video){let L=Xe(t,u.videoPath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading video for "${u.title}"...`);try{await Ft(b.video,L),A=h.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${u.title}"`)}catch(ue){console.warn("[reporter] Video upload failed:",ue)}}}let M;if(u.tracePath&&b.trace){let L=Xe(t,u.tracePath);if(J.existsSync(L)){console.log(`[reporter] [3/4] Uploading trace for "${u.title}"...`);try{await Ft(b.trace,L),M=h.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${u.title}"`)}catch(ue){console.warn("[reporter] Trace upload failed:",ue)}}}console.log(`[reporter] [3/4] Uploading report for "${u.title}"...`);let tt=_i(u,y,A,M),Bt=Buffer.from(JSON.stringify(tt)),Ht=Ut(Bt),Wt=await be.post(`${s}/v1/local-runs/${p.testRunId}/results/${h.testCaseResultId}/report-url`,{md5:Ht},{headers:o}),zr=Wt.data.reportUrl,Yr=Wt.data.reportS3Uri;return await be.put(zr,Bt,{headers:{"Content-Type":"application/json","Content-MD5":Ht}}),console.log(`[reporter] [3/4] Report uploaded for "${u.title}"`),{testCaseResultId:h.testCaseResultId,result:bi(u.status),durationMs:u.duration,startTime:u.startTime,endTime:u.endTime,error:u.error,reportS3Uri:Yr,videoS3Uri:A,traceS3Uri:M,metadata:{suiteName:u.suiteName,file:u.file}}}))).filter(u=>!!u);console.log("[reporter] [4/4] Finalising run...");let m=vi(e.tests);console.log(`[reporter] [4/4] Overall status: ${m}`);let g=await be.put(`${s}/v1/local-runs/${p.testRunId}/complete`,{status:m,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:f},{headers:o});console.log(`
1560
- Shiplight cloud report: ${g.data.reportUrl}`)}var Mr=_(()=>{"use strict";Pt()});var Cr={};ee(Cr,{buildGitHubSummary:()=>Rr,runReport:()=>xi});import*as P from"fs";import*as E from"path";async function xi(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"),n=e.includes("--merge"),r=e.includes("--github-summary");!n&&r&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),n?await ki(e,t,r):await Ti(e,t)}async function Ti(e,t){let n=e.find(a=>!a.startsWith("--"))||"shiplight-report",r=E.isAbsolute(n)?n:E.join(process.cwd(),n),s=E.join(r,"report-data.json");P.existsSync(s)||(console.error(`Error: ${s} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(P.readFileSync(s,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${s}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=E.join(r,"index.html");if(P.writeFileSync(i,Dt(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Lr(o,r),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function ki(e,t,n){let r=E.join(process.cwd(),"shiplight-report"),s=e.findIndex(m=>m==="-o"||m==="--output");if(s!==-1&&e[s+1]){let m=e[s+1];r=E.isAbsolute(m)?m:E.join(process.cwd(),m)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let m=0;m<e.length;m++){let g=e[m];if(o.has(g)){m++;continue}if(i.has(g))continue;let u=E.isAbsolute(g)?g:E.join(process.cwd(),g);a.push(u)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;P.mkdirSync(E.join(r,"screenshots"),{recursive:!0});for(let m=0;m<a.length;m++){let g=a[m],u=`shard-${m}`,v=E.join(g,"report-data.json");if(!P.existsSync(v)){console.warn(`Warning: No report-data.json found in ${g}, skipping.`);continue}let h;try{h=JSON.parse(P.readFileSync(v,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${v}, skipping.`);continue}console.log(`Merging ${u}: ${h.tests.length} tests from ${g}`),l+=h.totalDuration||0;let b=E.join(g,"screenshots");P.existsSync(b)&&Or(b,E.join(r,"screenshots",u));let y=E.join(r,u);for(let x of h.tests){let A=[x,...x.attempts||[]];for(let M of A)Ai(M.steps,u),M.videoPath&&Ir(g,M.videoPath,y)&&(M.videoPath=`${u}/${M.videoPath}`),M.tracePath&&Ir(g,M.tracePath,y)&&(M.tracePath=`${u}/${M.tracePath}`);c.push(x)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:ve};P.writeFileSync(E.join(r,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let f=E.join(r,"index.html");if(P.writeFileSync(f,Dt(d),"utf-8"),console.log(`
1561
- Merged ${c.length} tests from ${p} shards into: ${f}`),await Lr(d,r),n&&Ei(c),t)try{let m=(await import("open")).default;await m(f)}catch{}}function Ir(e,t,n){let r=E.resolve(e,t);return r.startsWith(E.resolve(e)+E.sep)?P.existsSync(r)?(P.mkdirSync(n,{recursive:!0}),P.copyFileSync(r,E.join(n,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Or(e,t){P.mkdirSync(t,{recursive:!0});for(let n of P.readdirSync(e,{withFileTypes:!0})){let r=E.join(e,n.name),s=E.join(t,n.name);n.isDirectory()?Or(r,s):P.copyFileSync(r,s)}}function Ai(e,t){for(let n of e)n.screenshot?.startsWith("screenshots/")&&(n.screenshot=n.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function Lr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let r=e.tests.map(o=>o.startTime).filter(o=>!!o),s=r.length>0?r.sort()[0]:e.timestamp??new Date().toISOString();try{await $r(e,t,s,n)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function jt(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),n=E.join("tests",E.basename(t));return{name:e.title||E.basename(t),yamlPath:n}}function Rr(e){let t=e.filter(a=>!a.file.includes("auth.setup")),n=t.filter(a=>a.flaky),r=t.filter(a=>a.status==="passed"&&!a.flaky),s=t.filter(a=>a.status!=="passed"),o=t.length,i=`## Test Results
1560
+ Shiplight cloud report: ${xi(g.data.reportUrl,s)}`)}function xi(e,t){if(/^https?:\/\//i.test(e))return e;let n=e.startsWith("/")?e:`/${e}`;return`${Ti(t)}${n}`}function Ti(e){return e==="https://api.shiplight.ai"?"https://app.shiplight.ai":e==="https://nova-api.shiplight.ai"?"https://nova.shiplight.ai":e.endsWith("/")?e.slice(0,-1):e}var Mr=_(()=>{"use strict";Pt()});var Cr={};ee(Cr,{buildGitHubSummary:()=>Rr,runReport:()=>ki});import*as P from"fs";import*as E from"path";async function ki(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"),n=e.includes("--merge"),r=e.includes("--github-summary");!n&&r&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),n?await Ei(e,t,r):await Ai(e,t)}async function Ai(e,t){let n=e.find(a=>!a.startsWith("--"))||"shiplight-report",r=E.isAbsolute(n)?n:E.join(process.cwd(),n),s=E.join(r,"report-data.json");P.existsSync(s)||(console.error(`Error: ${s} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(P.readFileSync(s,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${s}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=E.join(r,"index.html");if(P.writeFileSync(i,Dt(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Lr(o,r),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Ei(e,t,n){let r=E.join(process.cwd(),"shiplight-report"),s=e.findIndex(m=>m==="-o"||m==="--output");if(s!==-1&&e[s+1]){let m=e[s+1];r=E.isAbsolute(m)?m:E.join(process.cwd(),m)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let m=0;m<e.length;m++){let g=e[m];if(o.has(g)){m++;continue}if(i.has(g))continue;let u=E.isAbsolute(g)?g:E.join(process.cwd(),g);a.push(u)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;P.mkdirSync(E.join(r,"screenshots"),{recursive:!0});for(let m=0;m<a.length;m++){let g=a[m],u=`shard-${m}`,v=E.join(g,"report-data.json");if(!P.existsSync(v)){console.warn(`Warning: No report-data.json found in ${g}, skipping.`);continue}let h;try{h=JSON.parse(P.readFileSync(v,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${v}, skipping.`);continue}console.log(`Merging ${u}: ${h.tests.length} tests from ${g}`),l+=h.totalDuration||0;let b=E.join(g,"screenshots");P.existsSync(b)&&Or(b,E.join(r,"screenshots",u));let y=E.join(r,u);for(let x of h.tests){let A=[x,...x.attempts||[]];for(let M of A)Pi(M.steps,u),M.videoPath&&Ir(g,M.videoPath,y)&&(M.videoPath=`${u}/${M.videoPath}`),M.tracePath&&Ir(g,M.tracePath,y)&&(M.tracePath=`${u}/${M.tracePath}`);c.push(x)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:ve};P.writeFileSync(E.join(r,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let f=E.join(r,"index.html");if(P.writeFileSync(f,Dt(d),"utf-8"),console.log(`
1561
+ Merged ${c.length} tests from ${p} shards into: ${f}`),await Lr(d,r),n&&$i(c),t)try{let m=(await import("open")).default;await m(f)}catch{}}function Ir(e,t,n){let r=E.resolve(e,t);return r.startsWith(E.resolve(e)+E.sep)?P.existsSync(r)?(P.mkdirSync(n,{recursive:!0}),P.copyFileSync(r,E.join(n,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Or(e,t){P.mkdirSync(t,{recursive:!0});for(let n of P.readdirSync(e,{withFileTypes:!0})){let r=E.join(e,n.name),s=E.join(t,n.name);n.isDirectory()?Or(r,s):P.copyFileSync(r,s)}}function Pi(e,t){for(let n of e)n.screenshot?.startsWith("screenshots/")&&(n.screenshot=n.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function Lr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let n=process.env.SHIPLIGHT_API_TOKEN;if(!n){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}let r=e.tests.map(o=>o.startTime).filter(o=>!!o),s=r.length>0?r.sort()[0]:e.timestamp??new Date().toISOString();try{await $r(e,t,s,n)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function jt(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),n=E.join("tests",E.basename(t));return{name:e.title||E.basename(t),yamlPath:n}}function Rr(e){let t=e.filter(a=>!a.file.includes("auth.setup")),n=t.filter(a=>a.flaky),r=t.filter(a=>a.status==="passed"&&!a.flaky),s=t.filter(a=>a.status!=="passed"),o=t.length,i=`## Test Results
1562
1562
 
1563
1563
  `;if(s.length===0&&n.length===0?i+=`\u2705 All ${o} tests passed
1564
1564
 
@@ -1583,9 +1583,9 @@ Merged ${c.length} tests from ${p} shards into: ${f}`),await Lr(d,r),n&&Ei(c),t)
1583
1583
  `;for(let c of a)i+=`- ${c}
1584
1584
  `;i+=`
1585
1585
  </details>
1586
- `}return i}function Ei(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,Rr(e)),console.log("GitHub step summary written.")}var Dr=_(()=>{"use strict";Ar();Mr();Pe()});var Nr,Fr=_(()=>{"use strict";Nr="0.1.59"});var Ur={};ee(Ur,{runTranspile:()=>$i});import*as Ze from"path";import{glob as Pi}from"glob";async function $i(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",n=process.cwd(),r=await Pi(t,{cwd:n,ignore:["node_modules/**","*.yaml.spec.ts"]});r.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of r.sort()){let c=Ze.resolve(n,a),l=Xn(c,{version:Nr});if(!l.valid){s++,console.log(`
1586
+ `}return i}function $i(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,Rr(e)),console.log("GitHub step summary written.")}var Dr=_(()=>{"use strict";Ar();Mr();Pe()});var Nr,Fr=_(()=>{"use strict";Nr="0.1.60"});var Ur={};ee(Ur,{runTranspile:()=>Ii});import*as Ze from"path";import{glob as Mi}from"glob";async function Ii(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",n=process.cwd(),r=await Mi(t,{cwd:n,ignore:["node_modules/**","*.yaml.spec.ts"]});r.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let s=0,o=0,i=0;for(let a of r.sort()){let c=Ze.resolve(n,a),l=Xn(c,{version:Nr});if(!l.valid){s++,console.log(`
1587
1587
  \u2717 ${a}`);for(let d of l.errors)console.log(` ERROR: ${d}`);continue}i++;let p=Ze.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${p}`);for(let d of l.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${a} \u2192 ${p}`)}console.log(`
1588
- ${r.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var jr=_(()=>{"use strict";xt();Fr()});var Wr={};ee(Wr,{runInspect:()=>Mi});import*as Qe from"fs";import*as Br from"path";async function Mi(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"),n=e.includes("--stats"),r=e.find(i=>!i.startsWith("--"));r||(console.error("Error: no file specified"),process.exit(1));let s=Br.resolve(process.cwd(),r);Qe.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=Qe.readFileSync(s,"utf-8");try{let i=xe(o),a=C(o);if(n)Ii(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${r}: ${i.message}`),process.exit(1)}}function Ii(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 n=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${n.tests.length}`);for(let r of n.tests){let s=r.skip?` [SKIP${typeof r.skip=="string"?`: ${r.skip}`:""}]`:"";console.log(` - ${r.name}: ${r.statements.length} statements${r.teardown?`, ${r.teardown.length} teardown`:""}${s}`)}n.beforeAll?.length&&console.log(`Hooks: beforeAll (${n.beforeAll.length})`),n.afterAll?.length&&console.log(`Hooks: afterAll (${n.afterAll.length})`),n.beforeEach?.length&&console.log(`Hooks: beforeEach (${n.beforeEach.length})`),n.afterEach?.length&&console.log(`Hooks: afterEach (${n.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 n=Hr(e.statements??[]);console.log(` DRAFT: ${n.drafts}, ACTION: ${n.actions}, STEP: ${n.steps}`)}}function Hr(e){let t={drafts:0,actions:0,steps:0};for(let n of e)if(n.type==="DRAFT")t.drafts++;else if(n.type==="ACTION")t.actions++;else if(n.type==="STEP"){t.steps++;let r=Hr(n.statements??[]);t.drafts+=r.drafts,t.actions+=r.actions,t.steps+=r.steps}return t}var Gr=_(()=>{"use strict";ae()});var Kr=Xr((Mp,Oi)=>{Oi.exports={name:"shiplightai",version:"0.1.59",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.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Pe();import Li from"dotenv";Li.config();Gt();Kt();function Vr(){console.log(`
1588
+ ${r.length} file(s): ${i} transpiled, ${s} error(s), ${o} warning(s)`),process.exit(s>0?1:0)}var jr=_(()=>{"use strict";xt();Fr()});var Wr={};ee(Wr,{runInspect:()=>Oi});import*as Qe from"fs";import*as Br from"path";async function Oi(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"),n=e.includes("--stats"),r=e.find(i=>!i.startsWith("--"));r||(console.error("Error: no file specified"),process.exit(1));let s=Br.resolve(process.cwd(),r);Qe.existsSync(s)||(console.error(`Error: file not found: ${s}`),process.exit(1));let o=Qe.readFileSync(s,"utf-8");try{let i=xe(o),a=C(o);if(n)Li(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${r}: ${i.message}`),process.exit(1)}}function Li(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 n=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${n.tests.length}`);for(let r of n.tests){let s=r.skip?` [SKIP${typeof r.skip=="string"?`: ${r.skip}`:""}]`:"";console.log(` - ${r.name}: ${r.statements.length} statements${r.teardown?`, ${r.teardown.length} teardown`:""}${s}`)}n.beforeAll?.length&&console.log(`Hooks: beforeAll (${n.beforeAll.length})`),n.afterAll?.length&&console.log(`Hooks: afterAll (${n.afterAll.length})`),n.beforeEach?.length&&console.log(`Hooks: beforeEach (${n.beforeEach.length})`),n.afterEach?.length&&console.log(`Hooks: afterEach (${n.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 n=Hr(e.statements??[]);console.log(` DRAFT: ${n.drafts}, ACTION: ${n.actions}, STEP: ${n.steps}`)}}function Hr(e){let t={drafts:0,actions:0,steps:0};for(let n of e)if(n.type==="DRAFT")t.drafts++;else if(n.type==="ACTION")t.actions++;else if(n.type==="STEP"){t.steps++;let r=Hr(n.statements??[]);t.drafts+=r.drafts,t.actions+=r.actions,t.steps+=r.steps}return t}var Gr=_(()=>{"use strict";ae()});var Kr=Xr((Op,Ri)=>{Ri.exports={name:"shiplightai",version:"0.1.60",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.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Pe();import Ci from"dotenv";Ci.config();Gt();Kt();function Vr(){console.log(`
1589
1589
  Usage: shiplight <command> [options]
1590
1590
 
1591
1591
  Commands:
package/dist/index.js CHANGED
@@ -4366,7 +4366,7 @@ ${p.join(`
4366
4366
  `)}function fc(e,t){let i=[],a=t?.version||"unknown";i.push(`// @generated by shiplightai v${a}`),i.push(...go()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=Cn(t?.tags);i.push(`test.describe.serial('${s}${_e(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...fi("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...fi("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=po(d.testFlow,p.values);i.push(...yi(g,`${_e(d.name)} [${_e(p.name)}]`,o,1,h)),i.push("")}else i.push(...yi(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(...fi("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...fi("afterAll",e.afterAll,o,1)),i.push("});"),fo(i,n),i.join(`
4367
4367
  `)}function Cn(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var wc=["testContext","request"];function bi(e){let t=new Set;function i(a){for(let n of a)switch(n.type){case ue.ACTION:{let r=n.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&wc.includes(s)&&t.add(s);break}case ue.STEP:i(n.statements);break;case ue.IF_ELSE:{let o=n;i(o.then),o.else&&i(o.else);break}case ue.WHILE_LOOP:i(n.body);break}}return i(e),t}function bc(e){let t=bi(e.statements??[]);if(e.teardown)for(let i of bi(e.teardown))t.add(i);return t}function Nn(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function yi(e,t,i,a=0,n){let o=" ".repeat(a),r=[],s=bc(e),l=Nn(s),c=n?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),n?.skip===!0?r.push(`${o} test.skip();`):typeof n?.skip=="string"&&r.push(`${o} test.skip(true, '${_e(n.skip)}');`),n?.fail===!0?r.push(`${o} test.fail();`):typeof n?.fail=="string"&&r.push(`${o} test.fail(true, '${_e(n.fail)}');`),n?.slow&&r.push(`${o} test.slow();`),n?.timeout&&r.push(`${o} test.setTimeout(${n.timeout});`);let d=e.teardown&&e.teardown.length>0,h=a+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 co(e,t,i){let a=[],n=mo(t),o=bi(n),r=Nn(o);return a.push(`test.${e}(async (${r}) => {`),a.push(...pe(n,1,i,e)),a.push("});"),a}function fi(e,t,i,a){let n=" ".repeat(a),o=[],r=mo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...pe(r,a+1,l,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let l=bi(r),c=Nn(l);o.push(`${n}test.${e}(async (${c}) => {`),o.push(...pe(r,a+1,i,e)),o.push(`${n}});`)}return o}function mo(e){let i=tc({goal:"_hook",statements:e});return me(i).statements??[]}function po(e,t){let i=An(e);for(let[a,n]of Object.entries(t))i=i.split(`<<${a}>>`).join(String(n));return me(i)}function go(){return["import { test, expect } from 'shiplightai/fixture';"]}function fo(e,t){if(t.size>0){let i=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(i=n+1);let a=Array.from(t);e.splice(i,0,...a)}}var ho=5;function bo(e,t,i){let a={expandingPaths:new Set([On(t)]),depth:0,referencedPaths:new Set,basePath:i},n={...e};Array.isArray(n.statements)&&(n.statements=Ne(n.statements,t,a)),Array.isArray(n.teardown)&&(n.teardown=Ne(n.teardown,t,a));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=Ne(n[o],t,a));return{doc:n,referencedTemplatePaths:Array.from(a.referencedPaths)}}function Ne(e,t,i){let a=[];for(let n of e)if(Ec(n)){let o=Tc(n,t,i);a.push(o)}else a.push(Ac(n,t,i));return a}function Ec(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Tc(e,t,i){if(i.depth>=ho)throw new Error(`Template expansion exceeded maximum depth of ${ho}. Check for deeply nested or circular template references.`);let a=On(kc(t),e.template),n=!_c(a)&&i.basePath?On(i.basePath,e.template):a;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 o;try{o=vc(n,"utf-8")}catch(w){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${w.message}`)}let r=uo(o);if(!r||typeof r!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let s=r.params||[],l=e.params||{};for(let w of s)if(!(w in l))throw new Error(`Template ${e.template} requires param "${w}" but it was not provided. Required params: [${s.join(", ")}]`);let c=r.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let y=Sc(c);for(let[b,m]of Object.entries(l))y=y.split(`<<${b}>>`).join(String(m));c=uo(y)}let d={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths},h=Ne(c,n,d),g={STEP:r.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:h};return Object.keys(l).length>0&&(g.template_params=l),g}function Ac(e,t,i){if(typeof e!="object"||e===null)return e;let a={...e};return Array.isArray(a.statements)&&(a.statements=Ne(a.statements,t,i)),Array.isArray(a.THEN)&&(a.THEN=Ne(a.THEN,t,i)),Array.isArray(a.ELSE)&&(a.ELSE=Ne(a.ELSE,t,i)),Array.isArray(a.DO)&&(a.DO=Ne(a.DO,t,i)),a}var Dn=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Rn(e,t,i){let a=xc(e),n=a?.name,o=a?.tags,r=a?.use;if(a&&(a.name!==void 0||a.tags!==void 0||a.use!==void 0)&&(delete a.name,delete a.tags,delete a.use),a?.suite){if(a.goal||a.statements)throw new Dn('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Mc(a,n,o,r,t,i)}return $c(a,n,o,r,t,i)}function $c(e,t,i,a,n,o){let r=e?.beforeEach,s=e?.afterEach,l=yo(e?.parameters),c=e?.timeout,d=e?.skip,h=e?.fail,p=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new Dn(`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 w=[];if(n&&e&&typeof e=="object"){let m=bo(e,n,o);e=m.doc,w=m.referencedTemplatePaths}let y=wo(e),b=me(y);return n&&(Xe(b.statements??[],n,"main"),b.teardown&&Xe(b.teardown,n,"teardown")),{testFlow:b,name:t,tags:i,use:a,beforeEach:r,afterEach:s,parameters:l,timeout:c,skip:d,fail:h,only:p,slow:g,referencedTemplatePaths:w}}function Mc(e,t,i,a,n,o){let r=e.suite;if(!Array.isArray(r.tests)||r.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=r.beforeAll,l=r.afterAll,c=r.beforeEach,d=r.afterEach,h=[],p=r.tests.map(y=>{if(!y.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(y.statements)||y.statements.length===0)throw new Error(`Suite test "${y.name}" must have a non-empty "statements" array.`);let b={goal:y.name,statements:y.statements};y.teardown&&(b.teardown=y.teardown);let m=[],x=b;if(n&&typeof b=="object"){let $=bo(b,n,o);x=$.doc,m=$.referencedTemplatePaths,h.push(...m)}let S=wo(x),E=me(S),A=yo(y.parameters);return{testFlow:E,name:y.name,tags:Array.isArray(y.tags)?y.tags:void 0,parameters:A,timeout:y.timeout,skip:y.skip,fail:y.fail,only:y.only,slow:y.slow}}),g=r.base_url,w=g?{...a,baseURL:g}:a;return{suite:{beforeAll:s,afterAll:l,beforeEach:c,afterEach:d,tests:p},name:t,tags:i,use:w,referencedTemplatePaths:h}}function yo(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 Xe(e,t,i){for(let a=0;a<e.length;a++){let n=e[a],o=`${i}.${a}`,r=n.description||"";if(n.uid=Ic(t,o,r),n.type===ue.STEP)Xe(n.statements,t,o);else if(n.type===ue.IF_ELSE){let s=n;Xe(s.then,t,`${o}.then`),s.else&&Xe(s.else,t,`${o}.else`)}else n.type===ue.WHILE_LOOP&&Xe(n.body,t,`${o}.body`)}}function Ic(e,t,i){let a=yc("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${a.slice(0,8)}-${a.slice(8,12)}-${a.slice(12,16)}-${a.slice(16,20)}-${a.slice(20,32)}`}function xo(e,t,i){let a=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=a||n?null:Mn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Rn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},h=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=fc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=gc(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"),Cc(Oc(s),{recursive:!0}),Pc(s,r)}catch(c){let d=c instanceof Dn?"":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 Fn="0.1.59";function vo(e){try{return Lc(e).mtimeMs}catch{return 0}}var Rc=`// @generated by shiplightai v${Fn}`;function Fc(e,t){if(!ko(e)||_o(e,"utf-8").split(`
4369
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Cc(Oc(s),{recursive:!0}),Pc(s,r)}catch(c){let d=c instanceof Dn?"":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 Fn="0.1.60";function vo(e){try{return Lc(e).mtimeMs}catch{return 0}}var Rc=`// @generated by shiplightai v${Fn}`;function Fc(e,t){if(!ko(e)||_o(e,"utf-8").split(`
4370
4370
  `,1)[0]!==Rc)return!1;let a=vo(e);for(let n of t)if(vo(n)>a)return!1;return!0}function Hc(e){let t=process.argv.slice(2),i=[],a=Hn(e);for(let n of t){if(n.startsWith("-"))continue;let o=n.endsWith(".yaml.spec.ts")?n.replace(/\.yaml\.spec\.ts$/,".test.yaml"):n;if(!o.endsWith(".test.yaml"))continue;let r=Hn(e,o);ko(r)&&i.push(r.startsWith(a)?r.slice(a.length+1):o)}return i.length>0?i:null}function So(e){let t=Hc(e.cwd),i=t??Dc("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),a=[];for(let n of i){let o=Hn(e.cwd,n),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=_o(o,"utf-8");try{let l=Rn(s,o,e.projectRoot??e.cwd),c=Nc(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&Fc(r,[o,...l.referencedTemplatePaths]))continue;let p=xo(s,o,{version:Fn,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${n}:`,l),a.push({file:n,error:l})}}if(a.length>0){let n=`[shiplight] Transpilation failed for ${a.length} file(s):
4371
4371
  `+a.map(o=>` - ${o.file}`).join(`
4372
4372
  `);if(t)throw new Error(n);console.warn(n+" (skipped)")}}import*as ae from"fs";import*as gt from"path";import{globSync as Jc}from"glob";_i();function Po(e){let t=vi().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new jn:new Gn(e)}var qc=".shiplight/action-cache";function Io(e){return e.replace(/\//g,"__")+".json"}function Zc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Gn=class{constructor(t){this.cwd=t;this.cacheDir=gt.join(t,qc)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!ae.existsSync(this.cacheDir))return i;for(let a of t){let n=gt.join(this.cacheDir,Io(a));try{if(ae.existsSync(n)){let o=ae.readFileSync(n,"utf-8");i.set(a,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[a,n]of t)try{let o=gt.join(this.cacheDir,Io(a)),r=In();if(ae.existsSync(o))try{r=JSON.parse(ae.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...n.entries}};ae.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!ae.existsSync(this.cacheDir))return;let t=Jc("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,a=0;for(let n of t)try{let o=ae.readFileSync(gt.join(this.cacheDir,n),"utf-8"),r=JSON.parse(o),s=Zc(n);i.set(s,r),a+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${a} cached action entit${a===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},jn=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(Bn(),Un));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(Bn(),Un));return i(t)}loadAll(){}};_i();function td(e={}){e.dotenv!==!1&&nd(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),a=Po(t).loadAll();return So({cwd:t,projectRoot:process.cwd(),actionEntityStores:a}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function id(e,...t){return ed(e,...t)}function nd(e){To(e);for(let t of Wn(e))Qc.config({path:t})}R();R();Ct();Qe();import{z as Zs}from"zod";var _h=Zs.object({instruction:Zs.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 Qs(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:_h,usesElementIndex:!1,async execute(t,i){let{instruction:a}=t,n={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await Ze(a,n,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:a,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:a}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await It(r,n);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 kh}from"ai";import{convert as Sh}from"html-to-text";async function Eh(e,t,i){let{apiKey:a,domain:n}=e;if(!a||!n)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${n}/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:${a}`).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 w=(d.message||{}).headers||{},y=w.subject||"",b=w.from||"",m=w.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!y.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:y,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:w["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let w=`https://api.mailgun.net/v3/domains/${n}/messages/${g}`;try{let y=await fetch(w,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`,Accept:"application/json"}});if(y.ok){let b=await y.json(),m=b.Subject||"",x=b.From||"",S=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${x}`),u.info(`to_addr: ${S}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let $=b["body-html"]||b["body-plain"]||"";if($&&$.includes("<")&&($=Sh($)),u.info(`Body: ${$.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!$.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:x,to:S,date:E,body:$,message_id:A});continue}else u.warn(`Messages API returned ${y.status}`)}catch(y){u.warn(`Failed to parse JSON response: ${y}`)}}try{let w=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`}});if(!w.ok){u.warn(`Could not fetch stored message: ${w.status}`);continue}let y=await w.text();u.info(`Fallback: Raw email length: ${y.length}`);let b=y.split(`
package/dist/reporter.js CHANGED
@@ -875,7 +875,7 @@ import*as k from"fs";import*as v from"path";import{v4 as vt}from"uuid";import{z
875
875
  });
876
876
  </script>
877
877
  </body>
878
- </html>`}var Te="0.1.59",Ae=Te!=="dev"?Te:void 0;var yn=3600*1e3,bn=10080*60*1e3;var pt={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return pt[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function ht(e){return[...e].sort(([t],[r])=>{let i=$e(t),n=$e(r);if(i!==n)return i-n;let a=Ee(t),o=Ee(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function ft(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 mt(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function q(e,t="main",r,i,n){r===void 0&&(r=!ft(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=mt(o.title,t);a.push(...q(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,g={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(g.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(g),o.steps.length>0&&a.push(...q(o.steps,u,r,i,n))}}return a}var Z=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 u of this.collected){let g=u.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(u)}let i=[];for(let[,u]of r.entries()){let g=u[0].test.location.file,y=[],d,p,m;for(let w=0;w<u.length;w++){let{test:T,result:M}=u[w],A=await this.buildReportTest(T,M,g);d=A,p||(p=A.startTime),m=A.endTime,y.push({attemptNumber:w+1,status:M.status,duration:M.duration,steps:A.steps,error:A.error,videoPath:A.videoPath,tracePath:A.tracePath})}let h=y[y.length-1],{test:S}=u[u.length-1],b={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),g),status:h.status,duration:h.duration,steps:h.steps,error:h.error,videoPath:h.videoPath,tracePath:h.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:p,endTime:m,suiteName:d?.suiteName},_=y.some(w=>w.status==="failed"||w.status==="timedOut");y.length>1&&_&&h.status==="passed"&&(b.flaky=!0,b.retries=y.length-1,b.attempts=y),i.push(b)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);k.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let g=n.tests[u],y=g.attempts&&g.attempts.length>0,d=[{obj:g,prefix:y?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(g.attempts)for(let p=0;p<g.attempts.length;p++)d.push({obj:g.attempts[p],prefix:`test-${u}-attempt-${p+1}`,screenshotSubDir:`test-${u}/attempt-${p}`});for(let{obj:p,prefix:m,screenshotSubDir:h}of d){let S=v.join(o,h),b=!1;for(let _ of p.steps)if(_.screenshot&&v.isAbsolute(_.screenshot))try{b||(k.mkdirSync(S,{recursive:!0}),b=!0);let w=`${_.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(_.screenshot,v.join(S,w)),_.screenshot=`screenshots/${h}/${w}`}catch(w){console.warn(`[reporter] Failed to copy screenshot for ${_.stepId}:`,w)}if(p.videoPath&&v.isAbsolute(p.videoPath)){let _=v.extname(p.videoPath)||".webm",w=`${m}-video${_}`;try{k.copyFileSync(p.videoPath,v.join(a,w)),p.videoPath=w}catch{p.videoPath=void 0}}if(p.tracePath&&v.isAbsolute(p.tracePath)){let _=v.extname(p.tracePath)||".zip",w=`${m}-trace${_}`;try{k.copyFileSync(p.tracePath,v.join(a,w)),p.tracePath=w}catch{p.tracePath=void 0}}}}let c=v.join(a,"report-data.json");k.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(k.writeFileSync(l,ke(n),"utf-8"),console.log(`
878
+ </html>`}var Te="0.1.60",Ae=Te!=="dev"?Te:void 0;var yn=3600*1e3,bn=10080*60*1e3;var pt={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return pt[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function ht(e){return[...e].sort(([t],[r])=>{let i=$e(t),n=$e(r);if(i!==n)return i-n;let a=Ee(t),o=Ee(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function ft(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 mt(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function q(e,t="main",r,i,n){r===void 0&&(r=!ft(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=mt(o.title,t);a.push(...q(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,g={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(g.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(g),o.steps.length>0&&a.push(...q(o.steps,u,r,i,n))}}return a}var Z=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 u of this.collected){let g=u.test.titlePath().join(" > "),y=r.get(g);y||(y=[],r.set(g,y)),y.push(u)}let i=[];for(let[,u]of r.entries()){let g=u[0].test.location.file,y=[],d,p,m;for(let w=0;w<u.length;w++){let{test:T,result:M}=u[w],A=await this.buildReportTest(T,M,g);d=A,p||(p=A.startTime),m=A.endTime,y.push({attemptNumber:w+1,status:M.status,duration:M.duration,steps:A.steps,error:A.error,videoPath:A.videoPath,tracePath:A.tracePath})}let h=y[y.length-1],{test:S}=u[u.length-1],b={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),g),status:h.status,duration:h.duration,steps:h.steps,error:h.error,videoPath:h.videoPath,tracePath:h.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:p,endTime:m,suiteName:d?.suiteName},_=y.some(w=>w.status==="failed"||w.status==="timedOut");y.length>1&&_&&h.status==="passed"&&(b.flaky=!0,b.retries=y.length-1,b.attempts=y),i.push(b)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);k.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let g=n.tests[u],y=g.attempts&&g.attempts.length>0,d=[{obj:g,prefix:y?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(g.attempts)for(let p=0;p<g.attempts.length;p++)d.push({obj:g.attempts[p],prefix:`test-${u}-attempt-${p+1}`,screenshotSubDir:`test-${u}/attempt-${p}`});for(let{obj:p,prefix:m,screenshotSubDir:h}of d){let S=v.join(o,h),b=!1;for(let _ of p.steps)if(_.screenshot&&v.isAbsolute(_.screenshot))try{b||(k.mkdirSync(S,{recursive:!0}),b=!0);let w=`${_.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(_.screenshot,v.join(S,w)),_.screenshot=`screenshots/${h}/${w}`}catch(w){console.warn(`[reporter] Failed to copy screenshot for ${_.stepId}:`,w)}if(p.videoPath&&v.isAbsolute(p.videoPath)){let _=v.extname(p.videoPath)||".webm",w=`${m}-video${_}`;try{k.copyFileSync(p.videoPath,v.join(a,w)),p.videoPath=w}catch{p.videoPath=void 0}}if(p.tracePath&&v.isAbsolute(p.tracePath)){let _=v.extname(p.tracePath)||".zip",w=`${m}-trace${_}`;try{k.copyFileSync(p.tracePath,v.join(a,w)),p.tracePath=w}catch{p.tracePath=void 0}}}}let c=v.join(a,"report-data.json");k.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(k.writeFileSync(l,ke(n),"utf-8"),console.log(`
879
879
  Shiplight report written to: ${l}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(l)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let n={title:t.title,file:v.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&&(n.error=r.errors.map(d=>d.message||d.stack||String(d)).join(`
880
880
 
881
881
  `)),r.stdout.length>0&&(n.stdout=r.stdout.map(d=>typeof d=="string"?d:d.toString()).join("")),r.stderr.length>0&&(n.stderr=r.stderr.map(d=>typeof d=="string"?d:d.toString()).join(""));for(let d of r.attachments)d.name==="video"&&d.path&&(n.videoPath=d.path),d.name==="trace"&&d.path&&(n.tracePath=d.path);let a=r.attachments.find(d=>d.name==="shiplight-results"),o=null;if(a)try{if(a.body)o=JSON.parse(a.body.toString("utf-8"));else if(a.path){let d=k.readFileSync(a.path,"utf-8");o=JSON.parse(d)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),l={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),g=u?u[1]:t.title,y=u?u[2]:void 0;if(y&&(n.parameterSetName=y),k.existsSync(c))try{let d=k.readFileSync(c,"utf-8"),p=_e(d,c);if(p.suite){let m=p.suite.tests.find(h=>h.name===g);m&&(l=K(m.testFlow),m.tags?.length&&(n.tags=m.tags),m.skip!==void 0&&(n.skip=m.skip),m.slow&&(n.slow=m.slow),m.timeout!==void 0&&(n.timeout=m.timeout),n.baseTitle=m.name||m.testFlow?.goal),n.suiteName=p.name,p.tags?.length&&(n.suiteTags=p.tags),p.use?.baseURL&&(n.baseUrl=p.use.baseURL)}else p.testFlow&&(l=K(p.testFlow),n.baseTitle=p.name||p.testFlow?.goal,p.tags?.length&&(n.tags=p.tags),p.use?.baseURL&&(n.baseUrl=p.use.baseURL))}catch{}if(o||Object.keys(l).length>0){let d=new Set([...Object.keys(l),...Object.keys(o||{})]),p=Array.from(d).map(h=>[h,null]),m=ht(p);for(let[h]of m){let S=l[h],b=o?.[h],_=S?.description;if(!_||_==="Action"||_==="Draft"){let T=S?.action_entity;_=T?.action_description||T?.action_data?.kwargs?.description||b?.description||h}let w={stepId:h,description:_,status:b?.status||"pending",duration:b?.duration};if(b?.message){let T=typeof b.message=="string"?b.message:JSON.stringify(b.message,null,2);b.status==="failure"?w.error=T:w.message=T}if(b?.screenshot){let T=b.screenshot,M=a?.path?v.dirname(a.path):"",A=v.isAbsolute(T)?T:v.join(M,T);k.existsSync(A)&&(w.screenshot=A)}b?.code&&(w.code=b.code),n.steps.push(w)}}if(o===null&&Object.keys(l).length===0&&!i.endsWith(".yaml.spec.ts")){let d=new Map;if(n.steps=q(r.steps,"main",void 0,d),n.steps.length>0){let p=new Map;n.actionStepsMap=Object.fromEntries(n.steps.map(m=>{let h=d.get(m.stepId),S;if(h?.file){if(!p.has(h.file))try{p.set(h.file,k.readFileSync(h.file,"utf-8").split(`
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "shiplightai",
3
- "version": "0.1.59",
3
+ "version": "0.1.60",
4
4
  "type": "module",
5
5
  "description": "Shiplight CLI for running and debugging .test.yaml files",
6
6
  "main": "dist/index.js",
@@ -94,8 +94,8 @@
94
94
  "mcp-tools": "1.0.0",
95
95
  "sdk-internal": "0.1.1",
96
96
  "shiplight-tools": "1.0.0",
97
- "@loggia/common": "1.0.0",
98
- "shiplight-types": "0.1.0"
97
+ "shiplight-types": "0.1.0",
98
+ "@loggia/common": "1.0.0"
99
99
  },
100
100
  "peerDependencies": {
101
101
  "@playwright/test": "^1.60.0"