aws-delivlib
Version:
A fabulous library for defining continuous pipelines for building, testing and releasing code libraries.
113 lines • 13.5 kB
JavaScript
;
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":"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","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"]}