forked from guyarb/golang-test-annotations
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
96 lines (92 loc) · 3.69 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
const core = require('@actions/core');
const lineReader = require('line-by-line');
const fs = require('fs');
const path = require('path');
try {
const testResultsPath = core.getInput('test-results');;
if (!fs.existsSync(testResultsPath)) {
core.warning(
`No file was found with the provided path: ${testResultsPath}.`
)
return
}
let obj = {};
let lr = new lineReader(testResultsPath);
lr.on('line', function (line) {
const currentLine = JSON.parse(line);
const testName = currentLine.Test;
if (typeof testName === "undefined") {
return;
}
// Strip github.com/owner/repo package from the path by default
let packageName = currentLine.Package.split("/").slice(3).join("/");
let key = packageName + "/" + testName;
if (currentLine.Action === "output") {
let output = currentLine.Output;
if (typeof output === "undefined") {
return;
}
// output = output.replace("\n", "%0A").replace("\r", "%0D")
if (/^(=== RUN|\s*--- (FAIL|PASS): )/.test(output ?? '')) {
return;
}
obj[key] ??= { output: [], packageName };
obj[key].output.push(output);
}
if (currentLine.Action === "fail") {
obj[key] ??= { output: [], packageName };
obj[key].status = "FAIL";
}
});
lr.on('end', function () {
const messages = [];
for (const { output, status, packageName } of Object.values(obj)) {
if (status !== "FAIL") {
continue;
}
let current
for (const line of output) {
// ^(?:.*\s+|\s*) - non-greedy match of any chars followed by a space or, a space.
// (?<file>\S+\.go):(?<line>\d+): - gofile:line: followed by a space.
// (?<message>.\n)$ - all remaining message up to $.
const m = line.match(/^.*\s+(?<file>\S+\.go):(?<line>\d+): (?<message>.*\n)$/);
if (m?.groups) {
const file = m.groups.file && path.isAbsolute(m.groups.file) ? m.groups.file : path.join(packageName, m.groups.file);
const startLine = Number(m.groups.line);
current = { file ,startLine };
messages.push({ message: m.groups.message, location: current });
} else if (current) {
messages.push({ message: line, location: current });
}
}
if (!current) {
// parse golang stack trace
const testingStackFrame = /(\S+testing\.go):(\d+).*\n?/g;
const stackFrame = /(?<file>\S+\.go):(?<startLine>\d+).*\n?/g;
const lastStackFrame = output
.join("")
.replace(testingStackFrame, "")
.match(stackFrame)
?.pop() ?? "";
const { file, startLine } = stackFrame.exec(lastStackFrame)?.groups ?? {};
if (file !== undefined && startLine !== undefined) {
messages.push({ message: output.join(""), location: { file: file && path.isAbsolute(file) ? file : path.join(packageName, file), startLine: Number(startLine) } });
}
}
}
const merged = new Map();
for (const { message, location } of messages) {
const loc = `${location?.file}:${location?.startLine}`;
if (merged.has(loc)) {
merged.get(loc).message += '' + message;
} else {
merged.set(loc, { message, location });
}
}
[...merged.values()].forEach(({ message, location }) => {
core.error(message, location);
});
});
} catch (error) {
core.setFailed(error.message);
}