apex-code-coverage-transformer 2.4.1 → 2.5.1
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/CHANGELOG.md +15 -0
- package/README.md +38 -260
- package/lib/commands/acc-transformer/transform.d.ts +1 -1
- package/lib/commands/acc-transformer/transform.js +15 -8
- package/lib/commands/acc-transformer/transform.js.map +1 -1
- package/lib/handlers/cloverCoverageHandler.d.ts +7 -0
- package/lib/handlers/cloverCoverageHandler.js +69 -0
- package/lib/handlers/cloverCoverageHandler.js.map +1 -0
- package/lib/handlers/coberturaCoverageHandler.d.ts +8 -0
- package/lib/handlers/coberturaCoverageHandler.js +62 -0
- package/lib/handlers/coberturaCoverageHandler.js.map +1 -0
- package/lib/handlers/getCoverageHandler.d.ts +2 -0
- package/lib/handlers/getCoverageHandler.js +19 -0
- package/lib/handlers/getCoverageHandler.js.map +1 -0
- package/lib/handlers/lcovCoverageHandler.d.ts +7 -0
- package/lib/handlers/lcovCoverageHandler.js +26 -0
- package/lib/handlers/lcovCoverageHandler.js.map +1 -0
- package/lib/handlers/sonarCoverageHandler.d.ts +7 -0
- package/lib/handlers/sonarCoverageHandler.js +27 -0
- package/lib/handlers/sonarCoverageHandler.js.map +1 -0
- package/lib/helpers/constants.js +1 -1
- package/lib/helpers/constants.js.map +1 -1
- package/lib/helpers/findFilePath.js +2 -0
- package/lib/helpers/findFilePath.js.map +1 -1
- package/lib/helpers/generateReport.d.ts +2 -0
- package/lib/helpers/generateReport.js +31 -0
- package/lib/helpers/generateReport.js.map +1 -0
- package/lib/helpers/setCoveredLines.d.ts +1 -0
- package/lib/helpers/setCoveredLines.js +27 -0
- package/lib/helpers/setCoveredLines.js.map +1 -0
- package/lib/helpers/transformDeployCoverageReport.js +11 -89
- package/lib/helpers/transformDeployCoverageReport.js.map +1 -1
- package/lib/helpers/transformTestCoverageReport.js +8 -88
- package/lib/helpers/transformTestCoverageReport.js.map +1 -1
- package/lib/helpers/types.d.ts +20 -3
- package/lib/hooks/postrun.js +4 -4
- package/lib/hooks/postrun.js.map +1 -1
- package/messages/transformer.transform.md +7 -6
- package/oclif.lock +523 -521
- package/oclif.manifest.json +12 -10
- package/package.json +5 -5
- package/lib/helpers/generateXml.d.ts +0 -2
- package/lib/helpers/generateXml.js +0 -14
- package/lib/helpers/generateXml.js.map +0 -1
- package/lib/helpers/initializeCoverageObject.d.ts +0 -5
- package/lib/helpers/initializeCoverageObject.js +0 -68
- package/lib/helpers/initializeCoverageObject.js.map +0 -1
- package/lib/helpers/setCoveredLinesClover.d.ts +0 -2
- package/lib/helpers/setCoveredLinesClover.js +0 -35
- package/lib/helpers/setCoveredLinesClover.js.map +0 -1
- package/lib/helpers/setCoveredLinesCobertura.d.ts +0 -2
- package/lib/helpers/setCoveredLinesCobertura.js +0 -34
- package/lib/helpers/setCoveredLinesCobertura.js.map +0 -1
- package/lib/helpers/setCoveredLinesSonar.d.ts +0 -2
- package/lib/helpers/setCoveredLinesSonar.js +0 -30
- package/lib/helpers/setCoveredLinesSonar.js.map +0 -1
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
export class CoberturaCoverageHandler {
|
|
3
|
+
coverageObj;
|
|
4
|
+
packageObj;
|
|
5
|
+
constructor() {
|
|
6
|
+
this.coverageObj = {
|
|
7
|
+
coverage: {
|
|
8
|
+
'@lines-valid': 0,
|
|
9
|
+
'@lines-covered': 0,
|
|
10
|
+
'@line-rate': 0,
|
|
11
|
+
'@branches-valid': 0,
|
|
12
|
+
'@branches-covered': 0,
|
|
13
|
+
'@branch-rate': 1,
|
|
14
|
+
'@timestamp': Date.now(),
|
|
15
|
+
'@complexity': 0,
|
|
16
|
+
'@version': '0.1',
|
|
17
|
+
sources: { source: ['.'] },
|
|
18
|
+
packages: { package: [] },
|
|
19
|
+
},
|
|
20
|
+
};
|
|
21
|
+
this.packageObj = {
|
|
22
|
+
'@name': 'main',
|
|
23
|
+
'@line-rate': 0,
|
|
24
|
+
'@branch-rate': 1,
|
|
25
|
+
classes: { class: [] },
|
|
26
|
+
};
|
|
27
|
+
this.coverageObj.coverage.packages.package.push(this.packageObj);
|
|
28
|
+
}
|
|
29
|
+
processFile(filePath, fileName, lines, uncoveredLines, coveredLines) {
|
|
30
|
+
const classObj = {
|
|
31
|
+
'@name': fileName,
|
|
32
|
+
'@filename': filePath,
|
|
33
|
+
'@line-rate': '0',
|
|
34
|
+
'@branch-rate': '1',
|
|
35
|
+
methods: {},
|
|
36
|
+
lines: { line: [] },
|
|
37
|
+
};
|
|
38
|
+
const totalLines = uncoveredLines.length + coveredLines.length;
|
|
39
|
+
const coveredLineCount = coveredLines.length;
|
|
40
|
+
for (const [lineNumber, isCovered] of Object.entries(lines)) {
|
|
41
|
+
classObj.lines.line.push({
|
|
42
|
+
'@number': Number(lineNumber),
|
|
43
|
+
'@hits': isCovered === 1 ? 1 : 0,
|
|
44
|
+
'@branch': 'false',
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
// Calculate and set the line rate for this class
|
|
48
|
+
if (totalLines > 0) {
|
|
49
|
+
const lineRate = (coveredLineCount / totalLines).toFixed(4);
|
|
50
|
+
classObj['@line-rate'] = lineRate;
|
|
51
|
+
}
|
|
52
|
+
this.coverageObj.coverage['@lines-valid'] += totalLines;
|
|
53
|
+
this.coverageObj.coverage['@lines-covered'] += coveredLineCount;
|
|
54
|
+
this.packageObj['@line-rate'] = Number((this.coverageObj.coverage['@lines-covered'] / this.coverageObj.coverage['@lines-valid']).toFixed(4));
|
|
55
|
+
this.coverageObj.coverage['@line-rate'] = this.packageObj['@line-rate'];
|
|
56
|
+
this.packageObj.classes.class.push(classObj);
|
|
57
|
+
}
|
|
58
|
+
finalize() {
|
|
59
|
+
return this.coverageObj;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
//# sourceMappingURL=coberturaCoverageHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"coberturaCoverageHandler.js","sourceRoot":"","sources":["../../src/handlers/coberturaCoverageHandler.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAIb,MAAM,OAAO,wBAAwB;IAClB,WAAW,CAA0B;IAC9C,UAAU,CAAmB;IAErC;QACE,IAAI,CAAC,WAAW,GAAG;YACjB,QAAQ,EAAE;gBACR,cAAc,EAAE,CAAC;gBACjB,gBAAgB,EAAE,CAAC;gBACnB,YAAY,EAAE,CAAC;gBACf,iBAAiB,EAAE,CAAC;gBACpB,mBAAmB,EAAE,CAAC;gBACtB,cAAc,EAAE,CAAC;gBACjB,YAAY,EAAE,IAAI,CAAC,GAAG,EAAE;gBACxB,aAAa,EAAE,CAAC;gBAChB,UAAU,EAAE,KAAK;gBACjB,OAAO,EAAE,EAAE,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE;gBAC1B,QAAQ,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE;aAC1B;SACF,CAAC;QACF,IAAI,CAAC,UAAU,GAAG;YAChB,OAAO,EAAE,MAAM;YACf,YAAY,EAAE,CAAC;YACf,cAAc,EAAE,CAAC;YACjB,OAAO,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;SACvB,CAAC;QACF,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACnE,CAAC;IAEM,WAAW,CAChB,QAAgB,EAChB,QAAgB,EAChB,KAA6B,EAC7B,cAAwB,EACxB,YAAsB;QAEtB,MAAM,QAAQ,GAAmB;YAC/B,OAAO,EAAE,QAAQ;YACjB,WAAW,EAAE,QAAQ;YACrB,YAAY,EAAE,GAAG;YACjB,cAAc,EAAE,GAAG;YACnB,OAAO,EAAE,EAAE;YACX,KAAK,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE;SACpB,CAAC;QAEF,MAAM,UAAU,GAAG,cAAc,CAAC,MAAM,GAAG,YAAY,CAAC,MAAM,CAAC;QAC/D,MAAM,gBAAgB,GAAG,YAAY,CAAC,MAAM,CAAC;QAE7C,KAAK,MAAM,CAAC,UAAU,EAAE,SAAS,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YAC5D,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC;gBACvB,SAAS,EAAE,MAAM,CAAC,UAAU,CAAC;gBAC7B,OAAO,EAAE,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChC,SAAS,EAAE,OAAO;aACnB,CAAC,CAAC;QACL,CAAC;QAED,iDAAiD;QACjD,IAAI,UAAU,GAAG,CAAC,EAAE,CAAC;YACnB,MAAM,QAAQ,GAAG,CAAC,gBAAgB,GAAG,UAAU,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YAC5D,QAAQ,CAAC,YAAY,CAAC,GAAG,QAAQ,CAAC;QACpC,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,UAAU,CAAC;QACxD,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,gBAAgB,CAAC,IAAI,gBAAgB,CAAC;QAEhE,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,GAAG,MAAM,CACpC,CAAC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CACrG,CAAC;QACF,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;QAExE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAC/C,CAAC;IAEM,QAAQ;QACb,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;CACF"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
import { CloverCoverageHandler } from './cloverCoverageHandler.js';
|
|
3
|
+
import { CoberturaCoverageHandler } from './coberturaCoverageHandler.js';
|
|
4
|
+
import { SonarCoverageHandler } from './sonarCoverageHandler.js';
|
|
5
|
+
import { LcovCoverageHandler } from './lcovCoverageHandler.js';
|
|
6
|
+
export function getCoverageHandler(format) {
|
|
7
|
+
const handlers = {
|
|
8
|
+
sonar: new SonarCoverageHandler(),
|
|
9
|
+
cobertura: new CoberturaCoverageHandler(),
|
|
10
|
+
clover: new CloverCoverageHandler(),
|
|
11
|
+
lcovonly: new LcovCoverageHandler(),
|
|
12
|
+
};
|
|
13
|
+
const handler = handlers[format];
|
|
14
|
+
if (!handler) {
|
|
15
|
+
throw new Error(`Unsupported format: ${format}`);
|
|
16
|
+
}
|
|
17
|
+
return handler;
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=getCoverageHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"getCoverageHandler.js","sourceRoot":"","sources":["../../src/handlers/getCoverageHandler.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAGb,OAAO,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AACnE,OAAO,EAAE,wBAAwB,EAAE,MAAM,+BAA+B,CAAC;AACzE,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAE/D,MAAM,UAAU,kBAAkB,CAAC,MAAc;IAC/C,MAAM,QAAQ,GAAoC;QAChD,KAAK,EAAE,IAAI,oBAAoB,EAAE;QACjC,SAAS,EAAE,IAAI,wBAAwB,EAAE;QACzC,MAAM,EAAE,IAAI,qBAAqB,EAAE;QACnC,QAAQ,EAAE,IAAI,mBAAmB,EAAE;KACpC,CAAC;IAEF,MAAM,OAAO,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC;IACjC,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,MAAM,IAAI,KAAK,CAAC,uBAAuB,MAAM,EAAE,CAAC,CAAC;IACnD,CAAC;IACD,OAAO,OAAO,CAAC;AACjB,CAAC"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { CoverageHandler, LcovCoverageObject } from '../helpers/types.js';
|
|
2
|
+
export declare class LcovCoverageHandler implements CoverageHandler {
|
|
3
|
+
private readonly coverageObj;
|
|
4
|
+
constructor();
|
|
5
|
+
processFile(filePath: string, fileName: string, lines: Record<string, number>, uncoveredLines: number[], coveredLines: number[]): void;
|
|
6
|
+
finalize(): LcovCoverageObject;
|
|
7
|
+
}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
export class LcovCoverageHandler {
|
|
3
|
+
coverageObj;
|
|
4
|
+
constructor() {
|
|
5
|
+
this.coverageObj = { files: [] };
|
|
6
|
+
}
|
|
7
|
+
processFile(filePath, fileName, lines, uncoveredLines, coveredLines) {
|
|
8
|
+
const lcovFile = {
|
|
9
|
+
sourceFile: filePath,
|
|
10
|
+
lines: [],
|
|
11
|
+
totalLines: uncoveredLines.length + coveredLines.length,
|
|
12
|
+
coveredLines: coveredLines.length,
|
|
13
|
+
};
|
|
14
|
+
for (const [lineNumber, isCovered] of Object.entries(lines)) {
|
|
15
|
+
lcovFile.lines.push({
|
|
16
|
+
lineNumber: Number(lineNumber),
|
|
17
|
+
hitCount: isCovered === 1 ? 1 : 0,
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
this.coverageObj.files.push(lcovFile);
|
|
21
|
+
}
|
|
22
|
+
finalize() {
|
|
23
|
+
return this.coverageObj;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=lcovCoverageHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lcovCoverageHandler.js","sourceRoot":"","sources":["../../src/handlers/lcovCoverageHandler.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAIb,MAAM,OAAO,mBAAmB;IACb,WAAW,CAAqB;IAEjD;QACE,IAAI,CAAC,WAAW,GAAG,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC;IACnC,CAAC;IAEM,WAAW,CAChB,QAAgB,EAChB,QAAgB,EAChB,KAA6B,EAC7B,cAAwB,EACxB,YAAsB;QAEtB,MAAM,QAAQ,GAAa;YACzB,UAAU,EAAE,QAAQ;YACpB,KAAK,EAAE,EAAE;YACT,UAAU,EAAE,cAAc,CAAC,MAAM,GAAG,YAAY,CAAC,MAAM;YACvD,YAAY,EAAE,YAAY,CAAC,MAAM;SAClC,CAAC;QAEF,KAAK,MAAM,CAAC,UAAU,EAAE,SAAS,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YAC5D,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC;gBAClB,UAAU,EAAE,MAAM,CAAC,UAAU,CAAC;gBAC9B,QAAQ,EAAE,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;aAClC,CAAC,CAAC;QACL,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACxC,CAAC;IAEM,QAAQ;QACb,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;CACF"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { SonarCoverageObject, CoverageHandler } from '../helpers/types.js';
|
|
2
|
+
export declare class SonarCoverageHandler implements CoverageHandler {
|
|
3
|
+
private readonly coverageObj;
|
|
4
|
+
constructor();
|
|
5
|
+
processFile(filePath: string, _fileName: string, lines: Record<string, number>): void;
|
|
6
|
+
finalize(): SonarCoverageObject;
|
|
7
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
export class SonarCoverageHandler {
|
|
3
|
+
coverageObj;
|
|
4
|
+
constructor() {
|
|
5
|
+
this.coverageObj = { coverage: { '@version': '1', file: [] } };
|
|
6
|
+
}
|
|
7
|
+
processFile(filePath, _fileName, lines) {
|
|
8
|
+
const fileObj = {
|
|
9
|
+
'@path': filePath,
|
|
10
|
+
lineToCover: [],
|
|
11
|
+
};
|
|
12
|
+
for (const lineNumberString in lines) {
|
|
13
|
+
if (!Object.hasOwn(lines, lineNumberString))
|
|
14
|
+
continue;
|
|
15
|
+
const covered = lines[lineNumberString] === 1 ? 'true' : 'false';
|
|
16
|
+
fileObj.lineToCover.push({
|
|
17
|
+
'@lineNumber': Number(lineNumberString),
|
|
18
|
+
'@covered': covered,
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
this.coverageObj.coverage.file.push(fileObj);
|
|
22
|
+
}
|
|
23
|
+
finalize() {
|
|
24
|
+
return this.coverageObj;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=sonarCoverageHandler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sonarCoverageHandler.js","sourceRoot":"","sources":["../../src/handlers/sonarCoverageHandler.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAIb,MAAM,OAAO,oBAAoB;IACd,WAAW,CAAsB;IAElD;QACE,IAAI,CAAC,WAAW,GAAG,EAAE,QAAQ,EAAE,EAAE,UAAU,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC;IACjE,CAAC;IAEM,WAAW,CAChB,QAAgB,EAChB,SAAiB,EACjB,KAA6B;QAE7B,MAAM,OAAO,GAAe;YAC1B,OAAO,EAAE,QAAQ;YACjB,WAAW,EAAE,EAAE;SAChB,CAAC;QACF,KAAK,MAAM,gBAAgB,IAAI,KAAK,EAAE,CAAC;YACrC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,EAAE,gBAAgB,CAAC;gBAAE,SAAS;YACtD,MAAM,OAAO,GAAG,KAAK,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC;YACjE,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC;gBACvB,aAAa,EAAE,MAAM,CAAC,gBAAgB,CAAC;gBACvC,UAAU,EAAE,OAAO;aACpB,CAAC,CAAC;QACL,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC/C,CAAC;IAEM,QAAQ;QACb,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;CACF"}
|
package/lib/helpers/constants.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export const formatOptions = ['sonar', 'cobertura', 'clover'];
|
|
1
|
+
export const formatOptions = ['sonar', 'cobertura', 'clover', 'lcovonly'];
|
|
2
2
|
//# sourceMappingURL=constants.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"constants.js","sourceRoot":"","sources":["../../src/helpers/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,aAAa,GAAa,CAAC,OAAO,EAAE,WAAW,EAAE,QAAQ,CAAC,CAAC"}
|
|
1
|
+
{"version":3,"file":"constants.js","sourceRoot":"","sources":["../../src/helpers/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,aAAa,GAAa,CAAC,OAAO,EAAE,WAAW,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC"}
|
|
@@ -2,11 +2,13 @@
|
|
|
2
2
|
/* eslint-disable no-await-in-loop */
|
|
3
3
|
import { readdir, stat } from 'node:fs/promises';
|
|
4
4
|
import { join, relative } from 'node:path';
|
|
5
|
+
import { normalizePathToUnix } from './normalizePathToUnix.js';
|
|
5
6
|
export async function findFilePath(fileName, packageDirectories, repoRoot) {
|
|
6
7
|
let relativeFilePath;
|
|
7
8
|
for (const directory of packageDirectories) {
|
|
8
9
|
relativeFilePath = await findFilePathinDirectory(fileName, directory, repoRoot);
|
|
9
10
|
if (relativeFilePath !== undefined) {
|
|
11
|
+
relativeFilePath = normalizePathToUnix(relativeFilePath);
|
|
10
12
|
break;
|
|
11
13
|
}
|
|
12
14
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"findFilePath.js","sourceRoot":"","sources":["../../src/helpers/findFilePath.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AACb,qCAAqC;AAErC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;
|
|
1
|
+
{"version":3,"file":"findFilePath.js","sourceRoot":"","sources":["../../src/helpers/findFilePath.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AACb,qCAAqC;AAErC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,WAAW,CAAC;AAC3C,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAE/D,MAAM,CAAC,KAAK,UAAU,YAAY,CAChC,QAAgB,EAChB,kBAA4B,EAC5B,QAAgB;IAEhB,IAAI,gBAAoC,CAAC;IACzC,KAAK,MAAM,SAAS,IAAI,kBAAkB,EAAE,CAAC;QAC3C,gBAAgB,GAAG,MAAM,uBAAuB,CAAC,QAAQ,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;QAChF,IAAI,gBAAgB,KAAK,SAAS,EAAE,CAAC;YACnC,gBAAgB,GAAG,mBAAmB,CAAC,gBAAgB,CAAC,CAAA;YACxD,MAAM;QACR,CAAC;IACH,CAAC;IACD,OAAO,gBAAgB,CAAC;AAC1B,CAAC;AAED,KAAK,UAAU,iBAAiB,CAAC,QAAgB,EAAE,WAAmB;IACpE,MAAM,KAAK,GAAG,MAAM,OAAO,CAAC,WAAW,CAAC,CAAC;IACzC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;QACzC,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,CAAC;QACnC,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;YACxB,MAAM,MAAM,GAAG,MAAM,iBAAiB,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;YAC3D,IAAI,MAAM,EAAE,CAAC;gBACX,OAAO,MAAM,CAAC;YAChB,CAAC;QACH,CAAC;aAAM,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,OAAO,QAAQ,CAAC;QAClB,CAAC;IACH,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,KAAK,UAAU,uBAAuB,CACpC,QAAgB,EAChB,WAAmB,EACnB,QAAgB;IAEhB,MAAM,aAAa,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC7D,IAAI,gBAAoC,CAAC;IAEzC,IAAI,aAAa,EAAE,CAAC;QAClB,uEAAuE;QACvE,MAAM,gBAAgB,GAAG,MAAM,iBAAiB,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QACxE,IAAI,gBAAgB,KAAK,SAAS;YAAE,gBAAgB,GAAG,QAAQ,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;IAC9F,CAAC;SAAM,CAAC;QACN,+DAA+D;QAC/D,MAAM,QAAQ,GAAa,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;QAC9C,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;YAC3B,MAAM,gBAAgB,GAAG,MAAM,iBAAiB,CAAC,GAAG,QAAQ,IAAI,GAAG,EAAE,EAAE,WAAW,CAAC,CAAC;YACpF,IAAI,gBAAgB,KAAK,SAAS,EAAE,CAAC;gBACnC,gBAAgB,GAAG,QAAQ,CAAC,QAAQ,EAAE,gBAAgB,CAAC,CAAC;gBACxD,MAAM;YACR,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,gBAAgB,CAAC;AAC1B,CAAC"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import { SonarCoverageObject, CoberturaCoverageObject, CloverCoverageObject, LcovCoverageObject } from './types.js';
|
|
2
|
+
export declare function generateReport(coverageObj: SonarCoverageObject | CoberturaCoverageObject | CloverCoverageObject | LcovCoverageObject, format: string): string;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
import { create } from 'xmlbuilder2';
|
|
3
|
+
export function generateReport(coverageObj, format) {
|
|
4
|
+
if (format === 'lcovonly') {
|
|
5
|
+
if ('files' in coverageObj) {
|
|
6
|
+
let lcovOutput = '';
|
|
7
|
+
for (const file of coverageObj.files) {
|
|
8
|
+
lcovOutput += `TN:\nSF:${file.sourceFile}\n`;
|
|
9
|
+
lcovOutput += 'FNF:0\nFNH:0\n';
|
|
10
|
+
for (const line of file.lines) {
|
|
11
|
+
lcovOutput += `DA:${line.lineNumber},${line.hitCount}\n`;
|
|
12
|
+
}
|
|
13
|
+
lcovOutput += `LF:${file.totalLines}\n`;
|
|
14
|
+
lcovOutput += `LH:${file.coveredLines}\n`;
|
|
15
|
+
lcovOutput += 'BRF:0\nBRH:0\n';
|
|
16
|
+
lcovOutput += 'end_of_record\n';
|
|
17
|
+
}
|
|
18
|
+
return lcovOutput;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
const isHeadless = format === 'cobertura' || format === 'clover';
|
|
22
|
+
let xml = create(coverageObj).end({ prettyPrint: true, indent: ' ', headless: isHeadless });
|
|
23
|
+
if (format === 'cobertura') {
|
|
24
|
+
xml = `<?xml version="1.0" ?>\n<!DOCTYPE coverage SYSTEM "http://cobertura.sourceforge.net/xml/coverage-04.dtd">\n${xml}`;
|
|
25
|
+
}
|
|
26
|
+
else if (format === 'clover') {
|
|
27
|
+
xml = `<?xml version="1.0" encoding="UTF-8"?>\n${xml}`;
|
|
28
|
+
}
|
|
29
|
+
return xml;
|
|
30
|
+
}
|
|
31
|
+
//# sourceMappingURL=generateReport.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generateReport.js","sourceRoot":"","sources":["../../src/helpers/generateReport.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAGrC,MAAM,UAAU,cAAc,CAC5B,WAAsG,EACtG,MAAc;IAEd,IAAI,MAAM,KAAK,UAAU,EAAE,CAAC;QAC1B,IAAI,OAAO,IAAI,WAAW,EAAE,CAAC;YAC3B,IAAI,UAAU,GAAG,EAAE,CAAC;YACpB,KAAK,MAAM,IAAI,IAAI,WAAW,CAAC,KAAK,EAAE,CAAC;gBACrC,UAAU,IAAI,WAAW,IAAI,CAAC,UAAU,IAAI,CAAC;gBAC7C,UAAU,IAAI,gBAAgB,CAAC;gBAE/B,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;oBAC9B,UAAU,IAAI,MAAM,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,QAAQ,IAAI,CAAC;gBAC3D,CAAC;gBAED,UAAU,IAAI,MAAM,IAAI,CAAC,UAAU,IAAI,CAAC;gBACxC,UAAU,IAAI,MAAM,IAAI,CAAC,YAAY,IAAI,CAAC;gBAC1C,UAAU,IAAI,gBAAgB,CAAC;gBAC/B,UAAU,IAAI,iBAAiB,CAAC;YAClC,CAAC;YACD,OAAO,UAAU,CAAC;QACpB,CAAC;IACH,CAAC;IAED,MAAM,UAAU,GAAG,MAAM,KAAK,WAAW,IAAI,MAAM,KAAK,QAAQ,CAAC;IACjE,IAAI,GAAG,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC,GAAG,CAAC,EAAE,WAAW,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAC;IAE7F,IAAI,MAAM,KAAK,WAAW,EAAE,CAAC;QAC3B,GAAG,GAAG,8GAA8G,GAAG,EAAE,CAAC;IAC5H,CAAC;SAAM,IAAI,MAAM,KAAK,QAAQ,EAAE,CAAC;QAC/B,GAAG,GAAG,2CAA2C,GAAG,EAAE,CAAC;IACzD,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function renumberLines(filePath: string, repoRoot: string, lines: Record<string, number>): Promise<Record<string, number>>;
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
import { getTotalLines } from './getTotalLines.js';
|
|
4
|
+
export async function renumberLines(filePath, repoRoot, lines) {
|
|
5
|
+
const totalLines = await getTotalLines(join(repoRoot, filePath));
|
|
6
|
+
const updatedLines = {};
|
|
7
|
+
const usedLines = new Set();
|
|
8
|
+
for (const [line, status] of Object.entries(lines)) {
|
|
9
|
+
const lineNumber = parseInt(line, 10);
|
|
10
|
+
if (status === 1 && lineNumber > totalLines) {
|
|
11
|
+
// Find the first valid line number not already used
|
|
12
|
+
for (let randomLine = 1; randomLine <= totalLines; randomLine++) {
|
|
13
|
+
if (!usedLines.has(randomLine)) {
|
|
14
|
+
updatedLines[randomLine.toString()] = status;
|
|
15
|
+
usedLines.add(randomLine);
|
|
16
|
+
break;
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
else {
|
|
21
|
+
updatedLines[line] = status;
|
|
22
|
+
usedLines.add(lineNumber);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
return updatedLines;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=setCoveredLines.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"setCoveredLines.js","sourceRoot":"","sources":["../../src/helpers/setCoveredLines.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AAEb,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAEnD,MAAM,CAAC,KAAK,UAAU,aAAa,CACjC,QAAgB,EAChB,QAAgB,EAChB,KAA6B;IAE7B,MAAM,UAAU,GAAG,MAAM,aAAa,CAAC,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC;IACjE,MAAM,YAAY,GAA2B,EAAE,CAAC;IAChD,MAAM,SAAS,GAAG,IAAI,GAAG,EAAU,CAAC;IAEpC,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QACnD,MAAM,UAAU,GAAG,QAAQ,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAEtC,IAAI,MAAM,KAAK,CAAC,IAAI,UAAU,GAAG,UAAU,EAAE,CAAC;YAC5C,oDAAoD;YACpD,KAAK,IAAI,UAAU,GAAG,CAAC,EAAE,UAAU,IAAI,UAAU,EAAE,UAAU,EAAE,EAAE,CAAC;gBAChE,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC;oBAC/B,YAAY,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC,GAAG,MAAM,CAAC;oBAC7C,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;oBAC1B,MAAM;gBACR,CAAC;YACH,CAAC;QACH,CAAC;aAAM,CAAC;YACN,YAAY,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC;YAC5B,SAAS,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC;IAED,OAAO,YAAY,CAAC;AACtB,CAAC"}
|
|
@@ -1,28 +1,22 @@
|
|
|
1
1
|
'use strict';
|
|
2
|
+
/* eslint-disable no-await-in-loop */
|
|
3
|
+
import { getCoverageHandler } from '../handlers/getCoverageHandler.js';
|
|
2
4
|
import { getPackageDirectories } from './getPackageDirectories.js';
|
|
3
5
|
import { findFilePath } from './findFilePath.js';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
6
|
-
import { setCoveredLinesClover } from './setCoveredLinesClover.js';
|
|
7
|
-
import { normalizePathToUnix } from './normalizePathToUnix.js';
|
|
8
|
-
import { generateXml } from './generateXml.js';
|
|
9
|
-
import { formatOptions } from './constants.js';
|
|
10
|
-
import { initializeCoverageObject } from './initializeCoverageObject.js';
|
|
6
|
+
import { generateReport } from './generateReport.js';
|
|
7
|
+
import { renumberLines } from './setCoveredLines.js';
|
|
11
8
|
export async function transformDeployCoverageReport(data, format) {
|
|
12
|
-
if (!formatOptions.includes(format)) {
|
|
13
|
-
throw new Error(`Unsupported format: ${format}`);
|
|
14
|
-
}
|
|
15
9
|
const warnings = [];
|
|
16
10
|
let filesProcessed = 0;
|
|
17
11
|
const { repoRoot, packageDirectories } = await getPackageDirectories();
|
|
18
|
-
const
|
|
12
|
+
const handler = getCoverageHandler(format);
|
|
19
13
|
for (const fileName in data) {
|
|
20
14
|
if (!Object.hasOwn(data, fileName))
|
|
21
15
|
continue;
|
|
22
16
|
const fileInfo = data[fileName];
|
|
23
17
|
const formattedFileName = fileName.replace(/no-map[\\/]+/, '');
|
|
24
18
|
const relativeFilePath = await findFilePath(formattedFileName, packageDirectories, repoRoot);
|
|
25
|
-
if (relativeFilePath
|
|
19
|
+
if (!relativeFilePath) {
|
|
26
20
|
warnings.push(`The file name ${formattedFileName} was not found in any package directory.`);
|
|
27
21
|
continue;
|
|
28
22
|
}
|
|
@@ -32,85 +26,13 @@ export async function transformDeployCoverageReport(data, format) {
|
|
|
32
26
|
const coveredLines = Object.keys(fileInfo.s)
|
|
33
27
|
.filter((lineNumber) => fileInfo.s[lineNumber] === 1)
|
|
34
28
|
.map(Number);
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
else if (format === 'cobertura') {
|
|
39
|
-
await handleCoberturaFormat(relativeFilePath, formattedFileName, uncoveredLines, coveredLines, repoRoot, coverageObj, packageObj);
|
|
40
|
-
}
|
|
41
|
-
else {
|
|
42
|
-
await handleCloverFormat(relativeFilePath, formattedFileName, uncoveredLines, coveredLines, repoRoot, coverageObj);
|
|
43
|
-
}
|
|
29
|
+
const updatedLines = await renumberLines(relativeFilePath, repoRoot, fileInfo.s);
|
|
30
|
+
fileInfo.s = updatedLines; // Safe reassignment outside the function
|
|
31
|
+
handler.processFile(relativeFilePath, formattedFileName, fileInfo.s, uncoveredLines, coveredLines);
|
|
44
32
|
filesProcessed++;
|
|
45
33
|
}
|
|
46
|
-
const
|
|
34
|
+
const coverageObj = handler.finalize();
|
|
35
|
+
const xml = generateReport(coverageObj, format);
|
|
47
36
|
return { xml, warnings, filesProcessed };
|
|
48
37
|
}
|
|
49
|
-
async function handleSonarFormat(filePath, uncoveredLines, coveredLines, repoRoot, coverageObj) {
|
|
50
|
-
const fileObj = {
|
|
51
|
-
'@path': normalizePathToUnix(filePath),
|
|
52
|
-
lineToCover: uncoveredLines.map((lineNumber) => ({
|
|
53
|
-
'@lineNumber': lineNumber,
|
|
54
|
-
'@covered': 'false',
|
|
55
|
-
})),
|
|
56
|
-
};
|
|
57
|
-
await setCoveredLinesSonar(coveredLines, uncoveredLines, repoRoot, filePath, fileObj);
|
|
58
|
-
coverageObj.coverage.file.push(fileObj);
|
|
59
|
-
}
|
|
60
|
-
async function handleCoberturaFormat(filePath, fileName, uncoveredLines, coveredLines, repoRoot, coverageObj, packageObj) {
|
|
61
|
-
const classObj = {
|
|
62
|
-
'@name': fileName,
|
|
63
|
-
'@filename': normalizePathToUnix(filePath),
|
|
64
|
-
'@line-rate': (coveredLines.length / (coveredLines.length + uncoveredLines.length)).toFixed(4),
|
|
65
|
-
'@branch-rate': '1',
|
|
66
|
-
methods: {},
|
|
67
|
-
lines: {
|
|
68
|
-
line: [
|
|
69
|
-
...uncoveredLines.map((lineNumber) => ({
|
|
70
|
-
'@number': lineNumber,
|
|
71
|
-
'@hits': 0,
|
|
72
|
-
'@branch': 'false',
|
|
73
|
-
})),
|
|
74
|
-
],
|
|
75
|
-
},
|
|
76
|
-
};
|
|
77
|
-
await setCoveredLinesCobertura(coveredLines, uncoveredLines, repoRoot, filePath, classObj);
|
|
78
|
-
coverageObj.coverage['@lines-valid'] += uncoveredLines.length + coveredLines.length;
|
|
79
|
-
coverageObj.coverage['@lines-covered'] += coveredLines.length;
|
|
80
|
-
packageObj.classes.class.push(classObj);
|
|
81
|
-
packageObj['@line-rate'] = Number((coverageObj.coverage['@lines-covered'] / coverageObj.coverage['@lines-valid']).toFixed(4));
|
|
82
|
-
coverageObj.coverage['@line-rate'] = packageObj['@line-rate'];
|
|
83
|
-
}
|
|
84
|
-
async function handleCloverFormat(filePath, fileName, uncoveredLines, coveredLines, repoRoot, coverageObj) {
|
|
85
|
-
const cloverFile = {
|
|
86
|
-
'@name': fileName,
|
|
87
|
-
'@path': normalizePathToUnix(filePath),
|
|
88
|
-
metrics: {
|
|
89
|
-
'@statements': uncoveredLines.length + coveredLines.length,
|
|
90
|
-
'@coveredstatements': coveredLines.length,
|
|
91
|
-
'@conditionals': 0,
|
|
92
|
-
'@coveredconditionals': 0,
|
|
93
|
-
'@methods': 0,
|
|
94
|
-
'@coveredmethods': 0,
|
|
95
|
-
},
|
|
96
|
-
line: [
|
|
97
|
-
...uncoveredLines.map((lineNumber) => ({
|
|
98
|
-
'@num': lineNumber,
|
|
99
|
-
'@count': 0,
|
|
100
|
-
'@type': 'stmt',
|
|
101
|
-
})),
|
|
102
|
-
],
|
|
103
|
-
};
|
|
104
|
-
await setCoveredLinesClover(coveredLines, uncoveredLines, repoRoot, filePath, cloverFile);
|
|
105
|
-
coverageObj.coverage.project.file.push(cloverFile);
|
|
106
|
-
const projectMetrics = coverageObj.coverage.project.metrics;
|
|
107
|
-
projectMetrics['@statements'] += uncoveredLines.length + coveredLines.length;
|
|
108
|
-
projectMetrics['@coveredstatements'] += coveredLines.length;
|
|
109
|
-
projectMetrics['@elements'] += uncoveredLines.length + coveredLines.length;
|
|
110
|
-
projectMetrics['@coveredelements'] += coveredLines.length;
|
|
111
|
-
projectMetrics['@files'] += 1;
|
|
112
|
-
projectMetrics['@classes'] += 1;
|
|
113
|
-
projectMetrics['@loc'] += uncoveredLines.length + coveredLines.length;
|
|
114
|
-
projectMetrics['@ncloc'] += uncoveredLines.length + coveredLines.length;
|
|
115
|
-
}
|
|
116
38
|
//# sourceMappingURL=transformDeployCoverageReport.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transformDeployCoverageReport.js","sourceRoot":"","sources":["../../src/helpers/transformDeployCoverageReport.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"transformDeployCoverageReport.js","sourceRoot":"","sources":["../../src/helpers/transformDeployCoverageReport.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AACb,qCAAqC;AAErC,OAAO,EAAE,kBAAkB,EAAE,MAAM,mCAAmC,CAAC;AAEvE,OAAO,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AACnE,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACrD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,MAAM,CAAC,KAAK,UAAU,6BAA6B,CACjD,IAAwB,EACxB,MAAc;IAEd,MAAM,QAAQ,GAAa,EAAE,CAAC;IAC9B,IAAI,cAAc,GAAG,CAAC,CAAC;IACvB,MAAM,EAAE,QAAQ,EAAE,kBAAkB,EAAE,GAAG,MAAM,qBAAqB,EAAE,CAAC;IACvE,MAAM,OAAO,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC;IAE3C,KAAK,MAAM,QAAQ,IAAI,IAAI,EAAE,CAAC;QAC5B,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC;YAAE,SAAS;QAE7C,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;QAChC,MAAM,iBAAiB,GAAG,QAAQ,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;QAC/D,MAAM,gBAAgB,GAAG,MAAM,YAAY,CAAC,iBAAiB,EAAE,kBAAkB,EAAE,QAAQ,CAAC,CAAC;QAE7F,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,QAAQ,CAAC,IAAI,CAAC,iBAAiB,iBAAiB,0CAA0C,CAAC,CAAC;YAC5F,SAAS;QACX,CAAC;QAED,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;aAC3C,MAAM,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;aACpD,GAAG,CAAC,MAAM,CAAC,CAAC;QACf,MAAM,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;aACzC,MAAM,CAAC,CAAC,UAAU,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;aACpD,GAAG,CAAC,MAAM,CAAC,CAAC;QAEf,MAAM,YAAY,GAAG,MAAM,aAAa,CAAC,gBAAgB,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC;QACjF,QAAQ,CAAC,CAAC,GAAG,YAAY,CAAC,CAAC,yCAAyC;QAGpE,OAAO,CAAC,WAAW,CACjB,gBAAgB,EAChB,iBAAiB,EACjB,QAAQ,CAAC,CAAC,EACV,cAAc,EACd,YAAY,CACb,CAAC;QACF,cAAc,EAAE,CAAC;IACnB,CAAC;IAED,MAAM,WAAW,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IACvC,MAAM,GAAG,GAAG,cAAc,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAEhD,OAAO,EAAE,GAAG,EAAE,QAAQ,EAAE,cAAc,EAAE,CAAC;AAC3C,CAAC"}
|
|
@@ -1,21 +1,17 @@
|
|
|
1
1
|
'use strict';
|
|
2
|
+
/* eslint-disable no-await-in-loop */
|
|
3
|
+
import { getCoverageHandler } from '../handlers/getCoverageHandler.js';
|
|
2
4
|
import { getPackageDirectories } from './getPackageDirectories.js';
|
|
3
5
|
import { findFilePath } from './findFilePath.js';
|
|
4
|
-
import {
|
|
5
|
-
import { generateXml } from './generateXml.js';
|
|
6
|
-
import { formatOptions } from './constants.js';
|
|
7
|
-
import { initializeCoverageObject } from './initializeCoverageObject.js';
|
|
6
|
+
import { generateReport } from './generateReport.js';
|
|
8
7
|
export async function transformTestCoverageReport(testCoverageData, format) {
|
|
9
|
-
if (!formatOptions.includes(format)) {
|
|
10
|
-
throw new Error(`Unsupported format: ${format}`);
|
|
11
|
-
}
|
|
12
8
|
const warnings = [];
|
|
13
9
|
let filesProcessed = 0;
|
|
14
10
|
const { repoRoot, packageDirectories } = await getPackageDirectories();
|
|
15
|
-
const
|
|
11
|
+
const handler = getCoverageHandler(format);
|
|
16
12
|
let coverageData = testCoverageData;
|
|
17
13
|
if (!Array.isArray(coverageData)) {
|
|
18
|
-
coverageData = [coverageData];
|
|
14
|
+
coverageData = [coverageData]; // Ensure the data is an array
|
|
19
15
|
}
|
|
20
16
|
for (const data of coverageData) {
|
|
21
17
|
const name = data?.name;
|
|
@@ -34,87 +30,11 @@ export async function transformTestCoverageReport(testCoverageData, format) {
|
|
|
34
30
|
const coveredLines = Object.entries(lines)
|
|
35
31
|
.filter(([, isCovered]) => isCovered === 1)
|
|
36
32
|
.map(([lineNumber]) => Number(lineNumber));
|
|
37
|
-
|
|
38
|
-
handleSonarFormat(relativeFilePath, lines, coverageObj);
|
|
39
|
-
}
|
|
40
|
-
else if (format === 'cobertura') {
|
|
41
|
-
handleCoberturaFormat(relativeFilePath, formattedFileName, lines, uncoveredLines, coveredLines, coverageObj, packageObj);
|
|
42
|
-
}
|
|
43
|
-
else {
|
|
44
|
-
handleCloverFormat(relativeFilePath, formattedFileName, lines, uncoveredLines, coveredLines, coverageObj);
|
|
45
|
-
}
|
|
33
|
+
handler.processFile(relativeFilePath, formattedFileName, lines, uncoveredLines, coveredLines);
|
|
46
34
|
filesProcessed++;
|
|
47
35
|
}
|
|
48
|
-
const
|
|
36
|
+
const coverageObj = handler.finalize();
|
|
37
|
+
const xml = generateReport(coverageObj, format);
|
|
49
38
|
return { xml, warnings, filesProcessed };
|
|
50
39
|
}
|
|
51
|
-
function handleSonarFormat(filePath, lines, coverageObj) {
|
|
52
|
-
const fileObj = {
|
|
53
|
-
'@path': normalizePathToUnix(filePath),
|
|
54
|
-
lineToCover: [],
|
|
55
|
-
};
|
|
56
|
-
for (const [lineNumber, isCovered] of Object.entries(lines)) {
|
|
57
|
-
fileObj.lineToCover.push({
|
|
58
|
-
'@lineNumber': Number(lineNumber),
|
|
59
|
-
'@covered': `${isCovered === 1}`,
|
|
60
|
-
});
|
|
61
|
-
}
|
|
62
|
-
coverageObj.coverage.file.push(fileObj);
|
|
63
|
-
}
|
|
64
|
-
function handleCoberturaFormat(filePath, fileName, lines, uncoveredLines, coveredLines, coverageObj, packageObj) {
|
|
65
|
-
const classObj = {
|
|
66
|
-
'@name': fileName,
|
|
67
|
-
'@filename': normalizePathToUnix(filePath),
|
|
68
|
-
'@line-rate': (coveredLines.length / (coveredLines.length + uncoveredLines.length)).toFixed(4),
|
|
69
|
-
'@branch-rate': '1',
|
|
70
|
-
methods: {},
|
|
71
|
-
lines: {
|
|
72
|
-
line: [],
|
|
73
|
-
},
|
|
74
|
-
};
|
|
75
|
-
for (const [lineNumber, isCovered] of Object.entries(lines)) {
|
|
76
|
-
classObj.lines.line.push({
|
|
77
|
-
'@number': Number(lineNumber),
|
|
78
|
-
'@hits': isCovered === 1 ? 1 : 0,
|
|
79
|
-
'@branch': 'false',
|
|
80
|
-
});
|
|
81
|
-
}
|
|
82
|
-
coverageObj.coverage['@lines-valid'] += uncoveredLines.length + coveredLines.length;
|
|
83
|
-
coverageObj.coverage['@lines-covered'] += coveredLines.length;
|
|
84
|
-
packageObj.classes.class.push(classObj);
|
|
85
|
-
packageObj['@line-rate'] = Number((coverageObj.coverage['@lines-covered'] / coverageObj.coverage['@lines-valid']).toFixed(4));
|
|
86
|
-
coverageObj.coverage['@line-rate'] = packageObj['@line-rate'];
|
|
87
|
-
}
|
|
88
|
-
function handleCloverFormat(filePath, fileName, lines, uncoveredLines, coveredLines, coverageObj) {
|
|
89
|
-
const cloverFile = {
|
|
90
|
-
'@name': fileName,
|
|
91
|
-
'@path': normalizePathToUnix(filePath),
|
|
92
|
-
metrics: {
|
|
93
|
-
'@statements': uncoveredLines.length + coveredLines.length,
|
|
94
|
-
'@coveredstatements': coveredLines.length,
|
|
95
|
-
'@conditionals': 0,
|
|
96
|
-
'@coveredconditionals': 0,
|
|
97
|
-
'@methods': 0,
|
|
98
|
-
'@coveredmethods': 0,
|
|
99
|
-
},
|
|
100
|
-
line: [],
|
|
101
|
-
};
|
|
102
|
-
for (const [lineNumber, isCovered] of Object.entries(lines)) {
|
|
103
|
-
cloverFile.line.push({
|
|
104
|
-
'@num': Number(lineNumber),
|
|
105
|
-
'@count': isCovered === 1 ? 1 : 0,
|
|
106
|
-
'@type': 'stmt',
|
|
107
|
-
});
|
|
108
|
-
}
|
|
109
|
-
coverageObj.coverage.project.file.push(cloverFile);
|
|
110
|
-
const projectMetrics = coverageObj.coverage.project.metrics;
|
|
111
|
-
projectMetrics['@statements'] += uncoveredLines.length + coveredLines.length;
|
|
112
|
-
projectMetrics['@coveredstatements'] += coveredLines.length;
|
|
113
|
-
projectMetrics['@elements'] += uncoveredLines.length + coveredLines.length;
|
|
114
|
-
projectMetrics['@coveredelements'] += coveredLines.length;
|
|
115
|
-
projectMetrics['@files'] += 1;
|
|
116
|
-
projectMetrics['@classes'] += 1;
|
|
117
|
-
projectMetrics['@loc'] += uncoveredLines.length + coveredLines.length;
|
|
118
|
-
projectMetrics['@ncloc'] += uncoveredLines.length + coveredLines.length;
|
|
119
|
-
}
|
|
120
40
|
//# sourceMappingURL=transformTestCoverageReport.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"transformTestCoverageReport.js","sourceRoot":"","sources":["../../src/helpers/transformTestCoverageReport.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"transformTestCoverageReport.js","sourceRoot":"","sources":["../../src/helpers/transformTestCoverageReport.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;AACb,qCAAqC;AAErC,OAAO,EAAE,kBAAkB,EAAE,MAAM,mCAAmC,CAAC;AAEvE,OAAO,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AACnE,OAAO,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAErD,MAAM,CAAC,KAAK,UAAU,2BAA2B,CAC/C,gBAAoC,EACpC,MAAc;IAEd,MAAM,QAAQ,GAAa,EAAE,CAAC;IAC9B,IAAI,cAAc,GAAG,CAAC,CAAC;IACvB,MAAM,EAAE,QAAQ,EAAE,kBAAkB,EAAE,GAAG,MAAM,qBAAqB,EAAE,CAAC;IACvE,MAAM,OAAO,GAAG,kBAAkB,CAAC,MAAM,CAAC,CAAC;IAE3C,IAAI,YAAY,GAAG,gBAAgB,CAAC;IACpC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;QACjC,YAAY,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,8BAA8B;IAC/D,CAAC;IAED,KAAK,MAAM,IAAI,IAAI,YAAY,EAAE,CAAC;QAChC,MAAM,IAAI,GAAG,IAAI,EAAE,IAAI,CAAC;QACxB,MAAM,KAAK,GAAG,IAAI,EAAE,KAAK,CAAC;QAE1B,IAAI,CAAC,IAAI,IAAI,CAAC,KAAK;YAAE,SAAS;QAE9B,MAAM,iBAAiB,GAAG,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;QAC3D,MAAM,gBAAgB,GAAG,MAAM,YAAY,CAAC,iBAAiB,EAAE,kBAAkB,EAAE,QAAQ,CAAC,CAAC;QAE7F,IAAI,gBAAgB,KAAK,SAAS,EAAE,CAAC;YACnC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,iBAAiB,0CAA0C,CAAC,CAAC;YAC5F,SAAS;QACX,CAAC;QAED,MAAM,cAAc,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC;aACzC,MAAM,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,SAAS,KAAK,CAAC,CAAC;aAC1C,GAAG,CAAC,CAAC,CAAC,UAAU,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;QAC7C,MAAM,YAAY,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC;aACvC,MAAM,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,SAAS,KAAK,CAAC,CAAC;aAC1C,GAAG,CAAC,CAAC,CAAC,UAAU,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;QAE7C,OAAO,CAAC,WAAW,CACjB,gBAAgB,EAChB,iBAAiB,EACjB,KAAK,EACL,cAAc,EACd,YAAY,CACb,CAAC;QAEF,cAAc,EAAE,CAAC;IACnB,CAAC;IAED,MAAM,WAAW,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IACvC,MAAM,GAAG,GAAG,cAAc,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAEhD,OAAO,EAAE,GAAG,EAAE,QAAQ,EAAE,cAAc,EAAE,CAAC;AAC3C,CAAC"}
|
package/lib/helpers/types.d.ts
CHANGED
|
@@ -51,7 +51,7 @@ export type SonarCoverageObject = {
|
|
|
51
51
|
export type HookFile = {
|
|
52
52
|
deployCoverageJsonPath: string;
|
|
53
53
|
testCoverageJsonPath: string;
|
|
54
|
-
|
|
54
|
+
outputReportPath: string;
|
|
55
55
|
format: string;
|
|
56
56
|
};
|
|
57
57
|
export type CoberturaLine = {
|
|
@@ -114,7 +114,7 @@ export type CloverFile = {
|
|
|
114
114
|
};
|
|
115
115
|
line: CloverLine[];
|
|
116
116
|
};
|
|
117
|
-
|
|
117
|
+
type CloverProjectMetrics = {
|
|
118
118
|
'@statements': number;
|
|
119
119
|
'@coveredstatements': number;
|
|
120
120
|
'@conditionals': number;
|
|
@@ -130,7 +130,7 @@ export type CloverProjectMetrics = {
|
|
|
130
130
|
'@files': number;
|
|
131
131
|
'@classes': number;
|
|
132
132
|
};
|
|
133
|
-
|
|
133
|
+
type CloverProject = {
|
|
134
134
|
'@timestamp': number;
|
|
135
135
|
'@name': string;
|
|
136
136
|
metrics: CloverProjectMetrics;
|
|
@@ -143,4 +143,21 @@ export type CloverCoverageObject = {
|
|
|
143
143
|
project: CloverProject;
|
|
144
144
|
};
|
|
145
145
|
};
|
|
146
|
+
export type CoverageHandler = {
|
|
147
|
+
processFile(filePath: string, fileName: string, lines: Record<string, number>, uncoveredLines: number[], coveredLines: number[]): void;
|
|
148
|
+
finalize(): SonarCoverageObject | CoberturaCoverageObject | CloverCoverageObject | LcovCoverageObject;
|
|
149
|
+
};
|
|
150
|
+
type LcovLine = {
|
|
151
|
+
lineNumber: number;
|
|
152
|
+
hitCount: number;
|
|
153
|
+
};
|
|
154
|
+
export type LcovFile = {
|
|
155
|
+
sourceFile: string;
|
|
156
|
+
lines: LcovLine[];
|
|
157
|
+
totalLines: number;
|
|
158
|
+
coveredLines: number;
|
|
159
|
+
};
|
|
160
|
+
export type LcovCoverageObject = {
|
|
161
|
+
files: LcovFile[];
|
|
162
|
+
};
|
|
146
163
|
export {};
|