@cloudsnorkel/cdk-github-runners
Version:
CDK construct to create GitHub Actions self-hosted runners. Creates ephemeral runners on demand. Easy to deploy and highly customizable.
149 lines • 18.6 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.clearFailuresCache = clearFailuresCache;
exports.handler = handler;
const lambda_github_1 = require("./lambda-github");
/**
* Get webhook delivery failures since the last processed delivery ID.
*
* @internal
*/
async function newDeliveryFailures(octokit, sinceId) {
const deliveries = new Map();
const successfulDeliveries = new Set();
const timeLimitMs = 1000 * 60 * 30; // don't look at deliveries over 30 minutes old
let lastId = 0;
let processedCount = 0;
for await (const response of octokit.paginate.iterator('GET /app/hook/deliveries')) {
if (response.status !== 200) {
throw new Error('Failed to fetch webhook deliveries');
}
for (const delivery of response.data) {
const deliveredAt = new Date(delivery.delivered_at);
const success = delivery.status === 'OK';
if (delivery.id <= sinceId) {
// stop processing if we reach the last processed delivery ID
console.info({
notice: 'Reached last processed delivery ID',
sinceId: sinceId,
deliveryId: delivery.id,
guid: delivery.guid,
processedCount,
});
return { deliveries, lastId };
}
lastId = Math.max(lastId, delivery.id);
if (deliveredAt.getTime() < Date.now() - timeLimitMs) {
// stop processing if the delivery is too old (for first iteration and performance of further iterations)
console.info({
notice: 'Stopping at old delivery',
deliveryId: delivery.id,
guid: delivery.guid,
deliveredAt: deliveredAt,
processedCount,
});
return { deliveries, lastId };
}
console.debug({
notice: 'Processing webhook delivery',
deliveryId: delivery.id,
guid: delivery.guid,
status: delivery.status,
deliveredAt: delivery.delivered_at,
redelivery: delivery.redelivery,
});
processedCount++;
if (success) {
successfulDeliveries.add(delivery.guid);
continue;
}
if (successfulDeliveries.has(delivery.guid)) {
// do not redeliver deliveries that were already successful
continue;
}
deliveries.set(delivery.guid, { id: delivery.id, deliveredAt, redelivery: delivery.redelivery });
}
}
console.info({
notice: 'No more webhook deliveries to process',
deliveryId: 'DONE',
guid: 'DONE',
deliveredAt: 'DONE',
processedCount,
});
return { deliveries, lastId };
}
let lastDeliveryIdProcessed = 0;
const failures = new Map();
/**
* Clear the cache of webhook delivery failures.
*
* For unit testing purposes only.
*
* @internal
*/
function clearFailuresCache() {
lastDeliveryIdProcessed = 0;
failures.clear();
}
async function handler() {
const octokit = await (0, lambda_github_1.getAppOctokit)();
if (!octokit) {
console.info({
notice: 'Skipping webhook redelivery',
reason: 'App installation might not be configured or the app is not installed.',
});
return;
}
// fetch deliveries since the last processed delivery ID
// for any failures:
// 1. if this is not a redelivery, save the delivery ID and time, and finally retry
// 2. if this is a redelivery, check if the original delivery is still within the time limit and retry if it is
const { deliveries, lastId } = await newDeliveryFailures(octokit, lastDeliveryIdProcessed);
lastDeliveryIdProcessed = Math.max(lastDeliveryIdProcessed, lastId);
const timeLimitMs = 1000 * 60 * 60 * 3; // retry for up to 3 hours
for (const [guid, details] of deliveries) {
if (!details.redelivery) {
failures.set(guid, { id: details.id, firstDeliveredAt: details.deliveredAt });
console.log({
notice: 'Redelivering failed delivery',
deliveryId: details.id,
guid: guid,
firstDeliveredAt: details.deliveredAt,
});
await (0, lambda_github_1.redeliver)(octokit, details.id);
}
else {
// if this is a redelivery, check if the original delivery is still within the time limit
const originalFailure = failures.get(guid);
if (originalFailure) {
if (new Date().getTime() - originalFailure.firstDeliveredAt.getTime() < timeLimitMs) {
console.log({
notice: 'Redelivering failed delivery',
deliveryId: details.id,
guid: guid,
firstDeliveredAt: originalFailure.firstDeliveredAt,
});
await (0, lambda_github_1.redeliver)(octokit, details.id);
}
else {
failures.delete(guid); // no need to keep track of this anymore
console.log({
notice: 'Skipping redelivery of old failed delivery',
deliveryId: details.id,
guid: guid,
firstDeliveredAt: originalFailure?.firstDeliveredAt,
});
}
}
else {
console.log({
notice: 'Skipping redelivery of old failed delivery',
deliveryId: details.id,
guid: guid,
});
}
}
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhook-redelivery.lambda.js","sourceRoot":"","sources":["../src/webhook-redelivery.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { Octokit } from '@octokit/rest';\nimport { getAppOctokit, redeliver } from './lambda-github';\n\n/**\n * Get webhook delivery failures since the last processed delivery ID.\n *\n * @internal\n */\nasync function newDeliveryFailures(octokit: Octokit, sinceId: number) {\n  const deliveries: Map<string, { id: number; deliveredAt: Date; redelivery: boolean }> = new Map();\n  const successfulDeliveries: Set<string> = new Set();\n  const timeLimitMs = 1000 * 60 * 30; // don't look at deliveries over 30 minutes old\n  let lastId = 0;\n  let processedCount = 0;\n\n  for await (const response of octokit.paginate.iterator('GET /app/hook/deliveries')) {\n    if (response.status !== 200) {\n      throw new Error('Failed to fetch webhook deliveries');\n    }\n\n    for (const delivery of response.data) {\n      const deliveredAt = new Date(delivery.delivered_at);\n      const success = delivery.status === 'OK';\n\n      if (delivery.id <= sinceId) {\n        // stop processing if we reach the last processed delivery ID\n        console.info({\n          notice: 'Reached last processed delivery ID',\n          sinceId: sinceId,\n          deliveryId: delivery.id,\n          guid: delivery.guid,\n          processedCount,\n        });\n        return { deliveries, lastId };\n      }\n\n      lastId = Math.max(lastId, delivery.id);\n\n      if (deliveredAt.getTime() < Date.now() - timeLimitMs) {\n        // stop processing if the delivery is too old (for first iteration and performance of further iterations)\n        console.info({\n          notice: 'Stopping at old delivery',\n          deliveryId: delivery.id,\n          guid: delivery.guid,\n          deliveredAt: deliveredAt,\n          processedCount,\n        });\n        return { deliveries, lastId };\n      }\n\n      console.debug({\n        notice: 'Processing webhook delivery',\n        deliveryId: delivery.id,\n        guid: delivery.guid,\n        status: delivery.status,\n        deliveredAt: delivery.delivered_at,\n        redelivery: delivery.redelivery,\n      });\n      processedCount++;\n\n      if (success) {\n        successfulDeliveries.add(delivery.guid);\n        continue;\n      }\n\n      if (successfulDeliveries.has(delivery.guid)) {\n        // do not redeliver deliveries that were already successful\n        continue;\n      }\n\n      deliveries.set(delivery.guid, { id: delivery.id, deliveredAt, redelivery: delivery.redelivery });\n    }\n  }\n\n  console.info({\n    notice: 'No more webhook deliveries to process',\n    deliveryId: 'DONE',\n    guid: 'DONE',\n    deliveredAt: 'DONE',\n    processedCount,\n  });\n\n  return { deliveries, lastId };\n}\n\nlet lastDeliveryIdProcessed = 0;\nconst failures: Map<string, { id: number; firstDeliveredAt: Date }> = new Map();\n\n/**\n * Clear the cache of webhook delivery failures.\n *\n * For unit testing purposes only.\n *\n * @internal\n */\nexport function clearFailuresCache() {\n  lastDeliveryIdProcessed = 0;\n  failures.clear();\n}\n\nexport async function handler() {\n  const octokit = await getAppOctokit();\n  if (!octokit) {\n    console.info({\n      notice: 'Skipping webhook redelivery',\n      reason: 'App installation might not be configured or the app is not installed.',\n    });\n    return;\n  }\n\n  // fetch deliveries since the last processed delivery ID\n  // for any failures:\n  //  1. if this is not a redelivery, save the delivery ID and time, and finally retry\n  //  2. if this is a redelivery, check if the original delivery is still within the time limit and retry if it is\n  const { deliveries, lastId } = await newDeliveryFailures(octokit, lastDeliveryIdProcessed);\n  lastDeliveryIdProcessed = Math.max(lastDeliveryIdProcessed, lastId);\n  const timeLimitMs = 1000 * 60 * 60 * 3; // retry for up to 3 hours\n  for (const [guid, details] of deliveries) {\n    if (!details.redelivery) {\n      failures.set(guid, { id: details.id, firstDeliveredAt: details.deliveredAt });\n      console.log({\n        notice: 'Redelivering failed delivery',\n        deliveryId: details.id,\n        guid: guid,\n        firstDeliveredAt: details.deliveredAt,\n      });\n      await redeliver(octokit, details.id);\n    } else {\n      // if this is a redelivery, check if the original delivery is still within the time limit\n      const originalFailure = failures.get(guid);\n      if (originalFailure) {\n        if (new Date().getTime() - originalFailure.firstDeliveredAt.getTime() < timeLimitMs) {\n          console.log({\n            notice: 'Redelivering failed delivery',\n            deliveryId: details.id,\n            guid: guid,\n            firstDeliveredAt: originalFailure.firstDeliveredAt,\n          });\n          await redeliver(octokit, details.id);\n        } else {\n          failures.delete(guid); // no need to keep track of this anymore\n          console.log({\n            notice: 'Skipping redelivery of old failed delivery',\n            deliveryId: details.id,\n            guid: guid,\n            firstDeliveredAt: originalFailure?.firstDeliveredAt,\n          });\n        }\n      } else {\n        console.log({\n          notice: 'Skipping redelivery of old failed delivery',\n          deliveryId: details.id,\n          guid: guid,\n        });\n      }\n    }\n  }\n}\n"]}