@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.
114 lines • 13.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.baseUrlFromDomain = baseUrlFromDomain;
exports.getOctokit = getOctokit;
exports.getRunner = getRunner;
exports.deleteRunner = deleteRunner;
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 = {};
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);
if (octokitCache.octokit && octokitCache.installationId == installationId && octokitCache.secrets &&
octokitCache.secrets.domain == githubSecrets.domain && octokitCache.secrets.appId == githubSecrets.appId &&
octokitCache.secrets.personalAuthToken == githubSecrets.personalAuthToken) {
// test and use cache
try {
await octokitCache.octokit.rest.meta.getOctocat();
console.log('Using cached octokit');
return {
octokit: octokitCache.octokit,
githubSecrets: octokitCache.secrets,
};
}
catch (e) {
console.log('Octokit cache is invalid', e);
octokitCache.octokit = undefined;
}
}
let 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,
});
octokitCache.octokit = octokit;
octokitCache.installationId = installationId;
octokitCache.secrets = githubSecrets;
return {
octokit,
githubSecrets,
};
}
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,
});
}
}
//# 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 { createAppAuth } from '@octokit/auth-app';\nimport { Octokit } from '@octokit/rest';\nimport { getSecretValue, getSecretJsonValue } 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: {\n  installationId?: number;\n  secrets?: GitHubSecrets;\n  octokit?: Octokit;\n} = {};\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  if (octokitCache.octokit && octokitCache.installationId == installationId && octokitCache.secrets &&\n    octokitCache.secrets.domain == githubSecrets.domain && octokitCache.secrets.appId == githubSecrets.appId &&\n    octokitCache.secrets.personalAuthToken == githubSecrets.personalAuthToken) {\n    // test and use cache\n    try {\n      await octokitCache.octokit.rest.meta.getOctocat();\n      console.log('Using cached octokit');\n      return {\n        octokit: octokitCache.octokit,\n        githubSecrets: octokitCache.secrets,\n      };\n    } catch (e) {\n      console.log('Octokit cache is invalid', e);\n      octokitCache.octokit = undefined;\n    }\n  }\n\n  let 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  octokitCache.octokit = octokit;\n  octokitCache.installationId = installationId;\n  octokitCache.secrets = githubSecrets;\n\n  return {\n    octokit,\n    githubSecrets,\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"]}