shiplightai 0.1.69 → 0.1.70

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/cli.js CHANGED
@@ -17,7 +17,7 @@ Install it as a project dependency instead:
17
17
  `)}catch{}}function vn(e=ut()){try{let t=C.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<yn)return r}catch{}return null}function _n(e,t=ut()){try{C.mkdirSync(Y.dirname(t),{recursive:!0}),C.writeFileSync(t,JSON.stringify(e))}catch{}}async function xn(){try{let e=await fetch(mn,{headers:{Accept:"application/json"},signal:AbortSignal.timeout(3e3)});if(!e.ok)return null;let t=await e.json();return typeof t.version=="string"?t.version:null}catch{return null}}function Tn(e,t){let r=d=>{let h=d.indexOf("-");return h===-1?[d,!1]:[d.slice(0,h),!0]},s=d=>d.split(".").map(h=>parseInt(h,10)||0),[n,o]=r(e),[i,a]=r(t),c=s(n),l=s(i),u=Math.max(c.length,l.length);for(let d=0;d<u;d++){let h=c[d]??0,g=l[d]??0;if(h<g)return!0;if(h>g)return!1}return!!(o&&!a)}async function Zt(e={}){let t=e.runningVersion??lt,r=e.cwd??process.cwd(),s=e.cacheFile??ut(),n=e.fetchLatest??xn,o=e.env??process.env,i=e.warn??(l=>console.warn(l));if(o.CI||t==="dev"||!C.existsSync(Y.join(r,"package-lock.json")))return;let a=null,c=vn(s);if(c)a=c.latest;else{try{a=await n()}catch{a=null}a&&_n({latest:a,fetchedAt:Date.now()},s)}a&&Tn(t,a)&&i(`
