shiplightai 0.1.66 → 0.1.67

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.
@@ -4365,7 +4365,7 @@ ${p.join(`
4365
4365
  `)}function ed(e,t){let i=[],a=t?.version||"unknown";i.push(`// @generated by shiplightai v${a}`),i.push(...Zo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let n=new Set,o={imports:n,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=ia(t?.tags);i.push(`test.describe.serial('${s}${Se(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...Pi("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...Pi("beforeEach",e.beforeEach,o,1)),i.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],h=d.timeout||d.skip!==void 0||d.fail!==void 0||d.only||d.slow?{timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}:void 0;if(d.parameters&&d.parameters.length>0)for(let p of d.parameters){let g=qo(d.testFlow,p.values);i.push(...Li(g,`${Se(d.name)} [${Se(p.name)}]`,o,1,h)),i.push("")}else i.push(...Li(d.testFlow,Se(d.name),o,1,h)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&i.push("")}return e.afterEach&&e.afterEach.length>0&&(i.push(...Pi("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...Pi("afterAll",e.afterAll,o,1)),i.push("});"),Qo(i,n),i.join(`
4366
4366
  `)}function ia(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var td=["testContext","request"];function Oi(e){let t=new Set;function i(a){for(let n of a)switch(n.type){case me.ACTION:{let r=n.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&td.includes(s)&&t.add(s);break}case me.STEP:i(n.statements);break;case me.IF_ELSE:{let o=n;i(o.then),o.else&&i(o.else);break}case me.WHILE_LOOP:i(n.body);break}}return i(e),t}function id(e){let t=Oi(e.statements??[]);if(e.teardown)for(let i of Oi(e.teardown))t.add(i);return t}function aa(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function Li(e,t,i,a=0,n){let o=" ".repeat(a),r=[],s=id(e),l=aa(s),c=n?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),n?.skip===!0?r.push(`${o} test.skip();`):typeof n?.skip=="string"&&r.push(`${o} test.skip(true, '${Se(n.skip)}');`),n?.fail===!0?r.push(`${o} test.fail();`):typeof n?.fail=="string"&&r.push(`${o} test.fail(true, '${Se(n.fail)}');`),n?.slow&&r.push(`${o} test.slow();`),n?.timeout&&r.push(`${o} test.setTimeout(${n.timeout});`);let d=e.teardown&&e.teardown.length>0,h=a+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=ge(e.statements,h+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=ge(e.teardown,h+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=ge(e.statements,h,i);r.push(...p)}return r.push(`${o}});`),r}function Vo(e,t,i){let a=[],n=Jo(t),o=Oi(n),r=aa(o);return a.push(`test.${e}(async (${r}) => {`),a.push(...ge(n,1,i,e)),a.push("});"),a}function Pi(e,t,i,a){let n=" ".repeat(a),o=[],r=Jo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${n}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${n} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...ge(r,a+1,l,e)),o.push(`${n} await page.close();`),o.push(`${n}});`)}else{let l=Oi(r),c=aa(l);o.push(`${n}test.${e}(async (${c}) => {`),o.push(...ge(r,a+1,i,e)),o.push(`${n}});`)}return o}function Jo(e){let i=(0,Yo.stringify)({goal:"_hook",statements:e});return pe(i).statements??[]}function qo(e,t){let i=qn(e);for(let[a,n]of Object.entries(t))i=i.split(`<<${a}>>`).join(String(n));return pe(i)}function Zo(){return["import { test, expect } from 'shiplightai/fixture';"]}function Qo(e,t){if(t.size>0){let i=0;for(let n=0;n<e.length;n++)e[n].startsWith("import ")&&(i=n+1);let a=Array.from(t);e.splice(i,0,...a)}}var Xo=5;function tr(e,t,i){let a={expandingPaths:new Set([(0,qe.resolve)(t)]),depth:0,referencedPaths:new Set,basePath:i},n={...e};Array.isArray(n.statements)&&(n.statements=Re(n.statements,t,a)),Array.isArray(n.teardown)&&(n.teardown=Re(n.teardown,t,a));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(n[o])&&(n[o]=Re(n[o],t,a));return{doc:n,referencedTemplatePaths:Array.from(a.referencedPaths)}}function Re(e,t,i){let a=[];for(let n of e)if(nd(n)){let o=ad(n,t,i);a.push(o)}else a.push(od(n,t,i));return a}function nd(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function ad(e,t,i){if(i.depth>=Xo)throw new Error(`Template expansion exceeded maximum depth of ${Xo}. Check for deeply nested or circular template references.`);let a=(0,qe.resolve)((0,qe.dirname)(t),e.template),n=!(0,Ni.existsSync)(a)&&i.basePath?(0,qe.resolve)(i.basePath,e.template):a;if(i.expandingPaths.has(n))throw new Error(`Circular template reference detected: ${n} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${n}`);i.referencedPaths.add(n);let o;try{o=(0,Ni.readFileSync)(n,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${n} (referenced from ${t}): ${f.message}`)}let r=(0,vt.parse)(o);if(!r||typeof r!="object")throw new Error(`Invalid template file: ${n} \u2014 expected a YAML object`);let s=r.params||[],l=e.params||{};for(let f of s)if(!(f in l))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${s.join(", ")}]`);let c=r.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let y=(0,vt.stringify)(c);for(let[b,m]of Object.entries(l))y=y.split(`<<${b}>>`).join(String(m));c=(0,vt.parse)(y)}let d={expandingPaths:new Set([...i.expandingPaths,n]),depth:i.depth+1,referencedPaths:i.referencedPaths},h=Re(c,n,d),g={STEP:r.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:h};return Object.keys(l).length>0&&(g.template_params=l),g}function od(e,t,i){if(typeof e!="object"||e===null)return e;let a={...e};return Array.isArray(a.statements)&&(a.statements=Re(a.statements,t,i)),Array.isArray(a.THEN)&&(a.THEN=Re(a.THEN,t,i)),Array.isArray(a.ELSE)&&(a.ELSE=Re(a.ELSE,t,i)),Array.isArray(a.DO)&&(a.DO=Re(a.DO,t,i)),a}var oa=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function ra(e,t,i){let a=(0,_t.parse)(e),n=a?.name,o=a?.tags,r=a?.use;if(a&&(a.name!==void 0||a.tags!==void 0||a.use!==void 0)&&(delete a.name,delete a.tags,delete a.use),a?.suite){if(a.goal||a.statements)throw new oa('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return sd(a,n,o,r,t,i)}return rd(a,n,o,r,t,i)}function rd(e,t,i,a,n,o){let r=e?.beforeEach,s=e?.afterEach,l=ir(e?.parameters),c=e?.timeout,d=e?.skip,h=e?.fail,p=e?.only,g=e?.slow,f=e?.settings,y=a;if(f){let _={};f.auto_dismiss_modal!==void 0&&(_.autoDismissModal=!!f.auto_dismiss_modal),f.browser_timezone!==void 0&&f.browser_timezone!==null&&(_.timezoneId=String(f.browser_timezone)),f.browser_language!==void 0&&f.browser_language!==null&&(_.locale=String(f.browser_language)),f.extra_http_headers!==void 0&&f.extra_http_headers!==null&&typeof f.extra_http_headers=="object"&&(_.extraHTTPHeaders=f.extra_http_headers),Object.keys(_).length>0&&(y={...y,..._})}if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow,delete e.settings),e?.url)throw new oa(`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 b=[];if(n&&e&&typeof e=="object"){let _=tr(e,n,o);e=_.doc,b=_.referencedTemplatePaths}let m=(0,_t.stringify)(e),x=pe(m);return n&&(Je(x.statements??[],n,"main"),x.teardown&&Je(x.teardown,n,"teardown")),{testFlow:x,name:t,tags:i,use:y,beforeEach:r,afterEach:s,parameters:l,timeout:c,skip:d,fail:h,only:p,slow:g,referencedTemplatePaths:b}}function sd(e,t,i,a,n,o){let r=e.suite;if(!Array.isArray(r.tests)||r.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=r.beforeAll,l=r.afterAll,c=r.beforeEach,d=r.afterEach,h=[],p=r.tests.map(y=>{if(!y.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(y.statements)||y.statements.length===0)throw new Error(`Suite test "${y.name}" must have a non-empty "statements" array.`);let b={goal:y.name,statements:y.statements};y.teardown&&(b.teardown=y.teardown);let m=[],x=b;if(n&&typeof b=="object"){let $=tr(b,n,o);x=$.doc,m=$.referencedTemplatePaths,h.push(...m)}let _=(0,_t.stringify)(x),E=pe(_),A=ir(y.parameters);return{testFlow:E,name:y.name,tags:Array.isArray(y.tags)?y.tags:void 0,parameters:A,timeout:y.timeout,skip:y.skip,fail:y.fail,only:y.only,slow:y.slow}}),g=r.base_url,f=g?{...a,baseURL:g}:a;return{suite:{beforeAll:s,afterAll:l,beforeEach:c,afterEach:d,tests:p},name:t,tags:i,use:f,referencedTemplatePaths:h}}function ir(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Je(e,t,i){for(let a=0;a<e.length;a++){let n=e[a],o=`${i}.${a}`,r=n.description||"";if(n.uid=ld(t,o,r),n.type===me.STEP)Je(n.statements,t,o);else if(n.type===me.IF_ELSE){let s=n;Je(s.then,t,`${o}.then`),s.else&&Je(s.else,t,`${o}.else`)}else n.type===me.WHILE_LOOP&&Je(n.body,t,`${o}.body`)}}function ld(e,t,i){let a=(0,er.createHash)("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${a.slice(0,8)}-${a.slice(8,12)}-${a.slice(12,16)}-${a.slice(16,20)}-${a.slice(20,32)}`}function ar(e,t,i){let a=/\btemplate:\s/.test(e),n=/^suite:/m.test(e),o=a||n?null:Qn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??ra(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},h=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=ed(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=Qc(c.testFlow,{...d,testName:c.name,tags:c.tags,use:h,beforeEach:c.beforeEach,afterEach:c.afterEach,parameters:c.parameters,timeout:c.timeout,skip:c.skip,fail:c.fail,only:c.only,slow:c.slow});let p=r.split(`
4367
4367
  `).filter(g=>!g.startsWith("import ")).join(`
4368
- `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),(0,kt.mkdirSync)((0,nr.dirname)(s),{recursive:!0}),(0,kt.writeFileSync)(s,r)}catch(c){let d=c instanceof oa?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var sa="0.1.66";function or(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var cd=`// @generated by shiplightai v${sa}`;function dd(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4368
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),(0,kt.mkdirSync)((0,nr.dirname)(s),{recursive:!0}),(0,kt.writeFileSync)(s,r)}catch(c){let d=c instanceof oa?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var sa="0.1.67";function or(e){try{return(0,Fe.statSync)(e).mtimeMs}catch{return 0}}var cd=`// @generated by shiplightai v${sa}`;function dd(e,t){if(!(0,Fe.existsSync)(e)||(0,Fe.readFileSync)(e,"utf-8").split(`
4369
4369
  `,1)[0]!==cd)return!1;let a=or(e);for(let n of t)if(or(n)>a)return!1;return!0}function ud(e){let t=process.argv.slice(2),i=[],a=(0,Ze.resolve)(e);for(let n of t){if(n.startsWith("-"))continue;let o=n.endsWith(".yaml.spec.ts")?n.replace(/\.yaml\.spec\.ts$/,".test.yaml"):n;if(!o.endsWith(".test.yaml"))continue;let r=(0,Ze.resolve)(e,o);(0,Fe.existsSync)(r)&&i.push(r.startsWith(a)?r.slice(a.length+1):o)}return i.length>0?i:null}function sr(e){let t=ud(e.cwd),i=t??(0,rr.globSync)("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),a=[];for(let n of i){let o=(0,Ze.resolve)(e.cwd,n),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=(0,Fe.readFileSync)(o,"utf-8");try{let l=ra(s,o,e.projectRoot??e.cwd),c=(0,Ze.relative)(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&dd(r,[o,...l.referencedTemplatePaths]))continue;let p=ar(s,o,{version:sa,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${n}:`,l),a.push({file:n,error:l})}}if(a.length>0){let n=`[shiplight] Transpilation failed for ${a.length} file(s):
4370
4370
  `+a.map(o=>` - ${o.file}`).join(`
4371
4371
  `);if(t)throw new Error(n);console.warn(n+" (skipped)")}}var re=X(require("fs"),1),St=X(require("path"),1),gr=require("glob");Fi();function fr(e){let t=Ri().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new ha:new ua(e)}var vd=".shiplight/action-cache";function pr(e){return e.replace(/\//g,"__")+".json"}function _d(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var ua=class{constructor(t){this.cwd=t;this.cacheDir=St.join(t,vd)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!re.existsSync(this.cacheDir))return i;for(let a of t){let n=St.join(this.cacheDir,pr(a));try{if(re.existsSync(n)){let o=re.readFileSync(n,"utf-8");i.set(a,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;re.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[a,n]of t)try{let o=St.join(this.cacheDir,pr(a)),r=ea();if(re.existsSync(o))try{r=JSON.parse(re.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...n.entries}};re.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!re.existsSync(this.cacheDir))return;let t=(0,gr.globSync)("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,a=0;for(let n of t)try{let o=re.readFileSync(St.join(this.cacheDir,n),"utf-8"),r=JSON.parse(o),s=_d(n);i.set(s,r),a+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${a} cached action entit${a===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},ha=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(da(),ca));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(da(),ca));return i(t)}loadAll(){}};Fi();function yr(e={}){e.dotenv!==!1&&kd(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),a=fr(t).loadAll();return sr({cwd:t,projectRoot:process.cwd(),actionEntityStores:a}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function xr(e,...t){return(0,br.defineConfig)(e,...t)}function kd(e){dr(e);for(let t of la(e))wr.default.config({path:t})}F();F();jt();nt();var ro=require("zod"),Sh=ro.z.object({instruction:ro.z.string().describe('The instruction of the operation to perform. Can only include one operation. Do not inlcude element indexes just describe the element, e.g. "select the text "Hello, world!" in "Hello, world!""')});function Dl(e){e.register({name:"perform_accurate_operation",description:"Perform an operation that requires accurate interaction like dragging or interacting with a specific area of an element. Only use this action when neccecary.",schema:Sh,usesElementIndex:!1,async execute(t,i){let{instruction:a}=t,n={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await it(a,n,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:a,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:a}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Bt(r,n);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}$e();F();var Rl=require("ai"),Fl=require("html-to-text");async function Eh(e,t,i){let{apiKey:a,domain:n}=e;if(!a||!n)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${n}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}u.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let h=(d.storage||{}).url;if(u.info(`message_url: ${h}`),!h){let f=(d.message||{}).headers||{},y=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!y.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:y,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${n}/messages/${g}`;try{let y=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`,Accept:"application/json"}});if(y.ok){let b=await y.json(),m=b.Subject||"",x=b.From||"",_=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${x}`),u.info(`to_addr: ${_}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let $=b["body-html"]||b["body-plain"]||"";if($&&$.includes("<")&&($=(0,Fl.convert)($)),u.info(`Body: ${$.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!$.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:x,to:_,date:E,body:$,message_id:A});continue}else u.warn(`Messages API returned ${y.status}`)}catch(y){u.warn(`Failed to parse JSON response: ${y}`)}}try{let f=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${a}`).toString("base64")}`}});if(!f.ok){u.warn(`Could not fetch stored message: ${f.status}`);continue}let y=await f.text();u.info(`Fallback: Raw email length: ${y.length}`);let b=y.split(`
@@ -873,7 +873,7 @@
873
873
  });
874
874
  </script>
875
875
  </body>
876
- </html>`}var Pe="0.1.66",Oe=Pe!=="dev"?Pe:void 0;var sn=3600*1e3,cn=10080*60*1e3;var wt={before:0,main:1,teardown:2,after:3};function Ie(e){let t=e.split(".")[0];return wt[t]??1}function Le(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function _t(e){return[...e].sort(([t],[r])=>{let i=Ie(t),n=Ie(r);if(i!==n)return i-n;let a=Le(t),o=Le(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function vt(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function St(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function te(e,t="main",r,i,n){r===void 0&&(r=!vt(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=St(o.title,t);a.push(...te(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,y={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(y.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(y),o.steps.length>0&&a.push(...te(o.steps,u,r,i,n))}}return a}var re=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let u of this.collected){let y=u.test.titlePath().join(" > "),b=r.get(y);b||(b=[],r.set(y,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let y=u[0].test.location.file,b=[],d,h,f;for(let g=0;g<u.length;g++){let{test:A,result:P}=u[g],$=await this.buildReportTest(A,P,y);d=$,h||(h=$.startTime),f=$.endTime,b.push({attemptNumber:g+1,status:P.status,duration:P.duration,steps:$.steps,error:$.error,videoPath:$.videoPath,tracePath:$.tracePath})}let p=b[b.length-1],{test:S}=u[u.length-1],_={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),y),status:p.status,duration:p.duration,steps:p.steps,error:p.error,videoPath:p.videoPath,tracePath:p.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:h,endTime:f,suiteName:d?.suiteName},w=b.some(g=>g.status==="failed"||g.status==="timedOut");b.length>1&&w&&p.status==="passed"&&(_.flaky=!0,_.retries=b.length-1,_.attempts=b),i.push(_)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);T.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let y=n.tests[u],b=y.attempts&&y.attempts.length>0,d=[{obj:y,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(y.attempts)for(let h=0;h<y.attempts.length;h++)d.push({obj:y.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:f,screenshotSubDir:p}of d){let S=v.join(o,p),_=!1;for(let w of h.steps)if(w.screenshot&&v.isAbsolute(w.screenshot))try{_||(T.mkdirSync(S,{recursive:!0}),_=!0);let g=`${w.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(w.screenshot,v.join(S,g)),w.screenshot=`screenshots/${p}/${g}`}catch(g){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,g)}if(h.videoPath&&v.isAbsolute(h.videoPath)){let w=v.extname(h.videoPath)||".webm",g=`${f}-video${w}`;try{T.copyFileSync(h.videoPath,v.join(a,g)),h.videoPath=g}catch{h.videoPath=void 0}}if(h.tracePath&&v.isAbsolute(h.tracePath)){let w=v.extname(h.tracePath)||".zip",g=`${f}-trace${w}`;try{T.copyFileSync(h.tracePath,v.join(a,g)),h.tracePath=g}catch{h.tracePath=void 0}}}}let c=v.join(a,"report-data.json");T.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(T.writeFileSync(l,Me(n),"utf-8"),console.log(`
876
+ </html>`}var Pe="0.1.67",Oe=Pe!=="dev"?Pe:void 0;var sn=3600*1e3,cn=10080*60*1e3;var wt={before:0,main:1,teardown:2,after:3};function Ie(e){let t=e.split(".")[0];return wt[t]??1}function Le(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function _t(e){return[...e].sort(([t],[r])=>{let i=Ie(t),n=Ie(r);if(i!==n)return i-n;let a=Le(t),o=Le(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function vt(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function St(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function te(e,t="main",r,i,n){r===void 0&&(r=!vt(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=St(o.title,t);a.push(...te(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,y={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(y.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(y),o.steps.length>0&&a.push(...te(o.steps,u,r,i,n))}}return a}var re=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let u of this.collected){let y=u.test.titlePath().join(" > "),b=r.get(y);b||(b=[],r.set(y,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let y=u[0].test.location.file,b=[],d,h,f;for(let g=0;g<u.length;g++){let{test:A,result:P}=u[g],$=await this.buildReportTest(A,P,y);d=$,h||(h=$.startTime),f=$.endTime,b.push({attemptNumber:g+1,status:P.status,duration:P.duration,steps:$.steps,error:$.error,videoPath:$.videoPath,tracePath:$.tracePath})}let p=b[b.length-1],{test:S}=u[u.length-1],_={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),y),status:p.status,duration:p.duration,steps:p.steps,error:p.error,videoPath:p.videoPath,tracePath:p.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:h,endTime:f,suiteName:d?.suiteName},w=b.some(g=>g.status==="failed"||g.status==="timedOut");b.length>1&&w&&p.status==="passed"&&(_.flaky=!0,_.retries=b.length-1,_.attempts=b),i.push(_)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Oe},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);T.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let y=n.tests[u],b=y.attempts&&y.attempts.length>0,d=[{obj:y,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(y.attempts)for(let h=0;h<y.attempts.length;h++)d.push({obj:y.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:f,screenshotSubDir:p}of d){let S=v.join(o,p),_=!1;for(let w of h.steps)if(w.screenshot&&v.isAbsolute(w.screenshot))try{_||(T.mkdirSync(S,{recursive:!0}),_=!0);let g=`${w.stepId.replace(/\./g,"-")}.png`;T.copyFileSync(w.screenshot,v.join(S,g)),w.screenshot=`screenshots/${p}/${g}`}catch(g){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,g)}if(h.videoPath&&v.isAbsolute(h.videoPath)){let w=v.extname(h.videoPath)||".webm",g=`${f}-video${w}`;try{T.copyFileSync(h.videoPath,v.join(a,g)),h.videoPath=g}catch{h.videoPath=void 0}}if(h.tracePath&&v.isAbsolute(h.tracePath)){let w=v.extname(h.tracePath)||".zip",g=`${f}-trace${w}`;try{T.copyFileSync(h.tracePath,v.join(a,g)),h.tracePath=g}catch{h.tracePath=void 0}}}}let c=v.join(a,"report-data.json");T.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(T.writeFileSync(l,Me(n),"utf-8"),console.log(`
877
877
  Shiplight report written to: ${l}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(l)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let n={title:t.title,file:v.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(n.error=r.errors.map(d=>d.message||d.stack||String(d)).join(`
878
878
 
879
879
  `)),r.stdout.length>0&&(n.stdout=r.stdout.map(d=>typeof d=="string"?d:d.toString()).join("")),r.stderr.length>0&&(n.stderr=r.stderr.map(d=>typeof d=="string"?d:d.toString()).join(""));for(let d of r.attachments)d.name==="video"&&d.path&&(n.videoPath=d.path),d.name==="trace"&&d.path&&(n.tracePath=d.path);let a=r.attachments.find(d=>d.name==="shiplight-results"),o=null;if(a)try{if(a.body)o=JSON.parse(a.body.toString("utf-8"));else if(a.path){let d=T.readFileSync(a.path,"utf-8");o=JSON.parse(d)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),l={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),y=u?u[1]:t.title,b=u?u[2]:void 0;if(b&&(n.parameterSetName=b),T.existsSync(c))try{let d=T.readFileSync(c,"utf-8"),h=Te(d,c);if(h.suite){let f=h.suite.tests.find(p=>p.name===y);f&&(l=z(f.testFlow),f.tags?.length&&(n.tags=f.tags),f.skip!==void 0&&(n.skip=f.skip),f.slow&&(n.slow=f.slow),f.timeout!==void 0&&(n.timeout=f.timeout),n.baseTitle=f.name||f.testFlow?.goal),n.suiteName=h.name,h.tags?.length&&(n.suiteTags=h.tags),h.use?.baseURL&&(n.baseUrl=h.use.baseURL)}else h.testFlow&&(l=z(h.testFlow),n.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(n.tags=h.tags),h.use?.baseURL&&(n.baseUrl=h.use.baseURL))}catch{}if(o||Object.keys(l).length>0){let d=new Set([...Object.keys(l),...Object.keys(o||{})]),h=Array.from(d).map(p=>[p,null]),f=_t(h);for(let[p]of f){let S=l[p],_=o?.[p],w=S?.description;if(!w||w==="Action"||w==="Draft"){let A=S?.action_entity;w=A?.action_description||A?.action_data?.kwargs?.description||_?.description||p}let g={stepId:p,description:w,status:_?.status||"pending",duration:_?.duration};if(_?.message){let A=typeof _.message=="string"?_.message:JSON.stringify(_.message,null,2);_.status==="failure"?g.error=A:g.message=A}if(_?.screenshot){let A=_.screenshot,P=a?.path?v.dirname(a.path):"",$=v.isAbsolute(A)?A:v.join(P,A);T.existsSync($)&&(g.screenshot=$)}_?.code&&(g.code=_.code),n.steps.push(g)}}if(o===null&&Object.keys(l).length===0&&!i.endsWith(".yaml.spec.ts")){let d=new Map;if(n.steps=te(r.steps,"main",void 0,d),n.steps.length>0){let h=new Map;n.actionStepsMap=Object.fromEntries(n.steps.map(f=>{let p=d.get(f.stepId),S;if(p?.file){if(!h.has(p.file))try{h.set(p.file,T.readFileSync(p.file,"utf-8").split(`
package/dist/cli.js CHANGED
@@ -17,7 +17,7 @@ Install it as a project dependency instead:
17
17
  `)}catch{}}function mn(e=ct()){try{let t=O.readFileSync(e,"utf-8"),r=JSON.parse(t);if(typeof r.latest=="string"&&typeof r.fetchedAt=="number"&&Date.now()-r.fetchedAt<dn)return r}catch{}return null}function yn(e,t=ct()){try{O.mkdirSync(W.dirname(t),{recursive:!0}),O.writeFileSync(t,JSON.stringify(e))}catch{}}async function wn(){try{let e=await fetch(un,{headers:{Accept:"application/json"},signal:AbortSignal.timeout(3e3)});if(!e.ok)return null;let t=await e.json();return typeof t.version=="string"?t.version:null}catch{return null}}function bn(e,t){let r=d=>{let f=d.indexOf("-");return f===-1?[d,!1]:[d.slice(0,f),!0]},s=d=>d.split(".").map(f=>parseInt(f,10)||0),[n,o]=r(e),[i,a]=r(t),l=s(n),c=s(i),u=Math.max(l.length,c.length);for(let d=0;d<u;d++){let f=l[d]??0,g=c[d]??0;if(f<g)return!0;if(f>g)return!1}return!!(o&&!a)}async function Yt(e={}){let t=e.runningVersion??it,r=e.cwd??process.cwd(),s=e.cacheFile??ct(),n=e.fetchLatest??wn,o=e.env??process.env,i=e.warn??(c=>console.warn(c));if(o.CI||t==="dev"||!O.existsSync(W.join(r,"package-lock.json")))return;let a=null,l=mn(s);if(l)a=l.latest;else{try{a=await n()}catch{a=null}a&&yn({latest:a,fetchedAt:Date.now()},s)}a&&bn(t,a)&&i(`
18
18
  \x1B[33m\u26A0 shiplightai ${a} is available (you have ${t}).
19
19
  Run: npm update shiplightai\x1B[0m
20
- `)}var it,_e,un,dn,hn,Le=_(()=>{"use strict";it="0.1.66",_e=it!=="dev"?it:void 0,un="https://registry.npmjs.org/shiplightai/latest",dn=3600*1e3,hn=10080*60*1e3});import*as re from"fs";import*as ae from"path";function lt(e){let{projectPath:t}=e,r=e.projectName??ae.basename(ae.resolve(t));if(re.existsSync(t)&&re.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);re.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let i=ae.join(t,n);re.mkdirSync(ae.dirname(i),{recursive:!0}),re.writeFileSync(i,o)};return s("package.json",vn.replace("{{name}}",r)),s("playwright.config.ts",Sn),s(".gitignore",_n),s(".env.example",xn),s(".mcp.json",Tn),s("auth/example.login.ts",Pn),s("environments/example.env.yaml",kn),s("tests/example.test.yaml",En),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var vn,Sn,_n,xn,Tn,Pn,kn,En,Jt=_(()=>{"use strict";vn=`{
20
+ `)}var it,_e,un,dn,hn,Le=_(()=>{"use strict";it="0.1.67",_e=it!=="dev"?it:void 0,un="https://registry.npmjs.org/shiplightai/latest",dn=3600*1e3,hn=10080*60*1e3});import*as re from"fs";import*as ae from"path";function lt(e){let{projectPath:t}=e,r=e.projectName??ae.basename(ae.resolve(t));if(re.existsSync(t)&&re.readdirSync(t).length>0)throw new Error(`Cannot scaffold into non-empty directory: ${t}`);re.mkdirSync(t,{recursive:!0});let s=(n,o)=>{let i=ae.join(t,n);re.mkdirSync(ae.dirname(i),{recursive:!0}),re.writeFileSync(i,o)};return s("package.json",vn.replace("{{name}}",r)),s("playwright.config.ts",Sn),s(".gitignore",_n),s(".env.example",xn),s(".mcp.json",Tn),s("auth/example.login.ts",Pn),s("environments/example.env.yaml",kn),s("tests/example.test.yaml",En),{projectPath:t,projectName:r,filesCreated:["package.json","playwright.config.ts",".gitignore",".env.example",".mcp.json","auth/example.login.ts","environments/example.env.yaml","tests/example.test.yaml"]}}var vn,Sn,_n,xn,Tn,Pn,kn,En,Jt=_(()=>{"use strict";vn=`{
21
21
  "name": "{{name}}",
22
22
  "type": "module",
23
23
  "scripts": {
@@ -1152,9 +1152,9 @@ Merged ${l.length} tests from ${u} shards into: ${f}`),await Gs(d,s),r&&Ni(l),t)
1152
1152
  `;for(let l of a)i+=`- ${l}
1153
1153
  `;i+=`
1154
1154
  </details>
1155
- `}return i}function Ni(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}$.appendFileSync(t,Ks(e)),console.log("GitHub step summary written.")}var Vs=_(()=>{"use strict";Ds();Bs();Le()});var Ys,Js=_(()=>{"use strict";Ys="0.1.66"});var Xs={};te(Xs,{runTranspile:()=>ji});import*as st from"path";import{glob as Di}from"glob";async function ji(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),s=await Di(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,i=0;for(let a of s.sort()){let l=st.resolve(r,a),c=is(l,{version:Ys});if(!c.valid){n++,console.log(`
1155
+ `}return i}function Ni(e){let t=process.env.GITHUB_STEP_SUMMARY;if(!t){console.warn("Warning: $GITHUB_STEP_SUMMARY not set, skipping GitHub summary.");return}$.appendFileSync(t,Ks(e)),console.log("GitHub step summary written.")}var Vs=_(()=>{"use strict";Ds();Bs();Le()});var Ys,Js=_(()=>{"use strict";Ys="0.1.67"});var Xs={};te(Xs,{runTranspile:()=>ji});import*as st from"path";import{glob as Di}from"glob";async function ji(e){(e.includes("--help")||e.includes("-h"))&&(console.log("Usage: shiplight transpile [glob]"),console.log(""),console.log("Transpiles YAML test files to Playwright spec files (.yaml.spec.ts)."),console.log("Validates syntax and reports action coverage warnings."),console.log("Default glob: **/*.test.yaml"),console.log(""),console.log("Examples:"),console.log(" shiplight transpile # transpile all YAML tests"),console.log(' shiplight transpile "tests/**/*.test.yaml" # transpile specific directory'),console.log(" shiplight transpile tests/login.test.yaml # transpile a single file"),process.exit(0));let t=e[0]||"**/*.test.yaml",r=process.cwd(),s=await Di(t,{cwd:r,ignore:["node_modules/**","*.yaml.spec.ts"]});s.length===0&&(console.log(`No files matched: ${t}`),process.exit(0));let n=0,o=0,i=0;for(let a of s.sort()){let l=st.resolve(r,a),c=is(l,{version:Ys});if(!c.valid){n++,console.log(`
1156
1156
  \u2717 ${a}`);for(let d of c.errors)console.log(` ERROR: ${d}`);continue}i++;let u=st.basename(c.specFile);if(c.warnings.length>0){o++,console.log(`\u26A0 ${a} \u2192 ${u}`);for(let d of c.warnings)console.log(` WARNING: ${d}`)}else console.log(`\u2713 ${a} \u2192 ${u}`)}console.log(`
1157
- ${s.length} file(s): ${i} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var qs=_(()=>{"use strict";kt();Js()});var en={};te(en,{runInspect:()=>Fi});import*as nt from"fs";import*as Zs from"path";async function Fi(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),s=e.find(i=>!i.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=Zs.resolve(process.cwd(),s);nt.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=nt.readFileSync(n,"utf-8");try{let i=ke(o),a=C(o);if(r)Ui(a,i);else{let l={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(l,null,t?0:2))}}catch(i){console.error(`Error parsing ${s}: ${i.message}`),process.exit(1)}}function Ui(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Qs(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Qs(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let s=Qs(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var tn=_(()=>{"use strict";ce()});var rn=ln((Bp,Bi)=>{Bi.exports={name:"shiplightai",version:"0.1.66",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Le();import Hi from"dotenv";Hi.config();Vt();Yt();function sn(){console.log(`
1157
+ ${s.length} file(s): ${i} transpiled, ${n} error(s), ${o} warning(s)`),process.exit(n>0?1:0)}var qs=_(()=>{"use strict";kt();Js()});var en={};te(en,{runInspect:()=>Fi});import*as nt from"fs";import*as Zs from"path";async function Fi(e){(e.includes("--help")||e.includes("-h")||e.length===0)&&(console.log("Usage: shiplight inspect <file.test.yaml> [options]"),console.log(""),console.log("Parse a YAML test file and output the resulting TestFlow JSON."),console.log("Useful for verifying YAML \u2192 JSON conversion."),console.log(""),console.log("Options:"),console.log(" --pretty Pretty-print JSON (default)"),console.log(" --compact Compact JSON output"),console.log(" --stats Show statement statistics only"),console.log(""),console.log("Examples:"),console.log(" shiplight inspect tests/login.test.yaml"),console.log(" shiplight inspect tests/suite.test.yaml --stats"),console.log(" shiplight inspect tests/login.test.yaml --compact | jq ."),process.exit(e.length===0?1:0));let t=e.includes("--compact"),r=e.includes("--stats"),s=e.find(i=>!i.startsWith("--"));s||(console.error("Error: no file specified"),process.exit(1));let n=Zs.resolve(process.cwd(),s);nt.existsSync(n)||(console.error(`Error: file not found: ${n}`),process.exit(1));let o=nt.readFileSync(n,"utf-8");try{let i=ke(o),a=C(o);if(r)Ui(a,i);else{let l={...i.test_case_id!==void 0?{test_case_id:i.test_case_id}:{},...i.name?{name:i.name}:{},testFlow:a};console.log(JSON.stringify(l,null,t?0:2))}}catch(i){console.error(`Error parsing ${s}: ${i.message}`),process.exit(1)}}function Ui(e,t){if(console.log(`File: ${t.name||"(unnamed)"}`),t.test_case_id!==void 0&&console.log(`Cloud ID: ${t.test_case_id}`),console.log(`Version: ${e.version||"unknown"}`),e.testGroup){let r=e.testGroup;console.log("Type: suite (testGroup)"),console.log(`Tests: ${r.tests.length}`);for(let s of r.tests){let n=s.skip?` [SKIP${typeof s.skip=="string"?`: ${s.skip}`:""}]`:"";console.log(` - ${s.name}: ${s.statements.length} statements${s.teardown?`, ${s.teardown.length} teardown`:""}${n}`)}r.beforeAll?.length&&console.log(`Hooks: beforeAll (${r.beforeAll.length})`),r.afterAll?.length&&console.log(`Hooks: afterAll (${r.afterAll.length})`),r.beforeEach?.length&&console.log(`Hooks: beforeEach (${r.beforeEach.length})`),r.afterEach?.length&&console.log(`Hooks: afterEach (${r.afterEach.length})`)}else{console.log("Type: single test"),console.log(`Goal: ${e.goal}`),e.url&&console.log(`URL: ${e.url}`),e.baseURL&&console.log(`Base URL: ${e.baseURL}`),console.log(`Statements: ${e.statements?.length??0}`),e.teardown?.length&&console.log(`Teardown: ${e.teardown.length}`);let r=Qs(e.statements??[]);console.log(` DRAFT: ${r.drafts}, ACTION: ${r.actions}, STEP: ${r.steps}`)}}function Qs(e){let t={drafts:0,actions:0,steps:0};for(let r of e)if(r.type==="DRAFT")t.drafts++;else if(r.type==="ACTION")t.actions++;else if(r.type==="STEP"){t.steps++;let s=Qs(r.statements??[]);t.drafts+=s.drafts,t.actions+=s.actions,t.steps+=s.steps}return t}var tn=_(()=>{"use strict";ce()});var rn=ln((Bp,Bi)=>{Bi.exports={name:"shiplightai",version:"0.1.67",type:"module",description:"Shiplight CLI for running and debugging .test.yaml files",main:"dist/index.js",types:"dist/index.d.ts",bin:{shiplight:"dist/cli.js"},exports:{".":{types:"./dist/index.d.ts",import:"./dist/index.js",require:"./dist/cjs/index.cjs",default:"./dist/index.js"},"./fixture":{types:"./dist/fixture.d.ts",import:"./dist/fixture.js",require:"./dist/cjs/fixture.cjs",default:"./dist/fixture.js"},"./debugger-pw":{types:"./dist/debugger-pw.d.ts",import:"./dist/debugger-pw.js",require:"./dist/cjs/debugger-pw.cjs",default:"./dist/debugger-pw.js"},"./debugger-manager":{types:"./dist/debugger-manager.d.ts",import:"./dist/debugger-manager.js",require:"./dist/cjs/debugger-manager.cjs",default:"./dist/debugger-manager.js"},"./debugger-server":{types:"./dist/debugger-server.d.ts",import:"./dist/debugger-server.js",require:"./dist/cjs/debugger-server.cjs",default:"./dist/debugger-server.js"},"./reporter":{types:"./dist/reporter.d.ts",import:"./dist/reporter.js",require:"./dist/cjs/reporter.cjs",default:"./dist/reporter.js"},"./package.json":"./package.json"},files:["dist","!dist/**/*.map","README.md"],publishConfig:{registry:"https://registry.npmjs.org",access:"public"},scripts:{prebuild:"pnpm typecheck",build:"tsup",pack:"pnpm build && pnpm pack",clean:"rm -rf dist",dev:"tsup --watch","dev:run":"node --import tsx/esm src/cli.ts",test:"pnpm test:unit && pnpm test:logic","test:unit":"tsx --test 'src/**/*.test.ts'","test:logic":"playwright test -c playwright.logic.config.ts","test:e2e":"playwright test -c playwright.config.ts",typecheck:"tsc --noEmit"},dependencies:{"@ai-sdk/anthropic":"^3.0.1","@ai-sdk/google":"^3.0.1","@ai-sdk/google-vertex":"^4.0.1","@ai-sdk/openai":"^3.0.1","@ai-sdk/provider":"^3.0.1","@anthropic-ai/claude-agent-sdk":"^0.1.72","@babel/parser":"^7.28.5","@babel/plugin-transform-typescript":"^7.27.0","@google/genai":"^1.34.0","google-auth-library":"^10.0.0","@babel/preset-env":"^7.26.9","@babel/preset-typescript":"^7.27.0","@modelcontextprotocol/sdk":"^1.29.0","@shiplightai/devtools-assets":"workspace:*",ai:"^6.0.3",axios:"^1.15.0",chalk:"^4.1.2",commander:"^11.0.0",dotenv:"^16.0.3",express:"^5.2.1","fs-extra":"^11.2.0",glob:"^13.0.0","html-to-text":"^9.0.5",open:"^10.1.0",openai:"^6.25.0",ora:"^5.4.1",otplib:"^13.4.0","p-retry":"^6.2.1",sharp:"^0.34.5",uuid:"^11.1.0",yaml:"^2.8.3",zod:"^3.22.0","zod-to-json-schema":"^3.24.6"},devDependencies:{"@playwright/test":"1.60.0","@types/express":"^4.17.21","@types/node":"^24.0.0","mcp-tools":"workspace:*","sdk-core":"workspace:*","sdk-internal":"workspace:*","shiplight-tools":"workspace:*","shiplight-types":"workspace:*","@loggia/common":"workspace:*",tsup:"^8.3.5",typescript:"5.5.4"},peerDependencies:{"@playwright/test":"^1.60.0"},engines:{node:">=22.0.0"},keywords:["playwright","yaml","testing","automation","ai","shiplight","mcp"],author:"Shiplight",license:"MIT"}});Le();import Hi from"dotenv";Hi.config();Vt();Yt();function sn(){console.log(`
1158
1158
  Usage: shiplight <command> [options]
1159
1159
 
1160
1160
  Commands:
package/dist/index.js CHANGED
@@ -4365,7 +4365,7 @@ ${p.join(`
4365
4365
  `)}function xc(e,t){let i=[],n=t?.version||"unknown";i.push(`// @generated by shiplightai v${n}`),i.push(...bo()),i.push(""),t?.use&&Object.keys(t.use).length>0&&(i.push(`test.use(${JSON.stringify(t.use,null,2)});`),i.push(""));let a=new Set,o={imports:a,actionEntityStore:t?.actionEntityStore},r=t?.testName||"Test Suite",s=Ln(t?.tags);i.push(`test.describe.serial('${s}${_e(r)}', () => {`),e.beforeAll&&e.beforeAll.length>0&&(i.push(...bi("beforeAll",e.beforeAll,o,1)),i.push("")),e.beforeEach&&e.beforeEach.length>0&&(i.push(...bi("beforeEach",e.beforeEach,o,1)),i.push(""));for(let c=0;c<e.tests.length;c++){let d=e.tests[c],h=d.timeout||d.skip!==void 0||d.fail!==void 0||d.only||d.slow?{timeout:d.timeout,skip:d.skip,fail:d.fail,only:d.only,slow:d.slow}:void 0;if(d.parameters&&d.parameters.length>0)for(let p of d.parameters){let g=wo(d.testFlow,p.values);i.push(...vi(g,`${_e(d.name)} [${_e(p.name)}]`,o,1,h)),i.push("")}else i.push(...vi(d.testFlow,_e(d.name),o,1,h)),(c<e.tests.length-1||e.afterEach||e.afterAll)&&i.push("")}return e.afterEach&&e.afterEach.length>0&&(i.push(...bi("afterEach",e.afterEach,o,1)),i.push("")),e.afterAll&&e.afterAll.length>0&&i.push(...bi("afterAll",e.afterAll,o,1)),i.push("});"),yo(i,a),i.join(`
4366
4366
  `)}function Ln(e){return e&&e.length>0?e.map(t=>`@${t}`).join(" ")+" ":""}var vc=["testContext","request"];function xi(e){let t=new Set;function i(n){for(let a of n)switch(a.type){case ue.ACTION:{let r=a.action_entity?.action_data?.kwargs;if(r?.args&&Array.isArray(r.args))for(let s of r.args)typeof s=="string"&&vc.includes(s)&&t.add(s);break}case ue.STEP:i(a.statements);break;case ue.IF_ELSE:{let o=a;i(o.then),o.else&&i(o.else);break}case ue.WHILE_LOOP:i(a.body);break}}return i(e),t}function _c(e){let t=xi(e.statements??[]);if(e.teardown)for(let i of xi(e.teardown))t.add(i);return t}function Rn(e){return`{ ${["page","agent",...Array.from(e).sort()].join(", ")} }`}function vi(e,t,i,n=0,a){let o=" ".repeat(n),r=[],s=_c(e),l=Rn(s),c=a?.only?"test.only":"test";r.push(`${o}${c}('${t}', async (${l}) => {`),a?.skip===!0?r.push(`${o} test.skip();`):typeof a?.skip=="string"&&r.push(`${o} test.skip(true, '${_e(a.skip)}');`),a?.fail===!0?r.push(`${o} test.fail();`):typeof a?.fail=="string"&&r.push(`${o} test.fail(true, '${_e(a.fail)}');`),a?.slow&&r.push(`${o} test.slow();`),a?.timeout&&r.push(`${o} test.setTimeout(${a.timeout});`);let d=e.teardown&&e.teardown.length>0,h=n+1;if(d){if(r.push(`${o} try {`),e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let g=pe(e.statements,h+1,i);r.push(...g)}r.push(`${o} } finally {`),r.push(`${o} // Teardown`);let p=pe(e.teardown,h+1,i,"teardown");r.push(...p),r.push(`${o} }`)}else if(e.statements&&e.statements.length>0){r.push(`${o} // Test steps`);let p=pe(e.statements,h,i);r.push(...p)}return r.push(`${o}});`),r}function mo(e,t,i){let n=[],a=fo(t),o=xi(a),r=Rn(o);return n.push(`test.${e}(async (${r}) => {`),n.push(...pe(a,1,i,e)),n.push("});"),n}function bi(e,t,i,n){let a=" ".repeat(n),o=[],r=fo(t);if(e==="beforeAll"||e==="afterAll"){let l={...i,noAgent:!0};o.push(`${a}test.${e}(async ({ browser }, workerInfo) => {`),o.push(`${a} const page = await browser.newPage({ baseURL: workerInfo.project.use.baseURL });`),o.push(...pe(r,n+1,l,e)),o.push(`${a} await page.close();`),o.push(`${a}});`)}else{let l=xi(r),c=Rn(l);o.push(`${a}test.${e}(async (${c}) => {`),o.push(...pe(r,n+1,i,e)),o.push(`${a}});`)}return o}function fo(e){let i=oc({goal:"_hook",statements:e});return me(i).statements??[]}function wo(e,t){let i=Mn(e);for(let[n,a]of Object.entries(t))i=i.split(`<<${n}>>`).join(String(a));return me(i)}function bo(){return["import { test, expect } from 'shiplightai/fixture';"]}function yo(e,t){if(t.size>0){let i=0;for(let a=0;a<e.length;a++)e[a].startsWith("import ")&&(i=a+1);let n=Array.from(t);e.splice(i,0,...n)}}var go=5;function vo(e,t,i){let n={expandingPaths:new Set([Nn(t)]),depth:0,referencedPaths:new Set,basePath:i},a={...e};Array.isArray(a.statements)&&(a.statements=Ne(a.statements,t,n)),Array.isArray(a.teardown)&&(a.teardown=Ne(a.teardown,t,n));for(let o of["beforeAll","afterAll","beforeEach","afterEach"])Array.isArray(a[o])&&(a[o]=Ne(a[o],t,n));return{doc:a,referencedTemplatePaths:Array.from(n.referencedPaths)}}function Ne(e,t,i){let n=[];for(let a of e)if(Mc(a)){let o=Ic(a,t,i);n.push(o)}else n.push(Pc(a,t,i));return n}function Mc(e){return typeof e=="object"&&e!==null&&typeof e.template=="string"}function Ic(e,t,i){if(i.depth>=go)throw new Error(`Template expansion exceeded maximum depth of ${go}. Check for deeply nested or circular template references.`);let n=Nn(Ac(t),e.template),a=!Tc(n)&&i.basePath?Nn(i.basePath,e.template):n;if(i.expandingPaths.has(a))throw new Error(`Circular template reference detected: ${a} is already being expanded. Stack: ${Array.from(i.expandingPaths).join(" \u2192 ")} \u2192 ${a}`);i.referencedPaths.add(a);let o;try{o=Ec(a,"utf-8")}catch(f){throw new Error(`Failed to read template file: ${a} (referenced from ${t}): ${f.message}`)}let r=po(o);if(!r||typeof r!="object")throw new Error(`Invalid template file: ${a} \u2014 expected a YAML object`);let s=r.params||[],l=e.params||{};for(let f of s)if(!(f in l))throw new Error(`Template ${e.template} requires param "${f}" but it was not provided. Required params: [${s.join(", ")}]`);let c=r.statements;if(!Array.isArray(c))throw new Error(`Template ${e.template} must have a "statements" array`);if(Object.keys(l).length>0){let y=$c(c);for(let[b,m]of Object.entries(l))y=y.split(`<<${b}>>`).join(String(m));c=po(y)}let d={expandingPaths:new Set([...i.expandingPaths,a]),depth:i.depth+1,referencedPaths:i.referencedPaths},h=Ne(c,a,d),g={STEP:r.name||e.template.replace(/\.yaml$/,"").split("/").pop()||e.template,template_path:e.template,statements:h};return Object.keys(l).length>0&&(g.template_params=l),g}function Pc(e,t,i){if(typeof e!="object"||e===null)return e;let n={...e};return Array.isArray(n.statements)&&(n.statements=Ne(n.statements,t,i)),Array.isArray(n.THEN)&&(n.THEN=Ne(n.THEN,t,i)),Array.isArray(n.ELSE)&&(n.ELSE=Ne(n.ELSE,t,i)),Array.isArray(n.DO)&&(n.DO=Ne(n.DO,t,i)),n}var Fn=class extends Error{constructor(e){super(e),this.name="YamlValidationError"}};function Hn(e,t,i){let n=Sc(e),a=n?.name,o=n?.tags,r=n?.use;if(n&&(n.name!==void 0||n.tags!==void 0||n.use!==void 0)&&(delete n.name,delete n.tags,delete n.use),n?.suite){if(n.goal||n.statements)throw new Fn('YAML file cannot have both "suite" and top-level "goal"/"statements". Use either suite format or single-test format.');return Oc(n,a,o,r,t,i)}return Cc(n,a,o,r,t,i)}function Cc(e,t,i,n,a,o){let r=e?.beforeEach,s=e?.afterEach,l=_o(e?.parameters),c=e?.timeout,d=e?.skip,h=e?.fail,p=e?.only,g=e?.slow,f=e?.settings,y=n;if(f){let _={};f.auto_dismiss_modal!==void 0&&(_.autoDismissModal=!!f.auto_dismiss_modal),f.browser_timezone!==void 0&&f.browser_timezone!==null&&(_.timezoneId=String(f.browser_timezone)),f.browser_language!==void 0&&f.browser_language!==null&&(_.locale=String(f.browser_language)),f.extra_http_headers!==void 0&&f.extra_http_headers!==null&&typeof f.extra_http_headers=="object"&&(_.extraHTTPHeaders=f.extra_http_headers),Object.keys(_).length>0&&(y={...y,..._})}if(e&&(delete e.beforeEach,delete e.afterEach,delete e.parameters,delete e.timeout,delete e.skip,delete e.fail,delete e.only,delete e.slow,delete e.settings),e?.url)throw new Fn(`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 b=[];if(a&&e&&typeof e=="object"){let _=vo(e,a,o);e=_.doc,b=_.referencedTemplatePaths}let m=xo(e),x=me(m);return a&&(Xe(x.statements??[],a,"main"),x.teardown&&Xe(x.teardown,a,"teardown")),{testFlow:x,name:t,tags:i,use:y,beforeEach:r,afterEach:s,parameters:l,timeout:c,skip:d,fail:h,only:p,slow:g,referencedTemplatePaths:b}}function Oc(e,t,i,n,a,o){let r=e.suite;if(!Array.isArray(r.tests)||r.tests.length===0)throw new Error('Suite must have a non-empty "tests" array.');let s=r.beforeAll,l=r.afterAll,c=r.beforeEach,d=r.afterEach,h=[],p=r.tests.map(y=>{if(!y.name)throw new Error('Each test in a suite must have a "name" field.');if(!Array.isArray(y.statements)||y.statements.length===0)throw new Error(`Suite test "${y.name}" must have a non-empty "statements" array.`);let b={goal:y.name,statements:y.statements};y.teardown&&(b.teardown=y.teardown);let m=[],x=b;if(a&&typeof b=="object"){let $=vo(b,a,o);x=$.doc,m=$.referencedTemplatePaths,h.push(...m)}let _=xo(x),E=me(_),A=_o(y.parameters);return{testFlow:E,name:y.name,tags:Array.isArray(y.tags)?y.tags:void 0,parameters:A,timeout:y.timeout,skip:y.skip,fail:y.fail,only:y.only,slow:y.slow}}),g=r.base_url,f=g?{...n,baseURL:g}:n;return{suite:{beforeAll:s,afterAll:l,beforeEach:c,afterEach:d,tests:p},name:t,tags:i,use:f,referencedTemplatePaths:h}}function _o(e){if(!(!Array.isArray(e)||e.length===0))return e.map((t,i)=>{if(!t.name)throw new Error(`Parameter set at index ${i} must have a "name" field.`);if(!t.values||typeof t.values!="object")throw new Error(`Parameter set "${t.name}" must have a "values" object.`);return{name:t.name,values:t.values}})}function Xe(e,t,i){for(let n=0;n<e.length;n++){let a=e[n],o=`${i}.${n}`,r=a.description||"";if(a.uid=Lc(t,o,r),a.type===ue.STEP)Xe(a.statements,t,o);else if(a.type===ue.IF_ELSE){let s=a;Xe(s.then,t,`${o}.then`),s.else&&Xe(s.else,t,`${o}.else`)}else a.type===ue.WHILE_LOOP&&Xe(a.body,t,`${o}.body`)}}function Lc(e,t,i){let n=kc("sha256").update(`${e}:${t}:${i}`).digest("hex");return`${n.slice(0,8)}-${n.slice(8,12)}-${n.slice(12,16)}-${n.slice(16,20)}-${n.slice(20,32)}`}function ko(e,t,i){let n=/\btemplate:\s/.test(e),a=/^suite:/m.test(e),o=n||a?null:Pn(e);if(o&&!o.valid)return{valid:!1,errors:o.errors,warnings:[],stats:o.stats};let r,s,l=[];try{let c=i?.parsed??Hn(e,t);l=c.referencedTemplatePaths;let d={version:i?.version,actionEntityStore:i?.actionEntityStore},h=c.testFlow?.baseURL?{...c.use,baseURL:c.testFlow.baseURL}:c.use;c.suite?r=xc(c.suite,{...d,testName:c.name,tags:c.tags,use:c.use}):r=yc(c.testFlow,{...d,testName:c.name,tags:c.tags,use:h,beforeEach:c.beforeEach,afterEach:c.afterEach,parameters:c.parameters,timeout:c.timeout,skip:c.skip,fail:c.fail,only:c.only,slow:c.slow});let p=r.split(`
4367
4367
  `).filter(g=>!g.startsWith("import ")).join(`
4368
- `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Dc(Rc(s),{recursive:!0}),Nc(s,r)}catch(c){let d=c instanceof Fn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var Wn="0.1.66";function So(e){try{return Fc(e).mtimeMs}catch{return 0}}var Uc=`// @generated by shiplightai v${Wn}`;function Bc(e,t){if(!To(e)||Eo(e,"utf-8").split(`
4368
+ `);new Function(p),s=t.replace(/\.test\.yaml$/,".yaml.spec.ts"),Dc(Rc(s),{recursive:!0}),Nc(s,r)}catch(c){let d=c instanceof Fn?"":c.message.includes("Unexpected token")?" This usually means a YAML escaping issue \u2014 in double-quoted strings, use \\\\/ instead of \\/ for regex patterns, or use single quotes / block scalars.":" This may indicate a transpiler bug \u2014 please report it.";return{valid:!1,errors:[`Transpilation failed: ${c.message}.${d}`],warnings:[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},referencedTemplatePaths:l}}return{valid:!0,errors:[],warnings:o?.warnings??[],stats:o?.stats??{total:0,action:0,draft:0,coverage:0},specFile:s,referencedTemplatePaths:l}}var Wn="0.1.67";function So(e){try{return Fc(e).mtimeMs}catch{return 0}}var Uc=`// @generated by shiplightai v${Wn}`;function Bc(e,t){if(!To(e)||Eo(e,"utf-8").split(`
4369
4369
  `,1)[0]!==Uc)return!1;let n=So(e);for(let a of t)if(So(a)>n)return!1;return!0}function Gc(e){let t=process.argv.slice(2),i=[],n=Un(e);for(let a of t){if(a.startsWith("-"))continue;let o=a.endsWith(".yaml.spec.ts")?a.replace(/\.yaml\.spec\.ts$/,".test.yaml"):a;if(!o.endsWith(".test.yaml"))continue;let r=Un(e,o);To(r)&&i.push(r.startsWith(n)?r.slice(n.length+1):o)}return i.length>0?i:null}function Ao(e){let t=Gc(e.cwd),i=t??Wc("**/*.test.yaml",{cwd:e.cwd,ignore:["**/node_modules/**"]}),n=[];for(let a of i){let o=Un(e.cwd,a),r=o.replace(/\.test\.yaml$/,".yaml.spec.ts"),s=Eo(o,"utf-8");try{let l=Hn(s,o,e.projectRoot??e.cwd),c=Hc(e.cwd,o),d=e.actionEntityStores?.get(c)??e.actionEntityStores?.get("*");if(!(d&&Object.keys(d.entries).length>0)&&Bc(r,[o,...l.referencedTemplatePaths]))continue;let p=ko(s,o,{version:Wn,actionEntityStore:d,parsed:l});if(!p.valid)throw new Error(p.errors.join("; "))}catch(l){console.error(`[shiplight] Failed to transpile ${a}:`,l),n.push({file:a,error:l})}}if(n.length>0){let a=`[shiplight] Transpilation failed for ${n.length} file(s):
4370
4370
  `+n.map(o=>` - ${o.file}`).join(`
4371
4371
  `);if(t)throw new Error(a);console.warn(a+" (skipped)")}}import*as ae from"fs";import*as ft from"path";import{globSync as ed}from"glob";Si();function Lo(e){let t=ki().SHIPLIGHT_API_TOKEN;return process.env.CI&&t?new zn:new Kn(e)}var td=".shiplight/action-cache";function Oo(e){return e.replace(/\//g,"__")+".json"}function id(e){return e.replace(/\.json$/,"").replace(/__/g,"/")}var Kn=class{constructor(t){this.cwd=t;this.cacheDir=ft.join(t,td)}isCloud=!1;cacheDir;async lookup(t){let i=new Map;if(t.length===0||!ae.existsSync(this.cacheDir))return i;for(let n of t){let a=ft.join(this.cacheDir,Oo(n));try{if(ae.existsSync(a)){let o=ae.readFileSync(a,"utf-8");i.set(n,JSON.parse(o))}}catch{}}return i}async update(t){if(t.size===0)return 0;ae.mkdirSync(this.cacheDir,{recursive:!0});let i=0;for(let[n,a]of t)try{let o=ft.join(this.cacheDir,Oo(n)),r=Cn();if(ae.existsSync(o))try{r=JSON.parse(ae.readFileSync(o,"utf-8"))}catch{}let s={...r,entries:{...r.entries,...a.entries}};ae.writeFileSync(o,JSON.stringify(s,null,2)),i++}catch{}return i}loadAll(){if(!ae.existsSync(this.cacheDir))return;let t=ed("*.json",{cwd:this.cacheDir});if(t.length===0)return;let i=new Map,n=0;for(let a of t)try{let o=ae.readFileSync(ft.join(this.cacheDir,a),"utf-8"),r=JSON.parse(o),s=id(a);i.set(s,r),n+=Object.keys(r.entries??{}).length}catch{}if(i.size!==0)return console.log(`[shiplight] Cache: loaded ${n} cached action entit${n===1?"y":"ies"} for ${i.size} test file${i.size!==1?"s":""}`),i}},zn=class{isCloud=!0;async lookup(t){let{lookupActionStores:i}=await Promise.resolve().then(()=>(jn(),Gn));return i(t)}async update(t){let{updateActionStores:i}=await Promise.resolve().then(()=>(jn(),Gn));return i(t)}loadAll(){}};Si();function od(e={}){e.dotenv!==!1&&sd(e.scanDir||process.cwd());let t=e.scanDir||process.cwd(),n=Lo(t).loadAll();return Ao({cwd:t,projectRoot:process.cwd(),actionEntityStores:n}),{reporter:[["list"],["shiplightai/reporter",{outputFolder:"shiplight-report",open:"never"}]]}}function rd(e,...t){return ad(e,...t)}function sd(e){Mo(e);for(let t of Bn(e))nd.config({path:t})}R();R();Lt();Qe();import{z as tl}from"zod";var Th=tl.object({instruction:tl.string().describe('The instruction of the operation to perform. Can only include one operation. Do not inlcude element indexes just describe the element, e.g. "select the text "Hello, world!" in "Hello, world!""')});function il(e){e.register({name:"perform_accurate_operation",description:"Perform an operation that requires accurate interaction like dragging or interacting with a specific area of an element. Only use this action when neccecary.",schema:Th,usesElementIndex:!1,async execute(t,i){let{instruction:n}=t,a={page:i.page,agentServices:i.agentServices,domService:i.domService},o=await Ze(n,a,{});if(o.status==="error"||!o.actionEntity)return{success:!1,actionEntity:{action_description:n,action_data:{action_name:"perform_accurate_operation",kwargs:{instruction:n}}},error:o.error||"Failed to generate action"};let{actionEntity:r}=o,s=await Ct(r,a);return{success:s.success,actionEntity:r,message:s.success?`Successfully executed action: ${r.action_data?.action_name}`:void 0,error:s.error}}})}Te();R();import{generateText as Ah}from"ai";import{convert as $h}from"html-to-text";async function Mh(e,t,i){let{apiKey:n,domain:a}=e;if(!n||!a)throw new Error("Mailgun configuration missing. Please provide apiKey and domain");let o=[];try{let r=`https://api.mailgun.net/v3/${a}/events`,s={event:"accepted",limit:"10",ascending:"yes",recipient:t};if(i.from_email&&(s.from=i.from_email),i.since)s.begin=Math.floor(i.since/1e3).toString();else{let d=new Date(Date.now()-6e5);s.begin=Math.floor(d.getTime()/1e3).toString()}u.info(`Mailgun params: ${JSON.stringify(s)}`);let l=await fetch(r+"?"+new URLSearchParams(s),{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!l.ok){let d=await l.text();throw new Error(`Mailgun events API error: ${d}`)}let c=(await l.json()).items||[];for(let d of c.slice(0,10)){if(d.event!=="accepted")continue;let h=(d.storage||{}).url;if(u.info(`message_url: ${h}`),!h){let f=(d.message||{}).headers||{},y=f.subject||"",b=f.from||"",m=f.to||"";if(i.from_email&&!b.toLowerCase().includes(i.from_email.toLowerCase())||i.to_email&&!m.toLowerCase().includes(i.to_email.toLowerCase())||i.subject&&!y.toLowerCase().includes(i.subject.toLowerCase()))continue;o.push({subject:y,from:b,to:m,date:new Date(d.timestamp*1e3).toUTCString(),body:"Message body not available (Mailgun storage disabled)",message_id:f["message-id"]||""});continue}let p=h.split("/"),g=p[p.length-1];if(u.info(`Storage key: ${g}`),g){let f=`https://api.mailgun.net/v3/domains/${a}/messages/${g}`;try{let y=await fetch(f,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`,Accept:"application/json"}});if(y.ok){let b=await y.json(),m=b.Subject||"",x=b.From||"",_=b.To||"",E=b.Date||"",A=b["Message-Id"]||"";u.info(`subject: ${m}`),u.info(`from_addr: ${x}`),u.info(`to_addr: ${_}`),u.info(`date: ${E}`),u.info(`message_id: ${A}`);let $=b["body-html"]||b["body-plain"]||"";if($&&$.includes("<")&&($=$h($)),u.info(`Body: ${$.substring(0,200)}...`),i.subject&&!m.toLowerCase().includes(i.subject.toLowerCase())||i.body_contains&&!$.toLowerCase().includes(i.body_contains.toLowerCase()))continue;o.push({subject:m,from:x,to:_,date:E,body:$,message_id:A});continue}else u.warn(`Messages API returned ${y.status}`)}catch(y){u.warn(`Failed to parse JSON response: ${y}`)}}try{let f=await fetch(h,{method:"GET",headers:{Authorization:`Basic ${Buffer.from(`api:${n}`).toString("base64")}`}});if(!f.ok){u.warn(`Could not fetch stored message: ${f.status}`);continue}let y=await f.text();u.info(`Fallback: Raw email length: ${y.length}`);let b=y.split(`
package/dist/reporter.js CHANGED
@@ -873,7 +873,7 @@ import*as k from"fs";import*as v from"path";import{v4 as vt}from"uuid";import{z
873
873
  });
874
874
  </script>
875
875
  </body>
876
- </html>`}var Te="0.1.66",Ae=Te!=="dev"?Te:void 0;var yn=3600*1e3,bn=10080*60*1e3;var ut={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return ut[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function ht(e){return[...e].sort(([t],[r])=>{let i=$e(t),n=$e(r);if(i!==n)return i-n;let a=Ee(t),o=Ee(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function ft(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function mt(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function q(e,t="main",r,i,n){r===void 0&&(r=!ft(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=mt(o.title,t);a.push(...q(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,y={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(y.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(y),o.steps.length>0&&a.push(...q(o.steps,u,r,i,n))}}return a}var Z=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let u of this.collected){let y=u.test.titlePath().join(" > "),b=r.get(y);b||(b=[],r.set(y,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let y=u[0].test.location.file,b=[],d,h,f;for(let g=0;g<u.length;g++){let{test:T,result:M}=u[g],A=await this.buildReportTest(T,M,y);d=A,h||(h=A.startTime),f=A.endTime,b.push({attemptNumber:g+1,status:M.status,duration:M.duration,steps:A.steps,error:A.error,videoPath:A.videoPath,tracePath:A.tracePath})}let p=b[b.length-1],{test:S}=u[u.length-1],_={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),y),status:p.status,duration:p.duration,steps:p.steps,error:p.error,videoPath:p.videoPath,tracePath:p.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:h,endTime:f,suiteName:d?.suiteName},w=b.some(g=>g.status==="failed"||g.status==="timedOut");b.length>1&&w&&p.status==="passed"&&(_.flaky=!0,_.retries=b.length-1,_.attempts=b),i.push(_)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);k.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let y=n.tests[u],b=y.attempts&&y.attempts.length>0,d=[{obj:y,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(y.attempts)for(let h=0;h<y.attempts.length;h++)d.push({obj:y.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:f,screenshotSubDir:p}of d){let S=v.join(o,p),_=!1;for(let w of h.steps)if(w.screenshot&&v.isAbsolute(w.screenshot))try{_||(k.mkdirSync(S,{recursive:!0}),_=!0);let g=`${w.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(w.screenshot,v.join(S,g)),w.screenshot=`screenshots/${p}/${g}`}catch(g){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,g)}if(h.videoPath&&v.isAbsolute(h.videoPath)){let w=v.extname(h.videoPath)||".webm",g=`${f}-video${w}`;try{k.copyFileSync(h.videoPath,v.join(a,g)),h.videoPath=g}catch{h.videoPath=void 0}}if(h.tracePath&&v.isAbsolute(h.tracePath)){let w=v.extname(h.tracePath)||".zip",g=`${f}-trace${w}`;try{k.copyFileSync(h.tracePath,v.join(a,g)),h.tracePath=g}catch{h.tracePath=void 0}}}}let c=v.join(a,"report-data.json");k.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(k.writeFileSync(l,ke(n),"utf-8"),console.log(`
876
+ </html>`}var Te="0.1.67",Ae=Te!=="dev"?Te:void 0;var yn=3600*1e3,bn=10080*60*1e3;var ut={before:0,main:1,teardown:2,after:3};function $e(e){let t=e.split(".")[0];return ut[t]??1}function Ee(e){return e.split(".").map(t=>{let r=Number(t);return Number.isNaN(r)?0:r})}function ht(e){return[...e].sort(([t],[r])=>{let i=$e(t),n=$e(r);if(i!==n)return i-n;let a=Ee(t),o=Ee(r);for(let c=0;c<Math.max(a.length,o.length);c++){let l=a[c]??-1,u=o[c]??-1;if(l!==u)return l-u}return 0})}function ft(e){let t=new Set;for(let r of e)if(t.add(r.category),r.category==="hook")for(let i of r.steps)t.add(i.category);return t}function mt(e,t){let r=e.toLowerCase();return r.includes("before")?"before":r.includes("after")?"after":t}function q(e,t="main",r,i,n){r===void 0&&(r=!ft(e).has("test.step")),n||(n=new Map);let a=[];for(let o of e){if(o.category==="fixture"||o.category==="test.attach")continue;if(o.category==="hook"){let l=mt(o.title,t);a.push(...q(o.steps,l,r,i,n));continue}if(o.category==="test.step"||r&&(o.category==="expect"||o.category==="pw:api")){let l=n.get(t)??0;n.set(t,l+1);let u=`${t}.${l}`,y={stepId:u,description:o.title,status:o.error?"failure":o.duration===-1?"skipped":"success",duration:o.duration>=0?o.duration:void 0};o.error&&(y.error=o.error.message??o.error.stack),i&&o.location&&i.set(u,o.location),a.push(y),o.steps.length>0&&a.push(...q(o.steps,u,r,i,n))}}return a}var Z=class{outputFolder;openMode;collected=[];config;runStartTime;constructor(t={}){this.outputFolder=t.outputFolder||"shiplight-report",this.openMode=t.open||"on-failure"}onBegin(t,r){this.config=t,this.runStartTime=new Date().toISOString()}onTestEnd(t,r){this.collected.push({test:t,result:r})}async onEnd(t){if(this.collected.length===0)return;let r=new Map;for(let u of this.collected){let y=u.test.titlePath().join(" > "),b=r.get(y);b||(b=[],r.set(y,b)),b.push(u)}let i=[];for(let[,u]of r.entries()){let y=u[0].test.location.file,b=[],d,h,f;for(let g=0;g<u.length;g++){let{test:T,result:M}=u[g],A=await this.buildReportTest(T,M,y);d=A,h||(h=A.startTime),f=A.endTime,b.push({attemptNumber:g+1,status:M.status,duration:M.duration,steps:A.steps,error:A.error,videoPath:A.videoPath,tracePath:A.tracePath})}let p=b[b.length-1],{test:S}=u[u.length-1],_={title:S.title,baseTitle:d?.baseTitle,file:v.relative(process.cwd(),y),status:p.status,duration:p.duration,steps:p.steps,error:p.error,videoPath:p.videoPath,tracePath:p.tracePath,actionStepsMap:d?.actionStepsMap,tags:d?.tags,baseUrl:d?.baseUrl,skip:d?.skip,slow:d?.slow,timeout:d?.timeout,parameterSetName:d?.parameterSetName,startTime:h,endTime:f,suiteName:d?.suiteName},w=b.some(g=>g.status==="failed"||g.status==="timedOut");b.length>1&&w&&p.status==="passed"&&(_.flaky=!0,_.retries=b.length-1,_.attempts=b),i.push(_)}let n={tests:i,totalDuration:t.duration,timestamp:new Date().toISOString(),shiplightVersion:Ae},a=v.isAbsolute(this.outputFolder)?this.outputFolder:v.join(process.cwd(),this.outputFolder);k.mkdirSync(a,{recursive:!0});let o=v.join(a,"screenshots");for(let u=0;u<n.tests.length;u++){let y=n.tests[u],b=y.attempts&&y.attempts.length>0,d=[{obj:y,prefix:b?`test-${u}-attempt-0`:`test-${u}`,screenshotSubDir:`test-${u}`}];if(y.attempts)for(let h=0;h<y.attempts.length;h++)d.push({obj:y.attempts[h],prefix:`test-${u}-attempt-${h+1}`,screenshotSubDir:`test-${u}/attempt-${h}`});for(let{obj:h,prefix:f,screenshotSubDir:p}of d){let S=v.join(o,p),_=!1;for(let w of h.steps)if(w.screenshot&&v.isAbsolute(w.screenshot))try{_||(k.mkdirSync(S,{recursive:!0}),_=!0);let g=`${w.stepId.replace(/\./g,"-")}.png`;k.copyFileSync(w.screenshot,v.join(S,g)),w.screenshot=`screenshots/${p}/${g}`}catch(g){console.warn(`[reporter] Failed to copy screenshot for ${w.stepId}:`,g)}if(h.videoPath&&v.isAbsolute(h.videoPath)){let w=v.extname(h.videoPath)||".webm",g=`${f}-video${w}`;try{k.copyFileSync(h.videoPath,v.join(a,g)),h.videoPath=g}catch{h.videoPath=void 0}}if(h.tracePath&&v.isAbsolute(h.tracePath)){let w=v.extname(h.tracePath)||".zip",g=`${f}-trace${w}`;try{k.copyFileSync(h.tracePath,v.join(a,g)),h.tracePath=g}catch{h.tracePath=void 0}}}}let c=v.join(a,"report-data.json");k.writeFileSync(c,JSON.stringify(n,null,2),"utf-8");let l=v.join(a,"index.html");if(k.writeFileSync(l,ke(n),"utf-8"),console.log(`
877
877
  Shiplight report written to: ${l}`),this.openMode==="always"||this.openMode==="on-failure"&&t.status!=="passed")try{let u=(await import("open")).default;await u(l)}catch{}}printsToStdio(){return!1}async buildReportTest(t,r,i){let n={title:t.title,file:v.relative(process.cwd(),i),status:r.status,duration:r.duration,steps:[],startTime:new Date(r.startTime).toISOString(),endTime:new Date(r.startTime.getTime()+r.duration).toISOString()};r.errors.length>0&&(n.error=r.errors.map(d=>d.message||d.stack||String(d)).join(`
878
878
 
879
879
  `)),r.stdout.length>0&&(n.stdout=r.stdout.map(d=>typeof d=="string"?d:d.toString()).join("")),r.stderr.length>0&&(n.stderr=r.stderr.map(d=>typeof d=="string"?d:d.toString()).join(""));for(let d of r.attachments)d.name==="video"&&d.path&&(n.videoPath=d.path),d.name==="trace"&&d.path&&(n.tracePath=d.path);let a=r.attachments.find(d=>d.name==="shiplight-results"),o=null;if(a)try{if(a.body)o=JSON.parse(a.body.toString("utf-8"));else if(a.path){let d=k.readFileSync(a.path,"utf-8");o=JSON.parse(d)}}catch{}let c=i.replace(/\.yaml\.spec\.ts$/,".test.yaml"),l={},u=t.title.match(/^(.*)\s+\[([^\]]+)\]$/),y=u?u[1]:t.title,b=u?u[2]:void 0;if(b&&(n.parameterSetName=b),k.existsSync(c))try{let d=k.readFileSync(c,"utf-8"),h=_e(d,c);if(h.suite){let f=h.suite.tests.find(p=>p.name===y);f&&(l=K(f.testFlow),f.tags?.length&&(n.tags=f.tags),f.skip!==void 0&&(n.skip=f.skip),f.slow&&(n.slow=f.slow),f.timeout!==void 0&&(n.timeout=f.timeout),n.baseTitle=f.name||f.testFlow?.goal),n.suiteName=h.name,h.tags?.length&&(n.suiteTags=h.tags),h.use?.baseURL&&(n.baseUrl=h.use.baseURL)}else h.testFlow&&(l=K(h.testFlow),n.baseTitle=h.name||h.testFlow?.goal,h.tags?.length&&(n.tags=h.tags),h.use?.baseURL&&(n.baseUrl=h.use.baseURL))}catch{}if(o||Object.keys(l).length>0){let d=new Set([...Object.keys(l),...Object.keys(o||{})]),h=Array.from(d).map(p=>[p,null]),f=ht(h);for(let[p]of f){let S=l[p],_=o?.[p],w=S?.description;if(!w||w==="Action"||w==="Draft"){let T=S?.action_entity;w=T?.action_description||T?.action_data?.kwargs?.description||_?.description||p}let g={stepId:p,description:w,status:_?.status||"pending",duration:_?.duration};if(_?.message){let T=typeof _.message=="string"?_.message:JSON.stringify(_.message,null,2);_.status==="failure"?g.error=T:g.message=T}if(_?.screenshot){let T=_.screenshot,M=a?.path?v.dirname(a.path):"",A=v.isAbsolute(T)?T:v.join(M,T);k.existsSync(A)&&(g.screenshot=A)}_?.code&&(g.code=_.code),n.steps.push(g)}}if(o===null&&Object.keys(l).length===0&&!i.endsWith(".yaml.spec.ts")){let d=new Map;if(n.steps=q(r.steps,"main",void 0,d),n.steps.length>0){let h=new Map;n.actionStepsMap=Object.fromEntries(n.steps.map(f=>{let p=d.get(f.stepId),S;if(p?.file){if(!h.has(p.file))try{h.set(p.file,k.readFileSync(p.file,"utf-8").split(`
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "shiplightai",
3
- "version": "0.1.66",
3
+ "version": "0.1.67",
4
4
  "type": "module",
5
5
  "description": "Shiplight CLI for running and debugging .test.yaml files",
6
6
  "main": "dist/index.js",
@@ -99,10 +99,10 @@
99
99
  "typescript": "5.5.4",
100
100
  "mcp-tools": "1.0.0",
101
101
  "sdk-core": "0.1.0",
102
- "sdk-internal": "0.1.1",
103
102
  "shiplight-tools": "1.0.0",
104
103
  "shiplight-types": "0.1.0",
105
- "@loggia/common": "1.0.0"
104
+ "@loggia/common": "1.0.0",
105
+ "sdk-internal": "0.1.1"
106
106
  },
107
107
  "peerDependencies": {
108
108
  "@playwright/test": "^1.60.0"