@testrelic/maestro-analytics 1.0.0-next.14

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/index.js ADDED
@@ -0,0 +1,1256 @@
1
+ import {isValidEndpointUrl,isValidCloudConfig,createError,ErrorCode,isValidQueueEntry}from'@testrelic/core';import {existsSync,statSync,readFileSync,readdirSync,mkdirSync,writeFileSync,renameSync,unlinkSync,createReadStream}from'fs';import {resolve,join,dirname,basename,extname}from'path';import {XMLParser}from'fast-xml-parser';import {parseDocument}from'yaml';import {exec,spawn,execSync}from'child_process';import {tmpdir}from'os';import {createHash,randomUUID,randomBytes}from'crypto';import {gzipSync}from'zlib';import {Readable}from'stream';var Xe=".testrelic",qt="testrelic-config.json",Jt=".testrelic",Kt=5,Wt=new Set(["__proto__","constructor","prototype"]),ae=Object.freeze({apiKey:null,endpoint:"https://platform.testrelic.ai/api/v1",uploadStrategy:"batch",timeout:3e4,projectName:null,queueMaxAge:6048e5,queueDirectory:`${Xe}/queue`,uploadArtifacts:true,artifactMaxSizeMb:50}),Xt={s:1e3,m:60*1e3,h:3600*1e3,d:1440*60*1e3};function le(e){let t=resolve(e);for(let r=0;r<=Kt;r++){let o=join(t,Xe,qt);if(existsSync(o))try{if(statSync(o).isFile())return o}catch{}let n=join(t,Jt);if(existsSync(n))try{if(statSync(n).isFile())return process.stderr.write(`[testrelic] Deprecation: config file ".testrelic" has moved to ".testrelic/testrelic-config.json". Please migrate your config file to the new location.
2
+ `),n}catch{}let s=dirname(t);if(s===t)break;t=s;}return null}function ce(e){try{let t=readFileSync(e,"utf-8"),r=JSON.parse(t);return typeof r!="object"||r===null||Array.isArray(r)||!Ye(r)?null:r}catch{return null}}function Ye(e){for(let t of Object.keys(e)){if(Wt.has(t))return false;let r=e[t];if(typeof r=="object"&&r!==null&&!Array.isArray(r)&&!Ye(r))return false}return true}function B(e){let t=/^\$\{([A-Za-z_][A-Za-z0-9_]*)\}$/.exec(e);if(t)return process.env[t[1]]??null;let r=/^\$([A-Za-z_][A-Za-z0-9_]*)$/.exec(e);return r?process.env[r[1]]??null:e}function H(e){let t=Object.create(null);for(let r of Object.keys(e)){let o=e[r];typeof o=="string"&&o.startsWith("$")?t[r]=B(o):typeof o=="object"&&o!==null&&!Array.isArray(o)?t[r]=H(o):t[r]=o;}return t}function ie(e){let t=/^(\d+)\s*([smhd])$/.exec(e.trim());if(!t)return null;let r=parseInt(t[1],10),o=t[2],n=Xt[o];return !n||r<=0?null:r*n}function de(e,t){let r=Object.create(null);if(Object.assign(r,ae),e){let l=e.cloud;l&&typeof l=="object"&&(typeof l.endpoint=="string"&&(r.endpoint=l.endpoint),typeof l.upload=="string"&&(r.uploadStrategy=l.upload),typeof l.timeout=="number"&&(r.timeout=l.timeout),typeof l.apiKey=="string"&&l.apiKey.length>0&&(r.apiKey=l.apiKey));let d=e["testrelic-repo"]??e.project;d&&typeof d=="object"&&typeof d.name=="string"&&(r.projectName=d.name);let c=e.queue;if(c&&typeof c=="object"){if(typeof c.maxAge=="string"){let p=ie(c.maxAge);p!==null&&(r.queueMaxAge=p);}typeof c.directory=="string"&&(r.queueDirectory=c.directory);}}if(t){if(typeof t.apiKey=="string"&&t.apiKey.length>0){let l=t.apiKey.startsWith("$")?B(t.apiKey):t.apiKey;l&&(r.apiKey=l);}if(typeof t.endpoint=="string"){let l=t.endpoint.startsWith("$")?B(t.endpoint):t.endpoint;l&&(r.endpoint=l);}if(typeof t.upload=="string"&&(r.uploadStrategy=t.upload),typeof t.timeout=="number"&&(r.timeout=t.timeout),typeof t.projectName=="string"&&(r.projectName=t.projectName),typeof t.queueMaxAge=="string"){let l=ie(t.queueMaxAge);l!==null&&(r.queueMaxAge=l);}typeof t.queueDirectory=="string"&&(r.queueDirectory=t.queueDirectory),typeof t.uploadArtifacts=="boolean"&&(r.uploadArtifacts=t.uploadArtifacts),typeof t.artifactMaxSizeMb=="number"&&(r.artifactMaxSizeMb=t.artifactMaxSizeMb);}let o=process.env.TESTRELIC_API_KEY;o&&o.length>0&&(r.apiKey=o);let n=process.env.TESTRELIC_CLOUD_ENDPOINT;n&&isValidEndpointUrl(n)&&(r.endpoint=n);let s=process.env.TESTRELIC_UPLOAD_STRATEGY;s&&["batch","realtime","both"].includes(s)&&(r.uploadStrategy=s);let a=process.env.TESTRELIC_CLOUD_TIMEOUT;if(a){let l=parseInt(a,10);!isNaN(l)&&l>=1e3&&l<=12e4&&(r.timeout=l);}let i=Object.freeze(r);return isValidCloudConfig(i)?i:ae}var Zt=new Set(["__proto__","constructor","prototype"]);function Qe(e){if(typeof e!="object"||e===null)return false;for(let t of Object.keys(e))if(Zt.has(t))return true;return false}function er(e){if(typeof e!="object"||e===null||Qe(e))return false;let t=e;return !(t.outputPath!==void 0&&typeof t.outputPath!="string"||t.htmlReportPath!==void 0&&typeof t.htmlReportPath!="string"||t.openReport!==void 0&&typeof t.openReport!="boolean"||t.includeScreenshots!==void 0&&typeof t.includeScreenshots!="boolean"||t.includeVideo!==void 0&&typeof t.includeVideo!="boolean"||t.includeAiAnalysis!==void 0&&typeof t.includeAiAnalysis!="boolean"||t.includeLogs!==void 0&&typeof t.includeLogs!="boolean"||t.includeFlowMetadata!==void 0&&typeof t.includeFlowMetadata!="boolean"||t.quiet!==void 0&&typeof t.quiet!="boolean"||t.metadata!==void 0&&t.metadata!==null&&(typeof t.metadata!="object"||Qe(t.metadata)))}function tr(e){if(e!==void 0&&!er(e))throw createError(ErrorCode.CONFIG_INVALID,"Invalid Maestro reporter configuration");let t=Object.create(null),r=e?.outputPath??"./test-results/testrelic-maestro.json";return t.outputPath=r,t.htmlReportPath=e?.htmlReportPath??r.replace(/\.json$/,".html"),t.openReport=e?.openReport??true,t.includeScreenshots=e?.includeScreenshots??true,t.includeVideo=e?.includeVideo??true,t.includeAiAnalysis=e?.includeAiAnalysis??true,t.includeLogs=e?.includeLogs??true,t.includeFlowMetadata=e?.includeFlowMetadata??true,t.flowsDir=e?.flowsDir??null,t.testRunId=e?.testRunId??null,t.metadata=e?.metadata??null,t.quiet=e?.quiet??false,t.reportMode=e?.reportMode??"batch",t.cloud=rr(e?.cloud??null),Object.freeze(t)}function rr(e){let t=le(process.cwd()),r=t?ce(t):null,o=r?H(r):null,n=de(o,e??void 0);return n.apiKey?n:null}var nr=new XMLParser({ignoreAttributes:false,attributeNamePrefix:"@_",allowBooleanAttributes:true,parseAttributeValue:true,trimValues:true});function G(e){return e==null?[]:Array.isArray(e)?e:[e]}function et(e){return !e||typeof e!="object"?[]:G(e.property).map(o=>({name:String(o["@_name"]??""),value:String(o["@_value"]??"")}))}function ar(e){let t=String(e["@_name"]??"unnamed"),r=String(e["@_classname"]??t),o=Number(e["@_time"]??0),n=e["@_id"]!=null?String(e["@_id"]):null,s=String(e["@_status"]??"").toUpperCase(),a=e.failure,i=e.error,l=e.skipped!==void 0,d="SUCCESS";return a?d="FAILURE":i?d="ERROR":l?d="SKIPPED":s==="FAILURE"||s==="FAILED"?d="FAILURE":s==="ERROR"?d="ERROR":s==="SKIPPED"&&(d="SKIPPED"),{id:n,name:t,classname:r,time:o,status:d,failureMessage:a?String(a["@_message"]??a["#text"]??""):null,failureType:a?String(a["@_type"]??""):null,errorMessage:i?String(i["@_message"]??i["#text"]??""):null,errorType:i?String(i["@_type"]??""):null,properties:et(e.properties)}}function Ze(e){let t=String(e["@_name"]??"Test Suite"),r=e["@_device"]!=null?String(e["@_device"]):null,o=Number(e["@_tests"]??0),n=Number(e["@_failures"]??0),s=Number(e["@_errors"]??0),a=Number(e["@_skipped"]??0),i=Number(e["@_time"]??0),d=G(e.testcase).map(ar);return {name:t,device:r,tests:o,failures:n,errors:s,skipped:a,time:i,testCases:d,properties:et(e.properties)}}function tt(e){let t=nr.parse(e),r=[];if(t.testsuites){let l=t.testsuites;r=G(l.testsuite).map(Ze);}else t.testsuite&&(r=G(t.testsuite).map(Ze));let o=0,n=0,s=0,a=0,i=0;for(let l of r)o+=l.tests,n+=l.failures,s+=l.errors,a+=l.skipped,i+=l.time;return {testSuites:r,totalTests:o,totalFailures:n,totalErrors:s,totalSkipped:a,totalTime:i}}function pe(e){let t=readFileSync(e,"utf-8");return tt(t)}var ur=new Set(["tapOn","doubleTapOn","longPressOn","inputText","eraseText","pasteText","swipe","scroll","scrollUntilVisible","hideKeyboard","pressKey","setClipboard","copyTextFrom"]),fr=new Set(["assertVisible","assertNotVisible","assertTrue","assertScreenshot","assertNoDefectsWithAI","assertWithAI","assertCondition"]),gr=new Set(["launchApp","killApp","stopApp","clearState","openLink","back","clearKeychain"]),mr=new Set(["setAirplaneMode","toggleAirplaneMode","setLocation","setOrientation","setPermissions","addMedia","travel"]),hr=new Set(["takeScreenshot","startRecording","stopRecording"]),br=new Set(["runScript","evalScript"]),yr=new Set(["runFlow","repeat","retry","waitForAnimationToEnd","extendedWaitUntil"]),xr=new Set(["assertWithAI","assertNoDefectsWithAI","extractTextWithAI"]);function ot(e){return xr.has(e)?"ai":fr.has(e)?"assertion":ur.has(e)?"interaction":gr.has(e)?"navigation":mr.has(e)?"device":hr.has(e)?"media":br.has(e)?"script":yr.has(e)?"flow_control":"other"}var vr={tapOnElement:"tapOn",assertCondition:"assertVisible",backPress:"back",applyConfiguration:"applyConfiguration",defineVariables:"defineVariables"};function wr(e){let t=e.replace(/Command$/,"");return vr[t]??t}function kr(e){if(e.selector&&typeof e.selector=="object"){let t=e.selector;if(typeof t.textRegex=="string")return t.textRegex;if(typeof t.text=="string")return t.text;if(typeof t.id=="string")return `#${t.id}`}if(e.condition&&typeof e.condition=="object"){let t=e.condition,r=t.visible??t.notVisible;if(r){if(typeof r.textRegex=="string")return r.textRegex;if(typeof r.text=="string")return r.text}}if(typeof e.path=="string")return e.path}function rt(e,t,r){let o,n;if(typeof e.command=="object"&&e.command!==null&&!Array.isArray(e.command)){let c=e.command,p=Object.keys(c)[0]??`step-${t}`;o=wr(p);let f=c[p]??{};n=kr(f);}else o=e.command??e.commandName??e.name??`step-${t}`,n=e.selector??void 0;let s=e.metadata??{},a=Cr(s.status??e.status),i=s.duration??e.duration??e.durationMs??0,l;typeof s.timestamp=="number"?l=new Date(s.timestamp).toISOString():l=e.timestamp??e.time??r;let d=e.error??e.errorMessage??void 0;return {command:o,category:ot(o),status:a,duration:i,timestamp:l,...n?{selector:n}:{},...d?{error:d}:{}}}function Cr(e){if(!e)return "completed";let t=e.toLowerCase();return t==="failed"||t==="error"?"failed":t==="skipped"?"skipped":"completed"}function st(e,t){let r=t??new Date().toISOString();try{let o=JSON.parse(e);if(Array.isArray(o))return o.map((n,s)=>rt(n,s,r));if(typeof o=="object"&&o!==null){let n=o,s=n.commands??n.steps??n.data;if(Array.isArray(s))return s.map((a,i)=>rt(a,i,r))}return []}catch{return []}}function ue(e,t){try{let r=readFileSync(e,"utf-8");return st(r,t)}catch{return []}}function fe(e){if(!existsSync(e))return [];try{return readdirSync(e,{recursive:!0}).map(String).filter(t=>basename(t).startsWith("commands")&&t.endsWith(".json")).map(t=>join(e,t))}catch{return []}}function Mr(e){let t=[];for(let r of e)if(typeof r=="object"&&r!==null){let o=r;if(typeof o.runFlow=="string")t.push(o.runFlow);else if(typeof o.runFlow=="object"&&o.runFlow!==null){let n=o.runFlow;typeof n.file=="string"&&t.push(n.file);}}return t}function at(e){if(!e)return [];let t=[],r=Array.isArray(e)?e:[e];for(let o of r)if(typeof o=="string")t.push(o);else if(typeof o=="object"&&o!==null){let n=o;if(typeof n.runFlow=="string")t.push(n.runFlow);else if(typeof n.runFlow=="object"&&n.runFlow!==null){let s=n.runFlow;typeof s.file=="string"&&t.push(s.file);}typeof n.runScript=="string"&&t.push(n.runScript);}return t}function it(e,t){let r=e.split(/^---\s*$/m,2),o=r[0]??"",n=r[1]??"",s={};try{let g=parseDocument(o).toJS();typeof g=="object"&&g!==null&&!Array.isArray(g)&&(s=g);}catch{}let a=[];if(n.trim())try{let g=parseDocument(n).toJS();Array.isArray(g)&&(a=g);}catch{}let i=typeof s.appId=="string"?s.appId:null,l=typeof s.name=="string"?s.name:null,d=[];if(Array.isArray(s.tags))for(let m of s.tags)typeof m=="string"&&d.push(m);let c={};if(typeof s.env=="object"&&s.env!==null&&!Array.isArray(s.env))for(let[m,g]of Object.entries(s.env))c[m]=String(g);let p={};if(typeof s.properties=="object"&&s.properties!==null&&!Array.isArray(s.properties))for(let[m,g]of Object.entries(s.properties))p[m]=String(g);let f=at(s.onFlowStart),x=at(s.onFlowComplete),v=Mr(a);return {appId:i,name:l,tags:d,env:c,properties:p,onFlowStart:f,onFlowComplete:x,subflowRefs:v,filePath:t}}function ge(e){let t=readFileSync(e,"utf-8");return it(t,e)}function me(e){if(!existsSync(e))return [];let t=[];function r(o){try{let n=readdirSync(o,{withFileTypes:!0});for(let s of n){let a=join(o,s.name);if(s.isDirectory())r(a);else if(s.isFile()){let i=extname(s.name).toLowerCase();(i===".yaml"||i===".yml")&&t.push(a);}}}catch{}}return r(e),t}var Dr=/^(\d{4}-\d{2}-\d{2}[T ]\d{2}:\d{2}:\d{2}(?:\.\d+)?(?:Z|[+-]\d{2}:?\d{2})?)\s+(\w+)\s+(.+)$/,Nr=/^\[?(\d{2}:\d{2}:\d{2}(?:\.\d+)?)\]?\s+(\w+)\s+(.+)$/;function lt(e){let t=e.toUpperCase();return t==="DEBUG"||t==="TRACE"||t==="VERBOSE"?"DEBUG":t==="INFO"?"INFO":t==="WARN"||t==="WARNING"?"WARN":t==="ERROR"||t==="FATAL"||t==="SEVERE"?"ERROR":"INFO"}function ct(e){let t=[],r=e.split(`
3
+ `),o=new Date().toISOString().split("T")[0];for(let n of r){let s=n.trim();if(!s)continue;let a=Dr.exec(s);if(a){t.push({timestamp:a[1],level:lt(a[2]),message:a[3],source:null});continue}if(a=Nr.exec(s),a){t.push({timestamp:`${o}T${a[1]}`,level:lt(a[2]),message:a[3],source:null});continue}if(t.length>0){let i=t[t.length-1];t[t.length-1]={...i,message:i.message+`
4
+ `+s};}}return t}function he(e){try{let t=readFileSync(e,"utf-8");return ct(t)}catch{return []}}function be(e){for(let t of e){let r=t.message.toLowerCase();if(r.includes("android")||r.includes("adb")||r.includes("uiautomator")||r.includes("emulator"))return "android";if(r.includes("ios")||r.includes("xcuitest")||r.includes("simulator")||r.includes("xctest"))return "ios";if(r.includes("web driver")||r.includes("chrome")||r.includes("chromium"))return "web"}return "unknown"}function ye(e){if(!existsSync(e))return [];try{return readdirSync(e,{recursive:!0}).map(String).filter(t=>basename(t)==="maestro.log"||t.endsWith(".log")).map(t=>join(e,t))}catch{return []}}var Hr=/(?:defect|issue|bug|error|warning|problem)/i,Gr=/(?:critical|severe|major|blocker)/i,Vr=/(?:warning|moderate|minor)/i,xe=/(?:cut[\s-]?off|overlap|truncat|misalign|overflow|clip|obscur|hidden\s+text|broken\s+layout)/i,V=/(?:spelling|typo|misspell|grammar)/i,q=/(?:i18n|internationali[sz]ation|locali[sz]ation|untranslated|missing\s+translation)/i,Ce=/(?:layout|spacing|padding|margin|alignment|centering|position)/i,Re=/(?:accessib|a11y|contrast|screen\s*reader|alt\s*text|aria)/i;function ve(e){return V.test(e)?"spelling":q.test(e)?"i18n":Re.test(e)?"accessibility":Ce.test(e)?"layout":"ui"}function we(e){return Gr.test(e)?"critical":Vr.test(e)?"warning":"info"}function ke(e){return e.replace(/<[^>]*>/g," ").replace(/\s+/g," ").trim()}function qr(e){let t=[],r=ke(e),o=/<li[^>]*>([\s\S]*?)<\/li>/gi,n;for(;(n=o.exec(e))!==null;){let s=n[1],a=ke(s);!a||a.length<5||(Hr.test(a)||xe.test(a)||V.test(a)||q.test(a))&&t.push({type:ve(a),severity:we(a),description:a,location:null,screenshot:null});}if(t.length===0){let s=/<p[^>]*>([\s\S]*?)<\/p>/gi;for(;(n=s.exec(e))!==null;){let a=ke(n[1]);!a||a.length<10||(xe.test(a)||V.test(a)||q.test(a)||Ce.test(a)||Re.test(a))&&t.push({type:ve(a),severity:we(a),description:a,location:null,screenshot:null});}}if(t.length===0&&r.length>20){let s=r.split(/[.!?]+/).filter(a=>a.trim().length>10);for(let a of s){let i=a.trim();(xe.test(i)||V.test(i)||q.test(i)||Ce.test(i)||Re.test(i))&&t.push({type:ve(i),severity:we(i),description:i,location:null,screenshot:null});}}return t}function dt(e){let t=qr(e);return {defects:t,totalDefects:t.length,hasIssues:t.length>0,rawHtml:e}}function Se(e){try{let t=readFileSync(e,"utf-8");return dt(t)}catch{return {defects:[],totalDefects:0,hasIssues:false,rawHtml:null}}}function Te(e){if(!existsSync(e))return [];try{return readdirSync(e,{recursive:!0}).map(String).filter(t=>{let r=basename(t).toLowerCase();return extname(t).toLowerCase()===".html"&&(r.includes("insight")||r.includes("ai")||r.includes("analysis"))}).map(t=>join(e,t))}catch{return []}}var Qr=new Set([".png",".jpg",".jpeg",".webp"]),Zr=new Set([".mp4",".webm",".mov"]);function pt(e){let t=[];if(!existsSync(e))return t;function r(o){try{let n=readdirSync(o,{withFileTypes:!0});for(let s of n){let a=join(o,s.name);s.isDirectory()?r(a):s.isFile()&&t.push(a);}}catch{}}return r(e),t}function Ae(e,t){let r=[],o=[],n=[],s=[],a=[],i=null,l=[];e&&l.push(...pt(e)),t&&l.push(...pt(t));let d=new Set;for(let c of l){if(d.has(c))continue;d.add(c);let p=basename(c).toLowerCase(),f=extname(c).toLowerCase();if(Qr.has(f)){r.push(c);continue}if(Zr.has(f)){o.push(c);continue}if(p==="maestro.log"||f===".log"&&p.includes("maestro")){n.push(c);continue}if(p.startsWith("commands")&&f===".json"){s.push(c);continue}if(f===".html"&&(p.includes("insight")||p.includes("ai")||p.includes("analysis"))){a.push(c);continue}if(f===".xml"&&(p.includes("report")||p.includes("junit"))){i||(i=c);continue}}return {screenshotPaths:r,videoPaths:o,logPaths:n,commandJsonPaths:s,aiReportPaths:a,junitReportPath:i}}function eo(e){return {screenshots:e.screenshotPaths.length,videos:e.videoPaths.length,logs:e.logPaths.length,commandFiles:e.commandJsonPaths.length,aiReports:e.aiReportPaths.length,junitReport:e.junitReportPath?1:0}}function to(e){switch(e){case "interaction":return "ui_action";case "assertion":return "assertion";case "ai":return "assertion";case "navigation":return "custom_step";case "device":return "custom_step";case "media":return "custom_step";case "script":return "custom_step";case "flow_control":return "custom_step";case "recording":return "custom_step";default:return "custom_step"}}function ut(e){return {title:e.selector?`${e.command} \u2192 ${e.selector}`:e.command,category:to(e.category),status:e.status==="failed"?"failed":"passed",duration:e.duration,timestamp:e.timestamp,videoOffset:null,error:e.error??null,children:[]}}function ro(e){let t=[...e.commands.map(ut),...e.assertions.map(ut)];t.sort((n,s)=>new Date(n.timestamp).getTime()-new Date(s.timestamp).getTime());let r=e.screenshotPaths.length>0||e.videoPath?{screenshot:e.screenshotPaths[0]??void 0,video:e.videoPath??void 0}:null,o=[...e.tags];return e.platform!=="unknown"&&!o.includes(e.platform)&&o.push(e.platform),{title:e.flowName,status:e.status,duration:e.duration,startedAt:e.startedAt,completedAt:e.completedAt,retryCount:0,tags:o,failure:e.failureMessage?{message:e.failureMessage,line:null,code:null,stack:null}:null,testId:`${e.flowFile}::${e.flowName}`,filePath:e.flowFile,suiteName:e.appId??"maestro-suite",testType:"mobile",isFlaky:false,retryStatus:null,expectedStatus:"passed",actualStatus:e.status,artifacts:r,networkRequests:null,apiCalls:null,apiAssertions:null,actions:t.length>0?t:null,consoleLogs:null}}function ft(e){let t=ro(e);return {url:e.appId??"maestro-flow",navigationType:"page_load",visitedAt:e.startedAt,duration:e.duration,specFile:e.flowFile,domContentLoadedAt:null,networkIdleAt:null,networkStats:null,tests:[t]}}function Ie(e){return e.map(ft)}var oo={"2xx":0,"3xx":0,"4xx":0,"5xx":0,error:0};function z(e){let t=0,r=0,o=0,n=0,s=0,a=0,i=0,l=0,d=0,c=e.length,p=0,f={},x=new Set;for(let v of e){x.add(v.url);for(let m of v.tests){switch(t++,m.status){case "passed":r++;break;case "failed":o++;break;case "flaky":n++;break;case "skipped":s++;break;case "timedout":a++;break}if(m.actions)for(let g of m.actions)if(p++,g.category==="assertion")i++,g.status==="passed"?l++:d++;else {let w=g.category;f[w]=(f[w]??0)+1;}}}return {total:t,passed:r,failed:o,flaky:n,skipped:s,timedout:a,totalApiCalls:0,uniqueApiUrls:0,apiCallsByMethod:{},apiCallsByStatusRange:oo,apiResponseTime:null,totalAssertions:i,passedAssertions:l,failedAssertions:d,totalNavigations:c,uniqueNavigationUrls:x.size,totalTimelineSteps:e.length,totalActionSteps:p,actionStepsByCategory:f}}var gt=`
5
+ /* \u2500\u2500 Theme Variables (verbatim from Appium) \u2500\u2500 */
6
+ :root,[data-theme="dark"]{
7
+ --bg:#0f1117;--bg-1:#161b22;--bg-2:#1c2128;--bg-3:#21262d;--bg-code:#13111c;
8
+ --fg:#e6edf3;--fg-1:#8b949e;--fg-2:#484f58;--fg-code:#d4d4d4;--fg-err:#f8d7da;
9
+ --bd:rgba(255,255,255,0.06);--bd-s:rgba(255,255,255,0.04);--bd-m:rgba(255,255,255,0.08);--bd-l:rgba(255,255,255,0.1);--bd-xs:rgba(255,255,255,0.03);
10
+ --hvr:rgba(255,255,255,0.025);--hvr-s:rgba(255,255,255,0.02);
11
+ --overlay-bg:rgba(0,0,0,.55);--shadow-c:rgba(0,0,0,.35);
12
+ --lb-bg:rgba(0,0,0,.92);--lb-shadow:rgba(0,0,0,.6);--lb-btn:rgba(255,255,255,.1);--lb-btn-h:rgba(255,255,255,.2);
13
+ --scroll-thumb:#21262d;
14
+ }
15
+ [data-theme="light"]{
16
+ --bg:#ffffff;--bg-1:#f6f8fa;--bg-2:#eaeef2;--bg-3:#d0d7de;--bg-code:#f6f8fa;
17
+ --fg:#1f2328;--fg-1:#656d76;--fg-2:#8b949e;--fg-code:#24292e;--fg-err:#82071e;
18
+ --bd:rgba(0,0,0,0.08);--bd-s:rgba(0,0,0,0.04);--bd-m:rgba(0,0,0,0.12);--bd-l:rgba(0,0,0,0.15);--bd-xs:rgba(0,0,0,0.03);
19
+ --hvr:rgba(0,0,0,0.04);--hvr-s:rgba(0,0,0,0.03);
20
+ --overlay-bg:rgba(0,0,0,.3);--shadow-c:rgba(0,0,0,.1);
21
+ --lb-bg:rgba(0,0,0,.85);--lb-shadow:rgba(0,0,0,.3);--lb-btn:rgba(0,0,0,.12);--lb-btn-h:rgba(0,0,0,.2);
22
+ --scroll-thumb:#d0d7de;
23
+ }
24
+ *,*::before,*::after{box-sizing:border-box;margin:0;padding:0}
25
+ body{font-family:Inter,system-ui,-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;background:var(--bg);color:var(--fg);line-height:1.5;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}
26
+
27
+ /* \u2500\u2500 Layout \u2500\u2500 */
28
+ .wrap{max-width:960px;margin:0 auto;padding:32px 24px 64px}
29
+
30
+ /* \u2500\u2500 Topbar (same as Appium) \u2500\u2500 */
31
+ .top-bar{display:flex;align-items:center;gap:14px;margin-bottom:20px;flex-wrap:wrap}
32
+ .top-bar-actions{display:flex;align-items:center;gap:8px;flex-shrink:0;margin-left:auto}
33
+ .brand{display:flex;align-items:center;gap:10px}
34
+ .brand svg{flex-shrink:0}
35
+ .brand-text{display:flex;flex-direction:column}
36
+ .brand-title{font-size:17px;font-weight:700;color:var(--fg);line-height:1.2}
37
+ .brand-sub{font-size:10px;font-weight:600;color:#03b79c;letter-spacing:.06em;text-transform:uppercase;margin-top:1px}
38
+ .run-meta{display:flex;flex-wrap:wrap;gap:4px 14px;font-size:11px;color:var(--fg-1);flex:1;min-width:0}
39
+ .run-meta-item{display:flex;align-items:center;gap:4px}
40
+ .run-meta-label{font-weight:600}
41
+ .run-id-tag{font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:10px;background:var(--bg-2);padding:1px 6px;border-radius:4px;color:var(--fg-1)}
42
+ .ci-tag{display:inline-flex;align-items:center;gap:4px;background:rgba(59,130,246,0.12);color:#60a5fa;padding:1px 6px;border-radius:4px;font-size:10px;font-weight:600}
43
+ .platform-tag{display:inline-flex;align-items:center;gap:4px;background:rgba(3,183,156,0.12);color:#2dd4a8;padding:1px 6px;border-radius:4px;font-size:10px;font-weight:600}
44
+
45
+ /* \u2500\u2500 Theme Toggle (same as Appium) \u2500\u2500 */
46
+ .theme-toggle{display:inline-flex;border:1px solid var(--bd-m);border-radius:8px;overflow:hidden;background:var(--bg);flex-shrink:0}
47
+ .theme-btn{padding:6px 10px;border:none;background:transparent;color:var(--fg-2);cursor:pointer;display:flex;align-items:center;justify-content:center;transition:all .15s;font-family:inherit}
48
+ .theme-btn:not(:last-child){border-right:1px solid var(--bd-m)}
49
+ .theme-btn.active{background:var(--bg-3);color:var(--fg)}
50
+ .theme-btn:hover:not(.active){color:var(--fg-1);background:var(--hvr)}
51
+ .theme-btn svg{width:14px;height:14px}
52
+
53
+ /* \u2500\u2500 CTA Button (same as Appium) \u2500\u2500 */
54
+ .cta-btn{display:inline-flex;align-items:center;gap:5px;font-size:11px;font-weight:500;color:#e6edf3;background:linear-gradient(135deg,rgba(3,183,156,0.15),rgba(14,165,233,0.15));border:1px solid rgba(3,183,156,0.3);padding:6px 14px;border-radius:8px;text-decoration:none;white-space:nowrap;flex-shrink:0;transition:all .2s;letter-spacing:.01em}
55
+ .cta-btn:hover{background:linear-gradient(135deg,rgba(3,183,156,0.25),rgba(14,165,233,0.25));border-color:rgba(3,183,156,0.5);box-shadow:0 0 16px rgba(3,183,156,0.15);color:#fff}
56
+ .cta-btn strong{font-weight:700;color:#2dd4a8}
57
+ .cta-btn:hover strong{color:#5eead4}
58
+ .cta-icon{width:13px;height:13px;color:#2dd4a8;flex-shrink:0}
59
+ .cta-arrow{width:11px;height:11px;color:#2dd4a8;flex-shrink:0;transition:transform .2s}
60
+ .cta-btn:hover .cta-arrow{transform:translateX(2px)}
61
+ [data-theme="light"] .cta-btn{color:#1f2328;background:linear-gradient(135deg,rgba(3,183,156,0.08),rgba(14,165,233,0.08));border-color:rgba(3,183,156,0.25)}
62
+ [data-theme="light"] .cta-btn:hover{background:linear-gradient(135deg,rgba(3,183,156,0.15),rgba(14,165,233,0.15));border-color:rgba(3,183,156,0.4);color:#0f172a}
63
+ [data-theme="light"] .cta-btn strong{color:#059669}
64
+ [data-theme="light"] .cta-icon,[data-theme="light"] .cta-arrow{color:#059669}
65
+
66
+ /* \u2500\u2500 Hero Strip (Maestro: progress rings + summary chips) \u2500\u2500 */
67
+ .hero-strip{border:1px solid var(--bd);border-radius:10px;background:var(--bg-1);padding:20px;margin-bottom:20px}
68
+ .hero-top{display:flex;align-items:center;gap:24px;margin-bottom:16px;flex-wrap:wrap}
69
+ .hero-rings{display:flex;gap:20px;flex-shrink:0}
70
+ .ring-wrap{text-align:center}
71
+ .ring-wrap svg{display:block;margin:0 auto 4px}
72
+ .ring-label{font-size:9px;font-weight:600;text-transform:uppercase;letter-spacing:.06em;color:var(--fg-2)}
73
+ .hero-chips{display:flex;gap:8px;flex:1;flex-wrap:wrap}
74
+ .summary-chip{flex:1;text-align:center;padding:12px 4px;border-radius:8px;border:1px solid var(--bd-s);min-width:70px;cursor:pointer;transition:border-color .15s}
75
+ .summary-chip:hover{border-color:var(--bd-m)}
76
+ .summary-chip.active{border-color:#03b79c}
77
+ .summary-chip .s-count{font-size:22px;font-weight:800;line-height:1}
78
+ .summary-chip .s-label{font-size:9px;font-weight:600;text-transform:uppercase;letter-spacing:.06em;margin-top:4px;color:var(--fg-2)}
79
+ .s-total{background:rgba(99,102,241,0.06)}.s-total .s-count{color:#818cf8}
80
+ .s-passed{background:rgba(34,197,94,0.06)}.s-passed .s-count{color:#22c55e}
81
+ .s-failed{background:rgba(239,68,68,0.06)}.s-failed .s-count{color:#ef4444}
82
+ .s-flaky{background:rgba(245,158,11,0.06)}.s-flaky .s-count{color:#f59e0b}
83
+ .s-skipped{background:rgba(107,114,128,0.06)}.s-skipped .s-count{color:#6b7280}
84
+ .s-ai{background:rgba(168,85,247,0.06)}.s-ai .s-count{color:#a855f7}
85
+ .hero-footer{display:flex;gap:14px;font-size:11px;color:var(--fg-1);flex-wrap:wrap;padding-top:12px;border-top:1px solid var(--bd-s)}
86
+
87
+ /* \u2500\u2500 Execution Waterfall \u2500\u2500 */
88
+ .waterfall-card{border:1px solid var(--bd);border-radius:10px;background:var(--bg-1);padding:16px;margin-bottom:20px}
89
+ .waterfall-title{font-size:11px;font-weight:600;color:var(--fg-2);text-transform:uppercase;letter-spacing:.06em;margin-bottom:10px}
90
+ .waterfall-row{display:flex;align-items:center;gap:8px;padding:3px 0;font-size:11px;cursor:pointer;transition:background .1s;border-radius:4px;padding:3px 6px;margin:0 -6px}
91
+ .waterfall-row:hover{background:var(--hvr)}
92
+ .waterfall-name{width:140px;flex-shrink:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;color:var(--fg-1);font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:10px}
93
+ .waterfall-track{flex:1;height:14px;background:var(--bg-2);border-radius:3px;overflow:hidden;position:relative}
94
+ .waterfall-bar{height:100%;border-radius:3px;min-width:2px;transition:width .3s}
95
+ .wb-passed{background:rgba(34,197,94,0.35)}
96
+ .wb-failed{background:rgba(239,68,68,0.35)}
97
+ .wb-skipped{background:rgba(107,114,128,0.2)}
98
+ .waterfall-dur{width:56px;flex-shrink:0;text-align:right;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:10px;color:var(--fg-2)}
99
+
100
+ /* \u2500\u2500 Filter Bar (same pattern as Appium: search + status chips + filter icon) \u2500\u2500 */
101
+ .filter-bar{display:flex;align-items:center;gap:8px;margin-bottom:24px;flex-wrap:wrap}
102
+ .filter-chips{display:flex;gap:5px;flex-wrap:wrap}
103
+ .filter-chip{font-size:12px;font-weight:500;padding:5px 14px;border-radius:9999px;border:1px solid var(--bd-m);background:transparent;color:var(--fg-1);cursor:pointer;transition:all .15s;font-family:inherit;white-space:nowrap}
104
+ .filter-chip:hover{border-color:#03b79c;color:var(--fg)}
105
+ .filter-chip.active{background:rgba(3,183,156,0.12);border-color:#03b79c;color:#2dd4a8}
106
+ .filter-chip--dimmed{opacity:.45}
107
+ .chip-count{font-weight:700;margin-left:3px}
108
+ .filter-clear{font-size:11px;font-weight:600;color:#03b79c;background:none;border:none;cursor:pointer;padding:4px 8px;border-radius:6px;font-family:inherit;transition:background .15s}
109
+ .filter-clear:hover{background:rgba(3,183,156,0.1)}
110
+ .filter-indicator{font-size:11px;color:var(--fg-2)}
111
+ .search-box{position:relative;width:260px;flex-shrink:1;min-width:160px;margin-left:0}
112
+ .search-input{width:100%;padding:6px 10px 6px 32px;border:1px solid var(--bd-m);border-radius:8px;background:var(--bg-1);color:var(--fg);font-size:12px;font-family:inherit;outline:none;transition:border-color .15s}
113
+ .search-input::placeholder{color:var(--fg-2)}
114
+ .search-input:focus{border-color:#03b79c}
115
+ .search-icon{position:absolute;left:10px;top:50%;transform:translateY(-50%);color:var(--fg-2);pointer-events:none}
116
+
117
+ /* \u2500\u2500 Filter Icon Button (opens tag drawer) \u2500\u2500 */
118
+ .filter-icon-btn{position:relative;display:flex;align-items:center;justify-content:center;width:34px;height:34px;border-radius:8px;border:1px solid var(--bd-m);background:transparent;color:var(--fg-2);cursor:pointer;transition:all .15s;flex-shrink:0;margin-left:auto}
119
+ .filter-icon-btn:hover{background:var(--hvr);color:var(--fg);border-color:var(--bd-l)}
120
+ .filter-icon-badge{position:absolute;top:5px;right:5px;width:7px;height:7px;border-radius:50%;background:#03b79c}
121
+
122
+ /* \u2500\u2500 Filter Drawer (right panel for tags + file filters) \u2500\u2500 */
123
+ .filter-drawer-backdrop{position:fixed;inset:0;background:var(--overlay-bg);z-index:100;opacity:0;pointer-events:none;transition:opacity .25s}
124
+ .filter-drawer-backdrop.open{opacity:1;pointer-events:auto}
125
+ .filter-drawer{position:fixed;top:0;right:0;bottom:0;width:320px;max-width:90vw;z-index:110;background:var(--bg-1);border-left:1px solid var(--bd);transform:translateX(100%);transition:transform .3s cubic-bezier(.4,0,.2,1);display:flex;flex-direction:column;overflow:hidden;box-shadow:-4px 0 24px var(--shadow-c)}
126
+ .filter-drawer.open{transform:translateX(0)}
127
+ .filter-drawer-bar{display:flex;align-items:center;justify-content:space-between;padding:14px 20px;border-bottom:1px solid var(--bd);background:linear-gradient(180deg,rgba(3,183,156,0.04) 0%,transparent 100%);flex-shrink:0}
128
+ .filter-drawer-title{font-size:12px;font-weight:600;color:var(--fg-1);text-transform:uppercase;letter-spacing:.06em}
129
+ .filter-drawer-close{width:32px;height:32px;border-radius:8px;border:1px solid var(--bd-m);background:transparent;color:var(--fg-1);font-size:18px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:background .15s,color .15s}
130
+ .filter-drawer-close:hover{background:var(--bd);color:var(--fg)}
131
+ .filter-drawer-body{flex:1;overflow-y:auto;padding:20px;scrollbar-width:thin;scrollbar-color:var(--scroll-thumb) transparent}
132
+ .filter-drawer-body::-webkit-scrollbar{width:5px}
133
+ .filter-drawer-body::-webkit-scrollbar-thumb{background:var(--bg-3);border-radius:3px}
134
+ .filter-drawer-section{margin-bottom:20px}
135
+ .filter-drawer-section-label{display:block;font-size:10px;font-weight:600;color:var(--fg-2);text-transform:uppercase;letter-spacing:.04em;margin-bottom:8px}
136
+ .filter-drawer-actions{display:flex;align-items:center;gap:12px;padding-top:16px;border-top:1px solid var(--bd)}
137
+
138
+ /* \u2500\u2500 Flow List (same pattern as Appium file-group/test-row) \u2500\u2500 */
139
+ .file-group{margin-bottom:16px}
140
+ .file-group-header{font-size:11px;font-weight:600;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;color:var(--fg-2);padding:0 4px 6px}
141
+ .file-group-card{border:1px solid var(--bd);border-radius:10px;overflow:hidden;background:var(--bg-1)}
142
+ .test-row{display:flex;align-items:center;gap:12px;padding:12px 18px;cursor:pointer;transition:background .1s;border-bottom:1px solid var(--bd-s)}
143
+ .test-row:last-child{border-bottom:none}
144
+ .test-row:hover{background:var(--hvr)}
145
+ .test-row.active{background:rgba(3,183,156,0.07)}
146
+ .status-indicator{width:10px;height:10px;border-radius:50%;flex-shrink:0}
147
+ .si-passed{background:#22c55e}.si-failed{background:#ef4444}.si-flaky{background:#f59e0b}.si-skipped{background:#6b7280}.si-timedout{background:#f97316}
148
+ .test-row-info{flex:1;min-width:0;display:flex;flex-direction:column;gap:3px}
149
+ .test-row-title{font-size:14px;font-weight:500;color:var(--fg);line-height:1.3;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
150
+ .test-row-badges{display:flex;gap:5px;align-items:center;flex-wrap:wrap}
151
+ .trb{font-size:10px;font-weight:600;padding:1px 7px;border-radius:4px}
152
+ .trb-app{background:rgba(59,130,246,0.12);color:#60a5fa}
153
+ .trb-tag{background:rgba(139,92,246,0.12);color:#a78bfa}
154
+ .trb-platform{background:rgba(3,183,156,0.12);color:#2dd4a8}
155
+ .test-row-dur{font-size:13px;font-weight:700;color:var(--fg-1);flex-shrink:0;white-space:nowrap}
156
+ .test-row-arrow{color:var(--fg-2);flex-shrink:0;transition:color .15s}
157
+ .test-row:hover .test-row-arrow{color:var(--fg-1)}
158
+ .test-row-error{font-size:11px;color:#ef4444;margin-top:2px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
159
+ .no-tests{text-align:center;padding:48px 20px;color:var(--fg-2);font-size:14px}
160
+
161
+ /* \u2500\u2500 Drawer (same as Appium) \u2500\u2500 */
162
+ .drawer-backdrop{position:fixed;inset:0;background:var(--overlay-bg);z-index:100;opacity:0;pointer-events:none;transition:opacity .25s}
163
+ .drawer-backdrop.open{opacity:1;pointer-events:auto}
164
+ .drawer{position:fixed;top:0;right:0;bottom:0;width:50vw;max-width:50vw;z-index:110;background:var(--bg-1);border-left:1px solid var(--bd);transform:translateX(100%);transition:transform .3s cubic-bezier(.4,0,.2,1);display:flex;flex-direction:column;overflow:hidden;box-shadow:-8px 0 40px var(--shadow-c)}
165
+ .drawer.open{transform:translateX(0)}
166
+ .drawer-bar{display:flex;align-items:center;justify-content:space-between;padding:14px 20px;border-bottom:1px solid var(--bd);background:linear-gradient(180deg,rgba(3,183,156,0.04) 0%,transparent 100%);flex-shrink:0}
167
+ .drawer-bar-title{font-size:12px;font-weight:600;color:var(--fg-1);text-transform:uppercase;letter-spacing:.06em}
168
+ .drawer-close{width:32px;height:32px;border-radius:8px;border:1px solid var(--bd-m);background:transparent;color:var(--fg-1);font-size:18px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:background .15s,color .15s}
169
+ .drawer-close:hover{background:var(--bd);color:var(--fg)}
170
+ .drawer-body{flex:1;overflow-y:auto;scrollbar-width:thin;scrollbar-color:var(--scroll-thumb) transparent}
171
+ .drawer-body::-webkit-scrollbar{width:5px}
172
+ .drawer-body::-webkit-scrollbar-track{background:transparent}
173
+ .drawer-body::-webkit-scrollbar-thumb{background:var(--bg-3);border-radius:3px}
174
+
175
+ /* \u2500\u2500 Drawer Tabs (same seg-ctrl as Appium) \u2500\u2500 */
176
+ .seg-ctrl{display:flex;gap:2px;padding:8px 12px 0;border-bottom:1px solid var(--bd);background:var(--bg-1);flex-wrap:wrap}
177
+ .seg-btn{display:flex;align-items:center;gap:5px;font-size:11px;padding:5px 12px;border-radius:6px 6px 0 0;border:none;background:none;color:var(--fg-2);cursor:pointer;font-weight:500;transition:all .15s;border-bottom:2px solid transparent;margin-bottom:-1px;font-family:inherit}
178
+ .seg-btn:hover{color:var(--fg-1);background:var(--hvr)}
179
+ .seg-btn.active{color:#03b79c;border-bottom-color:#03b79c;font-weight:600;background:var(--bg)}
180
+ .seg-pane{display:none;padding:20px 24px}
181
+ .seg-pane.active{display:block;animation:trFadeIn .15s ease-out}
182
+ @keyframes trFadeIn{from{opacity:0;transform:translateY(4px)}to{opacity:1;transform:none}}
183
+
184
+ /* \u2500\u2500 Overview Tab \u2500\u2500 */
185
+ .detail-header{margin-bottom:20px;padding-bottom:16px;border-bottom:1px solid var(--bd)}
186
+ .detail-top{display:flex;align-items:flex-start;gap:12px;margin-bottom:10px}
187
+ .detail-status-badge{font-size:11px;font-weight:700;padding:4px 14px;border-radius:9999px;text-transform:uppercase;letter-spacing:.04em;flex-shrink:0;margin-top:2px}
188
+ .dbg-passed{background:rgba(34,197,94,0.1);color:#22c55e;border:1px solid rgba(34,197,94,0.2)}
189
+ .dbg-failed{background:rgba(239,68,68,0.1);color:#ef4444;border:1px solid rgba(239,68,68,0.2)}
190
+ .dbg-skipped{background:rgba(107,114,128,0.1);color:#6b7280;border:1px solid rgba(107,114,128,0.2)}
191
+ .dbg-flaky{background:rgba(245,158,11,0.1);color:#f59e0b;border:1px solid rgba(245,158,11,0.2)}
192
+ .dbg-timedout{background:rgba(249,115,22,0.1);color:#f97316;border:1px solid rgba(249,115,22,0.2)}
193
+ .detail-title-section{flex:1;min-width:0}
194
+ .detail-title{font-size:16px;font-weight:700;color:var(--fg);line-height:1.35;margin-bottom:6px}
195
+ .detail-meta{display:flex;flex-wrap:wrap;gap:5px;align-items:center}
196
+ .dm-badge{font-size:10px;font-weight:600;padding:2px 8px;border-radius:4px}
197
+ .dm-app{background:rgba(59,130,246,0.12);color:#60a5fa}
198
+ .dm-tag{background:rgba(139,92,246,0.12);color:#a78bfa}
199
+ .dm-platform{background:rgba(3,183,156,0.12);color:#2dd4a8}
200
+ .detail-duration{font-size:24px;font-weight:800;color:var(--fg);flex-shrink:0;white-space:nowrap}
201
+ .detail-props{display:grid;grid-template-columns:auto 1fr;gap:4px 16px;font-size:12px;margin-top:12px}
202
+ .detail-props-k{color:var(--fg-2);font-weight:600;text-transform:uppercase;letter-spacing:.04em;font-size:10px}
203
+ .detail-props-v{color:var(--fg);font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:11px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
204
+
205
+ /* \u2500\u2500 Failure Panel (same as Appium) \u2500\u2500 */
206
+ .failure-panel{margin-bottom:20px;border:1px solid rgba(239,68,68,0.2);border-radius:10px;overflow:hidden;background:var(--bg-2)}
207
+ .failure-panel-header{display:flex;align-items:center;gap:8px;padding:10px 14px;background:rgba(239,68,68,0.08);font-size:12px;font-weight:600;color:#ef4444}
208
+ .failure-message{padding:12px 14px;background:var(--bg-code);color:var(--fg-err);font-size:12px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;white-space:pre-wrap;word-break:break-word;line-height:1.6;position:relative}
209
+ .copy-btn{position:absolute;top:8px;right:8px;font-size:10px;padding:3px 10px;border-radius:4px;border:1px solid var(--bd-m);background:var(--bg-3);color:var(--fg-1);cursor:pointer;font-family:inherit;transition:all .15s}
210
+ .copy-btn:hover{background:var(--bg-2);color:var(--fg)}
211
+
212
+ /* \u2500\u2500 Fix Hint (Maestro-specific, teal accent) \u2500\u2500 */
213
+ .fix-hint{margin-bottom:20px;border:1px solid rgba(3,183,156,0.2);border-radius:10px;overflow:hidden;background:var(--bg-2)}
214
+ .fix-hint-header{display:flex;align-items:center;gap:8px;padding:10px 14px;background:rgba(3,183,156,0.06);font-size:12px;font-weight:600;color:#2dd4a8}
215
+ .fix-hint-body{padding:12px 14px;font-size:12px;color:var(--fg-1);line-height:1.7}
216
+ .fix-hint-body ul{margin:4px 0 0 16px;list-style:disc}
217
+ .fix-hint-body li{margin:2px 0}
218
+ .fix-hint-body code{font-size:11px;padding:1px 5px;background:var(--bg-code);border-radius:3px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;color:var(--fg-code)}
219
+ .fix-hint-section{margin-top:10px;padding-top:8px;border-top:1px solid var(--bd-s)}
220
+ .fix-hint-section-title{font-size:10px;font-weight:600;color:var(--fg-2);text-transform:uppercase;letter-spacing:.04em;margin-bottom:4px}
221
+ .fix-code{margin:6px 0;padding:8px 12px;background:var(--bg-code);border:1px solid var(--bd);border-radius:6px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:11px;color:var(--fg-code);white-space:pre-wrap;line-height:1.5}
222
+
223
+ /* \u2500\u2500 Subflow Graph \u2500\u2500 */
224
+ .subflow-tree{margin-top:16px;padding:12px;background:var(--bg-2);border:1px solid var(--bd-s);border-radius:8px;font-size:11px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;color:var(--fg-1)}
225
+ .subflow-tree .sf-node{padding:2px 0}
226
+ .subflow-tree .sf-child{padding-left:20px;border-left:1px solid var(--bd-m)}
227
+
228
+ /* \u2500\u2500 Steps Tab \u2500\u2500 */
229
+ .steps-header{font-size:11px;font-weight:700;color:var(--fg-1);text-transform:uppercase;letter-spacing:.06em;margin-bottom:12px;display:flex;align-items:center;gap:8px}
230
+ .steps-stat{font-weight:400;color:var(--fg-2);font-size:10px}
231
+ .steps-waterfall{margin-bottom:16px;border:1px solid var(--bd);border-radius:8px;background:var(--bg-2);padding:8px 12px;overflow-x:auto}
232
+ .sw-row{display:flex;align-items:center;gap:6px;padding:2px 0;font-size:10px}
233
+ .sw-name{width:120px;flex-shrink:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;color:var(--fg-2);font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace}
234
+ .sw-track{flex:1;height:10px;background:var(--bg-3);border-radius:2px;overflow:hidden}
235
+ .sw-bar{height:100%;border-radius:2px;min-width:1px}
236
+ .sw-bar-green{background:#22c55e}.sw-bar-amber{background:#f59e0b}.sw-bar-red{background:#ef4444}.sw-bar-grey{background:#6b7280}
237
+ .sw-dur{width:52px;flex-shrink:0;text-align:right;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;color:var(--fg-2)}
238
+ .cmd-step{display:flex;align-items:center;gap:8px;padding:6px 12px;border-radius:6px;font-size:12px;transition:background .1s;border-left:3px solid transparent}
239
+ .cmd-step:hover{background:var(--hvr)}
240
+ .cmd-step-failed{background:rgba(239,68,68,0.04);border-left-color:#ef4444}
241
+ .cmd-step-failed:hover{background:rgba(239,68,68,0.08)}
242
+ .cmd-step-slow{border-left-color:#f59e0b;background:rgba(245,158,11,0.03)}
243
+ .cmd-step-slow:hover{background:rgba(245,158,11,0.07)}
244
+ .cmd-step-fast{border-left-color:#22c55e}
245
+ .cmd-dot{width:8px;height:8px;border-radius:50%;flex-shrink:0}
246
+ .cmd-dot-pass{background:#22c55e}.cmd-dot-fail{background:#ef4444}
247
+ .cmd-badge{font-size:9px;font-weight:600;padding:2px 7px;border-radius:4px;text-transform:uppercase;letter-spacing:.03em;flex-shrink:0;white-space:nowrap}
248
+ .cmd-badge-ui{background:rgba(59,130,246,0.12);color:#60a5fa}
249
+ .cmd-badge-assert{background:rgba(34,197,94,0.12);color:#22c55e}
250
+ .cmd-badge-nav{background:rgba(168,85,247,0.12);color:#a78bfa}
251
+ .cmd-badge-device{background:rgba(249,115,22,0.12);color:#fb923c}
252
+ .cmd-badge-ai{background:rgba(236,72,153,0.12);color:#f472b6}
253
+ .cmd-badge-cmd{background:var(--bg-3);color:var(--fg-2)}
254
+ .cmd-badge-slow{background:rgba(245,158,11,0.15);color:#f59e0b;margin-left:auto}
255
+ .cmd-title{flex:1;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;color:var(--fg);font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:11px}
256
+ .cmd-dur{font-size:10px;color:var(--fg-2);flex-shrink:0;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace}
257
+ .cmd-error{font-size:11px;color:#ef4444;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;white-space:pre-wrap;word-break:break-word;line-height:1.5;padding:4px 12px 8px 31px}
258
+ .cmd-connector{width:2px;height:4px;background:var(--bd-m);margin-left:15px}
259
+ .steps-legend{display:flex;flex-wrap:wrap;gap:10px;margin-top:16px;padding-top:12px;border-top:1px solid var(--bd);font-size:10px;color:var(--fg-2)}
260
+ .steps-legend span{display:inline-flex;align-items:center;gap:4px}
261
+
262
+ /* \u2500\u2500 Assertions Tab \u2500\u2500 */
263
+ .assert-header{display:flex;align-items:center;gap:16px;margin-bottom:16px;flex-wrap:wrap}
264
+ .assert-ring{flex-shrink:0}
265
+ .assert-stats{font-size:12px;color:var(--fg-1);line-height:1.6}
266
+ .assert-stats strong{font-weight:700}
267
+ .assert-bar-wrap{flex:1;min-width:100px}
268
+ .assert-bar{height:6px;background:var(--bg-3);border-radius:3px;overflow:hidden}
269
+ .assert-bar-fill{height:100%;background:#22c55e;border-radius:3px;transition:width .3s}
270
+ .assert-pct{font-size:11px;font-weight:700;color:var(--fg-1);margin-top:2px;text-align:right}
271
+ .assert-table{width:100%;border-collapse:collapse;font-size:12px}
272
+ .assert-table th{text-align:left;padding:6px 10px;font-size:10px;font-weight:600;color:var(--fg-2);text-transform:uppercase;letter-spacing:.04em;border-bottom:1px solid var(--bd)}
273
+ .assert-table td{padding:6px 10px;border-bottom:1px solid var(--bd-s);vertical-align:top}
274
+ .assert-table tr:hover{background:var(--hvr)}
275
+ .assert-result{font-weight:700;font-size:13px}
276
+ .ar-pass{color:#22c55e}.ar-fail{color:#ef4444}
277
+ .assert-error-row td{padding:2px 10px 8px 40px;border-bottom:1px solid var(--bd-s);color:#ef4444;font-size:11px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace}
278
+
279
+ /* \u2500\u2500 Screenshots Tab \u2500\u2500 */
280
+ .screenshot-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(180px,1fr));gap:12px}
281
+ .screenshot-card{border:1px solid var(--bd);border-radius:8px;overflow:hidden;background:var(--bg-2);cursor:pointer;transition:border-color .15s}
282
+ .screenshot-card:hover{border-color:#03b79c}
283
+ .screenshot-card img{width:100%;display:block;object-fit:contain}
284
+ .screenshot-card-label{padding:6px 10px;font-size:10px;font-weight:600;color:var(--fg-2);border-top:1px solid var(--bd-s)}
285
+ .screenshot-card.failure-shot{border-color:rgba(239,68,68,0.3)}
286
+ .screenshot-card.failure-shot .screenshot-card-label{background:rgba(239,68,68,0.06);color:#ef4444}
287
+ .screenshot-empty{padding:24px 14px;text-align:center;font-size:12px;color:var(--fg-2);font-style:italic;border:1px dashed var(--bd-m);border-radius:10px}
288
+
289
+ /* \u2500\u2500 AI Defects Tab \u2500\u2500 */
290
+ .ai-card{border:1px solid var(--bd);border-radius:10px;overflow:hidden;background:var(--bg-2);margin-bottom:12px;border-left:3px solid transparent}
291
+ .ai-card-critical{border-left-color:rgba(239,68,68,0.5)}
292
+ .ai-card-warning{border-left-color:rgba(245,158,11,0.5)}
293
+ .ai-card-info{border-left-color:rgba(59,130,246,0.5)}
294
+ .ai-card-header{display:flex;align-items:center;gap:8px;padding:10px 14px;border-bottom:1px solid var(--bd-s)}
295
+ .ai-severity{font-size:10px;font-weight:700;padding:2px 10px;border-radius:9999px;text-transform:uppercase;letter-spacing:.04em}
296
+ .ai-sev-critical{background:rgba(239,68,68,0.12);color:#ef4444;border:1px solid rgba(239,68,68,0.2)}
297
+ .ai-sev-warning{background:rgba(245,158,11,0.12);color:#f59e0b;border:1px solid rgba(245,158,11,0.2)}
298
+ .ai-sev-info{background:rgba(59,130,246,0.12);color:#60a5fa;border:1px solid rgba(59,130,246,0.2)}
299
+ .ai-type{font-size:11px;font-weight:600;color:var(--fg-1);text-transform:capitalize}
300
+ .ai-desc{padding:10px 14px;font-size:12px;color:var(--fg);line-height:1.6}
301
+ .ai-fix{padding:10px 14px;border-top:1px solid var(--bd-s);background:rgba(3,183,156,0.03)}
302
+ .ai-fix-label{font-size:10px;font-weight:700;color:#2dd4a8;text-transform:uppercase;letter-spacing:.06em;margin-bottom:6px}
303
+ .ai-fix-body{font-size:12px;color:var(--fg-1);line-height:1.7}
304
+ .ai-fix-body ul{margin:0 0 0 16px;list-style:disc}
305
+ .ai-fix-body li{margin:2px 0}
306
+ .ai-fix-code{margin:6px 0;padding:8px 12px;background:var(--bg-code);border:1px solid var(--bd);border-radius:6px;font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:11px;color:var(--fg-code);white-space:pre-wrap;line-height:1.5}
307
+ .ai-fix-platform{font-size:10px;font-weight:600;color:var(--fg-2);margin:8px 0 2px;text-transform:uppercase;letter-spacing:.04em}
308
+ .ai-empty{text-align:center;padding:32px;color:#22c55e;font-weight:600;font-size:14px}
309
+
310
+ /* \u2500\u2500 Lightbox (same as Appium) \u2500\u2500 */
311
+ .lightbox-overlay{position:fixed;inset:0;background:var(--lb-bg);z-index:1000;display:flex;align-items:center;justify-content:center;cursor:zoom-out;animation:trFadeIn .15s}
312
+ .lightbox-overlay img{max-width:92vw;max-height:92vh;border-radius:8px;box-shadow:0 8px 40px var(--lb-shadow)}
313
+ .lightbox-close{position:fixed;top:16px;right:16px;z-index:1001;width:40px;height:40px;border-radius:50%;border:none;background:var(--lb-btn);color:#fff;font-size:22px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:background .15s;backdrop-filter:blur(8px)}
314
+ .lightbox-close:hover{background:var(--lb-btn-h)}
315
+ .lightbox-nav{position:fixed;top:50%;z-index:1001;width:40px;height:40px;border-radius:50%;border:none;background:var(--lb-btn);color:#fff;font-size:18px;cursor:pointer;display:flex;align-items:center;justify-content:center;transition:background .15s;transform:translateY(-50%);backdrop-filter:blur(8px)}
316
+ .lightbox-nav:hover{background:var(--lb-btn-h)}
317
+ .lightbox-prev{left:16px}
318
+ .lightbox-next{right:16px}
319
+ .lightbox-caption{position:fixed;bottom:20px;left:50%;transform:translateX(-50%);z-index:1001;font-size:12px;color:#fff;background:rgba(0,0,0,.6);padding:4px 16px;border-radius:8px;backdrop-filter:blur(8px)}
320
+
321
+ /* \u2500\u2500 Footer \u2500\u2500 */
322
+ .report-footer{text-align:center;padding:24px;font-size:12px;color:var(--fg-2)}
323
+ .report-footer a{color:#03b79c;text-decoration:none}
324
+ .report-footer a:hover{text-decoration:underline}
325
+
326
+ /* \u2500\u2500 Loading (same as Appium) \u2500\u2500 */
327
+ .loading-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background:var(--bg);z-index:9999;display:flex;flex-direction:column;align-items:center;justify-content:center;gap:16px}
328
+ .loading-spinner{width:36px;height:36px;border:3px solid var(--bd-l);border-top-color:var(--fg-1);border-radius:50%;animation:spin .8s linear infinite}
329
+ .loading-text{font:13px/1 inherit;color:var(--fg-1)}
330
+ @keyframes spin{to{transform:rotate(360deg)}}
331
+
332
+ /* \u2500\u2500 Scrollbar (same as Appium) \u2500\u2500 */
333
+ *{scrollbar-width:thin;scrollbar-color:var(--scroll-thumb) transparent}
334
+ ::-webkit-scrollbar{width:6px;height:6px}
335
+ ::-webkit-scrollbar-track{background:transparent}
336
+ ::-webkit-scrollbar-thumb{background:var(--scroll-thumb);border-radius:3px}
337
+ ::-webkit-scrollbar-thumb:hover{background:var(--fg-2)}
338
+ ::-webkit-scrollbar-corner{background:transparent}
339
+
340
+ /* \u2500\u2500 Responsive (same breakpoints as Appium) \u2500\u2500 */
341
+ @media(max-width:1100px){.drawer{width:60vw;max-width:60vw}}
342
+ @media(max-width:800px){.drawer{width:85vw;max-width:85vw}}
343
+ @media(max-width:640px){
344
+ .wrap{padding:16px 12px 48px}
345
+ .top-bar{flex-direction:column;align-items:flex-start;gap:8px}
346
+ .top-bar-actions{margin-left:0}
347
+ .hero-top{flex-direction:column;align-items:stretch}
348
+ .hero-rings{justify-content:center}
349
+ .hero-chips{flex-wrap:wrap}
350
+ .summary-chip{min-width:60px}
351
+ .filter-bar{flex-direction:column;align-items:stretch}
352
+ .search-box{width:100%;margin-left:0}
353
+ .drawer{width:100%;max-width:100%}
354
+ .filter-drawer{width:100%;max-width:100%}
355
+ .cta-btn{font-size:10px;padding:5px 10px;order:10}
356
+ .detail-title{font-size:14px}
357
+ .detail-duration{font-size:20px}
358
+ .waterfall-name{width:80px}
359
+ }
360
+
361
+ /* \u2500\u2500 Print (same as Appium) \u2500\u2500 */
362
+ @media print{
363
+ body{background:#fff;color:#000}
364
+ .drawer-backdrop,.drawer,.lightbox-overlay,.lightbox-close,.lightbox-nav,.lightbox-caption{display:none}
365
+ .test-row-arrow{display:none}
366
+ .summary-chip,.filter-chip,.status-indicator,.detail-status-badge,.dm-badge,.trb,.cmd-badge,.ai-severity{print-color-adjust:exact;-webkit-print-color-adjust:exact}
367
+ }
368
+ `;var mt=`<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 196 247" fill="none">
369
+ <path fill-rule="evenodd" clip-rule="evenodd" d="M4.75 1.08009C2.034 2.66209 -0.130999 7.63009 0.610001 10.5821C0.969001 12.0121 3.021 15.2131 5.17 17.6971C14.375 28.3321 18 39.7791 18 58.2101V71.0001H12.434C1.16501 71.0001 0 73.4641 0 97.3051C0 106.858 0.298003 128.922 0.662003 146.337L1.323 178H33.606C65.786 178 65.897 178.007 68.544 180.284C72.228 183.453 72.244 189.21 68.579 192.877L65.957 195.5L33.479 195.801L1 196.103V204.551V213H24.577H48.154L51.077 215.923C55.007 219.853 55.007 224.147 51.077 228.077L48.154 231H26.469H4.783L5.41901 233.534C5.76901 234.927 7.143 238.527 8.472 241.534L10.89 247H40.945H71L71.006 241.75C71.017 230.748 76.027 221.606 84.697 216.767C97.854 209.424 114.086 213.895 121.323 226.857C123.659 231.041 124.418 233.833 124.789 239.607L125.263 247H155.187H185.11L187.528 241.534C188.857 238.527 190.231 234.927 190.581 233.534L191.217 231H169.531H147.846L144.923 228.077C142.928 226.082 142 224.152 142 222C142 219.848 142.928 217.918 144.923 215.923L147.846 213H171.423H195V204.551V196.103L162.521 195.801L130.043 195.5L127.421 192.877C123.991 189.445 123.835 183.869 127.074 180.421L129.349 178H162.013H194.677L195.338 146.337C195.702 128.922 196 106.858 196 97.3051C196 73.4641 194.835 71.0001 183.566 71.0001H178V58.2101C178 39.6501 181.397 28.7731 190.538 18.0651C195.631 12.0971 196.572 9.00809 194.511 5.02109C192.672 1.46509 190.197 9.12233e-05 186.028 9.12233e-05C179.761 9.12233e-05 168.713 14.8831 163.388 30.5001C160.975 37.5771 160.608 40.3751 160.213 54.7501L159.765 71.0001H150.732H141.7L142.286 53.2501C142.904 34.5511 144.727 24.3761 148.938 16.1211C151.823 10.4671 151.628 5.90109 148.364 2.63609C145 -0.726907 140.105 -0.887909 136.596 2.25009C133.481 5.03609 128.686 17.0811 126.507 27.5921C125.569 32.1191 124.617 43.0901 124.28 53.2501L123.69 71.0001H115.345H107V38.4231V5.84609L104.077 2.92309C102.082 0.928088 100.152 9.12233e-05 98 9.12233e-05C95.848 9.12233e-05 93.918 0.928088 91.923 2.92309L89 5.84609V38.4231V71.0001H80.655H72.31L71.72 53.2501C71.383 43.0901 70.431 32.1191 69.493 27.5921C67.314 17.0811 62.519 5.03609 59.404 2.25009C55.998 -0.795909 51.059 -0.710905 47.646 2.45209C44.221 5.62609 44.191 9.92109 47.539 17.4911C51.71 26.9241 53.007 34.4791 53.676 53.2501L54.31 71.0001H45.272H36.235L35.787 54.7501C35.392 40.3751 35.025 37.5771 32.612 30.5001C27.194 14.6091 16.228 -0.02891 9.787 0.03009C7.979 0.04709 5.712 0.519093 4.75 1.08009ZM42.687 108.974C33.431 112.591 20.036 125.024 18.408 131.512C17.476 135.223 20.677 140.453 28.253 147.599C37.495 156.319 44.191 159.471 53.5 159.485C59.317 159.494 61.645 158.953 67.274 156.289C77.634 151.385 88.987 139.161 88.996 132.9C89.004 127.304 76.787 114.707 66.745 109.956C59.16 106.368 50.285 106.006 42.687 108.974ZM129.255 109.904C119.151 114.768 106.996 127.33 107.004 132.9C107.013 139.108 118.562 151.475 128.939 156.389C134.338 158.945 136.744 159.496 142.521 159.498C152.526 159.501 160.369 155.502 169.771 145.605C180.444 134.368 180.278 130.975 168.486 119.388C160.043 111.094 152.727 107.595 143 107.201C136.364 106.933 134.78 107.244 129.255 109.904ZM48.5 125.922C46.3 126.969 43.152 128.945 41.505 130.314L38.511 132.802L40.504 135.005C41.601 136.216 44.434 138.342 46.8 139.728C52.577 143.114 57.36 142.466 64.009 137.395L68.978 133.606L66.756 131.24C60.944 125.054 54.357 123.135 48.5 125.922ZM138.386 125.063C136.674 125.571 133.375 127.677 131.057 129.743L126.841 133.5L131.901 137.343C138.65 142.468 143.407 143.124 149.2 139.728C151.566 138.342 154.351 136.269 155.389 135.122C157.684 132.587 156.742 131.097 150.58 127.511C145.438 124.519 142.329 123.895 138.386 125.063ZM91.923 162.923C89.043 165.804 89 166.008 89 176.973C89 184.805 89.435 188.941 90.47 190.941C92.356 194.589 96.918 196.273 101.03 194.84C105.82 193.17 107 189.638 107 176.973C107 166.008 106.957 165.804 104.077 162.923C102.082 160.928 100.152 160 98 160C95.848 160 93.918 160.928 91.923 162.923ZM94.5 232.155C91.026 234.055 90 236.229 90 241.691V247H98H106V242.082C106 235.732 105.37 234.242 101.928 232.463C98.591 230.737 97.197 230.679 94.5 232.155Z" fill="url(#paint0_linear_55_11)"/>
370
+ <defs><linearGradient id="paint0_linear_55_11" x1="98" y1="0.000244141" x2="98" y2="247" gradientUnits="userSpaceOnUse">
371
+ <stop stop-color="#03B79C"/><stop offset="0.504808" stop-color="#4EDAA4"/><stop offset="0.865285" stop-color="#84F3AA"/>
372
+ </linearGradient></defs></svg>`,ht=`<svg width="32" height="40" viewBox="0 0 196 247" fill="none" xmlns="http://www.w3.org/2000/svg">
373
+ <path fill-rule="evenodd" clip-rule="evenodd" d="M4.75 1.08009C2.034 2.66209 -0.130999 7.63009 0.610001 10.5821C0.969001 12.0121 3.021 15.2131 5.17 17.6971C14.375 28.3321 18 39.7791 18 58.2101V71.0001H12.434C1.16501 71.0001 0 73.4641 0 97.3051C0 106.858 0.298003 128.922 0.662003 146.337L1.323 178H33.606C65.786 178 65.897 178.007 68.544 180.284C72.228 183.453 72.244 189.21 68.579 192.877L65.957 195.5L33.479 195.801L1 196.103V204.551V213H24.577H48.154L51.077 215.923C55.007 219.853 55.007 224.147 51.077 228.077L48.154 231H26.469H4.783L5.41901 233.534C5.76901 234.927 7.143 238.527 8.472 241.534L10.89 247H40.945H71L71.006 241.75C71.017 230.748 76.027 221.606 84.697 216.767C97.854 209.424 114.086 213.895 121.323 226.857C123.659 231.041 124.418 233.833 124.789 239.607L125.263 247H155.187H185.11L187.528 241.534C188.857 238.527 190.231 234.927 190.581 233.534L191.217 231H169.531H147.846L144.923 228.077C142.928 226.082 142 224.152 142 222C142 219.848 142.928 217.918 144.923 215.923L147.846 213H171.423H195V204.551V196.103L162.521 195.801L130.043 195.5L127.421 192.877C123.991 189.445 123.835 183.869 127.074 180.421L129.349 178H162.013H194.677L195.338 146.337C195.702 128.922 196 106.858 196 97.3051C196 73.4641 194.835 71.0001 183.566 71.0001H178V58.2101C178 39.6501 181.397 28.7731 190.538 18.0651C195.631 12.0971 196.572 9.00809 194.511 5.02109C192.672 1.46509 190.197 9.12233e-05 186.028 9.12233e-05C179.761 9.12233e-05 168.713 14.8831 163.388 30.5001C160.975 37.5771 160.608 40.3751 160.213 54.7501L159.765 71.0001H150.732H141.7L142.286 53.2501C142.904 34.5511 144.727 24.3761 148.938 16.1211C151.823 10.4671 151.628 5.90109 148.364 2.63609C145 -0.726907 140.105 -0.887909 136.596 2.25009C133.481 5.03609 128.686 17.0811 126.507 27.5921C125.569 32.1191 124.617 43.0901 124.28 53.2501L123.69 71.0001H115.345H107V38.4231V5.84609L104.077 2.92309C102.082 0.928088 100.152 9.12233e-05 98 9.12233e-05C95.848 9.12233e-05 93.918 0.928088 91.923 2.92309L89 5.84609V38.4231V71.0001H80.655H72.31L71.72 53.2501C71.383 43.0901 70.431 32.1191 69.493 27.5921C67.314 17.0811 62.519 5.03609 59.404 2.25009C55.998 -0.795909 51.059 -0.710905 47.646 2.45209C44.221 5.62609 44.191 9.92109 47.539 17.4911C51.71 26.9241 53.007 34.4791 53.676 53.2501L54.31 71.0001H45.272H36.235L35.787 54.7501C35.392 40.3751 35.025 37.5771 32.612 30.5001C27.194 14.6091 16.228 -0.02891 9.787 0.03009C7.979 0.04709 5.712 0.519093 4.75 1.08009ZM42.687 108.974C33.431 112.591 20.036 125.024 18.408 131.512C17.476 135.223 20.677 140.453 28.253 147.599C37.495 156.319 44.191 159.471 53.5 159.485C59.317 159.494 61.645 158.953 67.274 156.289C77.634 151.385 88.987 139.161 88.996 132.9C89.004 127.304 76.787 114.707 66.745 109.956C59.16 106.368 50.285 106.006 42.687 108.974ZM129.255 109.904C119.151 114.768 106.996 127.33 107.004 132.9C107.013 139.108 118.562 151.475 128.939 156.389C134.338 158.945 136.744 159.496 142.521 159.498C152.526 159.501 160.369 155.502 169.771 145.605C180.444 134.368 180.278 130.975 168.486 119.388C160.043 111.094 152.727 107.595 143 107.201C136.364 106.933 134.78 107.244 129.255 109.904ZM48.5 125.922C46.3 126.969 43.152 128.945 41.505 130.314L38.511 132.802L40.504 135.005C41.601 136.216 44.434 138.342 46.8 139.728C52.577 143.114 57.36 142.466 64.009 137.395L68.978 133.606L66.756 131.24C60.944 125.054 54.357 123.135 48.5 125.922ZM138.386 125.063C136.674 125.571 133.375 127.677 131.057 129.743L126.841 133.5L131.901 137.343C138.65 142.468 143.407 143.124 149.2 139.728C151.566 138.342 154.351 136.269 155.389 135.122C157.684 132.587 156.742 131.097 150.58 127.511C145.438 124.519 142.329 123.895 138.386 125.063ZM91.923 162.923C89.043 165.804 89 166.008 89 176.973C89 184.805 89.435 188.941 90.47 190.941C92.356 194.589 96.918 196.273 101.03 194.84C105.82 193.17 107 189.638 107 176.973C107 166.008 106.957 165.804 104.077 162.923C102.082 160.928 100.152 160 98 160C95.848 160 93.918 160.928 91.923 162.923ZM94.5 232.155C91.026 234.055 90 236.229 90 241.691V247H98H106V242.082C106 235.732 105.37 234.242 101.928 232.463C98.591 230.737 97.197 230.679 94.5 232.155Z" fill="url(#paint0_linear_55_11)"/>
374
+ <defs><linearGradient id="paint0_linear_55_11" x1="98" y1="0.000244141" x2="98" y2="247" gradientUnits="userSpaceOnUse">
375
+ <stop stop-color="#03B79C"/><stop offset="0.504808" stop-color="#4EDAA4"/><stop offset="0.865285" stop-color="#84F3AA"/>
376
+ </linearGradient></defs></svg>`;var bt=`
377
+ /* \u2500\u2500 Utilities \u2500\u2500 */
378
+ function esc(s){if(!s)return '';return String(s).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/"/g,'&quot;').replace(/'/g,'&#39;')}
379
+ function fmtDur(ms){if(!ms||ms<=0)return '\\u2014';if(ms<1000)return Math.round(ms)+'ms';if(ms<60000)return (ms/1000).toFixed(1)+'s';var m=Math.floor(ms/60000),s=Math.round((ms%60000)/1000);return s>0?m+'m '+s+'s':m+'m'}
380
+ function fmtDate(iso){try{return new Date(iso).toLocaleString()}catch(e){return iso}}
381
+ function $(id){return document.getElementById(id)}
382
+
383
+ /* \u2500\u2500 Data Transform \u2500\u2500 */
384
+ var flows=[];
385
+ (report.timeline||[]).forEach(function(entry,ei){
386
+ if(!entry.tests)return;
387
+ entry.tests.forEach(function(t,ti){
388
+ flows.push({entry:entry,test:t,idx:ei*1000+ti});
389
+ });
390
+ });
391
+ flows.sort(function(a,b){
392
+ if(a.test.filePath!==b.test.filePath)return a.test.filePath<b.test.filePath?-1:1;
393
+ return (a.test.startedAt||'')<(b.test.startedAt||'')?-1:1;
394
+ });
395
+
396
+ var summary=report.summary||{};
397
+ var currentFilter='all';
398
+ var searchQuery='';
399
+ var openFlowIdx=-1;
400
+
401
+ function detectPlatform(){
402
+ for(var i=0;i<flows.length;i++){
403
+ var tags=flows[i].test.tags||[];
404
+ for(var j=0;j<tags.length;j++){
405
+ var t=tags[j].toLowerCase();
406
+ if(t==='android'||t==='ios'||t==='web')return t;
407
+ }
408
+ }
409
+ return '';
410
+ }
411
+
412
+ /* \u2500\u2500 SVG Progress Ring \u2500\u2500 */
413
+ function svgRing(pct,size,stroke,color){
414
+ var r=(size-stroke)/2;
415
+ var c=2*Math.PI*r;
416
+ var offset=c*(1-pct/100);
417
+ return '<svg width="'+size+'" height="'+size+'" viewBox="0 0 '+size+' '+size+'">'
418
+ +'<circle cx="'+(size/2)+'" cy="'+(size/2)+'" r="'+r+'" fill="none" stroke="var(--bd-l)" stroke-width="'+stroke+'"/>'
419
+ +'<circle cx="'+(size/2)+'" cy="'+(size/2)+'" r="'+r+'" fill="none" stroke="'+color+'" stroke-width="'+stroke+'"'
420
+ +' stroke-dasharray="'+c+'" stroke-dashoffset="'+offset+'" stroke-linecap="round" transform="rotate(-90 '+(size/2)+' '+(size/2)+')"/>'
421
+ +'<text x="'+(size/2)+'" y="'+(size/2+5)+'" text-anchor="middle" fill="var(--fg)" font-size="'+(size>60?14:12)+'" font-weight="700">'+Math.round(pct)+'%</text>'
422
+ +'</svg>';
423
+ }
424
+
425
+ /* \u2500\u2500 Render: Run Meta \u2500\u2500 */
426
+ function renderRunMeta(){
427
+ var el=$('run-meta');if(!el)return;
428
+ var h='';
429
+ h+='<span class="run-meta-item"><span class="run-meta-label">Run</span><span class="run-id-tag">'+esc((report.testRunId||'').substring(0,12))+'</span></span>';
430
+ h+='<span class="run-meta-item"><span class="run-meta-label">Duration</span>'+fmtDur(report.totalDuration)+'</span>';
431
+ h+='<span class="run-meta-item">'+fmtDate(report.startedAt)+'</span>';
432
+ if(report.ci){
433
+ h+='<span class="run-meta-item"><span class="ci-tag">'+esc(report.ci.provider)+'</span></span>';
434
+ if(report.ci.branch)h+='<span class="run-meta-item"><span class="run-meta-label">Branch</span>'+esc(report.ci.branch)+'</span>';
435
+ if(report.ci.commitSha)h+='<span class="run-meta-item"><span class="run-meta-label">Commit</span><span class="run-id-tag">'+esc(report.ci.commitSha.substring(0,8))+'</span></span>';
436
+ }
437
+ var plat=detectPlatform();
438
+ if(plat)h+='<span class="platform-tag">'+esc(plat)+'</span>';
439
+ el.innerHTML=h;
440
+ }
441
+
442
+ /* \u2500\u2500 Render: Hero Strip (rings + chips) \u2500\u2500 */
443
+ function renderHeroStrip(){
444
+ var el=$('hero-strip');if(!el)return;
445
+ var passRate=summary.total>0?Math.round((summary.passed/summary.total)*100):0;
446
+ var assertRate=summary.totalAssertions>0?Math.round((summary.passedAssertions/summary.totalAssertions)*100):0;
447
+
448
+ var h='<div class="hero-top">';
449
+ h+='<div class="hero-rings">';
450
+ h+='<div class="ring-wrap">'+svgRing(passRate,64,5,'#22c55e')+'<div class="ring-label">Pass Rate</div></div>';
451
+ h+='<div class="ring-wrap">'+svgRing(assertRate,64,5,'#a78bfa')+'<div class="ring-label">Assert Rate</div></div>';
452
+ h+='</div>';
453
+
454
+ h+='<div class="hero-chips">';
455
+ var chips=[
456
+ {cls:'s-total',n:summary.total||0,l:'Total',f:'all'},
457
+ {cls:'s-failed',n:summary.failed||0,l:'Failed',f:'failed'},
458
+ {cls:'s-passed',n:summary.passed||0,l:'Passed',f:'passed'},
459
+ {cls:'s-skipped',n:summary.skipped||0,l:'Skipped',f:'skipped'},
460
+ {cls:'s-ai',n:aiDefects.length,l:'AI Defects',f:null}
461
+ ];
462
+ chips.forEach(function(c){
463
+ h+='<div class="summary-chip '+c.cls+'" data-filter="'+(c.f||'')+'">'
464
+ +'<div class="s-count">'+c.n+'</div>'
465
+ +'<div class="s-label">'+c.l+'</div></div>';
466
+ });
467
+ h+='</div></div>';
468
+
469
+ h+='<div class="hero-footer">';
470
+ h+='<span>Total Duration: <strong>'+fmtDur(report.totalDuration)+'</strong></span>';
471
+ var plat=detectPlatform();
472
+ if(plat)h+='<span>Platform: <strong>'+esc(plat)+'</strong></span>';
473
+ if(summary.totalActionSteps)h+='<span>Steps: <strong>'+summary.totalActionSteps+'</strong></span>';
474
+ if(summary.totalAssertions)h+='<span>Assertions: <strong>'+summary.totalAssertions+'</strong></span>';
475
+ h+='</div>';
476
+
477
+ el.innerHTML=h;
478
+
479
+ el.querySelectorAll('.summary-chip[data-filter]').forEach(function(chip){
480
+ chip.addEventListener('click',function(){
481
+ var f=chip.dataset.filter;
482
+ if(f){currentFilter=f;renderFilterBar();renderFlowList()}
483
+ });
484
+ });
485
+ }
486
+
487
+ /* \u2500\u2500 Render: Execution Waterfall \u2500\u2500 */
488
+ function renderWaterfall(){
489
+ var el=$('waterfall');if(!el)return;
490
+ if(flows.length===0){el.style.display='none';return}
491
+ var maxDur=0;
492
+ flows.forEach(function(f){if(f.test.duration>maxDur)maxDur=f.test.duration});
493
+ if(maxDur===0)maxDur=1;
494
+
495
+ var h='<div class="waterfall-title">Execution Timeline</div>';
496
+ flows.forEach(function(f){
497
+ var t=f.test;
498
+ var pct=Math.max(1,Math.round((t.duration/maxDur)*100));
499
+ var cls=t.status==='passed'?'wb-passed':t.status==='failed'?'wb-failed':'wb-skipped';
500
+ h+='<div class="waterfall-row" data-wf-idx="'+f.idx+'">'
501
+ +'<div class="waterfall-name" title="'+esc(t.title)+'">'+esc(t.title)+'</div>'
502
+ +'<div class="waterfall-track"><div class="waterfall-bar '+cls+'" style="width:'+pct+'%"></div></div>'
503
+ +'<div class="waterfall-dur">'+fmtDur(t.duration)+'</div>'
504
+ +'</div>';
505
+ });
506
+ el.innerHTML=h;
507
+ }
508
+
509
+ /* \u2500\u2500 Tag filter state \u2500\u2500 */
510
+ var activeTagFilter='';
511
+
512
+ /* \u2500\u2500 Compute tag counts \u2500\u2500 */
513
+ function getTagCounts(){
514
+ var tags={};
515
+ flows.forEach(function(f){(f.test.tags||[]).forEach(function(t){
516
+ var low=t.toLowerCase();if(low!=='android'&&low!=='ios'&&low!=='web')tags[t]=(tags[t]||0)+1;
517
+ })});
518
+ return tags;
519
+ }
520
+
521
+ /* \u2500\u2500 Render: Filter Bar (clean: search + status pills + filter icon) \u2500\u2500 */
522
+ function renderFilterBar(){
523
+ var el=$('filter-bar');if(!el)return;
524
+ var h='';
525
+
526
+ h+='<div class="search-box"><svg class="search-icon" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><path d="M21 21l-4.35-4.35"/></svg>'
527
+ +'<input class="search-input" id="search-input" type="text" placeholder="Search flows..." value="'+esc(searchQuery)+'"></div>';
528
+
529
+ h+='<div class="filter-chips">';
530
+ var statuses=['passed','failed','skipped'];
531
+ var statusLabels={passed:'Passed',failed:'Failed',skipped:'Skipped'};
532
+ statuses.forEach(function(s){
533
+ var cnt=flows.filter(function(f){return f.test.status===s}).length;
534
+ var cls='filter-chip'+(currentFilter===s?' active':'')+(cnt===0?' filter-chip--dimmed':'');
535
+ h+='<button class="'+cls+'" data-status="'+s+'">'+statusLabels[s]+' <span class="chip-count">'+cnt+'</span></button>';
536
+ });
537
+ h+='</div>';
538
+
539
+ if(currentFilter!=='all'||activeTagFilter){
540
+ var parts=[];
541
+ if(currentFilter!=='all')parts.push(currentFilter);
542
+ if(activeTagFilter)parts.push('#'+activeTagFilter);
543
+ h+='<span class="filter-indicator">Filtering: '+esc(parts.join(' + '))+'</span>';
544
+ h+='<button class="filter-clear">Clear all</button>';
545
+ }
546
+
547
+ var tagCounts=getTagCounts();
548
+ var hasTagFilter=!!activeTagFilter;
549
+ h+='<button class="filter-icon-btn" title="Tags &amp; filters">'
550
+ +'<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M22 3H2l8 9.46V19l4 2v-8.54L22 3z"/></svg>'
551
+ +(hasTagFilter?'<span class="filter-icon-badge"></span>':'')
552
+ +'</button>';
553
+
554
+ el.innerHTML=h;
555
+
556
+ renderFilterDrawerBody(tagCounts);
557
+ }
558
+
559
+ /* \u2500\u2500 Render: Filter Drawer Body \u2500\u2500 */
560
+ function renderFilterDrawerBody(tagCounts){
561
+ var el=$('filter-drawer-body');if(!el)return;
562
+ var tags=Object.keys(tagCounts).sort();
563
+ if(tags.length===0){el.innerHTML='<div style="font-size:12px;color:var(--fg-2)">No tags found in flow metadata.</div>';return}
564
+
565
+ var h='<div class="filter-drawer-section">';
566
+ h+='<span class="filter-drawer-section-label">Tags</span>';
567
+ h+='<div class="filter-chips">';
568
+ tags.forEach(function(t){
569
+ var cnt=tagCounts[t];
570
+ var cls='filter-chip'+(activeTagFilter===t?' active':'');
571
+ h+='<button class="'+cls+'" data-tag="'+esc(t)+'">'+esc(t)+' <span class="chip-count">'+cnt+'</span></button>';
572
+ });
573
+ h+='</div></div>';
574
+
575
+ var specCounts={};
576
+ flows.forEach(function(f){
577
+ var key=f.test.filePath||f.entry.specFile||'';
578
+ if(key)specCounts[key]=(specCounts[key]||0)+1;
579
+ });
580
+ var specs=Object.keys(specCounts).sort();
581
+ if(specs.length>1){
582
+ h+='<div class="filter-drawer-section">';
583
+ h+='<span class="filter-drawer-section-label">Flow Files</span>';
584
+ h+='<div class="filter-chips">';
585
+ specs.forEach(function(sp){
586
+ var short=sp.split(/[\\/\\\\]/).pop()||sp;
587
+ h+='<button class="filter-chip" data-tag="'+esc(short)+'" title="'+esc(sp)+'">'+esc(short)+' <span class="chip-count">'+specCounts[sp]+'</span></button>';
588
+ });
589
+ h+='</div></div>';
590
+ }
591
+
592
+ if(activeTagFilter){
593
+ h+='<div class="filter-drawer-actions"><button class="filter-clear">Clear all filters</button></div>';
594
+ }
595
+
596
+ el.innerHTML=h;
597
+ }
598
+
599
+ /* \u2500\u2500 Render: Flow List \u2500\u2500 */
600
+ function getFilteredFlows(){
601
+ return flows.filter(function(f){
602
+ if(currentFilter!=='all'&&f.test.status!==currentFilter)return false;
603
+ if(activeTagFilter){
604
+ var tags=(f.test.tags||[]).map(function(t){return t.toLowerCase()});
605
+ if(tags.indexOf(activeTagFilter.toLowerCase())<0)return false;
606
+ }
607
+ if(searchQuery){
608
+ var q=searchQuery.toLowerCase();
609
+ var title=(f.test.title||'').toLowerCase();
610
+ var tagStr=(f.test.tags||[]).join(' ').toLowerCase();
611
+ var file=(f.test.filePath||'').toLowerCase();
612
+ if(title.indexOf(q)<0&&tagStr.indexOf(q)<0&&file.indexOf(q)<0)return false;
613
+ }
614
+ return true;
615
+ });
616
+ }
617
+
618
+ function renderFlowList(){
619
+ var el=$('flow-list');if(!el)return;
620
+ var filtered=getFilteredFlows();
621
+ if(filtered.length===0){el.innerHTML='<div class="no-tests">No flows match the current filter.</div>';return}
622
+
623
+ var groups={};
624
+ filtered.forEach(function(f){
625
+ var key=f.test.filePath||f.entry.specFile||'flows';
626
+ if(!groups[key])groups[key]=[];
627
+ groups[key].push(f);
628
+ });
629
+
630
+ var h='';
631
+ for(var key in groups){
632
+ if(!groups.hasOwnProperty(key))continue;
633
+ h+='<div class="file-group">';
634
+ h+='<div class="file-group-header">'+esc(key)+'</div>';
635
+ h+='<div class="file-group-card">';
636
+ groups[key].forEach(function(f){
637
+ var t=f.test;
638
+ var badges='';
639
+ if(f.entry.url&&f.entry.url!=='maestro-flow')badges+='<span class="trb trb-app">'+esc(f.entry.url)+'</span>';
640
+ (t.tags||[]).forEach(function(tag){
641
+ var low=tag.toLowerCase();
642
+ if(low==='android'||low==='ios'||low==='web')badges+='<span class="trb trb-platform">'+esc(tag)+'</span>';
643
+ else badges+='<span class="trb trb-tag">'+esc(tag)+'</span>';
644
+ });
645
+ var activeClass=f.idx===openFlowIdx?' active':'';
646
+ h+='<div class="test-row'+activeClass+'" data-flow-idx="'+f.idx+'">'
647
+ +'<div class="status-indicator si-'+esc(t.status)+'"></div>'
648
+ +'<div class="test-row-info">'
649
+ +'<div class="test-row-title">'+esc(t.title)+'</div>'
650
+ +'<div class="test-row-badges">'+badges+'</div>'
651
+ +(t.failure?'<div class="test-row-error">'+esc(t.failure.message||'')+'</div>':'')
652
+ +'</div>'
653
+ +'<div class="test-row-dur">'+fmtDur(t.duration)+'</div>'
654
+ +'<svg class="test-row-arrow" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="m9 18 6-6-6-6"/></svg>'
655
+ +'</div>';
656
+ });
657
+ h+='</div></div>';
658
+ }
659
+ el.innerHTML=h;
660
+ }
661
+ `;var yt=`
662
+ /* \u2500\u2500 Drawer Open/Close \u2500\u2500 */
663
+ function openDrawer(flowIdx){
664
+ openFlowIdx=flowIdx;
665
+ renderFlowList();
666
+ var flow=flows.find(function(f){return f.idx===flowIdx});
667
+ if(!flow)return;
668
+ $('drawer-backdrop').classList.add('open');
669
+ $('drawer').classList.add('open');
670
+ renderDrawerContent(flow);
671
+ }
672
+ function closeDrawer(){
673
+ openFlowIdx=-1;
674
+ renderFlowList();
675
+ $('drawer-backdrop').classList.remove('open');
676
+ $('drawer').classList.remove('open');
677
+ }
678
+
679
+ function renderDrawerContent(flow){
680
+ var tabs=['Overview','Steps','Assertions','Screenshots','AI Defects'];
681
+ var h='<div class="seg-ctrl">';
682
+ tabs.forEach(function(name,i){
683
+ h+='<button class="seg-btn'+(i===0?' active':'')+'" data-seg="tab'+i+'">'+name+'</button>';
684
+ });
685
+ h+='</div>';
686
+
687
+ h+='<div class="seg-pane active" data-seg-pane="tab0">'+renderOverviewTab(flow)+'</div>';
688
+ h+='<div class="seg-pane" data-seg-pane="tab1">'+renderStepsTab(flow)+'</div>';
689
+ h+='<div class="seg-pane" data-seg-pane="tab2">'+renderAssertionsTab(flow)+'</div>';
690
+ h+='<div class="seg-pane" data-seg-pane="tab3">'+renderScreenshotsTab()+'</div>';
691
+ h+='<div class="seg-pane" data-seg-pane="tab4">'+renderAiDefectsTab()+'</div>';
692
+
693
+ $('drawer-body').innerHTML=h;
694
+ }
695
+
696
+ /* \u2500\u2500 Overview Tab \u2500\u2500 */
697
+ function renderOverviewTab(flow){
698
+ var t=flow.test;
699
+ var h='<div class="detail-header"><div class="detail-top">';
700
+ h+='<span class="detail-status-badge dbg-'+esc(t.status)+'">'+esc(t.status)+'</span>';
701
+ h+='<div class="detail-title-section"><div class="detail-title">'+esc(t.title)+'</div>';
702
+ h+='<div class="detail-meta">';
703
+ if(flow.entry.url&&flow.entry.url!=='maestro-flow')h+='<span class="dm-badge dm-app">'+esc(flow.entry.url)+'</span>';
704
+ (t.tags||[]).forEach(function(tag){
705
+ var low=tag.toLowerCase();
706
+ if(low==='android'||low==='ios'||low==='web')h+='<span class="dm-badge dm-platform">'+esc(tag)+'</span>';
707
+ else h+='<span class="dm-badge dm-tag">'+esc(tag)+'</span>';
708
+ });
709
+ h+='</div></div>';
710
+ h+='<div class="detail-duration">'+fmtDur(t.duration)+'</div>';
711
+ h+='</div>';
712
+
713
+ h+='<div class="detail-props">';
714
+ h+='<span class="detail-props-k">Flow File</span><span class="detail-props-v">'+esc(t.filePath||flow.entry.specFile||'\\u2014')+'</span>';
715
+ h+='<span class="detail-props-k">App ID</span><span class="detail-props-v">'+esc(flow.entry.url||'\\u2014')+'</span>';
716
+ h+='<span class="detail-props-k">Suite</span><span class="detail-props-v">'+esc(t.suiteName||'\\u2014')+'</span>';
717
+ h+='<span class="detail-props-k">Started</span><span class="detail-props-v">'+esc(t.startedAt?fmtDate(t.startedAt):'\\u2014')+'</span>';
718
+ h+='<span class="detail-props-k">Test ID</span><span class="detail-props-v">'+esc(t.testId||'\\u2014')+'</span>';
719
+ h+='</div></div>';
720
+
721
+ if(t.failure){
722
+ h+='<div class="failure-panel">';
723
+ h+='<div class="failure-panel-header"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><path d="m15 9-6 6M9 9l6 6"/></svg> Failure Diagnostic</div>';
724
+ h+='<div class="failure-message">'+esc(t.failure.message||'Unknown error')+'<button class="copy-btn" data-copy="'+esc(t.failure.message||'')+'">Copy</button></div>';
725
+ h+='</div>';
726
+ h+=renderRootCauseAnalysis(t,flow);
727
+ }
728
+
729
+ if(t.tags&&t.tags.length>0){
730
+ var subflows=[];
731
+ (t.actions||[]).forEach(function(a){
732
+ if(a.title&&a.title.indexOf('runFlow')>=0)subflows.push(a.title.replace('runFlow \\u2192 ',''));
733
+ });
734
+ if(subflows.length>0){
735
+ h+='<div class="subflow-tree"><div class="sf-node">'+esc(t.filePath||t.title)+'</div>';
736
+ subflows.forEach(function(sf){h+='<div class="sf-child"><div class="sf-node">\\u2514\\u2500 '+esc(sf)+' (subflow)</div></div>'});
737
+ h+='</div>';
738
+ }
739
+ }
740
+
741
+ return h;
742
+ }
743
+
744
+ /* \u2500\u2500 Root Cause Analysis \u2500\u2500 */
745
+ function renderRootCauseAnalysis(t,flow){
746
+ var msg=(t.failure&&t.failure.message)||'';
747
+ var lower=msg.toLowerCase();
748
+ var actions=t.actions||[];
749
+ var failedSteps=actions.filter(function(a){return a.status==='failed'});
750
+
751
+ var analysis='';
752
+ if(lower.indexOf('not found')>=0||lower.indexOf('not visible')>=0||lower.indexOf('could not find')>=0){
753
+ var prevFailed=failedSteps.length>1;
754
+ analysis+='<p>The test attempted to interact with an element that was not present in the view hierarchy.';
755
+ if(prevFailed)analysis+=' The preceding step also failed, suggesting the target screen did not fully load.';
756
+ analysis+='</p>';
757
+ }else if(lower.indexOf('timeout')>=0||lower.indexOf('timed out')>=0){
758
+ analysis+='<p>The operation exceeded its timeout threshold. This typically indicates a slow network response, a heavy UI render, or the target element appearing after the timeout window.</p>';
759
+ }else if(lower.indexOf('assert')>=0){
760
+ analysis+='<p>An assertion did not match the expected state. This could indicate a UI regression, dynamic content change, or a race condition between the app and the test.</p>';
761
+ }else{
762
+ analysis+='<p>Review the step timeline to identify the exact failing command and its preceding context.</p>';
763
+ }
764
+
765
+ var h='<div class="fix-hint">';
766
+ h+='<div class="fix-hint-header"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><path d="M12 16v-4M12 8h.01"/></svg> Root Cause Analysis</div>';
767
+ h+='<div class="fix-hint-body">';
768
+ h+='<div class="fix-hint-section"><div class="fix-hint-section-title">What Went Wrong</div>'+analysis+'</div>';
769
+
770
+ h+='<div class="fix-hint-section"><div class="fix-hint-section-title">Recommended Actions</div><ul>';
771
+ if(lower.indexOf('not found')>=0||lower.indexOf('not visible')>=0){
772
+ h+='<li>Add <code>extendedWaitUntil</code> before the interaction:</li>';
773
+ h+='</ul><div class="fix-code">- extendedWaitUntil:\\n visible: "Pay Now"\\n timeout: 10000</div><ul>';
774
+ h+='<li>Verify the selector with <strong>Maestro Studio</strong> on the target device</li>';
775
+ h+='<li>Add an <code>assertVisible</code> guard before <code>tapOn</code></li>';
776
+ h+='<li>Check if the element requires scrolling into view first</li>';
777
+ h+='<li>Inspect network requests \\u2014 the backend API may be slow</li>';
778
+ }else if(lower.indexOf('timeout')>=0||lower.indexOf('timed out')>=0){
779
+ h+='<li>Increase timeout with <code>extendedWaitUntil</code> (default 5s may be too short)</li>';
780
+ h+='<li>Check for loading spinners or network requests blocking the UI</li>';
781
+ h+='<li>Run on a faster emulator or physical device to rule out performance</li>';
782
+ h+='<li>Add <code>waitForAnimationToEnd</code> before the assertion</li>';
783
+ }else if(lower.indexOf('assert')>=0){
784
+ h+='<li>Verify the expected text/element exists on the current screen</li>';
785
+ h+='<li>Check for dynamic content that changes between runs</li>';
786
+ h+='<li>Use <code>assertWithAI</code> for fuzzy visual assertions</li>';
787
+ h+='<li>Add <code>waitForAnimationToEnd</code> before <code>assertVisible</code></li>';
788
+ }else{
789
+ h+='<li>Review the step timeline for the exact failing command</li>';
790
+ h+='<li>Run the flow in <strong>Maestro Studio</strong> to debug interactively</li>';
791
+ h+='<li>Check device logs for crash traces or ANR</li>';
792
+ h+='<li>Try <code>clearState</code> before <code>launchApp</code></li>';
793
+ }
794
+ h+='</ul></div>';
795
+
796
+ h+='</div></div>';
797
+ return h;
798
+ }
799
+ `;var xt=`
800
+ function renderStepsTab(flow){
801
+ var actions=flow.test.actions||[];
802
+ if(actions.length===0)return '<div class="no-tests">No step data available for this flow.</div>';
803
+
804
+ var failCount=actions.filter(function(a){return a.status==='failed'}).length;
805
+ var slowCount=actions.filter(function(a){return a.duration>2000}).length;
806
+ var maxDur=0;
807
+ actions.forEach(function(a){if(a.duration>maxDur)maxDur=a.duration});
808
+
809
+ var h='<div class="steps-header">'+actions.length+' steps'
810
+ +'<span class="steps-stat">'+failCount+' failed</span>'
811
+ +(slowCount?'<span class="steps-stat">'+slowCount+' slow (&gt;2s)</span>':'')
812
+ +'<span class="steps-stat">slowest: '+fmtDur(maxDur)+'</span></div>';
813
+
814
+ /* Mini waterfall */
815
+ h+='<div class="steps-waterfall">';
816
+ var wfMax=maxDur||1;
817
+ actions.forEach(function(a){
818
+ var pct=Math.max(1,Math.round((a.duration/wfMax)*100));
819
+ var barCls=a.status==='failed'?'sw-bar-red':a.duration>2000?'sw-bar-amber':a.duration>500?'sw-bar-amber':'sw-bar-green';
820
+ var nameStr=a.title||'step';
821
+ if(nameStr.length>18)nameStr=nameStr.substring(0,18)+'\\u2026';
822
+ h+='<div class="sw-row">'
823
+ +'<div class="sw-name" title="'+esc(a.title)+'">'+esc(nameStr)+'</div>'
824
+ +'<div class="sw-track"><div class="sw-bar '+barCls+'" style="width:'+pct+'%"></div></div>'
825
+ +'<div class="sw-dur">'+fmtDur(a.duration)+'</div></div>';
826
+ });
827
+ h+='</div>';
828
+
829
+ /* Step list */
830
+ actions.forEach(function(a,i){
831
+ var isFailed=a.status==='failed';
832
+ var isSlow=a.duration>2000&&!isFailed;
833
+ var isFast=a.duration<=500&&!isFailed;
834
+ var rowCls='cmd-step';
835
+ if(isFailed)rowCls+=' cmd-step-failed';
836
+ else if(isSlow)rowCls+=' cmd-step-slow';
837
+ else if(isFast)rowCls+=' cmd-step-fast';
838
+ var dotCls='cmd-dot '+(isFailed?'cmd-dot-fail':'cmd-dot-pass');
839
+ h+='<div class="'+rowCls+'">';
840
+ h+='<div class="'+dotCls+'"></div>';
841
+ h+=getCmdBadge(a.category);
842
+ h+='<div class="cmd-title" title="'+esc(a.title)+'">'+esc(a.title)+'</div>';
843
+ if(isSlow||isFailed&&a.duration>2000)h+='<span class="cmd-badge cmd-badge-slow">SLOW</span>';
844
+ h+='<div class="cmd-dur">'+fmtDur(a.duration)+'</div>';
845
+ h+='</div>';
846
+ if(a.error)h+='<div class="cmd-error">'+esc(a.error)+'</div>';
847
+ if(i<actions.length-1)h+='<div class="cmd-connector"></div>';
848
+ });
849
+
850
+ h+='<div class="steps-legend">';
851
+ h+='<span><span class="cmd-badge cmd-badge-ui">UI</span> ui_action</span>';
852
+ h+='<span><span class="cmd-badge cmd-badge-assert">ASSERT</span> assertion</span>';
853
+ h+='<span><span class="cmd-badge cmd-badge-ai">AI</span> ai</span>';
854
+ h+='<span><span class="cmd-badge cmd-badge-cmd">CMD</span> custom_step</span>';
855
+ h+='<span><span class="cmd-badge cmd-badge-slow">SLOW</span> &gt; 2s</span>';
856
+ h+='</div>';
857
+ h+='<div class="steps-legend" style="margin-top:4px">';
858
+ h+='<span>Duration: <span style="color:#22c55e;font-weight:700">&lt;500ms</span></span>';
859
+ h+='<span style="color:#f59e0b;font-weight:700">500ms\\u20132s</span>';
860
+ h+='<span style="color:#ef4444;font-weight:700">&gt;2s</span>';
861
+ h+='</div>';
862
+ return h;
863
+ }
864
+
865
+ function getCmdBadge(cat){
866
+ var m={ui_action:['UI','cmd-badge-ui'],assertion:['ASSERT','cmd-badge-assert'],navigation:['NAV','cmd-badge-nav'],custom_step:['CMD','cmd-badge-cmd'],ai:['AI','cmd-badge-ai'],device:['DEV','cmd-badge-device']};
867
+ var pair=m[cat]||['CMD','cmd-badge-cmd'];
868
+ return '<span class="cmd-badge '+pair[1]+'">'+pair[0]+'</span>';
869
+ }
870
+ `;var vt=`
871
+ function renderAssertionsTab(flow){
872
+ var actions=flow.test.actions||[];
873
+ var asserts=actions.filter(function(a){return a.category==='assertion'});
874
+ if(asserts.length===0)return '<div class="no-tests">No assertions recorded for this flow.</div>';
875
+
876
+ var passCount=asserts.filter(function(a){return a.status==='passed'}).length;
877
+ var failCount=asserts.length-passCount;
878
+ var pct=Math.round((passCount/asserts.length)*100);
879
+
880
+ var h='<div class="assert-header">';
881
+ h+='<div class="assert-ring">'+svgRing(pct,56,4,'#22c55e')+'</div>';
882
+ h+='<div class="assert-stats"><strong>'+asserts.length+'</strong> assertions<br>';
883
+ h+='<span style="color:#22c55e;font-weight:700">'+passCount+' passed</span> &middot; ';
884
+ h+='<span style="color:#ef4444;font-weight:700">'+failCount+' failed</span></div>';
885
+ h+='<div class="assert-bar-wrap"><div class="assert-bar"><div class="assert-bar-fill" style="width:'+pct+'%"></div></div>';
886
+ h+='<div class="assert-pct">'+pct+'% pass rate</div></div>';
887
+ h+='</div>';
888
+
889
+ h+='<table class="assert-table"><thead><tr><th>Result</th><th>Command</th><th>Duration</th></tr></thead><tbody>';
890
+ asserts.forEach(function(a){
891
+ var icon=a.status==='passed'?'\\u2713':'\\u2717';
892
+ var cls=a.status==='passed'?'ar-pass':'ar-fail';
893
+ h+='<tr><td class="assert-result '+cls+'">'+icon+'</td>';
894
+ h+='<td style="font-family:ui-monospace,SFMono-Regular,Menlo,Consolas,monospace;font-size:11px">'+esc(a.title)+'</td>';
895
+ h+='<td class="cmd-dur">'+fmtDur(a.duration)+'</td></tr>';
896
+ if(a.error)h+='<tr class="assert-error-row"><td></td><td colspan="2">'+esc(a.error)+'</td></tr>';
897
+ });
898
+ h+='</tbody></table>';
899
+ return h;
900
+ }
901
+ `;var wt=`
902
+ function renderAiDefectsTab(){
903
+ if(aiDefects.length===0)return '<div class="ai-empty">\\u2714 No AI defects detected.</div>';
904
+ var h='<div style="font-size:12px;color:var(--fg-1);margin-bottom:16px">'+aiDefects.length+' defect'+(aiDefects.length>1?'s':'')+' found by Maestro AI analysis</div>';
905
+
906
+ aiDefects.forEach(function(d){
907
+ var sev=d.severity||'info';
908
+ var type=(d.type||'ui').toLowerCase();
909
+ h+='<div class="ai-card ai-card-'+sev+'">';
910
+ h+='<div class="ai-card-header"><span class="ai-severity ai-sev-'+sev+'">'+esc(sev)+'</span><span class="ai-type">'+esc(d.type||'UI')+'</span></div>';
911
+ h+='<div class="ai-desc">'+esc(d.description||'')+'</div>';
912
+ h+='<div class="ai-fix"><div class="ai-fix-label">Fix Playbook</div><div class="ai-fix-body">'+getFixPlaybook(type,d)+'</div></div>';
913
+ h+='</div>';
914
+ });
915
+ return h;
916
+ }
917
+
918
+ function getFixPlaybook(type,d){
919
+ if(type==='ui')return uiFixPlaybook(d);
920
+ if(type==='spelling')return spellingFixPlaybook(d);
921
+ if(type==='layout')return layoutFixPlaybook(d);
922
+ if(type==='i18n')return i18nFixPlaybook(d);
923
+ if(type==='accessibility')return a11yFixPlaybook(d);
924
+ return genericFixPlaybook(d);
925
+ }
926
+
927
+ function uiFixPlaybook(d){
928
+ return '<div class="ai-fix-platform">Android (XML)</div>'
929
+ +'<div class="ai-fix-code">android:layout_width="wrap_content"\\nandroid:minWidth="120dp"\\nandroid:paddingHorizontal="16dp"</div>'
930
+ +'<div class="ai-fix-platform">Jetpack Compose</div>'
931
+ +'<div class="ai-fix-code">Modifier\\n .widthIn(min = 120.dp)\\n .padding(horizontal = 16.dp)</div>'
932
+ +'<div class="ai-fix-platform">Verify</div>'
933
+ +'<div class="ai-fix-code">maestro test --device Pixel_4a ./flows</div>'
934
+ +'<ul><li>Test on smallest supported screen (360dp width)</li>'
935
+ +'<li>Run <code>assertNoDefectsWithAI</code> to catch regressions</li></ul>';
936
+ }
937
+
938
+ function spellingFixPlaybook(d){
939
+ var desc=d.description||'';
940
+ var match=desc.match(/"([^"]+)"/);
941
+ var typo=match?match[1]:'typo';
942
+ return '<div class="ai-fix-platform">Find &amp; Replace</div>'
943
+ +'<div class="ai-fix-code">grep -r "'+esc(typo)+'" --include="*.xml" --include="*.strings"\\ngrep -r "'+esc(typo)+'" --include="*.kt" --include="*.swift"</div>'
944
+ +'<ul><li>Android: <code>res/values/strings.xml</code></li>'
945
+ +'<li>iOS: <code>Localizable.strings</code></li>'
946
+ +'<li>Add a CI linting rule for common typos</li></ul>';
947
+ }
948
+
949
+ function layoutFixPlaybook(d){
950
+ return '<div class="ai-fix-platform">Android (XML)</div>'
951
+ +'<div class="ai-fix-code">android:maxLines="1"\\nandroid:ellipsize="end"</div>'
952
+ +'<div class="ai-fix-platform">iOS (Swift)</div>'
953
+ +'<div class="ai-fix-code">label.lineBreakMode = .byTruncatingTail\\nlabel.numberOfLines = 1</div>'
954
+ +'<ul><li>Use ConstraintLayout / Auto Layout to prevent overlap</li>'
955
+ +'<li>Test with long strings (20+ characters)</li></ul>';
956
+ }
957
+
958
+ function i18nFixPlaybook(d){
959
+ return '<ul><li>Audit translation files for missing keys</li>'
960
+ +'<li>Run extraction tool to find untranslated strings</li>'
961
+ +'<li>Verify all locales in CI with locale-specific flows</li></ul>'
962
+ +'<div class="ai-fix-platform">Example Flow</div>'
963
+ +'<div class="ai-fix-code">env:\\n LANG: fr_FR\\n---\\n- launchApp\\n- assertVisible: "Bienvenue"</div>';
964
+ }
965
+
966
+ function a11yFixPlaybook(d){
967
+ return '<div class="ai-fix-platform">Android</div>'
968
+ +'<div class="ai-fix-code">android:contentDescription="1 item in cart"</div>'
969
+ +'<div class="ai-fix-platform">iOS (Swift)</div>'
970
+ +'<div class="ai-fix-code">cartBadge.accessibilityLabel = "1 item in cart"</div>'
971
+ +'<ul><li>WCAG 2.1 SC 1.1.1: Non-text Content</li>'
972
+ +'<li>Ensure 4.5:1 contrast ratio (SC 1.4.3)</li>'
973
+ +'<li>Test with TalkBack (Android) / VoiceOver (iOS)</li></ul>';
974
+ }
975
+
976
+ function genericFixPlaybook(d){
977
+ return '<ul><li>Review the defect description and identify the affected screen</li>'
978
+ +'<li>Reproduce on a physical device</li>'
979
+ +'<li>Add <code>assertNoDefectsWithAI</code> for regression detection</li></ul>';
980
+ }
981
+ `;var kt=`
982
+ /* \u2500\u2500 Theme \u2500\u2500 */
983
+ function applyTheme(){
984
+ var pref=localStorage.getItem('tr-theme')||'system';
985
+ var resolved=pref==='system'?(window.matchMedia('(prefers-color-scheme:light)').matches?'light':'dark'):pref;
986
+ document.documentElement.setAttribute('data-theme',resolved);
987
+ document.querySelectorAll('.theme-btn').forEach(function(b){
988
+ b.classList.toggle('active',b.dataset.themeVal===pref);
989
+ });
990
+ }
991
+
992
+ /* \u2500\u2500 Screenshots Tab \u2500\u2500 */
993
+ function renderScreenshotsTab(){
994
+ if(screenshotPaths.length===0)return '<div class="screenshot-empty">No screenshots captured for this run.</div>';
995
+ var h='<div class="screenshot-grid">';
996
+ screenshotPaths.forEach(function(p,i){
997
+ var name=p.split(/[\\/\\\\]/).pop()||'screenshot';
998
+ var isFail=name.toLowerCase().indexOf('fail')>=0||name.toLowerCase().indexOf('error')>=0;
999
+ h+='<div class="screenshot-card'+(isFail?' failure-shot':'')+'" data-lb-idx="'+i+'">'
1000
+ +'<img src="'+esc(p)+'" alt="'+esc(name)+'" loading="lazy" onerror="this.parentElement.style.display=\\'none\\'"/>'
1001
+ +'<div class="screenshot-card-label">'+(isFail?'Failure Screenshot':esc(name))+'</div></div>';
1002
+ });
1003
+ h+='</div>';
1004
+ return h;
1005
+ }
1006
+
1007
+ /* \u2500\u2500 Lightbox \u2500\u2500 */
1008
+ function openLightbox(idx){
1009
+ if(idx<0||idx>=screenshotPaths.length)return;
1010
+ var existing=document.querySelector('.lightbox-overlay');
1011
+ if(existing)existing.remove();
1012
+
1013
+ var name=screenshotPaths[idx].split(/[\\/\\\\]/).pop()||'screenshot';
1014
+ var div=document.createElement('div');
1015
+ div.className='lightbox-overlay';
1016
+ div.innerHTML='<img src="'+esc(screenshotPaths[idx])+'"/>'
1017
+ +'<button class="lightbox-close">&times;</button>'
1018
+ +(idx>0?'<button class="lightbox-nav lightbox-prev">&lsaquo;</button>':'')
1019
+ +(idx<screenshotPaths.length-1?'<button class="lightbox-nav lightbox-next">&rsaquo;</button>':'')
1020
+ +'<div class="lightbox-caption">'+esc(name)+' ('+(idx+1)+' / '+screenshotPaths.length+')</div>';
1021
+ document.body.appendChild(div);
1022
+
1023
+ function cleanup(){div.remove();document.removeEventListener('keydown',keyHandler)}
1024
+ function keyHandler(e){
1025
+ if(e.key==='Escape'){cleanup()}
1026
+ if(e.key==='ArrowLeft'&&idx>0){cleanup();openLightbox(idx-1)}
1027
+ if(e.key==='ArrowRight'&&idx<screenshotPaths.length-1){cleanup();openLightbox(idx+1)}
1028
+ }
1029
+ div.querySelector('.lightbox-close').addEventListener('click',cleanup);
1030
+ div.addEventListener('click',function(e){if(e.target===div)cleanup()});
1031
+ var prev=div.querySelector('.lightbox-prev');
1032
+ var next=div.querySelector('.lightbox-next');
1033
+ if(prev)prev.addEventListener('click',function(e){e.stopPropagation();cleanup();openLightbox(idx-1)});
1034
+ if(next)next.addEventListener('click',function(e){e.stopPropagation();cleanup();openLightbox(idx+1)});
1035
+ document.addEventListener('keydown',keyHandler);
1036
+ }
1037
+
1038
+ /* \u2500\u2500 Event Delegation \u2500\u2500 */
1039
+ document.addEventListener('click',function(e){
1040
+ var target=e.target;
1041
+
1042
+ /* Test row -> open drawer */
1043
+ var row=target.closest('.test-row');
1044
+ if(row){openDrawer(parseInt(row.dataset.flowIdx));return}
1045
+
1046
+ /* Waterfall row -> open drawer */
1047
+ var wfRow=target.closest('.waterfall-row');
1048
+ if(wfRow){openDrawer(parseInt(wfRow.dataset.wfIdx));return}
1049
+
1050
+ /* Drawer close */
1051
+ if(target.closest('#drawer-close')){closeDrawer();return}
1052
+ if(target.closest('#drawer-backdrop')){closeDrawer();return}
1053
+
1054
+ /* Filter chip (status) */
1055
+ var statusChip=target.closest('.filter-chip[data-status]');
1056
+ if(statusChip){
1057
+ var s=statusChip.dataset.status;
1058
+ currentFilter=currentFilter===s?'all':s;
1059
+ renderFilterBar();renderFlowList();
1060
+ return;
1061
+ }
1062
+
1063
+ /* Filter chip (tag \u2014 in filter drawer) */
1064
+ var tagChip=target.closest('.filter-chip[data-tag]');
1065
+ if(tagChip){
1066
+ var tag=tagChip.dataset.tag;
1067
+ activeTagFilter=activeTagFilter===tag?'':tag;
1068
+ renderFilterBar();renderFlowList();
1069
+ return;
1070
+ }
1071
+
1072
+ /* Filter icon -> open filter drawer */
1073
+ if(target.closest('.filter-icon-btn')){
1074
+ $('filter-drawer-backdrop').classList.add('open');
1075
+ $('filter-drawer').classList.add('open');
1076
+ return;
1077
+ }
1078
+
1079
+ /* Filter drawer close */
1080
+ if(target.closest('#filter-drawer-close')||target.closest('#filter-drawer-backdrop')){
1081
+ $('filter-drawer-backdrop').classList.remove('open');
1082
+ $('filter-drawer').classList.remove('open');
1083
+ return;
1084
+ }
1085
+
1086
+ /* Clear all filters */
1087
+ if(target.closest('.filter-clear')){
1088
+ currentFilter='all';searchQuery='';activeTagFilter='';
1089
+ $('filter-drawer-backdrop').classList.remove('open');
1090
+ $('filter-drawer').classList.remove('open');
1091
+ renderFilterBar();renderFlowList();
1092
+ return;
1093
+ }
1094
+
1095
+ /* Seg-ctrl tab switching */
1096
+ var segBtn=target.closest('.seg-btn');
1097
+ if(segBtn){
1098
+ var parent=segBtn.parentElement;
1099
+ if(parent&&parent.classList.contains('seg-ctrl')){
1100
+ var container=parent.parentElement;
1101
+ parent.querySelectorAll('.seg-btn').forEach(function(b){b.classList.remove('active')});
1102
+ segBtn.classList.add('active');
1103
+ container.querySelectorAll('.seg-pane').forEach(function(p){p.classList.remove('active')});
1104
+ var pane=container.querySelector('[data-seg-pane="'+segBtn.dataset.seg+'"]');
1105
+ if(pane)pane.classList.add('active');
1106
+ }
1107
+ return;
1108
+ }
1109
+
1110
+ /* Copy button */
1111
+ var copyBtn=target.closest('.copy-btn');
1112
+ if(copyBtn){
1113
+ e.stopPropagation();
1114
+ var text=copyBtn.dataset.copy||'';
1115
+ navigator.clipboard.writeText(text).then(function(){copyBtn.textContent='Copied!'});
1116
+ setTimeout(function(){copyBtn.textContent='Copy'},1500);
1117
+ return;
1118
+ }
1119
+
1120
+ /* Screenshot lightbox */
1121
+ var card=target.closest('.screenshot-card');
1122
+ if(card&&card.dataset.lbIdx!==undefined){openLightbox(parseInt(card.dataset.lbIdx));return}
1123
+ });
1124
+
1125
+ /* Theme buttons */
1126
+ document.querySelectorAll('.theme-btn').forEach(function(b){
1127
+ b.addEventListener('click',function(){
1128
+ localStorage.setItem('tr-theme',b.dataset.themeVal);
1129
+ applyTheme();
1130
+ });
1131
+ });
1132
+
1133
+ /* Search input */
1134
+ document.addEventListener('input',function(e){
1135
+ if(e.target&&e.target.id==='search-input'){
1136
+ searchQuery=e.target.value.toLowerCase();
1137
+ renderFlowList();
1138
+ }
1139
+ });
1140
+
1141
+ /* Escape to close drawer */
1142
+ document.addEventListener('keydown',function(e){
1143
+ if(e.key==='Escape'&&$('drawer').classList.contains('open'))closeDrawer();
1144
+ });
1145
+ `;var so=`
1146
+ (function(){
1147
+ var payload=JSON.parse(document.getElementById('report-data').textContent);
1148
+ var report=payload.report;
1149
+ var aiDefects=payload.aiDefects||[];
1150
+ var screenshotPaths=payload.screenshotPaths||[];
1151
+
1152
+ ${bt}
1153
+ ${yt}
1154
+ ${xt}
1155
+ ${vt}
1156
+ ${wt}
1157
+ ${kt}
1158
+
1159
+ applyTheme();
1160
+ renderRunMeta();
1161
+ renderHeroStrip();
1162
+ renderWaterfall();
1163
+ renderFilterBar();
1164
+ renderFlowList();
1165
+
1166
+ var _lo=document.getElementById('loading-overlay');if(_lo)_lo.remove();
1167
+ })();`;function Me(e){let t=e.replace(/<\//g,"<\\/");return `<!-- TestRelic Maestro Report \u2014 self-contained, no external dependencies -->
1168
+ <!DOCTYPE html>
1169
+ <html lang="en" data-theme="">
1170
+ <head>
1171
+ <meta charset="UTF-8">
1172
+ <meta name="viewport" content="width=device-width,initial-scale=1">
1173
+ <meta http-equiv="Content-Security-Policy" content="default-src 'none'; style-src 'unsafe-inline'; script-src 'unsafe-inline'; img-src data: blob: file: 'self'; media-src blob: 'self'; connect-src 'self';">
1174
+ <title>TestRelic Maestro Test Report</title>
1175
+ <link rel="icon" type="image/svg+xml" href="data:image/svg+xml;base64,${Buffer.from(mt).toString("base64")}">
1176
+ <style>${gt}</style>
1177
+ <script>(function(){var p=localStorage.getItem('tr-theme')||'system';var t=p==='system'?(window.matchMedia('(prefers-color-scheme:light)').matches?'light':'dark'):p;document.documentElement.setAttribute('data-theme',t);})()</script>
1178
+ </head>
1179
+ <body>
1180
+ <div class="wrap">
1181
+ <div class="top-bar">
1182
+ <div class="brand">
1183
+ ${ht}
1184
+ <div class="brand-text">
1185
+ <span class="brand-title">TestRelic AI</span>
1186
+ <span class="brand-sub">Maestro Test Report</span>
1187
+ </div>
1188
+ </div>
1189
+ <div class="run-meta" id="run-meta"></div>
1190
+ <div class="top-bar-actions">
1191
+ <a class="cta-btn" href="https://testrelic.ai" target="_blank" rel="noopener noreferrer"><svg class="cta-icon" viewBox="0 0 16 16" fill="none"><path d="M8 1l1.545 4.955L14.5 7.5l-4.955 1.545L8 14l-1.545-4.955L1.5 7.5l4.955-1.545L8 1z" fill="currentColor"/></svg>Advanced Insights with AI<span style="color:var(--fg-2);margin:0 2px">&ndash;</span><strong>Get Started</strong><svg class="cta-arrow" viewBox="0 0 12 12" fill="none" stroke="currentColor" stroke-width="1.5"><path d="M2.5 6h7M6.5 3l3 3-3 3"/></svg></a>
1192
+ <div class="theme-toggle">
1193
+ <button class="theme-btn" data-theme-val="system" title="System"><svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="2" y="3" width="20" height="14" rx="2"/><path d="M8 21h8M12 17v4"/></svg></button>
1194
+ <button class="theme-btn" data-theme-val="light" title="Light"><svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="5"/><path d="M12 1v2M12 21v2M4.22 4.22l1.42 1.42M18.36 18.36l1.42 1.42M1 12h2M21 12h2M4.22 19.78l1.42-1.42M18.36 5.64l1.42-1.42"/></svg></button>
1195
+ <button class="theme-btn" data-theme-val="dark" title="Dark"><svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 12.79A9 9 0 1 1 11.21 3 7 7 0 0 0 21 12.79z"/></svg></button>
1196
+ </div>
1197
+ </div>
1198
+ </div>
1199
+
1200
+ <div id="hero-strip" class="hero-strip"></div>
1201
+ <div id="waterfall" class="waterfall-card"></div>
1202
+ <div id="filter-bar" class="filter-bar"></div>
1203
+ <div id="flow-list"></div>
1204
+
1205
+ <div class="report-footer">
1206
+ Generated by <a href="https://testrelic.ai">TestRelic AI</a> &middot; <code>@testrelic/maestro-analytics</code> &middot; <a href="https://docs.testrelic.ai">Documentation</a>
1207
+ </div>
1208
+ </div>
1209
+
1210
+ <div class="loading-overlay" id="loading-overlay"><div class="loading-spinner"></div><div class="loading-text">Loading report...</div></div>
1211
+ <div class="filter-drawer-backdrop" id="filter-drawer-backdrop"></div>
1212
+ <aside class="filter-drawer" id="filter-drawer">
1213
+ <div class="filter-drawer-bar">
1214
+ <span class="filter-drawer-title">Filters</span>
1215
+ <button class="filter-drawer-close" id="filter-drawer-close">&times;</button>
1216
+ </div>
1217
+ <div class="filter-drawer-body" id="filter-drawer-body"></div>
1218
+ </aside>
1219
+ <div class="drawer-backdrop" id="drawer-backdrop"></div>
1220
+ <aside class="drawer" id="drawer">
1221
+ <div class="drawer-bar">
1222
+ <span class="drawer-bar-title">Flow Details</span>
1223
+ <button class="drawer-close" id="drawer-close">&times;</button>
1224
+ </div>
1225
+ <div class="drawer-body" id="drawer-body"></div>
1226
+ </aside>
1227
+ <script id="report-data" type="application/json">${t}</script>
1228
+ <script>${so}</script>
1229
+ </body>
1230
+ </html>`}function Ee(e,t,r,o){let n=resolve(o),s=dirname(n);existsSync(s)||mkdirSync(s,{recursive:true});let i=JSON.stringify({report:e,aiDefects:t,screenshotPaths:r}),l=Me(i);writeFileSync(n,l,"utf-8");}function po(e){let t=60-e.length;return t>0?e+" ".repeat(t):e}function _(e){return `\u2502 ${po(e)} \u2502
1231
+ `}function Fe(e,t,r,o,n){if(o||e.total===0)return;let s=`\u250C${"\u2500".repeat(62)}\u2510
1232
+ `,a=`\u2514${"\u2500".repeat(62)}\u2518
1233
+ `,i=_(""),l=s;l+=_("TestRelic AI - Maestro Test Report"),l+=i;let d=[];if(e.passed>0&&d.push(`${e.passed} \u2713`),e.failed>0&&d.push(`${e.failed} \u2717`),e.flaky>0&&d.push(`${e.flaky} \u26A0`),e.skipped>0&&d.push(`${e.skipped} skipped`),e.timedout>0&&d.push(`${e.timedout} timedout`),l+=_(`Flows: ${e.total} total (${d.join(" ")})`),e.totalAssertions>0&&(l+=_(`Assertions: ${e.totalAssertions} total (${e.passedAssertions} \u2713 ${e.failedAssertions} \u2717)`)),e.totalActionSteps>0){let c=[],p=e.actionStepsByCategory;p.ui_action&&c.push(`${p.ui_action} UI`),p.navigation&&c.push(`${p.navigation} nav`),p.custom_step&&c.push(`${p.custom_step} custom`);let f=c.length>0?` (${c.join(" ")})`:"";l+=_(`Steps: ${e.totalActionSteps} total${f}`);}if(n&&n.length>0){let c=n.filter(v=>v.severity==="critical").length,p=n.filter(v=>v.severity==="warning").length,f=n.filter(v=>v.severity==="info").length,x=[];c>0&&x.push(`${c} critical`),p>0&&x.push(`${p} warning`),f>0&&x.push(`${f} info`),l+=_(`AI Defects: ${n.length} found (${x.join(" ")})`);}l+=_(`Report: ${r}`),l+=_(`Data: ${t}`),l+=a,l+=`For more information visit us at https://docs.testrelic.ai
1234
+ `,process.stderr.write(l);}function _e(e){let t=process.platform,r;t==="darwin"?r=`open "${e}"`:t==="win32"?r=`start "" "${e}"`:r=`xdg-open "${e}"`,exec(r,()=>{});}function bo(){let e=randomBytes(8).toString("hex");return join(tmpdir(),`testrelic-maestro-${e}`)}function Rt(e,t){let r=[];e.platform&&r.push("--platform",e.platform),e.device&&r.push("--device",e.device),r.push("test");let o=join(t,"report.xml"),n=join(t,"artifacts"),s=join(t,"debug");if(r.push("--format","junit"),r.push("--output",o),r.push("--test-output-dir",n),r.push("--debug-output",s),e.analyze&&r.push("--analyze"),e.includeTags&&r.push("--include-tags",e.includeTags),e.excludeTags&&r.push("--exclude-tags",e.excludeTags),e.shards&&e.shards>1&&r.push("--shards",String(e.shards)),e.config&&r.push("--config",e.config),e.env)for(let[a,i]of Object.entries(e.env))r.push("-e",`${a}=${i}`);return e.maestroArgs&&r.push(...e.maestroArgs),r.push(...e.flowPaths),r}async function yo(e){let t=e.outputDir?resolve(e.outputDir):bo();mkdirSync(join(t,"artifacts"),{recursive:true}),mkdirSync(join(t,"debug"),{recursive:true});let r=Rt(e,t),o=join(t,"report.xml"),n=join(t,"artifacts"),s=join(t,"debug");return new Promise(a=>{let i=[],l=[],d=spawn("maestro",r,{stdio:["inherit","pipe","pipe"],shell:true});d.stdout.on("data",c=>{i.push(c),e.quiet||process.stdout.write(c);}),d.stderr.on("data",c=>{l.push(c),e.quiet||process.stderr.write(c);}),d.on("close",c=>{a({exitCode:c??1,junitPath:o,testOutputDir:n,debugOutputDir:s,stdout:Buffer.concat(i).toString("utf-8"),stderr:Buffer.concat(l).toString("utf-8")});}),d.on("error",c=>{a({exitCode:127,junitPath:o,testOutputDir:n,debugOutputDir:s,stdout:"",stderr:`Failed to start maestro CLI: ${c.message}`});});})}var wo=new Set(["localhost","127.0.0.1","0.0.0.0"]);function Le(e){let t=new URL(e);if(t.protocol!=="https:"&&!(t.protocol==="http:"&&wo.has(t.hostname)))throw createError(ErrorCode.CLOUD_CONFIG_INVALID,`HTTPS is required for cloud communication. Got: ${t.protocol}//${t.hostname}`)}var ko=3e3;async function J(e){let t=new AbortController,r=setTimeout(()=>t.abort(),ko);try{return (await fetch(`${e}/health`,{method:"GET",signal:t.signal})).ok}catch{return false}finally{clearTimeout(r);}}async function Tt(e){return new Promise(t=>setTimeout(t,e))}async function St(e){try{let r=(await e.json()).error;if(r&&typeof r.message=="string")return r.message}catch{}return `HTTP ${e.status}`}async function K(e,t,r){let o=new AbortController,n=setTimeout(()=>o.abort(),r);try{let s=await fetch(`${e}/sdk/auth/token`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({apiKey:t}),signal:o.signal});if(s.ok){let i=await s.json();return {accessToken:i.accessToken,refreshToken:i.refreshToken,expiresIn:i.expiresIn,orgId:i.orgId,orgName:i.orgName,userId:i.userId,userName:i.userName}}if(s.status===429){let i=s.headers.get("Retry-After"),l=i?parseInt(i,10)*1e3:5e3;if(!isNaN(l)&&l>0){await Tt(l);let d=await fetch(`${e}/sdk/auth/token`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({apiKey:t}),signal:o.signal});if(d.ok){let c=await d.json();return {accessToken:c.accessToken,refreshToken:c.refreshToken,expiresIn:c.expiresIn,orgId:c.orgId,orgName:c.orgName,userId:c.userId,userName:c.userName}}}return {code:"rate_limited",message:"Rate limited during token exchange.",statusCode:429}}return s.status===400?{code:"validation_error",message:await St(s),statusCode:400}:s.status===401?{code:"invalid_key",message:"API key is invalid or has been revoked.",statusCode:401}:s.status===403?{code:"expired_key",message:"API key has expired.",statusCode:403}:{code:"server_error",message:await St(s),statusCode:s.status}}catch(s){return s instanceof DOMException&&s.name==="AbortError"?{code:"timeout",message:"Token exchange timed out.",statusCode:null}:{code:"network_error",message:"Failed to reach cloud for token exchange.",statusCode:null}}finally{clearTimeout(n);}}function W(e){return "code"in e&&typeof e.code=="string"&&["invalid_key","expired_key","validation_error","rate_limited","server_error","network_error","timeout"].includes(e.code)}async function Pe(e,t,r){let o=new AbortController,n=setTimeout(()=>o.abort(),r);try{let s=await fetch(`${e}/sdk/auth/refresh`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({refreshToken:t}),signal:o.signal});if(s.status===429){let i=s.headers.get("Retry-After"),l=i?parseInt(i,10)*1e3:5e3;!isNaN(l)&&l>0&&(await Tt(l),s=await fetch(`${e}/sdk/auth/refresh`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({refreshToken:t}),signal:o.signal}));}if(!s.ok)return null;let a=await s.json();return {accessToken:a.accessToken,refreshToken:a.refreshToken,expiresIn:a.expiresIn}}catch{return null}finally{clearTimeout(n);}}async function De(e,t,r,o,n,s){let a=new AbortController,i=setTimeout(()=>a.abort(),n);try{let l={gitId:r,displayName:o};s&&(l.branch=s);let d=await fetch(`${e}/repos/resolve`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(l),signal:a.signal});if(!d.ok)return null;let c=await d.json();return {repoId:c.repoId,displayName:c.displayName}}catch{return null}finally{clearTimeout(i);}}var Ao=5e3;function L(e,t){try{return execSync(e,{cwd:t,timeout:Ao,encoding:"utf-8",stdio:["pipe","pipe","pipe"]}).trim()||null}catch{return null}}function Ne(e){let t=L("git rev-parse --abbrev-ref HEAD",e),r=L("git rev-parse --short HEAD",e),o=L("git log -1 --pretty=%s",e),n=L("git log -1 --format=%an",e)??L("git config user.name",e),s=Io(e),a=s?P(s):null;return {branch:t,commitSha:r,commitMessage:o,commitAuthor:n,remoteUrl:a}}function Io(e){let t=L("git remote get-url origin",e);if(t)return t;let r=L("git remote",e);if(!r)return null;let o=r.split(`
1235
+ `)[0]?.trim();return o?L(`git remote get-url ${o}`,e):null}function P(e){let t=e.trim();return t=t.replace(/^[a-z+]+:\/\//,""),t=t.replace(/^[^@]+@/,""),t=t.replace(/:(?!\d)/,"/"),t=t.replace(/\.git$/,""),t=t.replace(/\/+$/,""),t=t.replace(/^[^@/]+@/,""),t.toLowerCase()}function Ue(e){let t=e.split("/").filter(Boolean);return t[t.length-1]??e}function At(e){try{let t=readFileSync(join(e,"package.json"),"utf-8"),r=JSON.parse(t);return typeof r.name=="string"&&r.name.length>0?r.name:null}catch{return null}}function D(e){let t=At(e);return t||`local/${basename(e)}`}var Po="1.0.0",It=10;function Y(e,t,r,o,n,s,a,i){try{mkdirSync(e,{recursive:!0});let l=Date.now(),d=`${l}-${t}-${r}.json`,c=join(e,d),p={version:Po,queuedAt:new Date(l).toISOString(),reason:o,retryCount:0,targetEndpoint:n,method:s,payload:a,headers:i},f=c+".tmp";writeFileSync(f,JSON.stringify(p,null,2),"utf-8"),renameSync(f,c);}catch(l){l.code==="ENOSPC"?process.stderr.write(`\u26A0 TestRelic: Disk full \u2014 unable to queue upload data.
1236
+ `):process.stderr.write(`\u26A0 TestRelic: Unable to queue upload data.
1237
+ `);}}async function ze(e,t,r){let o;try{o=readdirSync(e).filter(s=>s.endsWith(".json")&&!s.endsWith(".tmp")).sort();}catch{return}if(o.length===0)return;let n=[];for(let s=0;s<o.length;s+=It)n.push(o.slice(s,s+It));for(let s of n)for(let a of s){let i=join(e,a);try{if(!statSync(i).isFile())continue;let d=readFileSync(i,"utf-8"),c=JSON.parse(d);if(!isValidQueueEntry(c)){unlinkSync(i);continue}let p=c;if((await fetch(p.targetEndpoint,{method:p.method,headers:{...p.headers,Authorization:`Bearer ${r}`},body:JSON.stringify(p.payload)})).ok)unlinkSync(i);else return}catch{return}}}function je(e,t){try{let r=readdirSync(e).filter(n=>n.endsWith(".json")&&!n.endsWith(".tmp")),o=Date.now();for(let n of r){let s=join(e,n);try{let a=readFileSync(s,"utf-8"),l=JSON.parse(a).queuedAt;if(typeof l=="string"){let d=new Date(l).getTime();o-d>t&&unlinkSync(s);}}catch{try{unlinkSync(s);}catch{}}}}catch{}}var $o=3e5,Bo=864e5,Ho=6e4,Ft=3e4,_t="https://platform.testrelic.ai/settings/api-keys",j=class{constructor(t){this.gitMetadata=null;this.repoId=null;this.failureReason=null;this.flushPromise=null;this.healthCheckTimer=null;this.config=t,this.authState={mode:"pending",accessToken:null,refreshToken:null,expiresAt:null,orgId:null,orgName:null,userId:null,userName:null};}async initialize(){if(!this.config||!this.config.apiKey){this.setLocalMode("no_api_key"),process.stderr.write(`\u2139 TestRelic: No API key configured. Running in local mode.
1238
+ `);return}try{if(Le(this.config.endpoint),this.gitMetadata=Ne(),!await J(this.config.endpoint)){this.setLocalMode("cloud_unreachable"),process.stderr.write(`\u26A0 TestRelic: Cloud unreachable. Switching to local mode.
1239
+ `);return}let r=await K(this.config.endpoint,this.config.apiKey,this.config.timeout);if(W(r)){this.handleAuthError(r.code,r.statusCode);return}let o=r;this.authState={mode:"cloud",accessToken:o.accessToken,refreshToken:o.refreshToken,expiresAt:Date.now()+o.expiresIn*1e3,orgId:o.orgId,orgName:o.orgName,userId:o.userId,userName:o.userName},process.stderr.write(`\u2713 TestRelic: Connected to cloud (${o.orgName} / ${o.userName})
1240
+ `),await this.resolveRepoId(),this.config.queueDirectory&&(je(this.config.queueDirectory,this.config.queueMaxAge),this.startBackgroundFlush()),this.startHealthCheck();}catch(t){this.setLocalMode("unexpected_error"),process.stderr.write(`\u26A0 TestRelic: Unexpected error during cloud initialization. ${t instanceof Error?t.message:String(t)}
1241
+ `);}}getMode(){return this.authState.mode}isCloudMode(){return this.authState.mode==="cloud"}isLocalMode(){return this.authState.mode==="local"}getAccessToken(){return this.authState.accessToken}getRepoId(){return this.repoId}getGitMetadata(){return this.gitMetadata}getConfig(){return this.config}getFailureReason(){return this.failureReason}getEndpoint(){return this.config?.endpoint??"https://platform.testrelic.ai/api/v1"}async ensureValidToken(){if(!this.isCloudMode()||!this.authState.accessToken)return false;let t=this.authState.expiresAt??0;if(Date.now()+$o<t)return true;if(!this.config||!this.authState.refreshToken)return this.switchToLocalMode("token_expired_no_refresh"),false;try{let r=await Pe(this.config.endpoint,this.authState.refreshToken,this.config.timeout);return r?(this.authState.accessToken=r.accessToken,this.authState.refreshToken=r.refreshToken,this.authState.expiresAt=Date.now()+r.expiresIn*1e3,!0):(this.switchToLocalMode("token_refresh_failed"),!1)}catch{return this.switchToLocalMode("token_refresh_error"),false}}switchToLocalMode(t){this.authState.mode!=="local"&&(this.authState.mode="local",this.failureReason=t,process.stderr.write(`\u26A0 TestRelic: Switched to local mode (${t}).
1242
+ `));}async dispose(){if(this.healthCheckTimer&&(clearInterval(this.healthCheckTimer),this.healthCheckTimer=null),this.flushPromise){try{await Promise.race([this.flushPromise,new Promise(t=>setTimeout(t,Ft))]);}catch{}this.flushPromise=null;}this.authState={mode:"local",accessToken:null,refreshToken:null,expiresAt:null,orgId:null,orgName:null,userId:null,userName:null},this.repoId=null,this.gitMetadata=null;}setLocalMode(t){this.authState.mode="local",this.failureReason=t;}handleAuthError(t,r){switch(t){case "invalid_key":this.setLocalMode("invalid_api_key"),process.stderr.write(`\u26A0 TestRelic: API key is invalid or revoked. Running in local mode.
1243
+ \u2192 Manage keys: ${_t}
1244
+ `);break;case "expired_key":this.setLocalMode("expired_api_key"),process.stderr.write(`\u26A0 TestRelic: API key has expired. Running in local mode.
1245
+ \u2192 Generate a new key: ${_t}
1246
+ `);break;case "rate_limited":this.setLocalMode("rate_limited"),process.stderr.write(`\u26A0 TestRelic: Rate limited during authentication. Running in local mode.
1247
+ `);break;default:this.setLocalMode(`auth_error_${r??"unknown"}`),process.stderr.write(`\u26A0 TestRelic: Cloud authentication failed. Running in local mode.
1248
+ `);break}}async resolveRepoId(){if(!this.config||!this.authState.accessToken)return;let t=this.gitMetadata?.remoteUrl?P(this.gitMetadata.remoteUrl):null,r=t?Ue(t):this.config.projectName??D(process.cwd()),o=t??this.config.projectName??D(process.cwd()),n=this.readRepoCache(o);if(n){this.repoId=n.repoId;return}try{let s=await De(this.config.endpoint,this.authState.accessToken,o,r,this.config.timeout,this.gitMetadata?.branch);s&&(this.repoId=s.repoId,this.writeRepoCache(o,s.repoId,s.displayName));}catch{}}readRepoCache(t){if(!this.config)return null;let r=join(this.config.queueDirectory,"..","cache","repo.json");try{if(!existsSync(r))return null;let o=readFileSync(r,"utf-8"),n=JSON.parse(o);if(n.gitId!==t||Date.now()-n.resolvedAt>Bo)return null;let s=this.hashApiKey();return s&&n.apiKeyHash!==s?null:n}catch{return null}}writeRepoCache(t,r,o){if(!this.config)return;let n=join(this.config.queueDirectory,"..","cache"),s=join(n,"repo.json");try{mkdirSync(n,{recursive:!0});let a={repoId:r,gitId:t,displayName:o,resolvedAt:Date.now(),apiKeyHash:this.hashApiKey()??""},i=s+".tmp";writeFileSync(i,JSON.stringify(a,null,2),"utf-8"),renameSync(i,s);}catch{}}hashApiKey(){return this.config?.apiKey?createHash("sha256").update(this.config.apiKey).digest("hex").substring(0,16):null}startBackgroundFlush(){if(!this.config||!this.authState.accessToken)return;let t=this.authState.accessToken,r=this.config.queueDirectory,o=this.config.endpoint;this.flushPromise=Promise.race([ze(r,o,t),new Promise(n=>setTimeout(n,Ft))]).catch(()=>{});}startHealthCheck(){if(!this.config)return;let t=this.config.endpoint;this.healthCheckTimer=setInterval(async()=>{if(!this.isCloudMode()&&!(!this.failureReason||!["cloud_unreachable","auth_timeout","network_error"].includes(this.failureReason)))try{if(!await J(t))return;if(this.config?.apiKey){let o=await K(this.config.endpoint,this.config.apiKey,this.config.timeout);if(!W(o)){let n=o;this.authState={mode:"cloud",accessToken:n.accessToken,refreshToken:n.refreshToken,expiresAt:Date.now()+n.expiresIn*1e3,orgId:n.orgId,orgName:n.orgName,userId:n.userId,userName:n.userName},this.failureReason=null,process.stderr.write(`\u2713 TestRelic: Cloud connectivity restored.
1249
+ `),this.startBackgroundFlush();}}}catch{}},Ho);}};var Vo=1048576,Q=[1e3,3e3,9e3],Z=3;function te(e,t,r,o,n){let s=o?.provider&&o.provider!=="unknown"?o.provider:"local";return {runId:e.testRunId,repoGitId:t,startedAt:e.startedAt,summary:e.summary,timeline:e.timeline,environment:s,testFramework:"maestro",...n&&n.length>0?{tests:n}:{},...r?.branch?{branch:r.branch}:{},...r?.commitSha?{commit:r.commitSha}:{},...r?.commitMessage?{commitMessage:r.commitMessage}:{},...r?.commitAuthor?{commitAuthor:r.commitAuthor}:{},...e.completedAt?{finishedAt:e.completedAt}:{},...e.totalDuration?{duration:e.totalDuration}:{},...o?.provider?{ciProvider:o.provider}:{},...o?.runUrl?{ciRunUrl:o.runUrl}:{}}}async function ee(e){return new Promise(t=>setTimeout(t,e))}async function Lt(e,t,r){for(let o=0;o<Z;o++)try{let n=await fetch(e,t);if(n.ok)return n;if(n.status===401&&r&&o===0){let s=await r();if(s){let a={...t,headers:{...t.headers,Authorization:`Bearer ${s}`}},i=await fetch(e,a);if(i.ok)return i}return null}if(n.status===429&&o<Z-1){let s=n.headers.get("Retry-After"),a=s?parseInt(s,10)*1e3:Q[o];!isNaN(a)&&a>0?await ee(a):await ee(Q[o]);continue}if(n.status>=500&&o<Z-1){await ee(Q[o]);continue}return n}catch{if(o<Z-1){await ee(Q[o]);continue}return null}return null}function qo(e){let t=JSON.stringify(e),r={"Content-Type":"application/json"};if(Buffer.byteLength(t,"utf-8")>Vo){let o=gzipSync(Buffer.from(t,"utf-8"));return r["Content-Encoding"]="gzip",{body:o,headers:r}}return {body:t,headers:r}}async function Be(e,t,r,o){let n=`${e}/runs`,{body:s,headers:a}=qo(r);a.Authorization=`Bearer ${t}`;let i=await Lt(n,{method:"POST",headers:a,body:s},o);return i?.ok?{success:true,statusCode:i.status,error:null}:i?.status===409?{success:true,statusCode:409,error:null}:{success:false,reason:i?`Upload failed with status ${i.status}`:"Upload failed after retries",statusCode:i?.status??null,payload:r,targetEndpoint:n,method:"POST"}}function Jo(e){let t={};for(let[r,o]of Object.entries(e))o!=null&&(t[r]=o);return t}async function Ko(e,t,r){let o=`${e}/runs/init`;try{let n=await fetch(o,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(Jo(r))});return n.ok?{runId:(await n.json()).runId}:null}catch{return null}}async function Wo(e,t,r,o){let n=`${e}/runs/${r}/finalize`;return (await Lt(n,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify(o)}))?.ok??false}var es=5,re=[1e3,3e3,9e3],N=3,ts={".png":"image/png",".jpg":"image/jpeg",".jpeg":"image/jpeg",".webp":"image/webp",".webm":"video/webm",".mp4":"video/mp4",".mov":"video/quicktime",".zip":"application/zip"},He=0,Ge=[];async function rs(){He>=es&&await new Promise(e=>Ge.push(e)),He++;}function os(){He--,Ge.length>0&&Ge.shift()();}async function oe(e){return new Promise(t=>setTimeout(t,e))}function ss(e){let t=e.substring(e.lastIndexOf(".")).toLowerCase();return ts[t]??"application/octet-stream"}function ns(e){try{return statSync(e).size}catch{return 0}}async function as(e,t,r,o,n){let s=`${e}/artifacts/upload-url`,a=JSON.stringify({runId:r.runId,testId:r.testId,fileName:basename(r.filePath),contentType:n,type:r.type,sizeBytes:o});for(let i=0;i<N;i++)try{let l=await fetch(s,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:a});if(l.ok)return await l.json();if(l.status>=500&&i<N-1){await oe(re[i]);continue}return null}catch{if(i<N-1){await oe(re[i]);continue}return null}return null}async function is(e,t,r,o){for(let n=0;n<N;n++)try{let s=createReadStream(t),a=Readable.toWeb(s),i=await fetch(e,{method:"PUT",headers:{"Content-Type":r,"Content-Length":String(o)},body:a,duplex:"half"});if(i.ok)return !0;if(i.status>=500&&n<N-1){await oe(re[n]);continue}return !1}catch{if(n<N-1){await oe(re[n]);continue}return false}return false}async function ls(e,t,r){try{return (await fetch(`${e}/artifacts/confirm`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify({artifactId:r})})).ok}catch{return false}}async function Pt(e,t,r,o){let n=ns(r.filePath);if(n===0)return {success:false,storageKey:null,artifactId:null,error:"file_not_found_or_empty"};if(n>o*1024*1024)return {success:false,storageKey:null,artifactId:null,error:"file_too_large"};let s=ss(r.filePath);await rs();try{let a=await as(e,t,r,n,s);if(!a)return {success:!1,storageKey:null,artifactId:null,error:"upload_url_request_failed"};if(!await is(a.uploadUrl,r.filePath,s,n))return {success:!1,storageKey:a.storageKey,artifactId:a.artifactId,error:"presigned_put_failed"};let l=await ls(e,t,a.artifactId);return {success:l,storageKey:a.storageKey,artifactId:a.artifactId,error:l?null:"confirm_failed"}}finally{os();}}async function Ve(e,t,r,o){let n=new Map,s=r.map(async a=>{let i=await Pt(e,t,a,o);n.set(a.filePath,i);});return await Promise.allSettled(s),n}function cs(e){return e.GITHUB_ACTIONS!=="true"?null:{provider:"github-actions",buildId:e.GITHUB_RUN_ID??null,commitSha:e.GITHUB_SHA??null,branch:e.GITHUB_REF_NAME??null,runUrl:e.GITHUB_SERVER_URL&&e.GITHUB_REPOSITORY&&e.GITHUB_RUN_ID?`${e.GITHUB_SERVER_URL}/${e.GITHUB_REPOSITORY}/actions/runs/${e.GITHUB_RUN_ID}`:null}}function ds(e){return e.GITLAB_CI!=="true"?null:{provider:"gitlab-ci",buildId:e.CI_PIPELINE_ID??null,commitSha:e.CI_COMMIT_SHA??null,branch:e.CI_COMMIT_BRANCH??e.CI_COMMIT_REF_NAME??null,runUrl:e.CI_PIPELINE_URL??null}}function ps(e){if(!e.JENKINS_URL)return null;let t=e.GIT_BRANCH??null;return t?.startsWith("origin/")&&(t=t.slice(7)),{provider:"jenkins",buildId:e.BUILD_ID??null,commitSha:e.GIT_COMMIT??null,branch:t,runUrl:e.BUILD_URL??null}}function us(e){return e.CIRCLECI!=="true"?null:{provider:"circleci",buildId:e.CIRCLE_BUILD_NUM??null,commitSha:e.CIRCLE_SHA1??null,branch:e.CIRCLE_BRANCH??null,runUrl:e.CIRCLE_BUILD_URL??null}}function fs(e){if(!e.BITBUCKET_PIPELINE_UUID)return null;let t=e.BITBUCKET_WORKSPACE&&e.BITBUCKET_REPO_SLUG&&e.BITBUCKET_PIPELINE_UUID?`https://bitbucket.org/${e.BITBUCKET_WORKSPACE}/${e.BITBUCKET_REPO_SLUG}/pipelines/results/${e.BITBUCKET_PIPELINE_UUID}`:null;return {provider:"bitbucket-pipelines",buildId:e.BITBUCKET_BUILD_NUMBER??null,commitSha:e.BITBUCKET_COMMIT??null,branch:e.BITBUCKET_BRANCH??null,runUrl:t}}var gs=[cs,ds,ps,us,fs];function se(e){let t=e??process.env;for(let r of gs){let o=r(t);if(o)return o}return null}function ms(e){let t=[];for(let r of e){let o=r,n=o.tests;if(!Array.isArray(n)||n.length===0){t.push(r);continue}for(let s of n){t.push({timestamp:s.startedAt,title:s.title,category:"test",navigationType:o.navigationType,url:o.url,visitedAt:o.visitedAt,duration:s.duration,status:s.status,test_status:s.status,test_id:s.testId,test_title:s.title,specfile:s.filePath,suiteName:s.suiteName,type:"test",...s.failure?{error:s.failure.message,errorMessage:s.failure.message}:{}});for(let a of s.actions??[])t.push({timestamp:a.timestamp,title:a.title,category:a.category,status:a.status,test_status:s.status,duration:a.duration,error:a.error,test_id:s.testId,test_title:s.title,specfile:s.filePath,url:o.url,type:a.category==="assertion"?"assert":"action"});}}return t}function Dt(e){let r=e.getGitMetadata()?.remoteUrl;return r?P(r):e.getConfig()?.projectName??D(process.cwd())}async function qe(e,t,r,o,n,s,a,i,l,d,c){try{let p=c?.length??0,f=(i?.length??0)+(l?.length??0)+p>0;if(e.isCloudMode()&&f&&await e.ensureValidToken()){let v=e.getEndpoint(),m=e.getAccessToken(),g=t?.artifactMaxSizeMb??50,w=[];for(let u of i??[])w.push({filePath:u,runId:r,testId:"maestro-suite",type:"screenshot"});let I=new Set;for(let u of c??[])w.push({filePath:u.path,runId:r,testId:u.testId,type:"video"}),I.add(u.path);for(let u of l??[])I.has(u)||w.push({filePath:u,runId:r,testId:"maestro-suite",type:"video"});if(w.length>0){let u=await Ve(v,m,w,g),b=Array.from(u.entries()),h=b.filter(([,k])=>k.success).length,S=b.filter(([,k])=>!k.success);if(h>0){let k=w.filter(T=>T.type==="video").length,C=c?.length??0,y=w.filter(T=>T.type==="screenshot").length,R=[];if(y>0&&R.push(`${y} screenshot(s)`),k>0){let T=C>0?` (${C} per-flow)`:"";R.push(`${k} video(s)${T}`);}process.stderr.write(`\u2713 TestRelic: Uploaded ${h} artifact(s) [${R.join(", ")}] to cloud storage.
1250
+ `);}if(S.length>0){process.stderr.write(`\u26A0 TestRelic: ${S.length} artifact(s) failed to upload:
1251
+ `);for(let[k,C]of S){let y=w.find(T=>T.filePath===k),R=y?`[${y.type}] ${k}`:k;process.stderr.write(` \u2717 ${R} \u2014 reason: ${C.error??"unknown"}
1252
+ `);}}}}if(e.isCloudMode())if(await e.ensureValidToken()){let v=e.getGitMetadata(),m=se(),g=Dt(e),w=ms(o.timeline),I={...o,timeline:w},u=te(I,g,v,m,d),b=await Be(e.getEndpoint(),e.getAccessToken(),u,async()=>await e.ensureValidToken()?e.getAccessToken():null);if(b.success)process.stderr.write(`\u2713 TestRelic: Cloud upload succeeded \u2192 ${e.getEndpoint()}/runs/${r}
1253
+ `);else {let h=b,S=t?.queueDirectory??".testrelic/queue";Y(S,r,"batch",h.reason,h.targetEndpoint,h.method,h.payload,{"Content-Type":"application/json"}),process.stderr.write(`\u26A0 TestRelic: Cloud upload failed, queued for retry.
1254
+ `);}}else {let v=t?.queueDirectory??".testrelic/queue",m=e.getGitMetadata(),g=se(),w=Dt(e),I=te(o,w,m,g);Y(v,r,"batch",e.getFailureReason()??"token_invalid",`${e.getEndpoint()}/runs`,"POST",I,{"Content-Type":"application/json"});}await e.dispose();}catch{try{await e.dispose();}catch{}}}function Ut(e,t,r){if(t.length===0)return null;let o=basename(e,extname(e)).toLowerCase(),n=t.find(i=>basename(i,extname(i)).toLowerCase()===o);if(n)return n;let s=t.find(i=>basename(i).toLowerCase().includes(o));if(s)return s;let a=t.find(i=>{let l=basename(i,extname(i)).toLowerCase();return l.length>=3&&(o.startsWith(l)||o.includes(l))});if(a)return a;if(r){let i=basename(r,extname(r)).toLowerCase(),l=t.find(c=>basename(c,extname(c)).toLowerCase()===i);if(l)return l;let d=t.find(c=>basename(c).toLowerCase().includes(i));if(d)return d}return null}function vs(e,t){if(t.length===0)return [];let r=basename(e,extname(e)).toLowerCase(),o=t.filter(n=>{let s=basename(n).toLowerCase(),a=basename(n,extname(n)).toLowerCase();return s.startsWith(r)||s.includes(r)?true:a.length>=3&&(r.startsWith(a)||r.includes(a))});return o.length>0?o:t.slice()}function ws(e){for(let t of e)if(t.command==="startRecording"){let r=t.metadata,o=r?.path??r?.recording??r?.fileName;if(typeof o=="string")return o;if(typeof t.selector=="string")return t.selector}return null}function ks(e){let t=new Map;for(let[r,o]of e){let n=basename(r).replace(/^commands[-_]?/i,"").replace(/\.json$/i,"").toLowerCase(),s=ws(o);s&&n&&t.set(n,s);}return t}function Cs(e){switch(e){case "android":return "Android";case "ios":return "iOS";case "web":return "Web";default:return}}function Rs(e){return {testId:`${e.flowFile}::${e.flowName}`,title:e.flowName,status:e.status,duration:e.duration,suiteName:e.appId??"maestro-suite",filePath:e.flowFile,testType:"mobile",isFlaky:false,startedAt:e.startedAt,completedAt:e.completedAt,tags:e.tags,failure:e.failureMessage?{message:e.failureMessage}:null,retry:0,retryCount:0,retryStatus:null,source:"maestro",platform:e.platform!=="unknown"?e.platform:void 0,os:Cs(e.platform),deviceName:e.deviceId}}async function Ss(e){let{config:t}=e,r=new Date().toISOString(),o=t.testRunId??randomUUID(),n=Ae(e.testOutputDir,e.debugOutputDir),s=e.junitPath??n.junitReportPath,a=e.platform??"unknown";if(a==="unknown"){let u=e.debugOutputDir?ye(e.debugOutputDir):n.logPaths;for(let b of u){let h=he(b),S=be(h);if(S!=="unknown"){a=S;break}}}let i=new Map;if(e.flowsDir&&existsSync(e.flowsDir)){let u=me(e.flowsDir);for(let b of u){let h=ge(b),S=h.name??b;i.set(S,h);}}let l=new Map,d=e.testOutputDir?fe(e.testOutputDir):n.commandJsonPaths;for(let u of d)l.set(u,ue(u));let c=[],p=e.testOutputDir?Te(e.testOutputDir):n.aiReportPaths;for(let u of p){let b=Se(u);c.push(...b.defects);}let f=ks(l),x=[];if(s&&existsSync(s)){let u=pe(s),b=Array.from(l.values()).flat(),h=b.filter(k=>k.category==="assertion"),S=b.filter(k=>k.category!=="assertion");for(let k of u.testSuites)for(let C of k.testCases){let y=C.name,R=i.get(y),T=C.status==="SUCCESS"?"passed":C.status==="SKIPPED"?"skipped":"failed",F=C.time*1e3,U=r,$=new Date(new Date(U).getTime()+F).toISOString(),O=C.classname||y,zt=basename(O,extname(O)).toLowerCase(),jt=f.get(zt)??null;x.push({flowName:y,flowFile:O,appId:R?.appId??null,platform:a,deviceId:e.device,status:T,duration:F,startedAt:U,completedAt:$,tags:R?.tags??[],properties:R?.properties??{},commands:S,assertions:h,screenshotPaths:vs(O,n.screenshotPaths),videoPath:Ut(O,n.videoPaths,jt),aiDefects:c,logEntries:[],failureMessage:C.failureMessage??C.errorMessage??null,failureType:C.failureType??C.errorType??null,subflowRefs:R?.subflowRefs??[],metadata:R??null});}}if(x.length===0&&i.size>0)for(let[,u]of i)x.push({flowName:u.name??u.filePath,flowFile:u.filePath,appId:u.appId,platform:a,deviceId:e.device,status:"passed",duration:0,startedAt:r,completedAt:r,tags:u.tags,properties:u.properties,commands:[],assertions:[],screenshotPaths:[],videoPath:null,aiDefects:[],logEntries:[],failureMessage:null,failureType:null,subflowRefs:u.subflowRefs,metadata:u});let v=Ie(x),m=z(v),g=new Date().toISOString(),w=Date.now()-new Date(r).getTime(),I={schemaVersion:"1.0.0",testRunId:o,startedAt:r,completedAt:g,totalDuration:w,summary:m,ci:null,metadata:t.metadata??null,timeline:v,shardRunIds:null};if(mkdirSync(dirname(resolve(t.outputPath)),{recursive:true}),writeFileSync(resolve(t.outputPath),JSON.stringify(I,null,2),"utf-8"),Ee(I,c,n.screenshotPaths,resolve(t.htmlReportPath)),Fe(m,t.outputPath,t.htmlReportPath,t.quiet,c),t.openReport&&_e(resolve(t.htmlReportPath)),t.cloud){let u=new j(t.cloud);await u.initialize();let b=t.includeVideo?n.videoPaths:[],h=[];if(b.length>0){for(let y of x){let R=basename(y.flowFile,extname(y.flowFile)).toLowerCase(),T=f.get(R)??null,F=Ut(y.flowFile,b,T);if(F){let U=`${y.flowFile}::${y.flowName}`;h.some($=>$.path===F&&$.testId===U)||h.push({path:F,testId:U});}}if(b.length===1){let y=b[0];for(let R of x){let T=`${R.flowFile}::${R.flowName}`;h.some(F=>F.testId===T)||h.push({path:y,testId:T});}}}let S=new Set(h.map(y=>y.path)),k=b.filter(y=>!S.has(y)),C=x.map(Rs);await qe(u,t.cloud,o,I,g,w,m,t.includeScreenshots?n.screenshotPaths:[],k,C,h);}return {report:I,flowResults:x,aiDefects:c,artifacts:n}}function _s(e){return "tests"in e&&"visitedAt"in e}function Ot(e){let t=[],r=[],o="",n="",s="";for(let d of e)try{let c=readFileSync(d,"utf-8"),p=JSON.parse(c);s||(s=p.testRunId),(!o||p.startedAt<o)&&(o=p.startedAt),(!n||p.completedAt&&p.completedAt>n)&&(n=p.completedAt);for(let f of p.timeline)t.push(f),_s(f)&&r.push(f);}catch{process.stderr.write(`\u26A0 TestRelic: Unable to read report file: ${d}
1255
+ `);}let a=z(r),i=o?new Date(o).getTime():Date.now(),l=n?new Date(n).getTime():Date.now();return {schemaVersion:"1.0.0",testRunId:s||`maestro-merged-${Date.now()}`,startedAt:o||new Date().toISOString(),completedAt:n||new Date().toISOString(),totalDuration:l-i,summary:a,ci:null,metadata:null,timeline:t,shardRunIds:null}}function Ls(e,t){if(!existsSync(e))throw new Error(`Directory does not exist: ${e}`);let r=readdirSync(e).filter(n=>extname(n)===".json"&&n.includes("testrelic")).map(n=>join(e,n)),o=Ot(r);return writeFileSync(t,JSON.stringify(o,null,2),"utf-8"),o}export{j as CloudClient,ae as DEFAULT_CLOUD_CONFIG,Rt as buildMaestroArgs,z as buildSummary,Ie as buildTimeline,ft as buildTimelineEntry,te as buildUploadPayload,ot as categorizeCommand,je as cleanupExpiredQueue,Ae as collectArtifacts,Ne as collectGitMetadata,D as deriveNonGitProjectId,Ue as deriveRepoDisplayName,se as detectCI,be as detectPlatformFromLogs,Te as discoverAiReports,fe as discoverCommandFiles,le as discoverConfigFile,me as discoverFlowFiles,ye as discoverLogFiles,Le as enforceHttps,K as exchangeToken,qe as finalizeAndUpload,Wo as finalizeRun,ze as flushQueue,Ee as generateHtmlReport,eo as getArtifactStats,J as healthCheck,Ko as initRealtimeRun,W as isAuthError,de as mergeCloudConfig,Ot as mergeReports,Ls as mergeReportsFromDirectory,P as normalizeGitRemoteUrl,_e as openInBrowser,Ss as orchestrateReport,Se as parseAiReportFile,dt as parseAiReportHtml,ue as parseCommandsFile,st as parseCommandsJson,ce as parseConfigFile,ie as parseDuration,ge as parseFlowFile,it as parseFlowYaml,pe as parseJUnitFile,tt as parseJUnitXml,ct as parseLogContent,he as parseLogFile,Fe as printConsoleSummary,At as readPackageJsonName,Pe as refreshAccessToken,Me as renderHtmlDocument,tr as resolveConfig,B as resolveEnvVar,H as resolveEnvVars,De as resolveRepo,yo as runMaestro,Pt as uploadArtifact,Ve as uploadArtifacts,Be as uploadBatchRun,Y as writeToQueue};//# sourceMappingURL=index.js.map
1256
+ //# sourceMappingURL=index.js.map