@testrelic/playwright-analytics 2.2.7 → 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.cjs +621 -30
- package/dist/index.cjs +530 -335
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +7 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +530 -335
- package/dist/index.js.map +1 -1
- package/dist/merge.cjs +1 -1
- package/dist/merge.cjs.map +1 -1
- package/dist/merge.js +1 -1
- package/dist/merge.js.map +1 -1
- package/package.json +2 -2
package/dist/merge.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
'use strict';var crypto=require('crypto'),fs=require('fs'),path=require('path'),core=require('@testrelic/core');
|
|
1
|
+
'use strict';var crypto=require('crypto'),fs=require('fs'),path=require('path'),core=require('@testrelic/core');function k(s){try{return fs.statSync(s).isDirectory()&&fs.existsSync(path.join(s,"index.json"))}catch{return false}}function J(s){let o=path.join(s,"summary.json"),e=path.join(s,"index.json"),i;try{i=JSON.parse(fs.readFileSync(o,"utf-8"));}catch(n){throw core.createError(core.ErrorCode.MERGE_READ_FAILED,`Failed to read summary.json from: ${s}`,n)}let d;try{d=JSON.parse(fs.readFileSync(e,"utf-8"));}catch(n){throw core.createError(core.ErrorCode.MERGE_READ_FAILED,`Failed to read index.json from: ${s}`,n)}return {summary:i,index:d,dir:s}}async function U(s,o){let e=[],i=[];for(let t of s){if(k(t)){i.push(J(t));continue}let r;try{r=fs.readFileSync(t,"utf-8");}catch(l){throw core.createError(core.ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,l)}let m;try{m=JSON.parse(r);}catch(l){throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,l)}if(!core.isValidTestRunReport(m))throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);e.push(m);}let d=e.map(t=>t.testRunId),n=[];for(let t of e)n.push(...t.timeline);if(n.sort((t,r)=>{let m="visitedAt"in t?t.visitedAt:t.timestamp,l="visitedAt"in r?r.visitedAt:r.timestamp;return new Date(m).getTime()-new Date(l).getTime()}),i.length>0){let t=path.dirname(o.output),r=path.join(t,".testrelic-report"),m=path.join(r,"tests");fs.mkdirSync(m,{recursive:true});let l=new Set,I=[];for(let D of i){let x=path.join(D.dir,"tests");if(fs.existsSync(x)){let f=fs.readdirSync(x);for(let h of f)h.endsWith(".json")&&fs.cpSync(path.join(x,h),path.join(m,h));}for(let f of D.index)l.has(f.id)||(l.add(f.id),I.push(f));}fs.writeFileSync(path.join(r,"index.json"),JSON.stringify(I),"utf-8");}let A=[...e.map(t=>t.summary),...i.map(t=>t.summary)],g=L(A,n.length),c=e.reduce((t,r)=>r.startedAt<t?r.startedAt:t,e[0]?.startedAt??new Date().toISOString()),u=e.reduce((t,r)=>r.completedAt>t?r.completedAt:t,e[0]?.completedAt??new Date().toISOString()),R=new Date(u).getTime()-new Date(c).getTime(),p={schemaVersion:e[0]?.schemaVersion??"1.0.0",testRunId:o.testRunId??crypto.randomUUID(),startedAt:c,completedAt:u,totalDuration:R,summary:g,ci:e.find(t=>t.ci!==null)?.ci??null,metadata:e.find(t=>t.metadata!==null)?.metadata??null,timeline:n,shardRunIds:d},E=path.dirname(o.output);return fs.mkdirSync(E,{recursive:true}),fs.writeFileSync(o.output,JSON.stringify(p,null,2),"utf-8"),p}function L(s,o){let e=0,i=0,d=0,n=0,A=0,g=0,c=0,u=0,R=0,p=0,E=0;for(let t of s)e+=t.total,i+=t.passed,d+=t.failed,n+=t.flaky,A+=t.skipped,g+=t.timedout??0,c+=t.totalApiCalls??0,u+=t.totalAssertions??0,R+=t.passedAssertions??0,p+=t.failedAssertions??0,E+=t.totalNavigations??0;return {total:e,passed:i,failed:d,flaky:n,skipped:A,timedout:g,totalApiCalls:c,uniqueApiUrls:0,apiCallsByMethod:{},apiCallsByStatusRange:{"2xx":0,"3xx":0,"4xx":0,"5xx":0,error:0},apiResponseTime:null,totalAssertions:u,passedAssertions:R,failedAssertions:p,totalNavigations:E,uniqueNavigationUrls:0,totalTimelineSteps:o,totalActionSteps:0,actionStepsByCategory:{}}}exports.mergeReports=U;//# sourceMappingURL=merge.cjs.map
|
|
2
2
|
//# sourceMappingURL=merge.cjs.map
|
package/dist/merge.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/merge.ts"],"names":["mergeReports","files","options","reports","file","raw","readFileSync","err","createError","ErrorCode","parsed","isValidTestRunReport","shardRunIds","r","allTimelines","report","a","b","timeA","timeB","summary","recalculateSummary","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","dir","dirname","mkdirSync","writeFileSync","timelineLength","total","passed","failed","flaky","skipped","timedout","totalApiCalls","totalAssertions","passedAssertions","failedAssertions","totalNavigations","apiUrlSet","navUrlSet","methodCounts","statusRanges"],"mappings":"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","file":"merge.cjs","sourcesContent":["/**\n * @testrelic/playwright-analytics/merge\n *\n * Merges multiple shard report files into a single unified timeline.\n * Tree-shakeable: separate entry point from the main reporter.\n */\n\nimport { randomUUID } from 'node:crypto';\nimport { readFileSync, writeFileSync, mkdirSync } from 'node:fs';\nimport { dirname } from 'node:path';\nimport type {\n TestRunReport,\n MergeOptions,\n Summary,\n TimelineEntry,\n TimelineStep,\n} from '@testrelic/core';\nimport { isValidTestRunReport, createError, ErrorCode } from '@testrelic/core';\n\nexport async function mergeReports(\n files: string[],\n options: MergeOptions,\n): Promise<TestRunReport> {\n const reports: TestRunReport[] = [];\n\n for (const file of files) {\n let raw: string;\n try {\n raw = readFileSync(file, 'utf-8');\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_READ_FAILED,\n `Failed to read file: ${file}`,\n err,\n );\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(raw);\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid JSON in file: ${file}`,\n err,\n );\n }\n\n if (!isValidTestRunReport(parsed)) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid report schema in file: ${file}`,\n );\n }\n\n reports.push(parsed);\n }\n\n // Collect shard IDs\n const shardRunIds = reports.map((r) => r.testRunId);\n\n // Merge timelines chronologically\n const allTimelines: (TimelineEntry | TimelineStep)[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) => {\n const timeA = 'visitedAt' in a ? a.visitedAt : a.timestamp;\n const timeB = 'visitedAt' in b ? b.visitedAt : b.timestamp;\n return new Date(timeA).getTime() - new Date(timeB).getTime();\n });\n\n // Recalculate summary across all shards\n const summary = recalculateSummary(reports, allTimelines.length);\n\n // Compute timing\n const startedAt = reports.reduce(\n (earliest, r) => (r.startedAt < earliest ? r.startedAt : earliest),\n reports[0]?.startedAt ?? new Date().toISOString(),\n );\n const completedAt = reports.reduce(\n (latest, r) => (r.completedAt > latest ? r.completedAt : latest),\n reports[0]?.completedAt ?? new Date().toISOString(),\n );\n const totalDuration = new Date(completedAt).getTime() - new Date(startedAt).getTime();\n\n const merged: TestRunReport = {\n schemaVersion: reports[0]?.schemaVersion ?? '1.0.0',\n testRunId: options.testRunId ?? randomUUID(),\n startedAt,\n completedAt,\n totalDuration,\n summary,\n ci: reports.find((r) => r.ci !== null)?.ci ?? null,\n metadata: reports.find((r) => r.metadata !== null)?.metadata ?? null,\n timeline: allTimelines,\n shardRunIds,\n };\n\n // Write to disk\n const dir = dirname(options.output);\n mkdirSync(dir, { recursive: true });\n writeFileSync(options.output, JSON.stringify(merged, null, 2), 'utf-8');\n\n return merged;\n}\n\nfunction recalculateSummary(reports: TestRunReport[], timelineLength: number): Summary {\n let total = 0;\n let passed = 0;\n let failed = 0;\n let flaky = 0;\n let skipped = 0;\n let timedout = 0;\n let totalApiCalls = 0;\n let totalAssertions = 0;\n let passedAssertions = 0;\n let failedAssertions = 0;\n let totalNavigations = 0;\n const apiUrlSet = new Set<string>();\n const navUrlSet = new Set<string>();\n const methodCounts: Record<string, number> = {};\n const statusRanges = { '2xx': 0, '3xx': 0, '4xx': 0, '5xx': 0, error: 0 };\n const allResponseTimes: number[] = [];\n\n for (const report of reports) {\n total += report.summary.total;\n passed += report.summary.passed;\n failed += report.summary.failed;\n flaky += report.summary.flaky;\n skipped += report.summary.skipped;\n timedout += report.summary.timedout ?? 0;\n totalApiCalls += report.summary.totalApiCalls ?? 0;\n totalAssertions += report.summary.totalAssertions ?? 0;\n passedAssertions += report.summary.passedAssertions ?? 0;\n failedAssertions += report.summary.failedAssertions ?? 0;\n totalNavigations += report.summary.totalNavigations ?? 0;\n }\n\n return {\n total, passed, failed, flaky, skipped, timedout,\n totalApiCalls, uniqueApiUrls: apiUrlSet.size,\n apiCallsByMethod: methodCounts,\n apiCallsByStatusRange: statusRanges,\n apiResponseTime: null,\n totalAssertions, passedAssertions, failedAssertions,\n totalNavigations, uniqueNavigationUrls: navUrlSet.size,\n totalTimelineSteps: timelineLength,\n totalActionSteps: 0,\n actionStepsByCategory: {},\n };\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/merge.ts"],"names":["isStreamingDir","path","statSync","existsSync","join","loadStreamingReport","dir","summaryPath","indexPath","summary","readFileSync","err","createError","ErrorCode","index","mergeReports","files","options","reports","streamingReports","file","raw","parsed","isValidTestRunReport","shardRunIds","r","allTimelines","report","a","b","timeA","timeB","outputDir","dirname","mergedReportDir","mergedTestsDir","mkdirSync","seenTestIds","mergedIndex","sr","srcTestsDir","testFiles","readdirSync","f","cpSync","entry","writeFileSync","allSummaries","recalculateSummaryFromList","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","summaries","timelineLength","total","passed","failed","flaky","skipped","timedout","totalApiCalls","totalAssertions","passedAssertions","failedAssertions","totalNavigations","s"],"mappings":"gHAuBA,SAASA,CAAAA,CAAeC,CAAAA,CAAuB,CAC7C,GAAI,CAEF,OADaC,WAAAA,CAASD,CAAI,CAAA,CACd,WAAA,EAAY,EAAKE,aAAAA,CAAWC,UAAKH,CAAAA,CAAM,YAAY,CAAC,CAClE,MAAQ,CACN,OAAO,MACT,CACF,CAKA,SAASI,CAAAA,CAAoBC,CAAAA,CAAyE,CACpG,IAAMC,CAAAA,CAAcH,SAAAA,CAAKE,CAAAA,CAAK,cAAc,EACtCE,CAAAA,CAAYJ,SAAAA,CAAKE,CAAAA,CAAK,YAAY,EAEpCG,CAAAA,CACJ,GAAI,CACFA,CAAAA,CAAU,IAAA,CAAK,MAAMC,eAAAA,CAAaH,CAAAA,CAAa,OAAO,CAAC,EACzD,CAAA,MAASI,CAAAA,CAAK,CACZ,MAAMC,iBAAYC,cAAAA,CAAU,iBAAA,CAAmB,CAAA,kCAAA,EAAqCP,CAAG,GAAIK,CAAG,CAChG,CAEA,IAAIG,EACJ,GAAI,CACFA,CAAAA,CAAQ,IAAA,CAAK,MAAMJ,eAAAA,CAAaF,CAAAA,CAAW,OAAO,CAAC,EACrD,CAAA,MAASG,CAAAA,CAAK,CACZ,MAAMC,iBAAYC,cAAAA,CAAU,iBAAA,CAAmB,mCAAmCP,CAAG,CAAA,CAAA,CAAIK,CAAG,CAC9F,CAEA,OAAO,CAAE,QAAAF,CAAAA,CAAS,KAAA,CAAAK,CAAAA,CAAO,GAAA,CAAAR,CAAI,CAC/B,CAEA,eAAsBS,CAAAA,CACpBC,EACAC,CAAAA,CACwB,CACxB,IAAMC,CAAAA,CAA2B,GAC3BC,CAAAA,CAAsF,EAAC,CAE7F,IAAA,IAAWC,KAAQJ,CAAAA,CAAO,CAExB,GAAIhB,CAAAA,CAAeoB,CAAI,CAAA,CAAG,CACxBD,CAAAA,CAAiB,IAAA,CAAKd,EAAoBe,CAAI,CAAC,EAC/C,QACF,CAEA,IAAIC,CAAAA,CACJ,GAAI,CACFA,CAAAA,CAAMX,gBAAaU,CAAAA,CAAM,OAAO,EAClC,CAAA,MAAST,EAAK,CACZ,MAAMC,gBAAAA,CACJC,cAAAA,CAAU,kBACV,CAAA,qBAAA,EAAwBO,CAAI,CAAA,CAAA,CAC5BT,CACF,CACF,CAEA,IAAIW,CAAAA,CACJ,GAAI,CACFA,CAAAA,CAAS,IAAA,CAAK,KAAA,CAAMD,CAAG,EACzB,CAAA,MAASV,CAAAA,CAAK,CACZ,MAAMC,iBACJC,cAAAA,CAAU,oBAAA,CACV,yBAAyBO,CAAI,CAAA,CAAA,CAC7BT,CACF,CACF,CAEA,GAAI,CAACY,0BAAqBD,CAAM,CAAA,CAC9B,MAAMV,gBAAAA,CACJC,eAAU,oBAAA,CACV,CAAA,+BAAA,EAAkCO,CAAI,CAAA,CACxC,EAGFF,CAAAA,CAAQ,IAAA,CAAKI,CAAM,EACrB,CAGA,IAAME,CAAAA,CAAcN,CAAAA,CAAQ,GAAA,CAAKO,CAAAA,EAAMA,EAAE,SAAS,CAAA,CAG5CC,CAAAA,CAAiD,GACvD,IAAA,IAAWC,CAAAA,IAAUT,CAAAA,CACnBQ,CAAAA,CAAa,KAAK,GAAGC,CAAAA,CAAO,QAAQ,CAAA,CAStC,GAPAD,EAAa,IAAA,CAAK,CAACE,CAAAA,CAAGC,CAAAA,GAAM,CAC1B,IAAMC,CAAAA,CAAQ,WAAA,GAAeF,CAAAA,CAAIA,EAAE,SAAA,CAAYA,CAAAA,CAAE,SAAA,CAC3CG,CAAAA,CAAQ,cAAeF,CAAAA,CAAIA,CAAAA,CAAE,SAAA,CAAYA,CAAAA,CAAE,UACjD,OAAO,IAAI,IAAA,CAAKC,CAAK,EAAE,OAAA,EAAQ,CAAI,IAAI,IAAA,CAAKC,CAAK,CAAA,CAAE,OAAA,EACrD,CAAC,EAGGZ,CAAAA,CAAiB,MAAA,CAAS,EAAG,CAC/B,IAAMa,EAAYC,YAAAA,CAAQhB,CAAAA,CAAQ,MAAM,CAAA,CAClCiB,EAAkB9B,SAAAA,CAAK4B,CAAAA,CAAW,mBAAmB,CAAA,CACrDG,EAAiB/B,SAAAA,CAAK8B,CAAAA,CAAiB,OAAO,CAAA,CACpDE,aAAUD,CAAAA,CAAgB,CAAE,UAAW,IAAK,CAAC,EAE7C,IAAME,CAAAA,CAAc,IAAI,GAAA,CAClBC,EAAgC,EAAC,CAEvC,IAAA,IAAWC,CAAAA,IAAMpB,EAAkB,CAEjC,IAAMqB,CAAAA,CAAcpC,SAAAA,CAAKmC,EAAG,GAAA,CAAK,OAAO,EACxC,GAAIpC,aAAAA,CAAWqC,CAAW,CAAA,CAAG,CAC3B,IAAMC,CAAAA,CAAYC,eAAYF,CAAW,CAAA,CACzC,IAAA,IAAWG,CAAAA,IAAKF,EACVE,CAAAA,CAAE,QAAA,CAAS,OAAO,CAAA,EACpBC,UAAOxC,SAAAA,CAAKoC,CAAAA,CAAaG,CAAC,CAAA,CAAGvC,UAAK+B,CAAAA,CAAgBQ,CAAC,CAAC,EAG1D,CAGA,IAAA,IAAWE,CAAAA,IAASN,CAAAA,CAAG,KAAA,CAChBF,EAAY,GAAA,CAAIQ,CAAAA,CAAM,EAAE,CAAA,GAC3BR,EAAY,GAAA,CAAIQ,CAAAA,CAAM,EAAE,CAAA,CACxBP,CAAAA,CAAY,KAAKO,CAAK,CAAA,EAG5B,CAGAC,gBAAAA,CAAc1C,UAAK8B,CAAAA,CAAiB,YAAY,CAAA,CAAG,IAAA,CAAK,UAAUI,CAAW,CAAA,CAAG,OAAO,EACzF,CAGA,IAAMS,CAAAA,CAAe,CACnB,GAAG7B,CAAAA,CAAQ,IAAKO,CAAAA,EAAMA,CAAAA,CAAE,OAAO,CAAA,CAC/B,GAAGN,CAAAA,CAAiB,GAAA,CAAKoB,CAAAA,EAAOA,CAAAA,CAAG,OAAO,CAC5C,CAAA,CACM9B,CAAAA,CAAUuC,CAAAA,CAA2BD,EAAcrB,CAAAA,CAAa,MAAM,EAGtEuB,CAAAA,CAAY/B,CAAAA,CAAQ,OACxB,CAACgC,CAAAA,CAAU,CAAA,GAAO,CAAA,CAAE,UAAYA,CAAAA,CAAW,CAAA,CAAE,SAAA,CAAYA,CAAAA,CACzDhC,EAAQ,CAAC,CAAA,EAAG,SAAA,EAAa,IAAI,MAAK,CAAE,WAAA,EACtC,CAAA,CACMiC,EAAcjC,CAAAA,CAAQ,MAAA,CAC1B,CAACkC,CAAAA,CAAQ,IAAO,CAAA,CAAE,WAAA,CAAcA,CAAAA,CAAS,CAAA,CAAE,YAAcA,CAAAA,CACzDlC,CAAAA,CAAQ,CAAC,CAAA,EAAG,aAAe,IAAI,IAAA,GAAO,WAAA,EACxC,EACMmC,CAAAA,CAAgB,IAAI,IAAA,CAAKF,CAAW,EAAE,OAAA,EAAQ,CAAI,IAAI,IAAA,CAAKF,CAAS,CAAA,CAAE,OAAA,EAAQ,CAE9EK,CAAAA,CAAwB,CAC5B,aAAA,CAAepC,CAAAA,CAAQ,CAAC,CAAA,EAAG,aAAA,EAAiB,QAC5C,SAAA,CAAWD,CAAAA,CAAQ,SAAA,EAAasC,iBAAAA,GAChC,SAAA,CAAAN,CAAAA,CACA,WAAA,CAAAE,CAAAA,CACA,cAAAE,CAAAA,CACA,OAAA,CAAA5C,CAAAA,CACA,EAAA,CAAIS,EAAQ,IAAA,CAAMO,CAAAA,EAAMA,EAAE,EAAA,GAAO,IAAI,GAAG,EAAA,EAAM,IAAA,CAC9C,QAAA,CAAUP,CAAAA,CAAQ,KAAMO,CAAAA,EAAMA,CAAAA,CAAE,QAAA,GAAa,IAAI,GAAG,QAAA,EAAY,IAAA,CAChE,QAAA,CAAUC,CAAAA,CACV,YAAAF,CACF,CAAA,CAGMlB,CAAAA,CAAM2B,YAAAA,CAAQhB,EAAQ,MAAM,CAAA,CAClC,OAAAmB,YAAAA,CAAU9B,EAAK,CAAE,SAAA,CAAW,IAAK,CAAC,EAClCwC,gBAAAA,CAAc7B,CAAAA,CAAQ,MAAA,CAAQ,IAAA,CAAK,UAAUqC,CAAAA,CAAQ,IAAA,CAAM,CAAC,CAAA,CAAG,OAAO,EAE/DA,CACT,CAEA,SAASN,CAAAA,CAA2BQ,EAAsBC,CAAAA,CAAiC,CACzF,IAAIC,CAAAA,CAAQ,EACRC,CAAAA,CAAS,CAAA,CACTC,CAAAA,CAAS,CAAA,CACTC,EAAQ,CAAA,CACRC,CAAAA,CAAU,EACVC,CAAAA,CAAW,CAAA,CACXC,EAAgB,CAAA,CAChBC,CAAAA,CAAkB,CAAA,CAClBC,CAAAA,CAAmB,EACnBC,CAAAA,CAAmB,CAAA,CACnBC,CAAAA,CAAmB,CAAA,CAEvB,QAAWC,CAAAA,IAAKb,CAAAA,CACdE,CAAAA,EAASW,CAAAA,CAAE,MACXV,CAAAA,EAAUU,CAAAA,CAAE,OACZT,CAAAA,EAAUS,CAAAA,CAAE,OACZR,CAAAA,EAASQ,CAAAA,CAAE,KAAA,CACXP,CAAAA,EAAWO,EAAE,OAAA,CACbN,CAAAA,EAAYM,CAAAA,CAAE,QAAA,EAAY,EAC1BL,CAAAA,EAAiBK,CAAAA,CAAE,aAAA,EAAiB,CAAA,CACpCJ,GAAmBI,CAAAA,CAAE,eAAA,EAAmB,CAAA,CACxCH,CAAAA,EAAoBG,EAAE,gBAAA,EAAoB,CAAA,CAC1CF,CAAAA,EAAoBE,CAAAA,CAAE,kBAAoB,CAAA,CAC1CD,CAAAA,EAAoBC,CAAAA,CAAE,gBAAA,EAAoB,EAG5C,OAAO,CACL,KAAA,CAAAX,CAAAA,CAAO,OAAAC,CAAAA,CAAQ,MAAA,CAAAC,EAAQ,KAAA,CAAAC,CAAAA,CAAO,QAAAC,CAAAA,CAAS,QAAA,CAAAC,CAAAA,CACvC,aAAA,CAAAC,EAAe,aAAA,CAAe,CAAA,CAC9B,gBAAA,CAAkB,GAClB,qBAAA,CAAuB,CAAE,KAAA,CAAO,CAAA,CAAG,MAAO,CAAA,CAAG,KAAA,CAAO,EAAG,KAAA,CAAO,CAAA,CAAG,MAAO,CAAE,CAAA,CAC1E,eAAA,CAAiB,IAAA,CACjB,gBAAAC,CAAAA,CAAiB,gBAAA,CAAAC,CAAAA,CAAkB,gBAAA,CAAAC,EACnC,gBAAA,CAAAC,CAAAA,CAAkB,oBAAA,CAAsB,CAAA,CACxC,mBAAoBX,CAAAA,CACpB,gBAAA,CAAkB,EAClB,qBAAA,CAAuB,EACzB,CACF","file":"merge.cjs","sourcesContent":["/**\n * @testrelic/playwright-analytics/merge\n *\n * Merges multiple shard report files into a single unified timeline.\n * Tree-shakeable: separate entry point from the main reporter.\n */\n\nimport { randomUUID } from 'node:crypto';\nimport { readFileSync, writeFileSync, mkdirSync, existsSync, statSync, readdirSync, cpSync } from 'node:fs';\nimport { dirname, join } from 'node:path';\nimport type {\n TestRunReport,\n MergeOptions,\n Summary,\n TimelineEntry,\n TimelineStep,\n TestIndexEntry,\n} from '@testrelic/core';\nimport { isValidTestRunReport, createError, ErrorCode } from '@testrelic/core';\n\n/**\n * Check if a path is a streaming report directory (contains index.json).\n */\nfunction isStreamingDir(path: string): boolean {\n try {\n const stat = statSync(path);\n return stat.isDirectory() && existsSync(join(path, 'index.json'));\n } catch {\n return false;\n }\n}\n\n/**\n * Load a streaming report directory as a TestRunReport with summary data.\n */\nfunction loadStreamingReport(dir: string): { summary: Summary; index: TestIndexEntry[]; dir: string } {\n const summaryPath = join(dir, 'summary.json');\n const indexPath = join(dir, 'index.json');\n\n let summary: Summary;\n try {\n summary = JSON.parse(readFileSync(summaryPath, 'utf-8'));\n } catch (err) {\n throw createError(ErrorCode.MERGE_READ_FAILED, `Failed to read summary.json from: ${dir}`, err);\n }\n\n let index: TestIndexEntry[];\n try {\n index = JSON.parse(readFileSync(indexPath, 'utf-8'));\n } catch (err) {\n throw createError(ErrorCode.MERGE_READ_FAILED, `Failed to read index.json from: ${dir}`, err);\n }\n\n return { summary, index, dir };\n}\n\nexport async function mergeReports(\n files: string[],\n options: MergeOptions,\n): Promise<TestRunReport> {\n const reports: TestRunReport[] = [];\n const streamingReports: Array<{ summary: Summary; index: TestIndexEntry[]; dir: string }> = [];\n\n for (const file of files) {\n // Check if this is a streaming report directory\n if (isStreamingDir(file)) {\n streamingReports.push(loadStreamingReport(file));\n continue;\n }\n\n let raw: string;\n try {\n raw = readFileSync(file, 'utf-8');\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_READ_FAILED,\n `Failed to read file: ${file}`,\n err,\n );\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(raw);\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid JSON in file: ${file}`,\n err,\n );\n }\n\n if (!isValidTestRunReport(parsed)) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid report schema in file: ${file}`,\n );\n }\n\n reports.push(parsed);\n }\n\n // Collect shard IDs\n const shardRunIds = reports.map((r) => r.testRunId);\n\n // Merge timelines chronologically\n const allTimelines: (TimelineEntry | TimelineStep)[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) => {\n const timeA = 'visitedAt' in a ? a.visitedAt : a.timestamp;\n const timeB = 'visitedAt' in b ? b.visitedAt : b.timestamp;\n return new Date(timeA).getTime() - new Date(timeB).getTime();\n });\n\n // Merge streaming report data: copy test detail files and merge indexes\n if (streamingReports.length > 0) {\n const outputDir = dirname(options.output);\n const mergedReportDir = join(outputDir, '.testrelic-report');\n const mergedTestsDir = join(mergedReportDir, 'tests');\n mkdirSync(mergedTestsDir, { recursive: true });\n\n const seenTestIds = new Set<string>();\n const mergedIndex: TestIndexEntry[] = [];\n\n for (const sr of streamingReports) {\n // Copy test detail files\n const srcTestsDir = join(sr.dir, 'tests');\n if (existsSync(srcTestsDir)) {\n const testFiles = readdirSync(srcTestsDir);\n for (const f of testFiles) {\n if (f.endsWith('.json')) {\n cpSync(join(srcTestsDir, f), join(mergedTestsDir, f));\n }\n }\n }\n\n // Merge index entries (dedup by id)\n for (const entry of sr.index) {\n if (!seenTestIds.has(entry.id)) {\n seenTestIds.add(entry.id);\n mergedIndex.push(entry);\n }\n }\n }\n\n // Write merged index and summary\n writeFileSync(join(mergedReportDir, 'index.json'), JSON.stringify(mergedIndex), 'utf-8');\n }\n\n // Recalculate summary across all shards (including streaming)\n const allSummaries = [\n ...reports.map((r) => r.summary),\n ...streamingReports.map((sr) => sr.summary),\n ];\n const summary = recalculateSummaryFromList(allSummaries, allTimelines.length);\n\n // Compute timing\n const startedAt = reports.reduce(\n (earliest, r) => (r.startedAt < earliest ? r.startedAt : earliest),\n reports[0]?.startedAt ?? new Date().toISOString(),\n );\n const completedAt = reports.reduce(\n (latest, r) => (r.completedAt > latest ? r.completedAt : latest),\n reports[0]?.completedAt ?? new Date().toISOString(),\n );\n const totalDuration = new Date(completedAt).getTime() - new Date(startedAt).getTime();\n\n const merged: TestRunReport = {\n schemaVersion: reports[0]?.schemaVersion ?? '1.0.0',\n testRunId: options.testRunId ?? randomUUID(),\n startedAt,\n completedAt,\n totalDuration,\n summary,\n ci: reports.find((r) => r.ci !== null)?.ci ?? null,\n metadata: reports.find((r) => r.metadata !== null)?.metadata ?? null,\n timeline: allTimelines,\n shardRunIds,\n };\n\n // Write to disk\n const dir = dirname(options.output);\n mkdirSync(dir, { recursive: true });\n writeFileSync(options.output, JSON.stringify(merged, null, 2), 'utf-8');\n\n return merged;\n}\n\nfunction recalculateSummaryFromList(summaries: Summary[], timelineLength: number): Summary {\n let total = 0;\n let passed = 0;\n let failed = 0;\n let flaky = 0;\n let skipped = 0;\n let timedout = 0;\n let totalApiCalls = 0;\n let totalAssertions = 0;\n let passedAssertions = 0;\n let failedAssertions = 0;\n let totalNavigations = 0;\n\n for (const s of summaries) {\n total += s.total;\n passed += s.passed;\n failed += s.failed;\n flaky += s.flaky;\n skipped += s.skipped;\n timedout += s.timedout ?? 0;\n totalApiCalls += s.totalApiCalls ?? 0;\n totalAssertions += s.totalAssertions ?? 0;\n passedAssertions += s.passedAssertions ?? 0;\n failedAssertions += s.failedAssertions ?? 0;\n totalNavigations += s.totalNavigations ?? 0;\n }\n\n return {\n total, passed, failed, flaky, skipped, timedout,\n totalApiCalls, uniqueApiUrls: 0,\n apiCallsByMethod: {},\n apiCallsByStatusRange: { '2xx': 0, '3xx': 0, '4xx': 0, '5xx': 0, error: 0 },\n apiResponseTime: null,\n totalAssertions, passedAssertions, failedAssertions,\n totalNavigations, uniqueNavigationUrls: 0,\n totalTimelineSteps: timelineLength,\n totalActionSteps: 0,\n actionStepsByCategory: {},\n };\n}\n"]}
|
package/dist/merge.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import {randomUUID}from'crypto';import {readFileSync,mkdirSync,writeFileSync}from'fs';import {dirname}from'path';import {createError,ErrorCode,isValidTestRunReport}from'@testrelic/core';
|
|
1
|
+
import {randomUUID}from'crypto';import {readFileSync,mkdirSync,existsSync,readdirSync,cpSync,writeFileSync,statSync}from'fs';import {dirname,join}from'path';import {createError,ErrorCode,isValidTestRunReport}from'@testrelic/core';function k(s){try{return statSync(s).isDirectory()&&existsSync(join(s,"index.json"))}catch{return false}}function J(s){let o=join(s,"summary.json"),e=join(s,"index.json"),i;try{i=JSON.parse(readFileSync(o,"utf-8"));}catch(n){throw createError(ErrorCode.MERGE_READ_FAILED,`Failed to read summary.json from: ${s}`,n)}let d;try{d=JSON.parse(readFileSync(e,"utf-8"));}catch(n){throw createError(ErrorCode.MERGE_READ_FAILED,`Failed to read index.json from: ${s}`,n)}return {summary:i,index:d,dir:s}}async function U(s,o){let e=[],i=[];for(let t of s){if(k(t)){i.push(J(t));continue}let r;try{r=readFileSync(t,"utf-8");}catch(l){throw createError(ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,l)}let m;try{m=JSON.parse(r);}catch(l){throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,l)}if(!isValidTestRunReport(m))throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);e.push(m);}let d=e.map(t=>t.testRunId),n=[];for(let t of e)n.push(...t.timeline);if(n.sort((t,r)=>{let m="visitedAt"in t?t.visitedAt:t.timestamp,l="visitedAt"in r?r.visitedAt:r.timestamp;return new Date(m).getTime()-new Date(l).getTime()}),i.length>0){let t=dirname(o.output),r=join(t,".testrelic-report"),m=join(r,"tests");mkdirSync(m,{recursive:true});let l=new Set,I=[];for(let D of i){let x=join(D.dir,"tests");if(existsSync(x)){let f=readdirSync(x);for(let h of f)h.endsWith(".json")&&cpSync(join(x,h),join(m,h));}for(let f of D.index)l.has(f.id)||(l.add(f.id),I.push(f));}writeFileSync(join(r,"index.json"),JSON.stringify(I),"utf-8");}let A=[...e.map(t=>t.summary),...i.map(t=>t.summary)],g=L(A,n.length),c=e.reduce((t,r)=>r.startedAt<t?r.startedAt:t,e[0]?.startedAt??new Date().toISOString()),u=e.reduce((t,r)=>r.completedAt>t?r.completedAt:t,e[0]?.completedAt??new Date().toISOString()),R=new Date(u).getTime()-new Date(c).getTime(),p={schemaVersion:e[0]?.schemaVersion??"1.0.0",testRunId:o.testRunId??randomUUID(),startedAt:c,completedAt:u,totalDuration:R,summary:g,ci:e.find(t=>t.ci!==null)?.ci??null,metadata:e.find(t=>t.metadata!==null)?.metadata??null,timeline:n,shardRunIds:d},E=dirname(o.output);return mkdirSync(E,{recursive:true}),writeFileSync(o.output,JSON.stringify(p,null,2),"utf-8"),p}function L(s,o){let e=0,i=0,d=0,n=0,A=0,g=0,c=0,u=0,R=0,p=0,E=0;for(let t of s)e+=t.total,i+=t.passed,d+=t.failed,n+=t.flaky,A+=t.skipped,g+=t.timedout??0,c+=t.totalApiCalls??0,u+=t.totalAssertions??0,R+=t.passedAssertions??0,p+=t.failedAssertions??0,E+=t.totalNavigations??0;return {total:e,passed:i,failed:d,flaky:n,skipped:A,timedout:g,totalApiCalls:c,uniqueApiUrls:0,apiCallsByMethod:{},apiCallsByStatusRange:{"2xx":0,"3xx":0,"4xx":0,"5xx":0,error:0},apiResponseTime:null,totalAssertions:u,passedAssertions:R,failedAssertions:p,totalNavigations:E,uniqueNavigationUrls:0,totalTimelineSteps:o,totalActionSteps:0,actionStepsByCategory:{}}}export{U as mergeReports};//# sourceMappingURL=merge.js.map
|
|
2
2
|
//# sourceMappingURL=merge.js.map
|
package/dist/merge.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/merge.ts"],"names":["mergeReports","files","options","reports","file","raw","readFileSync","err","createError","ErrorCode","parsed","isValidTestRunReport","shardRunIds","r","allTimelines","report","a","b","timeA","timeB","summary","recalculateSummary","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","dir","dirname","mkdirSync","writeFileSync","timelineLength","total","passed","failed","flaky","skipped","timedout","totalApiCalls","totalAssertions","passedAssertions","failedAssertions","totalNavigations","apiUrlSet","navUrlSet","methodCounts","statusRanges"],"mappings":"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","file":"merge.js","sourcesContent":["/**\n * @testrelic/playwright-analytics/merge\n *\n * Merges multiple shard report files into a single unified timeline.\n * Tree-shakeable: separate entry point from the main reporter.\n */\n\nimport { randomUUID } from 'node:crypto';\nimport { readFileSync, writeFileSync, mkdirSync } from 'node:fs';\nimport { dirname } from 'node:path';\nimport type {\n TestRunReport,\n MergeOptions,\n Summary,\n TimelineEntry,\n TimelineStep,\n} from '@testrelic/core';\nimport { isValidTestRunReport, createError, ErrorCode } from '@testrelic/core';\n\nexport async function mergeReports(\n files: string[],\n options: MergeOptions,\n): Promise<TestRunReport> {\n const reports: TestRunReport[] = [];\n\n for (const file of files) {\n let raw: string;\n try {\n raw = readFileSync(file, 'utf-8');\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_READ_FAILED,\n `Failed to read file: ${file}`,\n err,\n );\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(raw);\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid JSON in file: ${file}`,\n err,\n );\n }\n\n if (!isValidTestRunReport(parsed)) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid report schema in file: ${file}`,\n );\n }\n\n reports.push(parsed);\n }\n\n // Collect shard IDs\n const shardRunIds = reports.map((r) => r.testRunId);\n\n // Merge timelines chronologically\n const allTimelines: (TimelineEntry | TimelineStep)[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) => {\n const timeA = 'visitedAt' in a ? a.visitedAt : a.timestamp;\n const timeB = 'visitedAt' in b ? b.visitedAt : b.timestamp;\n return new Date(timeA).getTime() - new Date(timeB).getTime();\n });\n\n // Recalculate summary across all shards\n const summary = recalculateSummary(reports, allTimelines.length);\n\n // Compute timing\n const startedAt = reports.reduce(\n (earliest, r) => (r.startedAt < earliest ? r.startedAt : earliest),\n reports[0]?.startedAt ?? new Date().toISOString(),\n );\n const completedAt = reports.reduce(\n (latest, r) => (r.completedAt > latest ? r.completedAt : latest),\n reports[0]?.completedAt ?? new Date().toISOString(),\n );\n const totalDuration = new Date(completedAt).getTime() - new Date(startedAt).getTime();\n\n const merged: TestRunReport = {\n schemaVersion: reports[0]?.schemaVersion ?? '1.0.0',\n testRunId: options.testRunId ?? randomUUID(),\n startedAt,\n completedAt,\n totalDuration,\n summary,\n ci: reports.find((r) => r.ci !== null)?.ci ?? null,\n metadata: reports.find((r) => r.metadata !== null)?.metadata ?? null,\n timeline: allTimelines,\n shardRunIds,\n };\n\n // Write to disk\n const dir = dirname(options.output);\n mkdirSync(dir, { recursive: true });\n writeFileSync(options.output, JSON.stringify(merged, null, 2), 'utf-8');\n\n return merged;\n}\n\nfunction recalculateSummary(reports: TestRunReport[], timelineLength: number): Summary {\n let total = 0;\n let passed = 0;\n let failed = 0;\n let flaky = 0;\n let skipped = 0;\n let timedout = 0;\n let totalApiCalls = 0;\n let totalAssertions = 0;\n let passedAssertions = 0;\n let failedAssertions = 0;\n let totalNavigations = 0;\n const apiUrlSet = new Set<string>();\n const navUrlSet = new Set<string>();\n const methodCounts: Record<string, number> = {};\n const statusRanges = { '2xx': 0, '3xx': 0, '4xx': 0, '5xx': 0, error: 0 };\n const allResponseTimes: number[] = [];\n\n for (const report of reports) {\n total += report.summary.total;\n passed += report.summary.passed;\n failed += report.summary.failed;\n flaky += report.summary.flaky;\n skipped += report.summary.skipped;\n timedout += report.summary.timedout ?? 0;\n totalApiCalls += report.summary.totalApiCalls ?? 0;\n totalAssertions += report.summary.totalAssertions ?? 0;\n passedAssertions += report.summary.passedAssertions ?? 0;\n failedAssertions += report.summary.failedAssertions ?? 0;\n totalNavigations += report.summary.totalNavigations ?? 0;\n }\n\n return {\n total, passed, failed, flaky, skipped, timedout,\n totalApiCalls, uniqueApiUrls: apiUrlSet.size,\n apiCallsByMethod: methodCounts,\n apiCallsByStatusRange: statusRanges,\n apiResponseTime: null,\n totalAssertions, passedAssertions, failedAssertions,\n totalNavigations, uniqueNavigationUrls: navUrlSet.size,\n totalTimelineSteps: timelineLength,\n totalActionSteps: 0,\n actionStepsByCategory: {},\n };\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/merge.ts"],"names":["isStreamingDir","path","statSync","existsSync","join","loadStreamingReport","dir","summaryPath","indexPath","summary","readFileSync","err","createError","ErrorCode","index","mergeReports","files","options","reports","streamingReports","file","raw","parsed","isValidTestRunReport","shardRunIds","r","allTimelines","report","a","b","timeA","timeB","outputDir","dirname","mergedReportDir","mergedTestsDir","mkdirSync","seenTestIds","mergedIndex","sr","srcTestsDir","testFiles","readdirSync","f","cpSync","entry","writeFileSync","allSummaries","recalculateSummaryFromList","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","summaries","timelineLength","total","passed","failed","flaky","skipped","timedout","totalApiCalls","totalAssertions","passedAssertions","failedAssertions","totalNavigations","s"],"mappings":"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","file":"merge.js","sourcesContent":["/**\n * @testrelic/playwright-analytics/merge\n *\n * Merges multiple shard report files into a single unified timeline.\n * Tree-shakeable: separate entry point from the main reporter.\n */\n\nimport { randomUUID } from 'node:crypto';\nimport { readFileSync, writeFileSync, mkdirSync, existsSync, statSync, readdirSync, cpSync } from 'node:fs';\nimport { dirname, join } from 'node:path';\nimport type {\n TestRunReport,\n MergeOptions,\n Summary,\n TimelineEntry,\n TimelineStep,\n TestIndexEntry,\n} from '@testrelic/core';\nimport { isValidTestRunReport, createError, ErrorCode } from '@testrelic/core';\n\n/**\n * Check if a path is a streaming report directory (contains index.json).\n */\nfunction isStreamingDir(path: string): boolean {\n try {\n const stat = statSync(path);\n return stat.isDirectory() && existsSync(join(path, 'index.json'));\n } catch {\n return false;\n }\n}\n\n/**\n * Load a streaming report directory as a TestRunReport with summary data.\n */\nfunction loadStreamingReport(dir: string): { summary: Summary; index: TestIndexEntry[]; dir: string } {\n const summaryPath = join(dir, 'summary.json');\n const indexPath = join(dir, 'index.json');\n\n let summary: Summary;\n try {\n summary = JSON.parse(readFileSync(summaryPath, 'utf-8'));\n } catch (err) {\n throw createError(ErrorCode.MERGE_READ_FAILED, `Failed to read summary.json from: ${dir}`, err);\n }\n\n let index: TestIndexEntry[];\n try {\n index = JSON.parse(readFileSync(indexPath, 'utf-8'));\n } catch (err) {\n throw createError(ErrorCode.MERGE_READ_FAILED, `Failed to read index.json from: ${dir}`, err);\n }\n\n return { summary, index, dir };\n}\n\nexport async function mergeReports(\n files: string[],\n options: MergeOptions,\n): Promise<TestRunReport> {\n const reports: TestRunReport[] = [];\n const streamingReports: Array<{ summary: Summary; index: TestIndexEntry[]; dir: string }> = [];\n\n for (const file of files) {\n // Check if this is a streaming report directory\n if (isStreamingDir(file)) {\n streamingReports.push(loadStreamingReport(file));\n continue;\n }\n\n let raw: string;\n try {\n raw = readFileSync(file, 'utf-8');\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_READ_FAILED,\n `Failed to read file: ${file}`,\n err,\n );\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(raw);\n } catch (err) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid JSON in file: ${file}`,\n err,\n );\n }\n\n if (!isValidTestRunReport(parsed)) {\n throw createError(\n ErrorCode.MERGE_INVALID_SCHEMA,\n `Invalid report schema in file: ${file}`,\n );\n }\n\n reports.push(parsed);\n }\n\n // Collect shard IDs\n const shardRunIds = reports.map((r) => r.testRunId);\n\n // Merge timelines chronologically\n const allTimelines: (TimelineEntry | TimelineStep)[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) => {\n const timeA = 'visitedAt' in a ? a.visitedAt : a.timestamp;\n const timeB = 'visitedAt' in b ? b.visitedAt : b.timestamp;\n return new Date(timeA).getTime() - new Date(timeB).getTime();\n });\n\n // Merge streaming report data: copy test detail files and merge indexes\n if (streamingReports.length > 0) {\n const outputDir = dirname(options.output);\n const mergedReportDir = join(outputDir, '.testrelic-report');\n const mergedTestsDir = join(mergedReportDir, 'tests');\n mkdirSync(mergedTestsDir, { recursive: true });\n\n const seenTestIds = new Set<string>();\n const mergedIndex: TestIndexEntry[] = [];\n\n for (const sr of streamingReports) {\n // Copy test detail files\n const srcTestsDir = join(sr.dir, 'tests');\n if (existsSync(srcTestsDir)) {\n const testFiles = readdirSync(srcTestsDir);\n for (const f of testFiles) {\n if (f.endsWith('.json')) {\n cpSync(join(srcTestsDir, f), join(mergedTestsDir, f));\n }\n }\n }\n\n // Merge index entries (dedup by id)\n for (const entry of sr.index) {\n if (!seenTestIds.has(entry.id)) {\n seenTestIds.add(entry.id);\n mergedIndex.push(entry);\n }\n }\n }\n\n // Write merged index and summary\n writeFileSync(join(mergedReportDir, 'index.json'), JSON.stringify(mergedIndex), 'utf-8');\n }\n\n // Recalculate summary across all shards (including streaming)\n const allSummaries = [\n ...reports.map((r) => r.summary),\n ...streamingReports.map((sr) => sr.summary),\n ];\n const summary = recalculateSummaryFromList(allSummaries, allTimelines.length);\n\n // Compute timing\n const startedAt = reports.reduce(\n (earliest, r) => (r.startedAt < earliest ? r.startedAt : earliest),\n reports[0]?.startedAt ?? new Date().toISOString(),\n );\n const completedAt = reports.reduce(\n (latest, r) => (r.completedAt > latest ? r.completedAt : latest),\n reports[0]?.completedAt ?? new Date().toISOString(),\n );\n const totalDuration = new Date(completedAt).getTime() - new Date(startedAt).getTime();\n\n const merged: TestRunReport = {\n schemaVersion: reports[0]?.schemaVersion ?? '1.0.0',\n testRunId: options.testRunId ?? randomUUID(),\n startedAt,\n completedAt,\n totalDuration,\n summary,\n ci: reports.find((r) => r.ci !== null)?.ci ?? null,\n metadata: reports.find((r) => r.metadata !== null)?.metadata ?? null,\n timeline: allTimelines,\n shardRunIds,\n };\n\n // Write to disk\n const dir = dirname(options.output);\n mkdirSync(dir, { recursive: true });\n writeFileSync(options.output, JSON.stringify(merged, null, 2), 'utf-8');\n\n return merged;\n}\n\nfunction recalculateSummaryFromList(summaries: Summary[], timelineLength: number): Summary {\n let total = 0;\n let passed = 0;\n let failed = 0;\n let flaky = 0;\n let skipped = 0;\n let timedout = 0;\n let totalApiCalls = 0;\n let totalAssertions = 0;\n let passedAssertions = 0;\n let failedAssertions = 0;\n let totalNavigations = 0;\n\n for (const s of summaries) {\n total += s.total;\n passed += s.passed;\n failed += s.failed;\n flaky += s.flaky;\n skipped += s.skipped;\n timedout += s.timedout ?? 0;\n totalApiCalls += s.totalApiCalls ?? 0;\n totalAssertions += s.totalAssertions ?? 0;\n passedAssertions += s.passedAssertions ?? 0;\n failedAssertions += s.failedAssertions ?? 0;\n totalNavigations += s.totalNavigations ?? 0;\n }\n\n return {\n total, passed, failed, flaky, skipped, timedout,\n totalApiCalls, uniqueApiUrls: 0,\n apiCallsByMethod: {},\n apiCallsByStatusRange: { '2xx': 0, '3xx': 0, '4xx': 0, '5xx': 0, error: 0 },\n apiResponseTime: null,\n totalAssertions, passedAssertions, failedAssertions,\n totalNavigations, uniqueNavigationUrls: 0,\n totalTimelineSteps: timelineLength,\n totalActionSteps: 0,\n actionStepsByCategory: {},\n };\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@testrelic/playwright-analytics",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.3.0",
|
|
4
4
|
"description": "Playwright test analytics reporter with E2E navigation tracking, API call capture, network stats, failure diagnostics, and interactive HTML reports",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"playwright",
|
|
@@ -81,7 +81,7 @@
|
|
|
81
81
|
"@playwright/test": ">=1.35.0"
|
|
82
82
|
},
|
|
83
83
|
"dependencies": {
|
|
84
|
-
"@testrelic/core": "2.
|
|
84
|
+
"@testrelic/core": "2.3.0"
|
|
85
85
|
},
|
|
86
86
|
"devDependencies": {
|
|
87
87
|
"@playwright/test": "^1.35.0",
|