From cbad58b59d7bbd1df2ef2a6a2b5c8b76a122771f Mon Sep 17 00:00:00 2001 From: Egor Bryzgalov Date: Mon, 13 Jul 2020 12:31:32 +0300 Subject: [PATCH] [JenkinsQueueJobV2] Improved error handling (#13242) * Improved http errors handling * Reworked logging code * Replaced console.log with console.error and fixed tests * Passed Error instance to console.error instead of message field to show more info --- Tasks/JenkinsQueueJobV2/Tests/L0.ts | 2 +- .../JenkinsQueueJobV2/jenkinsqueuejobtask.ts | 16 ++++++++-- Tasks/JenkinsQueueJobV2/job.ts | 2 +- Tasks/JenkinsQueueJobV2/task.json | 2 +- Tasks/JenkinsQueueJobV2/task.loc.json | 2 +- Tasks/JenkinsQueueJobV2/util.ts | 29 ++++++++++++++----- 6 files changed, 38 insertions(+), 15 deletions(-) diff --git a/Tasks/JenkinsQueueJobV2/Tests/L0.ts b/Tasks/JenkinsQueueJobV2/Tests/L0.ts index dd980b76f1c9..6a025c7b0822 100644 --- a/Tasks/JenkinsQueueJobV2/Tests/L0.ts +++ b/Tasks/JenkinsQueueJobV2/Tests/L0.ts @@ -141,7 +141,7 @@ describe('JenkinsQueueJob L0 Suite', function () { }); it('run JenkinsQueueJob with bogus url with parameters', (done) => { - const tp: string = path.join(__dirname, 'L0BogusUrlNoParameters.js'); + const tp: string = path.join(__dirname, 'L0BogusUrlParameters.js'); const tr: ttm.MockTestRunner = new ttm.MockTestRunner(tp); try { diff --git a/Tasks/JenkinsQueueJobV2/jenkinsqueuejobtask.ts b/Tasks/JenkinsQueueJobV2/jenkinsqueuejobtask.ts index f18d2142d0ee..326dc46083d7 100644 --- a/Tasks/JenkinsQueueJobV2/jenkinsqueuejobtask.ts +++ b/Tasks/JenkinsQueueJobV2/jenkinsqueuejobtask.ts @@ -116,9 +116,19 @@ async function doWork() { //store the job name in the output variable tl.setVariable('JENKINS_JOB_ID', rootJob.ExecutableNumber.toString()); } catch (e) { - tl.debug(e.message); - process.stderr.write(e + os.EOL); - tl.setResult(tl.TaskResult.Failed, e.message); + let message: string; + if (e instanceof util.HttpError) { + message = e.message; + console.error(e.fullMessage); + console.error(e.body); + } else if (e instanceof Error) { + message = e.message; + console.error(e); + } else { + message = e; + console.error(e); + } + tl.setResult(tl.TaskResult.Failed, message); } } diff --git a/Tasks/JenkinsQueueJobV2/job.ts b/Tasks/JenkinsQueueJobV2/job.ts index b9f16d296d1f..2f498e19a670 100644 --- a/Tasks/JenkinsQueueJobV2/job.ts +++ b/Tasks/JenkinsQueueJobV2/job.ts @@ -296,8 +296,8 @@ export class Job { thisJob.stopWork(thisJob.queue.TaskOptions.pollIntervalMillis, thisJob.State); return; } else if (httpResponse.statusCode !== 200) { + console.error(`Job was killed because of an response with unexpected status code from Jenkins - ${httpResponse.statusCode}`); Util.failReturnCode(httpResponse, 'Job progress tracking failed to read job result'); - tl.error(`Job was killed because of an response with unexpected status code from Jenkins - ${httpResponse.statusCode}`); thisJob.stopWork(0, JobState.Killed); } else { const parsedBody: {result: string, timestamp: number} = JSON.parse(body); diff --git a/Tasks/JenkinsQueueJobV2/task.json b/Tasks/JenkinsQueueJobV2/task.json index bf129b9906c8..50f0e03fa24d 100644 --- a/Tasks/JenkinsQueueJobV2/task.json +++ b/Tasks/JenkinsQueueJobV2/task.json @@ -14,7 +14,7 @@ "demands": [], "version": { "Major": 2, - "Minor": 171, + "Minor": 173, "Patch": 0 }, "groups": [ diff --git a/Tasks/JenkinsQueueJobV2/task.loc.json b/Tasks/JenkinsQueueJobV2/task.loc.json index 8d9dd0327b6a..1315a711233e 100644 --- a/Tasks/JenkinsQueueJobV2/task.loc.json +++ b/Tasks/JenkinsQueueJobV2/task.loc.json @@ -14,7 +14,7 @@ "demands": [], "version": { "Major": 2, - "Minor": 171, + "Minor": 173, "Patch": 0 }, "groups": [ diff --git a/Tasks/JenkinsQueueJobV2/util.ts b/Tasks/JenkinsQueueJobV2/util.ts index 5e98da7a9fca..976d28981533 100644 --- a/Tasks/JenkinsQueueJobV2/util.ts +++ b/Tasks/JenkinsQueueJobV2/util.ts @@ -19,9 +19,7 @@ export function getFullErrorMessage(httpResponse, message: string): string { export function failReturnCode(httpResponse, message: string): void { const fullMessage = getFullErrorMessage(httpResponse, message); - tl.debug(message); - tl.error(fullMessage); - process.stderr.write(message + os.EOL); + console.error(fullMessage); tl.setResult(tl.TaskResult.Failed, message); } @@ -40,6 +38,22 @@ export function fail(message: string): void { export class FailTaskError extends Error { } +/** + * @class Represents error based on HttpResponse + * @extends {Error} Error class + */ +export class HttpError extends Error { + public body: string; + public fullMessage: string; + + constructor(httpResponse: any, message: string) { + super(); + this.fullMessage = getFullErrorMessage(httpResponse, message); + this.message = message; + this.body = httpResponse.body; + } +} + export function convertJobName(jobName: string): string { return '/job/' + jobName.replace('/', '/job/'); } @@ -135,7 +149,7 @@ function createRootJob(queueUri: string, jobQueue: JobQueue, taskOptions: TaskOp defer.reject(error); } } else if (httpResponse.statusCode !== 200) { - defer.reject(getFullErrorMessage(httpResponse, 'Job progress tracking failed to read job queue')); + defer.reject(new HttpError(httpResponse, 'Job progress tracking failed to read job queue')); } else { const parsedBody: any = JSON.parse(body); tl.debug(`parsedBody for: ${queueUri} : ${JSON.stringify(parsedBody)}`); @@ -256,14 +270,14 @@ function submitJob(taskOptions: TaskOptions): Q.Promise { defer.reject(err); } } else if (httpResponse.statusCode !== 201) { - defer.reject(getFullErrorMessage(httpResponse, 'Job creation failed.')); + defer.reject(new HttpError(httpResponse, 'Job creation failed.')); } else { const queueUri: string = addUrlSegment(httpResponse.headers.location, 'api/json'); defer.resolve(queueUri); } }).auth(taskOptions.username, taskOptions.password, true); } else if (httpResponse.statusCode !== 201) { - defer.reject(getFullErrorMessage(httpResponse, 'Job creation failed.')); + defer.reject(new HttpError(httpResponse, 'Job creation failed.')); } else { taskOptions.teamBuildPluginAvailable = true; const jsonBody: any = JSON.parse(body); @@ -293,8 +307,7 @@ function getCrumb(taskOptions: TaskOptions): Q.Promise { taskOptions.crumb = taskOptions.NO_CRUMB; defer.resolve(taskOptions.NO_CRUMB); } else if (httpResponse.statusCode !== 200) { - failReturnCode(httpResponse, 'crumb request failed.'); - defer.reject(getFullErrorMessage(httpResponse, 'Crumb request failed.')); + defer.reject(new HttpError(httpResponse, 'Crumb request failed.')); } else { taskOptions.crumb = body; tl.debug('crumb: ' + taskOptions.crumb);