diff --git a/lib/ci/ci_failure_parser.js b/lib/ci/ci_failure_parser.js
index e782eb3e..8a91d935 100644
--- a/lib/ci/ci_failure_parser.js
+++ b/lib/ci/ci_failure_parser.js
@@ -33,10 +33,11 @@ const CC_TEST_FAILURE = 'CC_TEST_FAILURE';
const JENKINS_FAILURE = 'JENKINS_FAILURE';
const GIT_FAILURE = 'GIT_FAILURE';
const NCU_FAILURE = 'NCU_FAILURE';
+const RESUME_FAILURE = 'RESUME_FAILURE';
const FAILURE_TYPES = {
BUILD_FAILURE, JS_TEST_FAILURE, CC_TEST_FAILURE,
- JENKINS_FAILURE, GIT_FAILURE, NCU_FAILURE
+ JENKINS_FAILURE, GIT_FAILURE, NCU_FAILURE, RESUME_FAILURE
};
class CIResult {
@@ -103,6 +104,14 @@ class NCUFailure extends CIResult {
}
}
+// Refs: https://github.com/nodejs/build/issues/1496
+class ResumeFailure extends CIResult {
+ constructor(ctx, reason) {
+ super(ctx, reason);
+ this.type = RESUME_FAILURE;
+ }
+}
+
function failureMatcher(Failure, patterns, ctx, text) {
for (const pattern of patterns) {
const matches = text.match(pattern.pattern);
@@ -277,7 +286,8 @@ CIFailureParser.FAILURE_CONSTRUCTORS = {
JS_TEST_FAILURE: JSTestFailure,
CC_TEST_FAILURE: CCTestFailure,
GIT_FAILURE: GitFailure,
- NCU_FAILURE: NCUFailure
+ NCU_FAILURE: NCUFailure,
+ RESUME_FAILURE: ResumeFailure
};
CIFailureParser.CIResult = CIResult;
CIFailureParser.FAILURE_TYPES_NAME = {
@@ -286,6 +296,7 @@ CIFailureParser.FAILURE_TYPES_NAME = {
JS_TEST_FAILURE: 'JSTest Failure',
CC_TEST_FAILURE: 'CCTest Failure',
GIT_FAILURE: 'Git Failure',
- NCU_FAILURE: 'node-core-utils failure'
+ NCU_FAILURE: 'node-core-utils failure',
+ RESUME_FAILURE: 'resume failure'
};
module.exports = CIFailureParser;
diff --git a/lib/ci/ci_result_parser.js b/lib/ci/ci_result_parser.js
index 9d0567ba..e7da6d05 100644
--- a/lib/ci/ci_result_parser.js
+++ b/lib/ci/ci_result_parser.js
@@ -8,11 +8,14 @@ const CIFailureParser = require('./ci_failure_parser');
const {
FAILURE_TYPES: {
BUILD_FAILURE,
- NCU_FAILURE
+ NCU_FAILURE,
+ GIT_FAILURE,
+ RESUME_FAILURE
},
FAILURE_CONSTRUCTORS: {
[BUILD_FAILURE]: BuildFailure,
- [NCU_FAILURE]: NCUFailure
+ [NCU_FAILURE]: NCUFailure,
+ [RESUME_FAILURE]: ResumeFailure
},
CIResult,
FAILURE_TYPES_NAME
@@ -46,11 +49,12 @@ const COMMIT_TREE =
`subBuilds[${BUILD_FIELDS}]`;
// com.tikal.jenkins.plugins.multijob.MultiJobBuild
const FANNED_TREE =
- `result,url,number,subBuilds[phaseName,${BUILD_FIELDS}],builtOn`;
+ `result,url,number,subBuilds[phaseName,${BUILD_FIELDS}]`;
// hudson.matrix.MatrixBuild
const BUILD_TREE = 'result,runs[url,number,result],builtOn';
const LINTER_TREE = 'result,url,number,builtOn';
-const RUN_TREE = 'actions[causes[upstreamBuild,upstreamProject]],builtOn';
+const CAUSE_TREE = 'upstreamBuild,upstreamProject,shortDescription,_class';
+const RUN_TREE = `actions[causes[${CAUSE_TREE}]],builtOn`;
function getPath(url) {
return url.replace(`https://${CI_DOMAIN}/`, '').replace('api/json', '');
@@ -128,6 +132,13 @@ class Job {
return data;
}
+ getCause(actions) {
+ if (actions && actions.find(item => item.causes)) {
+ const action = actions.find(item => item.causes);
+ return action.causes[0];
+ }
+ }
+
async getAPIData() {
const { cli, request, path } = this;
const url = this.apiUrl;
@@ -355,7 +366,9 @@ function getHighlight(f) {
)
.replace(
/fatal: loose object \w+ \(stored in .git\/objects\/.+\) is corrupt/,
- 'fatal: loose object ... (stored in .git/objects/...) is corrupt');
+ 'fatal: loose object ... (stored in .git/objects/...) is corrupt')
+ .replace(/hudson\.plugins\.git\.GitException: /, '')
+ .replace(/java\.io\.IOException: /, '');
}
function markdownRow(...args) {
@@ -427,13 +440,15 @@ class FailureAggregator {
output += `[${jobName}/${first.jobid}](${first.link}) to `;
output += `[${jobName}/${last.jobid}](${last.link}) `;
output += `that failed more than 2 PRs\n`;
+ output += `(Generated with \`ncu-ci `;
+ output += `${process.argv.slice(2).join(' ')}\`)\n`;
const todo = [];
for (const type of Object.keys(aggregates)) {
output += `\n### ${FAILURE_TYPES_NAME[type]}\n\n`;
for (const item of aggregates[type]) {
const { reason, type, prs, failures, machines } = item;
if (prs.length < 2) { continue; }
- todo.push(reason);
+ todo.push({ count: prs.length, reason });
output += markdownRow('Reason', `${reason}
`);
output += markdownRow('-', ':-');
output += markdownRow('Type', type);
@@ -449,16 +464,19 @@ class FailureAggregator {
}
output += markdownRow('Last CI', `${prs[prs.length - 1].upstream}`);
output += '\n';
+ const example = failures[0].reason;
output += fold(
`Example`,
- failures[0].reason
+ (example.length > 1024 ? example.slice(0, 1024) + '...' : example)
);
output += '\n\n-------\n\n';
}
}
output += '### Progress\n\n';
- output += todo.map(i => `- \`${i}\``).join('\n');
+ output += todo.map(
+ ({count, reason}) => `- \`${reason}\` (${count})`).join('\n'
+ );
return output + '\n';
}
@@ -559,7 +577,10 @@ class CommitBuild extends TestBuild {
cli.startSpinner(`Querying failures of ${path}`);
const promises = builds.failed.map(({jobName, buildNumber, url}) => {
if (jobName.includes('fanned')) {
- return new FannedBuild(cli, request, jobName, buildNumber).getResults();
+ const cause = this.getCause(data.actions);
+ const isResumed = cause && cause._class.includes('ResumeCause');
+ return new FannedBuild(cli, request, jobName, buildNumber, isResumed)
+ .getResults();
} else if (jobName.includes('linter')) {
return new LinterBuild(cli, request, jobName, buildNumber).getResults();
} else if (jobName.includes('freestyle')) {
@@ -622,6 +643,7 @@ class PRBuild extends TestBuild {
const allBuilds = commitBuild.build.subBuilds;
// TODO: fetch result, builtOn, timestamp in the commit build's own data
// ..or maybe they do not worth an additional API call?
+ // Note that we have to pass the actions down to detect resume builds.
const buildData = {
result, subBuilds: allBuilds, changeSet, actions, timestamp
};
@@ -682,7 +704,7 @@ async function listBuilds(cli, request, type) {
}
class FannedBuild extends Job {
- constructor(cli, request, jobName, id) {
+ constructor(cli, request, jobName, id, isResumed) {
// assert(jobName.includes('fanned'));
const path = `job/${jobName}/${id}/`;
const tree = FANNED_TREE;
@@ -690,6 +712,7 @@ class FannedBuild extends Job {
this.failures = [];
this.builtOn = undefined;
+ this.isResumed = isResumed;
}
// Get the failures and their reasons of this build
@@ -724,8 +747,9 @@ class FannedBuild extends Job {
!failedPhase.phaseName.toLowerCase().includes('compilation')) {
this.failures = [
new BuildFailure(
- { url: failedPhase.url, builtOn: failedPhase.builtOn },
- `Failed in ${failedPhase.phaseName} phase`
+ { url: failedPhase.url },
+ `Failed in ${failedPhase.phaseName} phase (${failedPhase.jobName})`
+ // TODO: parse console text for the failed phase
)
];
return this.failures;
@@ -733,8 +757,23 @@ class FannedBuild extends Job {
const { jobName, buildNumber } = failedPhase;
const build = new NormalBuild(cli, request, jobName, buildNumber);
- const failures = await build.getResults();
- this.failures = flatten(failures);
+ let failures = await build.getResults();
+ failures = flatten(failures);
+
+ if (this.isResumed) {
+ // XXX: if it's a resumed build, we replace the build/git failures
+ // with resume failures. Probably just a random guess, though
+ for (let i = 0; i < failures.length; ++i) {
+ const item = failures[i];
+ if (item.type === BUILD_FAILURE || item.type === GIT_FAILURE) {
+ failures[i] = new ResumeFailure(
+ item,
+ `Possible resume failure\n${item.reason}`
+ );
+ }
+ }
+ }
+ this.failures = failures;
return this.failures;
}
}
@@ -853,10 +892,7 @@ class TestRun extends Job {
];
return this.failures;
}
- if (data.actions && data.actions.find(item => item.causes)) {
- const actions = data.actions.find(item => item.causes);
- this.cause = actions.causes[0];
- }
+ this.causes = this.getCause(data.actions) || {};
this.builtOn = data.builtOn;
}