UNPKG

@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
"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 type { 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"]}