shiplightai 0.1.47 → 0.1.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/reporter.js CHANGED
@@ -1,12 +1,13 @@
1
1
  import { createRequire as __createRequire } from "module";
2
2
  const require = __createRequire(import.meta.url);
3
- var Ve=Object.defineProperty;var Je=(e,t)=>()=>(e&&(t=e(e=0)),t);var Ye=(e,t)=>{for(var r in t)Ve(e,r,{get:t[r],enumerable:!0})};var Ue={};Ye(Ue,{uploadToCloud:()=>Ut});import*as L from"fs";import*as C from"path";import{execFileSync as It}from"child_process";import{createHash as Ot}from"crypto";import z from"axios";function De(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 N(...e){try{return It("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Lt(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=L.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Ct(){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??"",i=process.env.GITHUB_RUN_ID??"",o=process.env.GITHUB_EVENT_NAME??"",s=Lt().pull_request,a=s?.head?.sha,d=process.env.SHIPLIGHT_GIT_SHA??a??process.env.GITHUB_SHA??"",u=process.env.GITHUB_REF??"",f=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??u.match(/^refs\/pull\/(\d+)\//)?.[1],b=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,p=process.env.SHIPLIGHT_PR_TITLE??s?.title,h=N("log","-1","--pretty=%s"),g=N("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Actions",gitCommit:d,gitBranch:b,gitRepo:r,commitMessage:h,authorEmail:g,prNumber:f,prTitle:p,prUrl:f&&r?`${t}/${r}/pull/${f}`:void 0,ciBuildId:i,ciBuildUrl:i&&r?`${t}/${r}/actions/runs/${i}`:void 0,commitUrl:d&&r?`${t}/${r}/commit/${d}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:o,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",i=process.env.CI_MERGE_REQUEST_IID,o=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:o,prNumber:i,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:i&&t?`${t}/-/merge_requests/${i}`: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??"",i=process.env.CIRCLE_PROJECT_REPONAME??"",o=r&&i?`${r}/${i}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],a=process.env.CIRCLE_REPOSITORY_URL,d=a?De(a):void 0,u=N("log","-1","--pretty=%s"),f=N("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:o,commitMessage:u,authorEmail:f,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&d?`${d}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=N("rev-parse","HEAD"),r=N("rev-parse","--abbrev-ref","HEAD"),i=N("log","-1","--pretty=%s"),o=N("log","-1","--pretty=%ae"),n=N("remote","get-url","origin"),s=n?De(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:i,authorEmail:o,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function Nt(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 Be(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Rt(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 Dt(e,t){let r=new Map;for(let i of t){i.screenshotS3Uris={};let o=r.get(i.testCaseName);o?o.push(i):r.set(i.testCaseName,[i])}return e.map(i=>r.get(i.title)?.shift())}function Q(e,t){return C.isAbsolute(t)?t:C.join(e,t)}function Bt(e,t,r,i){let o={};for(let n of e.steps)o[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:Be(e.status),flaky:!1,segments:[{outcome:Be(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:o,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:i,actionStepsMap:e.actionStepsMap??{}}]}}function ne(e){return Ot("md5").update(e).digest("base64")}function ie(e){return ne(L.readFileSync(e))}async function oe(e,t){let r=L.readFileSync(t),i=C.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[i]??"application/octet-stream";await z.put(e,r,{headers:{"Content-Type":n,"Content-MD5":ne(r)}})}async function Ut(e,t,r,i){let o=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${i}`,"Content-Type":"application/json"},s=Ct(),a=e.tests.map(l=>{let T={testCaseName:l.title,testCaseBaseName:l.baseTitle,suiteName:l.suiteName,file:l.file,tags:l.tags,suiteTags:l.suiteTags,baseUrl:l.baseUrl,skip:l.skip,slow:l.slow,timeout:l.timeout,parameterSetName:l.parameterSetName,flaky:l.flaky,retries:l.retries};if(l.videoPath){let y=Q(t,l.videoPath);L.existsSync(y)&&(T.videoMd5=ie(y))}if(l.tracePath){let y=Q(t,l.tracePath);L.existsSync(y)&&(T.traceMd5=ie(y))}return T}),d=e.tests.length;console.log(`[reporter] Uploading ${d} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let f=(await z.post(`${o}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:a},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${f.testRunId})`);let b=Dt(e.tests,f.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(l,T)=>{let y=b[T];if(!y)return;let w=l.steps.filter(_=>_.screenshot);if(!w.length)return;let v=w.map(_=>_.stepId),x={};for(let _ of w){let k=C.isAbsolute(_.screenshot)?_.screenshot:C.join(t,_.screenshot);L.existsSync(k)&&(x[_.stepId]=ie(k))}try{let _=await z.post(`${o}/v1/local-runs/${f.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:v,md5s:x},{headers:n});y.uploadUrls.screenshots=_.data.screenshots,y.screenshotS3Uris=_.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${v.length} screenshot URL(s) for "${l.title}"`)}catch(_){console.warn(`[reporter] Failed to get screenshot URLs for "${l.title}":`,_)}})),console.log("[reporter] [3/4] Uploading assets...");let p=(await Promise.all(e.tests.map(async(l,T)=>{let y=b[T];if(!y){console.warn(`[reporter] No result slot found for test "${l.title}", skipping.`);return}let w=y.uploadUrls,v={},x=0;await Promise.all(l.steps.map(async E=>{if(E.screenshot&&w.screenshots?.[E.stepId]){let W=C.isAbsolute(E.screenshot)?E.screenshot:C.join(t,E.screenshot);if(L.existsSync(W))try{await oe(w.screenshots[E.stepId],W),v[E.stepId]=y.screenshotS3Uris[E.stepId],x++}catch(ze){console.warn(`[reporter] Screenshot upload failed for step ${E.stepId}:`,ze)}}})),x>0&&console.log(`[reporter] [3/4] Uploaded ${x} screenshot(s) for "${l.title}"`);let _;if(l.videoPath&&w.video){let E=Q(t,l.videoPath);if(L.existsSync(E)){console.log(`[reporter] [3/4] Uploading video for "${l.title}"...`);try{await oe(w.video,E),_=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${l.title}"`)}catch(W){console.warn("[reporter] Video upload failed:",W)}}}let k;if(l.tracePath&&w.trace){let E=Q(t,l.tracePath);if(L.existsSync(E)){console.log(`[reporter] [3/4] Uploading trace for "${l.title}"...`);try{await oe(w.trace,E),k=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${l.title}"`)}catch(W){console.warn("[reporter] Trace upload failed:",W)}}}console.log(`[reporter] [3/4] Uploading report for "${l.title}"...`);let He=Bt(l,v,_,k),ae=Buffer.from(JSON.stringify(He)),ce=ne(ae),le=await z.post(`${o}/v1/local-runs/${f.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:ce},{headers:n}),Ke=le.data.reportUrl,je=le.data.reportS3Uri;return await z.put(Ke,ae,{headers:{"Content-Type":"application/json","Content-MD5":ce}}),console.log(`[reporter] [3/4] Report uploaded for "${l.title}"`),{testCaseResultId:y.testCaseResultId,result:Nt(l.status),durationMs:l.duration,startTime:l.startTime,endTime:l.endTime,error:l.error,reportS3Uri:je,videoS3Uri:_,traceS3Uri:k,metadata:{suiteName:l.suiteName,file:l.file}}}))).filter(l=>!!l);console.log("[reporter] [4/4] Finalising run...");let h=Rt(e.tests),g=await z.put(`${o}/v1/local-runs/${f.testRunId}/complete`,{status:h,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:p},{headers:n});console.log(`
4
- Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});import*as $ from"fs";import*as S from"path";import{z as c}from"zod";var ue=c.enum(["JS_CODE","AI_MODE"]),ee=c.object({type:ue,expression:c.string()}),de=c.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),G=c.object({uid:c.string(),type:de,comment:c.string().optional()}),pe=c.object({action_data:c.object({action_name:c.string(),kwargs:c.record(c.any()).optional(),args:c.array(c.any()).optional()}),action_description:c.string().optional(),url:c.string().optional(),xpath:c.string().nullable().optional(),locator:c.string().nullable().optional(),css_selector:c.string().nullable().optional(),unique_selector:c.string().nullable().optional(),element_index:c.number().nullable().optional(),frame_path:c.array(c.any()).optional(),artifacts:c.record(c.any()).optional(),feedback:c.string().optional(),original_browser_use_action:c.any().optional()}).passthrough(),he=G.extend({type:c.literal("DRAFT"),description:c.string()}),fe=G.extend({type:c.literal("ACTION"),description:c.string(),action_entity:pe.optional(),locator:c.string().optional(),use_pure_vision:c.boolean().optional()}),P=c.lazy(()=>c.union([he,fe,G.extend({type:c.literal("STEP"),description:c.string().optional().default(""),statements:c.array(P),reference_id:c.number().optional()}),G.extend({type:c.literal("IF_ELSE"),description:c.string().optional(),condition:ee,then:c.array(P),else:c.array(P).optional()}),G.extend({type:c.literal("WHILE_LOOP"),description:c.string().optional(),condition:ee,body:c.array(P),timeout_ms:c.number().optional()})])),me=c.object({name:c.string(),statements:c.array(P),teardown:c.array(P).optional(),skip:c.union([c.boolean(),c.string()]).optional(),timeout:c.number().optional(),fail:c.union([c.boolean(),c.string()]).optional(),only:c.boolean().optional(),slow:c.boolean().optional()}),J=c.object({tests:c.array(me).min(1),beforeAll:c.array(P).optional(),afterAll:c.array(P).optional(),beforeEach:c.array(P).optional(),afterEach:c.array(P).optional()}),te=c.object({comment:c.string().optional(),version:c.string().optional(),goal:c.string().optional(),url:c.string().optional(),baseURL:c.string().optional(),final_feedback:c.string().optional(),completed:c.boolean().optional(),success:c.boolean().optional(),statements:c.array(P).optional(),teardown:c.array(P).optional(),last_modified_at:c.string().optional(),testGroup:J.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 Jt,parse as Xe,parseAllDocuments as Yt,parseDocument as qe,Document as Xt,isMap as Y,isSeq as H}from"yaml";import{v4 as O}from"uuid";function re(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(re);let t=e,r=Object.keys(t);if(r.length===1){let o=r[0];if(o.startsWith("{ ")&&o.endsWith(" }")&&t[o]===null)return`{{${o.slice(2,-2)}}}`}let i={};for(let[o,n]of Object.entries(t))i[o]=re(n);return i}var ge=1024*1024;function D(e){if(e.length>ge)throw new Error(`YAML input too large (${e.length} bytes, max ${ge})`);let t=re(Xe(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Ze(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:I(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=I(t.teardown));let i=te.safeParse(r);if(!i.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(i.error.errors)}`);let o=i.data;return we(e,o),o}function Ze(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 o={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 d={name:a.name,statements:I(a.statements)};return Array.isArray(a.teardown)&&a.teardown.length>0&&(d.teardown=I(a.teardown)),a.skip!==void 0&&(d.skip=a.skip),typeof a.timeout=="number"&&(d.timeout=a.timeout),a.fail!==void 0&&(d.fail=a.fail),a.only===!0&&(d.only=!0),a.slow===!0&&(d.slow=!0),d})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(o.beforeAll=I(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(o.afterAll=I(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(o.beforeEach=I(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(o.afterEach=I(t.afterEach));let n=J.safeParse(o);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function I(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(Qe)}function Qe(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 et(t);if("WHILE"in t)return tt(t);if("STEP"in t)return rt(t);if("VERIFY"in t){let r=t.VERIFY,i={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(i.code=t.js),{uid:O(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:i}}}}if("URL"in t){let r=t.URL,i=t.new_tab===!0?!0:void 0,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return i&&(n.new_tab=!0),o!==void 0&&(n.timeout_seconds=o),{uid:O(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:O(),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:i}}}}}if("WAIT"in t){let r=t.WAIT,i=typeof t.seconds=="number"?t.seconds:3;return{uid:O(),type:"ACTION",description:typeof r=="string"?r:`Wait ${i}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${i}s`,action_data:{action_name:"wait",kwargs:{seconds:i}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:O(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,i=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:O(),type:"ACTION",description:i,action_entity:{action_description:i,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...i}=t;return ye({...i,action:"function",functionName:r})}if("action"in t)return ye(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:O(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function be(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 et(e){let t=be(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let i={uid:O(),type:"IF_ELSE",condition:t,then:I(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(i.else=I(e.ELSE)),i}function tt(e){let t=be(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let i={uid:O(),type:"WHILE_LOOP",condition:t,body:I(r)};return typeof e.timeout_ms=="number"&&(i.timeout_ms=e.timeout_ms),i}function rt(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:O(),type:"STEP",description:t,statements:I(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}var it=new Set(["action","intent","desc","locator","xpath","use_pure_vision"]);function ye(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:"",i=typeof e.locator=="string"?e.locator:void 0,o=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[u,f]of Object.entries(e))it.has(u)||(s[u]=f);let a={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};i&&(a.locator=i),o&&(a.xpath=o);let d={uid:O(),type:"ACTION",description:r,action_entity:a};return n&&(d.use_pure_vision=!0),d}function we(e,t){let r;try{r=qe(e)}catch{return}let i=r.contents;if(!i||!Y(i))return;if(r.commentBefore)t.comment=r.commentBefore;else{let d=i.items?.[0];d?.key&&d.key.commentBefore&&(t.comment=d.key.commentBefore)}let o=i,n=o.get("statements",!0);H(n)&&t.statements&&K(n,t.statements);let s=o.get("teardown",!0);H(s)&&t.teardown&&K(s,t.teardown)}function K(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 i=e.items[r];i.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=i.commentBefore);let o=t[r];if(o.type==="STEP"&&Y(i)){let n=i.get("statements",!0);H(n)&&K(n,o.statements)}else if(o.type==="IF_ELSE"&&Y(i)){let n=i.get("THEN",!0);H(n)&&K(n,o.then);let s=i.get("ELSE",!0);H(s)&&o.else&&K(s,o.else)}else if(o.type==="WHILE_LOOP"&&Y(i)){let n=i.get("DO",!0);H(n)&&K(n,o.body)}}}import{parse as tr,stringify as rr}from"yaml";var B=(e,t,r)=>{e.forEach((i,o)=>{let n=`${t}.${o}`;i.type==="DRAFT"?r[n]={description:i.description||"Draft",action_entity:void 0}:i.type==="ACTION"?r[n]={description:i.description||"Action",action_entity:i.action_entity}:i.type==="STEP"&&i.statements?B(i.statements,n,r):i.type==="IF_ELSE"?(r[n]={description:"IF "+(i.condition?.expression||""),action_entity:void 0},i.then&&B(i.then,`${n}.then`,r),i.else&&B(i.else,`${n}.else`,r)):i.type==="WHILE_LOOP"&&(r[n]={description:"WHILE "+(i.condition?.expression||""),action_entity:void 0},i.body&&B(i.body,`${n}.body`,r))})};function X(e){if(!e?.statements||!Array.isArray(e.statements))return{};let t={};return B(e.statements,"main",t),e.teardown&&Array.isArray(e.teardown)&&B(e.teardown,"teardown",t),t}import{v4 as dr}from"uuid";var U=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(U||{});var ve=112;var at=1080-ve;var _e={"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"}};var Se={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"]},V=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),Se[e].map(i=>_e[i]).filter(i=>i.defaultBrowserType&&r.includes(i.defaultBrowserType))};var ct={desktop:{label:"Desktop",type:"desktop",devices:V("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:V("mobile")}},lt={desktop:{label:"Desktop",type:"desktop",devices:V("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:V("mobile",!0)}};import{stringify as Lo}from"yaml";import{createHash as gt}from"crypto";import{parse as yt,stringify as Ee}from"yaml";import{readFileSync as bt}from"fs";import{resolve as Me,dirname as wt}from"path";import{parse as xe,stringify as vt}from"yaml";var Ae=5e3;function ft(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function mt(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 $e(e){let t=ft(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let i=mt(e);if(i){let o=JSON.stringify(i);return`${t}.locator(${o}).first()`}return null}var M=new Map;function m(e,t){M.set(e,t)}function F(e,t,r=[]){let i=[...r];return t.locator?i.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&i.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&i.push(`frame_path: ${JSON.stringify(t.frame_path)}`),i.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...i.map(o=>` ${o},`),"});"]}m("click",e=>{let t=$e(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??Ae;return[`await ${t}.click({ timeout: ${r} });`]});m("click_element",M.get("click"));m("click_element_by_index",M.get("click"));m("double_click",e=>F("double_click",e));m("double_click_on_element",M.get("double_click"));m("right_click",e=>F("right_click",e));m("right_click_on_element",M.get("right_click"));m("hover",e=>F("hover",e));m("hover_element_by_index",M.get("hover"));m("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return F("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});m("fill",M.get("input_text"));m("clear_input",e=>F("clear_input",e));m("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});m("send_keys",M.get("press"));m("send_keys_on_element",e=>{let t=$e(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 i=e.action_data?.kwargs?.timeout_ms??Ae;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${i} });`]});m("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return F("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});m("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)})');`]});m("scroll_down",M.get("scroll"));m("scroll_up",M.get("scroll"));m("scroll_element",M.get("scroll"));m("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});m("scroll_on_element",e=>F("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));m("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)} } },`,"});"]});m("open_tab",M.get("go_to_url"));m("go_back",()=>['await agent.execAction("go_back", page, {});']);m("reload_page",()=>['await agent.execAction("reload_page", page, {});']);m("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);m("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);m("verify",(e,t)=>{let r=e.action_data?.kwargs,i=typeof r?.code=="string",o=i?r?.statement||e.action_description:e.action_description||r?.statement;if(i&&o){let s=r.code.split(`
5
- `),a=JSON.stringify(o);return["{ const _t = Date.now(); try {",...s.map(d=>` ${d}`),` 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 i?r.code.split(`
6
- `):o?[`await agent.assert(page, ${JSON.stringify(o)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});m("ai_assert",M.get("verify"));m("assert",M.get("verify"));m("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let i=JSON.stringify(r),o=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${i}, '${t||""}', ${o});`]});m("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"]});m("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,i=e.action_data?.kwargs?.variable_name;if(!r||!i)return["// Skipping ai_extract: missing element_description or variable_name"];let o=JSON.stringify(r),n=JSON.stringify(i);return[`await agent.extract(page, ${o}, ${n}, '${t||""}');`]});m("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,i=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${i}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});m("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)} } },`,"});"]});m("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],i=t.split(`
7
- `);for(let o of i)r.push(` ${o}`);return r.push("}"),r});m("function",(e,t,r)=>{let i=e.action_data?.kwargs||{},o=i.functionName;if(o&&o.includes("#")){let[s,a]=o.split("#");if(s&&a){let d=s.replace(/\.(ts|js|mjs)$/,""),u=`import { ${a} } from '${d}';`;r?.imports?.add(u);let f={...i,functionName:a},b=Te(f);return b?[b.endsWith(";")?b:`${b};`]:["// Skipping function: invalid export pattern"]}}let n=Te(i);return n?[n.endsWith(";")?n:`${n};`]:["// Skipping function: missing functionName"]});m("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)} } },`,"});"]});m("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],i={};return t.paths?i.paths=t.paths:t.path&&(i.path=t.path),t.use_file_input&&(i.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(i)} }`),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(o=>` ${o},`),"});"]});m("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} } },`,"});"]);m("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} } },`,"});"]);m("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} } },`,"});"]});m("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(i=>` ${i},`),"});"]});m("done",()=>["// Done - no action needed"]);m("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
8
- `):["// Skipping js_action: missing code"]});function Te(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 i=["page","testContext","request","agent"],o=["undefined","null","true","false"],n=r.map(s=>i.includes(s)||o.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}var ke=5;function Pe(e,t){let r={expandingPaths:new Set([Me(t)]),depth:0,referencedPaths:new Set},i={...e};Array.isArray(i.statements)&&(i.statements=R(i.statements,t,r)),Array.isArray(i.teardown)&&(i.teardown=R(i.teardown,t,r));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(i[o])&&(i[o]=R(i[o],t,r));return{doc:i,referencedTemplatePaths:Array.from(r.referencedPaths)}}function R(e,t,r){let i=[];for(let o of e)if(_t(o)){let n=St(o,t,r);i.push(...n)}else i.push(Tt(o,t,r));return i}function _t(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function St(e,t,r){if(r.depth>=ke)throw new Error(`Template expansion exceeded maximum depth of ${ke}. Check for deeply nested or circular template references.`);let i=Me(wt(t),e.template);if(r.expandingPaths.has(i))throw new Error(`Circular template reference detected: ${i} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${i}`);r.referencedPaths.add(i);let o;try{o=bt(i,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${i} (referenced from ${t}): ${f.message}`)}let n=xe(o);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${i} \u2014 expected a YAML object`);let s=n.params||[],a=e.params||{};for(let f of s)if(!(f in a))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${s.join(", ")}]`);let d=n.statements;if(!Array.isArray(d))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(a).length>0){let b=vt(d);for(let[p,h]of Object.entries(a))b=b.split(`<<${p}>>`).join(String(h));d=xe(b)}let u={expandingPaths:new Set([...r.expandingPaths,i]),depth:r.depth+1,referencedPaths:r.referencedPaths};return R(d,i,u)}function Tt(e,t,r){if(typeof e!="object"||e===null)return e;let i={...e};return Array.isArray(i.statements)&&(i.statements=R(i.statements,t,r)),Array.isArray(i.THEN)&&(i.THEN=R(i.THEN,t,r)),Array.isArray(i.ELSE)&&(i.ELSE=R(i.ELSE,t,r)),Array.isArray(i.DO)&&(i.DO=R(i.DO,t,r)),i}var Ie=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Oe(e,t){let r=yt(e),i=r?.name,o=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new Ie('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return kt(r,i,o,n,t)}return xt(r,i,o,n,t)}function xt(e,t,r,i,o){let n=e?.beforeEach,s=e?.afterEach,a=Le(e?.parameters),d=e?.timeout,u=e?.skip,f=e?.fail,b=e?.only,p=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new Ie(`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 h=[];if(o&&e&&typeof e=="object"){let T=Pe(e,o);e=T.doc,h=T.referencedTemplatePaths}let g=Ee(e),l=D(g);return o&&(j(l.statements??[],o,"main"),l.teardown&&j(l.teardown,o,"teardown")),{testFlow:l,name:t,tags:r,use:i,beforeEach:n,afterEach:s,parameters:a,timeout:d,skip:u,fail:f,only:b,slow:p,referencedTemplatePaths:h}}function kt(e,t,r,i,o){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,a=n.afterAll,d=n.beforeEach,u=n.afterEach,f=[],b=n.tests.map(g=>{if(!g.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(g.statements)||g.statements.length===0)throw new Error(`Suite test "${g.name}" must have a non-empty "statements" array.`);let l={goal:g.name,statements:g.statements};g.teardown&&(l.teardown=g.teardown);let T=[],y=l;if(o&&typeof l=="object"){let _=Pe(l,o);y=_.doc,T=_.referencedTemplatePaths,f.push(...T)}let w=Ee(y),v=D(w),x=Le(g.parameters);return{testFlow:v,name:g.name,tags:Array.isArray(g.tags)?g.tags:void 0,parameters:x,timeout:g.timeout,skip:g.skip,fail:g.fail,only:g.only,slow:g.slow}}),p=n.base_url,h=p?{...i,baseURL:p}:i;return{suite:{beforeAll:s,afterAll:a,beforeEach:d,afterEach:u,tests:b},name:t,tags:r,use:h,referencedTemplatePaths:f}}function Le(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 j(e,t,r){for(let i=0;i<e.length;i++){let o=e[i],n=`${r}.${i}`,s=o.description||"";if(o.uid=At(t,n,s),o.type===U.STEP)j(o.statements,t,n);else if(o.type===U.IF_ELSE){let a=o;j(a.then,t,`${n}.then`),a.else&&j(a.else,t,`${n}.else`)}else o.type===U.WHILE_LOOP&&j(o.body,t,`${n}.body`)}}function At(e,t,r){let i=gt("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${i.slice(0,8)}-${i.slice(8,12)}-${i.slice(12,16)}-${i.slice(16,20)}-${i.slice(20,32)}`}function A(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Z(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 Ce(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function q(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 $t(e){return`<span class="badge badge-${e}">${e}</span>`}function Et(e){let t=e.duration!=null?`<span class="step-duration">${Z(e.duration)}</span>`:"",r=q(e.status);if(e.screenshot||e.message||e.error){let o="";e.screenshot&&(o=`<img src="${A(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n="";e.error&&(n=`<div class="step-error"><pre>${A(e.error)}</pre></div>`);let s="";e.message&&!e.error&&(s=`<div class="step-message">${A(e.message)}</div>`);let a=e.status==="failure"?" open":"";return`
9
- <details class="step-details step step-${e.status}"${a}>
3
+ var Ye=Object.defineProperty;var Je=(e,t)=>()=>(e&&(t=e(e=0)),t);var Xe=(e,t)=>{for(var r in t)Ye(e,r,{get:t[r],enumerable:!0})};var Fe={};Xe(Fe,{uploadToCloud:()=>Wt});import*as L from"fs";import*as C from"path";import{execFileSync as Lt}from"child_process";import{createHash as Ct}from"crypto";import z from"axios";function Be(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 N(...e){try{return Lt("git",e,{stdio:["pipe","pipe","ignore"]}).toString().trim()||void 0}catch{return}}function Nt(){let e=process.env.GITHUB_EVENT_PATH;if(!e)return{};try{let t=L.readFileSync(e,"utf8");return JSON.parse(t)}catch{return{}}}function Rt(){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??"",i=process.env.GITHUB_RUN_ID??"",o=process.env.GITHUB_EVENT_NAME??"",s=Nt().pull_request,c=s?.head?.sha,d=process.env.SHIPLIGHT_GIT_SHA??c??process.env.GITHUB_SHA??"",p=process.env.GITHUB_REF??"",m=process.env.SHIPLIGHT_PR_NUMBER??process.env.GITHUB_PR_NUMBER??p.match(/^refs\/pull\/(\d+)\//)?.[1],b=(process.env.SHIPLIGHT_GIT_BRANCH??process.env.GITHUB_HEAD_REF)||process.env.GITHUB_REF_NAME,u=process.env.SHIPLIGHT_PR_TITLE??s?.title,h=N("log","-1","--pretty=%s"),f=N("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"GitHub Action",gitCommit:d,gitBranch:b,gitRepo:r,commitMessage:h,authorEmail:f,prNumber:m,prTitle:u,prUrl:m&&r?`${t}/${r}/pull/${m}`:void 0,ciBuildId:i,ciBuildUrl:i&&r?`${t}/${r}/actions/runs/${i}`:void 0,commitUrl:d&&r?`${t}/${r}/commit/${d}`:void 0,triggeredBy:process.env.GITHUB_ACTOR,eventName:o,workflow:process.env.GITHUB_WORKFLOW})}else if(process.env.GITLAB_CI){let t=process.env.CI_PROJECT_URL??"",r=process.env.CI_COMMIT_SHA??"",i=process.env.CI_MERGE_REQUEST_IID,o=process.env.CI_COMMIT_AUTHOR_EMAIL??process.env.GITLAB_USER_EMAIL,n=process.env.CI_PROJECT_PATH;Object.assign(e,{ciProvider:"GitLab CI",gitCommit:r,gitBranch:process.env.CI_COMMIT_REF_NAME,gitRepo:n,commitMessage:process.env.CI_COMMIT_MESSAGE,authorEmail:o,prNumber:i,prTitle:process.env.CI_MERGE_REQUEST_TITLE,prUrl:i&&t?`${t}/-/merge_requests/${i}`: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??"",i=process.env.CIRCLE_PROJECT_REPONAME??"",o=r&&i?`${r}/${i}`:void 0,n=process.env.CIRCLE_PULL_REQUEST,s=process.env.CIRCLE_PR_NUMBER??n?.match(/\/pull\/(\d+)$/)?.[1],c=process.env.CIRCLE_REPOSITORY_URL,d=c?Be(c):void 0,p=N("log","-1","--pretty=%s"),m=N("log","-1","--pretty=%ae");Object.assign(e,{ciProvider:"CircleCI",gitCommit:t,gitBranch:process.env.CIRCLE_BRANCH,gitRepo:o,commitMessage:p,authorEmail:m,prNumber:s,prUrl:n,ciBuildId:process.env.CIRCLE_BUILD_NUM,ciBuildUrl:process.env.CIRCLE_BUILD_URL,commitUrl:t&&d?`${d}/commit/${t}`:void 0,triggeredBy:process.env.CIRCLE_USERNAME})}else{e.ciProvider="Local";let t=N("rev-parse","HEAD"),r=N("rev-parse","--abbrev-ref","HEAD"),i=N("log","-1","--pretty=%s"),o=N("log","-1","--pretty=%ae"),n=N("remote","get-url","origin"),s=n?Be(n):void 0;Object.assign(e,{gitCommit:t,gitBranch:r,commitMessage:i,authorEmail:o,commitUrl:s&&t?`${s}/commit/${t}`:void 0})}return e}function Dt(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 Ue(e){switch(e){case"passed":return"passed";case"skipped":return"skipped";default:return"failed"}}function Bt(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 Ut(e,t){let r=new Map;for(let i of t){i.screenshotS3Uris={};let o=r.get(i.testCaseName);o?o.push(i):r.set(i.testCaseName,[i])}return e.map(i=>r.get(i.title)?.shift())}function Q(e,t){return C.isAbsolute(t)?t:C.join(e,t)}function Ft(e,t,r,i){let o={};for(let n of e.steps)o[n.stepId]={description:n.description,status:n.status,duration:n.duration,message:n.error??n.message,screenshotS3Uri:t[n.stepId]};return{schemaVersion:2,result:Ue(e.status),flaky:!1,segments:[{outcome:Ue(e.status),createdAt:e.endTime??new Date().toISOString(),fixId:null,resultJson:o,consoleLogs:[],stdout:e.stdout??"",stderr:e.stderr??"",videoS3Uri:r,traceS3Uri:i,actionStepsMap:e.actionStepsMap??{}}]}}function ne(e){return Ct("md5").update(e).digest("base64")}function ie(e){return ne(L.readFileSync(e))}async function oe(e,t){let r=L.readFileSync(t),i=C.extname(t).toLowerCase(),n={".png":"image/png",".webm":"video/webm",".zip":"application/zip",".json":"application/json"}[i]??"application/octet-stream";await z.put(e,r,{headers:{"Content-Type":n,"Content-MD5":ne(r)}})}async function Wt(e,t,r,i){let o=process.env.SHIPLIGHT_API_URL??"https://api.shiplight.ai",n={Authorization:`Bearer ${i}`,"Content-Type":"application/json"},s=Rt(),c=e.tests.map(a=>{let S={testCaseName:a.title,testCaseBaseName:a.baseTitle,suiteName:a.suiteName,file:a.file,tags:a.tags,suiteTags:a.suiteTags,baseUrl:a.baseUrl,skip:a.skip,slow:a.slow,timeout:a.timeout,parameterSetName:a.parameterSetName,flaky:a.flaky,retries:a.retries};if(a.videoPath){let y=Q(t,a.videoPath);L.existsSync(y)&&(S.videoMd5=ie(y))}if(a.tracePath){let y=Q(t,a.tracePath);L.existsSync(y)&&(S.traceMd5=ie(y))}return S}),d=e.tests.length;console.log(`[reporter] Uploading ${d} test result(s) to Shiplight cloud...`),console.log("[reporter] [1/4] Creating run record...");let m=(await z.post(`${o}/v1/local-runs`,{trigger:s.ciProvider,startTime:r,metadata:s,tests:c},{headers:n})).data;console.log(`[reporter] [1/4] Run record created (testRunId=${m.testRunId})`);let b=Ut(e.tests,m.testCaseResults);console.log("[reporter] [2/4] Requesting screenshot upload URLs..."),await Promise.all(e.tests.map(async(a,S)=>{let y=b[S];if(!y)return;let w=a.steps.filter(_=>_.screenshot);if(!w.length)return;let v=w.map(_=>_.stepId),T={};for(let _ of w){let k=C.isAbsolute(_.screenshot)?_.screenshot:C.join(t,_.screenshot);L.existsSync(k)&&(T[_.stepId]=ie(k))}try{let _=await z.post(`${o}/v1/local-runs/${m.testRunId}/results/${y.testCaseResultId}/screenshot-urls`,{stepIds:v,md5s:T},{headers:n});y.uploadUrls.screenshots=_.data.screenshots,y.screenshotS3Uris=_.data.screenshotS3Uris,console.log(`[reporter] [2/4] Got ${v.length} screenshot URL(s) for "${a.title}"`)}catch(_){console.warn(`[reporter] Failed to get screenshot URLs for "${a.title}":`,_)}})),console.log("[reporter] [3/4] Uploading assets...");let u=(await Promise.all(e.tests.map(async(a,S)=>{let y=b[S];if(!y){console.warn(`[reporter] No result slot found for test "${a.title}", skipping.`);return}let w=y.uploadUrls,v={},T=0;await Promise.all(a.steps.map(async E=>{if(E.screenshot&&w.screenshots?.[E.stepId]){let W=C.isAbsolute(E.screenshot)?E.screenshot:C.join(t,E.screenshot);if(L.existsSync(W))try{await oe(w.screenshots[E.stepId],W),v[E.stepId]=y.screenshotS3Uris[E.stepId],T++}catch(Ve){console.warn(`[reporter] Screenshot upload failed for step ${E.stepId}:`,Ve)}}})),T>0&&console.log(`[reporter] [3/4] Uploaded ${T} screenshot(s) for "${a.title}"`);let _;if(a.videoPath&&w.video){let E=Q(t,a.videoPath);if(L.existsSync(E)){console.log(`[reporter] [3/4] Uploading video for "${a.title}"...`);try{await oe(w.video,E),_=y.s3Uris.video,console.log(`[reporter] [3/4] Video uploaded for "${a.title}"`)}catch(W){console.warn("[reporter] Video upload failed:",W)}}}let k;if(a.tracePath&&w.trace){let E=Q(t,a.tracePath);if(L.existsSync(E)){console.log(`[reporter] [3/4] Uploading trace for "${a.title}"...`);try{await oe(w.trace,E),k=y.s3Uris.trace,console.log(`[reporter] [3/4] Trace uploaded for "${a.title}"`)}catch(W){console.warn("[reporter] Trace upload failed:",W)}}}console.log(`[reporter] [3/4] Uploading report for "${a.title}"...`);let Ke=Ft(a,v,_,k),ce=Buffer.from(JSON.stringify(Ke)),le=ne(ce),de=await z.post(`${o}/v1/local-runs/${m.testRunId}/results/${y.testCaseResultId}/report-url`,{md5:le},{headers:n}),je=de.data.reportUrl,ze=de.data.reportS3Uri;return await z.put(je,ce,{headers:{"Content-Type":"application/json","Content-MD5":le}}),console.log(`[reporter] [3/4] Report uploaded for "${a.title}"`),{testCaseResultId:y.testCaseResultId,result:Dt(a.status),durationMs:a.duration,startTime:a.startTime,endTime:a.endTime,error:a.error,reportS3Uri:ze,videoS3Uri:_,traceS3Uri:k,metadata:{suiteName:a.suiteName,file:a.file}}}))).filter(a=>!!a);console.log("[reporter] [4/4] Finalising run...");let h=Bt(e.tests),f=await z.put(`${o}/v1/local-runs/${m.testRunId}/complete`,{status:h,endTime:new Date().toISOString(),totalDuration:e.totalDuration,results:u},{headers:n});console.log(`
4
+ Shiplight cloud report: ${f.data.reportUrl}`)}var We=Je(()=>{"use strict"});import*as $ from"fs";import*as x from"path";import{v4 as Zt}from"uuid";import{z as l}from"zod";var ue=l.enum(["JS_CODE","AI_MODE"]),ee=l.object({type:ue,expression:l.string()}),pe=l.enum(["DRAFT","STEP","ACTION","IF_ELSE","WHILE_LOOP"]),G=l.object({uid:l.string(),type:pe,comment:l.string().optional()}),he=l.object({action_data:l.object({action_name:l.string(),kwargs:l.record(l.any()).optional(),args:l.array(l.any()).optional()}),action_description:l.string().optional(),url:l.string().optional(),xpath:l.string().nullable().optional(),locator:l.string().nullable().optional(),css_selector:l.string().nullable().optional(),unique_selector:l.string().nullable().optional(),element_index:l.number().nullable().optional(),frame_path:l.array(l.any()).optional(),artifacts:l.record(l.any()).optional(),feedback:l.string().optional(),original_browser_use_action:l.any().optional()}).passthrough(),fe=G.extend({type:l.literal("DRAFT"),description:l.string()}),me=G.extend({type:l.literal("ACTION"),description:l.string(),action_entity:he.optional(),locator:l.string().optional(),use_pure_vision:l.boolean().optional()}),P=l.lazy(()=>l.union([fe,me,G.extend({type:l.literal("STEP"),description:l.string().optional().default(""),statements:l.array(P),reference_id:l.number().optional()}),G.extend({type:l.literal("IF_ELSE"),description:l.string().optional(),condition:ee,then:l.array(P),else:l.array(P).optional()}),G.extend({type:l.literal("WHILE_LOOP"),description:l.string().optional(),condition:ee,body:l.array(P),timeout_ms:l.number().optional()})])),ge=l.object({name:l.string(),statements:l.array(P),teardown:l.array(P).optional(),skip:l.union([l.boolean(),l.string()]).optional(),timeout:l.number().optional(),fail:l.union([l.boolean(),l.string()]).optional(),only:l.boolean().optional(),slow:l.boolean().optional()}),Y=l.object({tests:l.array(ge).min(1),beforeAll:l.array(P).optional(),afterAll:l.array(P).optional(),beforeEach:l.array(P).optional(),afterEach:l.array(P).optional()}),te=l.object({comment:l.string().optional(),version:l.string().optional(),goal:l.string().optional(),url:l.string().optional(),baseURL:l.string().optional(),final_feedback:l.string().optional(),completed:l.boolean().optional(),success:l.boolean().optional(),statements:l.array(P).optional(),teardown:l.array(P).optional(),last_modified_at:l.string().optional(),testGroup:Y.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 or,parse as qe,parseAllDocuments as nr,parseDocument as Ze,Document as sr,isMap as J,isSeq as H}from"yaml";import{v4 as O}from"uuid";function re(e){if(e==null||typeof e!="object")return e;if(Array.isArray(e))return e.map(re);let t=e,r=Object.keys(t);if(r.length===1){let o=r[0];if(o.startsWith("{ ")&&o.endsWith(" }")&&t[o]===null)return`{{${o.slice(2,-2)}}}`}let i={};for(let[o,n]of Object.entries(t))i[o]=re(n);return i}var ye=1024*1024;function D(e){if(e.length>ye)throw new Error(`YAML input too large (${e.length} bytes, max ${ye})`);let t=re(qe(e));if(!t||typeof t!="object")throw new Error("Invalid YAML: expected an object at root level");if(t.suite)return Qe(t);let r={version:"1.3.0",goal:t.goal,url:t.url,baseURL:t.base_url,statements:I(t.statements??[])};t.final_feedback&&(r.final_feedback=t.final_feedback),t.teardown&&Array.isArray(t.teardown)&&(r.teardown=I(t.teardown));let i=te.safeParse(r);if(!i.success)throw new Error(`Invalid TestFlow after YAML conversion: ${JSON.stringify(i.error.errors)}`);let o=i.data;return ve(e,o),o}function Qe(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 o={tests:r.map(c=>{if(!c.name)throw new Error('Each test in a suite must have a "name" field');if(!Array.isArray(c.statements)||c.statements.length===0)throw new Error(`Suite test "${c.name}" must have a non-empty "statements" array`);let d={name:c.name,statements:I(c.statements)};return Array.isArray(c.teardown)&&c.teardown.length>0&&(d.teardown=I(c.teardown)),c.skip!==void 0&&(d.skip=c.skip),typeof c.timeout=="number"&&(d.timeout=c.timeout),c.fail!==void 0&&(d.fail=c.fail),c.only===!0&&(d.only=!0),c.slow===!0&&(d.slow=!0),d})};Array.isArray(t.beforeAll)&&t.beforeAll.length>0&&(o.beforeAll=I(t.beforeAll)),Array.isArray(t.afterAll)&&t.afterAll.length>0&&(o.afterAll=I(t.afterAll)),Array.isArray(t.beforeEach)&&t.beforeEach.length>0&&(o.beforeEach=I(t.beforeEach)),Array.isArray(t.afterEach)&&t.afterEach.length>0&&(o.afterEach=I(t.afterEach));let n=Y.safeParse(o);if(!n.success)throw new Error(`Invalid TestGroup: ${JSON.stringify(n.error.errors)}`);return{version:"1.3.0",baseURL:t.base_url||void 0,testGroup:n.data}}function I(e){if(!Array.isArray(e))throw new Error("Expected an array of statements");return e.map(et)}function et(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 tt(t);if("WHILE"in t)return rt(t);if("STEP"in t)return it(t);if("VERIFY"in t){let r=t.VERIFY,i={statement:typeof r=="string"?r:String(r)};return typeof t.js=="string"&&(i.code=t.js),{uid:O(),type:"ACTION",description:String(r),action_entity:{action_description:String(r),action_data:{action_name:"verify",kwargs:i}}}}if("URL"in t){let r=t.URL,i=t.new_tab===!0?!0:void 0,o=typeof t.timeout_seconds=="number"?t.timeout_seconds:void 0,n={url:typeof r=="string"?r:String(r)};return i&&(n.new_tab=!0),o!==void 0&&(n.timeout_seconds=o),{uid:O(),type:"ACTION",description:`Navigate to ${r}`,action_entity:{action_description:`Navigate to ${r}`,action_data:{action_name:"go_to_url",kwargs:n}}}}if("WAIT_UNTIL"in t){let r=t.WAIT_UNTIL,i=typeof t.timeout_seconds=="number"?t.timeout_seconds:60;return{uid:O(),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:i}}}}}if("WAIT"in t){let r=t.WAIT,i=typeof t.seconds=="number"?t.seconds:3;return{uid:O(),type:"ACTION",description:typeof r=="string"?r:`Wait ${i}s`,action_entity:{action_description:typeof r=="string"?r:`Wait ${i}s`,action_data:{action_name:"wait",kwargs:{seconds:i}}}}}if("CODE"in t){let r=t.CODE;if(r==null)throw new Error('CODE statement has no code. Use "CODE: |" followed by indented code on the next line.');return{uid:O(),type:"ACTION",description:"Code block",action_entity:{action_description:"Code block",action_data:{action_name:"js_code",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("js"in t&&("intent"in t||"desc"in t)&&!("VERIFY"in t)){let r=t.js,i=typeof t.intent=="string"?t.intent:typeof t.desc=="string"?t.desc:"";return{uid:O(),type:"ACTION",description:i,action_entity:{action_description:i,action_data:{action_name:"js_action",kwargs:{code:typeof r=="string"?r:String(r)}}}}}if("call"in t&&typeof t.call=="string"){let{call:r,...i}=t;return be({...i,action:"function",functionName:r})}if("action"in t)return be(t);if("intent"in t&&typeof t.intent=="string"||"desc"in t&&typeof t.desc=="string")return{uid:O(),type:"DRAFT",description:typeof t.intent=="string"?t.intent:t.desc};throw new Error(`Cannot infer statement type from object: ${JSON.stringify(t)}`)}function we(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 tt(e){let t=we(e.IF),r=e.THEN;if(!Array.isArray(r))throw new Error("IF_ELSE requires a THEN array");let i={uid:O(),type:"IF_ELSE",condition:t,then:I(r)};return"ELSE"in e&&Array.isArray(e.ELSE)&&(i.else=I(e.ELSE)),i}function rt(e){let t=we(e.WHILE),r=e.DO;if(!Array.isArray(r))throw new Error("WHILE_LOOP requires a DO array");let i={uid:O(),type:"WHILE_LOOP",condition:t,body:I(r)};return typeof e.timeout_ms=="number"&&(i.timeout_ms=e.timeout_ms),i}function it(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:O(),type:"STEP",description:t,statements:I(e.statements)};return typeof e.reference_id=="number"&&(r.reference_id=e.reference_id),r}var ot=new Set(["action","intent","desc","locator","xpath","use_pure_vision"]);function be(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:"",i=typeof e.locator=="string"?e.locator:void 0,o=typeof e.xpath=="string"?e.xpath:void 0,n=typeof e.use_pure_vision=="boolean"?e.use_pure_vision:void 0,s={};for(let[p,m]of Object.entries(e))ot.has(p)||(s[p]=m);let c={action_description:r,action_data:{action_name:t,kwargs:Object.keys(s).length>0?s:{}}};i&&(c.locator=i),o&&(c.xpath=o);let d={uid:O(),type:"ACTION",description:r,action_entity:c};return n&&(d.use_pure_vision=!0),d}function ve(e,t){let r;try{r=Ze(e)}catch{return}let i=r.contents;if(!i||!J(i))return;if(r.commentBefore)t.comment=r.commentBefore;else{let d=i.items?.[0];d?.key&&d.key.commentBefore&&(t.comment=d.key.commentBefore)}let o=i,n=o.get("statements",!0);H(n)&&t.statements&&K(n,t.statements);let s=o.get("teardown",!0);H(s)&&t.teardown&&K(s,t.teardown)}function K(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 i=e.items[r];i.commentBefore&&!(r===0&&e.commentBefore)&&(t[r].comment=i.commentBefore);let o=t[r];if(o.type==="STEP"&&J(i)){let n=i.get("statements",!0);H(n)&&K(n,o.statements)}else if(o.type==="IF_ELSE"&&J(i)){let n=i.get("THEN",!0);H(n)&&K(n,o.then);let s=i.get("ELSE",!0);H(s)&&o.else&&K(s,o.else)}else if(o.type==="WHILE_LOOP"&&J(i)){let n=i.get("DO",!0);H(n)&&K(n,o.body)}}}import{parse as ur,stringify as pr}from"yaml";var B=(e,t,r)=>{e.forEach((i,o)=>{let n=`${t}.${o}`;i.type==="DRAFT"?r[n]={description:i.description||"Draft",action_entity:void 0}:i.type==="ACTION"?r[n]={description:i.description||"Action",action_entity:i.action_entity,locator:i.locator}:i.type==="STEP"&&i.statements?B(i.statements,n,r):i.type==="IF_ELSE"?(r[n]={description:"IF "+(i.condition?.expression||""),action_entity:void 0},i.then&&B(i.then,`${n}.then`,r),i.else&&B(i.else,`${n}.else`,r)):i.type==="WHILE_LOOP"&&(r[n]={description:"WHILE "+(i.condition?.expression||""),action_entity:void 0},i.body&&B(i.body,`${n}.body`,r))})};function X(e){if(!e?.statements||!Array.isArray(e.statements))return{};let t={};return B(e.statements,"main",t),e.teardown&&Array.isArray(e.teardown)&&B(e.teardown,"teardown",t),t}var _e=112;var at=1080-_e;var Se={"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"}};var xe={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"]},V=(e,t=!1)=>{let r=["chromium"];return t&&r.push("webkit"),xe[e].map(i=>Se[i]).filter(i=>i.defaultBrowserType&&r.includes(i.defaultBrowserType))};var ct={desktop:{label:"Desktop",type:"desktop",devices:V("desktop")},mobile:{label:"Mobile Web",type:"mobile",devices:V("mobile")}},lt={desktop:{label:"Desktop",type:"desktop",devices:V("desktop",!0)},mobile:{label:"Mobile Web",type:"mobile",devices:V("mobile",!0)}};var U=(e=>(e.DRAFT="DRAFT",e.STEP="STEP",e.ACTION="ACTION",e.IF_ELSE="IF_ELSE",e.WHILE_LOOP="WHILE_LOOP",e))(U||{});import{stringify as Ro}from"yaml";import{createHash as yt}from"crypto";import{parse as bt,stringify as Me}from"yaml";import{readFileSync as wt}from"fs";import{resolve as Pe,dirname as vt}from"path";import{parse as ke,stringify as _t}from"yaml";var $e=5e3;function mt(e){let t=e.frame_path;return!t||t.length===0?"page":`page.frameLocator('${t[0]}')`}function gt(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 Ee(e){let t=mt(e),r=e.locator;if(typeof r=="string"&&r.trim())return r=r.trim(),r.endsWith("first()")?`${t}.${r}`:`${t}.${r}.first()`;let i=gt(e);if(i){let o=JSON.stringify(i);return`${t}.locator(${o}).first()`}return null}var M=new Map;function g(e,t){M.set(e,t)}function F(e,t,r=[]){let i=[...r];return t.locator?i.push(`locator: ${JSON.stringify(t.locator)}`):t.xpath&&i.push(`xpath: ${JSON.stringify(t.xpath)}`),t.frame_path&&t.frame_path.length>0&&i.push(`frame_path: ${JSON.stringify(t.frame_path)}`),i.length===0?[`await agent.execAction("${e}", page, {});`]:[`await agent.execAction("${e}", page, {`,...i.map(o=>` ${o},`),"});"]}g("click",e=>{let t=Ee(e);if(!t)return['await agent.execAction("click", page, {});'];let r=e.action_data?.kwargs?.timeout_ms??$e;return[`await ${t}.click({ timeout: ${r} });`]});g("click_element",M.get("click"));g("click_element_by_index",M.get("click"));g("double_click",e=>F("double_click",e));g("double_click_on_element",M.get("double_click"));g("right_click",e=>F("right_click",e));g("right_click_on_element",M.get("right_click"));g("hover",e=>F("hover",e));g("hover_element_by_index",M.get("hover"));g("input_text",e=>{let t=e.action_data?.kwargs?.text??e.action_data?.kwargs?.value??"";return F("input_text",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});g("fill",M.get("input_text"));g("clear_input",e=>F("clear_input",e));g("press",e=>{let t=e.action_data?.kwargs?.keys;return[`await page.keyboard.press(${JSON.stringify(t)});`]});g("send_keys",M.get("press"));g("send_keys_on_element",e=>{let t=Ee(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 i=e.action_data?.kwargs?.timeout_ms??$e;return[`await ${t}.press(${JSON.stringify(r)}, { timeout: ${i} });`]});g("select_dropdown_option",e=>{let t=e.action_data?.kwargs?.text||e.action_data?.kwargs?.option||"";return F("select_dropdown_option",e,[`action_data: { kwargs: { text: ${JSON.stringify(t)} } }`])});g("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)})');`]});g("scroll_down",M.get("scroll"));g("scroll_up",M.get("scroll"));g("scroll_element",M.get("scroll"));g("scroll_to_text",e=>{let t=e.action_data?.kwargs?.text||"";return[`await page.getByText(${JSON.stringify(t)}, { exact: false }).first().scrollIntoViewIfNeeded();`]});g("scroll_on_element",e=>F("scroll_on_element",e,[`action_data: { kwargs: ${JSON.stringify(e.action_data?.kwargs||{})} }`]));g("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)} } },`,"});"]});g("open_tab",M.get("go_to_url"));g("go_back",()=>['await agent.execAction("go_back", page, {});']);g("reload_page",()=>['await agent.execAction("reload_page", page, {});']);g("wait",e=>[`await page.waitForTimeout(${(e.action_data?.kwargs?.seconds||1)*1e3});`]);g("wait_for_page_ready",()=>["await page.waitForLoadState('domcontentloaded');"]);g("verify",(e,t)=>{let r=e.action_data?.kwargs,i=typeof r?.code=="string",o=i?r?.statement||e.action_description:e.action_description||r?.statement;if(i&&o){let s=r.code.split(`
5
+ `),c=JSON.stringify(o);return["{ const _t = Date.now(); try {",...s.map(d=>` ${d}`),` console.log(\`[VERIFY:JS] \u2713 \${((Date.now()-_t)/1000).toFixed(1)}s: ${c}\`);`,"} 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: ${c}\`);`,` await agent.assert(page, ${c}, ${JSON.stringify(t||"")});`,"} }"]}return i?r.code.split(`
6
+ `):o?[`await agent.assert(page, ${JSON.stringify(o)}, ${JSON.stringify(t||"")});`]:["// Skipping verify: missing statement or code"]});g("ai_assert",M.get("verify"));g("assert",M.get("verify"));g("ai_action",(e,t)=>{let r=e.action_data?.kwargs?.statement;if(!r)return["// Skipping ai_action: missing statement"];let i=JSON.stringify(r),o=e.action_data?.kwargs?.use_pure_vision;return[`await agent.execute(page, ${i}, '${t||""}', ${o});`]});g("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"]});g("ai_extract",(e,t)=>{let r=e.action_data?.kwargs?.element_description,i=e.action_data?.kwargs?.variable_name;if(!r||!i)return["// Skipping ai_extract: missing element_description or variable_name"];let o=JSON.stringify(r),n=JSON.stringify(i);return[`await agent.extract(page, ${o}, ${n}, '${t||""}');`]});g("ai_wait_until",(e,t)=>{let r=e.action_data?.kwargs?.condition,i=e.action_data?.kwargs?.timeout_seconds||60;return r?[`await agent.waitUntilCondition(page, ${JSON.stringify(r)}, ${i}, '${t||""}');`]:["// Skipping ai_wait_until: missing condition"]});g("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)} } },`,"});"]});g("js_code",e=>{let t=e.action_data?.kwargs?.code;if(!t)return["// Skipping js_code: missing code"];let r=["{"],i=t.split(`
7
+ `);for(let o of i)r.push(` ${o}`);return r.push("}"),r});g("function",(e,t,r)=>{let i=e.action_data?.kwargs||{},o=i.functionName;if(o&&o.includes("#")){let[s,c]=o.split("#");if(s&&c){let d=s.replace(/\.(ts|js|mjs)$/,""),p=`import { ${c} } from '${d}';`;r?.imports?.add(p);let m={...i,functionName:c},b=Te(m);return b?[b.endsWith(";")?b:`${b};`]:["// Skipping function: invalid export pattern"]}}let n=Te(i);return n?[n.endsWith(";")?n:`${n};`]:["// Skipping function: missing functionName"]});g("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)} } },`,"});"]});g("upload_file",e=>{let t=e.action_data?.kwargs||{},r=[],i={};return t.paths?i.paths=t.paths:t.path&&(i.path=t.path),t.use_file_input&&(i.use_file_input=!0),r.push(`action_data: { kwargs: ${JSON.stringify(i)} }`),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(o=>` ${o},`),"});"]});g("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} } },`,"});"]);g("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} } },`,"});"]);g("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} } },`,"});"]});g("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(i=>` ${i},`),"});"]});g("done",()=>["// Done - no action needed"]);g("js_action",e=>{let t=e.action_data?.kwargs?.code;return t?t.split(`
8
+ `):["// Skipping js_action: missing code"]});function Te(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 i=["page","testContext","request","agent"],o=["undefined","null","true","false"],n=r.map(s=>i.includes(s)||o.includes(s)||/^-?\d+(\.\d+)?$/.test(s)?s:s.startsWith("$")?`agent.agentServices.readVariable('${s.substring(1)}')`:`"${s}"`);return`await ${t}(${n.join(", ")})`}var Ae=5;function Ie(e,t){let r={expandingPaths:new Set([Pe(t)]),depth:0,referencedPaths:new Set},i={...e};Array.isArray(i.statements)&&(i.statements=R(i.statements,t,r)),Array.isArray(i.teardown)&&(i.teardown=R(i.teardown,t,r));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(i[o])&&(i[o]=R(i[o],t,r));return{doc:i,referencedTemplatePaths:Array.from(r.referencedPaths)}}function R(e,t,r){let i=[];for(let o of e)if(St(o)){let n=xt(o,t,r);i.push(...n)}else i.push(Tt(o,t,r));return i}function St(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function xt(e,t,r){if(r.depth>=Ae)throw new Error(`Template expansion exceeded maximum depth of ${Ae}. Check for deeply nested or circular template references.`);let i=Pe(vt(t),e.template);if(r.expandingPaths.has(i))throw new Error(`Circular template reference detected: ${i} is already being expanded. Stack: ${Array.from(r.expandingPaths).join(" \u2192 ")} \u2192 ${i}`);r.referencedPaths.add(i);let o;try{o=wt(i,"utf-8")}catch(m){throw new Error(`Failed to read template file: ${i} (referenced from ${t}): ${m.message}`)}let n=ke(o);if(!n||typeof n!="object")throw new Error(`Invalid template file: ${i} \u2014 expected a YAML object`);let s=n.params||[],c=e.params||{};for(let m of s)if(!(m in c))throw new Error(`Template ${e.template} requires param "${m}" but it was not provided. Required params: [${s.join(", ")}]`);let d=n.statements;if(!Array.isArray(d))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(c).length>0){let b=_t(d);for(let[u,h]of Object.entries(c))b=b.split(`<<${u}>>`).join(String(h));d=ke(b)}let p={expandingPaths:new Set([...r.expandingPaths,i]),depth:r.depth+1,referencedPaths:r.referencedPaths};return R(d,i,p)}function Tt(e,t,r){if(typeof e!="object"||e===null)return e;let i={...e};return Array.isArray(i.statements)&&(i.statements=R(i.statements,t,r)),Array.isArray(i.THEN)&&(i.THEN=R(i.THEN,t,r)),Array.isArray(i.ELSE)&&(i.ELSE=R(i.ELSE,t,r)),Array.isArray(i.DO)&&(i.DO=R(i.DO,t,r)),i}var Oe=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Le(e,t){let r=bt(e),i=r?.name,o=r?.tags,n=r?.use;if(r&&(r.name!==void 0||r.tags!==void 0||r.use!==void 0)&&(delete r.name,delete r.tags,delete r.use),r?.suite){if(r.goal||r.statements)throw new Oe('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return At(r,i,o,n,t)}return kt(r,i,o,n,t)}function kt(e,t,r,i,o){let n=e?.beforeEach,s=e?.afterEach,c=Ce(e?.parameters),d=e?.timeout,p=e?.skip,m=e?.fail,b=e?.only,u=e?.slow;if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow),e?.url)throw new Oe(`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 h=[];if(o&&e&&typeof e=="object"){let S=Ie(e,o);e=S.doc,h=S.referencedTemplatePaths}let f=Me(e),a=D(f);return o&&(j(a.statements??[],o,"main"),a.teardown&&j(a.teardown,o,"teardown")),{testFlow:a,name:t,tags:r,use:i,beforeEach:n,afterEach:s,parameters:c,timeout:d,skip:p,fail:m,only:b,slow:u,referencedTemplatePaths:h}}function At(e,t,r,i,o){let n=e.suite;if(!Array.isArray(n.tests)||n.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=n.beforeAll,c=n.afterAll,d=n.beforeEach,p=n.afterEach,m=[],b=n.tests.map(f=>{if(!f.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(f.statements)||f.statements.length===0)throw new Error(`Suite test "${f.name}" must have a non-empty "statements" array.`);let a={goal:f.name,statements:f.statements};f.teardown&&(a.teardown=f.teardown);let S=[],y=a;if(o&&typeof a=="object"){let _=Ie(a,o);y=_.doc,S=_.referencedTemplatePaths,m.push(...S)}let w=Me(y),v=D(w),T=Ce(f.parameters);return{testFlow:v,name:f.name,tags:Array.isArray(f.tags)?f.tags:void 0,parameters:T,timeout:f.timeout,skip:f.skip,fail:f.fail,only:f.only,slow:f.slow}}),u=n.base_url,h=u?{...i,baseURL:u}:i;return{suite:{beforeAll:s,afterAll:c,beforeEach:d,afterEach:p,tests:b},name:t,tags:r,use:h,referencedTemplatePaths:m}}function Ce(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 j(e,t,r){for(let i=0;i<e.length;i++){let o=e[i],n=`${r}.${i}`,s=o.description||"";if(o.uid=$t(t,n,s),o.type===U.STEP)j(o.statements,t,n);else if(o.type===U.IF_ELSE){let c=o;j(c.then,t,`${n}.then`),c.else&&j(c.else,t,`${n}.else`)}else o.type===U.WHILE_LOOP&&j(o.body,t,`${n}.body`)}}function $t(e,t,r){let i=yt("sha256").update(`${e}:${t}:${r}`).digest("hex");return`${i.slice(0,8)}-${i.slice(8,12)}-${i.slice(12,16)}-${i.slice(16,20)}-${i.slice(20,32)}`}function A(e){return e.replace(/&/g,"&amp;").replace(/</g,"&lt;").replace(/>/g,"&gt;").replace(/"/g,"&quot;").replace(/'/g,"&#039;")}function Z(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 Ne(e){return`passed after ${e||"?"} ${e===1?"retry":"retries"}`}function q(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 Et(e){return`<span class="badge badge-${e}">${e}</span>`}function Mt(e){if(!e.code)return"";let t=e.code.split(`
9
+ `);return e.codeStartLine!=null&&e.codeLine!=null?`<div class="step-code"><pre class="code-block">${t.map((o,n)=>{let s=e.codeStartLine+n,c=s===e.codeLine,d=String(s).padStart(4);return`<span class="code-line${c?" code-line-active":""}">${d} \u2502 ${A(o)}</span>`}).join("")}</pre></div>`:`<div class="step-code"><pre class="code-block">${t.map(i=>`<span class="code-line code-line-body">${A(i)}</span>`).join("")}</pre></div>`}function Pt(e){let t=e.duration!=null?`<span class="step-duration">${Z(e.duration)}</span>`:"",r=q(e.status);if(e.screenshot||e.message||e.error||e.code){let o="";e.screenshot&&(o=`<img src="${A(e.screenshot)}" alt="Step screenshot" class="step-screenshot" />`);let n=e.screenshot?"":Mt(e),s="";e.error&&(s=`<div class="step-error"><pre>${A(e.error)}</pre></div>`);let c="";e.message&&!e.error&&(c=`<div class="step-message">${A(e.message)}</div>`);let d=e.status==="failure"?" open":"";return`
10
+ <details class="step-details step step-${e.status}"${d}>
10
11
  <summary class="step-header">
11
12
  ${r}
12
13
  <span class="step-id">${A(e.stepId)}</span>
@@ -15,9 +16,10 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
15
16
  </summary>
16
17
  <div class="step-expanded">
17
18
  ${o}
19
+ ${n}
18
20
  <div class="step-description-full">${A(e.description)}</div>
21
+ ${c}
19
22
  ${s}
20
- ${n}
21
23
  </div>
22
24
  </details>`}return`
23
25
  <div class="step step-${e.status}">
@@ -27,7 +29,7 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
27
29
  <span class="step-description">${A(e.description)}</span>
28
30
  ${t}
29
31
  </div>
30
- </div>`}function Mt(e,t,r,i){let o=[];e&&o.push(`
32
+ </div>`}function It(e,t,r,i){let o=[];e&&o.push(`
31
33
  <details class="artifact-section">
32
34
  <summary class="artifact-summary">Video</summary>
33
35
  <div class="artifact-content">
@@ -38,15 +40,15 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
38
40
  <button class="enlarge-btn" onclick="openVideoOverlay(this)" title="Enlarge">&#x26F6;</button>
39
41
  </div>
40
42
  </div>
41
- </details>`);let n=r.filter(s=>s.screenshot).map(s=>({src:s.screenshot,stepId:s.stepId,description:s.description,status:s.status,message:s.message||s.error||""}));if(n.length>0){let s=A(JSON.stringify(n)),a=n.map((d,u)=>`
42
- <div class="screenshot-thumb" onclick="openGalleryAt(this, ${u})" data-gallery="${s}">
43
+ </details>`);let n=r.filter(s=>s.screenshot).map(s=>({src:s.screenshot,stepId:s.stepId,description:s.description,status:s.status,message:s.message||s.error||""}));if(n.length>0){let s=A(JSON.stringify(n)),c=n.map((d,p)=>`
44
+ <div class="screenshot-thumb" onclick="openGalleryAt(this, ${p})" data-gallery="${s}">
43
45
  <img src="${A(d.src)}" alt="${A(d.stepId)}" />
44
46
  <span class="thumb-label">${A(d.stepId)}</span>
45
47
  </div>`).join("");o.push(`
46
48
  <details class="artifact-section">
47
49
  <summary class="artifact-summary">Screenshots (${n.length})</summary>
48
50
  <div class="artifact-content">
49
- <div class="screenshot-grid">${a}</div>
51
+ <div class="screenshot-grid">${c}</div>
50
52
  </div>
51
53
  </details>`)}if(t){let s=A(t);o.push(`
52
54
  <details class="artifact-section">
@@ -59,40 +61,40 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
59
61
  <p class="trace-hint">Run this command in your terminal to open the interactive Trace Viewer</p>
60
62
  <p class="trace-hint"><a href="${s}" class="attachment-link" download>Download trace.zip</a></p>
61
63
  </div>
62
- </details>`)}return o.length===0?"":`<div class="test-artifacts">${o.join("")}</div>`}function Ne(e,t,r,i,o){let n=e.map(Et).join(`
63
- `),s="";t&&!e.some(d=>d.error)&&(s=`<div class="test-error"><pre>${A(t)}</pre></div>`);let a=Mt(r,i,e,o);return`
64
+ </details>`)}return o.length===0?"":`<div class="test-artifacts">${o.join("")}</div>`}function Re(e,t,r,i,o){let n=e.map(Pt).join(`
65
+ `),s="";t&&!e.some(d=>d.error)&&(s=`<div class="test-error"><pre>${A(t)}</pre></div>`);let c=It(r,i,e,o);return`
64
66
  ${s}
65
67
  <div class="steps-list">
66
68
  ${n||'<div class="no-steps">No YAML step details available</div>'}
67
69
  </div>
68
- ${a}`}function Pt(e,t){let r=e.flaky?"flaky":e.status,i=q(r),o;if(e.flaky&&e.attempts&&e.attempts.length>1){let n=`tabs-${t}`,s=e.attempts.length,a=e.attempts.map((u,f)=>{let b=f===s-1,p=u.status==="passed"?"passed":"failed",h=`Attempt ${u.attemptNumber}`;return`<button class="attempt-tab ${b?"active":""} attempt-tab-${p}"
69
- onclick="switchAttemptTab('${n}', ${f})"
70
- data-tab-index="${f}">${q(p)} ${h} <span class="attempt-tab-badge badge-${p}">${u.status}</span></button>`}).join(""),d=e.attempts.map((u,f)=>{let b=f===s-1,p=Ne(u.steps,u.error,u.videoPath,u.tracePath,`${t}-attempt-${f}`);return`<div class="attempt-panel ${b?"active":""}" data-panel-index="${f}">
70
+ ${c}`}function Ot(e,t){let r=e.flaky?"flaky":e.status,i=q(r),o;if(e.flaky&&e.attempts&&e.attempts.length>1){let n=`tabs-${t}`,s=e.attempts.length,c=e.attempts.map((p,m)=>{let b=m===s-1,u=p.status==="passed"?"passed":"failed",h=`Attempt ${p.attemptNumber}`;return`<button class="attempt-tab ${b?"active":""} attempt-tab-${u}"
71
+ onclick="switchAttemptTab('${n}', ${m})"
72
+ data-tab-index="${m}">${q(u)} ${h} <span class="attempt-tab-badge badge-${u}">${p.status}</span></button>`}).join(""),d=e.attempts.map((p,m)=>{let b=m===s-1,u=Re(p.steps,p.error,p.videoPath,p.tracePath,`${t}-attempt-${m}`);return`<div class="attempt-panel ${b?"active":""}" data-panel-index="${m}">
71
73
  <div class="attempt-meta">
72
- ${q(u.status==="passed"?"passed":"failed")}
73
- <span class="attempt-meta-text">Attempt ${u.attemptNumber} &mdash; ${u.status} in ${Z(u.duration)}</span>
74
+ ${q(p.status==="passed"?"passed":"failed")}
75
+ <span class="attempt-meta-text">Attempt ${p.attemptNumber} &mdash; ${p.status} in ${Z(p.duration)}</span>
74
76
  </div>
75
- ${p}
77
+ ${u}
76
78
  </div>`}).join("");o=`
77
- <div class="flaky-note">Flaky &mdash; ${Ce(e.retries)}</div>
79
+ <div class="flaky-note">Flaky &mdash; ${Ne(e.retries)}</div>
78
80
  <div class="attempt-tabs" id="${n}">
79
- <div class="attempt-tab-bar">${a}</div>
81
+ <div class="attempt-tab-bar">${c}</div>
80
82
  ${d}
81
83
  </div>`}else o=`
82
- ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${Ce(e.retries)}</div>`:""}
83
- ${Ne(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
84
+ ${e.flaky?`<div class="flaky-note">Flaky &mdash; ${Ne(e.retries)}</div>`:""}
85
+ ${Re(e.steps,e.error,e.videoPath,e.tracePath,String(t))}`;return`
84
86
  <details class="test-details" ${e.status==="failed"||e.status==="timedOut"?"open":""}>
85
87
  <summary class="test-summary test-${r}">
86
88
  ${i}
87
89
  <span class="test-title">${A(e.title)}</span>
88
90
  <span class="test-file">${A(e.file)}</span>
89
- ${$t(r)}
91
+ ${Et(r)}
90
92
  <span class="test-duration">${Z(e.duration)}</span>
91
93
  </summary>
92
94
  <div class="test-body">
93
95
  ${o}
94
96
  </div>
95
- </details>`}function Re(e){let t=e.tests.filter(a=>a.flaky).length,r=e.tests.filter(a=>a.status==="passed"&&!a.flaky).length,i=e.tests.filter(a=>a.status==="failed"||a.status==="timedOut").length,o=e.tests.filter(a=>a.status==="skipped").length,n=e.tests.length,s=e.tests.map((a,d)=>Pt(a,d)).join(`
97
+ </details>`}function De(e){let t=e.tests.filter(c=>c.flaky).length,r=e.tests.filter(c=>c.status==="passed"&&!c.flaky).length,i=e.tests.filter(c=>c.status==="failed"||c.status==="timedOut").length,o=e.tests.filter(c=>c.status==="skipped").length,n=e.tests.length,s=e.tests.map((c,d)=>Ot(c,d)).join(`
96
98
  `);return`<!DOCTYPE html>
97
99
  <html lang="en">
98
100
  <head>
@@ -686,6 +688,38 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
686
688
  .gallery-status.failure { background: var(--color-failed); }
687
689
  .gallery-status.skipped { background: var(--color-skipped); }
688
690
  .gallery-status.pending { background: var(--color-pending); }
691
+
692
+ .step-code {
693
+ margin: 6px 0 4px;
694
+ border-radius: 4px;
695
+ overflow: hidden;
696
+ border: 1px solid var(--color-border);
697
+ }
698
+ .code-block {
699
+ background: #0d1117;
700
+ padding: 4px 0;
701
+ font-family: 'SFMono-Regular', Consolas, 'Liberation Mono', Menlo, monospace;
702
+ font-size: 12px;
703
+ line-height: 1.6;
704
+ overflow-x: auto;
705
+ display: block;
706
+ margin: 0;
707
+ }
708
+ .code-line {
709
+ display: block;
710
+ padding: 0 12px;
711
+ white-space: pre;
712
+ color: var(--color-text-secondary);
713
+ }
714
+ .code-line-active {
715
+ background: rgba(100, 181, 246, 0.1);
716
+ color: var(--color-text);
717
+ border-left: 2px solid var(--color-accent);
718
+ padding-left: 10px;
719
+ }
720
+ .code-line-body {
721
+ color: var(--color-text);
722
+ }
689
723
  </style>
690
724
  </head>
691
725
  <body>
@@ -842,7 +876,9 @@ Shiplight cloud report: ${g.data.reportUrl}`)}var Fe=Je(()=>{"use strict"});impo
842
876
  });
843
877
  </script>
844
878
  </body>
845
- </html>`}var Ft={before:0,main:1,teardown:2,after:3};function We(e){let t=e.split(".")[0];return Ft[t]??1}function Ge(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Wt(e){return[...e].sort(([t],[r])=>{let i=We(t),o=We(r);if(i!==o)return i-o;let n=Ge(t),s=Ge(r);for(let a=0;a<Math.max(n.length,s.length);a++){let d=n[a]??-1,u=s[a]??-1;if(d!==u)return d-u}return 0})}var se=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud??!1,this.apiKey=t.apiKey||process.env.__SHIPLIGHT_API_KEY}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let u of this.collected){let f=u.test.titlePath().join(" > "),b=r.get(f);b||(b=[],r.set(f,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let f=u[0].test.location.file,b=[],p,h,g;for(let v=0;v<u.length;v++){let{test:x,result:_}=u[v],k=await this.buildReportTest(x,_,f);p=k,h||(h=k.startTime),g=k.endTime,b.push({attemptNumber:v+1,status:_.status,duration:_.duration,steps:k.steps,error:k.error,videoPath:k.videoPath,tracePath:k.tracePath})}let l=b[b.length-1],{test:T}=u[u.length-1],y={title:T.title,baseTitle:p?.baseTitle,file:S.relative(process.cwd(),f),status:l.status,duration:l.duration,steps:l.steps,error:l.error,videoPath:l.videoPath,tracePath:l.tracePath,actionStepsMap:p?.actionStepsMap,tags:p?.tags,baseUrl:p?.baseUrl,skip:p?.skip,slow:p?.slow,timeout:p?.timeout,parameterSetName:p?.parameterSetName,startTime:h,endTime:g,suiteName:p?.suiteName},w=b.some(v=>v.status==="failed"||v.status==="timedOut");b.length>1&&w&&l.status==="passed"&&(y.flaky=!0,y.retries=b.length-1,y.attempts=b),i.push(y)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString()},n=S.isAbsolute(this.outputFolder)?this.outputFolder:S.join(process.cwd(),this.outputFolder);$.mkdirSync(n,{recursive:!0});let s=S.join(n,"screenshots");for(let u=0;u<o.tests.length;u++){let f=o.tests[u],b=f.attempts&&f.attempts.length>0,p=[{obj:f,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(f.attempts)for(let h=0;h<f.attempts.length;h++)p.push({obj:f.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:g,screenshotSubDir:l}of p){let T=S.join(s,l),y=!1;for(let w of h.steps)if(w.screenshot&&S.isAbsolute(w.screenshot))try{y||($.mkdirSync(T,{recursive:!0}),y=!0);let v=`${w.stepId.replace(/\./g,"-")}.png`;$.copyFileSync(w.screenshot,S.join(T,v)),w.screenshot=`screenshots/${l}/${v}`}catch(v){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,v)}if(h.videoPath&&S.isAbsolute(h.videoPath)){let w=S.extname(h.videoPath)||".webm",v=`${g}-video${w}`;try{$.copyFileSync(h.videoPath,S.join(n,v)),h.videoPath=v}catch{h.videoPath=void 0}}if(h.tracePath&&S.isAbsolute(h.tracePath)){let w=S.extname(h.tracePath)||".zip",v=`${g}-trace${w}`;try{$.copyFileSync(h.tracePath,S.join(n,v)),h.tracePath=v}catch{h.tracePath=void 0}}}}let a=S.join(n,"report-data.json");$.writeFileSync(a,JSON.stringify(o,null,2),"utf-8");let d=S.join(n,"index.html");if($.writeFileSync(d,Re(o),"utf-8"),console.log(`
846
- Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(d)}catch{}if(this.reportToCloud){if(!this.apiKey){console.warn("[reporter] reportToCloud is enabled but no apiKey found, skipping cloud upload.");return}try{let{uploadToCloud:u}=await Promise.resolve().then(()=>(Fe(),Ue));await u(o,n,this.runStartTime,this.apiKey)}catch(u){console.warn("[reporter] Cloud upload failed:",u)}}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:S.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(p=>p.message||p.stack||String(p)).join(`
879
+ </html>`}var Gt={before:0,main:1,teardown:2,after:3};function Ge(e){let t=e.split(".")[0];return Gt[t]??1}function He(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function Ht(e){return[...e].sort(([t],[r])=>{let i=Ge(t),o=Ge(r);if(i!==o)return i-o;let n=He(t),s=He(r);for(let c=0;c<Math.max(n.length,s.length);c++){let d=n[c]??-1,p=s[c]??-1;if(d!==p)return d-p}return 0})}function Kt(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function jt(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function se(e,t="main",r,i,o){r===void 0&&(r=!Kt(e).has("test.step")),o||(o=new Map);let n=[];for(let s of e){if(s.category==="fixture"||s.category==="test.attach")continue;if(s.category==="hook"){let d=jt(s.title,t);n.push(...se(s.steps,d,r,i,o));continue}if(s.category==="test.step"||r&&(s.category==="expect"||s.category==="pw:api")){let d=o.get(t)??0;o.set(t,d+1);let p=`${t}.${d}`,m={stepId:p,description:s.title,status:s.error?"failure":s.duration===-1?"skipped":"success",duration:s.duration>=0?s.duration:void 0};s.error&&(m.error=s.error.message??s.error.stack),i&&s.location&&i.set(p,s.location),n.push(m),s.steps.length>0&&n.push(...se(s.steps,p,r,i,o))}}return n}var ae=class{outputFolder;openMode;reportToCloud;apiKey;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure",this.reportToCloud=t.reportToCloud??!1,this.apiKey=t.apiKey||process.env.__SHIPLIGHT_API_KEY}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let p of this.collected){let m=p.test.titlePath().join(" > "),b=r.get(m);b||(b=[],r.set(m,b)),b.push(p)}let i=[];for(let[,p]of r.entries()){let m=p[0].test.location.file,b=[],u,h,f;for(let v=0;v<p.length;v++){let{test:T,result:_}=p[v],k=await this.buildReportTest(T,_,m);u=k,h||(h=k.startTime),f=k.endTime,b.push({attemptNumber:v+1,status:_.status,duration:_.duration,steps:k.steps,error:k.error,videoPath:k.videoPath,tracePath:k.tracePath})}let a=b[b.length-1],{test:S}=p[p.length-1],y={title:S.title,baseTitle:u?.baseTitle,file:x.relative(process.cwd(),m),status:a.status,duration:a.duration,steps:a.steps,error:a.error,videoPath:a.videoPath,tracePath:a.tracePath,actionStepsMap:u?.actionStepsMap,tags:u?.tags,baseUrl:u?.baseUrl,skip:u?.skip,slow:u?.slow,timeout:u?.timeout,parameterSetName:u?.parameterSetName,startTime:h,endTime:f,suiteName:u?.suiteName},w=b.some(v=>v.status==="failed"||v.status==="timedOut");b.length>1&&w&&a.status==="passed"&&(y.flaky=!0,y.retries=b.length-1,y.attempts=b),i.push(y)}let o={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString()},n=x.isAbsolute(this.outputFolder)?this.outputFolder:x.join(process.cwd(),this.outputFolder);$.mkdirSync(n,{recursive:!0});let s=x.join(n,"screenshots");for(let p=0;p<o.tests.length;p++){let m=o.tests[p],b=m.attempts&&m.attempts.length>0,u=[{obj:m,prefix:b?`test-${p}-attempt-0`:`test-${p}`,screenshotSubDir:`test-${p}`}];if(m.attempts)for(let h=0;h<m.attempts.length;h++)u.push({obj:m.attempts[h],prefix:`test-${p}-attempt-${h+1}`,screenshotSubDir:`test-${p}/attempt-${h}`});for(let{obj:h,prefix:f,screenshotSubDir:a}of u){let S=x.join(s,a),y=!1;for(let w of h.steps)if(w.screenshot&&x.isAbsolute(w.screenshot))try{y||($.mkdirSync(S,{recursive:!0}),y=!0);let v=`${w.stepId.replace(/\./g,"-")}.png`;$.copyFileSync(w.screenshot,x.join(S,v)),w.screenshot=`screenshots/${a}/${v}`}catch(v){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,v)}if(h.videoPath&&x.isAbsolute(h.videoPath)){let w=x.extname(h.videoPath)||".webm",v=`${f}-video${w}`;try{$.copyFileSync(h.videoPath,x.join(n,v)),h.videoPath=v}catch{h.videoPath=void 0}}if(h.tracePath&&x.isAbsolute(h.tracePath)){let w=x.extname(h.tracePath)||".zip",v=`${f}-trace${w}`;try{$.copyFileSync(h.tracePath,x.join(n,v)),h.tracePath=v}catch{h.tracePath=void 0}}}}let c=x.join(n,"report-data.json");$.writeFileSync(c,JSON.stringify(o,null,2),"utf-8");let d=x.join(n,"index.html");if($.writeFileSync(d,De(o),"utf-8"),console.log(`
880
+ Shiplight report written to: ${d}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let p=(await import("open")).default;await p(d)}catch{}if(this.reportToCloud){if(!this.apiKey){console.warn("[reporter] reportToCloud is enabled but no apiKey found, skipping cloud upload.");return}try{let{uploadToCloud:p}=await Promise.resolve().then(()=>(We(),Fe));await p(o,n,this.runStartTime,this.apiKey)}catch(p){console.warn("[reporter] Cloud upload failed:",p)}}}printsToStdio(){return!1}async buildReportTest(t,r,i){let o={title:t.title,file:x.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(o.error=r.errors.map(u=>u.message||u.stack||String(u)).join(`
847
881
 
848
- `)),r.stdout.length>0&&(o.stdout=r.stdout.map(p=>typeof p=="string"?p:p.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(p=>typeof p=="string"?p:p.toString()).join(""));for(let p of r.attachments)p.name==="video"&&p.path&&(o.videoPath=p.path),p.name==="trace"&&p.path&&(o.tracePath=p.path);let n=r.attachments.find(p=>p.name==="shiplight-results"),s=null;if(n)try{if(n.body)s=JSON.parse(n.body.toString("utf-8"));else if(n.path){let p=$.readFileSync(n.path,"utf-8");s=JSON.parse(p)}}catch{}let a=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),f=u?u[1]:t.title,b=u?u[2]:void 0;if(b&&(o.parameterSetName=b),$.existsSync(a))try{let p=$.readFileSync(a,"utf-8"),h=Oe(p,a);if(h.suite){let g=h.suite.tests.find(l=>l.name===f);g&&(d=X(g.testFlow),g.tags?.length&&(o.tags=g.tags),g.skip!==void 0&&(o.skip=g.skip),g.slow&&(o.slow=g.slow),g.timeout!==void 0&&(o.timeout=g.timeout),o.baseTitle=g.name||g.testFlow?.goal),o.suiteName=h.name,h.tags?.length&&(o.suiteTags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL)}else h.testFlow&&(d=X(h.testFlow),o.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(o.tags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL))}catch{}if(s||Object.keys(d).length>0){let p=new Set([...Object.keys(d),...Object.keys(s||{})]),h=Array.from(p).map(l=>[l,null]),g=Wt(h);for(let[l]of g){let T=d[l],y=s?.[l],w=T?.description;if(!w||w==="Action"||w==="Draft"){let x=T?.action_entity;w=x?.action_description||x?.action_data?.kwargs?.description||y?.description||l}let v={stepId:l,description:w,status:y?.status||"pending",duration:y?.duration};if(y?.message){let x=typeof y.message=="string"?y.message:JSON.stringify(y.message,null,2);y.status==="failure"?v.error=x:v.message=x}if(y?.screenshot){let x=y.screenshot,_=n?.path?S.dirname(n.path):"",k=S.isAbsolute(x)?x:S.join(_,x);$.existsSync(k)&&(v.screenshot=k)}o.steps.push(v)}}return Object.keys(d).length>0&&(o.actionStepsMap=d),o}},Gt=se;export{Gt as default};
882
+ `)),r.stdout.length>0&&(o.stdout=r.stdout.map(u=>typeof u=="string"?u:u.toString()).join("")),r.stderr.length>0&&(o.stderr=r.stderr.map(u=>typeof u=="string"?u:u.toString()).join(""));for(let u of r.attachments)u.name==="video"&&u.path&&(o.videoPath=u.path),u.name==="trace"&&u.path&&(o.tracePath=u.path);let n=r.attachments.find(u=>u.name==="shiplight-results"),s=null;if(n)try{if(n.body)s=JSON.parse(n.body.toString("utf-8"));else if(n.path){let u=$.readFileSync(n.path,"utf-8");s=JSON.parse(u)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),d={},p=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),m=p?p[1]:t.title,b=p?p[2]:void 0;if(b&&(o.parameterSetName=b),$.existsSync(c))try{let u=$.readFileSync(c,"utf-8"),h=Le(u,c);if(h.suite){let f=h.suite.tests.find(a=>a.name===m);f&&(d=X(f.testFlow),f.tags?.length&&(o.tags=f.tags),f.skip!==void 0&&(o.skip=f.skip),f.slow&&(o.slow=f.slow),f.timeout!==void 0&&(o.timeout=f.timeout),o.baseTitle=f.name||f.testFlow?.goal),o.suiteName=h.name,h.tags?.length&&(o.suiteTags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL)}else h.testFlow&&(d=X(h.testFlow),o.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(o.tags=h.tags),h.use?.baseURL&&(o.baseUrl=h.use.baseURL))}catch{}if(s||Object.keys(d).length>0){let u=new Set([...Object.keys(d),...Object.keys(s||{})]),h=Array.from(u).map(a=>[a,null]),f=Ht(h);for(let[a]of f){let S=d[a],y=s?.[a],w=S?.description;if(!w||w==="Action"||w==="Draft"){let T=S?.action_entity;w=T?.action_description||T?.action_data?.kwargs?.description||y?.description||a}let v={stepId:a,description:w,status:y?.status||"pending",duration:y?.duration};if(y?.message){let T=typeof y.message=="string"?y.message:JSON.stringify(y.message,null,2);y.status==="failure"?v.error=T:v.message=T}if(y?.screenshot){let T=y.screenshot,_=n?.path?x.dirname(n.path):"",k=x.isAbsolute(T)?T:x.join(_,T);$.existsSync(k)&&(v.screenshot=k)}v.code=y.code,o.steps.push(v)}}if(s===null&&Object.keys(d).length===0&&!i.endsWith(".yaml.spec.ts")){let u=new Map;if(o.steps=se(r.steps,"main",void 0,u),o.steps.length>0){let h=new Map;o.actionStepsMap=Object.fromEntries(o.steps.map(f=>{let a=u.get(f.stepId),S;if(a?.file){if(!h.has(a.file))try{h.set(a.file,$.readFileSync(a.file,"utf-8").split(`
883
+ `))}catch{h.set(a.file,[])}let w=h.get(a.file);S=w[a.line-1]?.trim();let v=a.line-2,T=a.line,_=[],k=a.line;v>=0&&(_.push(w[v]??""),k=a.line-1),_.push(w[a.line-1]??""),T<w.length&&_.push(w[T]??""),f.code=_.join(`
884
+ `),f.codeStartLine=k,f.codeLine=a.line}let y={description:f.description,...S&&{action_entity:{action_description:f.description,action_data:{action_name:"js_code",args:[],kwargs:{code:S}}}}};return[f.stepId,y]}))}}return Object.keys(d).length>0?o.actionStepsMap=d:s&&o.steps.length>0&&(o.actionStepsMap=Object.fromEntries(o.steps.map(u=>{let h=s[u.stepId],f=h?.type,a={description:u.description,...f&&{action_entity:{action_description:u.description,action_data:{action_name:f==="step"?"js_code":f,args:[],kwargs:f==="step"?{code:h?.code??u.description}:{statement:u.description}}}}};return[u.stepId,a]}))),o}},zt=ae;export{zt as default};
@@ -1,4 +1,4 @@
1
- import { r as reactExports, W as We } from "./index-DP4PtPwQ.js";
1
+ import { r as reactExports, W as We } from "./index-D0G607gU.js";
2
2
  function _defineProperty$1(obj, key, value) {
3
3
  if (key in obj) {
4
4
  Object.defineProperty(obj, key, {
@@ -649,4 +649,4 @@ export {
649
649
  loader,
650
650
  Le as useMonaco
651
651
  };
652
- //# sourceMappingURL=index-ClD-Ul7O.js.map
652
+ //# sourceMappingURL=index-ByMAsm4i.js.map