18
18
  \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var lt,ke,mn,yn,wn,Ne=_(()=>{"use strict";lt="0.1.69",ke=lt!=="dev"?lt:void 0,mn="https://registry.npmjs.org/shiplightai/latest",yn=3600*1e3,wn=10080*60*1e3});import*as ie from"fs";import*as ue from"path";function dt(e){let{projectPath:t}=e,r=e.projectName??ue.basename(ue.resolve(t));if(ie.existsSync(t)&&ie.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);ie.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let i=ue.join(t,n);ie.mkdirSync(ue.dirname(i),{recursive:!0}),ie.writeFileSync(i,o)};return s("package.json",Pn.replace("{{name}}",r)),s("playwright.config.ts",kn),s(".gitignore",En),s(".env.example",An),s(".mcp.json",$n),s("auth/example.login.ts",In),s("environments/example.env.yaml",Mn),s("tests/example.test.yaml",On),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var Pn,kn,En,An,$n,In,Mn,On,Qt=_(()=>{"use strict";Pn=`{
20
+ `)}var lt,ke,mn,yn,wn,Ne=_(()=>{"use strict";lt="0.1.70",ke=lt!=="dev"?lt:void 0,mn="https://registry.npmjs.org/shiplightai/latest",yn=3600*1e3,wn=10080*60*1e3});import*as ie from"fs";import*as ue from"path";function dt(e){let{projectPath:t}=e,r=e.projectName??ue.basename(ue.resolve(t));if(ie.existsSync(t)&&ie.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);ie.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let i=ue.join(t,n);ie.mkdirSync(ue.dirname(i),{recursive:!0}),ie.writeFileSync(i,o)};return s("package.json",Pn.replace("{{name}}",r)),s("playwright.config.ts",kn),s(".gitignore",En),s(".env.example",An),s(".mcp.json",$n),s("auth/example.login.ts",In),s("environments/example.env.yaml",Mn),s("tests/example.test.yaml",On),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var Pn,kn,En,An,$n,In,Mn,On,Qt=_(()=>{"use strict";Pn=`{
21
21
  "name": "{{name}}",
22
22
  "type": "module",
23
23
  "scripts": {
@@ -205,12 +205,12 @@ test('__shiplight_debug__', async ({ page, agent }) => {
205
205
  // Keep alive until the server is closed externally (Ctrl+C kills the process)
206
206
  await new Promise(() => {});
207
207
  });
208
- `}async function Un(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let o=t();o.once("error",()=>n(!1)),o.once("listening",()=>{o.close(()=>n(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function ft(e){let{yamlFilePath:t,configPath:r,tempSuffix:s="",headed:n}=e,o=te.dirname(r),i=await Un(16174),a;if(!ce.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let y=jn(ce.readFileSync(t,"utf-8"));y?.use&&typeof y.use=="object"&&!Array.isArray(y.use)&&(a=y.use),y?.base_url&&!a?.baseURL&&(a={...a,baseURL:y.base_url}),y?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!y.settings.auto_dismiss_modal}),y?.settings?.browser_timezone!=null&&(a={...a,timezoneId:String(y.settings.browser_timezone)}),y?.settings?.browser_language!=null&&(a={...a,locale:String(y.settings.browser_language)}),y?.settings?.extra_http_headers!=null&&typeof y.settings.extra_http_headers=="object"&&(a={...a,extraHTTPHeaders:y.settings.extra_http_headers})}catch(y){console.error("[debugger] Could not parse YAML for `use` block:",y)}let c=te.dirname(te.resolve(t)),l=s?`-${s}`:"",u=te.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=Fn(t,i,a,o);ce.writeFileSync(u,d);let h=fr(u),g=["playwright","test",u,...n?["--headed"]:[]],f=Dn("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});f.stdout?.on("data",y=>{process.stderr.write(y)}),f.stderr?.on("data",y=>{process.stderr.write(y)});let p=()=>{f.killed||f.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),f.on("close",y=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),h(),y!==0&&y!==null&&console.error(`[debugger] Playwright process exited with code ${y}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let S=["127.0.0.1","::1"];async function w(y){try{let E=y.includes(":")?`[${y}]`:y,T=await fetch(`http://${E}:${i}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let m=null;for(let y=0;y<180;y++){if(f.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${f.exitCode} before sandbox was ready`);for(let E of S)if(await w(E)){m=E;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${i}`);break}if(y===179)throw p(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(E=>setTimeout(E,1e3))}if(!m)throw p(),h(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:m,pid:f.pid??0,cleanup:async()=>{p(),h()}}}var ht=_(()=>{"use strict"});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});import{v4 as ua}from"uuid";var wr=_(()=>{"use strict"});import{z as b}from"zod";var br,gt,Sr,we,vr,_r,xr,B,Tr,Ue,mt,yt=_(()=>{"use strict";br=b.enum(["JS_CODE","AI_MODE"]),gt=b.object({type:br,expression:b.string()}),Sr=b.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=b.object({uid:b.string(),type:Sr,comment:b.string().optional()}),vr=b.object({action_data:b.object({action_name:b.string(),kwargs:b.record(b.any()).optional(),args:b.array(b.any()).optional()}),action_description:b.string().optional(),url:b.string().optional(),xpath:b.string().nullable().optional(),locator:b.string().nullable().optional(),css_selector:b.string().nullable().optional(),unique_selector:b.string().nullable().optional(),element_index:b.number().nullable().optional(),frame_path:b.array(b.any()).optional(),artifacts:b.record(b.any()).optional(),feedback:b.string().optional(),original_browser_use_action:b.any().optional()}).passthrough(),_r=we.extend({type:b.literal("DRAFT"),description:b.string()}),xr=we.extend({type:b.literal("ACTION"),description:b.string(),action_entity:vr.optional(),locator:b.string().optional(),use_pure_vision:b.boolean().optional()}),B=b.lazy(()=>b.union([_r,xr,we.extend({type:b.literal("STEP"),description:b.string().optional().default(""),statements:b.array(B),reference_id:b.number().optional(),template_path:b.string().optional(),template_params:b.record(b.string()).optional()}),we.extend({type:b.literal("IF_ELSE"),description:b.string().optional(),condition:gt,then:b.array(B),else:b.array(B).optional()}),we.extend({type:b.literal("WHILE_LOOP"),description:b.string().optional(),condition:gt,body:b.array(B),timeout_ms:b.number().optional()})])),Tr=b.object({name:b.string(),statements:b.array(B),teardown:b.array(B).optional(),skip:b.union([b.boolean(),b.string()]).optional(),timeout:b.number().optional(),fail:b.union([b.boolean(),b.string()]).optional(),only:b.boolean().optional(),slow:b.boolean().optional()}),Ue=b.object({tests:b.array(Tr).min(1),beforeAll:b.array(B).optional(),afterAll:b.array(B).optional(),beforeEach:b.array(B).optional(),afterEach:b.array(B).optional()}),mt=b.object({comment:b.string().optional(),version:b.string().optional(),goal:b.string().optional(),url:b.string().optional(),baseURL:b.string().optional(),final_feedback:b.string().optional(),completed:b.boolean().optional(),success:b.boolean().optional(),statements:b.array(B).optional(),teardown:b.array(B).optional(),last_modified_at:b.string().optional(),testGroup:Ue.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 Bn,parse as Ar,parseAllDocuments as ya,parseDocument as Hn,Document as Wn,isMap as Se,isSeq as q}from"yaml";import{v4 as X}from"uuid";function Be(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(G)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(G)),r}function $e(e,t){if(e.testGroup)return Ir(e,t);let r=Be(e,t),s=new Wn(r);return e.comment&&(s.commentBefore=e.comment),Pr(s,e.statements??[]),e.teardown&&Pr(s,e.teardown,"teardown"),s.toString($r)}function Pr(e,t,r="statements"){let s=e.contents;if(!s||!Se(s))return;let n=s.get(r,!0);q(n)&&Ae(n,t)}function Ae(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),s.comment&&(r===0?e.commentBefore=s.comment:n.commentBefore=s.comment),Se(n)){let o=n;if(s.type==="STEP"){let i=o.get("statements",!0);q(i)&&Ae(i,s.statements)}else if(s.type==="IF_ELSE"){let i=o.get("THEN",!0);q(i)&&Ae(i,s.then);let a=o.get("ELSE",!0);q(a)&&s.else&&Ae(a,s.else)}else if(s.type==="WHILE_LOOP"){let i=o.get("DO",!0);q(i)&&Ae(i,s.body)}}}}function Ir(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let s={};t?.test_case_id!==void 0&&(s.test_case_id=t.test_case_id),t?.name&&(s.name=t.name),t?.tags&&t.tags.length>0&&(s.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(s.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(s.settings=t.settings);let n={};return e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(G)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(G)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(G)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(G)),n.tests=r.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(G),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(G)),i}),s.suite=n,Bn(s,$r)}function G(e){switch(e.type){case"DRAFT":return Gn(e);case"ACTION":return Kn(e);case"STEP":return zn(e);case"IF_ELSE":return Vn(e);case"WHILE_LOOP":return Yn(e)}}function Gn(e){return{intent:e.description}}function Kn(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&a.trim()&&e.description)return{description:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:a}}if(!e.action_entity)return{intent:e.description};let s=e.action_entity.action_data??e.action_entity.action;if(!s)return{intent:e.description};let n={intent:e.description,action:s.action_name},o=e.locator??e.action_entity.locator;o&&(n.locator=o);let i=e.action_entity.xpath;if(i&&(n.xpath=i),e.use_pure_vision&&(n.use_pure_vision=!0),s.kwargs&&Object.keys(s.kwargs).length>0)for(let[a,c]of Object.entries(s.kwargs))a!=="uid"&&(a==="statement"&&(t==="ai_action"||t==="ai_step")||(n[a]=c));return s.args&&s.args.length>0&&(n.args=s.args),n}function zn(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(G)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Vn(e){let t={IF:Mr(e.condition),THEN:e.then.map(G)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(G)),t}function Yn(e){let t={WHILE:Mr(e.condition),DO:e.body.map(G)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Mr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ie(e){try{let t=Ar(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(s=>typeof s=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function wt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(wt);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let s={};for(let[n,o]of Object.entries(t))s[n]=wt(o);return s}function j(e){if(e.length>kr)throw new Error(`YAML input too large (${e.length} bytes, max ${kr})`);let t=wt(Ar(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Jn(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:K(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=K(t.teardown));let s=mt.safeParse(r);if(!s.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(s.error.errors)}`);let n=s.data;return He(e,n),n}function Jn(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let n={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:K(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=K(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=K(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=K(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=K(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=K(t.afterEach));let o=Ue.safeParse(n);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function K(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Xn)}function Xn(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 qn(t);if("WHILE"in t)return Zn(t);if("STEP"in t)return Qn(t);if("VERIFY"in t){let r=t.VERIFY,s={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(s.code=t.js),{uid:X(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:s}}}}if("URL"in t){let r=t.URL,s=t.new_tab===!0?!0:void 0,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return s&&(o.new_tab=!0),n!==void 0&&(o.timeout_seconds=n),{uid:X(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:X(),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:s}}}}}if("WAIT"in t){let r=t.WAIT,s=typeof t.seconds=="number"?t.seconds:3;return{uid:X(),type:"ACTION",description:typeof r=="string"?r:`Wait ${s}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${s}s`,action_data:{action_name:"wait",kwargs:{seconds:s}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');let s=typeof t.description=="string"?t.description:"Code block";return{uid:X(),type:"ACTION",description:s,action_entity:{action_description:s,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",s=t.js;return{uid:X(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof s=="string"?s:String(s)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...s}=t;return Er({...s,action:"function",functionName:r})}if("action"in t)return Er(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:X(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Or(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 qn(e){let t=Or(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let s={uid:X(),type:"IF_ELSE",condition:t,then:K(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(s.else=K(e.ELSE)),s}function Zn(e){let t=Or(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let s={uid:X(),type:"WHILE_LOOP",condition:t,body:K(r)};return typeof e.timeout_ms=="number"&&(s.timeout_ms=e.timeout_ms),s}function Qn(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:X(),type:"STEP",description:t,statements:K(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let s=e.template_params,n={};for(let[o,i]of Object.entries(s))n[o]=String(i);r.template_params=n}return r}function Er(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",s=typeof e.locator=="string"?e.locator:void 0,n=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[l,u]of Object.entries(e))eo.has(l)||(i[l]=u);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js),(t==="ai_action"||t==="ai_step")&&i.statement===void 0&&(i.statement=r);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};s&&(a.locator=s),n&&(a.xpath=n);let c={uid:X(),type:"ACTION",description:r,action_entity:a};return o&&(c.use_pure_vision=!0),c}function He(e,t){let r;try{r=Hn(e)}catch{return}let s=r.contents;if(!s||!Se(s))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=s.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let n=s,o=n.get("statements",!0);q(o)&&t.statements&&be(o,t.statements);let i=n.get("teardown",!0);q(i)&&t.teardown&&be(i,t.teardown)}function be(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=e.items[r];s.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=s.commentBefore);let n=t[r];if(n.type==="STEP"&&Se(s)){let o=s.get("statements",!0);q(o)&&be(o,n.statements)}else if(n.type==="IF_ELSE"&&Se(s)){let o=s.get("THEN",!0);q(o)&&be(o,n.then);let i=s.get("ELSE",!0);q(i)&&n.else&&be(i,n.else)}else if(n.type==="WHILE_LOOP"&&Se(s)){let o=s.get("DO",!0);q(o)&&be(o,n.body)}}}var $r,kr,eo,We=_(()=>{"use strict";yt();$r={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};kr=1024*1024;eo=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as _a,stringify as xa}from"yaml";var Lr=_(()=>{"use strict";We()});var bt,Ge,Ke=_(()=>{"use strict";bt=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},Ge=e=>{let t=[],r=s=>{for(let n of s){t.push(n);let o=bt(n);for(let i of o)r(i.statements)}};return r(e),t}});function jr(e){let t=0,r=0;for(let s of e)if(s.type==="DRAFT")r++;else if(s.type==="ACTION"){let n=s.action_entity?.action_data?.action_name??"";Dr.has(n)||t++}return{action:t,draft:r}}function ro(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Nr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function so(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,s=t.join(`
209
- `).match(/\bawait\b/g),n=s?s.length:0;return r<=Rr&&n<=Cr?null:`${r} non-blank line(s), ${n} await(s) \u2014 limits are ${Rr} lines and ${Cr} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function St(e,t){let r=t?.coverageThreshold??to,s=[],n=[],o;try{o=j(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||s.push('Missing required field: "goal"'),o.statements?.length||s.push('Missing required field: "statements"');let i=[...Ge(o.statements??[]),...o.teardown?Ge(o.teardown):[]],{action:a,draft:c}=jr(i),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let h of i){let g=h;if(g.reference_id!==void 0){let f=g.description||h.uid;s.push(`Unresolved cloud template reference on statement "${f}" (reference_id: ${g.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(h.type==="ACTION"){let f=h,p=f.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let S=f.action_entity?.action_data?.kwargs?.code;if(typeof S=="string"){let w=ro(S);if(w){let m=f.description||p;s.push(`Invalid JS in "${m}": ${w}. ${l}`)}else if(p==="verify"){let m=so(S);if(m){let y=f.description||p;s.push(`JS cache for "${y}" is too complex: ${m}`)}}}}}if(h.type==="IF_ELSE"){let f=h;if(f.condition.type==="JS_CODE"){let p=Nr(f.condition.expression);p&&s.push(`Invalid JS in IF condition "${f.condition.expression}": ${p}. ${l}`)}}if(h.type==="WHILE_LOOP"){let f=h;if(f.condition.type==="JS_CODE"){let p=Nr(f.condition.expression);p&&s.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${p}. ${l}`)}}}let u=a+c,d=u>0?Math.round(a/u*100):0;return u>0&&d/100<r&&n.push(`Low action coverage: ${a}/${u} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:s.length===0,errors:s,warnings:n,stats:{total:u,action:a,draft:c,coverage:d}}}var to,Rr,Cr,Dr,Fr=_(()=>{"use strict";We();Ke();to=.5,Rr=5,Cr=3,Dr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var ze=_(()=>{"use strict"});var Ur=_(()=>{"use strict";ze()});var Br,oo,Hr,Wr,Me,io,ao,Gr=_(()=>{"use strict";Br=112,oo=1080-Br,Hr={"Blackberry PlayBook":{name:"Blackberry PlayBook",userAgent:"Mozilla/5.0 (PlayBook; U; RIM Tablet OS 2.1.0; en-US) AppleWebKit/536.2+ (KHTML like Gecko) Version/26.0 Safari/536.2+",screen:{width:600,height:1024},viewport:{width:600,height:1024},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"BlackBerry Z30":{name:"BlackBerry Z30",userAgent:"Mozilla/5.0 (BB10; Touch) AppleWebKit/537.10+ (KHTML, like Gecko) Version/26.0 Mobile Safari/537.10+",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note 3":{name:"Galaxy Note 3",userAgent:"Mozilla/5.0 (Linux; U; Android 4.3; en-us; SM-N900T Build/JSS15J) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy Note II":{name:"Galaxy Note II",userAgent:"Mozilla/5.0 (Linux; U; Android 4.1; en-us; GT-N7100 Build/JRO03C) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S III":{name:"Galaxy S III",userAgent:"Mozilla/5.0 (Linux; U; Android 4.0; en-us; GT-I9300 Build/IMM76D) AppleWebKit/534.30 (KHTML, like Gecko) Version/26.0 Mobile Safari/534.30",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Galaxy S5":{name:"Galaxy S5",userAgent:"Mozilla/5.0 (Linux; Android 5.0; SM-G900P Build/LRX21T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S8":{name:"Galaxy S8",userAgent:"Mozilla/5.0 (Linux; Android 7.0; SM-G950U Build/NRD90M) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:740},viewport:{width:360,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S9+":{name:"Galaxy S9+",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; SM-G965U Build/R16NW) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:320,height:658},viewport:{width:320,height:658},deviceScaleFactor:4.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy S24":{name:"Galaxy S24",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-S921U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:780},viewport:{width:360,height:780},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy A55":{name:"Galaxy A55",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-A556B) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:480,height:1040},viewport:{width:480,height:1040},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S4":{name:"Galaxy Tab S4",userAgent:"Mozilla/5.0 (Linux; Android 8.1.0; SM-T837A) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:712,height:1138},viewport:{width:712,height:1138},deviceScaleFactor:2.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Galaxy Tab S9":{name:"Galaxy Tab S9",userAgent:"Mozilla/5.0 (Linux; Android 14; SM-X710) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:640,height:1024},viewport:{width:640,height:1024},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"iPad (gen 5)":{name:"iPad (gen 5)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 6)":{name:"iPad (gen 6)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 7)":{name:"iPad (gen 7)",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:810,height:1080},viewport:{width:810,height:1080},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad (gen 11)":{name:"iPad (gen 11)",userAgent:"Mozilla/5.0 (iPad; CPU OS 18_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/604.1",screen:{width:656,height:944},viewport:{width:656,height:944},deviceScaleFactor:2.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Mini":{name:"iPad Mini",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:768,height:1024},viewport:{width:768,height:1024},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPad Pro 11":{name:"iPad Pro 11",userAgent:"Mozilla/5.0 (iPad; CPU OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:834,height:1194},viewport:{width:834,height:1194},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6":{name:"iPhone 6",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 6 Plus":{name:"iPhone 6 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7":{name:"iPhone 7",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 7 Plus":{name:"iPhone 7 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8":{name:"iPhone 8",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 8 Plus":{name:"iPhone 8 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:414,height:736},viewport:{width:414,height:736},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE":{name:"iPhone SE",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 10_3_1 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/14E304 Safari/602.1",screen:{width:320,height:568},viewport:{width:320,height:568},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone SE (3rd gen)":{name:"iPhone SE (3rd gen)",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 18_5 like Mac OS X) AppleWebKit/603.1.30 (KHTML, like Gecko) Version/26.0 Mobile/19E241 Safari/602.1",screen:{width:375,height:667},viewport:{width:375,height:667},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone X":{name:"iPhone X",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/26.0 Mobile/15A372 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:812},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone XR":{name:"iPhone XR",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:896},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11":{name:"iPhone 11",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro":{name:"iPhone 11 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:635},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 11 Pro Max":{name:"iPhone 11 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 12_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:414,height:896},viewport:{width:414,height:715},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12":{name:"iPhone 12",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro":{name:"iPhone 12 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Pro Max":{name:"iPhone 12 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 12 Mini":{name:"iPhone 12 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 14_4 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13":{name:"iPhone 13",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro":{name:"iPhone 13 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Pro Max":{name:"iPhone 13 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 13 Mini":{name:"iPhone 13 Mini",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 15_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:375,height:812},viewport:{width:375,height:629},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14":{name:"iPhone 14",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:390,height:844},viewport:{width:390,height:664},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Plus":{name:"iPhone 14 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:428,height:926},viewport:{width:428,height:746},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro":{name:"iPhone 14 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:660},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 14 Pro Max":{name:"iPhone 14 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 16_0 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:740},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15":{name:"iPhone 15",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Plus":{name:"iPhone 15 Plus",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro":{name:"iPhone 15 Pro",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:393,height:852},viewport:{width:393,height:659},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"iPhone 15 Pro Max":{name:"iPhone 15 Pro Max",userAgent:"Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Mobile/15E148 Safari/604.1",screen:{width:430,height:932},viewport:{width:430,height:739},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Kindle Fire HDX":{name:"Kindle Fire HDX",userAgent:"Mozilla/5.0 (Linux; U; en-us; KFAPWI Build/JDQ39) AppleWebKit/535.19 (KHTML, like Gecko) Silk/3.13 Safari/535.19 Silk-Accelerated=true",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"LG Optimus L70":{name:"LG Optimus L70",userAgent:"Mozilla/5.0 (Linux; U; Android 4.4.2; en-us; LGMS323 Build/KOT49I.MS32310c) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:1.25,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 550":{name:"Microsoft Lumia 550",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 550) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Microsoft Lumia 950":{name:"Microsoft Lumia 950",userAgent:"Mozilla/5.0 (Windows Phone 10.0; Android 4.2.1; Microsoft; Lumia 950) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36 Edge/14.14263",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:4,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 10":{name:"Nexus 10",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 10 Build/MOB31T) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:800,height:1280},viewport:{width:800,height:1280},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 4":{name:"Nexus 4",userAgent:"Mozilla/5.0 (Linux; Android 4.4.2; Nexus 4 Build/KOT49H) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:384,height:640},viewport:{width:384,height:640},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5":{name:"Nexus 5",userAgent:"Mozilla/5.0 (Linux; Android 6.0; Nexus 5 Build/MRA58N) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 5X":{name:"Nexus 5X",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 5X Build/OPR4.170623.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6":{name:"Nexus 6",userAgent:"Mozilla/5.0 (Linux; Android 7.1.1; Nexus 6 Build/N6F26U) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 6P":{name:"Nexus 6P",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Nexus 6P Build/OPP3.170518.006) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:732},viewport:{width:412,height:732},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nexus 7":{name:"Nexus 7",userAgent:"Mozilla/5.0 (Linux; Android 6.0.1; Nexus 7 Build/MOB30X) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:600,height:960},viewport:{width:600,height:960},deviceScaleFactor:2,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia Lumia 520":{name:"Nokia Lumia 520",userAgent:"Mozilla/5.0 (compatible; MSIE 10.0; Windows Phone 8.0; Trident/6.0; IEMobile/10.0; ARM; Touch; NOKIA; Lumia 520)",screen:{width:320,height:533},viewport:{width:320,height:533},deviceScaleFactor:1.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Nokia N9":{name:"Nokia N9",userAgent:"Mozilla/5.0 (MeeGo; NokiaN9) AppleWebKit/534.13 (KHTML, like Gecko) NokiaBrowser/8.5.0 Mobile Safari/534.13",screen:{width:480,height:854},viewport:{width:480,height:854},deviceScaleFactor:1,isMobile:!0,hasTouch:!0,defaultBrowserType:"webkit"},"Pixel 2":{name:"Pixel 2",userAgent:"Mozilla/5.0 (Linux; Android 8.0; Pixel 2 Build/OPD3.170816.012) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:731},viewport:{width:411,height:731},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 2 XL":{name:"Pixel 2 XL",userAgent:"Mozilla/5.0 (Linux; Android 8.0.0; Pixel 2 XL Build/OPD1.170816.004) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:411,height:823},viewport:{width:411,height:823},deviceScaleFactor:3.5,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 3":{name:"Pixel 3",userAgent:"Mozilla/5.0 (Linux; Android 9; Pixel 3 Build/PQ1A.181105.017.A1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:786},viewport:{width:393,height:786},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4":{name:"Pixel 4",userAgent:"Mozilla/5.0 (Linux; Android 10; Pixel 4) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:353,height:745},viewport:{width:353,height:745},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 4a (5G)":{name:"Pixel 4a (5G)",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 4a (5G)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:892},viewport:{width:412,height:765},deviceScaleFactor:2.63,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 5":{name:"Pixel 5",userAgent:"Mozilla/5.0 (Linux; Android 11; Pixel 5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:393,height:851},viewport:{width:393,height:727},deviceScaleFactor:2.75,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Pixel 7":{name:"Pixel 7",userAgent:"Mozilla/5.0 (Linux; Android 14; Pixel 7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:412,height:915},viewport:{width:412,height:839},deviceScaleFactor:2.625,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Moto G4":{name:"Moto G4",userAgent:"Mozilla/5.0 (Linux; Android 7.0; Moto G (4)) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Mobile Safari/537.36",screen:{width:360,height:640},viewport:{width:360,height:640},deviceScaleFactor:3,isMobile:!0,hasTouch:!0,defaultBrowserType:"chromium"},"Desktop Chrome HiDPI":{name:"Desktop Chrome HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge HiDPI":{name:"Desktop Edge HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Firefox HiDPI":{name:"Desktop Firefox HiDPI",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"},"Desktop Safari":{name:"Desktop Safari",userAgent:"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.0 Safari/605.1.15",screen:{width:1792,height:1120},viewport:{width:1280,height:720},deviceScaleFactor:2,isMobile:!1,hasTouch:!1,defaultBrowserType:"webkit"},"Desktop Chrome":{name:"Desktop Chrome",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome Medium Resolution":{name:"Desktop Chrome Medium Resolution",displayName:"Playwright Chromium",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Chrome (Branded)":{name:"Desktop Chrome (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Chrome Medium Resolution (Branded)":{name:"Desktop Chrome Medium Resolution (Branded)",displayName:"Google Chrome",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"chrome"},"Desktop Edge":{name:"Desktop Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium"},"Desktop Edge (Branded)":{name:"Desktop Edge (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1920,height:1080},viewport:{width:1920,height:1080},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Edge Medium Resolution (Branded)":{name:"Desktop Edge Medium Resolution (Branded)",displayName:"Microsoft Edge",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/141.0.7390.16 Safari/537.36 Edg/141.0.7390.16",screen:{width:1280,height:720},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"chromium",channel:"msedge"},"Desktop Firefox":{name:"Desktop Firefox",userAgent:"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:142.0.1) Gecko/20100101 Firefox/142.0.1",screen:{width:1920,height:1080},viewport:{width:1280,height:720},deviceScaleFactor:1,isMobile:!1,hasTouch:!1,defaultBrowserType:"firefox"}},Wr={desktop:["Desktop Chrome","Desktop Chrome Medium Resolution","Desktop Chrome (Branded)","Desktop Chrome Medium Resolution (Branded)","Desktop Edge (Branded)","Desktop Edge Medium Resolution (Branded)","Desktop Safari"],mobile:["iPhone 15 Pro Max","iPhone 15 Pro","iPhone 15 Plus","iPhone 15","iPhone 14 Pro Max","iPhone 14 Pro","iPhone 14 Plus","iPhone 14","iPhone 13 Pro Max","iPhone 13 Pro","iPhone 13","iPhone 13 Mini","iPhone 12 Pro Max","iPhone 12 Pro","iPhone 12","iPhone 12 Mini","iPhone 11 Pro Max","iPhone 11 Pro","iPhone 11","iPhone XR","iPhone X","iPhone SE (3rd gen)","iPhone SE","iPhone 8 Plus","iPhone 8","iPhone 7 Plus","iPhone 7","iPhone 6 Plus","iPhone 6","Galaxy S24","Galaxy A55","Galaxy S9+","Galaxy S8","Galaxy S5","Galaxy Note 3","Galaxy Note II","Galaxy S III","Pixel 7","Pixel 5","Pixel 4a (5G)","Pixel 4","Pixel 3","Pixel 2 XL","Pixel 2","Nexus 6P","Nexus 6","Nexus 5X","Nexus 5","Nexus 4","Moto G4","LG Optimus L70","Microsoft Lumia 950","Microsoft Lumia 550","Nokia Lumia 520","Nokia N9","BlackBerry Z30"]},Me=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Wr[e].map(s=>Hr[s]).filter(s=>s.defaultBrowserType&&r.includes(s.defaultBrowserType))},io={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile")}},ao={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile",!0)}}});var Kr=_(()=>{"use strict"});function vt(){return{version:"1.0",entries:{}}}var zr=_(()=>{"use strict";Ke()});var Z,_t,Vr=_(()=>{"use strict";Z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(Z||{}),_t=18e4});var Yr=_(()=>{"use strict"});var Jr=_(()=>{"use strict"});var Xr=_(()=>{"use strict"});var qr=_(()=>{"use strict";ze()});var de=_(()=>{"use strict";gr();mr();yr();wr();Lr();Fr();We();yt();Ur();Gr();Kr();zr();Ke();Vr();Yr();Jr();Xr();qr();ze()});import{stringify as lo}from"yaml";import{createHash as Ao}from"crypto";import{parse as $o,stringify as as}from"yaml";import{readFileSync as Io,existsSync as Mo}from"fs";import{resolve as Tt,dirname as Oo}from"path";import{parse as ts,stringify as Lo}from"yaml";import{readFileSync as Uo,writeFileSync as Bo,mkdirSync as Ho}from"fs";import{dirname as Wo}from"path";function le(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function O(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function po(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 Ye(e){let t=po(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let s=uo(e);if(s){let n=JSON.stringify(s);return`${t}.locator(${n}).first()`}return null}function Zr(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:fo.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:!ho.includes(t)}function v(e,t){N.set(e,t)}function mo(e){return N.get(e)}function ge(e,t,r=[]){let s=[...r];return t.locator?s.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&s.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&s.push(`frame_path: ${JSON.stringify(t.frame_path)}`),s.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...s.map(n=>` ${n},`),"});"]}function Qr(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 s=["page","testContext","request","agent"],n=["undefined","null","true","false"],o=r.map(i=>s.includes(i)||n.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function re(e,t,r,s="main"){let n=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${s}.${o}`,c=yo(i,t,a,r);c.length>0&&(n.push(...c),o<e.length-1&&n.push(""))}return n}function yo(e,t,r,s){let n=" ".repeat(t);switch(e.type){case"DRAFT":return wo(e,t,r,s);case"ACTION":return bo(e,t,r,s);case"STEP":return So(e,t,r,s);case"IF_ELSE":return vo(e,t,r,s);case"WHILE_LOOP":return _o(e,t,r,s);default:return[`${n}// Unknown statement type: ${e.type}`]}}function wo(e,t,r,s){let n=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${i}, '${r}');`]}function bo(e,t,r,s){let n=" ".repeat(t),o=e.description,i=e.uid,c=s.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let y=JSON.stringify(o),E=!!e.use_pure_vision;return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${y}, '${r}', ${E});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let u=l.action_data?.action_name||"",d=l.action_description||"",h=mo(u);if(!h)return[`${n}// ${r}: Unknown action: ${u}`];let g={imports:s.imports},f=h(l,r,g);if(s.noAgent){if(Zr(l))return[`${n}// ${r}: ${O(d)}`,`${n}// AI action: ${O(d)} (requires agent - skipped in hook)`];let y=xo(l,u,n,r);return y||[`${n}// ${r}: ${O(d)}`,...f.map(E=>`${n}${E}`)]}if(Zr(l))return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,...f.map(y=>`${n}${y}`)];let p=JSON.stringify(d),S=f.map(y=>`${n} ${y}`),w=go(l),m=i?`'${i}'`:"undefined";return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...S,`${n}}, ${p}, '${r}', ${m}, ${w});`]}function So(e,t,r,s){let n=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${n}// Step: ${O(e.description)}`);let i=re(e.statements,t,s,r);return o.push(...i),o}function vo(e,t,r,s){let n=" ".repeat(t),o=[];if(o.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${n}if (${e.condition.expression}) {`);else{o.push(`${n}// AI Condition: ${O(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${n}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let i=re(e.then,t+1,s,`${r}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${n}} else {`);let a=re(e.else,t+1,s,`${r}.else`);o.push(...a)}return o.push(`${n}}`),o}function _o(e,t,r,s){let n=" ".repeat(t),o=[];o.push(`${n}// ${r}: Loop`);let i=e.timeout_ms??_t,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${n}const ${l}_start = Date.now();`),o.push(`${n}const ${l}_timeout = ${i};`),o.push(`${n}const ${l}_check = () => {`),o.push(`${n} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${n} throw new Error('${c}');`),o.push(`${n} }`),o.push(`${n} return true;`),o.push(`${n}};`),e.condition.type==="JS_CODE")o.push(`${n}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${n}// AI Loop Condition: ${O(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${n}while (${l}_check() && await agent.evaluate(page, ${d}, "${r}")) {`)}let u=re(e.body,t+1,s,`${r}.body`);return o.push(...u),o.push(`${n}}`),o}function xo(e,t,r,s){let n=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??Pt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${r}// ${s}: ${O(n)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goForward();`];case"input_text":{let a=o.text||"",c=Ye(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=Ye(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function To(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...os()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...es("beforeEach",t.beforeEach,o)),r.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=xt(t?.tags);for(let l of t.parameters){let u=ns(e,l.values);r.push(...Xe(u,`${c}${le(a)} [${le(l.name)}]`,o,0,i)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=xt(t?.tags);r.push(...Xe(e,`${c}${le(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...es("afterEach",t.afterEach,o))),is(r,n),r.join(`
208
+ `}async function Un(e){let{createServer:t}=await import("net");for(let r=e;r<e+20;r++)if(await new Promise(n=>{let o=t();o.once("error",()=>n(!1)),o.once("listening",()=>{o.close(()=>n(!0))}),o.listen(r,"127.0.0.1")}))return r;throw new Error(`No available port found in range ${e}-${e+19}`)}async function ft(e){let{yamlFilePath:t,configPath:r,tempSuffix:s="",headed:n}=e,o=te.dirname(r),i=await Un(16174),a;if(!ce.existsSync(t))throw new Error(`Please select a test file before starting the debug session. File not found: ${t}`);try{let y=jn(ce.readFileSync(t,"utf-8"));y?.use&&typeof y.use=="object"&&!Array.isArray(y.use)&&(a=y.use),y?.base_url&&!a?.baseURL&&(a={...a,baseURL:y.base_url}),y?.settings?.auto_dismiss_modal!==void 0&&(a={...a,autoDismissModal:!!y.settings.auto_dismiss_modal}),y?.settings?.browser_timezone!=null&&(a={...a,timezoneId:String(y.settings.browser_timezone)}),y?.settings?.browser_language!=null&&(a={...a,locale:String(y.settings.browser_language)}),y?.settings?.extra_http_headers!=null&&typeof y.settings.extra_http_headers=="object"&&(a={...a,extraHTTPHeaders:y.settings.extra_http_headers})}catch(y){console.error("[debugger] Could not parse YAML for `use` block:",y)}let c=te.dirname(te.resolve(t)),l=s?`-${s}`:"",u=te.join(c,`.__shiplight_debug__${l}.yaml.spec.ts`),d=Fn(t,i,a,o);ce.writeFileSync(u,d);let h=fr(u),g=["playwright","test",u,...n?["--headed"]:[]],f=Dn("npx",g,{stdio:["ignore","pipe","pipe"],shell:!0,cwd:o,env:{...process.env,PWDEBUG:"console",SHIPLIGHT_REGISTRY_URL:""}});f.stdout?.on("data",y=>{process.stderr.write(y)}),f.stderr?.on("data",y=>{process.stderr.write(y)});let p=()=>{f.killed||f.kill("SIGTERM")};process.on("SIGTERM",p),process.on("SIGINT",p),process.on("exit",p),f.on("close",y=>{process.removeListener("SIGTERM",p),process.removeListener("SIGINT",p),process.removeListener("exit",p),h(),y!==0&&y!==null&&console.error(`[debugger] Playwright process exited with code ${y}`)}),console.error("[debugger] Waiting for Playwright sandbox to start...");let S=["127.0.0.1","::1"];async function w(y){try{let E=y.includes(":")?`[${y}]`:y,T=await fetch(`http://${E}:${i}/api/test-flow`);if(T.ok){try{await T.text()}catch{}return!0}}catch{}return!1}let m=null;for(let y=0;y<180;y++){if(f.exitCode!==null)throw h(),new Error(`Playwright process exited with code ${f.exitCode} before sandbox was ready`);for(let E of S)if(await w(E)){m=E;break}if(m){console.error(`[debugger] Playwright sandbox ready on ${m}:${i}`);break}if(y===179)throw p(),h(),new Error("Timed out waiting for Playwright sandbox to start (180s)");await new Promise(E=>setTimeout(E,1e3))}if(!m)throw p(),h(),new Error("Sandbox poll finished without a reachable host");return{port:i,host:m,pid:f.pid??0,cleanup:async()=>{p(),h()}}}var ht=_(()=>{"use strict"});var gr=_(()=>{"use strict"});var mr=_(()=>{"use strict"});var yr=_(()=>{"use strict"});import{v4 as ua}from"uuid";var wr=_(()=>{"use strict"});import{z as b}from"zod";var br,gt,Sr,we,vr,_r,xr,B,Tr,Ue,mt,yt=_(()=>{"use strict";br=b.enum(["JS_CODE","AI_MODE"]),gt=b.object({type:br,expression:b.string()}),Sr=b.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),we=b.object({uid:b.string(),type:Sr,comment:b.string().optional()}),vr=b.object({action_data:b.object({action_name:b.string(),kwargs:b.record(b.any()).optional(),args:b.array(b.any()).optional()}),action_description:b.string().optional(),url:b.string().optional(),xpath:b.string().nullable().optional(),locator:b.string().nullable().optional(),css_selector:b.string().nullable().optional(),unique_selector:b.string().nullable().optional(),element_index:b.number().nullable().optional(),frame_path:b.array(b.any()).optional(),artifacts:b.record(b.any()).optional(),feedback:b.string().optional(),original_browser_use_action:b.any().optional()}).passthrough(),_r=we.extend({type:b.literal("DRAFT"),description:b.string()}),xr=we.extend({type:b.literal("ACTION"),description:b.string(),action_entity:vr.optional(),locator:b.string().optional(),use_pure_vision:b.boolean().optional()}),B=b.lazy(()=>b.union([_r,xr,we.extend({type:b.literal("STEP"),description:b.string().optional().default(""),statements:b.array(B),reference_id:b.number().optional(),template_path:b.string().optional(),template_params:b.record(b.string()).optional()}),we.extend({type:b.literal("IF_ELSE"),description:b.string().optional(),condition:gt,then:b.array(B),else:b.array(B).optional()}),we.extend({type:b.literal("WHILE_LOOP"),description:b.string().optional(),condition:gt,body:b.array(B),timeout_ms:b.number().optional()})])),Tr=b.object({name:b.string(),statements:b.array(B),teardown:b.array(B).optional(),skip:b.union([b.boolean(),b.string()]).optional(),timeout:b.number().optional(),fail:b.union([b.boolean(),b.string()]).optional(),only:b.boolean().optional(),slow:b.boolean().optional()}),Ue=b.object({tests:b.array(Tr).min(1),beforeAll:b.array(B).optional(),afterAll:b.array(B).optional(),beforeEach:b.array(B).optional(),afterEach:b.array(B).optional()}),mt=b.object({comment:b.string().optional(),version:b.string().optional(),goal:b.string().optional(),url:b.string().optional(),baseURL:b.string().optional(),final_feedback:b.string().optional(),completed:b.boolean().optional(),success:b.boolean().optional(),statements:b.array(B).optional(),teardown:b.array(B).optional(),last_modified_at:b.string().optional(),testGroup:Ue.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 Bn,parse as Ar,parseAllDocuments as ya,parseDocument as Hn,Document as Gn,isMap as Se,isSeq as q}from"yaml";import{v4 as X}from"uuid";function Be(e,t){let r={...t?.test_case_id!==void 0?{test_case_id:t.test_case_id}:{},...t?.name?{name:t.name}:{},...t?.tags&&t.tags.length>0?{tags:t.tags}:{},...t?.skip!==void 0?{skip:t.skip}:{},...t?.fail!==void 0?{fail:t.fail}:{},...t?.only?{only:t.only}:{},...t?.slow?{slow:t.slow}:{},goal:e.goal??"",url:e.url,base_url:e.baseURL,...t?.timeout!==void 0?{timeout:t.timeout}:{},...t?.settings&&Object.keys(t.settings).length>0?{settings:t.settings}:{},...t?.use&&Object.keys(t.use).length>0?{use:t.use}:{},...t?.beforeEach&&t.beforeEach.length>0?{beforeEach:t.beforeEach}:{},...t?.afterEach&&t.afterEach.length>0?{afterEach:t.afterEach}:{},...t?.parameters&&t.parameters.length>0?{parameters:t.parameters}:{},statements:(e.statements??[]).map(W)};return e.final_feedback&&(r.final_feedback=e.final_feedback),e.teardown&&e.teardown.length>0&&(r.teardown=e.teardown.map(W)),r}function $e(e,t){if(e.testGroup)return Ir(e,t);let r=Be(e,t),s=new Gn(r);return e.comment&&(s.commentBefore=e.comment),Pr(s,e.statements??[]),e.teardown&&Pr(s,e.teardown,"teardown"),s.toString($r)}function Pr(e,t,r="statements"){let s=e.contents;if(!s||!Se(s))return;let n=s.get(r,!0);q(n)&&Ae(n,t)}function Ae(e,t){for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=t[r],n=e.items[r];if(r>0&&(n.spaceBefore=!0),s.comment&&(r===0?e.commentBefore=s.comment:n.commentBefore=s.comment),Se(n)){let o=n;if(s.type==="STEP"){let i=o.get("statements",!0);q(i)&&Ae(i,s.statements)}else if(s.type==="IF_ELSE"){let i=o.get("THEN",!0);q(i)&&Ae(i,s.then);let a=o.get("ELSE",!0);q(a)&&s.else&&Ae(a,s.else)}else if(s.type==="WHILE_LOOP"){let i=o.get("DO",!0);q(i)&&Ae(i,s.body)}}}}function Ir(e,t){let r=e.testGroup;if(!r)throw new Error("suiteToYaml requires a TestFlow with testGroup");let s={};t?.test_case_id!==void 0&&(s.test_case_id=t.test_case_id),t?.name&&(s.name=t.name),t?.tags&&t.tags.length>0&&(s.tags=t.tags),t?.use&&Object.keys(t.use).length>0&&(s.use=t.use),t?.settings&&Object.keys(t.settings).length>0&&(s.settings=t.settings);let n={};return e.baseURL&&(n.base_url=e.baseURL),r.beforeAll&&r.beforeAll.length>0&&(n.beforeAll=r.beforeAll.map(W)),r.beforeEach&&r.beforeEach.length>0&&(n.beforeEach=r.beforeEach.map(W)),r.afterEach&&r.afterEach.length>0&&(n.afterEach=r.afterEach.map(W)),r.afterAll&&r.afterAll.length>0&&(n.afterAll=r.afterAll.map(W)),n.tests=r.tests.map(o=>{let i={name:o.name};return o.skip!==void 0&&(i.skip=o.skip),o.timeout!==void 0&&(i.timeout=o.timeout),o.fail!==void 0&&(i.fail=o.fail),o.only!==void 0&&(i.only=o.only),o.slow!==void 0&&(i.slow=o.slow),i.statements=o.statements.map(W),o.teardown&&o.teardown.length>0&&(i.teardown=o.teardown.map(W)),i}),s.suite=n,Bn(s,$r)}function W(e){switch(e.type){case"DRAFT":return Wn(e);case"ACTION":return Kn(e);case"STEP":return zn(e);case"IF_ELSE":return Vn(e);case"WHILE_LOOP":return Yn(e)}}function Wn(e){return{intent:e.description}}function Kn(e){let t=e.action_entity?.action_data?.action_name??e.action_entity?.action?.action_name,r=e.action_entity?.action_data?.kwargs??e.action_entity?.action?.kwargs;if(t==="verify"){let a=r?.statement;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c=r?.code;return typeof c=="string"&&c.trim()?{VERIFY:a,js:c}:{VERIFY:a}}}if(t==="go_to_url"){let a=r?.url;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath){let c={URL:a};return r?.new_tab===!0&&(c.new_tab=!0),typeof r?.timeout_seconds=="number"&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="js_action"){let a=r?.code;if(typeof a=="string"&&a.trim()&&e.description)return{description:e.description,js:a}}if(t==="ai_wait_until"){let a=r?.condition;if(typeof a=="string"){let c={WAIT_UNTIL:a};return typeof r?.timeout_seconds=="number"&&r.timeout_seconds!==60&&(c.timeout_seconds=r.timeout_seconds),c}}if(t==="wait"){let a=r?.seconds,l={WAIT:e.description||`Wait ${a}s`};return typeof a=="number"&&(l.seconds=a),l}if(t==="js_code"){let a=r?.code;if(typeof a=="string"&&!e.action_entity?.locator&&!e.action_entity?.xpath)return{description:e.description||"Code block",js:a}}if(!e.action_entity)return{intent:e.description};let s=e.action_entity.action_data??e.action_entity.action;if(!s)return{intent:e.description};let n={intent:e.description,action:s.action_name},o=e.locator??e.action_entity.locator;o&&(n.locator=o);let i=e.action_entity.xpath;if(i&&(n.xpath=i),e.use_pure_vision&&(n.use_pure_vision=!0),s.kwargs&&Object.keys(s.kwargs).length>0)for(let[a,c]of Object.entries(s.kwargs))a!=="uid"&&(a==="statement"&&(t==="ai_action"||t==="ai_step")||(n[a]=c));return s.args&&s.args.length>0&&(n.args=s.args),n}function zn(e){if(e.template_path){let r={template:e.template_path};return e.template_params&&Object.keys(e.template_params).length>0&&(r.params=e.template_params),r}let t={STEP:e.description,statements:e.statements.map(W)};return e.reference_id!==void 0&&(t.reference_id=e.reference_id),t}function Vn(e){let t={IF:Mr(e.condition),THEN:e.then.map(W)};return e.else&&e.else.length>0&&(t.ELSE=e.else.map(W)),t}function Yn(e){let t={WHILE:Mr(e.condition),DO:e.body.map(W)};return e.timeout_ms!==void 0&&(t.timeout_ms=e.timeout_ms),t}function Mr(e){return e.type==="JS_CODE"?`js:${e.expression}`:e.expression}function Ie(e){try{let t=Ar(e);if(!t||typeof t!="object")return{};let r={};return typeof t.test_case_id=="number"&&Number.isFinite(t.test_case_id)&&(r.test_case_id=t.test_case_id),typeof t.template_id=="number"&&Number.isFinite(t.template_id)&&(r.template_id=t.template_id),typeof t.name=="string"&&t.name.trim()&&(r.name=t.name.trim()),typeof t.timeout=="number"&&Number.isFinite(t.timeout)&&(r.timeout=t.timeout),t.settings&&typeof t.settings=="object"&&!Array.isArray(t.settings)&&(r.settings=t.settings),t.use&&typeof t.use=="object"&&!Array.isArray(t.use)&&(r.use=t.use),Array.isArray(t.tags)&&t.tags.every(s=>typeof s=="string")&&(r.tags=t.tags),(typeof t.skip=="boolean"||typeof t.skip=="string")&&(r.skip=t.skip),(typeof t.fail=="boolean"||typeof t.fail=="string")&&(r.fail=t.fail),t.only===!0&&(r.only=!0),t.slow===!0&&(r.slow=!0),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(r.beforeEach=t.beforeEach),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(r.afterEach=t.afterEach),Array.isArray(t.parameters)&&t.parameters.length>0&&(r.parameters=t.parameters),r}catch{return{}}}function wt(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(wt);let t=e,r=Object.keys(t);if(r.length===1){let n=r[0];if(n.startsWith("{ ")&&n.endsWith(" }")&&t[n]===null)return`{{${n.slice(2,-2)}}}`}let s={};for(let[n,o]of Object.entries(t))s[n]=wt(o);return s}function j(e){if(e.length>kr)throw new Error(`YAML input too large (${e.length} bytes, max ${kr})`);let t=wt(Ar(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Jn(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:K(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=K(t.teardown));let s=mt.safeParse(r);if(!s.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(s.error.errors)}`);let n=s.data;return He(e,n),n}function Jn(e){let t=e.suite;if(!t||typeof t!="object")throw new Error("Invalid suite: expected an object");let r=t.tests;if(!Array.isArray(r)||r.length===0)throw new Error('Suite must have a non-empty "tests" array');let n={tests:r.map(a=>{if(!a.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(a.statements)||a.statements.length===0)throw new Error(`Suite test "${a.name}" must have a non-empty "statements" array`);let c={name:a.name,statements:K(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(c.teardown=K(a.teardown)),a.skip!==void 0&&(c.skip=a.skip),typeof a.timeout=="number"&&(c.timeout=a.timeout),a.fail!==void 0&&(c.fail=a.fail),a.only===!0&&(c.only=!0),a.slow===!0&&(c.slow=!0),c})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(n.beforeAll=K(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(n.afterAll=K(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(n.beforeEach=K(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(n.afterEach=K(t.afterEach));let o=Ue.safeParse(n);if(!o.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(o.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:o.data}}function K(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Xn)}function Xn(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 qn(t);if("WHILE"in t)return Zn(t);if("STEP"in t)return Qn(t);if("VERIFY"in t){let r=t.VERIFY,s={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(s.code=t.js),{uid:X(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:s}}}}if("URL"in t){let r=t.URL,s=t.new_tab===!0?!0:void 0,n=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,o={url:typeof r=="string"?r:String(r)};return s&&(o.new_tab=!0),n!==void 0&&(o.timeout_seconds=n),{uid:X(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:o}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,s=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:X(),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:s}}}}}if("WAIT"in t){let r=t.WAIT,s=typeof t.seconds=="number"?t.seconds:3;return{uid:X(),type:"ACTION",description:typeof r=="string"?r:`Wait ${s}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${s}s`,action_data:{action_name:"wait",kwargs:{seconds:s}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');let s=typeof t.description=="string"?t.description:"Code block";return{uid:X(),type:"ACTION",description:s,action_entity:{action_description:s,action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&!("VERIFY"in t)&&!("action"in t)){if("intent"in t||"desc"in t)throw new Error("A `js:` statement uses `description:`, not `intent:`. Raw JS does not self-heal \u2014 use `description: + js:` for code, or express it as a structured action (`intent:` + `action:`/`locator:`) to keep self-healing.");let r=typeof t.description=="string"&&t.description.trim()!==""?t.description:"Code block",s=t.js;return{uid:X(),type:"ACTION",description:r,action_entity:{action_description:r,action_data:{action_name:"js_code",kwargs:{code:typeof s=="string"?s:String(s)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...s}=t;return Er({...s,action:"function",functionName:r})}if("action"in t)return Er(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:X(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function Or(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 qn(e){let t=Or(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let s={uid:X(),type:"IF_ELSE",condition:t,then:K(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(s.else=K(e.ELSE)),s}function Zn(e){let t=Or(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let s={uid:X(),type:"WHILE_LOOP",condition:t,body:K(r)};return typeof e.timeout_ms=="number"&&(s.timeout_ms=e.timeout_ms),s}function Qn(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:X(),type:"STEP",description:t,statements:K(e.statements)};if(typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),typeof e.template_path=="string"&&(r.template_path=e.template_path),e.template_params&&typeof e.template_params=="object"&&!Array.isArray(e.template_params)){let s=e.template_params,n={};for(let[o,i]of Object.entries(s))n[o]=String(i);r.template_params=n}return r}function Er(e){let t=typeof e.action=="string"?e.action:String(e.action),r=typeof e.intent=="string"?e.intent:typeof e.desc=="string"?e.desc:"",s=typeof e.locator=="string"?e.locator:void 0,n=typeof e.xpath=="string"?e.xpath:void 0,o=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,i={};for(let[l,u]of Object.entries(e))eo.has(l)||(i[l]=u);t==="verify"&&typeof i.js=="string"&&(i.code=i.js,delete i.js),(t==="ai_action"||t==="ai_step")&&i.statement===void 0&&(i.statement=r);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(i).length>0?i:{}}};s&&(a.locator=s),n&&(a.xpath=n);let c={uid:X(),type:"ACTION",description:r,action_entity:a};return o&&(c.use_pure_vision=!0),c}function He(e,t){let r;try{r=Hn(e)}catch{return}let s=r.contents;if(!s||!Se(s))return;if(r.commentBefore)t.comment=r.commentBefore;else{let c=s.items?.[0];c?.key&&c.key.commentBefore&&(t.comment=c.key.commentBefore)}let n=s,o=n.get("statements",!0);q(o)&&t.statements&&be(o,t.statements);let i=n.get("teardown",!0);q(i)&&t.teardown&&be(i,t.teardown)}function be(e,t){e.commentBefore&&t.length>0&&(t[0].comment=e.commentBefore);for(let r=0;r<Math.min(e.items.length,t.length);r++){let s=e.items[r];s.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=s.commentBefore);let n=t[r];if(n.type==="STEP"&&Se(s)){let o=s.get("statements",!0);q(o)&&be(o,n.statements)}else if(n.type==="IF_ELSE"&&Se(s)){let o=s.get("THEN",!0);q(o)&&be(o,n.then);let i=s.get("ELSE",!0);q(i)&&n.else&&be(i,n.else)}else if(n.type==="WHILE_LOOP"&&Se(s)){let o=s.get("DO",!0);q(o)&&be(o,n.body)}}}var $r,kr,eo,Ge=_(()=>{"use strict";yt();$r={lineWidth:120,defaultKeyType:"PLAIN",defaultStringType:"PLAIN"};kr=1024*1024;eo=new Set(["action","intent","desc","locator","xpath","use_pure_vision"])});import{parse as _a,stringify as xa}from"yaml";var Lr=_(()=>{"use strict";Ge()});var bt,We,Ke=_(()=>{"use strict";bt=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},We=e=>{let t=[],r=s=>{for(let n of s){t.push(n);let o=bt(n);for(let i of o)r(i.statements)}};return r(e),t}});function jr(e){let t=0,r=0;for(let s of e)if(s.type==="DRAFT")r++;else if(s.type==="ACTION"){let n=s.action_entity?.action_data?.action_name??"";Dr.has(n)||t++}return{action:t,draft:r}}function ro(e){try{return new Function(`return async function() { ${e} }`),null}catch(t){return t.message}}function Nr(e){try{return new Function(`return async function() { return (${e}) }`),null}catch(t){return t.message}}function so(e){let t=e.split(/\r?\n/).map(o=>o.trim()).filter(o=>o.length>0&&!o.startsWith("//")),r=t.length,s=t.join(`
209
+ `).match(/\bawait\b/g),n=s?s.length:0;return r<=Rr&&n<=Cr?null:`${r} non-blank line(s), ${n} await(s) \u2014 limits are ${Rr} lines and ${Cr} awaits. The VERIFY js: field is a cache for one natural-language assertion, not a place for multi-step logic. Break this into multiple statements \u2014 one VERIFY per assertion \u2014 so each step is visible in the debugger and self-healable. For freeform setup code (mocking, storage), use the description: + js: escape hatch, which has no complexity cap.`}function St(e,t){let r=t?.coverageThreshold??to,s=[],n=[],o;try{o=j(e)}catch(h){return{valid:!1,errors:[`Invalid YAML: ${h.message}`],warnings:[],stats:{total:0,action:0,draft:0,coverage:0}}}o.goal||s.push('Missing required field: "goal"'),o.statements?.length||s.push('Missing required field: "statements"');let i=[...We(o.statements??[]),...o.teardown?We(o.teardown):[]],{action:a,draft:c}=jr(i),l="Hint: in YAML double-quoted strings, backslashes are escape characters \u2014 use \\\\/ instead of \\/ for regex, or use single quotes.";for(let h of i){let g=h;if(g.reference_id!==void 0){let f=g.description||h.uid;s.push(`Unresolved cloud template reference on statement "${f}" (reference_id: ${g.reference_id}). Local YAML tests cannot use reference_id \u2014 inline the template statements or use the local "template:" key instead.`)}if(h.type==="ACTION"){let f=h,p=f.action_entity?.action_data?.action_name??"";if(p==="js_code"||p==="js_action"||p==="verify"||p==="ai_assert"){let S=f.action_entity?.action_data?.kwargs?.code;if(typeof S=="string"){let w=ro(S);if(w){let m=f.description||p;s.push(`Invalid JS in "${m}": ${w}. ${l}`)}else if(p==="verify"){let m=so(S);if(m){let y=f.description||p;s.push(`JS cache for "${y}" is too complex: ${m}`)}}}}}if(h.type==="IF_ELSE"){let f=h;if(f.condition.type==="JS_CODE"){let p=Nr(f.condition.expression);p&&s.push(`Invalid JS in IF condition "${f.condition.expression}": ${p}. ${l}`)}}if(h.type==="WHILE_LOOP"){let f=h;if(f.condition.type==="JS_CODE"){let p=Nr(f.condition.expression);p&&s.push(`Invalid JS in WHILE condition "${f.condition.expression}": ${p}. ${l}`)}}}let u=a+c,d=u>0?Math.round(a/u*100):0;return u>0&&d/100<r&&n.push(`Low action coverage: ${a}/${u} statements (${d}%) are enriched with action/js. ${c} draft statement(s) still need enrichment. Use MCP tools (act, get_locators) to convert drafts to actions.`),{valid:s.length===0,errors:s,warnings:n,stats:{total:u,action:a,draft:c,coverage:d}}}var to,Rr,Cr,Dr,Fr=_(()=>{"use strict";Ge();Ke();to=.5,Rr=5,Cr=3,Dr=new Set(["verify","ai_assert","done","go_to_url","ai_wait_until","wait","js_code"])});var ze=_(()=>{"use strict"});var Ur=_(()=>{"use strict";ze()});var Br,oo,Hr,Gr,Me,io,ao,Wr=_(()=>{"use strict";Br=112,oo=1080-Br,Hr={"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"}},Gr={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"]},Me=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Gr[e].map(s=>Hr[s]).filter(s=>s.defaultBrowserType&&r.includes(s.defaultBrowserType))},io={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile")}},ao={desktop:{label:"Desktop",type:"desktop",devices:Me("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:Me("mobile",!0)}}});var Kr=_(()=>{"use strict"});function vt(){return{version:"1.0",entries:{}}}var zr=_(()=>{"use strict";Ke()});var Z,_t,Vr=_(()=>{"use strict";Z=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(Z||{}),_t=18e4});var Yr=_(()=>{"use strict"});var Jr=_(()=>{"use strict"});var Xr=_(()=>{"use strict"});var qr=_(()=>{"use strict";ze()});var de=_(()=>{"use strict";gr();mr();yr();wr();Lr();Fr();Ge();yt();Ur();Wr();Kr();zr();Ke();Vr();Yr();Jr();Xr();qr();ze()});import{stringify as lo}from"yaml";import{createHash as Ao}from"crypto";import{parse as $o,stringify as as}from"yaml";import{readFileSync as Io,existsSync as Mo}from"fs";import{resolve as Tt,dirname as Oo}from"path";import{parse as ts,stringify as Lo}from"yaml";import{readFileSync as Uo,writeFileSync as Bo,mkdirSync as Ho}from"fs";import{dirname as Go}from"path";function le(e){return e.replace(/\\/g,"\\\\").replace(/'/g,"\\'").replace(/\n/g,"\\n").replace(/\r/g,"\\r").replace(/\t/g,"\\t")}function O(e){return e.replace(/\r\n/g," ").replace(/\n/g," ").replace(/\r/g," ").trim()}function po(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 Ye(e){let t=po(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let s=uo(e);if(s){let n=JSON.stringify(s);return`${t}.locator(${n}).first()`}return null}function Zr(e){let t=e.action_data?.action_name;return!t||(t==="verify"||t==="ai_assert"||t==="assert")&&e.action_data?.kwargs?.code?!1:fo.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:!ho.includes(t)}function v(e,t){N.set(e,t)}function mo(e){return N.get(e)}function ge(e,t,r=[]){let s=[...r];return t.locator?s.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&s.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&s.push(`frame_path: ${JSON.stringify(t.frame_path)}`),s.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...s.map(n=>` ${n},`),"});"]}function Qr(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 s=["page","testContext","request","agent"],n=["undefined","null","true","false"],o=r.map(i=>s.includes(i)||n.includes(i)||/^-?\d+(\.\d+)?$/.test(i)?i:i.startsWith("$")?`agent.agentServices.readVariable('${i.substring(1)}')`:`"${i}"`);return`await ${t}(${o.join(", ")})`}function re(e,t,r,s="main"){let n=[];for(let o=0;o<e.length;o++){let i=e[o],a=`${s}.${o}`,c=yo(i,t,a,r);c.length>0&&(n.push(...c),o<e.length-1&&n.push(""))}return n}function yo(e,t,r,s){let n=" ".repeat(t);switch(e.type){case"DRAFT":return wo(e,t,r,s);case"ACTION":return bo(e,t,r,s);case"STEP":return So(e,t,r,s);case"IF_ELSE":return vo(e,t,r,s);case"WHILE_LOOP":return _o(e,t,r,s);default:return[`${n}// Unknown statement type: ${e.type}`]}}function wo(e,t,r,s){let n=" ".repeat(t),o=e.description?.trim()||"";if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let i=JSON.stringify(o);return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.run(page, ${i}, '${r}');`]}function bo(e,t,r,s){let n=" ".repeat(t),o=e.description,i=e.uid,c=s.actionEntityStore?.entries[e.uid]?.action_entity??e.action_entity;if(!c){if(!o)return[`${n}// ${r}: Skipping - no description`];if(s.noAgent)return[`${n}// ${r}: ${O(o)}`,`${n}// DRAFT: ${O(o)} (requires agent - skipped in hook)`];let y=JSON.stringify(o),E=!!e.use_pure_vision;return[`${n}// ${r}: ${O(o)}`,`${n}// \u26A0 DRAFT: AI-resolved at runtime (~5-10s). Add a locator to make this <1s.`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.execute(page, ${y}, '${r}', ${E});`]}let l=e.locator?{...c,locator:e.locator}:c;o&&o!==l.action_description&&(l={...l,action_description:o});let u=l.action_data?.action_name||"",d=l.action_description||"",h=mo(u);if(!h)return[`${n}// ${r}: Unknown action: ${u}`];let g={imports:s.imports},f=h(l,r,g);if(s.noAgent){if(Zr(l))return[`${n}// ${r}: ${O(d)}`,`${n}// AI action: ${O(d)} (requires agent - skipped in hook)`];let y=xo(l,u,n,r);return y||[`${n}// ${r}: ${O(d)}`,...f.map(E=>`${n}${E}`)]}if(Zr(l))return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,...f.map(y=>`${n}${y}`)];let p=JSON.stringify(d),S=f.map(y=>`${n} ${y}`),w=go(l),m=i?`'${i}'`:"undefined";return[`${n}// ${r}: ${O(d)}`,`${n}page = agent.agentServices.validatePage(page);`,`${n}await agent.step(page, async () => {`,...S,`${n}}, ${p}, '${r}', ${m}, ${w});`]}function So(e,t,r,s){let n=" ".repeat(t),o=[];e.description&&e.description.trim()&&o.push(`${n}// Step: ${O(e.description)}`);let i=re(e.statements,t,s,r);return o.push(...i),o}function vo(e,t,r,s){let n=" ".repeat(t),o=[];if(o.push(`${n}// ${r}: Conditional check`),e.condition.type==="JS_CODE")o.push(`${n}if (${e.condition.expression}) {`);else{o.push(`${n}// AI Condition: ${O(e.condition.expression)}`);let a=JSON.stringify(e.condition.expression);o.push(`${n}if (await agent.evaluate(page, ${a}, "${r}")) {`)}let i=re(e.then,t+1,s,`${r}.then`);if(o.push(...i),e.else&&e.else.length>0){o.push(`${n}} else {`);let a=re(e.else,t+1,s,`${r}.else`);o.push(...a)}return o.push(`${n}}`),o}function _o(e,t,r,s){let n=" ".repeat(t),o=[];o.push(`${n}// ${r}: Loop`);let i=e.timeout_ms??_t,a=i/1e3,c=e.timeout_ms?`While loop exceeded timeout of ${a}s`:`While loop exceeded default timeout of ${a}s`,l=`loop_${r.replace(/\./g,"_")}`;if(o.push(`${n}const ${l}_start = Date.now();`),o.push(`${n}const ${l}_timeout = ${i};`),o.push(`${n}const ${l}_check = () => {`),o.push(`${n} if (Date.now() - ${l}_start > ${l}_timeout) {`),o.push(`${n} throw new Error('${c}');`),o.push(`${n} }`),o.push(`${n} return true;`),o.push(`${n}};`),e.condition.type==="JS_CODE")o.push(`${n}while (${l}_check() && (${e.condition.expression})) {`);else{o.push(`${n}// AI Loop Condition: ${O(e.condition.expression)}`);let d=JSON.stringify(e.condition.expression);o.push(`${n}while (${l}_check() && await agent.evaluate(page, ${d}, "${r}")) {`)}let u=re(e.body,t+1,s,`${r}.body`);return o.push(...u),o.push(`${n}}`),o}function xo(e,t,r,s){let n=e.action_description||"",o=e.action_data?.kwargs||{},i=o.timeout_ms??Pt;switch(t){case"go_to_url":case"open_tab":{let a=o.url||"";return[`${r}// ${s}: ${O(n)}`,`${r}await page.goto(${JSON.stringify(a)}, { waitUntil: 'domcontentloaded' });`]}case"go_back":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goBack();`];case"go_forward":return[`${r}// ${s}: ${O(n)}`,`${r}await page.goForward();`];case"input_text":{let a=o.text||"",c=Ye(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.fill(${JSON.stringify(a)}, { timeout: ${i} });`]:null}case"select_dropdown_option":{let a=o.text||o.label||"",c=Ye(e);return c?[`${r}// ${s}: ${O(n)}`,`${r}await ${c}.selectOption({ label: ${JSON.stringify(a)} }, { timeout: ${i} });`]:null}default:return null}}function To(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...os()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore};t?.beforeEach&&t.beforeEach.length>0&&(r.push(...es("beforeEach",t.beforeEach,o)),r.push(""));let i=t?.timeout||t?.skip!==void 0||t?.fail!==void 0||t?.only||t?.slow?{timeout:t.timeout,skip:t.skip,fail:t.fail,only:t.only,slow:t.slow}:void 0;if(t?.parameters&&t.parameters.length>0){let a=t?.testName||e.goal||"Generated test",c=xt(t?.tags);for(let l of t.parameters){let u=ns(e,l.values);r.push(...Xe(u,`${c}${le(a)} [${le(l.name)}]`,o,0,i)),r.push("")}}else{let a=t?.testName||e.goal||"Generated test",c=xt(t?.tags);r.push(...Xe(e,`${c}${le(a)}`,o,0,i))}return t?.afterEach&&t.afterEach.length>0&&(r.push(""),r.push(...es("afterEach",t.afterEach,o))),is(r,n),r.join(`
210
210
  `)}function Po(e,t){let r=[],s=t?.version||"unknown";r.push(`// @generated by shiplightai v${s}`),r.push(...os()),r.push(""),t?.use&&Object.keys(t.use).length>0&&(r.push(`test.use(${JSON.stringify(t.use,null,2)});`),r.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},i=t?.testName||"Test Suite",a=xt(t?.tags);r.push(`test.describe.serial('${a}${le(i)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(r.push(...Ve("beforeAll",e.beforeAll,o,1)),r.push("")),e.beforeEach&&e.beforeEach.length>0&&(r.push(...Ve("beforeEach",e.beforeEach,o,1)),r.push(""));for(let l=0;l<e.tests.length;l++){let u=e.tests[l],d=u.timeout||u.skip!==void 0||u.fail!==void 0||u.only||u.slow?{timeout:u.timeout,skip:u.skip,fail:u.fail,only:u.only,slow:u.slow}:void 0;if(u.parameters&&u.parameters.length>0)for(let h of u.parameters){let g=ns(u.testFlow,h.values);r.push(...Xe(g,`${le(u.name)} [${le(h.name)}]`,o,1,d)),r.push("")}else r.push(...Xe(u.testFlow,le(u.name),o,1,d)),(l<e.tests.length-1||e.afterEach||e.afterAll)&&r.push("")}return e.afterEach&&e.afterEach.length>0&&(r.push(...Ve("afterEach",e.afterEach,o,1)),r.push("")),e.afterAll&&e.afterAll.length>0&&r.push(...Ve("afterAll",e.afterAll,o,1)),r.push("});"),is(r,n),r.join(`
211
- `)}function xt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Je(e){let t=new Set;function r(s){for(let n of s)switch(n.type){case Z.ACTION:{let i=n.action_entity?.action_data?.kwargs;if(i?.args&&Array.isArray(i.args))for(let a of i.args)typeof a=="string"&&ko.includes(a)&&t.add(a);break}case Z.STEP:r(n.statements);break;case Z.IF_ELSE:{let o=n;r(o.then),o.else&&r(o.else);break}case Z.WHILE_LOOP:r(n.body);break}}return r(e),t}function Eo(e){let t=Je(e.statements??[]);if(e.teardown)for(let r of Je(e.teardown))t.add(r);return t}function kt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Xe(e,t,r,s=0,n){let o=" ".repeat(s),i=[],a=Eo(e),c=kt(a),l=n?.only?"test.only":"test";i.push(`${o}${l}('${t}', async (${c}) => {`),n?.skip===!0?i.push(`${o} test.skip();`):typeof n?.skip=="string"&&i.push(`${o} test.skip(true, '${le(n.skip)}');`),n?.fail===!0?i.push(`${o} test.fail();`):typeof n?.fail=="string"&&i.push(`${o} test.fail(true, '${le(n.fail)}');`),n?.slow&&i.push(`${o} test.slow();`),n?.timeout&&i.push(`${o} test.setTimeout(${n.timeout});`);let u=e.teardown&&e.teardown.length>0,d=s+1;if(u){if(i.push(`${o} try {`),e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let g=re(e.statements,d+1,r);i.push(...g)}i.push(`${o} } finally {`),i.push(`${o} // Teardown`);let h=re(e.teardown,d+1,r,"teardown");i.push(...h),i.push(`${o} }`)}else if(e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let h=re(e.statements,d,r);i.push(...h)}return i.push(`${o}});`),i}function es(e,t,r){let s=[],n=ss(t),o=Je(n),i=kt(o);return s.push(`test.${e}(async (${i}) => {`),s.push(...re(n,1,r,e)),s.push("});"),s}function Ve(e,t,r,s){let n=" ".repeat(s),o=[],i=ss(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...re(i,s+1,c,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let c=Je(i),l=kt(c);o.push(`${n}test.${e}(async (${l}) => {`),o.push(...re(i,s+1,r,e)),o.push(`${n}});`)}return o}function ss(e){let r=lo({goal:"_hook",statements:e});return j(r).statements??[]}function ns(e,t){let r=$e(e);for(let[s,n]of Object.entries(t))r=r.split(`<<${s}>>`).join(String(n));return j(r)}function os(){return["import { test, expect } from 'shiplightai/fixture';"]}function is(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let s=Array.from(t);e.splice(r,0,...s)}}function cs(e,t,r){let s={expandingPaths:new Set([Tt(t)]),depth:0,referencedPaths:new Set,basePath:r},n={...e};Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,s)),Array.isArray(n.teardown)&&(n.teardown=fe(n.teardown,t,s));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=fe(n[o],t,s));return{doc:n,referencedTemplatePaths:Array.from(s.referencedPaths)}}function fe(e,t,r){let s=[];for(let n of e)if(Ro(n)){let o=Co(n,t,r);s.push(o)}else s.push(No(n,t,r));return s}function Ro(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Co(e,t,r){if(r.depth>=rs)throw new Error(`Template expansion exceeded maximum depth of ${rs}. Check for deeply nested or circular template references.`);let s=Tt(Oo(t),e.template),n=!Mo(s)&&r.basePath?Tt(r.basePath,e.template):s;if(r.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);r.referencedPaths.add(n);let o;try{o=Io(n,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${f.message}`)}let i=ts(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let a=i.params||[],c=e.params||{};for(let f of a)if(!(f in c))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${a.join(", ")}]`);let l=i.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let p=Lo(l);for(let[S,w]of Object.entries(c))p=p.split(`<<${S}>>`).join(String(w));l=ts(p)}let u={expandingPaths:new Set([...r.expandingPaths,n]),depth:r.depth+1,referencedPaths:r.referencedPaths},d=fe(l,n,u),g={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(c).length>0&&(g.template_params=c),g}function No(e,t,r){if(typeof e!="object"||e===null)return e;let s={...e};return Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,r)),Array.isArray(s.THEN)&&(s.THEN=fe(s.THEN,t,r)),Array.isArray(s.ELSE)&&(s.ELSE=fe(s.ELSE,t,r)),Array.isArray(s.DO)&&(s.DO=fe(s.DO,t,r)),s}function At(e,t,r){let s=$o(e),n=s?.name,o=s?.tags,i=s?.use;if(s&&(s.name!==void 0||s.tags!==void 0||s.use!==void 0)&&(delete s.name,delete s.tags,delete s.use),s?.suite){if(s.goal||s.statements)throw new Et('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return jo(s,n,o,i,t,r)}return Do(s,n,o,i,t,r)}function Do(e,t,r,s,n,o){let i=e?.beforeEach,a=e?.afterEach,c=ls(e?.parameters),l=e?.timeout,u=e?.skip,d=e?.fail,h=e?.only,g=e?.slow,f=e?.settings,p=s;if(f){let y={};f.auto_dismiss_modal!==void 0&&(y.autoDismissModal=!!f.auto_dismiss_modal),f.browser_timezone!==void 0&&f.browser_timezone!==null&&(y.timezoneId=String(f.browser_timezone)),f.browser_language!==void 0&&f.browser_language!==null&&(y.locale=String(f.browser_language)),f.extra_http_headers!==void 0&&f.extra_http_headers!==null&&typeof f.extra_http_headers=="object"&&(y.extraHTTPHeaders=f.extra_http_headers),Object.keys(y).length>0&&(p={...p,...y})}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,delete e.settings),e?.url)throw new Et(`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 S=[];if(n&&e&&typeof e=="object"){let y=cs(e,n,o);e=y.doc,S=y.referencedTemplatePaths}let w=as(e),m=j(w);return n&&(ve(m.statements??[],n,"main"),m.teardown&&ve(m.teardown,n,"teardown")),{testFlow:m,name:t,tags:r,use:p,beforeEach:i,afterEach:a,parameters:c,timeout:l,skip:u,fail:d,only:h,slow:g,referencedTemplatePaths:S}}function jo(e,t,r,s,n,o){let i=e.suite;if(!Array.isArray(i.tests)||i.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=i.beforeAll,c=i.afterAll,l=i.beforeEach,u=i.afterEach,d=[],h=i.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let S={goal:p.name,statements:p.statements};p.teardown&&(S.teardown=p.teardown);let w=[],m=S;if(n&&typeof S=="object"){let $=cs(S,n,o);m=$.doc,w=$.referencedTemplatePaths,d.push(...w)}let y=as(m),E=j(y),T=ls(p.parameters);return{testFlow:E,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:T,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=i.base_url,f=g?{...s,baseURL:g}:s;return{suite:{beforeAll:a,afterAll:c,beforeEach:l,afterEach:u,tests:h},name:t,tags:r,use:f,referencedTemplatePaths:d}}function ls(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 ve(e,t,r){for(let s=0;s<e.length;s++){let n=e[s],o=`${r}.${s}`,i=n.description||"";if(n.uid=Fo(t,o,i),n.type===Z.STEP)ve(n.statements,t,o);else if(n.type===Z.IF_ELSE){let a=n;ve(a.then,t,`${o}.then`),a.else&&ve(a.else,t,`${o}.else`)}else n.type===Z.WHILE_LOOP&&ve(n.body,t,`${o}.body`)}}function Fo(e,t,r){let s=Ao("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${s.slice(0,8)}-${s.slice(8,12)}-${s.slice(12,16)}-${s.slice(16,20)}-${s.slice(20,32)}`}function ps(e,t){let r;try{r=Uo(e,"utf-8")}catch(s){return{valid:!1,errors:[`Failed to read file: ${s.message}`],warnings:[]}}return Go(r,e,t)}function Go(e,t,r){let s=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=s||n?null:St(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,c=[];try{let l=r?.parsed??At(e,t);c=l.referencedTemplatePaths;let u={version:r?.version,actionEntityStore:r?.actionEntityStore},d=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?i=Po(l.suite,{...u,testName:l.name,tags:l.tags,use:l.use}):i=To(l.testFlow,{...u,testName:l.name,tags:l.tags,use:d,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let h=i.split(`
211
+ `)}function xt(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}function Je(e){let t=new Set;function r(s){for(let n of s)switch(n.type){case Z.ACTION:{let i=n.action_entity?.action_data?.kwargs;if(i?.args&&Array.isArray(i.args))for(let a of i.args)typeof a=="string"&&ko.includes(a)&&t.add(a);break}case Z.STEP:r(n.statements);break;case Z.IF_ELSE:{let o=n;r(o.then),o.else&&r(o.else);break}case Z.WHILE_LOOP:r(n.body);break}}return r(e),t}function Eo(e){let t=Je(e.statements??[]);if(e.teardown)for(let r of Je(e.teardown))t.add(r);return t}function kt(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Xe(e,t,r,s=0,n){let o=" ".repeat(s),i=[],a=Eo(e),c=kt(a),l=n?.only?"test.only":"test";i.push(`${o}${l}('${t}', async (${c}) => {`),n?.skip===!0?i.push(`${o} test.skip();`):typeof n?.skip=="string"&&i.push(`${o} test.skip(true, '${le(n.skip)}');`),n?.fail===!0?i.push(`${o} test.fail();`):typeof n?.fail=="string"&&i.push(`${o} test.fail(true, '${le(n.fail)}');`),n?.slow&&i.push(`${o} test.slow();`),n?.timeout&&i.push(`${o} test.setTimeout(${n.timeout});`);let u=e.teardown&&e.teardown.length>0,d=s+1;if(u){if(i.push(`${o} try {`),e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let g=re(e.statements,d+1,r);i.push(...g)}i.push(`${o} } finally {`),i.push(`${o} // Teardown`);let h=re(e.teardown,d+1,r,"teardown");i.push(...h),i.push(`${o} }`)}else if(e.statements&&e.statements.length>0){i.push(`${o} // Test steps`);let h=re(e.statements,d,r);i.push(...h)}return i.push(`${o}});`),i}function es(e,t,r){let s=[],n=ss(t),o=Je(n),i=kt(o);return s.push(`test.${e}(async (${i}) => {`),s.push(...re(n,1,r,e)),s.push("});"),s}function Ve(e,t,r,s){let n=" ".repeat(s),o=[],i=ss(t);if(e==="beforeAll"||e==="afterAll"){let c={...r,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...re(i,s+1,c,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let c=Je(i),l=kt(c);o.push(`${n}test.${e}(async (${l}) => {`),o.push(...re(i,s+1,r,e)),o.push(`${n}});`)}return o}function ss(e){let r=lo({goal:"_hook",statements:e});return j(r).statements??[]}function ns(e,t){let r=$e(e);for(let[s,n]of Object.entries(t))r=r.split(`<<${s}>>`).join(String(n));return j(r)}function os(){return["import { test, expect } from 'shiplightai/fixture';"]}function is(e,t){if(t.size>0){let r=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(r=n+1);let s=Array.from(t);e.splice(r,0,...s)}}function cs(e,t,r){let s={expandingPaths:new Set([Tt(t)]),depth:0,referencedPaths:new Set,basePath:r},n={...e};Array.isArray(n.statements)&&(n.statements=fe(n.statements,t,s)),Array.isArray(n.teardown)&&(n.teardown=fe(n.teardown,t,s));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=fe(n[o],t,s));return{doc:n,referencedTemplatePaths:Array.from(s.referencedPaths)}}function fe(e,t,r){let s=[];for(let n of e)if(Ro(n)){let o=Co(n,t,r);s.push(o)}else s.push(No(n,t,r));return s}function Ro(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Co(e,t,r){if(r.depth>=rs)throw new Error(`Template expansion exceeded maximum depth of ${rs}. Check for deeply nested or circular template references.`);let s=Tt(Oo(t),e.template),n=!Mo(s)&&r.basePath?Tt(r.basePath,e.template):s;if(r.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);r.referencedPaths.add(n);let o;try{o=Io(n,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${f.message}`)}let i=ts(o);if(!i||typeof i!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let a=i.params||[],c=e.params||{};for(let f of a)if(!(f in c))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${a.join(", ")}]`);let l=i.statements;if(!Array.isArray(l))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let p=Lo(l);for(let[S,w]of Object.entries(c))p=p.split(`<<${S}>>`).join(String(w));l=ts(p)}let u={expandingPaths:new Set([...r.expandingPaths,n]),depth:r.depth+1,referencedPaths:r.referencedPaths},d=fe(l,n,u),g={STEP:i.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:d};return Object.keys(c).length>0&&(g.template_params=c),g}function No(e,t,r){if(typeof e!="object"||e===null)return e;let s={...e};return Array.isArray(s.statements)&&(s.statements=fe(s.statements,t,r)),Array.isArray(s.THEN)&&(s.THEN=fe(s.THEN,t,r)),Array.isArray(s.ELSE)&&(s.ELSE=fe(s.ELSE,t,r)),Array.isArray(s.DO)&&(s.DO=fe(s.DO,t,r)),s}function At(e,t,r){let s=$o(e),n=s?.name,o=s?.tags,i=s?.use;if(s&&(s.name!==void 0||s.tags!==void 0||s.use!==void 0)&&(delete s.name,delete s.tags,delete s.use),s?.suite){if(s.goal||s.statements)throw new Et('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return jo(s,n,o,i,t,r)}return Do(s,n,o,i,t,r)}function Do(e,t,r,s,n,o){let i=e?.beforeEach,a=e?.afterEach,c=ls(e?.parameters),l=e?.timeout,u=e?.skip,d=e?.fail,h=e?.only,g=e?.slow,f=e?.settings,p=s;if(f){let y={};f.auto_dismiss_modal!==void 0&&(y.autoDismissModal=!!f.auto_dismiss_modal),f.browser_timezone!==void 0&&f.browser_timezone!==null&&(y.timezoneId=String(f.browser_timezone)),f.browser_language!==void 0&&f.browser_language!==null&&(y.locale=String(f.browser_language)),f.extra_http_headers!==void 0&&f.extra_http_headers!==null&&typeof f.extra_http_headers=="object"&&(y.extraHTTPHeaders=f.extra_http_headers),Object.keys(y).length>0&&(p={...p,...y})}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,delete e.settings),e?.url)throw new Et(`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 S=[];if(n&&e&&typeof e=="object"){let y=cs(e,n,o);e=y.doc,S=y.referencedTemplatePaths}let w=as(e),m=j(w);return n&&(ve(m.statements??[],n,"main"),m.teardown&&ve(m.teardown,n,"teardown")),{testFlow:m,name:t,tags:r,use:p,beforeEach:i,afterEach:a,parameters:c,timeout:l,skip:u,fail:d,only:h,slow:g,referencedTemplatePaths:S}}function jo(e,t,r,s,n,o){let i=e.suite;if(!Array.isArray(i.tests)||i.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let a=i.beforeAll,c=i.afterAll,l=i.beforeEach,u=i.afterEach,d=[],h=i.tests.map(p=>{if(!p.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(p.statements)||p.statements.length===0)throw new Error(`Suite test "${p.name}" must have a non-empty "statements" array.`);let S={goal:p.name,statements:p.statements};p.teardown&&(S.teardown=p.teardown);let w=[],m=S;if(n&&typeof S=="object"){let $=cs(S,n,o);m=$.doc,w=$.referencedTemplatePaths,d.push(...w)}let y=as(m),E=j(y),T=ls(p.parameters);return{testFlow:E,name:p.name,tags:Array.isArray(p.tags)?p.tags:void 0,parameters:T,timeout:p.timeout,skip:p.skip,fail:p.fail,only:p.only,slow:p.slow}}),g=i.base_url,f=g?{...s,baseURL:g}:s;return{suite:{beforeAll:a,afterAll:c,beforeEach:l,afterEach:u,tests:h},name:t,tags:r,use:f,referencedTemplatePaths:d}}function ls(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 ve(e,t,r){for(let s=0;s<e.length;s++){let n=e[s],o=`${r}.${s}`,i=n.description||"";if(n.uid=Fo(t,o,i),n.type===Z.STEP)ve(n.statements,t,o);else if(n.type===Z.IF_ELSE){let a=n;ve(a.then,t,`${o}.then`),a.else&&ve(a.else,t,`${o}.else`)}else n.type===Z.WHILE_LOOP&&ve(n.body,t,`${o}.body`)}}function Fo(e,t,r){let s=Ao("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${s.slice(0,8)}-${s.slice(8,12)}-${s.slice(12,16)}-${s.slice(16,20)}-${s.slice(20,32)}`}function ps(e,t){let r;try{r=Uo(e,"utf-8")}catch(s){return{valid:!1,errors:[`Failed to read file: ${s.message}`],warnings:[]}}return Wo(r,e,t)}function Wo(e,t,r){let s=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=s||n?null:St(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let i,a,c=[];try{let l=r?.parsed??At(e,t);c=l.referencedTemplatePaths;let u={version:r?.version,actionEntityStore:r?.actionEntityStore},d=l.testFlow?.baseURL?{...l.use,baseURL:l.testFlow.baseURL}:l.use;l.suite?i=Po(l.suite,{...u,testName:l.name,tags:l.tags,use:l.use}):i=To(l.testFlow,{...u,testName:l.name,tags:l.tags,use:d,beforeEach:l.beforeEach,afterEach:l.afterEach,parameters:l.parameters,timeout:l.timeout,skip:l.skip,fail:l.fail,only:l.only,slow:l.slow});let h=i.split(`
212
212
  `).filter(g=>!g.startsWith("import ")).join(`
213
- `);new Function(h),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Ho(Wo(a),{recursive:!0}),Bo(a,i)}catch(l){let u=l instanceof Et?"":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}.${u}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var Pt,fo,ho,N,ko,rs,Et,$t=_(()=>{"use strict";de();de();de();de();Pt=5e3;fo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],ho=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];N=new Map;v("click",e=>{let t=Ye(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.click({ timeout: ${r} });`]});v("click_element",N.get("click"));v("click_element_by_index",N.get("click"));v("double_click",e=>ge("double_click",e));v("double_click_on_element",N.get("double_click"));v("right_click",e=>ge("right_click",e));v("right_click_on_element",N.get("right_click"));v("hover",e=>ge("hover",e));v("hover_element_by_index",N.get("hover"));v("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("fill",N.get("input_text"));v("clear_input",e=>ge("clear_input",e));v("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});v("send_keys",N.get("press"));v("send_keys_on_element",e=>{let t=Ye(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 s=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${s} });`]});v("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});v("scroll_down",N.get("scroll"));v("scroll_up",N.get("scroll"));v("scroll_element",N.get("scroll"));v("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});v("scroll_on_element",e=>ge("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));v("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});v("open_tab",N.get("go_to_url"));v("go_back",()=>['await agent.execAction("go_back", page, {});']);v("reload_page",()=>['await agent.execAction("reload_page", page, {});']);v("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);v("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);v("verify",(e,t)=>{let r=e.action_data?.kwargs,s=typeof r?.code=="string",n=s?r?.statement||e.action_description:e.action_description||r?.statement;if(s&&n){let i=r.code.split(`
213
+ `);new Function(h),a=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Ho(Go(a),{recursive:!0}),Bo(a,i)}catch(l){let u=l instanceof Et?"":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}.${u}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:c}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:a,referencedTemplatePaths:c}}var Pt,fo,ho,N,ko,rs,Et,$t=_(()=>{"use strict";de();de();de();de();Pt=5e3;fo=["ai_action","ai_step","ai_assert","ai_extract","ai_wait_until","verify","assert"],ho=["js_code","function","wait","wait_for_download_complete","wait_for_page_ready","extract_email_content","extract_activation_code"];N=new Map;v("click",e=>{let t=Ye(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.click({ timeout: ${r} });`]});v("click_element",N.get("click"));v("click_element_by_index",N.get("click"));v("double_click",e=>ge("double_click",e));v("double_click_on_element",N.get("double_click"));v("right_click",e=>ge("right_click",e));v("right_click_on_element",N.get("right_click"));v("hover",e=>ge("hover",e));v("hover_element_by_index",N.get("hover"));v("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return ge("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("fill",N.get("input_text"));v("clear_input",e=>ge("clear_input",e));v("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});v("send_keys",N.get("press"));v("send_keys_on_element",e=>{let t=Ye(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 s=e.action_data?.kwargs?.timeout_ms??Pt;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${s} });`]});v("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return ge("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});v("scroll",e=>{let t=e.action_data?.kwargs?.down??!0;return[`await page.evaluate('window.scrollBy(0, window.innerHeight * ${(e.action_data?.kwargs?.num_pages??1)*(t?1:-1)})');`]});v("scroll_down",N.get("scroll"));v("scroll_up",N.get("scroll"));v("scroll_element",N.get("scroll"));v("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});v("scroll_on_element",e=>ge("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));v("go_to_url",e=>{let t=e.action_data?.kwargs?.url||"";return e.action_data?.kwargs?.new_tab===!0?['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)}, new_tab: true } },`,"});"]:['await agent.execAction("go_to_url", page, {',` action_data: { kwargs: { url: ${JSON.stringify(t)} } },`,"});"]});v("open_tab",N.get("go_to_url"));v("go_back",()=>['await agent.execAction("go_back", page, {});']);v("reload_page",()=>['await agent.execAction("reload_page", page, {});']);v("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);v("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);v("verify",(e,t)=>{let r=e.action_data?.kwargs,s=typeof r?.code=="string",n=s?r?.statement||e.action_description:e.action_description||r?.statement;if(s&&n){let i=r.code.split(`
214
214
  `),a=JSON.stringify(n);return["{ const _t = Date.now(); try {",...i.map(c=>` ${c}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${a}\`);`,"} catch (_e) {",` console.log(\`[VERIFY:JS\u2192AI] JS failed \${((Date.now()-_t)/1000).toFixed(1)}s: (\${_e instanceof Error ? _e.message : String(_e)}), falling back to AI: ${a}\`);`,` await agent.assert(page, ${a}, ${JSON.stringify(t||"")});`,"} }"]}return s?r.code.split(`
215
215
  `):n?[`await agent.assert(page, ${JSON.stringify(n)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});v("ai_assert",N.get("verify"));v("assert",N.get("verify"));v("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let s=JSON.stringify(r),n=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${s}, '${t||""}', ${n});`]});v("ai_step",(e,t)=>{let r=e.action_data?.kwargs?.statement;return r?[`await agent.run(page, ${JSON.stringify(r)}, '${t||""}');`]:["// Skipping ai_step: missing statement"]});v("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,s=e.action_data?.kwargs?.variable_name;if(!r||!s)return["// Skipping ai_extract: missing element_description or variable_name"];let n=JSON.stringify(r),o=JSON.stringify(s);return[`await agent.extract(page, ${n}, ${o}, '${t||""}');`]});v("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,s=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${s}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});v("save_variable",e=>{let t=e.action_data?.kwargs?.name||"",r=e.action_data?.kwargs?.value;return['await agent.execAction("save_variable", page, {',` action_data: { kwargs: { name: ${JSON.stringify(t)}, value: ${JSON.stringify(r)} } },`,"});"]});v("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],s=t.split(`
216
216
  `);for(let n of s)r.push(` ${n}`);return r.push("}"),r});v("function",(e,t,r)=>{let s=e.action_data?.kwargs||{},n=s.functionName;if(n&&n.includes("#")){let[i,a]=n.split("#");if(i&&a){let c=i.replace(/\.(ts|js|mjs)$/,""),l=`import { ${a} } from '${c}';`;r?.imports?.add(l);let u={...s,functionName:a},d=Qr(u);return d?[d.endsWith(";")?d:`${d};`]:["// Skipping function: invalid export pattern"]}}let o=Qr(s);return o?[o.endsWith(";")?o:`${o};`]:["// Skipping function: missing functionName"]});v("generate_2fa_code",e=>{let t=e.action_data?.kwargs?.otp_secret_key||"";return['await agent.execAction("generate_2fa_code", page, {',` action_data: { kwargs: { otp_secret_key: ${JSON.stringify(t)} } },`,"});"]});v("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],s={};return t.paths?s.paths=t.paths:t.path&&(s.path=t.path),t.use_file_input&&(s.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(s)} }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("upload_file", page, {',...r.map(n=>` ${n},`),"});"]});v("wait_for_download_complete",e=>['await agent.execAction("wait_for_download_complete", page, {',` action_data: { kwargs: { timeout_seconds: ${e.action_data?.kwargs?.timeout_seconds||10} } },`,"});"]);v("switch_tab",e=>['await agent.execAction("switch_tab", page, {',` action_data: { kwargs: { page_id: ${e.action_data?.kwargs?.page_id??e.action_data?.kwargs?.tab_index??0} } },`,"});"]);v("close_tab",e=>{let t=e.action_data?.kwargs?.page_id;return t=t??e.action_data?.kwargs?.index,['await agent.execAction("close_tab", page, {',` action_data: { kwargs: { page_id: ${t} } },`,"});"]});v("set_date_for_native_date_picker",e=>{let t=e.action_data?.kwargs?.date??"",r=[];return r.push(`action_data: { kwargs: { date: ${JSON.stringify(t)} } }`),e.locator?r.push(`locator: ${JSON.stringify(e.locator)}`):e.xpath&&r.push(`xpath: ${JSON.stringify(e.xpath)}`),e.frame_path&&e.frame_path.length>0&&r.push(`frame_path: ${JSON.stringify(e.frame_path)}`),['await agent.execAction("set_date_for_native_date_picker", page, {',...r.map(s=>` ${s},`),"});"]});v("done",()=>["// Done - no action needed"]);v("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
@@ -252,9 +252,9 @@ Shutting down...`);let m=setTimeout(()=>{console.error("Cleanup timed out, force
252
252
  statements:
253
253
  - URL: ${e}
254
254
  - "TODO: Add your first step"
255
- `});import Sp from"dotenv";function ii(){return globalThis[oi]}function tt(){let e=ii();return e===void 0?process.env:e}var oi,Rt=_(()=>{"use strict";oi="__shiplightDotenvCache__"});function rt(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith(ai)?ci:li}var ai,ci,li,Ct=_(()=>{"use strict";ai="shp_",ci="https://nova-api.shiplight.ai",li="https://api.shiplight.ai"});var Nt={};oe(Nt,{lookupActionStores:()=>di,updateActionStores:()=>fi});function Is(){let e=tt(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:rt(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function di(e){let t=Is();if(!t||e.length===0)return new Map;try{let r=new AbortController,s=setTimeout(()=>r.abort(),pi),n=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(s),!n.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${n.status}`),new Map;let o=await n.json(),i=new Map;for(let[a,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function fi(e){let t=Is();if(!t||e.size===0)return 0;try{let r=new AbortController,s=setTimeout(()=>r.abort(),ui),n={};for(let[a,c]of e)n[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:n}),signal:r.signal});return clearTimeout(s),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var pi,ui,Dt=_(()=>{"use strict";Ct();Rt();pi=2e3,ui=5e3});import*as H from"fs";import*as Re from"path";import{globSync as hi}from"glob";function Ms(e){let t=tt().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ft:new jt(e)}function Ce(e){return e.replace(/\//g,"__")+".json"}function mi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var gi,jt,Ft,Os=_(()=>{"use strict";de();Rt();gi=".shiplight/action-cache";jt=class{constructor(t){this.cwd=t;this.cacheDir=Re.join(t,gi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!H.existsSync(this.cacheDir))return r;for(let s of t){let n=Re.join(this.cacheDir,Ce(s));try{if(H.existsSync(n)){let o=H.readFileSync(n,"utf-8");r.set(s,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;H.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[s,n]of t)try{let o=Re.join(this.cacheDir,Ce(s)),i=vt();if(H.existsSync(o))try{i=JSON.parse(H.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...n.entries}};H.writeFileSync(o,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!H.existsSync(this.cacheDir))return;let t=hi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,s=0;for(let n of t)try{let o=H.readFileSync(Re.join(this.cacheDir,n),"utf-8"),i=JSON.parse(o),a=mi(n);r.set(a,i),s+=Object.keys(i.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${s} cached action entit${s===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},Ft=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}loadAll(){}}});var Ds={};oe(Ds,{buildPlaywrightSpawnOptions:()=>Rs,buildTestPathsFromGitInfo:()=>Ns,cloudKeyToCwdRelPath:()=>Ht,runTests:()=>wi});import{spawn as yi,execFileSync as Ls}from"child_process";import*as W from"fs";import*as U from"path";import{globSync as Ut}from"glob";async function wi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(d=>W.existsSync(U.join(t,d)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
255
+ `});import Sp from"dotenv";function ii(){return globalThis[oi]}function tt(){let e=ii();return e===void 0?process.env:e}var oi,Rt=_(()=>{"use strict";oi="__shiplightDotenvCache__"});function rt(e,t){let r=t?.trim();return r?r.endsWith("/")?r.slice(0,-1):r:e.startsWith(ai)?ci:li}var ai,ci,li,Ct=_(()=>{"use strict";ai="shp_",ci="https://nova-api.shiplight.ai",li="https://api.shiplight.ai"});var Nt={};oe(Nt,{lookupActionStores:()=>di,updateActionStores:()=>fi});function Is(){let e=tt(),t=e.SHIPLIGHT_API_TOKEN;return t?{apiUrl:rt(t,e.SHIPLIGHT_API_URL),apiToken:t}:null}async function di(e){let t=Is();if(!t||e.length===0)return new Map;try{let r=new AbortController,s=setTimeout(()=>r.abort(),pi),n=await fetch(`${t.apiUrl}/action-entity-cache/lookup`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({test_paths:e}),signal:r.signal});if(clearTimeout(s),!n.ok)return console.warn(`[shiplight] Cache lookup failed: HTTP ${n.status}`),new Map;let o=await n.json(),i=new Map;for(let[a,c]of Object.entries(o.stores??{}))i.set(a,c);return i}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache lookup error:",r.message),new Map}}async function fi(e){let t=Is();if(!t||e.size===0)return 0;try{let r=new AbortController,s=setTimeout(()=>r.abort(),ui),n={};for(let[a,c]of e)n[a]=c;let o=await fetch(`${t.apiUrl}/action-entity-cache/update`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t.apiToken}`},body:JSON.stringify({stores:n}),signal:r.signal});return clearTimeout(s),o.ok?(await o.json()).updated??0:(console.warn(`[shiplight] Cache update failed: HTTP ${o.status}`),0)}catch(r){return r instanceof Error&&r.name!=="AbortError"&&console.warn("[shiplight] Cache update error:",r.message),0}}var pi,ui,Dt=_(()=>{"use strict";Ct();Rt();pi=2e3,ui=5e3});import*as H from"fs";import*as Re from"path";import{globSync as hi}from"glob";function Ms(e){let t=tt().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new Ft:new jt(e)}function Ce(e){return e.replace(/\//g,"__")+".json"}function mi(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var gi,jt,Ft,Os=_(()=>{"use strict";de();Rt();gi=".shiplight/action-cache";jt=class{constructor(t){this.cwd=t;this.cacheDir=Re.join(t,gi)}isCloud=!1;cacheDir;async lookup(t){let r=new Map;if(t.length===0||!H.existsSync(this.cacheDir))return r;for(let s of t){let n=Re.join(this.cacheDir,Ce(s));try{if(H.existsSync(n)){let o=H.readFileSync(n,"utf-8");r.set(s,JSON.parse(o))}}catch{}}return r}async update(t){if(t.size===0)return 0;H.mkdirSync(this.cacheDir,{recursive:!0});let r=0;for(let[s,n]of t)try{let o=Re.join(this.cacheDir,Ce(s)),i=vt();if(H.existsSync(o))try{i=JSON.parse(H.readFileSync(o,"utf-8"))}catch{}let a={...i,entries:{...i.entries,...n.entries}};H.writeFileSync(o,JSON.stringify(a,null,2)),r++}catch{}return r}loadAll(){if(!H.existsSync(this.cacheDir))return;let t=hi("*.json",{cwd:this.cacheDir});if(t.length===0)return;let r=new Map,s=0;for(let n of t)try{let o=H.readFileSync(Re.join(this.cacheDir,n),"utf-8"),i=JSON.parse(o),a=mi(n);r.set(a,i),s+=Object.keys(i.entries??{}).length}catch{}if(r.size!==0)return console.log(`[shiplight] Cache: loaded ${s} cached action entit${s===1?"y":"ies"} for ${r.size} test file${r.size!==1?"s":""}`),r}},Ft=class{isCloud=!0;async lookup(t){let{lookupActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}async update(t){let{updateActionStores:r}=await Promise.resolve().then(()=>(Dt(),Nt));return r(t)}loadAll(){}}});var Ds={};oe(Ds,{buildPlaywrightSpawnOptions:()=>Rs,buildTestPathsFromGitInfo:()=>Ns,cloudKeyToCwdRelPath:()=>Ht,runTests:()=>wi});import{spawn as yi,execFileSync as Ls}from"child_process";import*as G from"fs";import*as U from"path";import{globSync as Ut}from"glob";async function wi(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight test [playwright-args...]"),console.log(""),console.log("Delegates to `npx playwright test` with all arguments forwarded."),console.log("Auto-detects playwright.config.ts in the current directory."),console.log(""),console.log("Examples:"),console.log(" shiplight test # run all tests"),console.log(" shiplight test --headed # run tests with browser visible"),console.log(" shiplight test tests/login.test.yaml # run a specific YAML test"),console.log(" shiplight test tests/login.test.ts # run a specific TS test"),console.log(" shiplight test --grep 'login' # filter tests by name"),process.exit(0));let t=process.cwd();["playwright.config.ts","playwright.config.js","playwright.config.mjs"].some(d=>G.existsSync(U.join(t,d)))||(console.warn("Warning: No playwright.config.ts found in current directory."),console.warn(`Make sure you're running from your project root.
256
256
  `));let n=e.includes("--magic"),i=e.filter(d=>d!=="--magic").map(d=>d.endsWith(".test.yaml")?d.replace(/\.test\.yaml$/,".yaml.spec.ts"):d),a=Ms(t);await Si(t,a),ke&&process.stdout.write(`shiplightai v${ke}
257
- `);let c={...process.env};n&&(c.SHIPLIGHT_MAGIC="1");let l=yi("npx",["playwright","test",...i],Rs(t,c)),u=await new Promise(d=>{l.on("close",h=>d(h??1))});await vi(t,a),process.exit(u)}function Rs(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Bt(){try{return Ls("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function bi(){try{let e=Ls("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Cs(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let s=bi(),n=Bt();return Ns(e,t,s,n)}function Ns(e,t,r,s){return{testPaths:e.map(o=>{let i=s?U.relative(s,U.resolve(t,o)):o;return`${r}${i}`}),branchPrefix:r}}function Ht(e,t,r,s){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?U.relative(s,U.resolve(r,n)):n}async function Si(e,t){try{let r=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:s,branchPrefix:n}=Cs(r,e,t.isCloud),o=await t.lookup(s);if(o.size===0)return;let i=U.join(e,".shiplight","action-cache");W.mkdirSync(i,{recursive:!0});let a=Bt(),c=0;for(let[l,u]of o){let d=t.isCloud?Ht(l,n,a,e):l,h=U.join(i,Ce(d));W.writeFileSync(h,JSON.stringify(u,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 vi(e,t){try{let r=U.join(e,"test-results");if(!W.existsSync(r))return;let s=Ut("**/new-action-entities.json",{cwd:r});if(s.length===0)return;let n={};for(let d of s)try{let h=W.readFileSync(U.join(r,d),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(n,g.entries)}catch{}if(Object.keys(n).length===0)return;let o=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=Cs(o,e,t.isCloud),c=new Map;for(let d=0;d<o.length;d++){let h=o[d],g=i[d],f=U.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!W.existsSync(f))continue;let p=W.readFileSync(f,"utf-8"),S={};for(let[w,m]of Object.entries(n))p.includes(w)&&(S[w]=m);if(Object.keys(S).length>0){let w=t.isCloud?Ht(g,a,Bt(),e):g,m=U.join(e,".shiplight","action-cache",Ce(w)),y={};if(W.existsSync(m))try{y=JSON.parse(W.readFileSync(m,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...y,...S}})}}if(c.size===0)return;let l=await t.update(c),u=Array.from(c.values()).reduce((d,h)=>d+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${u} action entit${u!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var js=_(()=>{"use strict";Os();Ne()});function M(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function nt(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 _i(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function xi(e,t){let r=e||"?",s=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${s}`:`failed after ${r} ${s}`}function st(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function Ti(e){return`<span class="badge badge-${e}">${e}</span>`}function Pi(e){if(!e.code)return"";let t=e.code.split(`
257
+ `);let c={...process.env};n&&(c.SHIPLIGHT_MAGIC="1");let l=yi("npx",["playwright","test",...i],Rs(t,c)),u=await new Promise(d=>{l.on("close",h=>d(h??1))});await vi(t,a),process.exit(u)}function Rs(e,t){return{stdio:"inherit",shell:process.platform==="win32",cwd:e,env:t}}function Bt(){try{return Ls("git",["rev-parse","--show-toplevel"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()}catch{return null}}function bi(){try{let e=Ls("git",["rev-parse","--abbrev-ref","HEAD"],{encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim();if(e&&e!=="HEAD")return`${e}:`}catch{}return""}function Cs(e,t,r){if(!r)return{testPaths:[...e],branchPrefix:""};let s=bi(),n=Bt();return Ns(e,t,s,n)}function Ns(e,t,r,s){return{testPaths:e.map(o=>{let i=s?U.relative(s,U.resolve(t,o)):o;return`${r}${i}`}),branchPrefix:r}}function Ht(e,t,r,s){let n=t&&e.startsWith(t)?e.slice(t.length):e;return r?U.relative(s,U.resolve(r,n)):n}async function Si(e,t){try{let r=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]});if(r.length===0)return;let{testPaths:s,branchPrefix:n}=Cs(r,e,t.isCloud),o=await t.lookup(s);if(o.size===0)return;let i=U.join(e,".shiplight","action-cache");G.mkdirSync(i,{recursive:!0});let a=Bt(),c=0;for(let[l,u]of o){let d=t.isCloud?Ht(l,n,a,e):l,h=U.join(i,Ce(d));G.writeFileSync(h,JSON.stringify(u,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 vi(e,t){try{let r=U.join(e,"test-results");if(!G.existsSync(r))return;let s=Ut("**/new-action-entities.json",{cwd:r});if(s.length===0)return;let n={};for(let d of s)try{let h=G.readFileSync(U.join(r,d),"utf-8"),g=JSON.parse(h);g?.entries&&Object.assign(n,g.entries)}catch{}if(Object.keys(n).length===0)return;let o=Ut("**/*.test.yaml",{cwd:e,ignore:["**/node_modules/**"]}),{testPaths:i,branchPrefix:a}=Cs(o,e,t.isCloud),c=new Map;for(let d=0;d<o.length;d++){let h=o[d],g=i[d],f=U.join(e,h.replace(/\.test\.yaml$/,".yaml.spec.ts"));if(!G.existsSync(f))continue;let p=G.readFileSync(f,"utf-8"),S={};for(let[w,m]of Object.entries(n))p.includes(w)&&(S[w]=m);if(Object.keys(S).length>0){let w=t.isCloud?Ht(g,a,Bt(),e):g,m=U.join(e,".shiplight","action-cache",Ce(w)),y={};if(G.existsSync(m))try{y=JSON.parse(G.readFileSync(m,"utf-8")).entries}catch{}c.set(g,{version:"1.0",entries:{...y,...S}})}}if(c.size===0)return;let l=await t.update(c),u=Array.from(c.values()).reduce((d,h)=>d+Object.keys(h.entries).length,0);console.log(`[shiplight] Cache: saved ${u} action entit${u!==1?"ies":"y"} for ${l} test${l!==1?"s":""}`)}catch(r){console.warn("[shiplight] Cache upload failed:",r.message)}}var js=_(()=>{"use strict";Os();Ne()});function M(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function nt(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 _i(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function xi(e,t){let r=e||"?",s=e===1?"retry":"retries";return t==="passed"?`passed after ${r} ${s}`:`failed after ${r} ${s}`}function st(e){switch(e){case"passed":case"success":return'<span class="status-icon passed">&#x2714;</span>';case"flaky":return'<span class="status-icon flaky">&#x21BB;</span>';case"failed":case"failure":case"timedOut":return'<span class="status-icon failed">&#x2718;</span>';case"skipped":return'<span class="status-icon skipped">&#x2500;</span>';case"interrupted":return'<span class="status-icon failed">&#x26A0;</span>';default:return'<span class="status-icon pending">&#x25CB;</span>'}}function Ti(e){return`<span class="badge badge-${e}">${e}</span>`}function Pi(e){if(!e.code)return"";let t=e.code.split(`
258
258
  `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((n,o)=>{let i=e.codeStartLine+o,a=i===e.codeLine,c=String(i).padStart(4);return`<span class="code-line${a?" code-line-active":""}">${c} \u2502 ${M(n)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(s=>`<span class="code-line code-line-body">${M(s)}</span>`).join("")}</pre></div>`}function ki(e){let t=e.duration!=null?`<span class="step-duration">${nt(e.duration)}</span>`:"",r=st(e.status);if(e.screenshot||e.message||e.error||e.code){let n="";e.screenshot&&(n=`<img src="${M(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let o=e.screenshot?"":Pi(e),i="";e.error&&(i=`<div class="step-error"><pre>${M(e.error)}</pre></div>`);let a="";e.message&&!e.error&&(a=`<div class="step-message">${M(e.message)}</div>`);let c=e.status==="failure"?" open":"";return`
259
259
  <details class="step-details step step-${e.status}"${c}>
260
260
  <summary class="step-header">
@@ -341,7 +341,7 @@ statements:
341
341
  <div class="test-body">
342
342
  ${n}
343
343
  </div>
344
- </details>`}function Wt(e){let t=e.tests.filter(c=>c.flaky).length,r=e.tests.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0).length,s=e.tests.filter(c=>c.status==="passed"&&!c.flaky).length,n=e.tests.filter(c=>c.status==="failed"||c.status==="timedOut").length,o=e.tests.filter(c=>c.status==="skipped").length,i=e.tests.length,a=e.tests.map((c,l)=>Ai(c,l)).join(`
344
+ </details>`}function Gt(e){let t=e.tests.filter(c=>c.flaky).length,r=e.tests.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0).length,s=e.tests.filter(c=>c.status==="passed"&&!c.flaky).length,n=e.tests.filter(c=>c.status==="failed"||c.status==="timedOut").length,o=e.tests.filter(c=>c.status==="skipped").length,i=e.tests.length,a=e.tests.map((c,l)=>Ai(c,l)).join(`
345
345
  `);return`<!DOCTYPE html>
346
346
  <html lang="en">
347
347
  <head>
@@ -1135,9 +1135,9 @@ statements:
1135
1135
  });
1136
1136
  </script>
1137
1137
  </body>
1138
- </html>`}var Us=_(()=>{"use strict"});import*as D from"fs";import*as Pe from"path";import{execFileSync as $i}from"child_process";import{createHash as Ii}from"crypto";import Te from"axios";function Bs(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 pe(...e){try{return $i("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Mi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=D.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Oi(){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??"",s=process.env.GITHUB_RUN_ID??"",n=process.env.GITHUB_EVENT_NAME??"",i=Mi().pull_request,a=i?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",u=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,h=process.env.SHIPLIGHT_PR_TITLE??i?.title,g=pe("log","-1","--pretty=%s"),f=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:d,gitRepo:r,commitMessage:g,authorEmail:f,prNumber:u,prTitle:h,prUrl:u&&r?`${t}/${r}/pull/${u}`:void 0,ciBuildId:s,ciBuildUrl:s&&r?`${t}/${r}/actions/runs/${s}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:n,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",s=process.env.CI_MERGE_REQUEST_IID,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,prNumber:s,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:s&&t?`${t}/-/merge_requests/${s}`: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??"",s=process.env.CIRCLE_PROJECT_REPONAME??"",n=r&&s?`${r}/${s}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,i=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?Bs(a):void 0,l=pe("log","-1","--pretty=%s"),u=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:n,commitMessage:l,authorEmail:u,prNumber:i,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=pe("rev-parse","HEAD"),r=pe("rev-parse","--abbrev-ref","HEAD"),s=pe("log","-1","--pretty=%s"),n=pe("log","-1","--pretty=%ae"),o=pe("remote","get-url","origin"),i=o?Bs(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:s,authorEmail:n,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function Li(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 Gt(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Ri(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 Ci(e,t){let r=new Map;for(let s of t){s.screenshotS3Uris={};let n=r.get(s.testCaseName);n?n.push(s):r.set(s.testCaseName,[s])}return e.map(s=>r.get(s.title)?.shift())}function z(e,t){return Pe.isAbsolute(t)?t:Pe.join(e,t)}function Hs(e,t){let r={};for(let s of e)r[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return r}function Ni(e,t,r,s){let n=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let i=o.attemptNumber-1,a=t[i]??{},c={outcome:Gt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Hs(o.steps,a),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[i],traceS3Uri:s[i],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),n.push(c)}else n.push({outcome:Gt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Hs(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:s[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:Gt(e.status),flaky:e.flaky??!1,segments:n}}function Vt(e){return Ii("md5").update(e).digest("base64")}function Kt(e){return Vt(D.readFileSync(e))}async function zt(e,t){let r=D.readFileSync(t),s=Pe.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await Te.put(e,r,{headers:{"Content-Type":o,"Content-MD5":Vt(r)}})}async function Ws(e,t,r,s){let n=rt(s,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${s}`,"Content-Type":"application/json"},i=Oi(),a=e.tests.map(p=>{let S={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let w=z(t,p.videoPath);D.existsSync(w)&&(S.videoMd5=Kt(w))}if(p.tracePath){let w=z(t,p.tracePath);D.existsSync(w)&&(S.traceMd5=Kt(w))}return S}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let u=(await Te.post(`${n}/v1/local-runs`,{trigger:i.ciProvider,startTime:r,metadata:i,tests:a},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Ci(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,S)=>{let w=d[S];if(!w)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let T of p.attempts){let $=T.attemptNumber-1,V=T.attemptNumber===p.attempts.length;for(let L of T.steps)L.screenshot&&m.push({key:`attempt-${$}.${L.stepId}`,filePath:z(t,L.screenshot)});if(!V){if(T.videoPath){let L=z(t,T.videoPath);D.existsSync(L)&&m.push({key:`attempt-${$}.__video__`,filePath:L})}if(T.tracePath){let L=z(t,T.tracePath);D.existsSync(L)&&m.push({key:`attempt-${$}.__trace__`,filePath:L})}}}else for(let T of p.steps)T.screenshot&&m.push({key:T.stepId,filePath:z(t,T.screenshot)});if(!m.length)return;let y=m.map(T=>T.key),E={};for(let{key:T,filePath:$}of m)D.existsSync($)&&(E[T]=Kt($));try{let T=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:E},{headers:o});w.uploadUrls.screenshots=T.data.screenshots,w.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${p.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,T)}})),console.log("[reporter] [3/4] Uploading assets...");let h=(await Promise.all(e.tests.map(async(p,S)=>{let w=d[S];if(!w){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=w.uploadUrls,y={},E={},T={},$=0,V=[];if(p.attempts&&p.attempts.length>1)for(let P of p.attempts){let R=P.attemptNumber-1,dn=P.attemptNumber===p.attempts.length;for(let ee of P.steps)ee.screenshot&&V.push({key:`attempt-${R}.${ee.stepId}`,filePath:z(t,ee.screenshot),attemptIdx:R,assetType:"screenshot",originalStepId:ee.stepId});if(!dn){if(P.videoPath){let ee=z(t,P.videoPath);D.existsSync(ee)&&V.push({key:`attempt-${R}.__video__`,filePath:ee,attemptIdx:R,assetType:"video"})}if(P.tracePath){let ee=z(t,P.tracePath);D.existsSync(ee)&&V.push({key:`attempt-${R}.__trace__`,filePath:ee,attemptIdx:R,assetType:"trace"})}}}else for(let P of p.steps)P.screenshot&&V.push({key:P.stepId,filePath:z(t,P.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:P.stepId});await Promise.all(V.map(async P=>{if(m.screenshots?.[P.key]&&D.existsSync(P.filePath))try{await zt(m.screenshots[P.key],P.filePath);let R=w.screenshotS3Uris[P.key];switch(P.assetType){case"screenshot":y[P.attemptIdx]||(y[P.attemptIdx]={}),y[P.attemptIdx][P.originalStepId]=R;break;case"video":E[P.attemptIdx]=R;break;case"trace":T[P.attemptIdx]=R;break}$++}catch(R){console.warn(`[reporter] Asset upload failed for ${P.key}:`,R)}})),$>0&&console.log(`[reporter] [3/4] Uploaded ${$} asset(s) for "${p.title}"`);let L;if(p.videoPath&&m.video){let P=z(t,p.videoPath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await zt(m.video,P),L=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Video upload failed:",R)}}}let ye;if(p.tracePath&&m.trace){let P=z(t,p.tracePath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await zt(m.trace,P),ye=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Trace upload failed:",R)}}}if(p.attempts&&p.attempts.length>1){let P=p.attempts.length-1;L&&(E[P]=L),ye&&(T[P]=ye)}else L&&(E[0]=L),ye&&(T[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let ln=Ni(p,y,E,T),Yt=Buffer.from(JSON.stringify(ln)),Jt=Vt(Yt),Xt=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:Jt},{headers:o}),pn=Xt.data.reportUrl,un=Xt.data.reportS3Uri;return await Te.put(pn,Yt,{headers:{"Content-Type":"application/json","Content-MD5":Jt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:w.testCaseResultId,result:Li(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:un,videoS3Uri:L,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=Ri(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let f=await Te.put(`${n}/v1/local-runs/${u.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:o});console.log(`
1139
- Shiplight cloud report: ${Di(f.data.reportUrl,n)}`)}function Di(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${ji(t)}${r}`}function ji(e){let t=e.endsWith("/")?e.slice(0,-1):e;return t==="https://api.shiplight.ai"?"https://app.shiplight.ai":t==="https://nova-api.shiplight.ai"?"https://nova.shiplight.ai":t}var Gs=_(()=>{"use strict";Ct()});var Xs={};oe(Xs,{buildGitHubSummary:()=>Js,isReportToCloudEnabled:()=>Vs,runReport:()=>Fi});import*as I from"fs";import*as A from"path";async function Fi(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"),s=e.includes("--github-summary");!r&&s&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Bi(e,t,s):await Ui(e,t)}async function Ui(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",s=A.isAbsolute(r)?r:A.join(process.cwd(),r),n=A.join(s,"report-data.json");I.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(I.readFileSync(n,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${n}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=A.join(s,"index.html");if(I.writeFileSync(i,Wt(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Ys(o,s),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Bi(e,t,r){let s=A.join(process.cwd(),"shiplight-report"),n=e.findIndex(g=>g==="-o"||g==="--output");if(n!==-1&&e[n+1]){let g=e[n+1];s=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let g=0;g<e.length;g++){let f=e[g];if(o.has(f)){g++;continue}if(i.has(f))continue;let p=A.isAbsolute(f)?f:A.join(process.cwd(),f);a.push(p)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,u=0;I.mkdirSync(A.join(s,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let f=a[g],p=`shard-${g}`,S=A.join(f,"report-data.json");if(!I.existsSync(S)){console.warn(`Warning: No report-data.json found in ${f}, skipping.`);continue}let w;try{w=JSON.parse(I.readFileSync(S,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${S}, skipping.`);continue}console.log(`Merging ${p}: ${w.tests.length} tests from ${f}`),l+=w.totalDuration||0;let m=A.join(f,"screenshots");I.existsSync(m)&&zs(m,A.join(s,"screenshots",p));let y=A.join(s,p);for(let E of w.tests){let T=[E,...E.attempts||[]];for(let $ of T)Hi($.steps,p),$.videoPath&&Ks(f,$.videoPath,y)&&($.videoPath=`${p}/${$.videoPath}`),$.tracePath&&Ks(f,$.tracePath,y)&&($.tracePath=`${p}/${$.tracePath}`);c.push(E)}u++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:ke};I.writeFileSync(A.join(s,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let h=A.join(s,"index.html");if(I.writeFileSync(h,Wt(d),"utf-8"),console.log(`
1140
- Merged ${c.length} tests from ${u} shards into: ${h}`),await Ys(d,s),r&&Wi(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function Ks(e,t,r){let s=A.resolve(e,t);return s.startsWith(A.resolve(e)+A.sep)?I.existsSync(s)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(s,A.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function zs(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let s=A.join(e,r.name),n=A.join(t,r.name);r.isDirectory()?zs(s,n):I.copyFileSync(s,n)}}function Hi(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function Vs(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function Ys(e,t){if(!Vs())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let s=e.tests.map(o=>o.startTime).filter(o=>!!o),n=s.length>0?s.sort()[0]:e.timestamp??new Date().toISOString();try{await Ws(e,t,n,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ot(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:r}}function Js(e){let t=e.filter(c=>!c.file.includes("auth.setup")),r=t.filter(c=>c.flaky),s=t.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0),n=t.filter(c=>c.status==="passed"&&!c.flaky),o=t.filter(c=>c.status!=="passed"),i=t.length,a=`## Test Results
1138
+ </html>`}var Us=_(()=>{"use strict"});import*as D from"fs";import*as Pe from"path";import{execFileSync as $i}from"child_process";import{createHash as Ii}from"crypto";import Te from"axios";function Bs(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 pe(...e){try{return $i("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Mi(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=D.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Oi(){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??"",s=process.env.GITHUB_RUN_ID??"",n=process.env.GITHUB_EVENT_NAME??"",i=Mi().pull_request,a=i?.head?.sha,c=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",l=process.env.GITHUB_REF??"",u=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??l.match(/^refs\/pull\/(\d+)\//)?.[1],d=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,h=process.env.SHIPLIGHT_PR_TITLE??i?.title,g=pe("log","-1","--pretty=%s"),f=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:c,gitBranch:d,gitRepo:r,commitMessage:g,authorEmail:f,prNumber:u,prTitle:h,prUrl:u&&r?`${t}/${r}/pull/${u}`:void 0,ciBuildId:s,ciBuildUrl:s&&r?`${t}/${r}/actions/runs/${s}`:void 0,commitUrl:c&&r?`${t}/${r}/commit/${c}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:n,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",s=process.env.CI_MERGE_REQUEST_IID,n=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,o=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:o,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:n,prNumber:s,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:s&&t?`${t}/-/merge_requests/${s}`: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??"",s=process.env.CIRCLE_PROJECT_REPONAME??"",n=r&&s?`${r}/${s}`:void 0,o=process.env.CIRCLE_PULL_REQUEST,i=process.env.CIRCLE_PR_NUMBER??o?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,c=a?Bs(a):void 0,l=pe("log","-1","--pretty=%s"),u=pe("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:n,commitMessage:l,authorEmail:u,prNumber:i,prUrl:o,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&c?`${c}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=pe("rev-parse","HEAD"),r=pe("rev-parse","--abbrev-ref","HEAD"),s=pe("log","-1","--pretty=%s"),n=pe("log","-1","--pretty=%ae"),o=pe("remote","get-url","origin"),i=o?Bs(o):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:s,authorEmail:n,commitUrl:i&&t?`${i}/commit/${t}`:void 0})}return e}function Li(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 Wt(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Ri(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 Ci(e,t){let r=new Map;for(let s of t){s.screenshotS3Uris={};let n=r.get(s.testCaseName);n?n.push(s):r.set(s.testCaseName,[s])}return e.map(s=>r.get(s.title)?.shift())}function z(e,t){return Pe.isAbsolute(t)?t:Pe.join(e,t)}function Hs(e,t){let r={};for(let s of e)r[s.stepId]={description:s.description,status:s.status,duration:s.duration,message:s.error??s.message,screenshotS3Uri:t[s.stepId]};return r}function Ni(e,t,r,s){let n=[];if(e.attempts&&e.attempts.length>1)for(let o of e.attempts){let i=o.attemptNumber-1,a=t[i]??{},c={outcome:Wt(o.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Hs(o.steps,a),consoleLogs:[],stdout:o.attemptNumber===e.attempts.length?e.stdout??"":"",stderr:o.attemptNumber===e.attempts.length?e.stderr??"":"",videoS3Uri:r[i],traceS3Uri:s[i],actionStepsMap:e.actionStepsMap??{}};o.error&&(c.error={message:o.error}),o.status==="timedOut"&&(c.timedOut=!0),n.push(c)}else n.push({outcome:Wt(e.status),createdAt:e.endTime??new Date().toISOString(),resultJson:Hs(e.steps,t[0]??{}),consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r[0],traceS3Uri:s[0],actionStepsMap:e.actionStepsMap??{}});return{schemaVersion:2,result:Wt(e.status),flaky:e.flaky??!1,segments:n}}function Vt(e){return Ii("md5").update(e).digest("base64")}function Kt(e){return Vt(D.readFileSync(e))}async function zt(e,t){let r=D.readFileSync(t),s=Pe.extname(t).toLowerCase(),o={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[s]??"application/octet-stream";await Te.put(e,r,{headers:{"Content-Type":o,"Content-MD5":Vt(r)}})}async function Gs(e,t,r,s){let n=rt(s,process.env.SHIPLIGHT_API_URL),o={Authorization:`Bearer ${s}`,"Content-Type":"application/json"},i=Oi(),a=e.tests.map(p=>{let S={testCaseName:p.title,testCaseBaseName:p.baseTitle,suiteName:p.suiteName,file:p.file,tags:p.tags,suiteTags:p.suiteTags,baseUrl:p.baseUrl,skip:p.skip,slow:p.slow,timeout:p.timeout,parameterSetName:p.parameterSetName,flaky:p.flaky,retries:p.retries};if(p.videoPath){let w=z(t,p.videoPath);D.existsSync(w)&&(S.videoMd5=Kt(w))}if(p.tracePath){let w=z(t,p.tracePath);D.existsSync(w)&&(S.traceMd5=Kt(w))}return S}),c=e.tests.length;console.log(`[reporter] Uploading ${c} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let u=(await Te.post(`${n}/v1/local-runs`,{trigger:i.ciProvider,startTime:r,metadata:i,tests:a},{headers:o})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${u.testRunId})`);let d=Ci(e.tests,u.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(p,S)=>{let w=d[S];if(!w)return;let m=[];if(p.attempts&&p.attempts.length>1)for(let T of p.attempts){let $=T.attemptNumber-1,V=T.attemptNumber===p.attempts.length;for(let L of T.steps)L.screenshot&&m.push({key:`attempt-${$}.${L.stepId}`,filePath:z(t,L.screenshot)});if(!V){if(T.videoPath){let L=z(t,T.videoPath);D.existsSync(L)&&m.push({key:`attempt-${$}.__video__`,filePath:L})}if(T.tracePath){let L=z(t,T.tracePath);D.existsSync(L)&&m.push({key:`attempt-${$}.__trace__`,filePath:L})}}}else for(let T of p.steps)T.screenshot&&m.push({key:T.stepId,filePath:z(t,T.screenshot)});if(!m.length)return;let y=m.map(T=>T.key),E={};for(let{key:T,filePath:$}of m)D.existsSync($)&&(E[T]=Kt($));try{let T=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/screenshot-urls`,{stepIds:y,md5s:E},{headers:o});w.uploadUrls.screenshots=T.data.screenshots,w.screenshotS3Uris=T.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${y.length} screenshot URL(s) for "${p.title}"`)}catch(T){console.warn(`[reporter] Failed to get screenshot URLs for "${p.title}":`,T)}})),console.log("[reporter] [3/4] Uploading assets...");let h=(await Promise.all(e.tests.map(async(p,S)=>{let w=d[S];if(!w){console.warn(`[reporter] No result slot found for test "${p.title}", skipping.`);return}let m=w.uploadUrls,y={},E={},T={},$=0,V=[];if(p.attempts&&p.attempts.length>1)for(let P of p.attempts){let R=P.attemptNumber-1,dn=P.attemptNumber===p.attempts.length;for(let ee of P.steps)ee.screenshot&&V.push({key:`attempt-${R}.${ee.stepId}`,filePath:z(t,ee.screenshot),attemptIdx:R,assetType:"screenshot",originalStepId:ee.stepId});if(!dn){if(P.videoPath){let ee=z(t,P.videoPath);D.existsSync(ee)&&V.push({key:`attempt-${R}.__video__`,filePath:ee,attemptIdx:R,assetType:"video"})}if(P.tracePath){let ee=z(t,P.tracePath);D.existsSync(ee)&&V.push({key:`attempt-${R}.__trace__`,filePath:ee,attemptIdx:R,assetType:"trace"})}}}else for(let P of p.steps)P.screenshot&&V.push({key:P.stepId,filePath:z(t,P.screenshot),attemptIdx:0,assetType:"screenshot",originalStepId:P.stepId});await Promise.all(V.map(async P=>{if(m.screenshots?.[P.key]&&D.existsSync(P.filePath))try{await zt(m.screenshots[P.key],P.filePath);let R=w.screenshotS3Uris[P.key];switch(P.assetType){case"screenshot":y[P.attemptIdx]||(y[P.attemptIdx]={}),y[P.attemptIdx][P.originalStepId]=R;break;case"video":E[P.attemptIdx]=R;break;case"trace":T[P.attemptIdx]=R;break}$++}catch(R){console.warn(`[reporter] Asset upload failed for ${P.key}:`,R)}})),$>0&&console.log(`[reporter] [3/4] Uploaded ${$} asset(s) for "${p.title}"`);let L;if(p.videoPath&&m.video){let P=z(t,p.videoPath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading video for "${p.title}"...`);try{await zt(m.video,P),L=w.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Video upload failed:",R)}}}let ye;if(p.tracePath&&m.trace){let P=z(t,p.tracePath);if(D.existsSync(P)){console.log(`[reporter] [3/4] Uploading trace for "${p.title}"...`);try{await zt(m.trace,P),ye=w.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${p.title}"`)}catch(R){console.warn("[reporter] Trace upload failed:",R)}}}if(p.attempts&&p.attempts.length>1){let P=p.attempts.length-1;L&&(E[P]=L),ye&&(T[P]=ye)}else L&&(E[0]=L),ye&&(T[0]=ye);console.log(`[reporter] [3/4] Uploading report for "${p.title}"...`);let ln=Ni(p,y,E,T),Yt=Buffer.from(JSON.stringify(ln)),Jt=Vt(Yt),Xt=await Te.post(`${n}/v1/local-runs/${u.testRunId}/results/${w.testCaseResultId}/report-url`,{md5:Jt},{headers:o}),pn=Xt.data.reportUrl,un=Xt.data.reportS3Uri;return await Te.put(pn,Yt,{headers:{"Content-Type":"application/json","Content-MD5":Jt}}),console.log(`[reporter] [3/4] Report uploaded for "${p.title}"`),{testCaseResultId:w.testCaseResultId,result:Li(p.status),durationMs:p.duration,startTime:p.startTime,endTime:p.endTime,error:p.error,reportS3Uri:un,videoS3Uri:L,traceS3Uri:ye,metadata:{suiteName:p.suiteName,file:p.file,...p.retries!=null&&{retries:p.retries},...p.flaky&&{flaky:!0}}}}))).filter(p=>!!p);console.log("[reporter] [4/4] Finalising run...");let g=Ri(e.tests);console.log(`[reporter] [4/4] Overall status: ${g}`);let f=await Te.put(`${n}/v1/local-runs/${u.testRunId}/complete`,{status:g,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:h},{headers:o});console.log(`
1139
+ Shiplight cloud report: ${Di(f.data.reportUrl,n)}`)}function Di(e,t){if(/^https?:\/\//.test(e))return e;let r=e.startsWith("/")?e:`/${e}`;return`${ji(t)}${r}`}function ji(e){let t=e.endsWith("/")?e.slice(0,-1):e;return t==="https://api.shiplight.ai"?"https://app.shiplight.ai":t==="https://nova-api.shiplight.ai"?"https://nova.shiplight.ai":t}var Ws=_(()=>{"use strict";Ct()});var Xs={};oe(Xs,{buildGitHubSummary:()=>Js,isReportToCloudEnabled:()=>Vs,runReport:()=>Fi});import*as I from"fs";import*as A from"path";async function Fi(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"),s=e.includes("--github-summary");!r&&s&&console.warn("Warning: --github-summary is only supported with --merge, ignoring."),r?await Bi(e,t,s):await Ui(e,t)}async function Ui(e,t){let r=e.find(a=>!a.startsWith("--"))||"shiplight-report",s=A.isAbsolute(r)?r:A.join(process.cwd(),r),n=A.join(s,"report-data.json");I.existsSync(n)||(console.error(`Error: ${n} not found.`),console.error("Run a test first to generate report artifacts, then use this command to regenerate the HTML."),process.exit(1));let o;try{o=JSON.parse(I.readFileSync(n,"utf-8"))}catch(a){console.error(`Error: Failed to parse ${n}`),console.error(a instanceof Error?a.message:String(a)),process.exit(1)}let i=A.join(s,"index.html");if(I.writeFileSync(i,Gt(o),"utf-8"),console.log(`Shiplight report regenerated: ${i}`),await Ys(o,s),t)try{let a=(await import("open")).default;await a(i)}catch{}}async function Bi(e,t,r){let s=A.join(process.cwd(),"shiplight-report"),n=e.findIndex(g=>g==="-o"||g==="--output");if(n!==-1&&e[n+1]){let g=e[n+1];s=A.isAbsolute(g)?g:A.join(process.cwd(),g)}let o=new Set(["-o","--output"]),i=new Set(["--merge","--open","--github-summary","-o","--output"]),a=[];for(let g=0;g<e.length;g++){let f=e[g];if(o.has(f)){g++;continue}if(i.has(f))continue;let p=A.isAbsolute(f)?f:A.join(process.cwd(),f);a.push(p)}a.length===0&&(console.error("Error: --merge requires at least one input directory."),console.error("Usage: shiplight report --merge dir1/ dir2/ [-o output-dir]"),process.exit(1));let c=[],l=0,u=0;I.mkdirSync(A.join(s,"screenshots"),{recursive:!0});for(let g=0;g<a.length;g++){let f=a[g],p=`shard-${g}`,S=A.join(f,"report-data.json");if(!I.existsSync(S)){console.warn(`Warning: No report-data.json found in ${f}, skipping.`);continue}let w;try{w=JSON.parse(I.readFileSync(S,"utf-8"))}catch{console.warn(`Warning: Failed to parse ${S}, skipping.`);continue}console.log(`Merging ${p}: ${w.tests.length} tests from ${f}`),l+=w.totalDuration||0;let m=A.join(f,"screenshots");I.existsSync(m)&&zs(m,A.join(s,"screenshots",p));let y=A.join(s,p);for(let E of w.tests){let T=[E,...E.attempts||[]];for(let $ of T)Hi($.steps,p),$.videoPath&&Ks(f,$.videoPath,y)&&($.videoPath=`${p}/${$.videoPath}`),$.tracePath&&Ks(f,$.tracePath,y)&&($.tracePath=`${p}/${$.tracePath}`);c.push(E)}u++}c.length===0&&(console.error("Error: No tests found across any input directories."),process.exit(1));let d={tests:c,totalDuration:l,timestamp:new Date().toISOString(),shiplightVersion:ke};I.writeFileSync(A.join(s,"report-data.json"),JSON.stringify(d,null,2),"utf-8");let h=A.join(s,"index.html");if(I.writeFileSync(h,Gt(d),"utf-8"),console.log(`
1140
+ Merged ${c.length} tests from ${u} shards into: ${h}`),await Ys(d,s),r&&Gi(c),t)try{let g=(await import("open")).default;await g(h)}catch{}}function Ks(e,t,r){let s=A.resolve(e,t);return s.startsWith(A.resolve(e)+A.sep)?I.existsSync(s)?(I.mkdirSync(r,{recursive:!0}),I.copyFileSync(s,A.join(r,t)),!0):!1:(console.warn(`Warning: Skipping artifact with path traversal: ${t}`),!1)}function zs(e,t){I.mkdirSync(t,{recursive:!0});for(let r of I.readdirSync(e,{withFileTypes:!0})){let s=A.join(e,r.name),n=A.join(t,r.name);r.isDirectory()?zs(s,n):I.copyFileSync(s,n)}}function Hi(e,t){for(let r of e)r.screenshot?.startsWith("screenshots/")&&(r.screenshot=r.screenshot.replace("screenshots/",`screenshots/${t}/`))}function Vs(){let e=process.env.SHIPLIGHT_REPORT_TO_CLOUD??process.env.REPORT_TO_CLOUD;return e?["true","1","yes","on"].includes(e.trim().toLowerCase()):!1}async function Ys(e,t){if(!Vs())return;let r=process.env.SHIPLIGHT_API_TOKEN;if(!r){let o=process.env.SHIPLIGHT_REPORT_TO_CLOUD!==void 0?"SHIPLIGHT_REPORT_TO_CLOUD":"REPORT_TO_CLOUD";console.warn(`[report] ${o} is enabled but no SHIPLIGHT_API_TOKEN found, skipping cloud upload.`);return}let s=e.tests.map(o=>o.startTime).filter(o=>!!o),n=s.length>0?s.sort()[0]:e.timestamp??new Date().toISOString();try{await Gs(e,t,n,r)}catch(o){console.warn("[report] Cloud upload failed:",o)}}function ot(e){let t=e.file.replace(".yaml.spec.ts",".test.yaml"),r=A.join("tests",A.basename(t));return{name:e.title||A.basename(t),yamlPath:r}}function Js(e){let t=e.filter(c=>!c.file.includes("auth.setup")),r=t.filter(c=>c.flaky),s=t.filter(c=>!c.flaky&&c.retries!=null&&c.retries>0),n=t.filter(c=>c.status==="passed"&&!c.flaky),o=t.filter(c=>c.status!=="passed"),i=t.length,a=`## Test Results
1141
1141
 
1142
1142
  `;if(o.length===0&&r.length===0?a+=`\u2705 All ${i} tests passed
1143
1143
 
@@ -1166,9 +1166,9 @@ Merged ${c.length} tests from ${u} shards into: ${h}`),await Ys(d,s),r&&Wi(c),t)
1166
1166
  `;for(let l of c)a+=`- ${l}
1167
1167
  `;a+=`
1168
1168
  </details>
1169
- `}return a}function Wi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,Js(e)),console.log("GitHub step summary written.")}var qs=_(()=>{"use strict";Us();Gs();Ne()});var Zs,Qs=_(()=>{"use strict";Zs="0.1.69"});var en={};oe(en,{runTranspile:()=>Ki});import*as it from"path";import{glob as Gi}from"glob";async function Ki(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(),s=await Gi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,i=0;for(let a of s.sort()){let c=it.resolve(r,a),l=ps(c,{version:Zs});if(!l.valid){n++,console.log(`
1169
+ `}return a}function Gi(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}I.appendFileSync(t,Js(e)),console.log("GitHub step summary written.")}var qs=_(()=>{"use strict";Us();Ws();Ne()});var Zs,Qs=_(()=>{"use strict";Zs="0.1.70"});var en={};oe(en,{runTranspile:()=>Ki});import*as it from"path";import{glob as Wi}from"glob";async function Ki(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(),s=await Wi(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,i=0;for(let a of s.sort()){let c=it.resolve(r,a),l=ps(c,{version:Zs});if(!l.valid){n++,console.log(`
1170
1170
  \u2717 ${a}`);for(let d of l.errors)console.log(` ERROR: ${d}`);continue}i++;let u=it.basename(l.specFile);if(l.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${u}`);for(let d of l.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${a} \u2192 ${u}`)}console.log(`
1171
- ${s.length} file(s): ${i} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var tn=_(()=>{"use strict";$t();Qs()});var nn={};oe(nn,{runInspect:()=>zi});import*as at from"fs";import*as rn from"path";async function zi(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"),s=e.find(i=>!i.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=rn.resolve(process.cwd(),s);at.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=at.readFileSync(n,"utf-8");try{let i=Ie(o),a=j(o);if(r)Vi(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${s}: ${i.message}`),process.exit(1)}}function Vi(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 s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=sn(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function sn(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 s=sn(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var on=_(()=>{"use strict";de()});var an=hn((Vp,Yi)=>{Yi.exports={name:"shiplightai",version:"0.1.69",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Ne();import Ji from"dotenv";Ji.config();qt();Zt();function cn(){console.log(`
1171
+ ${s.length} file(s): ${i} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var tn=_(()=>{"use strict";$t();Qs()});var nn={};oe(nn,{runInspect:()=>zi});import*as at from"fs";import*as rn from"path";async function zi(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"),s=e.find(i=>!i.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=rn.resolve(process.cwd(),s);at.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=at.readFileSync(n,"utf-8");try{let i=Ie(o),a=j(o);if(r)Vi(a,i);else{let c={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(c,null,t?0:2))}}catch(i){console.error(`Error parsing ${s}: ${i.message}`),process.exit(1)}}function Vi(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 s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=sn(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function sn(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 s=sn(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var on=_(()=>{"use strict";de()});var an=hn((Vp,Yi)=>{Yi.exports={name:"shiplightai",version:"0.1.70",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Ne();import Ji from"dotenv";Ji.config();qt();Zt();function cn(){console.log(`
1172
1172
  Usage: shiplight <command> [options]
1173
1173
 
1174
1174
  Commands: