@testrelic/playwright-analytics 1.3.0 → 2.0.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/merge.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';var crypto=require('crypto'),fs=require('fs'),path=require('path'),core=require('@testrelic/core');async function k(l,o){let e=[];for(let t of l){let r;try{r=fs.readFileSync(t,"utf-8");}catch(d){throw core.createError(core.ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,d)}let u;try{u=JSON.parse(r);}catch(d){throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,d)}if(!core.isValidTestRunReport(u))throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);e.push(u);}let a=e.map(t=>t.testRunId),i=[];for(let t of e)i.push(...t.timeline);i.sort((t,r)=>new Date(t.visitedAt).getTime()-new Date(r.visitedAt).getTime());let m=w(e),s=e.reduce((t,r)=>r.startedAt<t?r.startedAt:t,e[0]?.startedAt??new Date().toISOString()),n=e.reduce((t,r)=>r.completedAt>t?r.completedAt:t,e[0]?.completedAt??new Date().toISOString()),y=new Date(n).getTime()-new Date(s).getTime(),f={schemaVersion:e[0]?.schemaVersion??"1.0.0",testRunId:o.testRunId??crypto.randomUUID(),startedAt:s,completedAt:n,totalDuration:y,summary:m,ci:e.find(t=>t.ci!==null)?.ci??null,metadata:e.find(t=>t.metadata!==null)?.metadata??null,timeline:i,shardRunIds:a},R=path.dirname(o.output);return fs.mkdirSync(R,{recursive:true}),fs.writeFileSync(o.output,JSON.stringify(f,null,2),"utf-8"),f}function w(l){let o=0,e=0,a=0,i=0,m=0,s=0;for(let n of l)o+=n.summary.total,e+=n.summary.passed,a+=n.summary.failed,i+=n.summary.flaky,m+=n.summary.skipped,s+=n.summary.timedout??0;return {total:o,passed:e,failed:a,flaky:i,skipped:m,timedout:s}}exports.mergeReports=k;//# sourceMappingURL=merge.cjs.map
1
+ 'use strict';var crypto=require('crypto'),fs=require('fs'),path=require('path'),core=require('@testrelic/core');async function k(y,a){let s=[];for(let t of y){let e;try{e=fs.readFileSync(t,"utf-8");}catch(o){throw core.createError(core.ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,o)}let n;try{n=JSON.parse(e);}catch(o){throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,o)}if(!core.isValidTestRunReport(n))throw core.createError(core.ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);s.push(n);}let d=s.map(t=>t.testRunId),i=[];for(let t of s)i.push(...t.timeline);i.sort((t,e)=>{let n="visitedAt"in t?t.visitedAt:t.timestamp,o="visitedAt"in e?e.visitedAt:e.timestamp;return new Date(n).getTime()-new Date(o).getTime()});let p=v(s,i.length),m=s.reduce((t,e)=>e.startedAt<t?e.startedAt:t,s[0]?.startedAt??new Date().toISOString()),l=s.reduce((t,e)=>e.completedAt>t?e.completedAt:t,s[0]?.completedAt??new Date().toISOString()),c=new Date(l).getTime()-new Date(m).getTime(),u={schemaVersion:s[0]?.schemaVersion??"1.0.0",testRunId:a.testRunId??crypto.randomUUID(),startedAt:m,completedAt:l,totalDuration:c,summary:p,ci:s.find(t=>t.ci!==null)?.ci??null,metadata:s.find(t=>t.metadata!==null)?.metadata??null,timeline:i,shardRunIds:d},f=path.dirname(a.output);return fs.mkdirSync(f,{recursive:true}),fs.writeFileSync(a.output,JSON.stringify(u,null,2),"utf-8"),u}function v(y,a){let s=0,d=0,i=0,p=0,m=0,l=0,c=0,u=0,f=0,t=0,e=0,n=new Set,o=new Set,S={},g={"2xx":0,"3xx":0,"4xx":0,"5xx":0,error:0};for(let r of y)s+=r.summary.total,d+=r.summary.passed,i+=r.summary.failed,p+=r.summary.flaky,m+=r.summary.skipped,l+=r.summary.timedout??0,c+=r.summary.totalApiCalls??0,u+=r.summary.totalAssertions??0,f+=r.summary.passedAssertions??0,t+=r.summary.failedAssertions??0,e+=r.summary.totalNavigations??0;return {total:s,passed:d,failed:i,flaky:p,skipped:m,timedout:l,totalApiCalls:c,uniqueApiUrls:n.size,apiCallsByMethod:S,apiCallsByStatusRange:g,apiResponseTime:null,totalAssertions:u,passedAssertions:f,failedAssertions:t,totalNavigations:e,uniqueNavigationUrls:o.size,totalTimelineSteps:a}}exports.mergeReports=k;//# sourceMappingURL=merge.cjs.map
2
2
  //# sourceMappingURL=merge.cjs.map
