shiplightai 0.1.44 → 0.1.45
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/index.cjs +1 -1
- package/dist/cjs/reporter.cjs +1 -1
- package/dist/cli.js +23 -23
- package/dist/index.js +1 -1
- package/dist/reporter.js +1 -1
- package/package.json +4 -4
package/dist/cjs/index.cjs
CHANGED
|
@@ -4366,7 +4366,7 @@ ${p.join(`
|
|
|
4366
4366
|
`)}function Pc(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...Ro()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let a=new Set,o={imports:a,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=zn(t?.tags);i.push(`test.describe.serial('${s}${Ee(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...Ti("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...Ti("beforeEach",e.beforeEach,o,1)),i.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],u=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=No(d.testFlow,p.values);i.push(...Mi(g,`${Ee(d.name)} [${Ee(p.name)}]`,o,1,u)),i.push("")}else i.push(...Mi(d.testFlow,Ee(d.name),o,1,u)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&i.push("")}return e.afterEach&&e.afterEach.length>0&&(i.push(...Ti("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...Ti("afterAll",e.afterAll,o,1)),i.push("});"),Fo(i,a),i.join(`
|
|
4367
4367
|
`)}function zn(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var Oc=["testContext","request"];function $i(e){let t=new Set;function i(n){for(let a of n)switch(a.type){case ge.ACTION:{let r=a.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&Oc.includes(s)&&t.add(s);break}case ge.STEP:i(a.statements);break;case ge.IF_ELSE:{let o=a;i(o.then),o.else&&i(o.else);break}case ge.WHILE_LOOP:i(a.body);break}}return i(e),t}function Dc(e){let t=$i(e.statements??[]);if(e.teardown)for(let i of $i(e.teardown))t.add(i);return t}function Xn(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Mi(e,t,i,n=0,a){let o=" ".repeat(n),r=[],s=Dc(e),l=Xn(s),c=a?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),a?.skip===!0?r.push(`${o} test.skip();`):typeof a?.skip=="string"&&r.push(`${o} test.skip(true, '${Ee(a.skip)}');`),a?.fail===!0?r.push(`${o} test.fail();`):typeof a?.fail=="string"&&r.push(`${o} test.fail(true, '${Ee(a.fail)}');`),a?.slow&&r.push(`${o} test.slow();`),a?.timeout&&r.push(`${o} test.setTimeout(${a.timeout});`);let d=e.teardown&&e.teardown.length>0,u=n+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=we(e.statements,u+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=we(e.teardown,u+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=we(e.statements,u,i);r.push(...p)}return r.push(`${o}});`),r}function Po(e,t,i){let n=[],a=Lo(t),o=$i(a),r=Xn(o);return n.push(`test.${e}(async (${r}) => {`),n.push(...we(a,1,i,e)),n.push("});"),n}function Ti(e,t,i,n){let a=" ".repeat(n),o=[],r=Lo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${a}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${a} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...we(r,n+1,l,e)),o.push(`${a} await page.close();`),o.push(`${a}});`)}else{let l=$i(r),c=Xn(l);o.push(`${a}test.${e}(async (${c}) => {`),o.push(...we(r,n+1,i,e)),o.push(`${a}});`)}return o}function Lo(e){let i=(0,Do.stringify)({goal:"_hook",statements:e});return fe(i).statements??[]}function No(e,t){let i=Bn(e);for(let[n,a]of Object.entries(t))i=i.split(`<<${n}>>`).join(String(a));return fe(i)}function Ro(){return["import { test, expect } from 'shiplightai/fixture';"]}function Fo(e,t){if(t.size>0){let i=0;for(let a=0;a<e.length;a++)e[a].startsWith("import ")&&(i=a+1);let n=Array.from(t);e.splice(i,0,...n)}}var Oo=5;function Bo(e,t){let i={expandingPaths:new Set([(0,bt.resolve)(t)]),depth:0,referencedPaths:new Set},n={...e};Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,i)),Array.isArray(n.teardown)&&(n.teardown=Re(n.teardown,t,i));for(let a of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[a])&&(n[a]=Re(n[a],t,i));return{doc:n,referencedTemplatePaths:Array.from(i.referencedPaths)}}function Re(e,t,i){let n=[];for(let a of e)if(Lc(a)){let o=Nc(a,t,i);n.push(...o)}else n.push(Rc(a,t,i));return n}function Lc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Nc(e,t,i){if(i.depth>=Oo)throw new Error(`Template expansion exceeded maximum depth of ${Oo}. Check for deeply nested or circular template references.`);let n=(0,bt.resolve)((0,bt.dirname)(t),e.template);if(i.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);i.referencedPaths.add(n);let a;try{a=(0,Wo.readFileSync)(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=(0,ft.parse)(a);if(!o||typeof o!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let r=o.params||[],s=e.params||{};for(let d of r)if(!(d in s))throw new Error(`Template ${e.template} requires param "${d}" but it was not provided. Required params: [${r.join(", ")}]`);let l=o.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(s).length>0){let u=(0,ft.stringify)(l);for(let[p,g]of Object.entries(s))u=u.split(`<<${p}>>`).join(String(g));l=(0,ft.parse)(u)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return Re(l,n,c)}function Rc(e,t,i){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,i)),Array.isArray(n.THEN)&&(n.THEN=Re(n.THEN,t,i)),Array.isArray(n.ELSE)&&(n.ELSE=Re(n.ELSE,t,i)),Array.isArray(n.DO)&&(n.DO=Re(n.DO,t,i)),n}var Yn=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Jn(e,t){let i=(0,wt.parse)(e),n=i?.name,a=i?.tags,o=i?.use;if(i&&(i.name!==void 0||i.tags!==void 0||i.use!==void 0)&&(delete i.name,delete i.tags,delete i.use),i?.suite){if(i.goal||i.statements)throw new Yn('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Hc(i,n,a,o,t)}return Fc(i,n,a,o,t)}function Fc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=Uo(e?.parameters),l=e?.timeout,c=e?.skip,d=e?.fail,u=e?.only,p=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new Yn(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let g=[];if(a&&e&&typeof e=="object"){let b=Bo(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=(0,wt.stringify)(e),v=fe(f);return a&&(Ye(v.statements??[],a,"main"),v.teardown&&Ye(v.teardown,a,"teardown")),{testFlow:v,name:t,tags:i,use:n,beforeEach:o,afterEach:r,parameters:s,timeout:l,skip:c,fail:d,only:u,slow:p,referencedTemplatePaths:g}}function Hc(e,t,i,n,a){let o=e.suite;if(!Array.isArray(o.tests)||o.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let r=o.beforeAll,s=o.afterAll,l=o.beforeEach,c=o.afterEach,d=[],u=o.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let v={goal:f.name,statements:f.statements};f.teardown&&(v.teardown=f.teardown);let b=[],m=v;if(a&&typeof v=="object"){let A=Bo(v,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=(0,wt.stringify)(m),S=fe(y),T=Uo(f.parameters);return{testFlow:S,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:T,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),p=o.base_url,g=p?{...n,baseURL:p}:n;return{suite:{beforeAll:r,afterAll:s,beforeEach:l,afterEach:c,tests:u},name:t,tags:i,use:g,referencedTemplatePaths:d}}function Uo(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Ye(e,t,i){for(let n=0;n<e.length;n++){let a=e[n],o=`${i}.${n}`,r=a.description||"";if(a.uid=Wc(t,o,r),a.type===ge.STEP)Ye(a.statements,t,o);else if(a.type===ge.IF_ELSE){let s=a;Ye(s.then,t,`${o}.then`),s.else&&Ye(s.else,t,`${o}.else`)}else a.type===ge.WHILE_LOOP&&Ye(a.body,t,`${o}.body`)}}function Wc(e,t,i){let n=(0,Ho.createHash)("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function jo(e,t,i){let n=/\btemplate:\s/.test(e),a=/^suite:/m.test(e),o=n||a?null:Gn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Jn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},u=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=Pc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Cc(c.testFlow,{...d,testName:c.name,tags:c.tags,use:u,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"),(0,yt.mkdirSync)((0,Go.dirname)(s),{recursive:!0}),(0,yt.writeFileSync)(s,r)}catch(c){let d=c instanceof Yn?"":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 qn="0.1.
|
|
4369
|
+
`);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),(0,yt.mkdirSync)((0,Go.dirname)(s),{recursive:!0}),(0,yt.writeFileSync)(s,r)}catch(c){let d=c instanceof Yn?"":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 qn="0.1.45";function Ko(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var Bc=`// @generated by shiplightai v${qn}`;function Uc(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
|
|
4370
4370
|
`,1)[0]!==Bc)return!1;let n=Ko(e);for(let a of t)if(Ko(a)>n)return!1;return!0}function Gc(e){let t=process.argv.slice(2),i=[],n=(0,Je.resolve)(e);for(let a of t){if(a.startsWith("-"))continue;let o=a.endsWith(".yaml.spec.ts")?a.replace(/\.yaml\.spec\.ts$/,".test.yaml"):a;if(!o.endsWith(".test.yaml"))continue;let r=(0,Je.resolve)(e,o);(0,Fe.existsSync)(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function Vo(e){let t=Gc(e.cwd),i=t??(0,zo.globSync)("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),n=[];for(let a of i){let o=(0,Je.resolve)(e.cwd,a),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=(0,Fe.readFileSync)(o,"utf-8");try{let l=Jn(s,o),c=(0,Je.relative)(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&Uc(r,[o,...l.referencedTemplatePaths]))continue;let p=jo(s,o,{version:qn,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${a}:`,l),n.push({file:a,error:l})}}if(n.length>0){let a=`[shiplight] Transpilation failed for ${n.length} file(s):
|
|
4371
4371
|
`+n.map(o=>` - ${o.file}`).join(`
|
|
4372
4372
|
`);if(t)throw new Error(a);console.warn(a+" (skipped)")}}var se=J(require("fs"),1),xt=J(require("path"),1),Jo=require("glob");function qo(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new ta:new ea(e)}var zc=".shiplight/action-cache";function Yo(e){return e.replace(/\//g,"__")+".json"}function Vc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ea=class{constructor(t){this.cwd=t;this.cacheDir=xt.join(t,zc)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!se.existsSync(this.cacheDir))return i;for(let n of t){let a=xt.join(this.cacheDir,Yo(n));try{if(se.existsSync(a)){let o=se.readFileSync(a,"utf-8");i.set(n,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;se.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[n,a]of t)try{let o=xt.join(this.cacheDir,Yo(n)),r=jn();if(se.existsSync(o))try{r=JSON.parse(se.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...a.entries}};se.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!se.existsSync(this.cacheDir))return;let t=(0,Jo.globSync)("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,n=0;for(let a of t)try{let o=se.readFileSync(xt.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=Vc(a);i.set(s,r),n+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},ta=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(Qn(),Zn));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(Qn(),Zn));return i(t)}loadAll(){}};function tr(e={}){e.dotenv!==!1&&Xc(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=qo(t).loadAll();return Vo({cwd:t,actionEntityStores:n}),e.apiKey&&(process.env.__SHIPLIGHT_API_KEY=e.apiKey),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never",reportToCloud:e.reportToCloud,apiKey:e.apiKey}]]}}function ir(e,...t){return(0,er.defineConfig)(e,...t)}function Xc(e){let t=[],i=Ue.resolve(e),n=Ue.resolve(process.cwd());for(;;){let a=Ue.join(i,".env");if(Zo.existsSync(a)&&t.push(a),i===n)break;let o=Ue.dirname(i);if(o===i)break;i=o}for(let a of t)Qo.default.config({path:a})}F();F();Ft();et();var Va=require("zod"),Bu=Va.z.object({instruction:Va.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 gl(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:Bu,usesElementIndex:!1,async execute(t,i){let{instruction:n}=t,a={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await Qe(n,a,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:n,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:n}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Nt(r,a);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}$e();F();var fl=require("ai"),wl=require("html-to-text");async function Uu(e,t,i){let{apiKey:n,domain:a}=e;if(!n||!a)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${a}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}h.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let u=(d.storage||{}).url;if(h.info(`message_url: ${u}`),!u){let f=(d.message||{}).headers||{},v=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!v.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:v,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=u.split("/"),g=p[p.length-1];if(h.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${a}/messages/${g}`;try{let v=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(v.ok){let b=await v.json(),m=b.Subject||"",y=b.From||"",S=b.To||"",T=b.Date||"",A=b["Message-Id"]||"";h.info(`subject: ${m}`),h.info(`from_addr: ${y}`),h.info(`to_addr: ${S}`),h.info(`date: ${T}`),h.info(`message_id: ${A}`);let M=b["body-html"]||b["body-plain"]||"";if(M&&M.includes("<")&&(M=(0,wl.convert)(M)),h.info(`Body: ${M.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!M.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:y,to:S,date:T,body:M,message_id:A});continue}else h.warn(`Messages API returned ${v.status}`)}catch(v){h.warn(`Failed to parse JSON response: ${v}`)}}try{let f=await fetch(u,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!f.ok){h.warn(`Could not fetch stored message: ${f.status}`);continue}let v=await f.text();h.info(`Fallback: Raw email length: ${v.length}`);let b=v.split(`
|
package/dist/cjs/reporter.cjs
CHANGED
|
@@ -840,7 +840,7 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var C,N,je,ze,G,Je=at(()=>{"use st
|
|
|
840
840
|
});
|
|
841
841
|
</script>
|
|
842
842
|
</body>
|
|
843
|
-
</html>`}var zt={before:0,main:1,teardown:2,after:3};function Ye(e){let t=e.split(".")[0];return zt[t]??1}function Xe(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Vt(e){return[...e].sort(([t],[r])=>{let i=Ye(t),o=Ye(r);if(i!==o)return i-o;let n=Xe(t),s=Xe(r);for(let a=0;a<Math.max(n.length,s.length);a++){let d=n[a]??-1,u=s[a]??-1;if(d!==u)return d-u}return 0})}var ue=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud
|
|
843
|
+
</html>`}var zt={before:0,main:1,teardown:2,after:3};function Ye(e){let t=e.split(".")[0];return zt[t]??1}function Xe(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Vt(e){return[...e].sort(([t],[r])=>{let i=Ye(t),o=Ye(r);if(i!==o)return i-o;let n=Xe(t),s=Xe(r);for(let a=0;a<Math.max(n.length,s.length);a++){let d=n[a]??-1,u=s[a]??-1;if(d!==u)return d-u}return 0})}var ue=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud??!1,this.apiKey=t.apiKey||process.env.__SHIPLIGHT_API_KEY}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 f=u.test.titlePath().join(" > "),b=r.get(f);b||(b=[],r.set(f,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let f=u[0].test.location.file,b=[],p,h,g;for(let v=0;v<u.length;v++){let{test:T,result:_}=u[v],A=await this.buildReportTest(T,_,f);p=A,h||(h=A.startTime),g=A.endTime,b.push({attemptNumber:v+1,status:_.status,duration:_.duration,steps:A.steps,error:A.error,videoPath:A.videoPath,tracePath:A.tracePath})}let l=b[b.length-1],{test:x}=u[u.length-1],y={title:x.title,baseTitle:p?.baseTitle,file:S.relative(process.cwd(),f),status:l.status,duration:l.duration,steps:l.steps,error:l.error,videoPath:l.videoPath,tracePath:l.tracePath,actionStepsMap:p?.actionStepsMap,tags:p?.tags,baseUrl:p?.baseUrl,skip:p?.skip,slow:p?.slow,timeout:p?.timeout,parameterSetName:p?.parameterSetName,startTime:h,endTime:g,suiteName:p?.suiteName},w=b.some(v=>v.status==="failed"||v.status==="timedOut");b.length>1&&w&&l.status==="passed"&&(y.flaky=!0,y.retries=b.length-1,y.attempts=b),i.push(y)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString()},n=S.isAbsolute(this.outputFolder)?this.outputFolder:S.join(process.cwd(),this.outputFolder);E.mkdirSync(n,{recursive:!0});let s=S.join(n,"screenshots");for(let u=0;u<o.tests.length;u++){let f=o.tests[u],b=f.attempts&&f.attempts.length>0,p=[{obj:f,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(f.attempts)for(let h=0;h<f.attempts.length;h++)p.push({obj:f.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:g,screenshotSubDir:l}of p){let x=S.join(s,l),y=!1;for(let w of h.steps)if(w.screenshot&&S.isAbsolute(w.screenshot))try{y||(E.mkdirSync(x,{recursive:!0}),y=!0);let v=`${w.stepId.replace(/\./g,"-")}.png`;E.copyFileSync(w.screenshot,S.join(x,v)),w.screenshot=`screenshots/${l}/${v}`}catch(v){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,v)}if(h.videoPath&&S.isAbsolute(h.videoPath)){let w=S.extname(h.videoPath)||".webm",v=`${g}-video${w}`;try{E.copyFileSync(h.videoPath,S.join(n,v)),h.videoPath=v}catch{h.videoPath=void 0}}if(h.tracePath&&S.isAbsolute(h.tracePath)){let w=S.extname(h.tracePath)||".zip",v=`${g}-trace${w}`;try{E.copyFileSync(h.tracePath,S.join(n,v)),h.tracePath=v}catch{h.tracePath=void 0}}}}let a=S.join(n,"report-data.json");E.writeFileSync(a,JSON.stringify(o,null,2),"utf-8");let d=S.join(n,"index.html");if(E.writeFileSync(d,Ge(o),"utf-8"),console.log(`
|
|
844
844
|
Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(d)}catch{}if(this.reportToCloud){if(!this.apiKey){console.warn("[reporter] reportToCloud is enabled but no apiKey found, skipping cloud upload.");return}try{let{uploadToCloud:u}=await Promise.resolve().then(()=>(Je(),Ve));await u(o,n,this.runStartTime,this.apiKey)}catch(u){console.warn("[reporter] Cloud upload failed:",u)}}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:S.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(p=>p.message||p.stack||String(p)).join(`
|
|
845
845
|
|
|
846
846
|
`)),r.stdout.length>0&&(o.stdout=r.stdout.map(p=>typeof p=="string"?p:p.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(p=>typeof p=="string"?p:p.toString()).join(""));for(let p of r.attachments)p.name==="video"&&p.path&&(o.videoPath=p.path),p.name==="trace"&&p.path&&(o.tracePath=p.path);let n=r.attachments.find(p=>p.name==="shiplight-results"),s=null;if(n)try{if(n.body)s=JSON.parse(n.body.toString("utf-8"));else if(n.path){let p=E.readFileSync(n.path,"utf-8");s=JSON.parse(p)}}catch{}let a=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),f=u?u[1]:t.title,b=u?u[2]:void 0;if(b&&(o.parameterSetName=b),E.existsSync(a))try{let p=E.readFileSync(a,"utf-8"),h=Be(p,a);if(h.suite){let g=h.suite.tests.find(l=>l.name===f);g&&(d=ee(g.testFlow),g.tags?.length&&(o.tags=g.tags),g.skip!==void 0&&(o.skip=g.skip),g.slow&&(o.slow=g.slow),g.timeout!==void 0&&(o.timeout=g.timeout),o.baseTitle=g.name||g.testFlow?.goal),o.suiteName=h.name,h.tags?.length&&(o.suiteTags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL)}else h.testFlow&&(d=ee(h.testFlow),o.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(o.tags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL))}catch{}if(s||Object.keys(d).length>0){let p=new Set([...Object.keys(d),...Object.keys(s||{})]),h=Array.from(p).map(l=>[l,null]),g=Vt(h);for(let[l]of g){let x=d[l],y=s?.[l],w=x?.description;if(!w||w==="Action"||w==="Draft"){let T=x?.action_entity;w=T?.action_description||T?.action_data?.kwargs?.description||y?.description||l}let v={stepId:l,description:w,status:y?.status||"pending",duration:y?.duration};if(y?.message){let T=typeof y.message=="string"?y.message:JSON.stringify(y.message,null,2);y.status==="failure"?v.error=T:v.message=T}if(y?.screenshot){let T=y.screenshot,_=n?.path?S.dirname(n.path):"",A=S.isAbsolute(T)?T:S.join(_,T);E.existsSync(A)&&(v.screenshot=A)}o.steps.push(v)}}return Object.keys(d).length>0&&(o.actionStepsMap=d),o}},qe=ue;
|
package/dist/cli.js
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import { createRequire as __cli_createRequire } from "module";
|
|
3
3
|
const require = __cli_createRequire(import.meta.url);
|
|
4
|
-
var
|
|
4
|
+
var so=Object.defineProperty;var _=(e,t)=>()=>(e&&(t=e(e=0)),t);var ao=(e,t)=>()=>(t||e((t={exports:{}}).exports,t),t.exports),te=(e,t)=>{for(var r in t)so(e,r,{get:t[r],enumerable:!0})};import*as G from"fs";import*as q from"path";function gt(e){let t=q.resolve(e);for(;;){if(G.existsSync(q.join(t,"package.json"))||G.existsSync(q.join(t,".shiplight")))return t;let r=q.dirname(t);if(r===t)break;t=r}return null}function mt(e){return q.join(e,co)}function yt(e,t){return q.join(mt(e),`${lo}${t}.json`)}function bt(e,t){let r=gt(t);if(!r)return;let o=mt(r);G.mkdirSync(o,{recursive:!0});let i={port:e,yamlFile:t,pid:process.pid,startedAt:new Date().toISOString()};G.writeFileSync(yt(r,e),JSON.stringify(i),"utf-8")}function wt(e,t){let r=gt(t);if(r)try{G.unlinkSync(yt(r,e))}catch{}}var co,lo,St=_(()=>{"use strict";co=".shiplight/run",lo="debug-"});var Fe={};te(Fe,{findPlaywrightConfig:()=>ho,spawnPlaywrightProcess:()=>go});import*as D from"fs";import*as Y from"path";import{spawn as po}from"child_process";import{parse as uo}from"yaml";function ho(e){let t=["playwright.config.ts","playwright.config.js","playwright.config.mjs"],r=Y.resolve(e);for(;;){for(let i of t){let n=Y.join(r,i);if(D.existsSync(n))return n}let o=Y.dirname(r);if(o===r)break;r=o}return null}function fo(e,t,r,o){let i={...o},n=i.launchOptions?.args??[];return i.launchOptions={...i.launchOptions??{},args:[...n,`--remote-debugging-port=${r}`]},`// @generated by shiplightai \u2014 temporary debug test
|
|
5
5
|
import { test } from 'shiplightai/fixture';
|
|
6
6
|
${`
|
|
7
7
|
test.use(${JSON.stringify(i)});
|
|
@@ -25,15 +25,15 @@ test('__shiplight_debug__', async ({ page, agent }) => {
|
|
|
25
25
|
// Keep alive until the server is closed externally (Ctrl+C kills the process)
|
|
26
26
|
await new Promise(() => {});
|
|
27
27
|
});
|
|
28
|
-
`}async function vt(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function fo(e){let{yamlFilePath:t,configPath:r}=e,o=Y.dirname(r),i=await vt(16174),n=await vt(9222),a;if(!D.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let u=po(D.readFileSync(t,"utf-8"));u?.use&&typeof u.use=="object"&&!Array.isArray(u.use)&&(a=u.use),u?.base_url&&!a?.baseURL&&(a={...a,baseURL:u.base_url}),u?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!u.settings.auto_dismiss_modal})}catch(u){console.error("[debugger] Could not parse YAML for `use` block:",u)}let s=Y.dirname(Y.resolve(t)),c=Y.join(s,".__shiplight_debug__.yaml.spec.ts"),l=ho(t,i,n,a);D.writeFileSync(c,l);let h=lo("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});h.stdout?.on("data",u=>{process.stderr.write(u)}),h.stderr?.on("data",u=>{process.stderr.write(u)});let g=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),h.on("close",u=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{D.unlinkSync(c)}catch{}u!==0&&u!==null&&console.error(`[debugger] Playwright process exited with code ${u}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let u=0;u<180;u++){if(h.exitCode!==null){try{D.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(u===179){g();try{D.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(d=>setTimeout(d,1e3))}return{port:i,cleanup:async()=>{g();try{D.unlinkSync(c)}catch{}}}}var Ue=_(()=>{"use strict"});var _t=_(()=>{"use strict"});var xt=_(()=>{"use strict"});var Tt=_(()=>{"use strict"});import{v4 as un}from"uuid";var kt=_(()=>{"use strict"});import{z as m}from"zod";var At,Be,$t,ae,Et,Pt,Mt,C,It,be,We,Ge=_(()=>{"use strict";At=m.enum(["JS_CODE","AI_MODE"]),Be=m.object({type:At,expression:m.string()}),$t=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ae=m.object({uid:m.string(),type:$t,comment:m.string().optional()}),Et=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Pt=ae.extend({type:m.literal("DRAFT"),description:m.string()}),Mt=ae.extend({type:m.literal("ACTION"),description:m.string(),action_entity:Et.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),C=m.lazy(()=>m.union([Pt,Mt,ae.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(C),reference_id:m.number().optional()}),ae.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Be,then:m.array(C),else:m.array(C).optional()}),ae.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Be,body:m.array(C),timeout_ms:m.number().optional()})])),It=m.object({name:m.string(),statements:m.array(C),teardown:m.array(C).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),be=m.object({tests:m.array(It).min(1),beforeAll:m.array(C).optional(),afterAll:m.array(C).optional(),beforeEach:m.array(C).optional(),afterEach:m.array(C).optional()}),We=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(C).optional(),teardown:m.array(C).optional(),last_modified_at:m.string().optional(),testGroup:be.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as go,parse as Rt,parseAllDocuments as yn,parseDocument as mo,Document as yo,isMap as le,isSeq as j}from"yaml";import{v4 as H}from"uuid";function Nt(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(F)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(F)),r}function he(e,t){if(e.testGroup)return Ft(e,t);let r=Nt(e,t),o=new yo(r);return e.comment&&(o.commentBefore=e.comment),Ot(o,e.statements??[]),e.teardown&&Ot(o,e.teardown,"teardown"),o.toString(Dt)}function Ot(e,t,r="statements"){let o=e.contents;if(!o||!le(o))return;let i=o.get(r,!0);j(i)&&de(i,t)}function de(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),le(i)){let n=i;if(o.type==="STEP"){let a=n.get("statements",!0);j(a)&&de(a,o.statements)}else if(o.type==="IF_ELSE"){let a=n.get("THEN",!0);j(a)&&de(a,o.then);let s=n.get("ELSE",!0);j(s)&&o.else&&de(s,o.else)}else if(o.type==="WHILE_LOOP"){let a=n.get("DO",!0);j(a)&&de(a,o.body)}}}}function Ft(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(F)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(F)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(F)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(F)),i.tests=r.tests.map(n=>{let a={name:n.name};return n.skip!==void 0&&(a.skip=n.skip),n.timeout!==void 0&&(a.timeout=n.timeout),n.fail!==void 0&&(a.fail=n.fail),n.only!==void 0&&(a.only=n.only),n.slow!==void 0&&(a.slow=n.slow),a.statements=n.statements.map(F),n.teardown&&n.teardown.length>0&&(a.teardown=n.teardown.map(F)),a}),o.suite=i,go(o,Dt)}function F(e){switch(e.type){case"DRAFT":return bo(e);case"ACTION":return wo(e);case"STEP":return So(e);case"IF_ELSE":return vo(e);case"WHILE_LOOP":return _o(e)}}function bo(e){return{intent:e.description}}function wo(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let s=r?.statement;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:s,js:c}:{VERIFY:s}}}if(t==="go_to_url"){let s=r?.url;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:s};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let s=r?.code;if(typeof s=="string"&&e.description)return{intent:e.description,js:s}}if(t==="ai_wait_until"){let s=r?.condition;if(typeof s=="string"){let c={WAIT_UNTIL:s};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let s=r?.seconds,l={WAIT:e.description||`Wait ${s}s`};return typeof s=="number"&&(l.seconds=s),l}if(t==="js_code"){let s=r?.code;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:s}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let a=e.action_entity.xpath;if(a&&(i.xpath=a),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[s,c]of Object.entries(o.kwargs))i[s]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function So(e){let t={STEP:e.description,statements:e.statements.map(F)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function vo(e){let t={IF:Ut(e.condition),THEN:e.then.map(F)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(F)),t}function _o(e){let t={WHILE:Ut(e.condition),DO:e.body.map(F)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Ut(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function fe(e){try{let t=Rt(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),r}catch{return{}}}function He(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(He);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=He(n);return o}function O(e){if(e.length>Lt)throw new Error(`YAML input too large (${e.length} bytes, max ${Lt})`);let t=He(Rt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return xo(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:U(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=U(t.teardown));let o=We.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return we(e,i),i}function xo(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let i={tests:r.map(s=>{if(!s.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(s.statements)||s.statements.length===0)throw new Error(`Suite test "${s.name}" must have a non-empty "statements" array`);let c={name:s.name,statements:U(s.statements)};return Array.isArray(s.teardown)&&s.teardown.length>0&&(c.teardown=U(s.teardown)),s.skip!==void 0&&(c.skip=s.skip),typeof s.timeout=="number"&&(c.timeout=s.timeout),s.fail!==void 0&&(c.fail=s.fail),s.only===!0&&(c.only=!0),s.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=U(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=U(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=U(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=U(t.afterEach));let n=be.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function U(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(To)}function To(e){if(typeof e=="string")throw new Error(`Plain string statements are not supported. Use an object with a "desc" key instead. Example: { "desc": "${e}" }`);if(typeof e!="object"||e===null)throw new Error(`Invalid statement: expected object, got ${typeof e}`);let t=e;if("IF"in t)return ko(t);if("WHILE"in t)return Ao(t);if("STEP"in t)return $o(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:H(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:H(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:H(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:H(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:H(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:H(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Ct({...o,action:"function",functionName:r})}if("action"in t)return Ct(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:H(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Bt(e){if(typeof e!="string")throw new Error(`Condition must be a string, got ${typeof e}`);return e.startsWith("js:")?{type:"JS_CODE",expression:e.slice(3)}:{type:"AI_MODE",expression:e}}function ko(e){let t=Bt(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:H(),type:"IF_ELSE",condition:t,then:U(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=U(e.ELSE)),o}function Ao(e){let t=Bt(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:H(),type:"WHILE_LOOP",condition:t,body:U(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function $o(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:H(),type:"STEP",description:t,statements:U(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Ct(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[l,p]of Object.entries(e))Eo.has(l)||(a[l]=p);let s={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};o&&(s.locator=o),i&&(s.xpath=i);let c={uid:H(),type:"ACTION",description:r,action_entity:s};return n&&(c.use_pure_vision=!0),c}function we(e,t){let r;try{r=mo(e)}catch{return}let o=r.contents;if(!o||!le(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);j(n)&&t.statements&&ce(n,t.statements);let a=i.get("teardown",!0);j(a)&&t.teardown&&ce(a,t.teardown)}function ce(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&le(o)){let n=o.get("statements",!0);j(n)&&ce(n,i.statements)}else if(i.type==="IF_ELSE"&&le(o)){let n=o.get("THEN",!0);j(n)&&ce(n,i.then);let a=o.get("ELSE",!0);j(a)&&i.else&&ce(a,i.else)}else if(i.type==="WHILE_LOOP"&&le(o)){let n=o.get("DO",!0);j(n)&&ce(n,i.body)}}}var Dt,Lt,Eo,Se=_(()=>{"use strict";Ge();Dt={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Lt=1024*1024;Eo=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as _n,stringify as xn}from"yaml";var Wt=_(()=>{"use strict";Se()});var je,ve,_e=_(()=>{"use strict";je=e=>{let t=[];switch(e.type){case"STEP":e.statements&&t.push({key:"statements",statements:e.statements});break;case"IF_ELSE":e.then&&t.push({key:"then",statements:e.then}),e.else&&t.push({key:"else",statements:e.else});break;case"WHILE_LOOP":e.body&&t.push({key:"body",statements:e.body});break}return t},ve=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=je(i);for(let a of n)r(a.statements)}};return r(e),t}});function jt(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";Ht.has(i)||t++}return{action:t,draft:r}}function Mo(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Gt(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Ke(e,t){let r=t?.coverageThreshold??Po,o=[],i=[],n;try{n=O(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let a=[...ve(n.statements??[]),...n.teardown?ve(n.teardown):[]],{action:s,draft:c}=jt(a),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of a){if(g.type==="ACTION"){let f=g,u=f.action_entity?.action_data?.action_name??"";if(u==="js_code"||u==="js_action"||u==="verify"||u==="ai_assert"){let d=f.action_entity?.action_data?.kwargs?.code;if(typeof d=="string"){let w=Mo(d);if(w){let b=f.description||u;o.push(`Invalid JS in "${b}": ${w}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let u=Gt(f.condition.expression);u&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${u}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let u=Gt(f.condition.expression);u&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${u}. ${l}`)}}}let p=s+c,h=p>0?Math.round(s/p*100):0;return p>0&&h/100<r&&i.push(`Low action coverage: ${s}/${p} statements (${h}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:s,draft:c,coverage:h}}}var Po,Ht,Kt=_(()=>{"use strict";Se();_e();Po=.5,Ht=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var xe=_(()=>{"use strict"});var zt=_(()=>{"use strict";xe()});var Vt,Oo,Yt,Jt,ge,Lo,Co,Xt=_(()=>{"use strict";Vt=112,Oo=1080-Vt,Yt={"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"}},Jt={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"]},ge=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Jt[e].map(o=>Yt[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},Lo={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile")}},Co={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile",!0)}}});var qt=_(()=>{"use strict"});function ze(){return{version:"1.0",entries:{}}}var Zt=_(()=>{"use strict";_e()});var K,Ve,Qt=_(()=>{"use strict";K=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(K||{}),Ve=18e4});var er=_(()=>{"use strict"});var tr=_(()=>{"use strict"});var rr=_(()=>{"use strict"});var or=_(()=>{"use strict";xe()});var re=_(()=>{"use strict";_t();xt();Tt();kt();Wt();Kt();Se();Ge();zt();Xt();qt();Zt();_e();Qt();er();tr();rr();or();xe()});import{stringify as No}from"yaml";import{createHash as ei}from"crypto";import{parse as ti,stringify as hr}from"yaml";import{readFileSync as ri}from"fs";import{resolve as fr,dirname as oi}from"path";import{parse as ar,stringify as ii}from"yaml";import{readFileSync as ui,writeFileSync as di,mkdirSync as hi}from"fs";import{dirname as fi}from"path";function Z(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function $(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Do(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Fo(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 ke(e){let t=Do(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=Fo(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function ir(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Uo.includes(t)}function Wo(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!Bo.includes(t)}function S(e,t){I.set(e,t)}function Go(e){return I.get(e)}function ne(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function nr(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(a=>o.includes(a)||i.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${n.join(", ")})`}function J(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let a=e[n],s=`${o}.${n}`,c=Ho(a,t,s,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function Ho(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return jo(e,t,r,o);case"ACTION":return Ko(e,t,r,o);case"STEP":return zo(e,t,r,o);case"IF_ELSE":return Vo(e,t,r,o);case"WHILE_LOOP":return Yo(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function jo(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${$(n)}`,`${i}// DRAFT: ${$(n)} (requires agent - skipped in hook)`];let a=JSON.stringify(n);return[`${i}// ${r}: ${$(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${a}, '${r}');`]}function Ko(e,t,r,o){let i=" ".repeat(t),n=e.description,a=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${$(n)}`,`${i}// DRAFT: ${$(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${$(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",h=l.action_description||"",g=Go(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},u=g(l,r,f);if(o.noAgent){if(ir(l))return[`${i}// ${r}: ${$(h)}`,`${i}// AI action: ${$(h)} (requires agent - skipped in hook)`];let y=Jo(l,p,i,r);return y||[`${i}// ${r}: ${$(h)}`,...u.map(A=>`${i}${A}`)]}if(ir(l))return[`${i}// ${r}: ${$(h)}`,`${i}page = agent.agentServices.validatePage(page);`,...u.map(y=>`${i}${y}`)];let d=JSON.stringify(h),w=u.map(y=>`${i} ${y}`),b=Wo(l),v=a?`'${a}'`:"undefined";return[`${i}// ${r}: ${$(h)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...w,`${i}}, ${d}, '${r}', ${v}, ${b});`]}function zo(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${$(e.description)}`);let a=J(e.statements,t,o,r);return n.push(...a),n}function Vo(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${$(e.condition.expression)}`);let s=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${s}, "${r}")) {`)}let a=J(e.then,t+1,o,`${r}.then`);if(n.push(...a),e.else&&e.else.length>0){n.push(`${i}} else {`);let s=J(e.else,t+1,o,`${r}.else`);n.push(...s)}return n.push(`${i}}`),n}function Yo(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let a=e.timeout_ms??Ve,s=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${s}s`:`While loop exceeded default timeout of ${s}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${a};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${$(e.condition.expression)}`);let h=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${h}, "${r}")) {`)}let p=J(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function Jo(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},a=n.timeout_ms??Je;switch(t){case"go_to_url":case"open_tab":{let s=n.url||"";return[`${r}// ${o}: ${$(i)}`,`${r}await page.goto(${JSON.stringify(s)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${$(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${$(i)}`,`${r}await page.goForward();`];case"input_text":{let s=n.text||"",c=ke(e);return c?[`${r}// ${o}: ${$(i)}`,`${r}await ${c}.fill(${JSON.stringify(s)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let s=n.text||n.label||"",c=ke(e);return c?[`${r}// ${o}: ${$(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(s)} }, { timeout: ${a} });`]:null}default:return null}}function Xo(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...ur()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...sr("beforeEach",t.beforeEach,n)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let s=t?.testName||e.goal||"Generated test",c=Ye(t?.tags);for(let l of t.parameters){let p=pr(e,l.values);r.push(...$e(p,`${c}${Z(s)} [${Z(l.name)}]`,n,0,a)),r.push("")}}else{let s=t?.testName||e.goal||"Generated test",c=Ye(t?.tags);r.push(...$e(e,`${c}${Z(s)}`,n,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...sr("afterEach",t.afterEach,n))),dr(r,i),r.join(`
|
|
29
|
-
`)}function
|
|
30
|
-
`)}function Ye(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Ae(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case K.ACTION:{let a=i.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let s of a.args)typeof s=="string"&&
|
|
28
|
+
`}async function vt(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(i=>{let n=t();n.once("error",()=>i(!1)),n.once("listening",()=>{n.close(()=>i(!0))}),n.listen(r,"localhost")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function go(e){let{yamlFilePath:t,configPath:r}=e,o=Y.dirname(r),i=await vt(16174),n=await vt(9222),a;if(!D.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let u=uo(D.readFileSync(t,"utf-8"));u?.use&&typeof u.use=="object"&&!Array.isArray(u.use)&&(a=u.use),u?.base_url&&!a?.baseURL&&(a={...a,baseURL:u.base_url}),u?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!u.settings.auto_dismiss_modal})}catch(u){console.error("[debugger] Could not parse YAML for `use` block:",u)}let s=Y.dirname(Y.resolve(t)),c=Y.join(s,".__shiplight_debug__.yaml.spec.ts"),l=fo(t,i,n,a);D.writeFileSync(c,l);let h=po("npx",["playwright","test",c,"--headed"],{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o});h.stdout?.on("data",u=>{process.stderr.write(u)}),h.stderr?.on("data",u=>{process.stderr.write(u)});let g=()=>{h.killed||h.kill("SIGTERM")};process.on("SIGTERM",g),process.on("SIGINT",g),process.on("exit",g),h.on("close",u=>{process.removeListener("SIGTERM",g),process.removeListener("SIGINT",g),process.removeListener("exit",g);try{D.unlinkSync(c)}catch{}u!==0&&u!==null&&console.error(`[debugger] Playwright process exited with code ${u}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let f=`http://localhost:${i}/api/test-flow`;for(let u=0;u<180;u++){if(h.exitCode!==null){try{D.unlinkSync(c)}catch{}throw new Error(`Playwright process exited with code ${h.exitCode} before sandbox was ready`)}try{if((await fetch(f)).ok){console.error(`[debugger] Playwright sandbox ready on internal port ${i}`);break}}catch{}if(u===179){g();try{D.unlinkSync(c)}catch{}throw new Error("Timed out waiting for Playwright sandbox to start (180s)")}await new Promise(d=>setTimeout(d,1e3))}return{port:i,cleanup:async()=>{g();try{D.unlinkSync(c)}catch{}}}}var Ue=_(()=>{"use strict"});var _t=_(()=>{"use strict"});var xt=_(()=>{"use strict"});var Tt=_(()=>{"use strict"});import{v4 as dn}from"uuid";var kt=_(()=>{"use strict"});import{z as m}from"zod";var At,Be,$t,ae,Et,Pt,Mt,C,It,be,We,Ge=_(()=>{"use strict";At=m.enum(["JS_CODE","AI_MODE"]),Be=m.object({type:At,expression:m.string()}),$t=m.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),ae=m.object({uid:m.string(),type:$t,comment:m.string().optional()}),Et=m.object({action_data:m.object({action_name:m.string(),kwargs:m.record(m.any()).optional(),args:m.array(m.any()).optional()}),action_description:m.string().optional(),url:m.string().optional(),xpath:m.string().nullable().optional(),locator:m.string().nullable().optional(),css_selector:m.string().nullable().optional(),unique_selector:m.string().nullable().optional(),element_index:m.number().nullable().optional(),frame_path:m.array(m.any()).optional(),artifacts:m.record(m.any()).optional(),feedback:m.string().optional(),original_browser_use_action:m.any().optional()}).passthrough(),Pt=ae.extend({type:m.literal("DRAFT"),description:m.string()}),Mt=ae.extend({type:m.literal("ACTION"),description:m.string(),action_entity:Et.optional(),locator:m.string().optional(),use_pure_vision:m.boolean().optional()}),C=m.lazy(()=>m.union([Pt,Mt,ae.extend({type:m.literal("STEP"),description:m.string().optional().default(""),statements:m.array(C),reference_id:m.number().optional()}),ae.extend({type:m.literal("IF_ELSE"),description:m.string().optional(),condition:Be,then:m.array(C),else:m.array(C).optional()}),ae.extend({type:m.literal("WHILE_LOOP"),description:m.string().optional(),condition:Be,body:m.array(C),timeout_ms:m.number().optional()})])),It=m.object({name:m.string(),statements:m.array(C),teardown:m.array(C).optional(),skip:m.union([m.boolean(),m.string()]).optional(),timeout:m.number().optional(),fail:m.union([m.boolean(),m.string()]).optional(),only:m.boolean().optional(),slow:m.boolean().optional()}),be=m.object({tests:m.array(It).min(1),beforeAll:m.array(C).optional(),afterAll:m.array(C).optional(),beforeEach:m.array(C).optional(),afterEach:m.array(C).optional()}),We=m.object({comment:m.string().optional(),version:m.string().optional(),goal:m.string().optional(),url:m.string().optional(),baseURL:m.string().optional(),final_feedback:m.string().optional(),completed:m.boolean().optional(),success:m.boolean().optional(),statements:m.array(C).optional(),teardown:m.array(C).optional(),last_modified_at:m.string().optional(),testGroup:be.optional()}).refine(e=>e.testGroup!==void 0?e.goal===void 0&&(e.statements===void 0||e.statements.length===0):e.goal!==void 0,{message:"TestFlow must have either goal/statements (single test) or testGroup (suite), not both"})});import{stringify as mo,parse as Rt,parseAllDocuments as bn,parseDocument as yo,Document as bo,isMap as le,isSeq as j}from"yaml";import{v4 as H}from"uuid";function Nt(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,statements:(e.statements??[]).map(F)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(F)),r}function he(e,t){if(e.testGroup)return Ft(e,t);let r=Nt(e,t),o=new bo(r);return e.comment&&(o.commentBefore=e.comment),Ot(o,e.statements??[]),e.teardown&&Ot(o,e.teardown,"teardown"),o.toString(Dt)}function Ot(e,t,r="statements"){let o=e.contents;if(!o||!le(o))return;let i=o.get(r,!0);j(i)&&de(i,t)}function de(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=t[r],i=e.items[r];if(r>0&&(i.spaceBefore=!0),o.comment&&(r===0?e.commentBefore=o.comment:i.commentBefore=o.comment),le(i)){let n=i;if(o.type==="STEP"){let a=n.get("statements",!0);j(a)&&de(a,o.statements)}else if(o.type==="IF_ELSE"){let a=n.get("THEN",!0);j(a)&&de(a,o.then);let s=n.get("ELSE",!0);j(s)&&o.else&&de(s,o.else)}else if(o.type==="WHILE_LOOP"){let a=n.get("DO",!0);j(a)&&de(a,o.body)}}}}function Ft(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let o={};t?.test_case_id!==void 0&&(o.test_case_id=t.test_case_id),t?.name&&(o.name=t.name),t?.tags&&t.tags.length>0&&(o.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(o.use=t.use);let i={};return e.baseURL&&(i.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(i.beforeAll=r.beforeAll.map(F)),r.beforeEach&&r.beforeEach.length>0&&(i.beforeEach=r.beforeEach.map(F)),r.afterEach&&r.afterEach.length>0&&(i.afterEach=r.afterEach.map(F)),r.afterAll&&r.afterAll.length>0&&(i.afterAll=r.afterAll.map(F)),i.tests=r.tests.map(n=>{let a={name:n.name};return n.skip!==void 0&&(a.skip=n.skip),n.timeout!==void 0&&(a.timeout=n.timeout),n.fail!==void 0&&(a.fail=n.fail),n.only!==void 0&&(a.only=n.only),n.slow!==void 0&&(a.slow=n.slow),a.statements=n.statements.map(F),n.teardown&&n.teardown.length>0&&(a.teardown=n.teardown.map(F)),a}),o.suite=i,mo(o,Dt)}function F(e){switch(e.type){case"DRAFT":return wo(e);case"ACTION":return So(e);case"STEP":return vo(e);case"IF_ELSE":return _o(e);case"WHILE_LOOP":return xo(e)}}function wo(e){return{intent:e.description}}function So(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let s=r?.statement;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"?{VERIFY:s,js:c}:{VERIFY:s}}}if(t==="go_to_url"){let s=r?.url;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:s};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let s=r?.code;if(typeof s=="string"&&e.description)return{intent:e.description,js:s}}if(t==="ai_wait_until"){let s=r?.condition;if(typeof s=="string"){let c={WAIT_UNTIL:s};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let s=r?.seconds,l={WAIT:e.description||`Wait ${s}s`};return typeof s=="number"&&(l.seconds=s),l}if(t==="js_code"){let s=r?.code;if(typeof s=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{CODE:s}}if(!e.action_entity)return{intent:e.description};let o=e.action_entity.action_data??e.action_entity.action;if(!o)return{intent:e.description};let i={intent:e.description,action:o.action_name},n=e.locator??e.action_entity.locator;n&&(i.locator=n);let a=e.action_entity.xpath;if(a&&(i.xpath=a),e.use_pure_vision&&(i.use_pure_vision=!0),o.kwargs&&Object.keys(o.kwargs).length>0)for(let[s,c]of Object.entries(o.kwargs))i[s]=c;return o.args&&o.args.length>0&&(i.args=o.args),i}function vo(e){let t={STEP:e.description,statements:e.statements.map(F)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function _o(e){let t={IF:Ut(e.condition),THEN:e.then.map(F)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(F)),t}function xo(e){let t={WHILE:Ut(e.condition),DO:e.body.map(F)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Ut(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function fe(e){try{let t=Rt(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),r}catch{return{}}}function He(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(He);let t=e,r=Object.keys(t);if(r.length===1){let i=r[0];if(i.startsWith("{ ")&&i.endsWith(" }")&&t[i]===null)return`{{${i.slice(2,-2)}}}`}let o={};for(let[i,n]of Object.entries(t))o[i]=He(n);return o}function O(e){if(e.length>Lt)throw new Error(`YAML input too large (${e.length} bytes, max ${Lt})`);let t=He(Rt(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return To(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:U(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=U(t.teardown));let o=We.safeParse(r);if(!o.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(o.error.errors)}`);let i=o.data;return we(e,i),i}function To(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let i={tests:r.map(s=>{if(!s.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(s.statements)||s.statements.length===0)throw new Error(`Suite test "${s.name}" must have a non-empty "statements" array`);let c={name:s.name,statements:U(s.statements)};return Array.isArray(s.teardown)&&s.teardown.length>0&&(c.teardown=U(s.teardown)),s.skip!==void 0&&(c.skip=s.skip),typeof s.timeout=="number"&&(c.timeout=s.timeout),s.fail!==void 0&&(c.fail=s.fail),s.only===!0&&(c.only=!0),s.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(i.beforeAll=U(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(i.afterAll=U(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(i.beforeEach=U(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(i.afterEach=U(t.afterEach));let n=be.safeParse(i);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function U(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(ko)}function ko(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 Ao(t);if("WHILE"in t)return $o(t);if("STEP"in t)return Eo(t);if("VERIFY"in t){let r=t.VERIFY,o={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(o.code=t.js),{uid:H(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:o}}}}if("URL"in t){let r=t.URL,o=t.new_tab===!0?!0:void 0,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return o&&(n.new_tab=!0),i!==void 0&&(n.timeout_seconds=i),{uid:H(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:H(),type:"ACTION",description:`Wait until: ${r}`,action_entity:{action_description:`Wait until: ${r}`,action_data:{action_name:"ai_wait_until",kwargs:{condition:typeof r=="string"?r:String(r),timeout_seconds:o}}}}}if("WAIT"in t){let r=t.WAIT,o=typeof t.seconds=="number"?t.seconds:3;return{uid:H(),type:"ACTION",description:typeof r=="string"?r:`Wait ${o}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${o}s`,action_data:{action_name:"wait",kwargs:{seconds:o}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:H(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,o=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:H(),type:"ACTION",description:o,action_entity:{action_description:o,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...o}=t;return Ct({...o,action:"function",functionName:r})}if("action"in t)return Ct(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:H(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Bt(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 Ao(e){let t=Bt(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let o={uid:H(),type:"IF_ELSE",condition:t,then:U(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(o.else=U(e.ELSE)),o}function $o(e){let t=Bt(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let o={uid:H(),type:"WHILE_LOOP",condition:t,body:U(r)};return typeof e.timeout_ms=="number"&&(o.timeout_ms=e.timeout_ms),o}function Eo(e){let t=typeof e.STEP=="string"?e.STEP:"";if(!Array.isArray(e.statements))throw new Error("STEP requires a statements array");let r={uid:H(),type:"STEP",description:t,statements:U(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}function Ct(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",o=typeof e.locator=="string"?e.locator:void 0,i=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,a={};for(let[l,p]of Object.entries(e))Po.has(l)||(a[l]=p);let s={action_description:r,action_data:{action_name:t,kwargs:Object.keys(a).length>0?a:{}}};o&&(s.locator=o),i&&(s.xpath=i);let c={uid:H(),type:"ACTION",description:r,action_entity:s};return n&&(c.use_pure_vision=!0),c}function we(e,t){let r;try{r=yo(e)}catch{return}let o=r.contents;if(!o||!le(o))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=o.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let i=o,n=i.get("statements",!0);j(n)&&t.statements&&ce(n,t.statements);let a=i.get("teardown",!0);j(a)&&t.teardown&&ce(a,t.teardown)}function ce(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let o=e.items[r];o.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=o.commentBefore);let i=t[r];if(i.type==="STEP"&&le(o)){let n=o.get("statements",!0);j(n)&&ce(n,i.statements)}else if(i.type==="IF_ELSE"&&le(o)){let n=o.get("THEN",!0);j(n)&&ce(n,i.then);let a=o.get("ELSE",!0);j(a)&&i.else&&ce(a,i.else)}else if(i.type==="WHILE_LOOP"&&le(o)){let n=o.get("DO",!0);j(n)&&ce(n,i.body)}}}var Dt,Lt,Po,Se=_(()=>{"use strict";Ge();Dt={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};Lt=1024*1024;Po=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as xn,stringify as Tn}from"yaml";var Wt=_(()=>{"use strict";Se()});var je,ve,_e=_(()=>{"use strict";je=e=>{let t=[];switch(e.type){case"STEP":e.statements&&t.push({key:"statements",statements:e.statements});break;case"IF_ELSE":e.then&&t.push({key:"then",statements:e.then}),e.else&&t.push({key:"else",statements:e.else});break;case"WHILE_LOOP":e.body&&t.push({key:"body",statements:e.body});break}return t},ve=e=>{let t=[],r=o=>{for(let i of o){t.push(i);let n=je(i);for(let a of n)r(a.statements)}};return r(e),t}});function jt(e){let t=0,r=0;for(let o of e)if(o.type==="DRAFT")r++;else if(o.type==="ACTION"){let i=o.action_entity?.action_data?.action_name??"";Ht.has(i)||t++}return{action:t,draft:r}}function Io(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Gt(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function Ke(e,t){let r=t?.coverageThreshold??Mo,o=[],i=[],n;try{n=O(e)}catch(g){return{valid:!1,errors:[`Invalid YAML: ${g.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}n.goal||o.push('Missing required field: "goal"'),n.statements?.length||o.push('Missing required field: "statements"');let a=[...ve(n.statements??[]),...n.teardown?ve(n.teardown):[]],{action:s,draft:c}=jt(a),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let g of a){if(g.type==="ACTION"){let f=g,u=f.action_entity?.action_data?.action_name??"";if(u==="js_code"||u==="js_action"||u==="verify"||u==="ai_assert"){let d=f.action_entity?.action_data?.kwargs?.code;if(typeof d=="string"){let w=Io(d);if(w){let b=f.description||u;o.push(`Invalid JS in "${b}": ${w}. ${l}`)}}}}if(g.type==="IF_ELSE"){let f=g;if(f.condition.type==="JS_CODE"){let u=Gt(f.condition.expression);u&&o.push(`Invalid JS in IF condition "${f.condition.expression}": ${u}. ${l}`)}}if(g.type==="WHILE_LOOP"){let f=g;if(f.condition.type==="JS_CODE"){let u=Gt(f.condition.expression);u&&o.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${u}. ${l}`)}}}let p=s+c,h=p>0?Math.round(s/p*100):0;return p>0&&h/100<r&&i.push(`Low action coverage: ${s}/${p} statements (${h}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:o.length===0,errors:o,warnings:i,stats:{total:p,action:s,draft:c,coverage:h}}}var Mo,Ht,Kt=_(()=>{"use strict";Se();_e();Mo=.5,Ht=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var xe=_(()=>{"use strict"});var zt=_(()=>{"use strict";xe()});var Vt,Lo,Yt,Jt,ge,Co,Ro,Xt=_(()=>{"use strict";Vt=112,Lo=1080-Vt,Yt={"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"}},Jt={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"]},ge=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Jt[e].map(o=>Yt[o]).filter(o=>o.defaultBrowserType&&r.includes(o.defaultBrowserType))},Co={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile")}},Ro={desktop:{label:"Desktop",type:"desktop",devices:ge("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:ge("mobile",!0)}}});var qt=_(()=>{"use strict"});function ze(){return{version:"1.0",entries:{}}}var Zt=_(()=>{"use strict";_e()});var K,Ve,Qt=_(()=>{"use strict";K=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(K||{}),Ve=18e4});var er=_(()=>{"use strict"});var tr=_(()=>{"use strict"});var rr=_(()=>{"use strict"});var or=_(()=>{"use strict";xe()});var re=_(()=>{"use strict";_t();xt();Tt();kt();Wt();Kt();Se();Ge();zt();Xt();qt();Zt();_e();Qt();er();tr();rr();or();xe()});import{stringify as Do}from"yaml";import{createHash as ti}from"crypto";import{parse as ri,stringify as hr}from"yaml";import{readFileSync as oi}from"fs";import{resolve as fr,dirname as ii}from"path";import{parse as ar,stringify as ni}from"yaml";import{readFileSync as di,writeFileSync as hi,mkdirSync as fi}from"fs";import{dirname as gi}from"path";function Z(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function $(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function Fo(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function Uo(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 ke(e){let t=Fo(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let o=Uo(e);if(o){let i=JSON.stringify(o);return`${t}.locator(${i}).first()`}return null}function ir(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:Bo.includes(t)}function Go(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:!Wo.includes(t)}function S(e,t){I.set(e,t)}function Ho(e){return I.get(e)}function ne(e,t,r=[]){let o=[...r];return t.locator?o.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&o.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&o.push(`frame_path: ${JSON.stringify(t.frame_path)}`),o.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...o.map(i=>` ${i},`),"});"]}function nr(e){let t=e.functionName;if(!t)return null;let r=Array.isArray(e.args)?e.args.map(String):[];if(r.length===0)return`await ${t}()`;let o=["page","testContext","request","agent"],i=["undefined","null","true","false"],n=r.map(a=>o.includes(a)||i.includes(a)||/^-?\d+(\.\d+)?$/.test(a)?a:a.startsWith("$")?`agent.agentServices.readVariable('${a.substring(1)}')`:`"${a}"`);return`await ${t}(${n.join(", ")})`}function J(e,t,r,o="main"){let i=[];for(let n=0;n<e.length;n++){let a=e[n],s=`${o}.${n}`,c=jo(a,t,s,r);c.length>0&&(i.push(...c),n<e.length-1&&i.push(""))}return i}function jo(e,t,r,o){let i=" ".repeat(t);switch(e.type){case"DRAFT":return Ko(e,t,r,o);case"ACTION":return zo(e,t,r,o);case"STEP":return Vo(e,t,r,o);case"IF_ELSE":return Yo(e,t,r,o);case"WHILE_LOOP":return Jo(e,t,r,o);default:return[`${i}// Unknown statement type: ${e.type}`]}}function Ko(e,t,r,o){let i=" ".repeat(t),n=e.description?.trim()||"";if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${$(n)}`,`${i}// DRAFT: ${$(n)} (requires agent - skipped in hook)`];let a=JSON.stringify(n);return[`${i}// ${r}: ${$(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.run(page, ${a}, '${r}');`]}function zo(e,t,r,o){let i=" ".repeat(t),n=e.description,a=e.uid,c=o.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!n)return[`${i}// ${r}: Skipping - no description`];if(o.noAgent)return[`${i}// ${r}: ${$(n)}`,`${i}// DRAFT: ${$(n)} (requires agent - skipped in hook)`];let y=JSON.stringify(n),A=!!e.use_pure_vision;return[`${i}// ${r}: ${$(n)}`,`${i}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.execute(page, ${y}, '${r}', ${A});`]}let l=e.locator?{...c,locator:e.locator}:c;n&&n!==l.action_description&&(l={...l,action_description:n});let p=l.action_data?.action_name||"",h=l.action_description||"",g=Ho(p);if(!g)return[`${i}// ${r}: Unknown action: ${p}`];let f={imports:o.imports},u=g(l,r,f);if(o.noAgent){if(ir(l))return[`${i}// ${r}: ${$(h)}`,`${i}// AI action: ${$(h)} (requires agent - skipped in hook)`];let y=Xo(l,p,i,r);return y||[`${i}// ${r}: ${$(h)}`,...u.map(A=>`${i}${A}`)]}if(ir(l))return[`${i}// ${r}: ${$(h)}`,`${i}page = agent.agentServices.validatePage(page);`,...u.map(y=>`${i}${y}`)];let d=JSON.stringify(h),w=u.map(y=>`${i} ${y}`),b=Go(l),v=a?`'${a}'`:"undefined";return[`${i}// ${r}: ${$(h)}`,`${i}page = agent.agentServices.validatePage(page);`,`${i}await agent.step(page, async () => {`,...w,`${i}}, ${d}, '${r}', ${v}, ${b});`]}function Vo(e,t,r,o){let i=" ".repeat(t),n=[];e.description&&e.description.trim()&&n.push(`${i}// Step: ${$(e.description)}`);let a=J(e.statements,t,o,r);return n.push(...a),n}function Yo(e,t,r,o){let i=" ".repeat(t),n=[];if(n.push(`${i}// ${r}: Conditional check`),e.condition.type==="JS_CODE")n.push(`${i}if (${e.condition.expression}) {`);else{n.push(`${i}// AI Condition: ${$(e.condition.expression)}`);let s=JSON.stringify(e.condition.expression);n.push(`${i}if (await agent.evaluate(page, ${s}, "${r}")) {`)}let a=J(e.then,t+1,o,`${r}.then`);if(n.push(...a),e.else&&e.else.length>0){n.push(`${i}} else {`);let s=J(e.else,t+1,o,`${r}.else`);n.push(...s)}return n.push(`${i}}`),n}function Jo(e,t,r,o){let i=" ".repeat(t),n=[];n.push(`${i}// ${r}: Loop`);let a=e.timeout_ms??Ve,s=a/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${s}s`:`While loop exceeded default timeout of ${s}s`,l=`loop_${r.replace(/\./g,"_")}`;if(n.push(`${i}const ${l}_start = Date.now();`),n.push(`${i}const ${l}_timeout = ${a};`),n.push(`${i}const ${l}_check = () => {`),n.push(`${i} if (Date.now() - ${l}_start > ${l}_timeout) {`),n.push(`${i} throw new Error('${c}');`),n.push(`${i} }`),n.push(`${i} return true;`),n.push(`${i}};`),e.condition.type==="JS_CODE")n.push(`${i}while (${l}_check() && (${e.condition.expression})) {`);else{n.push(`${i}// AI Loop Condition: ${$(e.condition.expression)}`);let h=JSON.stringify(e.condition.expression);n.push(`${i}while (${l}_check() && await agent.evaluate(page, ${h}, "${r}")) {`)}let p=J(e.body,t+1,o,`${r}.body`);return n.push(...p),n.push(`${i}}`),n}function Xo(e,t,r,o){let i=e.action_description||"",n=e.action_data?.kwargs||{},a=n.timeout_ms??Je;switch(t){case"go_to_url":case"open_tab":{let s=n.url||"";return[`${r}// ${o}: ${$(i)}`,`${r}await page.goto(${JSON.stringify(s)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${o}: ${$(i)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${o}: ${$(i)}`,`${r}await page.goForward();`];case"input_text":{let s=n.text||"",c=ke(e);return c?[`${r}// ${o}: ${$(i)}`,`${r}await ${c}.fill(${JSON.stringify(s)}, { timeout: ${a} });`]:null}case"select_dropdown_option":{let s=n.text||n.label||"",c=ke(e);return c?[`${r}// ${o}: ${$(i)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(s)} }, { timeout: ${a} });`]:null}default:return null}}function qo(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...ur()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...sr("beforeEach",t.beforeEach,n)),r.push(""));let a=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let s=t?.testName||e.goal||"Generated test",c=Ye(t?.tags);for(let l of t.parameters){let p=pr(e,l.values);r.push(...$e(p,`${c}${Z(s)} [${Z(l.name)}]`,n,0,a)),r.push("")}}else{let s=t?.testName||e.goal||"Generated test",c=Ye(t?.tags);r.push(...$e(e,`${c}${Z(s)}`,n,0,a))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...sr("afterEach",t.afterEach,n))),dr(r,i),r.join(`
|
|
29
|
+
`)}function Zo(e,t){let r=[],o=t?.version||"unknown";r.push(`// @generated by shiplightai v${o}`),r.push(...ur()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let i=new Set,n={imports:i,actionEntityStore:t?.actionEntityStore},a=t?.testName||"Test Suite",s=Ye(t?.tags);r.push(`test.describe.serial('${s}${Z(a)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Te("beforeAll",e.beforeAll,n,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Te("beforeEach",e.beforeEach,n,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let p=e.tests[l],h=p.timeout||p.skip!==void 0||p.fail!==void 0||p.only||p.slow?{timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}:void 0;if(p.parameters&&p.parameters.length>0)for(let g of p.parameters){let f=pr(p.testFlow,g.values);r.push(...$e(f,`${Z(p.name)} [${Z(g.name)}]`,n,1,h)),r.push("")}else r.push(...$e(p.testFlow,Z(p.name),n,1,h)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Te("afterEach",e.afterEach,n,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Te("afterAll",e.afterAll,n,1)),r.push("});"),dr(r,i),r.join(`
|
|
30
|
+
`)}function Ye(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Ae(e){let t=new Set;function r(o){for(let i of o)switch(i.type){case K.ACTION:{let a=i.action_entity?.action_data?.kwargs;if(a?.args&&Array.isArray(a.args))for(let s of a.args)typeof s=="string"&&Qo.includes(s)&&t.add(s);break}case K.STEP:r(i.statements);break;case K.IF_ELSE:{let n=i;r(n.then),n.else&&r(n.else);break}case K.WHILE_LOOP:r(i.body);break}}return r(e),t}function ei(e){let t=Ae(e.statements??[]);if(e.teardown)for(let r of Ae(e.teardown))t.add(r);return t}function Xe(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function $e(e,t,r,o=0,i){let n=" ".repeat(o),a=[],s=ei(e),c=Xe(s),l=i?.only?"test.only":"test";a.push(`${n}${l}('${t}', async (${c}) => {`),i?.skip===!0?a.push(`${n} test.skip();`):typeof i?.skip=="string"&&a.push(`${n} test.skip(true, '${Z(i.skip)}');`),i?.fail===!0?a.push(`${n} test.fail();`):typeof i?.fail=="string"&&a.push(`${n} test.fail(true, '${Z(i.fail)}');`),i?.slow&&a.push(`${n} test.slow();`),i?.timeout&&a.push(`${n} test.setTimeout(${i.timeout});`);let p=e.teardown&&e.teardown.length>0,h=o+1;if(p){if(a.push(`${n} try {`),e.statements&&e.statements.length>0){a.push(`${n} // Test steps`);let f=J(e.statements,h+1,r);a.push(...f)}a.push(`${n} } finally {`),a.push(`${n} // Teardown`);let g=J(e.teardown,h+1,r,"teardown");a.push(...g),a.push(`${n} }`)}else if(e.statements&&e.statements.length>0){a.push(`${n} // Test steps`);let g=J(e.statements,h,r);a.push(...g)}return a.push(`${n}});`),a}function sr(e,t,r){let o=[],i=lr(t),n=Ae(i),a=Xe(n);return o.push(`test.${e}(async (${a}) => {`),o.push(...J(i,1,r,e)),o.push("});"),o}function Te(e,t,r,o){let i=" ".repeat(o),n=[],a=lr(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};n.push(`${i}test.${e}(async ({ browser }, workerInfo) => {`),n.push(`${i} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),n.push(...J(a,o+1,c,e)),n.push(`${i} await page.close();`),n.push(`${i}});`)}else{let c=Ae(a),l=Xe(c);n.push(`${i}test.${e}(async (${l}) => {`),n.push(...J(a,o+1,r,e)),n.push(`${i}});`)}return n}function lr(e){let r=Do({goal:"_hook",statements:e});return O(r).statements??[]}function pr(e,t){let r=he(e);for(let[o,i]of Object.entries(t))r=r.split(`<<${o}>>`).join(String(i));return O(r)}function ur(){return["import { test, expect } from 'shiplightai/fixture';"]}function dr(e,t){if(t.size>0){let r=0;for(let i=0;i<e.length;i++)e[i].startsWith("import ")&&(r=i+1);let o=Array.from(t);e.splice(r,0,...o)}}function gr(e,t){let r={expandingPaths:new Set([fr(t)]),depth:0,referencedPaths:new Set},o={...e};Array.isArray(o.statements)&&(o.statements=oe(o.statements,t,r)),Array.isArray(o.teardown)&&(o.teardown=oe(o.teardown,t,r));for(let i of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(o[i])&&(o[i]=oe(o[i],t,r));return{doc:o,referencedTemplatePaths:Array.from(r.referencedPaths)}}function oe(e,t,r){let o=[];for(let i of e)if(si(i)){let n=ai(i,t,r);o.push(...n)}else o.push(ci(i,t,r));return o}function si(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function ai(e,t,r){if(r.depth>=cr)throw new Error(`Template expansion exceeded maximum depth of ${cr}. Check for deeply nested or circular template references.`);let o=fr(ii(t),e.template);if(r.expandingPaths.has(o))throw new Error(`Circular template reference detected: ${o} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${o}`);r.referencedPaths.add(o);let i;try{i=oi(o,"utf-8")}catch(p){throw new Error(`Failed to read template file: ${o} (referenced from ${t}): ${p.message}`)}let n=ar(i);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${o} \u2014 expected a YAML object`);let a=n.params||[],s=e.params||{};for(let p of a)if(!(p in s))throw new Error(`Template ${e.template} requires param "${p}" but it was not provided. Required params: [${a.join(", ")}]`);let c=n.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(s).length>0){let h=ni(c);for(let[g,f]of Object.entries(s))h=h.split(`<<${g}>>`).join(String(f));c=ar(h)}let l={expandingPaths:new Set([...r.expandingPaths,o]),depth:r.depth+1,referencedPaths:r.referencedPaths};return oe(c,o,l)}function ci(e,t,r){if(typeof e!="object"||e===null)return e;let o={...e};return Array.isArray(o.statements)&&(o.statements=oe(o.statements,t,r)),Array.isArray(o.THEN)&&(o.THEN=oe(o.THEN,t,r)),Array.isArray(o.ELSE)&&(o.ELSE=oe(o.ELSE,t,r)),Array.isArray(o.DO)&&(o.DO=oe(o.DO,t,r)),o}function Ze(e,t){let r=ri(e),o=r?.name,i=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new qe('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return pi(r,o,i,n,t)}return li(r,o,i,n,t)}function li(e,t,r,o,i){let n=e?.beforeEach,a=e?.afterEach,s=mr(e?.parameters),c=e?.timeout,l=e?.skip,p=e?.fail,h=e?.only,g=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new qe(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let f=[];if(i&&e&&typeof e=="object"){let w=gr(e,i);e=w.doc,f=w.referencedTemplatePaths}let u=hr(e),d=O(u);return i&&(pe(d.statements??[],i,"main"),d.teardown&&pe(d.teardown,i,"teardown")),{testFlow:d,name:t,tags:r,use:o,beforeEach:n,afterEach:a,parameters:s,timeout:c,skip:l,fail:p,only:h,slow:g,referencedTemplatePaths:f}}function pi(e,t,r,o,i){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=n.beforeAll,s=n.afterAll,c=n.beforeEach,l=n.afterEach,p=[],h=n.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 d={goal:u.name,statements:u.statements};u.teardown&&(d.teardown=u.teardown);let w=[],b=d;if(i&&typeof d=="object"){let k=gr(d,i);b=k.doc,w=k.referencedTemplatePaths,p.push(...w)}let v=hr(b),y=O(v),A=mr(u.parameters);return{testFlow:y,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}}),g=n.base_url,f=g?{...o,baseURL:g}:o;return{suite:{beforeAll:a,afterAll:s,beforeEach:c,afterEach:l,tests:h},name:t,tags:r,use:f,referencedTemplatePaths:p}}function mr(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,r)=>{if(!t.name)throw new Error(`Parameter set at index ${r} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function pe(e,t,r){for(let o=0;o<e.length;o++){let i=e[o],n=`${r}.${o}`,a=i.description||"";if(i.uid=ui(t,n,a),i.type===K.STEP)pe(i.statements,t,n);else if(i.type===K.IF_ELSE){let s=i;pe(s.then,t,`${n}.then`),s.else&&pe(s.else,t,`${n}.else`)}else i.type===K.WHILE_LOOP&&pe(i.body,t,`${n}.body`)}}function ui(e,t,r){let o=ti("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${o.slice(0,8)}-${o.slice(8,12)}-${o.slice(12,16)}-${o.slice(16,20)}-${o.slice(20,32)}`}function yr(e,t){let r;try{r=di(e,"utf-8")}catch(o){return{valid:!1,errors:[`Failed to read file: ${o.message}`],warnings:[]}}return mi(r,e,t)}function mi(e,t,r){let o=/\btemplate:\s/.test(e),i=/^suite:/m.test(e),n=o||i?null:Ke(e);if(n&&!n.valid)return{valid:!1,errors:n.errors,warnings:[],stats:n.stats};let a,s,c=[];try{let l=r?.parsed??Ze(e,t);c=l.referencedTemplatePaths;let p={version:r?.version,actionEntityStore:r?.actionEntityStore},h=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?a=Zo(l.suite,{...p,testName:l.name,tags:l.tags,use:l.use}):a=qo(l.testFlow,{...p,testName:l.name,tags:l.tags,use:h,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let g=a.split(`
|
|
31
31
|
`).filter(f=>!f.startsWith("import ")).join(`
|
|
32
|
-
`);new Function(g),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),
|
|
32
|
+
`);new Function(g),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),fi(gi(s),{recursive:!0}),hi(s,a)}catch(l){let p=l instanceof qe?"":l.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${l.message}.${p}`],warnings:[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:n?.warnings??[],stats:n?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:c}}var Je,Bo,Wo,I,Qo,cr,qe,Qe=_(()=>{"use strict";re();re();re();re();Je=5e3;Bo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],Wo=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];I=new Map;S("click",e=>{let t=ke(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Je;return[`await ${t}.click({ timeout: ${r} });`]});S("click_element",I.get("click"));S("click_element_by_index",I.get("click"));S("double_click",e=>ne("double_click",e));S("double_click_on_element",I.get("double_click"));S("right_click",e=>ne("right_click",e));S("right_click_on_element",I.get("right_click"));S("hover",e=>ne("hover",e));S("hover_element_by_index",I.get("hover"));S("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ne("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("fill",I.get("input_text"));S("clear_input",e=>ne("clear_input",e));S("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});S("send_keys",I.get("press"));S("send_keys_on_element",e=>{let t=ke(e),r=e.action_data?.kwargs?.keys||"";if(!t)return['await agent.execAction("send_keys_on_element", page, {',` action_data: { kwargs: { keys: ${JSON.stringify(r)} } },`,"});"];let o=e.action_data?.kwargs?.timeout_ms??Je;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${o} });`]});S("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ne("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});S("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});S("scroll_down",I.get("scroll"));S("scroll_up",I.get("scroll"));S("scroll_element",I.get("scroll"));S("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});S("scroll_on_element",e=>ne("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));S("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});S("open_tab",I.get("go_to_url"));S("go_back",()=>['await agent.execAction("go_back", page, {});']);S("reload_page",()=>['await agent.execAction("reload_page", page, {});']);S("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);S("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);S("verify",(e,t)=>{let r=e.action_data?.kwargs,o=typeof r?.code=="string",i=o?r?.statement||e.action_description:e.action_description||r?.statement;if(o&&i){let a=r.code.split(`
|
|
33
33
|
`),s=JSON.stringify(i);return["{ const _t = Date.now(); try {",...a.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${s}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${s}\`);`,` await agent.assert(page, ${s}, ${JSON.stringify(t||"")});`,"} }"]}return o?r.code.split(`
|
|
34
34
|
`):i?[`await agent.assert(page, ${JSON.stringify(i)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});S("ai_assert",I.get("verify"));S("assert",I.get("verify"));S("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let o=JSON.stringify(r),i=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${o}, '${t||""}', ${i});`]});S("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});S("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,o=e.action_data?.kwargs?.variable_name;if(!r||!o)return["// Skipping ai_extract: missing element_description or variable_name"];let i=JSON.stringify(r),n=JSON.stringify(o);return[`await agent.extract(page, ${i}, ${n}, '${t||""}');`]});S("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,o=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${o}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});S("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});S("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],o=t.split(`
|
|
35
35
|
`);for(let i of o)r.push(` ${i}`);return r.push("}"),r});S("function",(e,t,r)=>{let o=e.action_data?.kwargs||{},i=o.functionName;if(i&&i.includes("#")){let[a,s]=i.split("#");if(a&&s){let c=a.replace(/\.(ts|js|mjs)$/,""),l=`import { ${s} } from '${c}';`;r?.imports?.add(l);let p={...o,functionName:s},h=nr(p);return h?[h.endsWith(";")?h:`${h};`]:["// Skipping function: invalid export pattern"]}}let n=nr(o);return n?[n.endsWith(";")?n:`${n};`]:["// Skipping function: missing functionName"]});S("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});S("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],o={};return t.paths?o.paths=t.paths:t.path&&(o.path=t.path),t.use_file_input&&(o.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(o)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(i=>` ${i},`),"});"]});S("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);S("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);S("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});S("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(o=>` ${o},`),"});"]});S("done",()=>["// Done - no action needed"]);S("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
|
|
36
|
-
`):["// Skipping js_action: missing code"]});
|
|
36
|
+
`):["// Skipping js_action: missing code"]});Qo=["testContext","request"];cr=5;qe=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}}});import{Router as yi}from"express";import*as B from"fs/promises";import*as W from"path";import{stringify as bi}from"yaml";function Ee(e){if(!e||e.length===0)return[];let r=bi({goal:"_hook",statements:e});return O(r).statements??[]}async function br(e){try{let t=await B.readdir(e,{withFileTypes:!0});for(let r of t)if(!(r.name==="node_modules"||r.name.startsWith("."))&&(r.isFile()&&r.name.endsWith(".test.yaml")||r.isDirectory()&&await br(W.join(e,r.name))))return!0}catch{}return!1}function wr(e){let{initialDir:t,initialFile:r,projectRoot:o,onFileSelected:i}=e,n=yi();n.get("/api/files",async(s,c)=>{try{let l=typeof s.query.dir=="string"?s.query.dir:t,p=W.resolve(l),h=await B.readdir(p,{withFileTypes:!0}),g=[];for(let u of h)if(u.name!=="node_modules"&&!u.name.startsWith("."))if(u.isDirectory()){let d=W.join(p,u.name);await br(d)&&g.push({name:u.name,type:"directory",path:d})}else u.isFile()&&u.name.endsWith(".test.yaml")&&g.push({name:u.name,type:"file",path:W.join(p,u.name)});g.sort((u,d)=>u.type!==d.type?u.type==="directory"?-1:1:u.name.localeCompare(d.name));let f=W.dirname(p);c.json({dir:p,parent:f!==p?f:null,entries:g,initialFile:r??null,projectRoot:o??t})}catch(l){console.error("[debugger] Error listing files:",l),c.status(500).json({error:l.message})}});function a(s){if(typeof s=="string"&&s){let c=W.resolve(s);return c.endsWith(".test.yaml")?{filePath:c}:{error:"File must be a .test.yaml file"}}return r?{filePath:r}:{error:"No file specified. Pass ?file= parameter."}}return n.get("/api/test-flow",async(s,c)=>{let l=a(s.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath;try{i?.(p);let h=await B.readFile(p,"utf-8"),g=await B.stat(p),f=fe(h),u=Ze(h,p);if(u.suite){let d=u.suite,w={tests:d.tests.map(v=>({name:v.name,statements:v.testFlow.statements??[],teardown:v.testFlow.teardown,skip:v.skip,timeout:v.timeout,fail:v.fail,only:v.only,slow:v.slow})),beforeAll:Ee(d.beforeAll),afterAll:Ee(d.afterAll),beforeEach:Ee(d.beforeEach),afterEach:Ee(d.afterEach)},b={version:"1.3.0",baseURL:u.use?.baseURL,testGroup:w};c.json({isSuite:!0,testFlow:b,metadata:f,name:u.name,tags:u.tags,use:u.use,filePath:p,fileName:W.basename(p),lastModified:g.mtimeMs})}else{let d=u.testFlow;we(h,d),c.json({isSuite:!1,testFlow:d,metadata:f,name:u.name,tags:u.tags,use:u.use,filePath:p,fileName:W.basename(p),lastModified:g.mtimeMs})}}catch(h){if(h.code==="ENOENT")return c.status(404).json({error:`File not found: ${p}`});console.error("[debugger] Error loading test flow:",h),c.status(500).json({error:h.message})}}),n.put("/api/test-flow",async(s,c)=>{try{let l=a(s.query.file);if("error"in l)return c.status(400).json({error:l.error});let p=l.filePath,{testFlow:h,metadata:g}=s.body;if(!h)return c.status(400).json({error:"testFlow is required"});let f=he(h,g),u=p+".tmp";await B.writeFile(u,f,"utf-8"),await B.rename(u,p);let d=await B.stat(p);c.json({success:!0,lastModified:d.mtimeMs})}catch(l){console.error("[debugger] Error saving test flow:",l),c.status(500).json({error:l.message})}}),n}var Sr=_(()=>{"use strict";re();Qe()});import*as vr from"http";function _r(e){let{getPort:t,ensureReady:r}=e,o=null,i=0,n=5e3,a=async(s,c,l)=>{let p=t();if(p===null){if(!o&&Date.now()-i<n)return c.status(503).json({status:"error",message:"Sandbox not ready. Select a test file and try again."});o||(o=r().catch(f=>{throw i=Date.now(),o=null,f}));try{p=await o}catch(f){return c.status(503).json({status:"error",message:"Failed to start sandbox: "+f.message})}}let h=await new Promise((f,u)=>{let d=[];s.on("data",w=>d.push(w)),s.on("end",()=>f(Buffer.concat(d))),s.on("error",u),s.readableEnded&&f(Buffer.alloc(0))}),g=vr.request({hostname:"localhost",port:p,path:s.originalUrl,method:s.method,headers:{"content-type":s.headers["content-type"]||"application/json","content-length":String(h.length)},timeout:3e5},f=>{c.writeHead(f.statusCode,f.headers),f.on("data",u=>{c.write(u)}),f.on("end",()=>{c.end()}),f.on("error",()=>{c.writableEnded||c.end()})});g.on("error",f=>{c.headersSent||c.status(502).json({status:"error",message:"Inner server unavailable: "+f.message})}),g.end(h)};return a.reset=()=>{o=null},a.getPort=()=>t(),a.ensureReady=async()=>{let s=t();return s!==null?s:(o||(o=r().catch(c=>{throw i=Date.now(),o=null,c})),o)},a}var xr=_(()=>{"use strict"});var Tr={};te(Tr,{startDebuggerServer:()=>wi});import Pe from"express";import*as ie from"path";async function wi(e){let{initialDir:t,initialFile:r,projectRoot:o,port:i}=e,n=r,a=d=>{n=d,e.onFileSelected?.(d)},s=null,c=null,l=_r({getPort:()=>s,ensureReady:async()=>{console.error("[debugger] Initializing sandbox on first use...");let d=await e.innerServerFactory();return s=d.port,c=d.cleanup,d.port}}),p=Pe();p.use((d,w,b)=>{if(w.setHeader("Access-Control-Allow-Origin","*"),w.setHeader("Access-Control-Allow-Methods","GET, POST, PUT, DELETE, OPTIONS"),w.setHeader("Access-Control-Allow-Headers","Content-Type, Accept, Cache-Control, Idempotency-Key"),d.method==="OPTIONS")return w.sendStatus(204);b()}),p.post("/api/int-runner/create-session",Pe.json(),async(d,w)=>{n&&(d.body.testFilePath=n);try{let b=l.getPort();b||(b=await l.ensureReady());let v=await fetch(`http://localhost:${b}/api/int-runner/create-session`,{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify(d.body)}),y=await v.json();w.status(v.status).json(y)}catch(b){w.status(502).json({status:"error",message:b.message})}}),p.post("/api/int-runner/terminate-session",async(d,w)=>{let b=c;s=null,c=null,l.reset(),b&&(console.error("[debugger] Tearing down inner Playwright process for fresh restart..."),await b().catch(v=>console.error("[debugger] Cleanup error:",v.message))),w.json({status:"success",details:"Session terminated"})}),p.use("/api/int-runner",l),p.use("/api/browser-cdp",l),p.use(Pe.json({limit:"10mb"})),p.use(wr({initialDir:t,initialFile:r,projectRoot:o,onFileSelected:a}));let h=typeof import.meta.dirname=="string"?import.meta.dirname:__dirname,g=h.includes(ie.sep+"src"+ie.sep)?ie.resolve(h,"../../dist/static"):ie.join(h,"static");p.use(Pe.static(g)),p.get("*path",(d,w,b)=>{if(d.path.startsWith("/api/"))return b();w.sendFile(ie.join(g,"index.html"),v=>{v&&w.send(Si(t,i))})});let f=await new Promise((d,w)=>{let b=p.listen(i,"localhost",()=>{d(b)});b.on("error",w)}),u=`http://localhost:${i}`;return console.error(`[debugger] Server running at ${u}`),console.error(`[debugger] Directory: ${t}`),r&&console.error(`[debugger] File: ${r}`),{url:u,close:async()=>{c&&await c(),await new Promise((d,w)=>{f.close(b=>b?w(b):d())})}}}function Si(e,t){return`<!DOCTYPE html>
|
|
37
37
|
<html>
|
|
38
38
|
<head>
|
|
39
39
|
<meta charset="utf-8">
|
|
@@ -51,14 +51,14 @@ test('__shiplight_debug__', async ({ page, agent }) => {
|
|
|
51
51
|
<p>Directory: <code>${e}</code></p>
|
|
52
52
|
<p>Server: localhost:${t}</p>
|
|
53
53
|
</body>
|
|
54
|
-
</html>`}var kr=_(()=>{"use strict";Sr();xr()});var Ar={};te(Ar,{startDebugger:()=>
|
|
54
|
+
</html>`}var kr=_(()=>{"use strict";Sr();xr()});var Ar={};te(Ar,{startDebugger:()=>_i});import*as z from"fs";import*as Q from"path";async function _i(e){let t,r=6174,o,i=!1,n=!0;for(let y=0;y<e.length;y++)e[y]==="--port"&&e[y+1]?(r=parseInt(e[y+1],10),y++):e[y]==="--url"&&e[y+1]?(o=e[y+1],y++):e[y]==="--new"?i=!0:e[y]==="--open"?n=!1:e[y]==="--no-open"?n=!0:e[y]==="--help"||e[y]==="-h"?(xi(),process.exit(0)):e[y].startsWith("--")||(t=e[y]);let a,s;if(t){let y=Q.resolve(t);z.existsSync(y)&&z.statSync(y).isDirectory()?a=y:(a=Q.dirname(y),s=y)}else a=process.cwd();if(i&&s&&!z.existsSync(s)){let y=Q.dirname(s);z.existsSync(y)||z.mkdirSync(y,{recursive:!0}),z.writeFileSync(s,vi(o||"https://example.com"),"utf-8"),console.log(`Created new test file: ${s}`)}s&&!z.existsSync(s)&&(console.error(`Error: File not found: ${s}`),console.error("Hint: Use --new to create a new test file."),process.exit(1));let{findPlaywrightConfig:c}=await Promise.resolve().then(()=>(Ue(),Fe)),l=c(a),p=l?Q.dirname(l):a;(await import("dotenv")).config({path:Q.join(p,".env"),override:!0});let g=s;l||(console.error("Error: No Playwright config found in "+a),console.error("A Playwright config (playwright.config.ts) is required for the debugger."),process.exit(1));let f=async()=>{let{spawnPlaywrightProcess:y}=await Promise.resolve().then(()=>(Ue(),Fe));console.error("[debugger] Starting Playwright sandbox...");let A=g||Q.join(a,"_debug.test.yaml");return y({yamlFilePath:A,configPath:l})},u=y=>{g=y},{startDebuggerServer:d}=await Promise.resolve().then(()=>(kr(),Tr));console.log(s?`Starting debugger for: ${s}`:`Starting debugger in: ${a}`),l&&console.log(`Using Playwright config: ${l}`);let w;try{w=await d({initialDir:a,initialFile:s,projectRoot:p,port:r,innerServerFactory:f,onFileSelected:u})}catch(y){throw y?.code==="EADDRINUSE"&&(console.error(`Error: Port ${r} is already in use.`),console.error("Use --port <number> to specify a different port."),process.exit(1)),y}if(bt(r,a),console.log(`Debugger running at: ${w.url}`),console.log(""),console.log("Press Ctrl+C to stop."),!n)try{let{default:y}=await import("open");await y(w.url)}catch{}let b=!1,v=async()=>{b&&(console.log(`
|
|
55
55
|
Force exiting...`),process.exit(1)),b=!0,console.log(`
|
|
56
|
-
Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{wt(r,a),await w.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function
|
|
56
|
+
Shutting down...`);let y=setTimeout(()=>{console.error("Cleanup timed out, force exiting."),process.exit(1)},5e3);try{wt(r,a),await w.close()}catch{}clearTimeout(y),process.exit(0)};process.on("SIGINT",v),process.on("SIGTERM",v)}function xi(){console.log("Usage: shiplight debug [file-or-dir] [options]"),console.log(""),console.log("Arguments:"),console.log(" file-or-dir YAML test file or directory (default: cwd)"),console.log(""),console.log("Options:"),console.log(" --port <number> Server port (default: 6174)"),console.log(" --url <url> Override starting URL for the test"),console.log(" --new Create a new test file if it doesn't exist"),console.log(" --open Auto-open the debugger in your browser"),console.log(" --no-open Don't auto-open the browser (default)"),console.log(" -h, --help Show this help message"),console.log(""),console.log("Examples:"),console.log(" shiplight debug # browse cwd"),console.log(" shiplight debug tests/ # browse tests/ directory"),console.log(" shiplight debug tests/login.test.yaml # open specific file"),console.log(" shiplight debug tests/login.test.yaml --port 8080"),console.log(" shiplight debug tests/checkout.test.yaml --new --url https://myapp.com/checkout")}var vi,$r=_(()=>{"use strict";St();vi=e=>`goal: New test
|
|
57
57
|
statements:
|
|
58
58
|
- URL: ${e}
|
|
59
59
|
- "TODO: Add your first step"
|
|
60
|
-
`});var et={};te(et,{lookupActionStores:()=>
|
|
61
|
-
`));let i=e.includes("--magic"),a=e.filter(h=>h!=="--magic").map(h=>h.endsWith(".test.yaml")?h.replace(/\.test\.yaml$/,".yaml.spec.ts"):h),s=Pr(t);await
|
|
60
|
+
`});var et={};te(et,{lookupActionStores:()=>Ti,updateActionStores:()=>ki});function Er(){let e=process.env.SHIPLIGHT_API_TOKEN;return e?{apiUrl:process.env.SHIPLIGHT_API_URL||"https://api.shiplight.ai",apiToken:e}:null}async function Ti(e){let t=Er();if(!t||e.length===0)return new Map;try{let r=new AbortController,o=setTimeout(()=>r.abort(),2e3),i=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(o),!i.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${i.status}`),new Map;let n=await i.json(),a=new Map;for(let[s,c]of Object.entries(n.stores??{}))a.set(s,c);return a}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function ki(e){let t=Er();if(!t||e.size===0)return 0;try{let r=new AbortController,o=setTimeout(()=>r.abort(),5e3),i={};for(let[s,c]of e)i[s]=c;let n=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:i}),signal:r.signal});return clearTimeout(o),n.ok?(await n.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${n.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var tt=_(()=>{"use strict"});import*as R from"fs";import*as me from"path";import{globSync as Ai}from"glob";function Pr(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new ot:new rt(e)}function ye(e){return e.replace(/\//g,"__")+".json"}function Ei(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var $i,rt,ot,Mr=_(()=>{"use strict";re();$i=".shiplight/action-cache";rt=class{constructor(t){this.cwd=t;this.cacheDir=me.join(t,$i)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!R.existsSync(this.cacheDir))return r;for(let o of t){let i=me.join(this.cacheDir,ye(o));try{if(R.existsSync(i)){let n=R.readFileSync(i,"utf-8");r.set(o,JSON.parse(n))}}catch{}}return r}async update(t){if(t.size===0)return 0;R.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[o,i]of t)try{let n=me.join(this.cacheDir,ye(o)),a=ze();if(R.existsSync(n))try{a=JSON.parse(R.readFileSync(n,"utf-8"))}catch{}let s={...a,entries:{...a.entries,...i.entries}};R.writeFileSync(n,JSON.stringify(s,null,2)),r++}catch{}return r}loadAll(){if(!R.existsSync(this.cacheDir))return;let t=Ai("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,o=0;for(let i of t)try{let n=R.readFileSync(me.join(this.cacheDir,i),"utf-8"),a=JSON.parse(n),s=Ei(i);r.set(s,a),o+=Object.keys(a.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${o} cached action entit${o===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},ot=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(tt(),et));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(tt(),et));return r(t)}loadAll(){}}});var Cr={};te(Cr,{buildTestPathsFromGitInfo:()=>Lr,cloudKeyToCwdRelPath:()=>st,runTests:()=>Mi});import{spawn as Pi,execFileSync as Ir}from"child_process";import*as N from"fs";import*as L from"path";import{globSync as it}from"glob";async function Mi(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(h=>N.existsSync(L.join(t,h)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
|
|
61
|
+
`));let i=e.includes("--magic"),a=e.filter(h=>h!=="--magic").map(h=>h.endsWith(".test.yaml")?h.replace(/\.test\.yaml$/,".yaml.spec.ts"):h),s=Pr(t);await Oi(t,s);let c={...process.env};i&&(c.SHIPLIGHT_MAGIC="1");let l=Pi("npx",["playwright","test",...a],{stdio:"inherit",shell:!0,cwd:t,env:c}),p=await new Promise(h=>{l.on("close",g=>h(g??1))});await Li(t,s),process.exit(p)}function nt(){try{return Ir("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function Ii(){try{let e=Ir("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Or(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let o=Ii(),i=nt();return Lr(e,t,o,i)}function Lr(e,t,r,o){return{testPaths:e.map(n=>{let a=o?L.relative(o,L.resolve(t,n)):n;return`${r}${a}`}),branchPrefix:r}}function st(e,t,r,o){let i=t&&e.startsWith(t)?e.slice(t.length):e;return r?L.relative(o,L.resolve(r,i)):i}async function Oi(e,t){try{let r=it("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:o,branchPrefix:i}=Or(r,e,t.isCloud),n=await t.lookup(o);if(n.size===0)return;let a=L.join(e,".shiplight","action-cache");N.mkdirSync(a,{recursive:!0});let s=nt(),c=0;for(let[l,p]of n){let h=t.isCloud?st(l,i,s,e):l,g=L.join(a,ye(h));N.writeFileSync(g,JSON.stringify(p,null,2)),c++}console.log(`[shiplight] Cache: downloaded ${c} action store${c!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache download failed:",r.message)}}async function Li(e,t){try{let r=L.join(e,"test-results");if(!N.existsSync(r))return;let o=it("**/new-action-entities.json",{cwd:r});if(o.length===0)return;let i={};for(let h of o)try{let g=N.readFileSync(L.join(r,h),"utf-8"),f=JSON.parse(g);f?.entries&&Object.assign(i,f.entries)}catch{}if(Object.keys(i).length===0)return;let n=it("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:a,branchPrefix:s}=Or(n,e,t.isCloud),c=new Map;for(let h=0;h<n.length;h++){let g=n[h],f=a[h],u=L.join(e,g.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!N.existsSync(u))continue;let d=N.readFileSync(u,"utf-8"),w={};for(let[b,v]of Object.entries(i))d.includes(b)&&(w[b]=v);if(Object.keys(w).length>0){let b=t.isCloud?st(f,s,nt(),e):f,v=L.join(e,".shiplight","action-cache",ye(b)),y={};if(N.existsSync(v))try{y=JSON.parse(N.readFileSync(v,"utf-8")).entries}catch{}c.set(f,{version:"1.0",entries:{...y,...w}})}}if(c.size===0)return;let l=await t.update(c),p=Array.from(c.values()).reduce((h,g)=>h+Object.keys(g.entries).length,0);console.log(`[shiplight] Cache: saved ${p} action entit${p!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var Rr=_(()=>{"use strict";Mr()});function E(e){return e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""").replace(/'/g,"'")}function Ie(e){if(e<1e3)return`${e}ms`;if(e<6e4)return`${(e/1e3).toFixed(1)}s`;let t=Math.floor(e/6e4),r=(e%6e4/1e3).toFixed(0);return`${t}m ${r}s`}function Oe(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function Me(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">✔</span>';case"flaky":return'<span class="status-icon flaky">↻</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">✘</span>';case"skipped":return'<span class="status-icon skipped">─</span>';case"interrupted":return'<span class="status-icon failed">⚠</span>';default:return'<span class="status-icon pending">○</span>'}}function Ci(e){return`<span class="badge badge-${e}">${e}</span>`}function Ri(e){let t=e.duration!=null?`<span class="step-duration">${Ie(e.duration)}</span>`:"",r=Me(e.status);if(e.screenshot||e.message||e.error){let i="";e.screenshot&&(i=`<img src="${E(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n="";e.error&&(n=`<div class="step-error"><pre>${E(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${E(e.message)}</div>`);let s=e.status==="failure"?" open":"";return`
|
|
62
62
|
<details class="step-details step step-${e.status}"${s}>
|
|
63
63
|
<summary class="step-header">
|
|
64
64
|
${r}
|
|
@@ -80,7 +80,7 @@ statements:
|
|
|
80
80
|
<span class="step-description">${E(e.description)}</span>
|
|
81
81
|
${t}
|
|
82
82
|
</div>
|
|
83
|
-
</div>`}function
|
|
83
|
+
</div>`}function Ni(e,t,r,o){let i=[];e&&i.push(`
|
|
84
84
|
<details class="artifact-section">
|
|
85
85
|
<summary class="artifact-summary">Video</summary>
|
|
86
86
|
<div class="artifact-content">
|
|
@@ -112,13 +112,13 @@ statements:
|
|
|
112
112
|
<p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
|
|
113
113
|
<p class="trace-hint"><a href="${a}" class="attachment-link" download>Download trace.zip</a></p>
|
|
114
114
|
</div>
|
|
115
|
-
</details>`)}return i.length===0?"":`<div class="test-artifacts">${i.join("")}</div>`}function Nr(e,t,r,o,i){let n=e.map(
|
|
116
|
-
`),a="";t&&!e.some(c=>c.error)&&(a=`<div class="test-error"><pre>${E(t)}</pre></div>`);let s=
|
|
115
|
+
</details>`)}return i.length===0?"":`<div class="test-artifacts">${i.join("")}</div>`}function Nr(e,t,r,o,i){let n=e.map(Ri).join(`
|
|
116
|
+
`),a="";t&&!e.some(c=>c.error)&&(a=`<div class="test-error"><pre>${E(t)}</pre></div>`);let s=Ni(r,o,e,i);return`
|
|
117
117
|
${a}
|
|
118
118
|
<div class="steps-list">
|
|
119
119
|
${n||'<div class="no-steps">No YAML step details available</div>'}
|
|
120
120
|
</div>
|
|
121
|
-
${s}`}function
|
|
121
|
+
${s}`}function Di(e,t){let r=e.flaky?"flaky":e.status,o=Me(r),i;if(e.flaky&&e.attempts&&e.attempts.length>1){let n=`tabs-${t}`,a=e.attempts.length,s=e.attempts.map((l,p)=>{let h=p===a-1,g=l.status==="passed"?"passed":"failed",f=`Attempt ${l.attemptNumber}`;return`<button class="attempt-tab ${h?"active":""} attempt-tab-${g}"
|
|
122
122
|
onclick="switchAttemptTab('${n}', ${p})"
|
|
123
123
|
data-tab-index="${p}">${Me(g)} ${f} <span class="attempt-tab-badge badge-${g}">${l.status}</span></button>`}).join(""),c=e.attempts.map((l,p)=>{let h=p===a-1,g=Nr(l.steps,l.error,l.videoPath,l.tracePath,`${t}-attempt-${p}`);return`<div class="attempt-panel ${h?"active":""}" data-panel-index="${p}">
|
|
124
124
|
<div class="attempt-meta">
|
|
@@ -139,13 +139,13 @@ statements:
|
|
|
139
139
|
${o}
|
|
140
140
|
<span class="test-title">${E(e.title)}</span>
|
|
141
141
|
<span class="test-file">${E(e.file)}</span>
|
|
142
|
-
${
|
|
142
|
+
${Ci(r)}
|
|
143
143
|
<span class="test-duration">${Ie(e.duration)}</span>
|
|
144
144
|
</summary>
|
|
145
145
|
<div class="test-body">
|
|
146
146
|
${i}
|
|
147
147
|
</div>
|
|
148
|
-
</details>`}function at(e){let t=e.tests.filter(s=>s.flaky).length,r=e.tests.filter(s=>s.status==="passed"&&!s.flaky).length,o=e.tests.filter(s=>s.status==="failed"||s.status==="timedOut").length,i=e.tests.filter(s=>s.status==="skipped").length,n=e.tests.length,a=e.tests.map((s,c)=>
|
|
148
|
+
</details>`}function at(e){let t=e.tests.filter(s=>s.flaky).length,r=e.tests.filter(s=>s.status==="passed"&&!s.flaky).length,o=e.tests.filter(s=>s.status==="failed"||s.status==="timedOut").length,i=e.tests.filter(s=>s.status==="skipped").length,n=e.tests.length,a=e.tests.map((s,c)=>Di(s,c)).join(`
|
|
149
149
|
`);return`<!DOCTYPE html>
|
|
150
150
|
<html lang="en">
|
|
151
151
|
<head>
|
|
@@ -895,9 +895,9 @@ statements:
|
|
|
895
895
|
});
|
|
896
896
|
</script>
|
|
897
897
|
</body>
|
|
898
|
-
</html>`}var Dr=_(()=>{"use strict"});import*as V from"fs";import*as X from"path";import{execFileSync as
|
|
899
|
-
Shiplight cloud report: ${u.data.reportUrl}`)}var Wr=_(()=>{"use strict"});var
|
|
900
|
-
Merged ${c.length} tests from ${p} shards into: ${g}`),
|
|
898
|
+
</html>`}var Dr=_(()=>{"use strict"});import*as V from"fs";import*as X from"path";import{execFileSync as Fi}from"child_process";import{createHash as Ui}from"crypto";import ue from"axios";function Fr(e){let t=e.match(/^git@([^:]+):(.+?)(?:\.git)?$/);if(t)return`https://${t[1]}/${t[2]}`;let r=e.match(/^https?:\/\/([^/]+)\/(.+?)(?:\.git)?$/);if(r)return`https://${r[1]}/${r[2]}`}function ee(...e){try{return Fi("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Bi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=V.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",r=process.env.GITHUB_REPOSITORY??"",o=process.env.GITHUB_RUN_ID??"",i=process.env.GITHUB_EVENT_NAME??"",a=Bi().pull_request,s=a?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??s??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],h=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,g=process.env.SHIPLIGHT_PR_TITLE??a?.title,f=ee("log","-1","--pretty=%s"),u=ee("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Actions",gitCommit:c,gitBranch:h,gitRepo:r,commitMessage:f,authorEmail:u,prNumber:p,prTitle:g,prUrl:p&&r?`${t}/${r}/pull/${p}`:void 0,ciBuildId:o,ciBuildUrl:o&&r?`${t}/${r}/actions/runs/${o}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:i,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",o=process.env.CI_MERGE_REQUEST_IID,i=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:i,prNumber:o,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:o&&t?`${t}/-/merge_requests/${o}`:void 0,ciBuildId:process.env.CI_PIPELINE_ID,ciBuildUrl:process.env.CI_PIPELINE_URL,commitUrl:r&&t?`${t}/commit/${r}`:void 0,triggeredBy:process.env.GITLAB_USER_LOGIN})}else if(process.env.CIRCLECI){let t=process.env.CIRCLE_SHA1??"",r=process.env.CIRCLE_PROJECT_USERNAME??"",o=process.env.CIRCLE_PROJECT_REPONAME??"",i=r&&o?`${r}/${o}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,a=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],s=process.env.CIRCLE_REPOSITORY_URL,c=s?Fr(s):void 0,l=ee("log","-1","--pretty=%s"),p=ee("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:i,commitMessage:l,authorEmail:p,prNumber:a,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=ee("rev-parse","HEAD"),r=ee("rev-parse","--abbrev-ref","HEAD"),o=ee("log","-1","--pretty=%s"),i=ee("log","-1","--pretty=%ae"),n=ee("remote","get-url","origin"),a=n?Fr(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:o,authorEmail:i,commitUrl:a&&t?`${a}/commit/${t}`:void 0})}return e}function Gi(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 Ur(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Hi(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 ji(e,t){let r=new Map;for(let o of t){o.screenshotS3Uris={};let i=r.get(o.testCaseName);i?i.push(o):r.set(o.testCaseName,[o])}return e.map(o=>r.get(o.title)?.shift())}function Le(e,t){return X.isAbsolute(t)?t:X.join(e,t)}function Ki(e,t,r,o){let i={};for(let n of e.steps)i[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:Ur(e.status),flaky:!1,segments:[{outcome:Ur(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:i,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:o,actionStepsMap:e.actionStepsMap??{}}]}}function pt(e){return Ui("md5").update(e).digest("base64")}function ct(e){return pt(V.readFileSync(e))}async function lt(e,t){let r=V.readFileSync(t),o=X.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[o]??"application/octet-stream";await ue.put(e,r,{headers:{"Content-Type":n,"Content-MD5":pt(r)}})}async function Br(e,t,r,o){let i=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${o}`,"Content-Type":"application/json"},a=Wi(),s=e.tests.map(d=>{let w={testCaseName:d.title,testCaseBaseName:d.baseTitle,suiteName:d.suiteName,file:d.file,tags:d.tags,suiteTags:d.suiteTags,baseUrl:d.baseUrl,skip:d.skip,slow:d.slow,timeout:d.timeout,parameterSetName:d.parameterSetName,flaky:d.flaky,retries:d.retries};if(d.videoPath){let b=Le(t,d.videoPath);V.existsSync(b)&&(w.videoMd5=ct(b))}if(d.tracePath){let b=Le(t,d.tracePath);V.existsSync(b)&&(w.traceMd5=ct(b))}return w}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let p=(await ue.post(`${i}/v1/local-runs`,{trigger:a.ciProvider,startTime:r,metadata:a,tests:s},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${p.testRunId})`);let h=ji(e.tests,p.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(d,w)=>{let b=h[w];if(!b)return;let v=d.steps.filter(k=>k.screenshot);if(!v.length)return;let y=v.map(k=>k.stepId),A={};for(let k of v){let P=X.isAbsolute(k.screenshot)?k.screenshot:X.join(t,k.screenshot);V.existsSync(P)&&(A[k.stepId]=ct(P))}try{let k=await ue.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:A},{headers:n});b.uploadUrls.screenshots=k.data.screenshots,b.screenshotS3Uris=k.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${d.title}"`)}catch(k){console.warn(`[reporter] Failed to get screenshot URLs for "${d.title}":`,k)}})),console.log("[reporter] [3/4] Uploading assets...");let g=(await Promise.all(e.tests.map(async(d,w)=>{let b=h[w];if(!b){console.warn(`[reporter] No result slot found for test "${d.title}", skipping.`);return}let v=b.uploadUrls,y={},A=0;await Promise.all(d.steps.map(async M=>{if(M.screenshot&&v.screenshots?.[M.stepId]){let se=X.isAbsolute(M.screenshot)?M.screenshot:X.join(t,M.screenshot);if(V.existsSync(se))try{await lt(v.screenshots[M.stepId],se),y[M.stepId]=b.screenshotS3Uris[M.stepId],A++}catch(no){console.warn(`[reporter] Screenshot upload failed for step ${M.stepId}:`,no)}}})),A>0&&console.log(`[reporter] [3/4] Uploaded ${A} screenshot(s) for "${d.title}"`);let k;if(d.videoPath&&v.video){let M=Le(t,d.videoPath);if(V.existsSync(M)){console.log(`[reporter] [3/4] Uploading video for "${d.title}"...`);try{await lt(v.video,M),k=b.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${d.title}"`)}catch(se){console.warn("[reporter] Video upload failed:",se)}}}let P;if(d.tracePath&&v.trace){let M=Le(t,d.tracePath);if(V.existsSync(M)){console.log(`[reporter] [3/4] Uploading trace for "${d.title}"...`);try{await lt(v.trace,M),P=b.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${d.title}"`)}catch(se){console.warn("[reporter] Trace upload failed:",se)}}}console.log(`[reporter] [3/4] Uploading report for "${d.title}"...`);let De=Ki(d,y,k,P),dt=Buffer.from(JSON.stringify(De)),ht=pt(dt),ft=await ue.post(`${i}/v1/local-runs/${p.testRunId}/results/${b.testCaseResultId}/report-url`,{md5:ht},{headers:n}),oo=ft.data.reportUrl,io=ft.data.reportS3Uri;return await ue.put(oo,dt,{headers:{"Content-Type":"application/json","Content-MD5":ht}}),console.log(`[reporter] [3/4] Report uploaded for "${d.title}"`),{testCaseResultId:b.testCaseResultId,result:Gi(d.status),durationMs:d.duration,startTime:d.startTime,endTime:d.endTime,error:d.error,reportS3Uri:io,videoS3Uri:k,traceS3Uri:P,metadata:{suiteName:d.suiteName,file:d.file}}}))).filter(d=>!!d);console.log("[reporter] [4/4] Finalising run...");let f=Hi(e.tests),u=await ue.put(`${i}/v1/local-runs/${p.testRunId}/complete`,{status:f,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:g},{headers:n});console.log(`
|
|
899
|
+
Shiplight cloud report: ${u.data.reportUrl}`)}var Wr=_(()=>{"use strict"});var Kr={};te(Kr,{runReport:()=>zi});import*as T from"fs";import*as x from"path";async function zi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight report [folder] [options]"),console.log(" shiplight report --merge <dirs...> [options]"),console.log(""),console.log("Regenerates index.html from report-data.json in the given folder."),console.log("With --merge, combines multiple shard report directories into one."),console.log(""),console.log("Options:"),console.log(" --open Open the report in the default browser after generating"),console.log(" --merge Merge multiple report directories into one"),console.log(" -o, --output <dir> Output directory for merged report (default: ./shiplight-report)"),console.log(" --github-summary Write test summary to $GITHUB_STEP_SUMMARY"),console.log(""),console.log("Examples:"),console.log(" shiplight report # regenerate ./shiplight-report/index.html"),console.log(" shiplight report my-report --open # regenerate and open"),console.log(" shiplight report --merge all-shards/*/shiplight-report/ # merge shard reports"),console.log(" shiplight report --merge shard-0/ shard-1/ -o combined-report # merge with custom output"),process.exit(0));let t=e.includes("--open"),r=e.includes("--merge"),o=e.includes("--github-summary");!r&&o&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Yi(e,t,o):await Vi(e,t)}async function Vi(e,t){let r=e.find(s=>!s.startsWith("--"))||"shiplight-report",o=x.isAbsolute(r)?r:x.join(process.cwd(),r),i=x.join(o,"report-data.json");T.existsSync(i)||(console.error(`Error: ${i} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let n;try{n=JSON.parse(T.readFileSync(i,"utf-8"))}catch(s){console.error(`Error: Failed to parse ${i}`),console.error(s instanceof Error?s.message:String(s)),process.exit(1)}let a=x.join(o,"index.html");if(T.writeFileSync(a,at(n),"utf-8"),console.log(`Shiplight report regenerated: ${a}`),await jr(n,o),t)try{let s=(await import("open")).default;await s(a)}catch{}}async function Yi(e,t,r){let o=x.join(process.cwd(),"shiplight-report"),i=e.findIndex(f=>f==="-o"||f==="--output");if(i!==-1&&e[i+1]){let f=e[i+1];o=x.isAbsolute(f)?f:x.join(process.cwd(),f)}let n=new Set(["-o","--output"]),a=new Set(["--merge","--open","--github-summary","-o","--output"]),s=[];for(let f=0;f<e.length;f++){let u=e[f];if(n.has(u)){f++;continue}if(a.has(u))continue;let d=x.isAbsolute(u)?u:x.join(process.cwd(),u);s.push(d)}s.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,p=0;T.mkdirSync(x.join(o,"screenshots"),{recursive:!0});for(let f=0;f<s.length;f++){let u=s[f],d=`shard-${f}`,w=x.join(u,"report-data.json");if(!T.existsSync(w)){console.warn(`Warning: No report-data.json found in ${u}, skipping.`);continue}let b;try{b=JSON.parse(T.readFileSync(w,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${w}, skipping.`);continue}console.log(`Merging ${d}: ${b.tests.length} tests from ${u}`),l+=b.totalDuration||0;let v=x.join(u,"screenshots");T.existsSync(v)&&Hr(v,x.join(o,"screenshots",d));let y=x.join(o,d);for(let A of b.tests){let k=[A,...A.attempts||[]];for(let P of k)Ji(P.steps,d),P.videoPath&&Gr(u,P.videoPath,y)&&(P.videoPath=`${d}/${P.videoPath}`),P.tracePath&&Gr(u,P.tracePath,y)&&(P.tracePath=`${d}/${P.tracePath}`);c.push(A)}p++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let h={tests:c,totalDuration:l,timestamp:new Date().toISOString()};T.writeFileSync(x.join(o,"report-data.json"),JSON.stringify(h,null,2),"utf-8");let g=x.join(o,"index.html");if(T.writeFileSync(g,at(h),"utf-8"),console.log(`
|
|
900
|
+
Merged ${c.length} tests from ${p} shards into: ${g}`),await jr(h,o),r&&Xi(c),t)try{let f=(await import("open")).default;await f(g)}catch{}}function Gr(e,t,r){let o=x.resolve(e,t);return o.startsWith(x.resolve(e)+x.sep)?T.existsSync(o)?(T.mkdirSync(r,{recursive:!0}),T.copyFileSync(o,x.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function Hr(e,t){T.mkdirSync(t,{recursive:!0});for(let r of T.readdirSync(e,{withFileTypes:!0})){let o=x.join(e,r.name),i=x.join(t,r.name);r.isDirectory()?Hr(o,i):T.copyFileSync(o,i)}}function Ji(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}async function jr(e,t){if(process.env.REPORT_TO_CLOUD!=="true")return;let r=process.env.SHIPLIGHT_API_TOKEN||process.env.__SHIPLIGHT_API_KEY;if(!r){console.warn("[report] REPORT_TO_CLOUD is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.");return}try{await Br(e,t,new Date().toISOString(),r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ut(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=x.join("tests",x.basename(t));return{name:e.title||x.basename(t),yamlPath:r}}function Xi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}let r=e.filter(c=>!c.file.includes("auth.setup")),o=r.filter(c=>c.flaky),i=r.filter(c=>c.status==="passed"&&!c.flaky),n=r.filter(c=>c.status!=="passed"),a=r.length,s=`## Test Results
|
|
901
901
|
|
|
902
902
|
`;if(n.length===0&&o.length===0?s+=`All ${a} tests passed
|
|
903
903
|
|
|
@@ -920,9 +920,9 @@ Merged ${c.length} tests from ${p} shards into: ${g}`),process.env.REPORT_TO_CLO
|
|
|
920
920
|
`;for(let c of i){let{name:l}=ut(c);s+=`- <sub>${l}</sub>
|
|
921
921
|
`}s+=`
|
|
922
922
|
</details>
|
|
923
|
-
`}T.appendFileSync(t,s),console.log("GitHub step summary written.")}var
|
|
923
|
+
`}T.appendFileSync(t,s),console.log("GitHub step summary written.")}var zr=_(()=>{"use strict";Dr();Wr()});var Vr,Yr=_(()=>{"use strict";Vr="0.1.45"});var Jr={};te(Jr,{runTranspile:()=>Zi});import*as Ce from"path";import{glob as qi}from"glob";async function Zi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),o=await qi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});o.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let i=0,n=0,a=0;for(let s of o.sort()){let c=Ce.resolve(r,s),l=yr(c,{version:Vr});if(!l.valid){i++,console.log(`
|
|
924
924
|
\u2717 ${s}`);for(let h of l.errors)console.log(` ERROR: ${h}`);continue}a++;let p=Ce.basename(l.specFile);if(l.warnings.length>0){n++,console.log(`\u26A0 ${s} \u2192 ${p}`);for(let h of l.warnings)console.log(` WARNING: ${h}`)}else console.log(`\u2713 ${s} \u2192 ${p}`)}console.log(`
|
|
925
|
-
${o.length} file(s): ${a} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var
|
|
925
|
+
${o.length} file(s): ${a} transpiled, ${i} error(s), ${n} warning(s)`),process.exit(i>0?1:0)}var Xr=_(()=>{"use strict";Qe();Yr()});var Qr={};te(Qr,{runInspect:()=>Qi});import*as Re from"fs";import*as qr from"path";async function Qi(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),o=e.find(a=>!a.startsWith("--"));o||(console.error("Error: no file specified"),process.exit(1));let i=qr.resolve(process.cwd(),o);Re.existsSync(i)||(console.error(`Error: file not found: ${i}`),process.exit(1));let n=Re.readFileSync(i,"utf-8");try{let a=fe(n),s=O(n);if(r)en(s,a);else{let c={...a.test_case_id!==void 0?{test_case_id:a.test_case_id}:{},...a.name?{name:a.name}:{},testFlow:s};console.log(JSON.stringify(c,null,t?0:2))}}catch(a){console.error(`Error parsing ${o}: ${a.message}`),process.exit(1)}}function en(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let o of r.tests){let i=o.skip?` [SKIP${typeof o.skip=="string"?`: ${o.skip}`:""}]`:"";console.log(` - ${o.name}: ${o.statements.length} statements${o.teardown?`, ${o.teardown.length} teardown`:""}${i}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Zr(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Zr(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let o=Zr(r.statements??[]);t.drafts+=o.drafts,t.actions+=o.actions,t.steps+=o.steps}return t}var eo=_(()=>{"use strict";re()});var to=ao(($c,tn)=>{tn.exports={name:"shiplightai",version:"0.1.45",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"}},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{build:"tsup && cd ../frontend && npx vite build --config vite.debugger.config.ts",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch",test:"playwright test","test:unit":"tsx --test 'src/**/*.test.ts'",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0",ai:"^6.0.3",axios:"^1.6.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.58.2","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"1.58.2"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});import rn from"dotenv";rn.config();function ro(){console.log(`
|
|
926
926
|
Usage: shiplight <command> [options]
|
|
927
927
|
|
|
928
928
|
Commands:
|
|
@@ -942,5 +942,5 @@ Examples:
|
|
|
942
942
|
shiplight transpile
|
|
943
943
|
shiplight transpile "tests/**/*.test.yaml"
|
|
944
944
|
shiplight debug tests/login.test.yaml
|
|
945
|
-
`)}var Ne=process.argv[2];switch(Ne){case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>($r(),Ar));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Rr(),Cr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(
|
|
946
|
-
`),
|
|
945
|
+
`)}var Ne=process.argv[2];switch(Ne){case"debug":{let{startDebugger:e}=await Promise.resolve().then(()=>($r(),Ar));await e(process.argv.slice(3));break}case"test":{let{runTests:e}=await Promise.resolve().then(()=>(Rr(),Cr));await e(process.argv.slice(3));break}case"report":{let{runReport:e}=await Promise.resolve().then(()=>(zr(),Kr));await e(process.argv.slice(3));break}case"transpile":{let{runTranspile:e}=await Promise.resolve().then(()=>(Xr(),Jr));await e(process.argv.slice(3));break}case"inspect":{let{runInspect:e}=await Promise.resolve().then(()=>(eo(),Qr));await e(process.argv.slice(3));break}case"--version":case"-v":{let e=to().version,t=process.env.SHIPLIGHT_BUILD_TAG?`-${process.env.SHIPLIGHT_BUILD_TAG}`:"";console.log(`${e}${t}`);break}case"--help":case"-h":ro();break;default:Ne&&console.error(`Unknown command: ${Ne}
|
|
946
|
+
`),ro(),process.exit(Ne?1:0)}
|
package/dist/index.js
CHANGED
|
@@ -4368,7 +4368,7 @@ ${p.join(`
|
|
|
4368
4368
|
`)}function Jl(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...oo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let a=new Set,o={imports:a,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=En(t?.tags);i.push(`test.describe.serial('${s}${ke(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],u=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=ao(d.testFlow,p.values);i.push(...fi(g,`${ke(d.name)} [${ke(p.name)}]`,o,1,u)),i.push("")}else i.push(...fi(d.testFlow,ke(d.name),o,1,u)),(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("});"),ro(i,a),i.join(`
|
|
4369
4369
|
`)}function En(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var ql=["testContext","request"];function gi(e){let t=new Set;function i(n){for(let a of n)switch(a.type){case me.ACTION:{let r=a.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&ql.includes(s)&&t.add(s);break}case me.STEP:i(a.statements);break;case me.IF_ELSE:{let o=a;i(o.then),o.else&&i(o.else);break}case me.WHILE_LOOP:i(a.body);break}}return i(e),t}function Zl(e){let t=gi(e.statements??[]);if(e.teardown)for(let i of gi(e.teardown))t.add(i);return t}function An(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function fi(e,t,i,n=0,a){let o=" ".repeat(n),r=[],s=Zl(e),l=An(s),c=a?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),a?.skip===!0?r.push(`${o} test.skip();`):typeof a?.skip=="string"&&r.push(`${o} test.skip(true, '${ke(a.skip)}');`),a?.fail===!0?r.push(`${o} test.fail();`):typeof a?.fail=="string"&&r.push(`${o} test.fail(true, '${ke(a.fail)}');`),a?.slow&&r.push(`${o} test.slow();`),a?.timeout&&r.push(`${o} test.setTimeout(${a.timeout});`);let d=e.teardown&&e.teardown.length>0,u=n+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=fe(e.statements,u+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=fe(e.teardown,u+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=fe(e.statements,u,i);r.push(...p)}return r.push(`${o}});`),r}function eo(e,t,i){let n=[],a=no(t),o=gi(a),r=An(o);return n.push(`test.${e}(async (${r}) => {`),n.push(...fe(a,1,i,e)),n.push("});"),n}function mi(e,t,i,n){let a=" ".repeat(n),o=[],r=no(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${a}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${a} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...fe(r,n+1,l,e)),o.push(`${a} await page.close();`),o.push(`${a}});`)}else{let l=gi(r),c=An(l);o.push(`${a}test.${e}(async (${c}) => {`),o.push(...fe(r,n+1,i,e)),o.push(`${a}});`)}return o}function no(e){let i=Ll({goal:"_hook",statements:e});return ge(i).statements??[]}function ao(e,t){let i=xn(e);for(let[n,a]of Object.entries(t))i=i.split(`<<${n}>>`).join(String(a));return ge(i)}function oo(){return["import { test, expect } from 'shiplightai/fixture';"]}function ro(e,t){if(t.size>0){let i=0;for(let a=0;a<e.length;a++)e[a].startsWith("import ")&&(i=a+1);let n=Array.from(t);e.splice(i,0,...n)}}var io=5;function co(e,t){let i={expandingPaths:new Set([lo(t)]),depth:0,referencedPaths:new Set},n={...e};Array.isArray(n.statements)&&(n.statements=Le(n.statements,t,i)),Array.isArray(n.teardown)&&(n.teardown=Le(n.teardown,t,i));for(let a of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[a])&&(n[a]=Le(n[a],t,i));return{doc:n,referencedTemplatePaths:Array.from(i.referencedPaths)}}function Le(e,t,i){let n=[];for(let a of e)if(ac(a)){let o=oc(a,t,i);n.push(...o)}else n.push(rc(a,t,i));return n}function ac(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function oc(e,t,i){if(i.depth>=io)throw new Error(`Template expansion exceeded maximum depth of ${io}. Check for deeply nested or circular template references.`);let n=lo(ic(t),e.template);if(i.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);i.referencedPaths.add(n);let a;try{a=tc(n,"utf-8")}catch(d){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${d.message}`)}let o=to(a);if(!o||typeof o!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let r=o.params||[],s=e.params||{};for(let d of r)if(!(d in s))throw new Error(`Template ${e.template} requires param "${d}" but it was not provided. Required params: [${r.join(", ")}]`);let l=o.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(s).length>0){let u=nc(l);for(let[p,g]of Object.entries(s))u=u.split(`<<${p}>>`).join(String(g));l=to(u)}let c={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths};return Le(l,n,c)}function rc(e,t,i){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=Le(n.statements,t,i)),Array.isArray(n.THEN)&&(n.THEN=Le(n.THEN,t,i)),Array.isArray(n.ELSE)&&(n.ELSE=Le(n.ELSE,t,i)),Array.isArray(n.DO)&&(n.DO=Le(n.DO,t,i)),n}var $n=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Mn(e,t){let i=ec(e),n=i?.name,a=i?.tags,o=i?.use;if(i&&(i.name!==void 0||i.tags!==void 0||i.use!==void 0)&&(delete i.name,delete i.tags,delete i.use),i?.suite){if(i.goal||i.statements)throw new $n('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return lc(i,n,a,o,t)}return sc(i,n,a,o,t)}function sc(e,t,i,n,a){let o=e?.beforeEach,r=e?.afterEach,s=uo(e?.parameters),l=e?.timeout,c=e?.skip,d=e?.fail,u=e?.only,p=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new $n(`The "url" field is not supported in local YAML tests. Use "base_url: ${e.url}" and add "- URL: /" as the first statement instead.`);e&&!e.goal&&t&&(e.goal=t);let g=[];if(a&&e&&typeof e=="object"){let b=co(e,a);e=b.doc,g=b.referencedTemplatePaths}let f=so(e),v=ge(f);return a&&(Ve(v.statements??[],a,"main"),v.teardown&&Ve(v.teardown,a,"teardown")),{testFlow:v,name:t,tags:i,use:n,beforeEach:o,afterEach:r,parameters:s,timeout:l,skip:c,fail:d,only:u,slow:p,referencedTemplatePaths:g}}function lc(e,t,i,n,a){let o=e.suite;if(!Array.isArray(o.tests)||o.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let r=o.beforeAll,s=o.afterAll,l=o.beforeEach,c=o.afterEach,d=[],u=o.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let v={goal:f.name,statements:f.statements};f.teardown&&(v.teardown=f.teardown);let b=[],m=v;if(a&&typeof v=="object"){let A=co(v,a);m=A.doc,b=A.referencedTemplatePaths,d.push(...b)}let y=so(m),S=ge(y),T=uo(f.parameters);return{testFlow:S,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:T,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),p=o.base_url,g=p?{...n,baseURL:p}:n;return{suite:{beforeAll:r,afterAll:s,beforeEach:l,afterEach:c,tests:u},name:t,tags:i,use:g,referencedTemplatePaths:d}}function uo(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Ve(e,t,i){for(let n=0;n<e.length;n++){let a=e[n],o=`${i}.${n}`,r=a.description||"";if(a.uid=cc(t,o,r),a.type===me.STEP)Ve(a.statements,t,o);else if(a.type===me.IF_ELSE){let s=a;Ve(s.then,t,`${o}.then`),s.else&&Ve(s.else,t,`${o}.else`)}else a.type===me.WHILE_LOOP&&Ve(a.body,t,`${o}.body`)}}function cc(e,t,i){let n=Ql("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function ho(e,t,i){let n=/\btemplate:\s/.test(e),a=/^suite:/m.test(e),o=n||a?null:_n(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Mn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},u=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=Jl(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Yl(c.testFlow,{...d,testName:c.name,tags:c.tags,use:u,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(`
|
|
4370
4370
|
`).filter(g=>!g.startsWith("import ")).join(`
|
|
4371
|
-
`);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),uc(hc(s),{recursive:!0}),dc(s,r)}catch(c){let d=c instanceof $n?"":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 In="0.1.
|
|
4371
|
+
`);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),uc(hc(s),{recursive:!0}),dc(s,r)}catch(c){let d=c instanceof $n?"":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 In="0.1.45";function mo(e){try{return mc(e).mtimeMs}catch{return 0}}var fc=`// @generated by shiplightai v${In}`;function wc(e,t){if(!go(e)||po(e,"utf-8").split(`
|
|
4372
4372
|
`,1)[0]!==fc)return!1;let n=mo(e);for(let a of t)if(mo(a)>n)return!1;return!0}function bc(e){let t=process.argv.slice(2),i=[],n=Cn(e);for(let a of t){if(a.startsWith("-"))continue;let o=a.endsWith(".yaml.spec.ts")?a.replace(/\.yaml\.spec\.ts$/,".test.yaml"):a;if(!o.endsWith(".test.yaml"))continue;let r=Cn(e,o);go(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function fo(e){let t=bc(e.cwd),i=t??gc("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),n=[];for(let a of i){let o=Cn(e.cwd,a),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=po(o,"utf-8");try{let l=Mn(s,o),c=pc(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&wc(r,[o,...l.referencedTemplatePaths]))continue;let p=ho(s,o,{version:In,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${a}:`,l),n.push({file:a,error:l})}}if(n.length>0){let a=`[shiplight] Transpilation failed for ${n.length} file(s):
|
|
4373
4373
|
`+n.map(o=>` - ${o.file}`).join(`
|
|
4374
4374
|
`);if(t)throw new Error(a);console.warn(a+" (skipped)")}}import*as oe from"fs";import*as ht from"path";import{globSync as vc}from"glob";function yo(e){return process.env.CI&&process.env.SHIPLIGHT_API_TOKEN?new Ln:new Dn(e)}var _c=".shiplight/action-cache";function bo(e){return e.replace(/\//g,"__")+".json"}function kc(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Dn=class{constructor(t){this.cwd=t;this.cacheDir=ht.join(t,_c)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!oe.existsSync(this.cacheDir))return i;for(let n of t){let a=ht.join(this.cacheDir,bo(n));try{if(oe.existsSync(a)){let o=oe.readFileSync(a,"utf-8");i.set(n,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;oe.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[n,a]of t)try{let o=ht.join(this.cacheDir,bo(n)),r=kn();if(oe.existsSync(o))try{r=JSON.parse(oe.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...a.entries}};oe.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!oe.existsSync(this.cacheDir))return;let t=vc("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,n=0;for(let a of t)try{let o=oe.readFileSync(ht.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=kc(a);i.set(s,r),n+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},Ln=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(On(),Pn));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(On(),Pn));return i(t)}loadAll(){}};function Tc(e={}){e.dotenv!==!1&&$c(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=yo(t).loadAll();return fo({cwd:t,actionEntityStores:n}),e.apiKey&&(process.env.__SHIPLIGHT_API_KEY=e.apiKey),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never",reportToCloud:e.reportToCloud,apiKey:e.apiKey}]]}}function Ac(e,...t){return Ec(e,...t)}function $c(e){let t=[],i=We.resolve(e),n=We.resolve(process.cwd());for(;;){let a=We.join(i,".env");if(xo.existsSync(a)&&t.push(a),i===n)break;let o=We.dirname(i);if(o===i)break;i=o}for(let a of t)Sc.config({path:a})}R();R();$t();qe();import{z as Ps}from"zod";var Uu=Ps.object({instruction:Ps.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 Os(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:Uu,usesElementIndex:!1,async execute(t,i){let{instruction:n}=t,a={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await Je(n,a,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:n,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:n}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Tt(r,a);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}Te();R();import{generateText as Gu}from"ai";import{convert as ju}from"html-to-text";async function Ku(e,t,i){let{apiKey:n,domain:a}=e;if(!n||!a)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${a}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}h.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let u=(d.storage||{}).url;if(h.info(`message_url: ${u}`),!u){let f=(d.message||{}).headers||{},v=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!v.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:v,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=u.split("/"),g=p[p.length-1];if(h.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${a}/messages/${g}`;try{let v=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(v.ok){let b=await v.json(),m=b.Subject||"",y=b.From||"",S=b.To||"",T=b.Date||"",A=b["Message-Id"]||"";h.info(`subject: ${m}`),h.info(`from_addr: ${y}`),h.info(`to_addr: ${S}`),h.info(`date: ${T}`),h.info(`message_id: ${A}`);let M=b["body-html"]||b["body-plain"]||"";if(M&&M.includes("<")&&(M=ju(M)),h.info(`Body: ${M.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!M.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:y,to:S,date:T,body:M,message_id:A});continue}else h.warn(`Messages API returned ${v.status}`)}catch(v){h.warn(`Failed to parse JSON response: ${v}`)}}try{let f=await fetch(u,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!f.ok){h.warn(`Could not fetch stored message: ${f.status}`);continue}let v=await f.text();h.info(`Fallback: Raw email length: ${v.length}`);let b=v.split(`
|
package/dist/reporter.js
CHANGED
|
@@ -842,7 +842,7 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Ue=Je(()=>{"use strict"});impo
|
|
|
842
842
|
});
|
|
843
843
|
</script>
|
|
844
844
|
</body>
|
|
845
|
-
</html>`}var Ut={before:0,main:1,teardown:2,after:3};function We(e){let t=e.split(".")[0];return Ut[t]??1}function Ge(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Wt(e){return[...e].sort(([t],[r])=>{let i=We(t),o=We(r);if(i!==o)return i-o;let n=Ge(t),s=Ge(r);for(let a=0;a<Math.max(n.length,s.length);a++){let d=n[a]??-1,u=s[a]??-1;if(d!==u)return d-u}return 0})}var se=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud
|
|
845
|
+
</html>`}var Ut={before:0,main:1,teardown:2,after:3};function We(e){let t=e.split(".")[0];return Ut[t]??1}function Ge(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Wt(e){return[...e].sort(([t],[r])=>{let i=We(t),o=We(r);if(i!==o)return i-o;let n=Ge(t),s=Ge(r);for(let a=0;a<Math.max(n.length,s.length);a++){let d=n[a]??-1,u=s[a]??-1;if(d!==u)return d-u}return 0})}var se=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud??!1,this.apiKey=t.apiKey||process.env.__SHIPLIGHT_API_KEY}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 f=u.test.titlePath().join(" > "),b=r.get(f);b||(b=[],r.set(f,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let f=u[0].test.location.file,b=[],p,h,g;for(let v=0;v<u.length;v++){let{test:T,result:_}=u[v],k=await this.buildReportTest(T,_,f);p=k,h||(h=k.startTime),g=k.endTime,b.push({attemptNumber:v+1,status:_.status,duration:_.duration,steps:k.steps,error:k.error,videoPath:k.videoPath,tracePath:k.tracePath})}let l=b[b.length-1],{test:x}=u[u.length-1],y={title:x.title,baseTitle:p?.baseTitle,file:S.relative(process.cwd(),f),status:l.status,duration:l.duration,steps:l.steps,error:l.error,videoPath:l.videoPath,tracePath:l.tracePath,actionStepsMap:p?.actionStepsMap,tags:p?.tags,baseUrl:p?.baseUrl,skip:p?.skip,slow:p?.slow,timeout:p?.timeout,parameterSetName:p?.parameterSetName,startTime:h,endTime:g,suiteName:p?.suiteName},w=b.some(v=>v.status==="failed"||v.status==="timedOut");b.length>1&&w&&l.status==="passed"&&(y.flaky=!0,y.retries=b.length-1,y.attempts=b),i.push(y)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString()},n=S.isAbsolute(this.outputFolder)?this.outputFolder:S.join(process.cwd(),this.outputFolder);$.mkdirSync(n,{recursive:!0});let s=S.join(n,"screenshots");for(let u=0;u<o.tests.length;u++){let f=o.tests[u],b=f.attempts&&f.attempts.length>0,p=[{obj:f,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(f.attempts)for(let h=0;h<f.attempts.length;h++)p.push({obj:f.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:g,screenshotSubDir:l}of p){let x=S.join(s,l),y=!1;for(let w of h.steps)if(w.screenshot&&S.isAbsolute(w.screenshot))try{y||($.mkdirSync(x,{recursive:!0}),y=!0);let v=`${w.stepId.replace(/\./g,"-")}.png`;$.copyFileSync(w.screenshot,S.join(x,v)),w.screenshot=`screenshots/${l}/${v}`}catch(v){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,v)}if(h.videoPath&&S.isAbsolute(h.videoPath)){let w=S.extname(h.videoPath)||".webm",v=`${g}-video${w}`;try{$.copyFileSync(h.videoPath,S.join(n,v)),h.videoPath=v}catch{h.videoPath=void 0}}if(h.tracePath&&S.isAbsolute(h.tracePath)){let w=S.extname(h.tracePath)||".zip",v=`${g}-trace${w}`;try{$.copyFileSync(h.tracePath,S.join(n,v)),h.tracePath=v}catch{h.tracePath=void 0}}}}let a=S.join(n,"report-data.json");$.writeFileSync(a,JSON.stringify(o,null,2),"utf-8");let d=S.join(n,"index.html");if($.writeFileSync(d,Re(o),"utf-8"),console.log(`
|
|
846
846
|
Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(d)}catch{}if(this.reportToCloud){if(!this.apiKey){console.warn("[reporter] reportToCloud is enabled but no apiKey found, skipping cloud upload.");return}try{let{uploadToCloud:u}=await Promise.resolve().then(()=>(Ue(),Fe));await u(o,n,this.runStartTime,this.apiKey)}catch(u){console.warn("[reporter] Cloud upload failed:",u)}}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:S.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(p=>p.message||p.stack||String(p)).join(`
|
|
847
847
|
|
|
848
848
|
`)),r.stdout.length>0&&(o.stdout=r.stdout.map(p=>typeof p=="string"?p:p.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(p=>typeof p=="string"?p:p.toString()).join(""));for(let p of r.attachments)p.name==="video"&&p.path&&(o.videoPath=p.path),p.name==="trace"&&p.path&&(o.tracePath=p.path);let n=r.attachments.find(p=>p.name==="shiplight-results"),s=null;if(n)try{if(n.body)s=JSON.parse(n.body.toString("utf-8"));else if(n.path){let p=$.readFileSync(n.path,"utf-8");s=JSON.parse(p)}}catch{}let a=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),f=u?u[1]:t.title,b=u?u[2]:void 0;if(b&&(o.parameterSetName=b),$.existsSync(a))try{let p=$.readFileSync(a,"utf-8"),h=Oe(p,a);if(h.suite){let g=h.suite.tests.find(l=>l.name===f);g&&(d=X(g.testFlow),g.tags?.length&&(o.tags=g.tags),g.skip!==void 0&&(o.skip=g.skip),g.slow&&(o.slow=g.slow),g.timeout!==void 0&&(o.timeout=g.timeout),o.baseTitle=g.name||g.testFlow?.goal),o.suiteName=h.name,h.tags?.length&&(o.suiteTags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL)}else h.testFlow&&(d=X(h.testFlow),o.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(o.tags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL))}catch{}if(s||Object.keys(d).length>0){let p=new Set([...Object.keys(d),...Object.keys(s||{})]),h=Array.from(p).map(l=>[l,null]),g=Wt(h);for(let[l]of g){let x=d[l],y=s?.[l],w=x?.description;if(!w||w==="Action"||w==="Draft"){let T=x?.action_entity;w=T?.action_description||T?.action_data?.kwargs?.description||y?.description||l}let v={stepId:l,description:w,status:y?.status||"pending",duration:y?.duration};if(y?.message){let T=typeof y.message=="string"?y.message:JSON.stringify(y.message,null,2);y.status==="failure"?v.error=T:v.message=T}if(y?.screenshot){let T=y.screenshot,_=n?.path?S.dirname(n.path):"",k=S.isAbsolute(T)?T:S.join(_,T);$.existsSync(k)&&(v.screenshot=k)}o.steps.push(v)}}return Object.keys(d).length>0&&(o.actionStepsMap=d),o}},Gt=se;export{Gt as default};
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "shiplightai",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.45",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "Shiplight CLI for running and debugging .test.yaml files",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -83,11 +83,11 @@
|
|
|
83
83
|
"@types/node": "^24.0.0",
|
|
84
84
|
"tsup": "^8.3.5",
|
|
85
85
|
"typescript": "5.5.4",
|
|
86
|
-
"sdk-internal": "0.1.1",
|
|
87
|
-
"shiplight-tools": "1.0.0",
|
|
88
|
-
"shiplight-types": "0.1.0",
|
|
89
86
|
"mcp-tools": "1.0.0",
|
|
90
87
|
"sdk-core": "0.1.0",
|
|
88
|
+
"shiplight-tools": "1.0.0",
|
|
89
|
+
"shiplight-types": "0.1.0",
|
|
90
|
+
"sdk-internal": "0.1.1",
|
|
91
91
|
"@loggia/common": "1.0.0"
|
|
92
92
|
},
|
|
93
93
|
"peerDependencies": {
|