@cloudsnorkel/cdk-github-runners 0.14.15 → 0.14.17

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 (59) hide show
  1. package/.jsii +861 -335
  2. package/API.md +480 -6
  3. package/README.md +149 -0
  4. package/SETUP_GITHUB.md +99 -3
  5. package/assets/delete-failed-runner.lambda/index.js +40 -31
  6. package/assets/idle-runner-repear.lambda/index.js +40 -31
  7. package/assets/setup.lambda/index.html +12 -7
  8. package/assets/setup.lambda/index.js +26 -26
  9. package/assets/status.lambda/index.js +40 -31
  10. package/assets/token-retriever.lambda/index.js +40 -31
  11. package/assets/webhook-handler.lambda/index.js +110 -43
  12. package/assets/webhook-redelivery.lambda/index.js +40 -31
  13. package/lib/access.js +1 -1
  14. package/lib/image-builders/api.js +1 -1
  15. package/lib/image-builders/aws-image-builder/ami.d.ts +1 -2
  16. package/lib/image-builders/aws-image-builder/ami.js +6 -13
  17. package/lib/image-builders/aws-image-builder/builder.d.ts +4 -2
  18. package/lib/image-builders/aws-image-builder/builder.js +36 -34
  19. package/lib/image-builders/aws-image-builder/container.d.ts +2 -2
  20. package/lib/image-builders/aws-image-builder/container.js +7 -12
  21. package/lib/image-builders/aws-image-builder/deprecated/ami.js +1 -1
  22. package/lib/image-builders/aws-image-builder/deprecated/container.js +1 -1
  23. package/lib/image-builders/aws-image-builder/deprecated/linux-components.js +1 -1
  24. package/lib/image-builders/aws-image-builder/deprecated/windows-components.js +1 -1
  25. package/lib/image-builders/aws-image-builder/index.d.ts +0 -1
  26. package/lib/image-builders/aws-image-builder/index.js +1 -2
  27. package/lib/image-builders/aws-image-builder/workflow.d.ts +4 -4
  28. package/lib/image-builders/aws-image-builder/workflow.js +7 -10
  29. package/lib/image-builders/codebuild-deprecated.js +1 -1
  30. package/lib/image-builders/components.js +1 -1
  31. package/lib/image-builders/static.js +1 -1
  32. package/lib/index.d.ts +1 -0
  33. package/lib/index.js +2 -1
  34. package/lib/providers/codebuild.js +16 -10
  35. package/lib/providers/common.d.ts +61 -0
  36. package/lib/providers/common.js +33 -4
  37. package/lib/providers/composite.d.ts +61 -0
  38. package/lib/providers/composite.js +229 -0
  39. package/lib/providers/ec2.js +9 -8
  40. package/lib/providers/ecs.js +11 -6
  41. package/lib/providers/fargate.js +14 -9
  42. package/lib/providers/index.d.ts +1 -0
  43. package/lib/providers/index.js +2 -1
  44. package/lib/providers/lambda.js +6 -5
  45. package/lib/runner.d.ts +29 -5
  46. package/lib/runner.js +57 -24
  47. package/lib/secrets.js +1 -1
  48. package/lib/webhook-handler.lambda.d.ts +11 -0
  49. package/lib/webhook-handler.lambda.js +81 -14
  50. package/lib/webhook.d.ts +52 -7
  51. package/lib/webhook.js +4 -2
  52. package/package.json +15 -19
  53. package/assets/image-builders/aws-image-builder/versioner.lambda/index.js +0 -2115
  54. package/lib/image-builders/aws-image-builder/common.d.ts +0 -10
  55. package/lib/image-builders/aws-image-builder/common.js +0 -48
  56. package/lib/image-builders/aws-image-builder/versioner-function.d.ts +0 -13
  57. package/lib/image-builders/aws-image-builder/versioner-function.js +0 -23
  58. package/lib/image-builders/aws-image-builder/versioner.lambda.d.ts +0 -7
  59. package/lib/image-builders/aws-image-builder/versioner.lambda.js +0 -115
