@cloudsnorkel/cdk-github-runners 0.3.2 → 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/.gitattributes +1 -0
  2. package/.jsii +1413 -431
  3. package/API.md +1082 -126
  4. package/README.md +12 -10
  5. package/lib/index.d.ts +2 -1
  6. package/lib/index.js +4 -1
  7. package/lib/lambdas/aws-image-builder-versioner/index.js +2469 -0
  8. package/lib/lambdas/build-image/index.js +77 -43
  9. package/lib/lambdas/delete-runner/index.js +114 -79
  10. package/lib/lambdas/setup/index.html +10 -10
  11. package/lib/lambdas/setup/index.js +27 -12
  12. package/lib/lambdas/status/index.js +147 -82
  13. package/lib/lambdas/token-retriever/index.js +114 -79
  14. package/lib/lambdas/update-lambda/index.js +5 -2
  15. package/lib/lambdas/webhook-handler/index.js +11 -5
  16. package/lib/providers/codebuild.d.ts +5 -1
  17. package/lib/providers/codebuild.js +16 -6
  18. package/lib/providers/common.d.ts +25 -6
  19. package/lib/providers/common.js +4 -4
  20. package/lib/providers/docker-images/fargate/linux-arm64/Dockerfile +0 -2
  21. package/lib/providers/docker-images/fargate/linux-x64/Dockerfile +0 -3
  22. package/lib/providers/fargate.d.ts +6 -1
  23. package/lib/providers/fargate.js +56 -15
  24. package/lib/providers/image-builders/codebuild.d.ts +6 -2
  25. package/lib/providers/image-builders/codebuild.js +14 -11
  26. package/lib/providers/image-builders/container.d.ts +220 -0
  27. package/lib/providers/image-builders/container.js +509 -0
  28. package/lib/providers/image-builders/static.js +2 -3
  29. package/lib/providers/lambda.d.ts +6 -1
  30. package/lib/providers/lambda.js +79 -8
  31. package/lib/runner.js +29 -13
  32. package/lib/secrets.js +1 -1
  33. package/package.json +16 -12
  34. package/lib/providers/docker-images/fargate/linux-arm64/runner.sh +0 -5
  35. package/lib/providers/docker-images/fargate/linux-x64/runner.sh +0 -5
@@ -16,7 +16,10 @@ var __copyProps = (to, from, except, desc) => {
16
16
  }
17
17
  return to;
18
18
  };
19
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod));
19
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
21
+ mod
22
+ ));
20
23
 
21
24
  // node_modules/universal-user-agent/dist-node/index.js
