@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.
155 lines • 17.9 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.baseUrlFromDomain = baseUrlFromDomain;
exports.getOctokit = getOctokit;
exports.getAppOctokit = getAppOctokit;
exports.getRunner = getRunner;
exports.deleteRunner = deleteRunner;
exports.redeliver = redeliver;
const crypto_1 = require("crypto");
const auth_app_1 = require("@octokit/auth-app");
const rest_1 = require("@octokit/rest");
const lambda_helpers_1 = require("./lambda-helpers");
function baseUrlFromDomain(domain) {
if (domain == 'github.com') {
return 'https://api.github.com';
}
return `https://${domain}/api/v3`;
}
const octokitCache = new Map();
async function getOctokit(installationId) {
if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {
throw new Error('Missing environment variables');
}
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
// Create cache key from installation ID and secrets (hash to avoid exposing sensitive data by accident)
const cacheKey = (0, crypto_1.createHash)('sha256').update(`${installationId || 'no-install'}-${githubSecrets.domain}-${githubSecrets.appId}-${githubSecrets.personalAuthToken}`).digest('hex');
const cached = octokitCache.get(cacheKey);
if (cached) {
try {
// Test if the cached octokit is still valid
await cached.rest.meta.getOctocat();
console.log({
notice: 'Using cached octokit',
});
return {
octokit: cached,
githubSecrets,
};
}
catch (e) {
console.log({
notice: 'Octokit cache is invalid',
error: e,
});
octokitCache.delete(cacheKey);
}
}
const baseUrl = baseUrlFromDomain(githubSecrets.domain);
let token;
if (githubSecrets.personalAuthToken) {
token = githubSecrets.personalAuthToken;
}
else {
const privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
const appOctokit = new rest_1.Octokit({
baseUrl,
authStrategy: auth_app_1.createAppAuth,
auth: {
appId: githubSecrets.appId,
privateKey: privateKey,
},
});
token = (await appOctokit.auth({
type: 'installation',
installationId: installationId,
})).token;
}
const octokit = new rest_1.Octokit({
baseUrl,
auth: token,
});
// Store in cache
octokitCache.set(cacheKey, octokit);
return {
octokit,
githubSecrets,
};
}
// This function is used to get the Octokit instance for the app itself, not for a specific installation.
// With PAT authentication, it returns undefined.
async function getAppOctokit() {
if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {
throw new Error('Missing environment variables');
}
const githubSecrets = await (0, lambda_helpers_1.getSecretJsonValue)(process.env.GITHUB_SECRET_ARN);
const baseUrl = baseUrlFromDomain(githubSecrets.domain);
if (githubSecrets.personalAuthToken || !githubSecrets.appId) {
return undefined;
}
const privateKey = await (0, lambda_helpers_1.getSecretValue)(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);
return new rest_1.Octokit({
baseUrl,
authStrategy: auth_app_1.createAppAuth,
auth: {
appId: githubSecrets.appId,
privateKey: privateKey,
},
});
}
async function getRunner(octokit, runnerLevel, owner, repo, name) {
let page = 1;
while (true) {
let runners;
if ((runnerLevel ?? 'repo') === 'repo') {
runners = await octokit.rest.actions.listSelfHostedRunnersForRepo({
page: page,
owner: owner,
repo: repo,
});
}
else {
runners = await octokit.rest.actions.listSelfHostedRunnersForOrg({
page: page,
org: owner,
});
}
if (runners.data.runners.length == 0) {
return;
}
for (const runner of runners.data.runners) {
if (runner.name == name) {
return runner;
}
}
page++;
}
}
async function deleteRunner(octokit, runnerLevel, owner, repo, runnerId) {
if ((runnerLevel ?? 'repo') === 'repo') {
await octokit.rest.actions.deleteSelfHostedRunnerFromRepo({
owner: owner,
repo: repo,
runner_id: runnerId,
});
}
else {
await octokit.rest.actions.deleteSelfHostedRunnerFromOrg({
org: owner,
runner_id: runnerId,
});
}
}
async function redeliver(octokit, deliveryId) {
const response = await octokit.rest.apps.redeliverWebhookDelivery({
delivery_id: deliveryId,
});
if (response.status !== 202) {
throw new Error(`Failed to redeliver webhook delivery with ID ${deliveryId}`);
}
console.log({
notice: 'Successfully redelivered webhook delivery',
deliveryId,
});
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-github.js","sourceRoot":"","sources":["../src/lambda-github.ts"],"names":[],"mappings":"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","sourcesContent":["import { createHash } from 'crypto';\nimport { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/rest';\nimport { getSecretJsonValue, getSecretValue } from './lambda-helpers';\n\nexport function baseUrlFromDomain(domain: string): string {\n  if (domain == 'github.com') {\n    return 'https://api.github.com';\n  }\n  return `https://${domain}/api/v3`;\n}\n\ntype RunnerLevel = 'repo' | 'org' | undefined; // undefined is for backwards compatibility and should be treated as 'repo'\n\nexport interface GitHubSecrets {\n  domain: string;\n  appId: number;\n  personalAuthToken: string;\n  runnerLevel: RunnerLevel;\n}\n\nconst octokitCache = new Map<string, Octokit>();\n\nexport async function getOctokit(installationId?: number): Promise<{ octokit: Octokit; githubSecrets: GitHubSecrets }> {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n\n  // Create cache key from installation ID and secrets (hash to avoid exposing sensitive data by accident)\n  const cacheKey = createHash('sha256').update(`${installationId || 'no-install'}-${githubSecrets.domain}-${githubSecrets.appId}-${githubSecrets.personalAuthToken}`).digest('hex');\n\n  const cached = octokitCache.get(cacheKey);\n  if (cached) {\n    try {\n      // Test if the cached octokit is still valid\n      await cached.rest.meta.getOctocat();\n      console.log({\n        notice: 'Using cached octokit',\n      });\n      return {\n        octokit: cached,\n        githubSecrets,\n      };\n    } catch (e) {\n      console.log({\n        notice: 'Octokit cache is invalid',\n        error: e,\n      });\n      octokitCache.delete(cacheKey);\n    }\n  }\n\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  let token;\n  if (githubSecrets.personalAuthToken) {\n    token = githubSecrets.personalAuthToken;\n  } else {\n    const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n    const appOctokit = new Octokit({\n      baseUrl,\n      authStrategy: createAppAuth,\n      auth: {\n        appId: githubSecrets.appId,\n        privateKey: privateKey,\n      },\n    });\n\n    token = (await appOctokit.auth({\n      type: 'installation',\n      installationId: installationId,\n    }) as any).token;\n  }\n\n  const octokit = new Octokit({\n    baseUrl,\n    auth: token,\n  });\n\n  // Store in cache\n  octokitCache.set(cacheKey, octokit);\n\n  return {\n    octokit,\n    githubSecrets,\n  };\n}\n\n// This function is used to get the Octokit instance for the app itself, not for a specific installation.\n// With PAT authentication, it returns undefined.\nexport async function getAppOctokit() {\n  if (!process.env.GITHUB_SECRET_ARN || !process.env.GITHUB_PRIVATE_KEY_SECRET_ARN) {\n    throw new Error('Missing environment variables');\n  }\n\n  const githubSecrets: GitHubSecrets = await getSecretJsonValue(process.env.GITHUB_SECRET_ARN);\n  const baseUrl = baseUrlFromDomain(githubSecrets.domain);\n\n  if (githubSecrets.personalAuthToken || !githubSecrets.appId) {\n    return undefined;\n  }\n\n  const privateKey = await getSecretValue(process.env.GITHUB_PRIVATE_KEY_SECRET_ARN);\n\n  return new Octokit({\n    baseUrl,\n    authStrategy: createAppAuth,\n    auth: {\n      appId: githubSecrets.appId,\n      privateKey: privateKey,\n    },\n  });\n}\n\nexport async function getRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, name: string) {\n  let page = 1;\n  while (true) {\n    let runners;\n\n    if ((runnerLevel ?? 'repo') === 'repo') {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForRepo({\n        page: page,\n        owner: owner,\n        repo: repo,\n      });\n    } else {\n      runners = await octokit.rest.actions.listSelfHostedRunnersForOrg({\n        page: page,\n        org: owner,\n      });\n    }\n\n    if (runners.data.runners.length == 0) {\n      return;\n    }\n\n    for (const runner of runners.data.runners) {\n      if (runner.name == name) {\n        return runner;\n      }\n    }\n\n    page++;\n  }\n}\n\nexport async function deleteRunner(octokit: Octokit, runnerLevel: RunnerLevel, owner: string, repo: string, runnerId: number) {\n  if ((runnerLevel ?? 'repo') === 'repo') {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromRepo({\n      owner: owner,\n      repo: repo,\n      runner_id: runnerId,\n    });\n  } else {\n    await octokit.rest.actions.deleteSelfHostedRunnerFromOrg({\n      org: owner,\n      runner_id: runnerId,\n    });\n  }\n}\n\nexport async function redeliver(octokit: Octokit, deliveryId: number) {\n  const response = await octokit.rest.apps.redeliverWebhookDelivery({\n    delivery_id: deliveryId,\n  });\n\n  if (response.status !== 202) {\n    throw new Error(`Failed to redeliver webhook delivery with ID ${deliveryId}`);\n  }\n  console.log({\n    notice: 'Successfully redelivered webhook delivery',\n    deliveryId,\n  });\n}\n"]}