@@ -0,0 +1,229 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.CompositeProvider = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const constructs_1 = require("constructs");
8
+ const common_1 = require("./common");
9
+ /**
10
+ * A composite runner provider that implements fallback and distribution strategies.
11
+ */
12
+ class CompositeProvider {
13
+ /**
14
+ * Creates a fallback runner provider that tries each provider in order until one succeeds.
15
+ *
16
+ * For example, given providers A, B, C:
17
+ * - Try A first
18
+ * - If A fails, try B
19
+ * - If B fails, try C
20
+ *
21
+ * You can use this to try spot instance first, and switch to on-demand instances if spot is unavailable.
22
+ *
23
+ * Or you can use this to try different instance types in order of preference.
24
+ *
25
+ * @param scope The scope in which to define this construct
26
+ * @param id The scoped construct ID
27
+ * @param providers List of runner providers to try in order
28
+ */
29
+ static fallback(scope, id, providers) {
30
+ if (providers.length < 2) {
31
+ throw new Error('At least two providers must be specified for fallback');
32
+ }
33
+ this.validateLabels(providers);
34
+ return new FallbackRunnerProvider(scope, id, providers);
35
+ }
36
+ /**
37
+ * Creates a weighted distribution runner provider that randomly selects a provider based on weights.
38
+ *
39
+ * For example, given providers A (weight 10), B (weight 20), C (weight 30):
40
+ * - Total weight = 60
41
+ * - Probability of selecting A = 10/60 = 16.67%
42
+ * - Probability of selecting B = 20/60 = 33.33%
43
+ * - Probability of selecting C = 30/60 = 50%
44
+ *
45
+ * You can use this to distribute load across multiple instance types or availability zones.
46
+ *
47
+ * @param scope The scope in which to define this construct
48
+ * @param id The scoped construct ID
49
+ * @param weightedProviders List of weighted runner providers
50
+ */
51
+ static distribute(scope, id, weightedProviders) {
52
+ if (weightedProviders.length < 2) {
53
+ throw new Error('At least two providers must be specified for distribution');
54
+ }
55
+ // Validate labels
56
+ this.validateLabels(weightedProviders.map(wp => wp.provider));
57
+ // Validate weights
58
+ for (const wp of weightedProviders) {
59
+ if (wp.weight <= 0) {
60
+ throw new Error('All weights must be positive numbers');
61
+ }
62
+ }
63
+ return new DistributedRunnerProvider(scope, id, weightedProviders);
64
+ }
65
+ /**
66
+ * Validates that all providers have the exact same labels.
67
+ * This is required so that any provisioned runner can match the labels requested by the GitHub workflow job.
68
+ *
69
+ * @param providers Providers to validate
70
+ */
71
+ static validateLabels(providers) {
72
+ const firstLabels = new Set(providers[0].labels);
73
+ for (const provider of providers.slice(1)) {
74
+ const providerLabels = new Set(provider.labels);
75
+ if (firstLabels.size !== providerLabels.size || ![...firstLabels].every(label => providerLabels.has(label))) {
76
+ throw new Error(`All providers must have the exact same labels (${[...firstLabels].join(', ')} != ${[...providerLabels].join(', ')})`);
77
+ }
78
+ }
79
+ }
80
+ }
81
+ exports.CompositeProvider = CompositeProvider;
82
+ _a = JSII_RTTI_SYMBOL_1;
83
+ CompositeProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.CompositeProvider", version: "0.14.17" };
84
+ /**
85
+ * Internal implementation of fallback runner provider.
86
+ *
87
+ * @internal
88
+ */
89
+ class FallbackRunnerProvider extends constructs_1.Construct {
90
+ constructor(scope, id, providers) {
91
+ super(scope, id);
92
+ this.labels = providers[0].labels;
93
+ this.providers = providers;
94
+ }
95
+ /**
96
+ * Builds a Step Functions state machine that implements a fallback strategy.
97
+ *
98
+ * This method constructs a chain where each provider catches errors and falls back
99
+ * to the next provider in sequence. We iterate forward through providers, attaching
100
+ * catch handlers to each one (except the last) that route to the next provider.
101
+ *
102
+ * Example with providers [A, B, C]:
103
+ * - Save firstProvider = A (this will be returned)
104
+ * - Iteration 1 (i=0, provider A): A catches errors → falls back to B
105
+ * - Iteration 2 (i=1, provider B): B catches errors → falls back to C
106
+ * - Result: A → (on error) → B → (on error) → C
107
+ *
108
+ * Some providers generate one state while others (like EC2) may generate more complex chains.
109
+ * We try to avoid creating a complicated state machine, but complex chains may require wrapping in Parallel.
110
+ *
111
+ * @param parameters Runtime parameters for the step function task
112
+ * @returns A Step Functions chainable that implements the fallback logic
113
+ */
114
+ getStepFunctionTask(parameters) {
115
+ // Get all provider chainables upfront
116
+ const providerChainables = this.providers.map(p => p.getStepFunctionTask(parameters));
117
+ // Track the entry point - starts as first provider, but may be wrapped
118
+ let entryPoint = providerChainables[0];
119
+ // Attach catch handlers to each provider (except the last) to fall back to the next provider
120
+ for (let i = 0; i < this.providers.length - 1; i++) {
121
+ const currentProvider = providerChainables[i];
122
+ const nextProvider = providerChainables[i + 1];
123
+ // Try to attach catch handler directly to the provider's end state
124
+ // This is more efficient than wrapping in a Parallel state when possible
125
+ if (this.canAddCatchDirectly(currentProvider)) {
126
+ const endState = currentProvider.endStates[0];
127
+ endState.addCatch(nextProvider, {
128
+ errors: ['States.ALL'],
129
+ resultPath: `$.fallbackError${i + 1}`,
130
+ });
131
+ continue;
132
+ }
133
+ // Fallback: wrap in Parallel state to add catch capability
134
+ // This is needed when:
135
+ // - The provider is not a State instance
136
+ // - The provider has multiple end states
137
+ // - The end state doesn't support addCatch directly
138
+ const parallel = new aws_cdk_lib_1.aws_stepfunctions.Parallel(this, (0, common_1.generateStateName)(this, `attempt #${i + 1}`));
139
+ parallel.branch(currentProvider);
140
+ parallel.addCatch(nextProvider, {
141
+ errors: ['States.ALL'],
142
+ resultPath: `$.fallbackError${i + 1}`,
143
+ });
144
+ // If this is the first provider, update the entry point to the wrapped version
145
+ if (i === 0) {
146
+ entryPoint = parallel;
147
+ }
148
+ }
149
+ return entryPoint;
150
+ }
151
+ /**
152
+ * Checks if we can add a catch handler directly to the provider's end state.
153
+ * This avoids wrapping in a Parallel state when possible.
154
+ */
155
+ canAddCatchDirectly(provider) {
156
+ if (!(provider instanceof aws_cdk_lib_1.aws_stepfunctions.State)) {
157
+ return false;
158
+ }
159
+ const endStates = provider.endStates;
160
+ if (endStates.length !== 1 || !(endStates[0] instanceof aws_cdk_lib_1.aws_stepfunctions.State)) {
161
+ return false;
162
+ }
163
+ // Use 'any' type assertion because not all State types have addCatch in their type definition,
164
+ // but Task states and other executable states do support it at runtime
165
+ const endState = endStates[0];
166
+ return typeof endState.addCatch === 'function';
167
+ }
168
+ grantStateMachine(stateMachineRole) {
169
+ for (const provider of this.providers) {
170
+ provider.grantStateMachine(stateMachineRole);
171
+ }
172
+ }
173
+ status(statusFunctionRole) {
174
+ // Return statuses from all sub-providers
175
+ return this.providers.map(provider => provider.status(statusFunctionRole));
176
+ }
177
+ }
178
+ /**
179
+ * Internal implementation of distributed runner provider.
180
+ *
181
+ * @internal
182
+ */
183
+ class DistributedRunnerProvider extends constructs_1.Construct {
184
+ constructor(scope, id, weightedProviders) {
185
+ super(scope, id);
186
+ this.weightedProviders = weightedProviders;
187
+ this.labels = weightedProviders[0].provider.labels;
188
+ this.providers = weightedProviders.map(wp => wp.provider);
189
+ }
190
+ /**
191
+ * Weighted random selection algorithm:
192
+ * 1. Generate a random number in [1, totalWeight+1)
193
+ * 2. Build cumulative weight ranges for each provider (e.g., weights [10,20,30] -> ranges [1-10, 11-30, 31-60])
194
+ * 3. Use Step Functions Choice state to route to the provider whose range contains the random number
195
+ * The first matching condition wins, so we check if rand <= cumulativeWeight for each provider in order
196
+ *
197
+ * Note: States.MathRandom returns a value in [start, end) where end is exclusive. We use [1, totalWeight+1)
198
+ * to ensure the random value can be up to totalWeight (inclusive), which allows the last provider to be selected
199
+ * when rand equals totalWeight.
200
+ */
201
+ getStepFunctionTask(parameters) {
202
+ const totalWeight = this.weightedProviders.reduce((sum, wp) => sum + wp.weight, 0);
203
+ const rand = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, (0, common_1.generateStateName)(this, 'rand'), {
204
+ parameters: {
205
+ rand: aws_cdk_lib_1.aws_stepfunctions.JsonPath.mathRandom(1, totalWeight + 1),
206
+ },
207
+ resultPath: '$.composite',
208
+ });
209
+ const choice = new aws_cdk_lib_1.aws_stepfunctions.Choice(this, (0, common_1.generateStateName)(this, 'choice'));
210
+ rand.next(choice);
211
+ // Find provider with the highest weight
212
+ let rollingWeight = 0;
213
+ for (const wp of this.weightedProviders) {
214
+ rollingWeight += wp.weight;
215
+ choice.when(aws_cdk_lib_1.aws_stepfunctions.Condition.numberLessThanEquals('$.composite.rand', rollingWeight), wp.provider.getStepFunctionTask(parameters));
216
+ }
217
+ return rand;
218
+ }
219
+ grantStateMachine(stateMachineRole) {
220
+ for (const wp of this.weightedProviders) {
221
+ wp.provider.grantStateMachine(stateMachineRole);
222
+ }
223
+ }
224
+ status(statusFunctionRole) {
225
+ // Return statuses from all sub-providers
226
+ return this.providers.map(provider => provider.status(statusFunctionRole));
227
+ }
228
+ }
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"composite.js","sourceRoot":"","sources":["../../src/providers/composite.ts"],"names":[],"mappings":"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","sourcesContent":["import { aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { ICompositeProvider, IRunnerProvider, IRunnerProviderStatus, RunnerRuntimeParameters, generateStateName } from './common';\n\n/**\n * Configuration for weighted distribution of runners.\n */\nexport interface WeightedRunnerProvider {\n  /**\n   * The runner provider to use.\n   */\n  readonly provider: IRunnerProvider;\n\n  /**\n   * Weight for this provider. Higher weights mean higher probability of selection.\n   * Must be a positive number.\n   */\n  readonly weight: number;\n}\n\n/**\n * A composite runner provider that implements fallback and distribution strategies.\n */\nexport class CompositeProvider {\n  /**\n   * Creates a fallback runner provider that tries each provider in order until one succeeds.\n   *\n   * For example, given providers A, B, C:\n   * - Try A first\n   * - If A fails, try B\n   * - If B fails, try C\n   *\n   * You can use this to try spot instance first, and switch to on-demand instances if spot is unavailable.\n   *\n   * Or you can use this to try different instance types in order of preference.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param providers List of runner providers to try in order\n   */\n  public static fallback(scope: Construct, id: string, providers: IRunnerProvider[]): ICompositeProvider {\n    if (providers.length < 2) {\n      throw new Error('At least two providers must be specified for fallback');\n    }\n\n    this.validateLabels(providers);\n\n    return new FallbackRunnerProvider(scope, id, providers);\n  }\n\n  /**\n   * Creates a weighted distribution runner provider that randomly selects a provider based on weights.\n   *\n   * For example, given providers A (weight 10), B (weight 20), C (weight 30):\n   * - Total weight = 60\n   * - Probability of selecting A = 10/60 = 16.67%\n   * - Probability of selecting B = 20/60 = 33.33%\n   * - Probability of selecting C = 30/60 = 50%\n   *\n   * You can use this to distribute load across multiple instance types or availability zones.\n   *\n   * @param scope The scope in which to define this construct\n   * @param id The scoped construct ID\n   * @param weightedProviders List of weighted runner providers\n   */\n  public static distribute(scope: Construct, id: string, weightedProviders: WeightedRunnerProvider[]): ICompositeProvider {\n    if (weightedProviders.length < 2) {\n      throw new Error('At least two providers must be specified for distribution');\n    }\n\n    // Validate labels\n    this.validateLabels(weightedProviders.map(wp => wp.provider));\n\n    // Validate weights\n    for (const wp of weightedProviders) {\n      if (wp.weight <= 0) {\n        throw new Error('All weights must be positive numbers');\n      }\n    }\n\n    return new DistributedRunnerProvider(scope, id, weightedProviders);\n  }\n\n  /**\n   * Validates that all providers have the exact same labels.\n   * This is required so that any provisioned runner can match the labels requested by the GitHub workflow job.\n   *\n   * @param providers Providers to validate\n   */\n  private static validateLabels(providers: IRunnerProvider[]): void {\n    const firstLabels = new Set(providers[0].labels);\n    for (const provider of providers.slice(1)) {\n      const providerLabels = new Set(provider.labels);\n      if (firstLabels.size !== providerLabels.size || ![...firstLabels].every(label => providerLabels.has(label))) {\n        throw new Error(`All providers must have the exact same labels (${[...firstLabels].join(', ')} != ${[...providerLabels].join(', ')})`);\n      }\n    }\n  }\n}\n\n/**\n * Internal implementation of fallback runner provider.\n *\n * @internal\n */\nclass FallbackRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, providers: IRunnerProvider[]) {\n    super(scope, id);\n    this.labels = providers[0].labels;\n    this.providers = providers;\n  }\n\n  /**\n   * Builds a Step Functions state machine that implements a fallback strategy.\n   *\n   * This method constructs a chain where each provider catches errors and falls back\n   * to the next provider in sequence. We iterate forward through providers, attaching\n   * catch handlers to each one (except the last) that route to the next provider.\n   *\n   * Example with providers [A, B, C]:\n   * - Save firstProvider = A (this will be returned)\n   * - Iteration 1 (i=0, provider A): A catches errors → falls back to B\n   * - Iteration 2 (i=1, provider B): B catches errors → falls back to C\n   * - Result: A → (on error) → B → (on error) → C\n   *\n   * Some providers generate one state while others (like EC2) may generate more complex chains.\n   * We try to avoid creating a complicated state machine, but complex chains may require wrapping in Parallel.\n   *\n   * @param parameters Runtime parameters for the step function task\n   * @returns A Step Functions chainable that implements the fallback logic\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // Get all provider chainables upfront\n    const providerChainables = this.providers.map(p => p.getStepFunctionTask(parameters));\n\n    // Track the entry point - starts as first provider, but may be wrapped\n    let entryPoint = providerChainables[0];\n\n    // Attach catch handlers to each provider (except the last) to fall back to the next provider\n    for (let i = 0; i < this.providers.length - 1; i++) {\n      const currentProvider = providerChainables[i];\n      const nextProvider = providerChainables[i + 1];\n\n      // Try to attach catch handler directly to the provider's end state\n      // This is more efficient than wrapping in a Parallel state when possible\n      if (this.canAddCatchDirectly(currentProvider)) {\n        const endState = (currentProvider as stepfunctions.State).endStates[0] as any;\n        endState.addCatch(nextProvider, {\n          errors: ['States.ALL'],\n          resultPath: `$.fallbackError${i + 1}`,\n        });\n        continue;\n      }\n\n      // Fallback: wrap in Parallel state to add catch capability\n      // This is needed when:\n      // - The provider is not a State instance\n      // - The provider has multiple end states\n      // - The end state doesn't support addCatch directly\n      const parallel = new stepfunctions.Parallel(this, generateStateName(this, `attempt #${i + 1}`));\n      parallel.branch(currentProvider);\n      parallel.addCatch(nextProvider, {\n        errors: ['States.ALL'],\n        resultPath: `$.fallbackError${i + 1}`,\n      });\n\n      // If this is the first provider, update the entry point to the wrapped version\n      if (i === 0) {\n        entryPoint = parallel;\n      }\n    }\n\n    return entryPoint;\n  }\n\n  /**\n   * Checks if we can add a catch handler directly to the provider's end state.\n   * This avoids wrapping in a Parallel state when possible.\n   */\n  private canAddCatchDirectly(provider: stepfunctions.IChainable): boolean {\n    if (!(provider instanceof stepfunctions.State)) {\n      return false;\n    }\n    const endStates = provider.endStates;\n    if (endStates.length !== 1 || !(endStates[0] instanceof stepfunctions.State)) {\n      return false;\n    }\n    // Use 'any' type assertion because not all State types have addCatch in their type definition,\n    // but Task states and other executable states do support it at runtime\n    const endState = endStates[0] as any;\n    return typeof endState.addCatch === 'function';\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const provider of this.providers) {\n      provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n\n/**\n * Internal implementation of distributed runner provider.\n *\n * @internal\n */\nclass DistributedRunnerProvider extends Construct implements ICompositeProvider {\n  public readonly labels: string[];\n  public readonly providers: IRunnerProvider[];\n\n  constructor(scope: Construct, id: string, private readonly weightedProviders: WeightedRunnerProvider[]) {\n    super(scope, id);\n    this.labels = weightedProviders[0].provider.labels;\n    this.providers = weightedProviders.map(wp => wp.provider);\n  }\n\n  /**\n   * Weighted random selection algorithm:\n   * 1. Generate a random number in [1, totalWeight+1)\n   * 2. Build cumulative weight ranges for each provider (e.g., weights [10,20,30] -> ranges [1-10, 11-30, 31-60])\n   * 3. Use Step Functions Choice state to route to the provider whose range contains the random number\n   *    The first matching condition wins, so we check if rand <= cumulativeWeight for each provider in order\n   *\n   * Note: States.MathRandom returns a value in [start, end) where end is exclusive. We use [1, totalWeight+1)\n   * to ensure the random value can be up to totalWeight (inclusive), which allows the last provider to be selected\n   * when rand equals totalWeight.\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    const totalWeight = this.weightedProviders.reduce((sum, wp) => sum + wp.weight, 0);\n    const rand = new stepfunctions.Pass(this, generateStateName(this, 'rand'), {\n      parameters: {\n        rand: stepfunctions.JsonPath.mathRandom(1, totalWeight + 1),\n      },\n      resultPath: '$.composite',\n    });\n    const choice = new stepfunctions.Choice(this, generateStateName(this, 'choice'));\n    rand.next(choice);\n\n    // Find provider with the highest weight\n    let rollingWeight = 0;\n    for (const wp of this.weightedProviders) {\n      rollingWeight += wp.weight;\n      choice.when(\n        stepfunctions.Condition.numberLessThanEquals('$.composite.rand', rollingWeight),\n        wp.provider.getStepFunctionTask(parameters),\n      );\n    }\n\n    return rand;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable): void {\n    for (const wp of this.weightedProviders) {\n      wp.provider.grantStateMachine(stateMachineRole);\n    }\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus[] {\n    // Return statuses from all sub-providers\n    return this.providers.map(provider => provider.status(statusFunctionRole));\n  }\n}\n"]}
@@ -270,14 +270,14 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
270
270
  parameters.ownerPath,