22
25
  var require_dist_node = __commonJS({
@@ -5951,11 +5954,14 @@ var require_lib3 = __commonJS({
5951
5954
  blob() {
5952
5955
  let ct = this.headers && this.headers.get("content-type") || "";
5953
5956
  return consumeBody.call(this).then(function(buf) {
5954
- return Object.assign(new Blob([], {
5955
- type: ct.toLowerCase()
5956
- }), {
5957
- [BUFFER]: buf
5958
- });
5957
+ return Object.assign(
5958
+ new Blob([], {
5959
+ type: ct.toLowerCase()
5960
+ }),
5961
+ {
5962
+ [BUFFER]: buf
5963
+ }
5964
+ );
5959
5965
  });
5960
5966
  },
5961
5967
  json() {
@@ -7012,12 +7018,15 @@ var require_dist_node5 = __commonJS({
7012
7018
  let status;
7013
7019
  let url;
7014
7020
  const fetch = requestOptions.request && requestOptions.request.fetch || nodeFetch;
7015
- return fetch(requestOptions.url, Object.assign({
7016
- method: requestOptions.method,
7017
- body: requestOptions.body,
7018
- headers: requestOptions.headers,
7019
- redirect: requestOptions.redirect
7020
- }, requestOptions.request)).then(async (response) => {
7021
+ return fetch(requestOptions.url, Object.assign(
7022
+ {
7023
+ method: requestOptions.method,
7024
+ body: requestOptions.body,
7025
+ headers: requestOptions.headers,
7026
+ redirect: requestOptions.redirect
7027
+ },
7028
+ requestOptions.request
7029
+ )).then(async (response) => {
7021
7030
  url = response.url;
7022
7031
  status = response.status;
7023
7032
  for (const keyAndValue of response.headers) {
@@ -7391,8 +7400,6 @@ var require_dist_node6 = __commonJS({
7391
7400
  } else {
7392
7401
  if (Object.keys(remainingParameters).length) {
7393
7402
  body = remainingParameters;
7394
- } else {
7395
- headers["content-length"] = 0;
7396
7403
  }
7397
7404
  }
7398
7405
  }
@@ -7425,7 +7432,7 @@ var require_dist_node6 = __commonJS({
7425
7432
  parse
7426
7433
  });
7427
7434
  }
7428
- var VERSION = "7.0.0";
7435
+ var VERSION = "7.0.2";
7429
7436
  var userAgent = `octokit-endpoint.js/${VERSION} ${universalUserAgent.getUserAgent()}`;
7430
7437
  var DEFAULTS = {
7431
7438
  method: "GET",
@@ -7511,7 +7518,7 @@ var require_dist_node8 = __commonJS({
7511
7518
  var isPlainObject = require_is_plain_object();
7512
7519
  var nodeFetch = _interopDefault(require_lib3());
7513
7520
  var requestError = require_dist_node7();
7514
- var VERSION = "6.2.0";
7521
+ var VERSION = "6.2.1";
7515
7522
  function getBufferResponse(response) {
7516
7523
  return response.arrayBuffer();
7517
7524
  }
@@ -7524,12 +7531,15 @@ var require_dist_node8 = __commonJS({
7524
7531
  let status;
7525
7532
  let url;
7526
7533
  const fetch = requestOptions.request && requestOptions.request.fetch || globalThis.fetch || nodeFetch;
7527
- return fetch(requestOptions.url, Object.assign({
7528
- method: requestOptions.method,
7529
- body: requestOptions.body,
7530
- headers: requestOptions.headers,
7531
- redirect: requestOptions.redirect
7532
- }, requestOptions.request)).then(async (response) => {
7534
+ return fetch(requestOptions.url, Object.assign(
7535
+ {
7536
+ method: requestOptions.method,
7537
+ body: requestOptions.body,
7538
+ headers: requestOptions.headers,
7539
+ redirect: requestOptions.redirect
7540
+ },
7541
+ requestOptions.request
7542
+ )).then(async (response) => {
7533
7543
  url = response.url;
7534
7544
  status = response.status;
7535
7545
  for (const keyAndValue of response.headers) {
@@ -7946,8 +7956,6 @@ var require_dist_node10 = __commonJS({
7946
7956
  } else {
7947
7957
  if (Object.keys(remainingParameters).length) {
7948
7958
  body = remainingParameters;
7949
- } else {
7950
- headers["content-length"] = 0;
7951
7959
  }
7952
7960
  }
7953
7961
  }
@@ -7980,7 +7988,7 @@ var require_dist_node10 = __commonJS({
7980
7988
  parse
7981
7989
  });
7982
7990
  }
7983
- var VERSION = "7.0.0";
7991
+ var VERSION = "7.0.2";
7984
7992
  var userAgent = `octokit-endpoint.js/${VERSION} ${universalUserAgent.getUserAgent()}`;
7985
7993
  var DEFAULTS = {
7986
7994
  method: "GET",
@@ -8066,7 +8074,7 @@ var require_dist_node12 = __commonJS({
8066
8074
  var isPlainObject = require_is_plain_object();
8067
8075
  var nodeFetch = _interopDefault(require_lib3());
8068
8076
  var requestError = require_dist_node11();
8069
- var VERSION = "6.2.0";
8077
+ var VERSION = "6.2.1";
8070
8078
  function getBufferResponse(response) {
8071
8079
  return response.arrayBuffer();
8072
8080
  }
@@ -8079,12 +8087,15 @@ var require_dist_node12 = __commonJS({
8079
8087
  let status;
8080
8088
  let url;
8081
8089
  const fetch = requestOptions.request && requestOptions.request.fetch || globalThis.fetch || nodeFetch;
8082
- return fetch(requestOptions.url, Object.assign({
8083
- method: requestOptions.method,
8084
- body: requestOptions.body,
8085
- headers: requestOptions.headers,
8086
- redirect: requestOptions.redirect
8087
- }, requestOptions.request)).then(async (response) => {
8090
+ return fetch(requestOptions.url, Object.assign(
8091
+ {
8092
+ method: requestOptions.method,
8093
+ body: requestOptions.body,
8094
+ headers: requestOptions.headers,
8095
+ redirect: requestOptions.redirect
8096
+ },
8097
+ requestOptions.request
8098
+ )).then(async (response) => {
8088
8099
  url = response.url;
8089
8100
  status = response.status;
8090
8101
  for (const keyAndValue of response.headers) {
@@ -8217,7 +8228,7 @@ var require_dist_node13 = __commonJS({
8217
8228
  var request = require_dist_node12();
8218
8229
  var requestError = require_dist_node11();
8219
8230
  var btoa = _interopDefault(require_btoa_node());
8220
- var VERSION = "2.0.2";
8231
+ var VERSION = "2.0.3";
8221
8232
  function requestToOAuthBaseUrl(request2) {
8222
8233
  const endpointDefaults = request2.endpoint.DEFAULTS;
8223
8234
  return /^https:\/\/(api\.)?github\.com$/.test(endpointDefaults.baseUrl) ? "https://github.com" : endpointDefaults.baseUrl.replace("/api/v3", "");
@@ -8376,21 +8387,25 @@ var require_dist_node13 = __commonJS({
8376
8387
  }
8377
8388
  async function scopeToken(options) {
8378
8389
  const {
8379
- request: request$1,
8390
+ request: optionsRequest,
8380
8391
  clientType,
8381
8392
  clientId,
8382
8393
  clientSecret,
8383
8394
  token,
8384
8395
  ...requestOptions
8385
8396
  } = options;
8386
- const response = await (request$1 || request.request)("POST /applications/{client_id}/token/scoped", {
8387
- headers: {
8388
- authorization: `basic ${btoa(`${clientId}:${clientSecret}`)}`
8389
- },
8390
- client_id: clientId,
8391
- access_token: token,
8392
- ...requestOptions
8393
- });
8397
+ const request$1 = optionsRequest || request.request;
8398
+ const response = await request$1(
8399
+ "POST /applications/{client_id}/token/scoped",
8400
+ {
8401
+ headers: {
8402
+ authorization: `basic ${btoa(`${clientId}:${clientSecret}`)}`
8403
+ },
8404
+ client_id: clientId,
8405
+ access_token: token,
8406
+ ...requestOptions
8407
+ }
8408
+ );
8394
8409
  const authentication = Object.assign({
8395
8410
  clientType,
8396
8411
  clientId,
@@ -8407,13 +8422,16 @@ var require_dist_node13 = __commonJS({
8407
8422
  async function resetToken(options) {
8408
8423
  const request$1 = options.request || request.request;
8409
8424
  const auth = btoa(`${options.clientId}:${options.clientSecret}`);
8410
- const response = await request$1("PATCH /applications/{client_id}/token", {
8411
- headers: {
8412
- authorization: `basic ${auth}`
8413
- },
8414
- client_id: options.clientId,
8415
- access_token: options.token
8416
- });
8425
+ const response = await request$1(
8426
+ "PATCH /applications/{client_id}/token",
8427
+ {
8428
+ headers: {
8429
+ authorization: `basic ${auth}`
8430
+ },
8431
+ client_id: options.clientId,
8432
+ access_token: options.token
8433
+ }
8434
+ );
8417
8435
  const authentication = {
8418
8436
  clientType: options.clientType,
8419
8437
  clientId: options.clientId,
@@ -8434,24 +8452,30 @@ var require_dist_node13 = __commonJS({
8434
8452
  async function deleteToken(options) {
8435
8453
  const request$1 = options.request || request.request;
8436
8454
  const auth = btoa(`${options.clientId}:${options.clientSecret}`);
8437
- return request$1("DELETE /applications/{client_id}/token", {
8438
- headers: {
8439
- authorization: `basic ${auth}`
8440
- },
8441
- client_id: options.clientId,
8442
- access_token: options.token
8443
- });
8455
+ return request$1(
8456
+ "DELETE /applications/{client_id}/token",
8457
+ {
8458
+ headers: {
8459
+ authorization: `basic ${auth}`
8460
+ },
8461
+ client_id: options.clientId,
8462
+ access_token: options.token
8463
+ }
8464
+ );
8444
8465
  }
8445
8466
  async function deleteAuthorization(options) {
8446
8467
  const request$1 = options.request || request.request;
8447
8468
  const auth = btoa(`${options.clientId}:${options.clientSecret}`);
8448
- return request$1("DELETE /applications/{client_id}/grant", {
8449
- headers: {
8450
- authorization: `basic ${auth}`
8451
- },
8452
- client_id: options.clientId,
8453
- access_token: options.token
8454
- });
8469
+ return request$1(
8470
+ "DELETE /applications/{client_id}/grant",
8471
+ {
8472
+ headers: {
8473
+ authorization: `basic ${auth}`
8474
+ },
8475
+ client_id: options.clientId,
8476
+ access_token: options.token
8477
+ }
8478
+ );
8455
8479
  }
8456
8480
  exports2.VERSION = VERSION;
8457
8481
  exports2.checkToken = checkToken;
@@ -8565,7 +8589,7 @@ var require_dist_node14 = __commonJS({
8565
8589
  endpoint.headers.authorization = `token ${token}`;
8566
8590
  return request2(endpoint);
8567
8591
  }
8568
- var VERSION = "4.0.0";
8592
+ var VERSION = "4.0.2";
8569
8593
  function createOAuthDeviceAuth(options) {
8570
8594
  const requestWithDefaults = options.request || request.request.defaults({
8571
8595
  headers: {
@@ -8613,7 +8637,7 @@ var require_dist_node15 = __commonJS({
8613
8637
  var authOauthDevice = require_dist_node14();
8614
8638
  var oauthMethods = require_dist_node13();
8615
8639
  var btoa = _interopDefault(require_btoa_node());
8616
- var VERSION = "2.0.2";
8640
+ var VERSION = "2.0.3";
8617
8641
  async function getAuthentication(state) {
8618
8642
  if ("code" in state.strategyOptions) {
8619
8643
  const {
@@ -9792,14 +9816,18 @@ var require_ms = __commonJS({
9792
9816
  } else if (type === "number" && isFinite(val)) {
9793
9817
  return options.long ? fmtLong(val) : fmtShort(val);
9794
9818
  }
9795
- throw new Error("val is not a non-empty string or a valid number. val=" + JSON.stringify(val));
9819
+ throw new Error(
9820
+ "val is not a non-empty string or a valid number. val=" + JSON.stringify(val)
9821
+ );
9796
9822
  };
9797
9823
  function parse(str) {
9798
9824
  str = String(str);
9799
9825
  if (str.length > 100) {
9800
9826
  return;
9801
9827
  }
9802
- var match = /^(-?(?:\d+)?\.?\d+) *(milliseconds?|msecs?|ms|seconds?|secs?|s|minutes?|mins?|m|hours?|hrs?|h|days?|d|weeks?|w|years?|yrs?|y)?$/i.exec(str);
9828
+ var match = /^(-?(?:\d+)?\.?\d+) *(milliseconds?|msecs?|ms|seconds?|secs?|s|minutes?|mins?|m|hours?|hrs?|h|days?|d|weeks?|w|years?|yrs?|y)?$/i.exec(
9829
+ str
9830
+ );
9803
9831
  if (!match) {
9804
9832
  return;
9805
9833
  }
@@ -12705,8 +12733,6 @@ var require_dist_node18 = __commonJS({
12705
12733
  } else {
12706
12734
  if (Object.keys(remainingParameters).length) {
12707
12735
  body = remainingParameters;
12708
- } else {
12709
- headers["content-length"] = 0;
12710
12736
  }
12711
12737
  }
12712
12738
  }
@@ -12739,7 +12765,7 @@ var require_dist_node18 = __commonJS({
12739
12765
  parse
12740
12766
  });
12741
12767
  }
12742
- var VERSION = "7.0.0";
12768
+ var VERSION = "7.0.2";
12743
12769
  var userAgent = `octokit-endpoint.js/${VERSION} ${universalUserAgent.getUserAgent()}`;
12744
12770
  var DEFAULTS = {
12745
12771
  method: "GET",
@@ -12825,7 +12851,7 @@ var require_dist_node20 = __commonJS({
12825
12851
  var isPlainObject = require_is_plain_object();
12826
12852
  var nodeFetch = _interopDefault(require_lib3());
12827
12853
  var requestError = require_dist_node19();
12828
- var VERSION = "6.2.0";
12854
+ var VERSION = "6.2.1";
12829
12855
  function getBufferResponse(response) {
12830
12856
  return response.arrayBuffer();
12831
12857
  }
@@ -12838,12 +12864,15 @@ var require_dist_node20 = __commonJS({
12838
12864
  let status;
12839
12865
  let url;
12840
12866
  const fetch = requestOptions.request && requestOptions.request.fetch || globalThis.fetch || nodeFetch;
12841
- return fetch(requestOptions.url, Object.assign({
12842
- method: requestOptions.method,
12843
- body: requestOptions.body,
12844
- headers: requestOptions.headers,
12845
- redirect: requestOptions.redirect
12846
- }, requestOptions.request)).then(async (response) => {
12867
+ return fetch(requestOptions.url, Object.assign(
12868
+ {
12869
+ method: requestOptions.method,
12870
+ body: requestOptions.body,
12871
+ headers: requestOptions.headers,
12872
+ redirect: requestOptions.redirect
12873
+ },
12874
+ requestOptions.request
12875
+ )).then(async (response) => {
12847
12876
  url = response.url;
12848
12877
  status = response.status;
12849
12878
  for (const keyAndValue of response.headers) {
@@ -14096,7 +14125,9 @@ var require_dist_node25 = __commonJS({
14096
14125
  case "app":
14097
14126
  return getAppAuthentication(state);
14098
14127
  case "oauth":
14099
- state.log.warn(new deprecation.Deprecation(`[@octokit/auth-app] {type: "oauth"} is deprecated. Use {type: "oauth-app"} instead`));
14128
+ state.log.warn(
14129
+ new deprecation.Deprecation(`[@octokit/auth-app] {type: "oauth"} is deprecated. Use {type: "oauth-app"} instead`)
14130
+ );
14100
14131
  case "oauth-app":
14101
14132
  return state.oauthApp({
14102
14133
  type: "oauth-app"
@@ -14169,7 +14200,11 @@ var require_dist_node25 = __commonJS({
14169
14200
  const {
14170
14201
  token,
14171
14202
  createdAt
14172
- } = await getInstallationAuthentication(state, {}, request2);
14203
+ } = await getInstallationAuthentication(
14204
+ state,
14205
+ {},
14206
+ request2
14207
+ );
14173
14208
  endpoint.headers.authorization = `token ${token}`;
14174
14209
  return sendRequestWithRetries(state, request2, endpoint, createdAt);
14175
14210
  }
@@ -17,7 +17,10 @@ var __copyProps = (to, from, except, desc) => {
17
17
  }
18
18
  return to;
19
19
  };
20
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod));
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
22
+ mod
23
+ ));
21
24
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
22
25
 
23
26
  // src/lambdas/update-lambda/index.ts
@@ -31,7 +34,7 @@ var cfn = new AWS.CloudFormation();
31
34
  var lambda = new AWS.Lambda();
32
35
  async function handler(event) {
33
36
  var _a;
34
- console.log(event);
37
+ console.log(JSON.stringify(event));
35
38
  const stacks = await cfn.describeStacks({
36
39
  StackName: event.stackName
37
40
  }).promise();
@@ -13,7 +13,10 @@ var __copyProps = (to, from, except, desc) => {
13
13
  }
14
14
  return to;
15
15
  };
16
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target, mod));
16
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
17
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
18
+ mod
19
+ ));
17
20
 
18
21
  // src/lambdas/webhook-handler/index.ts
19
22
  var crypto = __toESM(require("crypto"));
@@ -40,11 +43,14 @@ async function getSecretJsonValue(arn) {
40
43
  var sf = new AWS2.StepFunctions();
41
44
  function verifyBody(event, secret) {
42
45
  const sig = Buffer.from(event.headers["x-hub-signature-256"] || "", "utf8");
43
- let body = event.body;
46
+ if (!event.body) {
47
+ throw new Error("No body");
48
+ }
49
+ let body;
44
50
  if (event.isBase64Encoded) {
45
- body = Buffer.from(body, "base64");
51
+ body = Buffer.from(event.body, "base64");
46
52
  } else {
47
- body = Buffer.from(body || "", "utf8");
53
+ body = Buffer.from(event.body || "", "utf8");
48
54
  }
49
55
  const hmac = crypto.createHmac("sha256", secret);
50
56
  hmac.update(body);
@@ -53,7 +59,7 @@ function verifyBody(event, secret) {
53
59
  if (sig.length !== expectedSig.length || !crypto.timingSafeEqual(sig, expectedSig)) {
54
60
  throw new Error(`Signature mismatch. Expected ${expectedSig.toString()} but got ${sig.toString()}`);
55
61
  }
56
- return body;
62
+ return body.toString();
57
63
  }
58
64
  exports.handler = async function(event) {
59
65
  var _a;
@@ -1,6 +1,6 @@
1
1
  import { aws_codebuild as codebuild, aws_ec2 as ec2, aws_iam as iam, aws_stepfunctions as stepfunctions, Duration } from 'aws-cdk-lib';
2
2
  import { Construct } from 'constructs';
3
- import { IImageBuilder, IRunnerProvider, RunnerProviderProps, RunnerRuntimeParameters } from './common';
3
+ import { IImageBuilder, IRunnerProvider, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';
4
4
  export interface CodeBuildRunnerProps extends RunnerProviderProps {
5
5
  /**
6
6
  * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist with access to Docker-in-Docker.
@@ -100,6 +100,10 @@ export declare class CodeBuildRunner extends Construct implements IRunnerProvide
100
100
  * Grant principal used to add permissions to the runner role.
101
101
  */
102
102
  readonly grantPrincipal: iam.IPrincipal;
103
+ /**
104
+ * Docker image in CodeBuild project.
105
+ */
106
+ readonly image: RunnerImage;
103
107
  constructor(scope: Construct, id: string, props: CodeBuildRunnerProps);
104
108
  /**
105
109
  * Generate step function task(s) to start a new runner.
@@ -24,7 +24,7 @@ class CodeBuildRunner extends constructs_1.Construct {
24
24
  this.label = props.label ?? 'codebuild';
25
25
  this.vpc = props.vpc;
26
26
  this.securityGroup = props.securityGroup;
27
- const buildSpec = {
27
+ let buildSpec = {
28
28
  version: '0.2',
29
29
  env: {
30
30
  variables: {
@@ -54,7 +54,17 @@ class CodeBuildRunner extends constructs_1.Construct {
54
54
  const imageBuilder = props.imageBuilder ?? new codebuild_1.CodeBuildImageBuilder(this, 'Image Builder', {
55
55
  dockerfilePath: CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH,
56
56
  });
57
- const image = imageBuilder.bind();
57
+ const image = this.image = imageBuilder.bind();
58
+ if (image.os.is(common_1.Os.WINDOWS)) {
59
+ buildSpec.phases.install.commands = [
60
+ 'cd \\actions',
61
+ './config.cmd --unattended --url "https://${Env:GITHUB_DOMAIN}/${Env:OWNER}/${Env:REPO}" --token "${Env:RUNNER_TOKEN}" --ephemeral --work _work --labels "${Env:RUNNER_LABEL}" --disableupdate --name "${Env:RUNNER_NAME}"',
62
+ ];
63
+ buildSpec.phases.build.commands = [
64
+ 'cd \\actions',
65
+ './run.cmd',
66
+ ];
67
+ }
58
68
  // choose build image
59
69
  let buildImage;
60
70
  if (image.os.is(common_1.Os.LINUX)) {
@@ -67,7 +77,7 @@ class CodeBuildRunner extends constructs_1.Construct {
67
77
  }
68
78
  if (image.os.is(common_1.Os.WINDOWS)) {
69
79
  if (image.architecture.is(common_1.Architecture.X86_64)) {
70
- buildImage = aws_cdk_lib_1.aws_codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);
80
+ buildImage = aws_cdk_lib_1.aws_codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag, aws_cdk_lib_1.aws_codebuild.WindowsImageType.SERVER_2019);
71
81
  }
72
82
  }
73
83
  if (buildImage === undefined) {
@@ -84,7 +94,7 @@ class CodeBuildRunner extends constructs_1.Construct {
84
94
  environment: {
85
95
  buildImage,
86
96
  computeType: props.computeType ?? aws_codebuild_1.ComputeType.SMALL,
87
- privileged: true,
97
+ privileged: image.os.is(common_1.Os.LINUX),
88
98
  },
89
99
  logging: {
90
100
  cloudWatch: {
@@ -145,7 +155,7 @@ class CodeBuildRunner extends constructs_1.Construct {
145
155
  }
146
156
  exports.CodeBuildRunner = CodeBuildRunner;
147
157
  _a = JSII_RTTI_SYMBOL_1;
148
- CodeBuildRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildRunner", version: "0.3.2" };
158
+ CodeBuildRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CodeBuildRunner", version: "0.5.0" };
149
159
  /**
150
160
  * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.
151
161
  *
@@ -170,4 +180,4 @@ CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images'
170
180
  * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.
171
181
  */
172
182
  CodeBuildRunner.LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');
173
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/providers/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_codebuild as codebuild,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport { Architecture, IImageBuilder, IRunnerProvider, Os, RunnerProviderProps, RunnerRuntimeParameters } from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n\nexport interface CodeBuildRunnerProps extends RunnerProviderProps {\n  /**\n   * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist with access to Docker-in-Docker.\n   *\n   * @default image builder with `CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile\n   */\n  readonly imageBuilder?: IImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default 'codebuild'\n   */\n  readonly label?: string;\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group to assign to this instance.\n   *\n   * @default public project with no security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * GitHub Actions runner provider using CodeBuild to execute the actions.\n *\n * Creates a project that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class CodeBuildRunner extends Construct implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stsable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stsable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');\n\n  /**\n   * CodeBuild project hosting the runner.\n   */\n  readonly project: codebuild.Project;\n\n  /**\n   * Label associated with this provider.\n   */\n  readonly label: string;\n\n  /**\n   * VPC used for hosting the project.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  constructor(scope: Construct, id: string, props: CodeBuildRunnerProps) {\n    super(scope, id);\n\n    this.label = props.label ?? 'codebuild';\n    this.vpc = props.vpc;\n    this.securityGroup = props.securityGroup;\n\n    const buildSpec = {\n      version: '0.2',\n      env: {\n        variables: {\n          RUNNER_TOKEN: 'unspecified',\n          RUNNER_NAME: 'unspecified',\n          RUNNER_LABEL: 'unspecified',\n          OWNER: 'unspecified',\n          REPO: 'unspecified',\n          GITHUB_DOMAIN: 'github.com',\n        },\n      },\n      phases: {\n        install: {\n          commands: [\n            'nohup /usr/local/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2 &',\n            'timeout 15 sh -c \"until docker info; do echo .; sleep 1; done\"',\n            'sudo -Hu runner /home/runner/config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" --disableupdate --name \"${RUNNER_NAME}\"',\n          ],\n        },\n        build: {\n          commands: [\n            'sudo -Hu runner /home/runner/run.sh',\n          ],\n        },\n      },\n    };\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = imageBuilder.bind();\n\n    // choose build image\n    let buildImage: codebuild.IBuildImage | undefined;\n    if (image.os.is(Os.LINUX)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.LinuxBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      } else if (image.architecture.is(Architecture.ARM64)) {\n        buildImage = codebuild.LinuxArmBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      }\n    }\n    if (image.os.is(Os.WINDOWS)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      }\n    }\n\n    if (buildImage === undefined) {\n      throw new Error(`Unable to find supported CodeBuild image for ${image.os.name}/${image.architecture.name}`);\n    }\n\n    // create project\n    this.project = new codebuild.Project(\n      this,\n      'CodeBuild',\n      {\n        description: `GitHub Actions self-hosted runner for label \"${this.label}\"`,\n        buildSpec: codebuild.BuildSpec.fromObject(buildSpec),\n        vpc: this.vpc,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        subnetSelection: props.subnetSelection,\n        timeout: props.timeout ?? Duration.hours(1),\n        environment: {\n          buildImage,\n          computeType: props.computeType ?? ComputeType.SMALL,\n          privileged: true,\n        },\n        logging: {\n          cloudWatch: {\n            logGroup: new logs.LogGroup(\n              this,\n              'Logs',\n              {\n                retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n                removalPolicy: RemovalPolicy.DESTROY,\n              },\n            ),\n          },\n        },\n      },\n    );\n\n    this.grantPrincipal = this.project.grantPrincipal;\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    return new stepfunctions_tasks.CodeBuildStartBuild(\n      this,\n      this.label,\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        project: this.project,\n        environmentVariablesOverride: {\n          RUNNER_TOKEN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerTokenPath,\n          },\n          RUNNER_NAME: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerNamePath,\n          },\n          RUNNER_LABEL: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: this.label,\n          },\n          GITHUB_DOMAIN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.githubDomainPath,\n          },\n          OWNER: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.ownerPath,\n          },\n          REPO: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.repoPath,\n          },\n        },\n      },\n    );\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return this.project.connections;\n  }\n}"]}
183
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/providers/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_codebuild as codebuild,\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { ComputeType } from 'aws-cdk-lib/aws-codebuild';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport { Architecture, IImageBuilder, IRunnerProvider, Os, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n\nexport interface CodeBuildRunnerProps extends RunnerProviderProps {\n  /**\n   * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist with access to Docker-in-Docker.\n   *\n   * @default image builder with `CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile\n   */\n  readonly imageBuilder?: IImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default 'codebuild'\n   */\n  readonly label?: string;\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default no VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security Group to assign to this instance.\n   *\n   * @default public project with no security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Where to place the network interfaces within the VPC.\n   *\n   * @default no subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * The type of compute to use for this build.\n   * See the {@link ComputeType} enum for the possible values.\n   *\n   * @default {@link ComputeType#SMALL}\n   */\n  readonly computeType?: codebuild.ComputeType;\n\n  /**\n   * The number of minutes after which AWS CodeBuild stops the build if it's\n   * not complete. For valid values, see the timeoutInMinutes field in the AWS\n   * CodeBuild User Guide.\n   *\n   * @default Duration.hours(1)\n   */\n  readonly timeout?: Duration;\n}\n\n/**\n * GitHub Actions runner provider using CodeBuild to execute the actions.\n *\n * Creates a project that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class CodeBuildRunner extends Construct implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stsable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirements for CodeBuild runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   * * `DOCKER_CHANNEL` overrides the channel from which Docker will be downloaded. Defaults to `\"stsable\"`.\n   * * `DIND_COMMIT` overrides the commit where dind is found.\n   * * `DOCKER_VERSION` overrides the installed Docker version.\n   * * `DOCKER_COMPOSE_VERSION` overrides the installed docker-compose version.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'codebuild', 'linux-arm64');\n\n  /**\n   * CodeBuild project hosting the runner.\n   */\n  readonly project: codebuild.Project;\n\n  /**\n   * Label associated with this provider.\n   */\n  readonly label: string;\n\n  /**\n   * VPC used for hosting the project.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Docker image in CodeBuild project.\n   */\n  readonly image: RunnerImage;\n\n  constructor(scope: Construct, id: string, props: CodeBuildRunnerProps) {\n    super(scope, id);\n\n    this.label = props.label ?? 'codebuild';\n    this.vpc = props.vpc;\n    this.securityGroup = props.securityGroup;\n\n    let buildSpec = {\n      version: '0.2',\n      env: {\n        variables: {\n          RUNNER_TOKEN: 'unspecified',\n          RUNNER_NAME: 'unspecified',\n          RUNNER_LABEL: 'unspecified',\n          OWNER: 'unspecified',\n          REPO: 'unspecified',\n          GITHUB_DOMAIN: 'github.com',\n        },\n      },\n      phases: {\n        install: {\n          commands: [\n            'nohup /usr/local/bin/dockerd --host=unix:///var/run/docker.sock --host=tcp://127.0.0.1:2375 --storage-driver=overlay2 &',\n            'timeout 15 sh -c \"until docker info; do echo .; sleep 1; done\"',\n            'sudo -Hu runner /home/runner/config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" --disableupdate --name \"${RUNNER_NAME}\"',\n          ],\n        },\n        build: {\n          commands: [\n            'sudo -Hu runner /home/runner/run.sh',\n          ],\n        },\n      },\n    };\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: CodeBuildRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    if (image.os.is(Os.WINDOWS)) {\n      buildSpec.phases.install.commands = [\n        'cd \\\\actions',\n        './config.cmd --unattended --url \"https://${Env:GITHUB_DOMAIN}/${Env:OWNER}/${Env:REPO}\" --token \"${Env:RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${Env:RUNNER_LABEL}\" --disableupdate --name \"${Env:RUNNER_NAME}\"',\n      ];\n      buildSpec.phases.build.commands = [\n        'cd \\\\actions',\n        './run.cmd',\n      ];\n    }\n\n    // choose build image\n    let buildImage: codebuild.IBuildImage | undefined;\n    if (image.os.is(Os.LINUX)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.LinuxBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      } else if (image.architecture.is(Architecture.ARM64)) {\n        buildImage = codebuild.LinuxArmBuildImage.fromEcrRepository(image.imageRepository, image.imageTag);\n      }\n    }\n    if (image.os.is(Os.WINDOWS)) {\n      if (image.architecture.is(Architecture.X86_64)) {\n        buildImage = codebuild.WindowsBuildImage.fromEcrRepository(image.imageRepository, image.imageTag, codebuild.WindowsImageType.SERVER_2019);\n      }\n    }\n\n    if (buildImage === undefined) {\n      throw new Error(`Unable to find supported CodeBuild image for ${image.os.name}/${image.architecture.name}`);\n    }\n\n    // create project\n    this.project = new codebuild.Project(\n      this,\n      'CodeBuild',\n      {\n        description: `GitHub Actions self-hosted runner for label \"${this.label}\"`,\n        buildSpec: codebuild.BuildSpec.fromObject(buildSpec),\n        vpc: this.vpc,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        subnetSelection: props.subnetSelection,\n        timeout: props.timeout ?? Duration.hours(1),\n        environment: {\n          buildImage,\n          computeType: props.computeType ?? ComputeType.SMALL,\n          privileged: image.os.is(Os.LINUX),\n        },\n        logging: {\n          cloudWatch: {\n            logGroup: new logs.LogGroup(\n              this,\n              'Logs',\n              {\n                retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n                removalPolicy: RemovalPolicy.DESTROY,\n              },\n            ),\n          },\n        },\n      },\n    );\n\n    this.grantPrincipal = this.project.grantPrincipal;\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    return new stepfunctions_tasks.CodeBuildStartBuild(\n      this,\n      this.label,\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        project: this.project,\n        environmentVariablesOverride: {\n          RUNNER_TOKEN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerTokenPath,\n          },\n          RUNNER_NAME: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.runnerNamePath,\n          },\n          RUNNER_LABEL: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: this.label,\n          },\n          GITHUB_DOMAIN: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.githubDomainPath,\n          },\n          OWNER: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.ownerPath,\n          },\n          REPO: {\n            type: codebuild.BuildEnvironmentVariableType.PLAINTEXT,\n            value: parameters.repoPath,\n          },\n        },\n      },\n    );\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return this.project.connections;\n  }\n}\n"]}
@@ -71,12 +71,6 @@ export interface RunnerImage {
71
71
  * Static image tag where the image will be pushed.
72
72
  */
73
73
  readonly imageTag: string;
74
- /**
75
- * Image digest for providers that need to know the digest like Lambda.
76
- *
77
- * WARNING: the digest might change when the builder automatically rebuilds the image on a schedule. Do not expect for this digest to stay the same between deploys.
78
- */
79
- readonly imageDigest: string;
80
74
  /**
81
75
  * Architecture of the image.
82
76
  */
@@ -85,6 +79,10 @@ export interface RunnerImage {
85
79
  * OS type of the image.
86
80
  */
87
81
  readonly os: Os;
82
+ /**
83
+ * Log group where image builds are logged.
84
+ */
85
+ readonly logGroup?: logs.LogGroup;
88
86
  }
89
87
  /**
90
88
  * Interface for constructs that build an image that can be used in {@link IRunnerProvider}.
@@ -149,6 +147,23 @@ export interface RunnerRuntimeParameters {
149
147
  */
150
148
  readonly repoPath: string;
151
149
  }
150
+ /**
151
+ * Interface for runner image status used by status.json.
152
+ */
153
+ export interface IRunnerImageStatus {
154
+ /**
155
+ * Image repository where runner image is pushed.
156
+ */
157
+ readonly imageRepository?: string;
158
+ /**
159
+ * Tag of image that should be used.
160
+ */
161
+ readonly imageTag?: string;
162
+ /**
163
+ * Log group name for the image builder where history of image builds can be analyzed.
164
+ */
165
+ readonly imageBuilderLogGroup?: string;
166
+ }
152
167
  /**
153
168
  * Interface for all runner providers. Implementations create all required resources and return a step function task that starts those resources from {@link getStepFunctionTask}.
154
169
  */
@@ -165,6 +180,10 @@ export interface IRunnerProvider extends ec2.IConnectable, iam.IGrantable {
165
180
  * Security group associated with runners.
166
181
  */
167
182
  readonly securityGroup?: ec2.ISecurityGroup;
183
+ /**
184
+ * Image used to create a new resource compute. Can be Docker image, AMI, or something else.
185
+ */
186
+ readonly image: RunnerImage;
168
187
  /**
169
188
  * Generate step function tasks that execute the runner.
170
189
  *