@@ -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","summary","recalculateSummary","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","dir","dirname","mkdirSync","writeFileSync","total","passed","failed","flaky","skipped","timedout"],"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} 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[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) =>\n new Date(a.visitedAt).getTime() - new Date(b.visitedAt).getTime(),\n );\n\n // Recalculate summary across all shards\n const summary = recalculateSummary(reports);\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[]): 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\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 }\n\n return { total, passed, failed, flaky, skipped, timedout };\n}\n"]}
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 };\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';async function k(l,o){let e=[];for(let t of l){let r;try{r=readFileSync(t,"utf-8");}catch(d){throw createError(ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,d)}let u;try{u=JSON.parse(r);}catch(d){throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,d)}if(!isValidTestRunReport(u))throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);e.push(u);}let a=e.map(t=>t.testRunId),i=[];for(let t of e)i.push(...t.timeline);i.sort((t,r)=>new Date(t.visitedAt).getTime()-new Date(r.visitedAt).getTime());let m=w(e),s=e.reduce((t,r)=>r.startedAt<t?r.startedAt:t,e[0]?.startedAt??new Date().toISOString()),n=e.reduce((t,r)=>r.completedAt>t?r.completedAt:t,e[0]?.completedAt??new Date().toISOString()),y=new Date(n).getTime()-new Date(s).getTime(),f={schemaVersion:e[0]?.schemaVersion??"1.0.0",testRunId:o.testRunId??randomUUID(),startedAt:s,completedAt:n,totalDuration:y,summary:m,ci:e.find(t=>t.ci!==null)?.ci??null,metadata:e.find(t=>t.metadata!==null)?.metadata??null,timeline:i,shardRunIds:a},R=dirname(o.output);return mkdirSync(R,{recursive:true}),writeFileSync(o.output,JSON.stringify(f,null,2),"utf-8"),f}function w(l){let o=0,e=0,a=0,i=0,m=0,s=0;for(let n of l)o+=n.summary.total,e+=n.summary.passed,a+=n.summary.failed,i+=n.summary.flaky,m+=n.summary.skipped,s+=n.summary.timedout??0;return {total:o,passed:e,failed:a,flaky:i,skipped:m,timedout:s}}export{k as mergeReports};//# sourceMappingURL=merge.js.map
1
+ import {randomUUID}from'crypto';import {readFileSync,mkdirSync,writeFileSync}from'fs';import {dirname}from'path';import {createError,ErrorCode,isValidTestRunReport}from'@testrelic/core';async function k(y,a){let s=[];for(let t of y){let e;try{e=readFileSync(t,"utf-8");}catch(o){throw createError(ErrorCode.MERGE_READ_FAILED,`Failed to read file: ${t}`,o)}let n;try{n=JSON.parse(e);}catch(o){throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid JSON in file: ${t}`,o)}if(!isValidTestRunReport(n))throw createError(ErrorCode.MERGE_INVALID_SCHEMA,`Invalid report schema in file: ${t}`);s.push(n);}let d=s.map(t=>t.testRunId),i=[];for(let t of s)i.push(...t.timeline);i.sort((t,e)=>{let n="visitedAt"in t?t.visitedAt:t.timestamp,o="visitedAt"in e?e.visitedAt:e.timestamp;return new Date(n).getTime()-new Date(o).getTime()});let p=v(s,i.length),m=s.reduce((t,e)=>e.startedAt<t?e.startedAt:t,s[0]?.startedAt??new Date().toISOString()),l=s.reduce((t,e)=>e.completedAt>t?e.completedAt:t,s[0]?.completedAt??new Date().toISOString()),c=new Date(l).getTime()-new Date(m).getTime(),u={schemaVersion:s[0]?.schemaVersion??"1.0.0",testRunId:a.testRunId??randomUUID(),startedAt:m,completedAt:l,totalDuration:c,summary:p,ci:s.find(t=>t.ci!==null)?.ci??null,metadata:s.find(t=>t.metadata!==null)?.metadata??null,timeline:i,shardRunIds:d},f=dirname(a.output);return mkdirSync(f,{recursive:true}),writeFileSync(a.output,JSON.stringify(u,null,2),"utf-8"),u}function v(y,a){let s=0,d=0,i=0,p=0,m=0,l=0,c=0,u=0,f=0,t=0,e=0,n=new Set,o=new Set,S={},g={"2xx":0,"3xx":0,"4xx":0,"5xx":0,error:0};for(let r of y)s+=r.summary.total,d+=r.summary.passed,i+=r.summary.failed,p+=r.summary.flaky,m+=r.summary.skipped,l+=r.summary.timedout??0,c+=r.summary.totalApiCalls??0,u+=r.summary.totalAssertions??0,f+=r.summary.passedAssertions??0,t+=r.summary.failedAssertions??0,e+=r.summary.totalNavigations??0;return {total:s,passed:d,failed:i,flaky:p,skipped:m,timedout:l,totalApiCalls:c,uniqueApiUrls:n.size,apiCallsByMethod:S,apiCallsByStatusRange:g,apiResponseTime:null,totalAssertions:u,passedAssertions:f,failedAssertions:t,totalNavigations:e,uniqueNavigationUrls:o.size,totalTimelineSteps:a}}export{k 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","summary","recalculateSummary","startedAt","earliest","completedAt","latest","totalDuration","merged","randomUUID","dir","dirname","mkdirSync","writeFileSync","total","passed","failed","flaky","skipped","timedout"],"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} 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[] = [];\n for (const report of reports) {\n allTimelines.push(...report.timeline);\n }\n allTimelines.sort((a, b) =>\n new Date(a.visitedAt).getTime() - new Date(b.visitedAt).getTime(),\n );\n\n // Recalculate summary across all shards\n const summary = recalculateSummary(reports);\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[]): 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\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 }\n\n return { total, passed, failed, flaky, skipped, timedout };\n}\n"]}
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 };\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@testrelic/playwright-analytics",
3
- "version": "1.3.0",
3
+ "version": "2.0.0",
4
4
  "description": "Playwright custom reporter and navigation-tracking fixture for test analytics",
5
5
  "type": "module",
6
6
  "main": "./dist/index.cjs",
@@ -27,6 +27,16 @@
27
27
  "default": "./dist/fixture.cjs"
28
28
  }
29
29
  },
30
+ "./api-fixture": {
31
+ "import": {
32
+ "types": "./dist/api-fixture.d.ts",
33
+ "default": "./dist/api-fixture.js"
34
+ },
35
+ "require": {
36
+ "types": "./dist/api-fixture.d.cts",
37
+ "default": "./dist/api-fixture.cjs"
38
+ }
39
+ },
30
40
  "./merge": {
31
41
  "import": {
32
42
  "types": "./dist/merge.d.ts",
@@ -46,11 +56,20 @@
46
56
  "dist",
47
57
  "timeline-schema.json"
48
58
  ],
59
+ "scripts": {
60
+ "build": "tsup",
61
+ "test": "vitest run --exclude '**/e2e/**'",
62
+ "test:watch": "vitest",
63
+ "test:integration": "vitest run --config vitest.integration.config.ts",
64
+ "typecheck": "tsc --noEmit",
65
+ "test:e2e": "pnpm run build && npx playwright test --config __tests__/e2e/playwright.config.ts",
66
+ "test:e2e:validate": "npx tsx __tests__/e2e/validate-report.ts"
67
+ },
49
68
  "peerDependencies": {
50
69
  "@playwright/test": ">=1.35.0"
51
70
  },
52
71
  "dependencies": {
53
- "@testrelic/core": "1.3.0"
72
+ "@testrelic/core": "workspace:*"
54
73
  },
55
74
  "devDependencies": {
56
75
  "@playwright/test": "^1.35.0",
@@ -63,14 +82,5 @@
63
82
  "node": ">=18"
64
83
  },
65
84
  "license": "MIT",
66
- "homepage": "https://testrelic.co",
67
- "scripts": {
68
- "build": "tsup",
69
- "test": "vitest run --exclude '**/e2e/**'",
70
- "test:watch": "vitest",
71
- "test:integration": "vitest run --config vitest.integration.config.ts",
72
- "typecheck": "tsc --noEmit",
73
- "test:e2e": "pnpm run build && npx playwright test --config __tests__/e2e/playwright.config.ts",
74
- "test:e2e:validate": "npx tsx __tests__/e2e/validate-report.ts"
75
- }
76
- }
85
+ "homepage": "https://testrelic.ai"
86
+ }