271
271
  parameters.repoPath,
272
272
  parameters.runnerTokenPath,
273
- this.labels.join(','),
273
+ parameters.labelsPath,
274
274
  parameters.registrationUrl,
275
275
  this.group ? '--runnergroup' : '',
276
276
  // this is split into 2 for powershell otherwise it will pass "--runnergroup name" as a single argument and config.sh will fail
277
277
  this.group ? this.group : '',
278
278
  this.defaultLabels ? '' : '--no-default-labels',
279
279
  ];
280
- const passUserData = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {
280
+ const passUserData = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, (0, common_1.generateStateName)(this, 'data'), {
281
281
  parameters: {
282
282
  userdataTemplate: this.ami.os.is(common_1.Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,
283
283
  },
@@ -295,9 +295,9 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
295
295
  });
296
296
  const rootDeviceResource = (0, common_1.amiRootDevice)(this, this.ami.launchTemplate.launchTemplateId);
297
297
  rootDeviceResource.node.addDependency(this.amiBuilder);
298
- const subnetRunners = this.subnets.map((subnet, index) => {
299
- return new aws_cdk_lib_1.aws_stepfunctions_tasks.CallAwsService(this, `${this.labels.join(', ')} subnet${index + 1}`, {
300
- comment: subnet.subnetId,
298
+ const subnetRunners = this.subnets.map(subnet => {
299
+ return new aws_cdk_lib_1.aws_stepfunctions_tasks.CallAwsService(this, (0, common_1.generateStateName)(this, subnet.subnetId), {
300
+ comment: subnet.availabilityZone,
301
301
  integrationPattern: aws_stepfunctions_1.IntegrationPattern.WAIT_FOR_TASK_TOKEN,
302
302
  service: 'ec2',
303
303
  action: 'runInstances',
@@ -402,6 +402,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
402
402
  return {
403
403
  type: this.constructor.name,
404
404
  labels: this.labels,
405
+ constructPath: this.node.path,
405
406
  securityGroups: this.securityGroups.map(sg => sg.securityGroupId),
406
407
  roleArn: this.role.roleArn,
407
408
  logGroup: this.logGroup.logGroupName,
@@ -420,7 +421,7 @@ class Ec2RunnerProvider extends common_1.BaseProvider {
420
421
  }
421
422
  exports.Ec2RunnerProvider = Ec2RunnerProvider;
422
423
  _a = JSII_RTTI_SYMBOL_1;
423
- Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.14.15" };
424
+ Ec2RunnerProvider[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2RunnerProvider", version: "0.14.17" };
424
425
  /**
425
426
  * @deprecated use {@link Ec2RunnerProvider}
426
427
  */
@@ -428,5 +429,5 @@ class Ec2Runner extends Ec2RunnerProvider {
428
429
  }
429
430
  exports.Ec2Runner = Ec2Runner;
430
431
  _b = JSII_RTTI_SYMBOL_1;
431
- Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.14.15" };
432
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\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  Stack,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  amiRootDevice,\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n  StorageOptions,\n} from './common';\nimport {\n  AwsImageBuilderRunnerImageBuilder,\n  IRunnerImageBuilder,\n  RunnerImageBuilder,\n  RunnerImageBuilderProps,\n  RunnerImageBuilderType,\n  RunnerImageComponent,\n} from '../image-builders';\nimport { MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT } from '../utils';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nlogGroupName=\"{}\"\nrunnerNamePath=\"{}\"\ngithubDomainPath=\"{}\"\nownerPath=\"{}\"\nrepoPath=\"{}\"\nrunnerTokenPath=\"{}\"\nlabels=\"{}\"\nregistrationURL=\"{}\"\nrunnerGroup1=\"{}\"\nrunnerGroup2=\"{}\"\ndefaultLabels=\"{}\"\n\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"$logGroupName\",\n              \"log_stream_name\": \"$runnerNamePath\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  # Determine the value of RUNNER_FLAGS\n  if [ \"$(< /home/runner/RUNNER_VERSION)\" = \"latest\" ]; then\n    RUNNER_FLAGS=\"\"\n  else\n    RUNNER_FLAGS=\"--disableupdate\"\n  fi\n\n  labelsTemplate=\"$labels,cdkghr:started:$(date +%s)\"\n\n  # Execute the configuration command for runner registration\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"$registrationURL\" --token \"$runnerTokenPath\" --ephemeral --work _work --labels \"$labelsTemplate\" $RUNNER_FLAGS --name \"$runnerNamePath\" $runnerGroup1 $runnerGroup2 $defaultLabels || exit 1\n\n  # Execute the run command\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n\n  # Retrieve the status\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\K.*\" /home/runner/_diag/ | tail -n1)\n\n  # Check and print the job status\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$labels\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\n$logGroupName=\"{}\"\n$runnerNamePath=\"{}\"\n$githubDomainPath=\"{}\"\n$ownerPath=\"{}\"\n$repoPath=\"{}\"\n$runnerTokenPath=\"{}\"\n$labels=\"{}\"\n$registrationURL=\"{}\"\n$runnerGroup1=\"{}\"\n$runnerGroup2=\"{}\"\n$defaultLabels=\"{}\"\n\n# EC2Launch only starts ssm agent after user data is done, so we need to start it ourselves (it is disabled by default)\nSet-Service -StartupType Manual AmazonSSMAgent\nStart-Service AmazonSSMAgent\n\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo \"{\n    \\`\"logs\\`\": {\n      \\`\"log_stream_name\\`\": \\`\"unknown\\`\",\n      \\`\"logs_collected\\`\": {\n        \\`\"files\\`\": {\n         \\`\"collect_list\\`\": [\n            {\n              \\`\"file_path\\`\": \\`\"/actions/runner.log\\`\",\n              \\`\"log_group_name\\`\": \\`\"$logGroupName\\`\",\n              \\`\"log_stream_name\\`\": \\`\"$runnerNamePath\\`\",\n              \\`\"timezone\\`\": \\`\"UTC\\`\"\n            }\n          ]\n        }\n      }\n    }\n  }\" | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  $RunnerVersion = Get-Content /actions/RUNNER_VERSION -Raw\n  if ($RunnerVersion -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" }\n  ./config.cmd --unattended --url \"\\${registrationUrl}\" --token \"\\${runnerTokenPath}\" --ephemeral --work _work --labels \"\\${labels},cdkghr:started:$(Get-Date -UFormat +%s)\" $RunnerFlags --name \"\\${runnerNamePath}\" \\${runnerGroup1} \\${runnerGroup2} \\${defaultLabels} 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\n\n  $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1\n\n  if ($STATUS) {\n      echo \"CDKGHA JOB DONE \\${labels} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log\n  }\n\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build AMI containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default Ec2RunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * @deprecated use imageBuilder\n   */\n  readonly amiBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * GitHub Actions runner group name.\n   *\n   * If specified, the runner will be registered with this group name. Setting a runner group can help managing access to self-hosted runners. It\n   * requires a paid GitHub account.\n   *\n   * The group must exist or the runner will not start.\n   *\n   * Users will still be able to trigger this runner with the correct labels. But the runner will only be able to run jobs from repos allowed to use the group.\n   *\n   * @default undefined\n   */\n  readonly group?: string;\n\n  /**\n   * Instance type for launched runner instances.\n   *\n   * @default m6i.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Options for runner instance storage volume.\n   */\n  readonly storageOptions?: StorageOptions;\n\n  /**\n   * Security Group to assign to launched runner instances.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\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 Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds EC2 specific runner images.\n   *\n   * You can customize the OS, architecture, VPC, subnet, security groups, etc. by passing in props.\n   *\n   * You can add components to the image builder by calling `imageBuilder.addComponent()`.\n   *\n   * The default OS is Ubuntu running on x64 architecture.\n   *\n   * Included components:\n   *  * `RunnerImageComponent.requiredPackages()`\n   *  * `RunnerImageComponent.cloudWatchAgent()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      builderType: RunnerImageBuilderType.AWS_IMAGE_BUILDER,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.cloudWatchAgent(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  readonly retryableErrors = [\n    'Ec2.Ec2Exception',\n    'States.Timeout',\n  ];\n\n  private readonly group?: string;\n  private readonly amiBuilder: IRunnerImageBuilder;\n  private readonly ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly storageOptions?: StorageOptions;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly defaultLabels: boolean;\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.group = props?.group;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M6I, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.storageOptions = props?.storageOptions;\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n    this.defaultLabels = props?.defaultLabels ?? true;\n\n    this.amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = this.amiBuilder.bindAmi();\n\n    if (this.amiBuilder instanceof AwsImageBuilderRunnerImageBuilder) {\n      if (this.amiBuilder.storageSize && this.storageSize.toBytes() < this.amiBuilder.storageSize.toBytes()) {\n        throw new Error(`Runner storage size (${this.storageSize.toGibibytes()} GiB) must be at least the same as the image builder storage size (${this.amiBuilder.storageSize.toGibibytes()} GiB)`);\n      }\n    }\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n      conditions: {\n        StringEquals: {\n          'aws:ResourceTag/aws:cloudformation:stack-id': cdk.Stack.of(this).stackId,\n        },\n      },\n    }));\n    this.grantPrincipal.addToPrincipalPolicy(MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT);\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\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    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      this.labels.join(','),\n      parameters.registrationUrl,\n      this.group ? '--runnergroup' : '',\n      // this is split into 2 for powershell otherwise it will pass \"--runnergroup name\" as a single argument and config.sh will fail\n      this.group ? this.group : '',\n      this.defaultLabels ? '' : '--no-default-labels',\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const rootDeviceResource = amiRootDevice(this, this.ami.launchTemplate.launchTemplateId);\n    rootDeviceResource.node.addDependency(this.amiBuilder);\n    const subnetRunners = this.subnets.map((subnet, index) => {\n      return new stepfunctions_tasks.CallAwsService(this, `${this.labels.join(', ')} subnet${index+1}`, {\n        comment: subnet.subnetId,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeatTimeout: stepfunctions.Timeout.duration(Duration.minutes(10)),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: rootDeviceResource.ref,\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n              VolumeType: this.storageOptions?.volumeType,\n              Iops: this.storageOptions?.iops,\n              Throughput: this.storageOptions?.throughput,\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n          TagSpecifications: [ // manually propagate tags\n            {\n              ResourceType: 'instance',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n            {\n              ResourceType: 'volume',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n          ],\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // start with the first subnet\n    passUserData.next(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i-1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    return passUserData;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n}\n\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}
432
+ Ec2Runner[_b] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.14.17" };
433
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\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  Stack,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  amiRootDevice,\n  Architecture,\n  BaseProvider,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n  generateStateName,\n  StorageOptions,\n} from './common';\nimport {\n  AwsImageBuilderRunnerImageBuilder,\n  IRunnerImageBuilder,\n  RunnerImageBuilder,\n  RunnerImageBuilderProps,\n  RunnerImageBuilderType,\n  RunnerImageComponent,\n} from '../image-builders';\nimport { MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT } from '../utils';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nlogGroupName=\"{}\"\nrunnerNamePath=\"{}\"\ngithubDomainPath=\"{}\"\nownerPath=\"{}\"\nrepoPath=\"{}\"\nrunnerTokenPath=\"{}\"\nlabels=\"{}\"\nregistrationURL=\"{}\"\nrunnerGroup1=\"{}\"\nrunnerGroup2=\"{}\"\ndefaultLabels=\"{}\"\n\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"$logGroupName\",\n              \"log_stream_name\": \"$runnerNamePath\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  # Determine the value of RUNNER_FLAGS\n  if [ \"$(< /home/runner/RUNNER_VERSION)\" = \"latest\" ]; then\n    RUNNER_FLAGS=\"\"\n  else\n    RUNNER_FLAGS=\"--disableupdate\"\n  fi\n\n  labelsTemplate=\"$labels,cdkghr:started:$(date +%s)\"\n\n  # Execute the configuration command for runner registration\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"$registrationURL\" --token \"$runnerTokenPath\" --ephemeral --work _work --labels \"$labelsTemplate\" $RUNNER_FLAGS --name \"$runnerNamePath\" $runnerGroup1 $runnerGroup2 $defaultLabels || exit 1\n\n  # Execute the run command\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n\n  # Retrieve the status\n  STATUS=$(grep -Phors \"finish job request for job [0-9a-f\\\\-]+ with result: \\K.*\" /home/runner/_diag/ | tail -n1)\n\n  # Check and print the job status\n  [ -n \"$STATUS\" ] && echo CDKGHA JOB DONE \"$labels\" \"$STATUS\"\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\nsleep 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\n$logGroupName=\"{}\"\n$runnerNamePath=\"{}\"\n$githubDomainPath=\"{}\"\n$ownerPath=\"{}\"\n$repoPath=\"{}\"\n$runnerTokenPath=\"{}\"\n$labels=\"{}\"\n$registrationURL=\"{}\"\n$runnerGroup1=\"{}\"\n$runnerGroup2=\"{}\"\n$defaultLabels=\"{}\"\n\n# EC2Launch only starts ssm agent after user data is done, so we need to start it ourselves (it is disabled by default)\nSet-Service -StartupType Manual AmazonSSMAgent\nStart-Service AmazonSSMAgent\n\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo \"{\n    \\`\"logs\\`\": {\n      \\`\"log_stream_name\\`\": \\`\"unknown\\`\",\n      \\`\"logs_collected\\`\": {\n        \\`\"files\\`\": {\n         \\`\"collect_list\\`\": [\n            {\n              \\`\"file_path\\`\": \\`\"/actions/runner.log\\`\",\n              \\`\"log_group_name\\`\": \\`\"$logGroupName\\`\",\n              \\`\"log_stream_name\\`\": \\`\"$runnerNamePath\\`\",\n              \\`\"timezone\\`\": \\`\"UTC\\`\"\n            }\n          ]\n        }\n      }\n    }\n  }\" | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  $RunnerVersion = Get-Content /actions/RUNNER_VERSION -Raw\n  if ($RunnerVersion -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" }\n  ./config.cmd --unattended --url \"\\${registrationUrl}\" --token \"\\${runnerTokenPath}\" --ephemeral --work _work --labels \"\\${labels},cdkghr:started:$(Get-Date -UFormat +%s)\" $RunnerFlags --name \"\\${runnerNamePath}\" \\${runnerGroup1} \\${runnerGroup2} \\${defaultLabels} 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\n\n  $STATUS = Select-String -Path './_diag/*.log' -Pattern 'finish job request for job [0-9a-f\\\\-]+ with result: (.*)' | %{$_.Matches.Groups[1].Value} | Select-Object -Last 1\n\n  if ($STATUS) {\n      echo \"CDKGHA JOB DONE \\${labels} $STATUS\" | Out-File -Encoding ASCII -Append /actions/runner.log\n  }\n\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStart-Sleep -Seconds 10  # give cloudwatch agent its default 5 seconds buffer duration to upload logs\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2RunnerProvider} construct.\n */\nexport interface Ec2RunnerProviderProps extends RunnerProviderProps {\n  /**\n   * Runner image builder used to build AMI containing GitHub Runner and all requirements.\n   *\n   * The image builder determines the OS and architecture of the runner.\n   *\n   * @default Ec2RunnerProvider.imageBuilder()\n   */\n  readonly imageBuilder?: IRunnerImageBuilder;\n\n  /**\n   * @deprecated use imageBuilder\n   */\n  readonly amiBuilder?: IRunnerImageBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * GitHub Actions runner group name.\n   *\n   * If specified, the runner will be registered with this group name. Setting a runner group can help managing access to self-hosted runners. It\n   * requires a paid GitHub account.\n   *\n   * The group must exist or the runner will not start.\n   *\n   * Users will still be able to trigger this runner with the correct labels. But the runner will only be able to run jobs from repos allowed to use the group.\n   *\n   * @default undefined\n   */\n  readonly group?: string;\n\n  /**\n   * Instance type for launched runner instances.\n   *\n   * @default m6i.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Options for runner instance storage volume.\n   */\n  readonly storageOptions?: StorageOptions;\n\n  /**\n   * Security Group to assign to launched runner instances.\n   *\n   * @default a new security group\n   *\n   * @deprecated use {@link securityGroups}\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Security groups to assign to launched runner instances.\n   *\n   * @default a new security group\n   */\n  readonly securityGroups?: ec2.ISecurityGroup[];\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   *\n   * @deprecated use {@link vpc} and {@link subnetSelection}\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * VPC where runner instances will be launched.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Where to place the network interfaces within the VPC. Only the first matched subnet will be used.\n   *\n   * @default default VPC subnet\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\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 Ec2RunnerProvider extends BaseProvider implements IRunnerProvider {\n  /**\n   * Create new image builder that builds EC2 specific runner images.\n   *\n   * You can customize the OS, architecture, VPC, subnet, security groups, etc. by passing in props.\n   *\n   * You can add components to the image builder by calling `imageBuilder.addComponent()`.\n   *\n   * The default OS is Ubuntu running on x64 architecture.\n   *\n   * Included components:\n   *  * `RunnerImageComponent.requiredPackages()`\n   *  * `RunnerImageComponent.cloudWatchAgent()`\n   *  * `RunnerImageComponent.runnerUser()`\n   *  * `RunnerImageComponent.git()`\n   *  * `RunnerImageComponent.githubCli()`\n   *  * `RunnerImageComponent.awsCli()`\n   *  * `RunnerImageComponent.docker()`\n   *  * `RunnerImageComponent.githubRunner()`\n   */\n  public static imageBuilder(scope: Construct, id: string, props?: RunnerImageBuilderProps) {\n    return RunnerImageBuilder.new(scope, id, {\n      os: Os.LINUX_UBUNTU,\n      architecture: Architecture.X86_64,\n      builderType: RunnerImageBuilderType.AWS_IMAGE_BUILDER,\n      components: [\n        RunnerImageComponent.requiredPackages(),\n        RunnerImageComponent.cloudWatchAgent(),\n        RunnerImageComponent.runnerUser(),\n        RunnerImageComponent.git(),\n        RunnerImageComponent.githubCli(),\n        RunnerImageComponent.awsCli(),\n        RunnerImageComponent.docker(),\n        RunnerImageComponent.githubRunner(props?.runnerVersion ?? RunnerVersion.latest()),\n      ],\n      ...props,\n    });\n  }\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * Log group where provided runners will save their logs.\n   *\n   * Note that this is not the job log, but the runner itself. It will not contain output from the GitHub Action but only metadata on its execution.\n   */\n  readonly logGroup: logs.ILogGroup;\n\n  readonly retryableErrors = [\n    'Ec2.Ec2Exception',\n    'States.Timeout',\n  ];\n\n  private readonly group?: string;\n  private readonly amiBuilder: IRunnerImageBuilder;\n  private readonly ami: RunnerAmi;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly storageOptions?: StorageOptions;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n  private readonly vpc: ec2.IVpc;\n  private readonly subnets: ec2.ISubnet[];\n  private readonly securityGroups: ec2.ISecurityGroup[];\n  private readonly defaultLabels: boolean;\n\n  constructor(scope: Construct, id: string, props?: Ec2RunnerProviderProps) {\n    super(scope, id, props);\n\n    this.labels = props?.labels ?? ['ec2'];\n    this.group = props?.group;\n    this.vpc = props?.vpc ?? ec2.Vpc.fromLookup(this, 'Default VPC', { isDefault: true });\n    this.securityGroups = props?.securityGroup ? [props.securityGroup] : (props?.securityGroups ?? [new ec2.SecurityGroup(this, 'SG', { vpc: this.vpc })]);\n    this.subnets = props?.subnet ? [props.subnet] : this.vpc.selectSubnets(props?.subnetSelection).subnets;\n    this.instanceType = props?.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M6I, ec2.InstanceSize.LARGE);\n    this.storageSize = props?.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.storageOptions = props?.storageOptions;\n    this.spot = props?.spot ?? false;\n    this.spotMaxPrice = props?.spotMaxPrice;\n    this.defaultLabels = props?.defaultLabels ?? true;\n\n    this.amiBuilder = props?.imageBuilder ?? props?.amiBuilder ?? Ec2RunnerProvider.imageBuilder(this, 'Ami Builder', {\n      vpc: props?.vpc,\n      subnetSelection: props?.subnetSelection,\n      securityGroups: this.securityGroups,\n    });\n    this.ami = this.amiBuilder.bindAmi();\n\n    if (this.amiBuilder instanceof AwsImageBuilderRunnerImageBuilder) {\n      if (this.amiBuilder.storageSize && this.storageSize.toBytes() < this.amiBuilder.storageSize.toBytes()) {\n        throw new Error(`Runner storage size (${this.storageSize.toGibibytes()} GiB) must be at least the same as the image builder storage size (${this.amiBuilder.storageSize.toGibibytes()} GiB)`);\n      }\n    }\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n      conditions: {\n        StringEquals: {\n          'aws:ResourceTag/aws:cloudformation:stack-id': cdk.Stack.of(this).stackId,\n        },\n      },\n    }));\n    this.grantPrincipal.addToPrincipalPolicy(MINIMAL_EC2_SSM_SESSION_MANAGER_POLICY_STATEMENT);\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props?.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\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    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      parameters.labelsPath,\n      parameters.registrationUrl,\n      this.group ? '--runnergroup' : '',\n      // this is split into 2 for powershell otherwise it will pass \"--runnergroup name\" as a single argument and config.sh will fail\n      this.group ? this.group : '',\n      this.defaultLabels ? '' : '--no-default-labels',\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, generateStateName(this, 'data'), {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we use ec2:RunInstances because we must\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n    // ec2:RunInstances also seemed like the only one to immediately return an error when spot capacity is not available\n\n    // we build a complicated chain of states here because ec2:RunInstances can only try one subnet at a time\n    // if someone can figure out a good way to use Map for this, please open a PR\n\n    // build a state for each subnet we want to try\n    const instanceProfile = new iam.CfnInstanceProfile(this, 'Instance Profile', {\n      roles: [this.role.roleName],\n    });\n    const rootDeviceResource = amiRootDevice(this, this.ami.launchTemplate.launchTemplateId);\n    rootDeviceResource.node.addDependency(this.amiBuilder);\n    const subnetRunners = this.subnets.map(subnet => {\n      return new stepfunctions_tasks.CallAwsService(this, generateStateName(this, subnet.subnetId), {\n        comment: subnet.availabilityZone,\n        integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n        service: 'ec2',\n        action: 'runInstances',\n        heartbeatTimeout: stepfunctions.Timeout.duration(Duration.minutes(10)),\n        parameters: {\n          LaunchTemplate: {\n            LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n          },\n          MinCount: 1,\n          MaxCount: 1,\n          InstanceType: this.instanceType.toString(),\n          UserData: stepfunctions.JsonPath.base64Encode(\n            stepfunctions.JsonPath.format(\n              stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n              ...params,\n            ),\n          ),\n          InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n          IamInstanceProfile: {\n            Arn: instanceProfile.attrArn,\n          },\n          MetadataOptions: {\n            HttpTokens: 'required',\n          },\n          SecurityGroupIds: this.securityGroups.map(sg => sg.securityGroupId),\n          SubnetId: subnet.subnetId,\n          BlockDeviceMappings: [{\n            DeviceName: rootDeviceResource.ref,\n            Ebs: {\n              DeleteOnTermination: true,\n              VolumeSize: this.storageSize.toGibibytes(),\n              VolumeType: this.storageOptions?.volumeType,\n              Iops: this.storageOptions?.iops,\n              Throughput: this.storageOptions?.throughput,\n            },\n          }],\n          InstanceMarketOptions: this.spot ? {\n            MarketType: 'spot',\n            SpotOptions: {\n              MaxPrice: this.spotMaxPrice,\n              SpotInstanceType: 'one-time',\n            },\n          } : undefined,\n          TagSpecifications: [ // manually propagate tags\n            {\n              ResourceType: 'instance',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n            {\n              ResourceType: 'volume',\n              Tags: [{\n                Key: 'GitHubRunners:Provider',\n                Value: this.node.path,\n              }],\n            },\n          ],\n        },\n        iamResources: ['*'],\n      });\n    });\n\n    // start with the first subnet\n    passUserData.next(subnetRunners[0]);\n\n    // chain up the rest of the subnets\n    for (let i = 1; i < subnetRunners.length; i++) {\n      subnetRunners[i - 1].addCatch(subnetRunners[i], {\n        errors: ['Ec2.Ec2Exception', 'States.Timeout'],\n        resultPath: stepfunctions.JsonPath.stringAt('$.lastSubnetError'),\n      });\n    }\n\n    return passUserData;\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:CreateServiceLinkedRole'],\n      resources: ['*'],\n      conditions: {\n        StringEquals: {\n          'iam:AWSServiceName': 'spot.amazonaws.com',\n        },\n      },\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      constructPath: this.node.path,\n      securityGroups: this.securityGroups.map(sg => sg.securityGroupId),\n      roleArn: this.role.roleArn,\n      logGroup: this.logGroup.logGroupName,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return new ec2.Connections({ securityGroups: this.securityGroups });\n  }\n}\n\n/**\n * @deprecated use {@link Ec2RunnerProvider}\n */\nexport class Ec2Runner extends Ec2RunnerProvider {\n}\n"]}