UNPKG

aws-delivlib

Version:

A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.

113 lines • 13.5 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = exports.codePipeline = void 0; const https = __importStar(require("https")); // eslint-disable-next-line import/no-extraneous-dependencies const client_codepipeline_1 = require("@aws-sdk/client-codepipeline"); // export for tests exports.codePipeline = new client_codepipeline_1.CodePipeline(); /** * Lambda handler for the codepipeline state change events * * { * "version": "0", * "id": event_Id, * "detail-type": "CodePipeline Pipeline Execution State Change", * "source": "aws.codepipeline", * "account": Pipeline_Account, * "time": TimeStamp, * "region": "us-east-1", * "resources": [ * "arn:aws:codepipeline:us-east-1:account_ID:myPipeline" * ], * "detail": { * "pipeline": "myPipeline", * "version": "1", * "state": "STARTED", * "execution-id": execution_Id * } * } */ async function handler(event) { // Log the event so we can have a look in CloudWatch logs process.stdout.write(`${JSON.stringify(event)}\n`); const webhookUrls = event.webhookUrls || []; if (webhookUrls.length === 0) { throw new Error("Expected event field 'webhookUrls'"); } const messageTemplate = event.message; if (!messageTemplate) { throw new Error("Expected event field 'message'"); } const details = event.detail || {}; const pipelineName = details.pipeline; const pipelineExecutionId = details['execution-id']; if (!pipelineName || !pipelineExecutionId) { process.stderr.write('Malformed event!\n'); return; } // Describe the revision that caused the pipeline to fail const response = await exports.codePipeline.getPipelineExecution({ pipelineName, pipelineExecutionId }); process.stdout.write(`${JSON.stringify(response)}\n`); const firstArtifact = (response.pipelineExecution?.artifactRevisions ?? [])[0]; const revisionSummary = firstArtifact?.revisionSummary ?? firstArtifact?.revisionId ?? `execution ${pipelineExecutionId}`; // Find the action that caused the pipeline to fail (no pagination for now) const actionResponse = await exports.codePipeline.listActionExecutions({ pipelineName, filter: { pipelineExecutionId } }); process.stdout.write(`${JSON.stringify(actionResponse)}\n`); const failingActionDetails = actionResponse.actionExecutionDetails?.find(d => d.status === 'Failed'); const failingAction = failingActionDetails?.actionName || 'UNKNOWN'; const failureUrl = failingActionDetails?.output?.executionResult?.externalExecutionUrl || '???'; const message = messageTemplate .replace(/\$PIPELINE/g, pipelineName) .replace(/\$REVISION/g, revisionSummary) .replace(/\$ACTION/g, failingAction) .replace(/\$URL/g, failureUrl); // Post the failure to all given Chime webhook URLs await Promise.all(webhookUrls.map(url => sendChimeNotification(url, message))); } exports.handler = handler; async function sendChimeNotification(url, message) { return new Promise((ok, ko) => { const req = https.request(url, { method: 'POST', headers: { 'Content-Type': 'application/json', }, }, (res) => { if (res.statusCode !== 200) { ko(new Error(`Server responded with ${res.statusCode}: ${JSON.stringify(res.headers)}`)); } res.setEncoding('utf8'); res.on('data', () => { }); res.on('error', ko); res.on('end', ok); }); req.on('error', ko); req.write(JSON.stringify({ Content: message })); req.end(); }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"notifier-handler.js","sourceRoot":"","sources":["notifier-handler.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,6CAA+B;AAC/B,6DAA6D;AAC7D,sEAA8E;AAG9E,mBAAmB;AACN,QAAA,YAAY,GAAG,IAAI,kCAAY,EAAE,CAAC;AAE/C;;;;;;;;;;;;;;;;;;;;;GAqBG;AACI,KAAK,UAAU,OAAO,CAAC,KAAU;IACtC,yDAAyD;IACzD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAEnD,MAAM,WAAW,GAAa,KAAK,CAAC,WAAW,IAAI,EAAE,CAAC;IACtD,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;QAAE,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;KAAE;IAExF,MAAM,eAAe,GAAG,KAAK,CAAC,OAAO,CAAC;IACtC,IAAI,CAAC,eAAe,EAAE;QAAE,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;KAAE;IAE5E,MAAM,OAAO,GAAG,KAAK,CAAC,MAAM,IAAI,EAAE,CAAC;IACnC,MAAM,YAAY,GAAG,OAAO,CAAC,QAAQ,CAAC;IACtC,MAAM,mBAAmB,GAAG,OAAO,CAAC,cAAc,CAAC,CAAC;IAEpD,IAAI,CAAC,YAAY,IAAI,CAAC,mBAAmB,EAAE;QACzC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;QAC3C,OAAO;KACR;IAED,yDAAyD;IACzD,MAAM,QAAQ,GAAG,MAAM,oBAAY,CAAC,oBAAoB,CAAC,EAAE,YAAY,EAAE,mBAAmB,EAAE,CAAC,CAAC;IAChG,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IACtD,MAAM,aAAa,GAAiC,CAAC,QAAQ,CAAC,iBAAiB,EAAE,iBAAiB,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7G,MAAM,eAAe,GAAG,aAAa,EAAE,eAAe,IAAI,aAAa,EAAE,UAAU,IAAI,aAAa,mBAAmB,EAAE,CAAC;IAE1H,2EAA2E;IAC3E,MAAM,cAAc,GAAG,MAAM,oBAAY,CAAC,oBAAoB,CAAC,EAAE,YAAY,EAAE,MAAM,EAAE,EAAE,mBAAmB,EAAE,EAAE,CAAC,CAAC;IAClH,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC5D,MAAM,oBAAoB,GAAG,cAAc,CAAC,sBAAsB,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;IACrG,MAAM,aAAa,GAAG,oBAAoB,EAAE,UAAU,IAAI,SAAS,CAAC;IACpE,MAAM,UAAU,GAAG,oBAAoB,EAAE,MAAM,EAAE,eAAe,EAAE,oBAAoB,IAAI,KAAK,CAAC;IAEhG,MAAM,OAAO,GAAG,eAAe;SAC5B,OAAO,CAAC,aAAa,EAAE,YAAY,CAAC;SACpC,OAAO,CAAC,aAAa,EAAE,eAAe,CAAC;SACvC,OAAO,CAAC,WAAW,EAAE,aAAa,CAAC;SACnC,OAAO,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;IAEjC,mDAAmD;IACnD,MAAM,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,qBAAqB,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC;AACjF,CAAC;AAxCD,0BAwCC;AAED,KAAK,UAAU,qBAAqB,CAAC,GAAW,EAAE,OAAe;IAC/D,OAAO,IAAI,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE;QAC5B,MAAM,GAAG,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE;YAC7B,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;aACnC;SACF,EAAE,CAAC,GAAG,EAAE,EAAE;YACT,IAAI,GAAG,CAAC,UAAU,KAAK,GAAG,EAAE;gBAC1B,EAAE,CAAC,IAAI,KAAK,CAAC,yBAAyB,GAAG,CAAC,UAAU,KAAK,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;aAC1F;YAED,GAAG,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YACxB,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE,GAAkC,CAAC,CAAC,CAAC;YACzD,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;YACpB,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACpB,CAAC,CAAC,CAAC;QAEH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;QACpB,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;QAChD,GAAG,CAAC,GAAG,EAAE,CAAC;IACZ,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import * as https from 'https';\n// eslint-disable-next-line import/no-extraneous-dependencies\nimport { ArtifactRevision, CodePipeline } from '@aws-sdk/client-codepipeline';\n\n\n// export for tests\nexport const codePipeline = new CodePipeline();\n\n/**\n * Lambda handler for the codepipeline state change events\n *\n * {\n *     \"version\": \"0\",\n *     \"id\": event_Id,\n *     \"detail-type\": \"CodePipeline Pipeline Execution State Change\",\n *     \"source\": \"aws.codepipeline\",\n *     \"account\": Pipeline_Account,\n *     \"time\": TimeStamp,\n *     \"region\": \"us-east-1\",\n *     \"resources\": [\n *         \"arn:aws:codepipeline:us-east-1:account_ID:myPipeline\"\n *     ],\n *     \"detail\": {\n *         \"pipeline\": \"myPipeline\",\n *         \"version\": \"1\",\n *         \"state\": \"STARTED\",\n *         \"execution-id\": execution_Id\n *     }\n * }\n */\nexport async function handler(event: any) {\n  // Log the event so we can have a look in CloudWatch logs\n  process.stdout.write(`${JSON.stringify(event)}\\n`);\n\n  const webhookUrls: string[] = event.webhookUrls || [];\n  if (webhookUrls.length === 0) { throw new Error(\"Expected event field 'webhookUrls'\"); }\n\n  const messageTemplate = event.message;\n  if (!messageTemplate) { throw new Error(\"Expected event field 'message'\"); }\n\n  const details = event.detail || {};\n  const pipelineName = details.pipeline;\n  const pipelineExecutionId = details['execution-id'];\n\n  if (!pipelineName || !pipelineExecutionId) {\n    process.stderr.write('Malformed event!\\n');\n    return;\n  }\n\n  // Describe the revision that caused the pipeline to fail\n  const response = await codePipeline.getPipelineExecution({ pipelineName, pipelineExecutionId });\n  process.stdout.write(`${JSON.stringify(response)}\\n`);\n  const firstArtifact: ArtifactRevision | undefined = (response.pipelineExecution?.artifactRevisions ?? [])[0];\n  const revisionSummary = firstArtifact?.revisionSummary ?? firstArtifact?.revisionId ?? `execution ${pipelineExecutionId}`;\n\n  // Find the action that caused the pipeline to fail (no pagination for now)\n  const actionResponse = await codePipeline.listActionExecutions({ pipelineName, filter: { pipelineExecutionId } });\n  process.stdout.write(`${JSON.stringify(actionResponse)}\\n`);\n  const failingActionDetails = actionResponse.actionExecutionDetails?.find(d => d.status === 'Failed');\n  const failingAction = failingActionDetails?.actionName || 'UNKNOWN';\n  const failureUrl = failingActionDetails?.output?.executionResult?.externalExecutionUrl || '???';\n\n  const message = messageTemplate\n    .replace(/\\$PIPELINE/g, pipelineName)\n    .replace(/\\$REVISION/g, revisionSummary)\n    .replace(/\\$ACTION/g, failingAction)\n    .replace(/\\$URL/g, failureUrl);\n\n  // Post the failure to all given Chime webhook URLs\n  await Promise.all(webhookUrls.map(url => sendChimeNotification(url, message)));\n}\n\nasync function sendChimeNotification(url: string, message: string) {\n  return new Promise((ok, ko) => {\n    const req = https.request(url, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n      },\n    }, (res) => {\n      if (res.statusCode !== 200) {\n        ko(new Error(`Server responded with ${res.statusCode}: ${JSON.stringify(res.headers)}`));\n      }\n\n      res.setEncoding('utf8');\n      res.on('data', () => { /* gobble gobble and ignore */ });\n      res.on('error', ko);\n      res.on('end', ok);\n    });\n\n    req.on('error', ko);\n    req.write(JSON.stringify({ Content: message }));\n    req.end();\n  });\n}\n"]}