terraform-cdk-serverless-github-actions-runner-controller 0.0.1 → 0.0.3

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.
package/lib/lib/aws.js CHANGED
@@ -1,6 +1,8 @@
1
1
  "use strict";
2
+ var _a;
2
3
  Object.defineProperty(exports, "__esModule", { value: true });
3
4
  exports.Aws = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
4
6
  const cloudwatch_log_group_1 = require("@cdktf/provider-aws/lib/cloudwatch-log-group");
5
7
  const data_aws_caller_identity_1 = require("@cdktf/provider-aws/lib/data-aws-caller-identity");
6
8
  const data_aws_region_1 = require("@cdktf/provider-aws/lib/data-aws-region");
@@ -9,6 +11,8 @@ const data_aws_subnets_1 = require("@cdktf/provider-aws/lib/data-aws-subnets");
9
11
  const ecs_cluster_1 = require("@cdktf/provider-aws/lib/ecs-cluster");
10
12
  const ecs_service_1 = require("@cdktf/provider-aws/lib/ecs-service");
11
13
  const ecs_task_definition_1 = require("@cdktf/provider-aws/lib/ecs-task-definition");
14
+ const efs_file_system_1 = require("@cdktf/provider-aws/lib/efs-file-system");
15
+ const efs_mount_target_1 = require("@cdktf/provider-aws/lib/efs-mount-target");
12
16
  const iam_policy_1 = require("@cdktf/provider-aws/lib/iam-policy");
13
17
  const iam_role_1 = require("@cdktf/provider-aws/lib/iam-role");
14
18
  const iam_role_policy_attachment_1 = require("@cdktf/provider-aws/lib/iam-role-policy-attachment");
@@ -16,62 +20,67 @@ const provider_1 = require("@cdktf/provider-aws/lib/provider");
16
20
  const cdktf_1 = require("cdktf");
17
21
  const constructs_1 = require("constructs");
18
22
  const variables_1 = require("./variables");
19
- const efs_file_system_1 = require("@cdktf/provider-aws/lib/efs-file-system");
20
- const efs_mount_target_1 = require("@cdktf/provider-aws/lib/efs-mount-target");
21
23
  class Aws extends constructs_1.Construct {
22
- constructor(scope, id) {
24
+ constructor(scope, id, props) {
23
25
  super(scope, id);
24
26
  new provider_1.AwsProvider(this, 'aws', {});
25
27
  const identity = new data_aws_caller_identity_1.DataAwsCallerIdentity(this, 'Identity', {});
26
28
  const region = new data_aws_region_1.DataAwsRegion(this, 'Region', {});
27
- const { pat, githubConfigUrl } = (0, variables_1.commonVariables)(this);
29
+ const { pat, githubConfigUrl, autoscalerImage } = (0, variables_1.commonVariables)(this);
28
30
  const cluster = new ecs_cluster_1.EcsCluster(this, 'Cluster', {
29
- name: 'gha-runner-cluster',
31
+ name: props.clusterName,
30
32
  });
31
- const runnerRole = new iam_role_1.IamRole(this, 'RunnerRole', {
33
+ const subnets = new data_aws_subnets_1.DataAwsSubnets(this, 'Subnets', {
34
+ filter: props.subnetFilters,
35
+ });
36
+ const securityGroups = new data_aws_security_groups_1.DataAwsSecurityGroups(this, 'SecurityGroups', {
37
+ filter: props.securityGroupFilters,
38
+ });
39
+ const autoscalerRole = new iam_role_1.IamRole(this, 'AutoscalerRole', {
32
40
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
33
- 'Version': '2012-10-17',
34
- 'Statement': [
41
+ Version: '2012-10-17',
42
+ Statement: [
35
43
  {
36
- 'Effect': 'Allow',
37
- 'Principal': {
38
- 'Service': 'ecs-tasks.amazonaws.com'
44
+ Effect: 'Allow',
45
+ Principal: {
46
+ Service: 'ecs-tasks.amazonaws.com',
39
47
  },
40
- 'Action': 'sts:AssumeRole'
41
- }
42
- ]
43
- })
48
+ Action: 'sts:AssumeRole',
49
+ },
50
+ ],
51
+ }),
44
52
  });
45
- const autoscalerRole = new iam_role_1.IamRole(this, 'AutoscalerRole', {
53
+ const runnerRole = new iam_role_1.IamRole(this, 'RunnerRole', {
46
54
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
47
- 'Version': '2012-10-17',
48
- 'Statement': [
55
+ Version: '2012-10-17',
56
+ Statement: [
49
57
  {
50
- 'Effect': 'Allow',
51
- 'Principal': {
52
- 'Service': 'ecs-tasks.amazonaws.com'
58
+ Effect: 'Allow',
59
+ Principal: {
60
+ Service: 'ecs-tasks.amazonaws.com',
53
61
  },
54
- 'Action': 'sts:AssumeRole'
55
- }
56
- ]
57
- })
62
+ Action: 'sts:AssumeRole',
63
+ },
64
+ ],
65
+ }),
58
66
  });
59
67
  const ecsTaskExecutionRole = new iam_role_1.IamRole(this, 'TaskExecutionRole', {
60
68
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
61
- 'Version': '2012-10-17',
62
- 'Statement': [
69
+ Version: '2012-10-17',
70
+ Statement: [
63
71
  {
64
- 'Effect': 'Allow',
65
- 'Principal': {
66
- 'Service': 'ecs-tasks.amazonaws.com'
72
+ Effect: 'Allow',
73
+ Principal: {
74
+ Service: 'ecs-tasks.amazonaws.com',
67
75
  },
68
- 'Action': 'sts:AssumeRole'
69
- }
70
- ]
76
+ Action: 'sts:AssumeRole',
77
+ },
78
+ ],
71
79
  }),
72
- managedPolicyArns: [
73
- 'arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'
74
- ]
80
+ });
81
+ new iam_role_policy_attachment_1.IamRolePolicyAttachment(this, 'TeskExecutionRoleAttachment', {
82
+ policyArn: 'arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy',
83
+ role: ecsTaskExecutionRole.name,
75
84
  });
76
85
  const runnerLogGroup = new cloudwatch_log_group_1.CloudwatchLogGroup(this, 'RunnerLogGroup', {
77
86
  name: '/ecs/GHA',
@@ -79,99 +88,152 @@ class Aws extends constructs_1.Construct {
79
88
  const autoscalerLogGroup = new cloudwatch_log_group_1.CloudwatchLogGroup(this, 'AutoscalerLogGroup', {
80
89
  name: '/ecs/Autoscaler',
81
90
  });
82
- const subnets = new data_aws_subnets_1.DataAwsSubnets(this, 'Subnets', {});
83
- const securityGroups = new data_aws_security_groups_1.DataAwsSecurityGroups(this, 'SecurityGroups');
84
- // EFS volume to allow sharing data between tasks
85
- const efs = new efs_file_system_1.EfsFileSystem(this, 'efs', {
86
- throughputMode: 'elastic',
87
- tags: {
88
- Name: 'work'
89
- }
90
- });
91
- const externalsEfs = new efs_file_system_1.EfsFileSystem(this, 'externalsEfs', {
92
- throughputMode: 'elastic',
93
- tags: {
94
- Name: 'externals'
95
- }
96
- });
97
- // Each subnet in VPC are on different AZs, so creating mountpoint to each
98
- const iterator = cdktf_1.TerraformIterator.fromList(subnets.ids);
99
- new efs_mount_target_1.EfsMountTarget(this, 'EfsMountTarget', {
100
- forEach: iterator,
101
- fileSystemId: efs.id,
102
- subnetId: iterator.value
103
- });
104
- new efs_mount_target_1.EfsMountTarget(this, 'ExternalsEfsMountTarget', {
105
- forEach: iterator,
106
- fileSystemId: externalsEfs.id,
107
- subnetId: iterator.value
108
- });
109
- const runnerVolumeName = 'work';
110
- const externalsVolumeName = 'externals';
111
- const runnerContainerDefinitions = [
91
+ const autoscalerContainerDefinition = {
92
+ name: 'autoscaler',
93
+ image: autoscalerImage.stringValue,
94
+ essential: true,
95
+ environment: [
96
+ {
97
+ name: 'PAT',
98
+ value: pat.value,
99
+ },
100
+ {
101
+ name: 'GITHUB_CONFIG_URL',
102
+ value: githubConfigUrl.value,
103
+ },
104
+ {
105
+ name: 'ECS_CLUSTER',
106
+ value: cluster.arn,
107
+ },
108
+ {
109
+ name: 'ECS_SUBNETS',
110
+ value: cdktf_1.Fn.join(',', subnets.ids),
111
+ },
112
+ {
113
+ name: 'ECS_SECURITY_GROUPS',
114
+ value: cdktf_1.Fn.join(',', securityGroups.ids),
115
+ },
116
+ {
117
+ name: 'SCALE_SET_NAME',
118
+ value: 'ecs-runner-set',
119
+ },
120
+ ],
121
+ logConfiguration: {
122
+ logDriver: 'awslogs',
123
+ options: {
124
+ 'awslogs-group': autoscalerLogGroup.name,
125
+ 'awslogs-region': region.name,
126
+ 'awslogs-stream-prefix': 'ecs',
127
+ },
128
+ },
129
+ };
130
+ const runnerVolume = [];
131
+ const runnerContainerDefinition = {
132
+ name: 'runner',
133
+ image: 'ghcr.io/actions/actions-runner:2.323.0',
134
+ essential: true,
135
+ entryPoint: ['/home/runner/run.sh'],
136
+ environment: [
137
+ {
138
+ name: 'ECS_CLUSTER_NAME',
139
+ value: cluster.name,
140
+ },
141
+ {
142
+ name: 'ACTIONS_RUNNER_POD_NAME',
143
+ value: 'gha-pod',
144
+ },
145
+ {
146
+ name: 'ECS_SUBNETS',
147
+ value: cdktf_1.Fn.join(',', subnets.ids),
148
+ },
149
+ {
150
+ name: 'ECS_SECURITY_GROUPS',
151
+ value: cdktf_1.Fn.join(',', securityGroups.ids),
152
+ },
153
+ {
154
+ name: 'ECS_TASK_ROLE',
155
+ value: runnerRole.arn,
156
+ },
157
+ {
158
+ name: 'ECS_EXECUTION_ROLE',
159
+ value: ecsTaskExecutionRole.arn,
160
+ },
161
+ ],
162
+ mountPoints: [],
163
+ logConfiguration: {
164
+ logDriver: 'awslogs',
165
+ options: {
166
+ 'awslogs-group': runnerLogGroup.name,
167
+ 'awslogs-region': region.name,
168
+ 'awslogs-stream-prefix': 'ecs',
169
+ },
170
+ },
171
+ };
172
+ if (props.containerSupport) {
173
+ // EFS volume to allow sharing data between tasks
174
+ const efs = new efs_file_system_1.EfsFileSystem(this, 'efs', {
175
+ throughputMode: 'elastic',
176
+ tags: {
177
+ Name: 'work',
178
+ },
179
+ });
180
+ const externalsEfs = new efs_file_system_1.EfsFileSystem(this, 'externalsEfs', {
181
+ throughputMode: 'elastic',
182
+ tags: {
183
+ Name: 'externals',
184
+ },
185
+ });
186
+ // Each subnet in VPC are on different AZs, so creating mountpoint to each
187
+ const iterator = cdktf_1.TerraformIterator.fromList(subnets.ids);
188
+ new efs_mount_target_1.EfsMountTarget(this, 'EfsMountTarget', {
189
+ forEach: iterator,
190
+ fileSystemId: efs.id,
191
+ subnetId: iterator.value,
192
+ });
193
+ new efs_mount_target_1.EfsMountTarget(this, 'ExternalsEfsMountTarget', {
194
+ forEach: iterator,
195
+ fileSystemId: externalsEfs.id,
196
+ subnetId: iterator.value,
197
+ });
198
+ const runnerVolumeName = 'work';
199
+ const externalsVolumeName = 'externals';
200
+ runnerVolume.push({
201
+ name: runnerVolumeName,
202
+ efsVolumeConfiguration: {
203
+ fileSystemId: efs.id,
204
+ },
205
+ },
206
+ // This doesn't work with same volume, as volume is initially empty so it can't map to it's "externals" directory
112
207
  {
113
- name: 'runner',
114
- image: 'ghcr.io/hi-fi/actions-runner:ecs',
115
- command: ['/bin/sh', '-c', 'export EXECID=$(cat /proc/sys/kernel/random/uuid) && sudo mkdir -p /tmp/_work/$EXECID && sudo chown runner:runner /tmp/_work/$EXECID && ln -s /tmp/_work/$EXECID _work && sudo chown runner:runner /tmp/externals && /home/runner/run.sh ; sudo rm -r /tmp/_work/$EXECID'],
116
- essential: true,
117
- environment: [
118
- {
119
- name: 'EFS_ID',
120
- value: efs.id
121
- },
122
- {
123
- name: 'EXTERNALS_EFS_ID',
124
- value: externalsEfs.id
125
- },
126
- {
127
- name: 'ECS_CLUSTER_NAME',
128
- value: cluster.name
129
- },
130
- {
131
- name: 'ACTIONS_RUNNER_POD_NAME',
132
- value: 'gha-pod'
133
- },
134
- {
135
- name: 'ACTIONS_RUNNER_REQUIRE_JOB_CONTAINER',
136
- value: 'false'
137
- },
138
- {
139
- name: 'ECS_SUBNETS',
140
- value: cdktf_1.Fn.join(',', subnets.ids)
141
- },
142
- {
143
- name: 'ECS_SECURITY_GROUPS',
144
- value: cdktf_1.Fn.join(',', securityGroups.ids)
145
- },
146
- {
147
- name: 'ECS_TASK_ROLE',
148
- value: runnerRole.arn
149
- },
150
- {
151
- name: 'ECS_EXECUTION_ROLE',
152
- value: ecsTaskExecutionRole.arn
153
- }
154
- ],
155
- mountPoints: [
156
- {
157
- sourceVolume: runnerVolumeName,
158
- containerPath: '/tmp/_work',
159
- },
160
- {
161
- sourceVolume: externalsVolumeName,
162
- containerPath: '/tmp/externals',
163
- }
164
- ],
165
- logConfiguration: {
166
- logDriver: 'awslogs',
167
- options: {
168
- "awslogs-group": runnerLogGroup.name,
169
- "awslogs-region": region.name,
170
- "awslogs-stream-prefix": "ecs",
171
- }
172
- }
173
- }
174
- ];
208
+ name: externalsVolumeName,
209
+ efsVolumeConfiguration: {
210
+ fileSystemId: externalsEfs.id,
211
+ },
212
+ });
213
+ runnerContainerDefinition.mountPoints?.push({
214
+ sourceVolume: runnerVolumeName,
215
+ containerPath: '/tmp/_work',
216
+ }, {
217
+ sourceVolume: externalsVolumeName,
218
+ containerPath: '/tmp/externals',
219
+ });
220
+ runnerContainerDefinition.environment?.push({
221
+ name: 'EFS_ID',
222
+ value: efs.id,
223
+ }, {
224
+ name: 'EXTERNALS_EFS_ID',
225
+ value: externalsEfs.id,
226
+ }, {
227
+ name: 'ACTIONS_RUNNER_REQUIRE_JOB_CONTAINER',
228
+ value: 'false',
229
+ });
230
+ runnerContainerDefinition.command = [
231
+ '/bin/sh',
232
+ '-c',
233
+ 'export EXECID=$(cat /proc/sys/kernel/random/uuid) && sudo mkdir -p /tmp/_work/$EXECID && sudo chown runner:runner /tmp/_work/$EXECID && ln -s /tmp/_work/$EXECID _work && sudo chown runner:runner /tmp/externals && /home/runner/run.sh ; sudo rm -r /tmp/_work/$EXECID',
234
+ ];
235
+ runnerContainerDefinition.image = 'ghcr.io/hi-fi/actions-runner:ecs';
236
+ }
175
237
  // TODO: Images through caching: https://docs.aws.amazon.com/AmazonECR/latest/userguide/pull-through-cache.html (requires authentication)
176
238
  // TODO: Pass Execution role to job task: https://www.ernestchiang.com/en/posts/2021/using-amazon-ecs-exec/#1-grant-permissions-ecs-task-iam-role
177
239
  // TODO: Pass Task role to job task
@@ -179,194 +241,131 @@ class Aws extends constructs_1.Construct {
179
241
  family: 'GHA',
180
242
  taskRoleArn: runnerRole.arn,
181
243
  executionRoleArn: ecsTaskExecutionRole.arn,
182
- containerDefinitions: cdktf_1.Fn.jsonencode(runnerContainerDefinitions),
244
+ containerDefinitions: cdktf_1.Fn.jsonencode([runnerContainerDefinition]),
183
245
  cpu: '1024',
184
246
  memory: '2048',
185
- requiresCompatibilities: [
186
- 'FARGATE'
187
- ],
247
+ requiresCompatibilities: ['FARGATE'],
188
248
  runtimePlatform: {
189
249
  cpuArchitecture: 'X86_64',
190
- operatingSystemFamily: 'LINUX'
250
+ operatingSystemFamily: 'LINUX',
191
251
  },
192
252
  networkMode: 'awsvpc',
193
- volume: [
194
- {
195
- name: runnerVolumeName,
196
- efsVolumeConfiguration: {
197
- fileSystemId: efs.id,
198
- },
199
- },
200
- // This doesn't work with same volume, as volume is initially empty so it can't map to it's "externals" directory
201
- {
202
- name: externalsVolumeName,
203
- efsVolumeConfiguration: {
204
- fileSystemId: externalsEfs.id,
205
- }
206
- }
207
- ]
253
+ volume: runnerVolume,
254
+ });
255
+ autoscalerContainerDefinition.environment?.push({
256
+ name: 'TASK_DEFINITION_ARN',
257
+ value: runnerTaskDefinition.arn,
208
258
  });
209
259
  const autoscalerTaskDefinition = new ecs_task_definition_1.EcsTaskDefinition(this, 'AutoscalerTaskDefinition', {
210
260
  family: 'Autoscaler',
211
261
  taskRoleArn: autoscalerRole.arn,
212
262
  executionRoleArn: ecsTaskExecutionRole.arn,
213
- containerDefinitions: cdktf_1.Fn.jsonencode([
214
- {
215
- name: 'autoscaler',
216
- image: 'ghcr.io/hi-fi/gha-runners-on-managed-env:test',
217
- essential: true,
218
- environment: [
219
- {
220
- name: 'PAT',
221
- value: pat.value
222
- },
223
- {
224
- name: 'GITHUB_CONFIG_URL',
225
- value: githubConfigUrl.value
226
- },
227
- {
228
- name: 'TASK_DEFINITION_ARN',
229
- value: runnerTaskDefinition.arn
230
- },
231
- {
232
- name: 'ECS_CLUSTER',
233
- value: cluster.arn
234
- },
235
- {
236
- name: 'ECS_SUBNETS',
237
- value: cdktf_1.Fn.join(',', subnets.ids)
238
- },
239
- {
240
- name: 'ECS_SECURITY_GROUPS',
241
- value: cdktf_1.Fn.join(',', securityGroups.ids)
242
- },
243
- {
244
- name: 'SCALE_SET_NAME',
245
- value: 'ecs-runner-set'
246
- },
247
- ],
248
- logConfiguration: {
249
- logDriver: 'awslogs',
250
- options: {
251
- "awslogs-group": autoscalerLogGroup.name,
252
- "awslogs-region": region.name,
253
- "awslogs-stream-prefix": "ecs",
254
- }
255
- }
256
- }
257
- ]),
263
+ containerDefinitions: cdktf_1.Fn.jsonencode([autoscalerContainerDefinition]),
258
264
  cpu: '256',
259
265
  memory: '512',
260
- requiresCompatibilities: [
261
- 'FARGATE'
262
- ],
266
+ requiresCompatibilities: ['FARGATE'],
263
267
  runtimePlatform: {
264
268
  cpuArchitecture: 'X86_64',
265
- operatingSystemFamily: 'LINUX'
269
+ operatingSystemFamily: 'LINUX',
266
270
  },
267
271
  networkMode: 'awsvpc',
268
272
  });
273
+ const runnerPolicyDefinition = {
274
+ Version: '2012-10-17',
275
+ Statement: [
276
+ {
277
+ Sid: 'StartandMonitorTask',
278
+ Effect: 'Allow',
279
+ Action: [
280
+ 'ecs:RunTask',
281
+ 'ecs:TagResource',
282
+ 'ecs:ListTaskDefinitions',
283
+ 'ecs:ListTasks',
284
+ 'ecs:StopTask',
285
+ 'ecs:RegisterTaskDefinition',
286
+ 'ecs:DescribeTaskDefinition',
287
+ 'ecs:DeregisterTaskDefinition',
288
+ 'ecs:DeleteTaskDefinitions',
289
+ 'ecs:ExecuteCommand',
290
+ // Needed for waiting
291
+ 'ecs:DescribeTasks',
292
+ 'logs:GetLogEvents',
293
+ 'iam:PassRole',
294
+ 'logs:StartLiveTail',
295
+ 'logs:CreateLogStream',
296
+ ],
297
+ Resource: [
298
+ `arn:aws:ecs:${region.name}:${identity.accountId}:task-definition/gha-pod-workflow:*`,
299
+ cluster.arn,
300
+ // Triggerer has to be allowed to pass both task and task execution role
301
+ ecsTaskExecutionRole.arn,
302
+ runnerRole.arn,
303
+ `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,
304
+ //TODO: reorder rights so that listing is only one with star
305
+ '*',
306
+ ],
307
+ },
308
+ {
309
+ Sid: 'GetVpcInfo',
310
+ Effect: 'Allow',
311
+ Action: ['ec2:DescribeSubnets', 'ec2:DescribeSecurityGroups'],
312
+ Resource: '*',
313
+ },
314
+ {
315
+ Sid: 'ExecCommands',
316
+ Effect: 'Allow',
317
+ Action: [
318
+ 'ssmmessages:CreateControlChannel',
319
+ 'ssmmessages:CreateDataChannel',
320
+ 'ssmmessages:OpenControlChannel',
321
+ 'ssmmessages:OpenDataChannel',
322
+ ],
323
+ Resource: '*',
324
+ },
325
+ ],
326
+ };
269
327
  const runnerPolicy = new iam_policy_1.IamPolicy(this, 'RunnerPolicy', {
270
- policy: cdktf_1.Fn.jsonencode({
271
- 'Version': '2012-10-17',
272
- 'Statement': [
273
- {
274
- 'Sid': 'StartandMonitorTask',
275
- 'Effect': 'Allow',
276
- 'Action': [
277
- 'ecs:RunTask',
278
- 'ecs:TagResource',
279
- 'ecs:ListTaskDefinitions',
280
- 'ecs:ListTasks',
281
- 'ecs:StopTask',
282
- 'ecs:RegisterTaskDefinition',
283
- 'ecs:DescribeTaskDefinition',
284
- 'ecs:DeregisterTaskDefinition',
285
- 'ecs:DeleteTaskDefinitions',
286
- 'ecs:ExecuteCommand',
287
- // Needed for waiting
288
- 'ecs:DescribeTasks',
289
- 'logs:GetLogEvents',
290
- 'iam:PassRole',
291
- 'logs:StartLiveTail',
292
- 'logs:CreateLogStream',
293
- ],
294
- 'Resource': [
295
- `arn:aws:ecs:${region.name}:${identity.accountId}:task-definition/gha-pod-workflow:*`,
296
- cluster.arn,
297
- // Triggerer has to be allowed to pass both task and task execution role
298
- ecsTaskExecutionRole.arn,
299
- runnerRole.arn,
300
- `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,
301
- //TODO: reorder rights so that listing is only one with star
302
- '*'
303
- ]
304
- },
305
- {
306
- 'Sid': 'GetVpcInfo',
307
- 'Effect': 'Allow',
308
- 'Action': [
309
- 'ec2:DescribeSubnets',
310
- 'ec2:DescribeSecurityGroups'
311
- ],
312
- 'Resource': '*'
313
- },
314
- {
315
- 'Sid': 'ExecCommands',
316
- 'Effect': 'Allow',
317
- 'Action': [
318
- 'ssmmessages:CreateControlChannel',
319
- 'ssmmessages:CreateDataChannel',
320
- 'ssmmessages:OpenControlChannel',
321
- 'ssmmessages:OpenDataChannel'
322
- ],
323
- 'Resource': '*'
324
- }
325
- ]
326
- })
328
+ policy: cdktf_1.Fn.jsonencode(runnerPolicyDefinition),
327
329
  });
328
330
  new iam_role_policy_attachment_1.IamRolePolicyAttachment(this, 'RunnerPolicyAttachment', {
329
331
  policyArn: runnerPolicy.arn,
330
- role: runnerRole.name
332
+ role: runnerRole.name,
331
333
  });
332
334
  const autoscalerPolicy = new iam_policy_1.IamPolicy(this, 'AutoscalerPolicy', {
333
335
  policy: cdktf_1.Fn.jsonencode({
334
- 'Version': '2012-10-17',
335
- 'Statement': [
336
+ Version: '2012-10-17',
337
+ Statement: [
336
338
  {
337
- 'Sid': 'StartandMonitorTask',
338
- 'Effect': 'Allow',
339
- 'Action': [
339
+ Sid: 'StartandMonitorTask',
340
+ Effect: 'Allow',
341
+ Action: [
340
342
  'ecs:RunTask',
341
343
  // Needed for waiting
342
344
  'ecs:DescribeTasks',
343
345
  'logs:GetLogEvents',
344
346
  'iam:PassRole',
345
347
  ],
346
- 'Resource': [
348
+ Resource: [
347
349
  `${runnerTaskDefinition.arnWithoutRevision}:*`,
348
350
  // Triggerer has to be allowed to pass both task and task execution role
349
351
  ecsTaskExecutionRole.arn,
350
352
  runnerRole.arn,
351
353
  `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,
352
354
  `${runnerLogGroup.arn}:log-stream:*`,
353
- ]
355
+ ],
354
356
  },
355
357
  {
356
- 'Sid': 'GetVpcInfo',
357
- 'Effect': 'Allow',
358
- 'Action': [
359
- 'ec2:DescribeSubnets',
360
- 'ec2:DescribeSecurityGroups'
361
- ],
362
- 'Resource': '*'
363
- }
364
- ]
365
- })
358
+ Sid: 'GetVpcInfo',
359
+ Effect: 'Allow',
360
+ Action: ['ec2:DescribeSubnets', 'ec2:DescribeSecurityGroups'],
361
+ Resource: '*',
362
+ },
363
+ ],
364
+ }),
366
365
  });
367
366
  new iam_role_policy_attachment_1.IamRolePolicyAttachment(this, 'AutoscalerPolicyAttachment', {
368
367
  policyArn: autoscalerPolicy.arn,
369
- role: autoscalerRole.name
368
+ role: autoscalerRole.name,
370
369
  });
371
370
  new ecs_service_1.EcsService(this, 'AutoscalerService', {
372
371
  cluster: cluster.arn,
@@ -377,15 +376,15 @@ class Aws extends constructs_1.Construct {
377
376
  networkConfiguration: {
378
377
  assignPublicIp: true,
379
378
  subnets: subnets.ids,
380
- securityGroups: securityGroups.ids
379
+ securityGroups: securityGroups.ids,
381
380
  },
382
381
  lifecycle: {
383
- ignoreChanges: [
384
- 'desired_count'
385
- ]
386
- }
382
+ ignoreChanges: ['desired_count'],
383
+ },
387
384
  });
388
385
  }
389
386
  }
390
387
  exports.Aws = Aws;
391
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["../../src/lib/aws.ts"],"names":[],"mappings":"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","sourcesContent":["import { CloudwatchLogGroup } from '@cdktf/provider-aws/lib/cloudwatch-log-group';\nimport { DataAwsCallerIdentity } from '@cdktf/provider-aws/lib/data-aws-caller-identity';\nimport { DataAwsRegion } from '@cdktf/provider-aws/lib/data-aws-region';\nimport { DataAwsSecurityGroups } from '@cdktf/provider-aws/lib/data-aws-security-groups';\nimport { DataAwsSubnets } from '@cdktf/provider-aws/lib/data-aws-subnets';\nimport { EcsCluster } from '@cdktf/provider-aws/lib/ecs-cluster';\nimport { EcsService } from '@cdktf/provider-aws/lib/ecs-service';\nimport { EcsTaskDefinition } from '@cdktf/provider-aws/lib/ecs-task-definition';\nimport { IamPolicy } from '@cdktf/provider-aws/lib/iam-policy';\nimport { IamRole } from '@cdktf/provider-aws/lib/iam-role';\nimport { IamRolePolicyAttachment } from '@cdktf/provider-aws/lib/iam-role-policy-attachment';\nimport { AwsProvider } from '@cdktf/provider-aws/lib/provider';\nimport { Fn, TerraformIterator } from 'cdktf';\nimport { Construct } from 'constructs';\nimport { commonVariables } from './variables';\nimport { EfsFileSystem } from '@cdktf/provider-aws/lib/efs-file-system';\nimport { type ContainerDefinition } from '@aws-sdk/client-ecs'\nimport { EfsMountTarget } from '@cdktf/provider-aws/lib/efs-mount-target';\n\nexport class Aws extends Construct {\n    constructor(scope: Construct, id: string) {\n        super(scope, id);\n\n        new AwsProvider(this, 'aws', {\n\n        });\n\n        const identity = new DataAwsCallerIdentity(this, 'Identity', {});\n\n        const region = new DataAwsRegion(this, 'Region', {})\n\n        const { pat, githubConfigUrl } = commonVariables(this);\n\n        const cluster = new EcsCluster(this, 'Cluster', {\n            name: 'gha-runner-cluster',\n        });\n\n        const runnerRole = new IamRole(this, 'RunnerRole', {\n            assumeRolePolicy: Fn.jsonencode({\n                'Version': '2012-10-17',\n                'Statement': [\n                    {\n                        'Effect': 'Allow',\n                        'Principal': {\n                            'Service': 'ecs-tasks.amazonaws.com'\n                        },\n                        'Action': 'sts:AssumeRole'\n                    }\n                ]\n            })\n        })\n\n        const autoscalerRole = new IamRole(this, 'AutoscalerRole', {\n            assumeRolePolicy: Fn.jsonencode({\n                'Version': '2012-10-17',\n                'Statement': [\n                    {\n                        'Effect': 'Allow',\n                        'Principal': {\n                            'Service': 'ecs-tasks.amazonaws.com'\n                        },\n                        'Action': 'sts:AssumeRole'\n                    }\n                ]\n            })\n        })\n\n        const ecsTaskExecutionRole = new IamRole(this, 'TaskExecutionRole', {\n            assumeRolePolicy: Fn.jsonencode({\n                'Version': '2012-10-17',\n                'Statement': [\n                    {\n                        'Effect': 'Allow',\n                        'Principal': {\n                            'Service': 'ecs-tasks.amazonaws.com'\n                        },\n                        'Action': 'sts:AssumeRole'\n                    }\n                ]\n            }),\n            managedPolicyArns: [\n                'arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'\n            ]\n        })\n\n        const runnerLogGroup = new CloudwatchLogGroup(this, 'RunnerLogGroup', {\n            name: '/ecs/GHA',\n        });\n\n        const autoscalerLogGroup = new CloudwatchLogGroup(this, 'AutoscalerLogGroup', {\n            name: '/ecs/Autoscaler',\n        });\n\n\n        const subnets = new DataAwsSubnets(this, 'Subnets', {});\n\n        const securityGroups = new DataAwsSecurityGroups(this, 'SecurityGroups');\n\n        // EFS volume to allow sharing data between tasks\n        const efs = new EfsFileSystem(this, 'efs', {\n            throughputMode: 'elastic',\n            tags: {\n                Name: 'work'\n            }\n        })\n\n        const externalsEfs = new EfsFileSystem(this, 'externalsEfs', {\n            throughputMode: 'elastic',\n            tags: {\n                Name: 'externals'\n            }\n        })\n\n        // Each subnet in VPC are on different AZs, so creating mountpoint to each\n        const iterator = TerraformIterator.fromList(subnets.ids)\n\n        new EfsMountTarget(this, 'EfsMountTarget', {\n            forEach: iterator,\n            fileSystemId: efs.id,\n            subnetId: iterator.value\n        });\n\n        new EfsMountTarget(this, 'ExternalsEfsMountTarget', {\n            forEach: iterator,\n            fileSystemId: externalsEfs.id,\n            subnetId: iterator.value\n        });\n\n        const runnerVolumeName = 'work';\n        const externalsVolumeName = 'externals';\n        const runnerContainerDefinitions: ContainerDefinition[] = [\n            {\n            name: 'runner',\n            image: 'ghcr.io/hi-fi/actions-runner:ecs',\n            command: ['/bin/sh', '-c', 'export EXECID=$(cat /proc/sys/kernel/random/uuid) && sudo mkdir -p /tmp/_work/$EXECID && sudo chown runner:runner /tmp/_work/$EXECID && ln -s /tmp/_work/$EXECID _work && sudo chown runner:runner /tmp/externals && /home/runner/run.sh ; sudo rm -r /tmp/_work/$EXECID'],\n            essential: true,\n            environment: [\n                {\n                    name: 'EFS_ID',\n                    value: efs.id\n                },\n                {\n                    name: 'EXTERNALS_EFS_ID',\n                    value: externalsEfs.id\n                },\n                {\n                    name: 'ECS_CLUSTER_NAME',\n                    value: cluster.name\n                },\n                {\n                    name: 'ACTIONS_RUNNER_POD_NAME',\n                    value: 'gha-pod'\n                },\n                {\n                    name: 'ACTIONS_RUNNER_REQUIRE_JOB_CONTAINER',\n                    value: 'false'\n                },\n                {\n                    name: 'ECS_SUBNETS',\n                    value: Fn.join(',', subnets.ids)\n                },\n                {\n                    name: 'ECS_SECURITY_GROUPS',\n                    value: Fn.join(',', securityGroups.ids)\n                },\n                {\n                    name: 'ECS_TASK_ROLE',\n                    value: runnerRole.arn\n                },\n                {\n                    name: 'ECS_EXECUTION_ROLE',\n                    value: ecsTaskExecutionRole.arn\n                }\n            ],\n            mountPoints: [\n                {\n                    sourceVolume: runnerVolumeName,\n                    containerPath: '/tmp/_work',\n                },\n                {\n                    sourceVolume: externalsVolumeName,\n                    containerPath: '/tmp/externals',\n                }\n            ],\n            logConfiguration: {\n                logDriver: 'awslogs',\n                options: {\n                    \"awslogs-group\": runnerLogGroup.name,\n                    \"awslogs-region\": region.name,\n                    \"awslogs-stream-prefix\": \"ecs\",\n                }\n            }\n        }]\n        // TODO: Images through caching: https://docs.aws.amazon.com/AmazonECR/latest/userguide/pull-through-cache.html (requires authentication)\n        // TODO: Pass Execution role to job task: https://www.ernestchiang.com/en/posts/2021/using-amazon-ecs-exec/#1-grant-permissions-ecs-task-iam-role\n        // TODO: Pass Task role to job task\n        const runnerTaskDefinition = new EcsTaskDefinition(this, 'RunnerTaskDefinition', {\n            family: 'GHA',\n            taskRoleArn: runnerRole.arn,\n            executionRoleArn: ecsTaskExecutionRole.arn,\n            containerDefinitions: Fn.jsonencode(runnerContainerDefinitions),\n            cpu: '1024',\n            memory: '2048',\n            requiresCompatibilities: [\n                'FARGATE'\n            ],\n            runtimePlatform: {\n                cpuArchitecture: 'X86_64',\n                operatingSystemFamily: 'LINUX'\n            },\n            networkMode: 'awsvpc',\n            volume: [\n                {\n                    name: runnerVolumeName,\n                    efsVolumeConfiguration: {\n                        fileSystemId: efs.id,\n                    },\n                },\n                // This doesn't work with same volume, as volume is initially empty so it can't map to it's \"externals\" directory\n                {\n                    name: externalsVolumeName,\n                    efsVolumeConfiguration: {\n                        fileSystemId: externalsEfs.id,\n                    }\n                }\n            ]\n        })\n\n        const autoscalerTaskDefinition = new EcsTaskDefinition(this, 'AutoscalerTaskDefinition', {\n            family: 'Autoscaler',\n            taskRoleArn: autoscalerRole.arn,\n            executionRoleArn: ecsTaskExecutionRole.arn,\n            containerDefinitions: Fn.jsonencode([\n                {\n                    name: 'autoscaler',\n                    image: 'ghcr.io/hi-fi/gha-runners-on-managed-env:test',\n                    essential: true,\n                    environment: [\n                        {\n                            name: 'PAT',\n                            value: pat.value\n                        },\n                        {\n                            name: 'GITHUB_CONFIG_URL',\n                            value: githubConfigUrl.value\n                        },\n                        {\n                            name: 'TASK_DEFINITION_ARN',\n                            value: runnerTaskDefinition.arn\n                        },\n                        {\n                            name: 'ECS_CLUSTER',\n                            value: cluster.arn\n                        },\n                        {\n                            name: 'ECS_SUBNETS',\n                            value: Fn.join(',', subnets.ids)\n                        },\n                        {\n                            name: 'ECS_SECURITY_GROUPS',\n                            value: Fn.join(',', securityGroups.ids)\n                        },\n                        {\n                            name: 'SCALE_SET_NAME',\n                            value: 'ecs-runner-set'\n                        },\n                    ],\n                    logConfiguration: {\n                        logDriver: 'awslogs',\n                        options: {\n                            \"awslogs-group\": autoscalerLogGroup.name,\n                            \"awslogs-region\": region.name,\n                            \"awslogs-stream-prefix\": \"ecs\",\n                        }\n                    }\n                }\n            ]),\n            cpu: '256',\n            memory: '512',\n            requiresCompatibilities: [\n                'FARGATE'\n            ],\n            runtimePlatform: {\n                cpuArchitecture: 'X86_64',\n                operatingSystemFamily: 'LINUX'\n            },\n            networkMode: 'awsvpc',\n        })\n\n        const runnerPolicy = new IamPolicy(this, 'RunnerPolicy', {\n            policy: Fn.jsonencode({\n                'Version': '2012-10-17',\n                'Statement': [\n                    {\n                        'Sid': 'StartandMonitorTask',\n                        'Effect': 'Allow',\n                        'Action': [\n                            'ecs:RunTask',\n                            'ecs:TagResource',\n                            'ecs:ListTaskDefinitions',\n                            'ecs:ListTasks',\n                            'ecs:StopTask',\n                            'ecs:RegisterTaskDefinition',\n                            'ecs:DescribeTaskDefinition',\n                            'ecs:DeregisterTaskDefinition',\n                            'ecs:DeleteTaskDefinitions',\n                            'ecs:ExecuteCommand',\n                            // Needed for waiting\n                            'ecs:DescribeTasks',\n                            'logs:GetLogEvents',\n                            'iam:PassRole',\n                            'logs:StartLiveTail',\n                            'logs:CreateLogStream',\n                        ],\n                        'Resource': [\n                            `arn:aws:ecs:${region.name}:${identity.accountId}:task-definition/gha-pod-workflow:*`,\n                            cluster.arn,\n                            // Triggerer has to be allowed to pass both task and task execution role\n                            ecsTaskExecutionRole.arn,\n                            runnerRole.arn,\n                            `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,\n                            //TODO: reorder rights so that listing is only one with star\n                            '*'\n                        ]\n                    },\n                    {\n                        'Sid': 'GetVpcInfo',\n                        'Effect': 'Allow',\n                        'Action': [\n                            'ec2:DescribeSubnets',\n                            'ec2:DescribeSecurityGroups'\n                        ],\n                        'Resource': '*'\n                    },\n                    {\n                        'Sid': 'ExecCommands',\n                        'Effect': 'Allow',\n                        'Action': [\n                            'ssmmessages:CreateControlChannel',\n                            'ssmmessages:CreateDataChannel',\n                            'ssmmessages:OpenControlChannel',\n                            'ssmmessages:OpenDataChannel'\n                        ],\n                        'Resource': '*'\n                    }\n                ]\n            }\n\n            )\n        })\n\n        new IamRolePolicyAttachment(this, 'RunnerPolicyAttachment', {\n            policyArn: runnerPolicy.arn,\n            role: runnerRole.name\n        })\n\n        const autoscalerPolicy = new IamPolicy(this, 'AutoscalerPolicy', {\n            policy: Fn.jsonencode({\n                'Version': '2012-10-17',\n                'Statement': [\n                    {\n                        'Sid': 'StartandMonitorTask',\n                        'Effect': 'Allow',\n                        'Action': [\n                            'ecs:RunTask',\n                            // Needed for waiting\n                            'ecs:DescribeTasks',\n                            'logs:GetLogEvents',\n                            'iam:PassRole',\n                        ],\n                        'Resource': [\n                            `${runnerTaskDefinition.arnWithoutRevision}:*`,\n                            // Triggerer has to be allowed to pass both task and task execution role\n                            ecsTaskExecutionRole.arn,\n                            runnerRole.arn,\n                            `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,\n                            `${runnerLogGroup.arn}:log-stream:*`,\n                        ]\n                    },\n                    {\n                        'Sid': 'GetVpcInfo',\n                        'Effect': 'Allow',\n                        'Action': [\n                            'ec2:DescribeSubnets',\n                            'ec2:DescribeSecurityGroups'\n                        ],\n                        'Resource': '*'\n                    }\n                ]\n            }\n\n            )\n        })\n        new IamRolePolicyAttachment(this, 'AutoscalerPolicyAttachment', {\n            policyArn: autoscalerPolicy.arn,\n            role: autoscalerRole.name\n        })\n\n        new EcsService(this, 'AutoscalerService', {\n            cluster: cluster.arn,\n            name: 'autoscaler-service',\n            desiredCount: 1,\n            launchType: 'FARGATE',\n            taskDefinition: autoscalerTaskDefinition.arnWithoutRevision,\n            networkConfiguration: {\n                assignPublicIp: true,\n                subnets: subnets.ids,\n                securityGroups: securityGroups.ids\n            },\n            lifecycle: {\n                ignoreChanges: [\n                    'desired_count'\n                ]\n            }\n        })\n    }\n}\n"]}
388
+ _a = JSII_RTTI_SYMBOL_1;
389
+ Aws[_a] = { fqn: "terraform-cdk-serverless-github-actions-runner-controller.Aws", version: "0.0.3" };
390
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["../../src/lib/aws.ts"],"names":[],"mappings":"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","sourcesContent":["import { type ContainerDefinition } from '@aws-sdk/client-ecs'\nimport { CloudwatchLogGroup } from '@cdktf/provider-aws/lib/cloudwatch-log-group'\nimport { DataAwsCallerIdentity } from '@cdktf/provider-aws/lib/data-aws-caller-identity'\nimport { DataAwsRegion } from '@cdktf/provider-aws/lib/data-aws-region'\nimport {\n  DataAwsSecurityGroups,\n  DataAwsSecurityGroupsFilter,\n} from '@cdktf/provider-aws/lib/data-aws-security-groups'\nimport {\n  DataAwsSubnets,\n  DataAwsSubnetsFilter,\n} from '@cdktf/provider-aws/lib/data-aws-subnets'\nimport { EcsCluster } from '@cdktf/provider-aws/lib/ecs-cluster'\nimport { EcsService } from '@cdktf/provider-aws/lib/ecs-service'\nimport {\n  EcsTaskDefinition,\n  EcsTaskDefinitionVolume,\n} from '@cdktf/provider-aws/lib/ecs-task-definition'\nimport { EfsFileSystem } from '@cdktf/provider-aws/lib/efs-file-system'\nimport { EfsMountTarget } from '@cdktf/provider-aws/lib/efs-mount-target'\nimport { IamPolicy } from '@cdktf/provider-aws/lib/iam-policy'\nimport { IamRole } from '@cdktf/provider-aws/lib/iam-role'\nimport { IamRolePolicyAttachment } from '@cdktf/provider-aws/lib/iam-role-policy-attachment'\nimport { AwsProvider } from '@cdktf/provider-aws/lib/provider'\nimport { Fn, IResolvable, TerraformIterator } from 'cdktf'\nimport { Construct } from 'constructs'\nimport { commonVariables } from './variables'\n\nexport interface AwsProps {\n  // Name of the ECS cluster to create and use.\n  readonly clusterName: string\n  // Utilize runner-container-hooks to allow support for service and build container as well as container actions and steps.\n  readonly containerSupport: boolean\n  /**\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/APIReference/API_DescribeSubnets.html\n   */\n  readonly subnetFilters?: DataAwsSubnetsFilter[] | IResolvable\n  /**\n   * @see https://docs.aws.amazon.com/cli/latest/reference/ec2/describe-security-groups.html\n   */\n  readonly securityGroupFilters?: DataAwsSecurityGroupsFilter[] | IResolvable\n}\n\nexport class Aws extends Construct {\n  constructor(scope: Construct, id: string, props: AwsProps) {\n    super(scope, id)\n\n    new AwsProvider(this, 'aws', {})\n\n    const identity = new DataAwsCallerIdentity(this, 'Identity', {})\n\n    const region = new DataAwsRegion(this, 'Region', {})\n\n    const { pat, githubConfigUrl, autoscalerImage } = commonVariables(this)\n\n    const cluster = new EcsCluster(this, 'Cluster', {\n      name: props.clusterName,\n    })\n\n    const subnets = new DataAwsSubnets(this, 'Subnets', {\n      filter: props.subnetFilters,\n    })\n\n    const securityGroups = new DataAwsSecurityGroups(this, 'SecurityGroups', {\n      filter: props.securityGroupFilters,\n    })\n\n    const autoscalerRole = new IamRole(this, 'AutoscalerRole', {\n      assumeRolePolicy: Fn.jsonencode({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'ecs-tasks.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      }),\n    })\n\n    const runnerRole = new IamRole(this, 'RunnerRole', {\n      assumeRolePolicy: Fn.jsonencode({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'ecs-tasks.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      }),\n    })\n\n    const ecsTaskExecutionRole = new IamRole(this, 'TaskExecutionRole', {\n      assumeRolePolicy: Fn.jsonencode({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Effect: 'Allow',\n            Principal: {\n              Service: 'ecs-tasks.amazonaws.com',\n            },\n            Action: 'sts:AssumeRole',\n          },\n        ],\n      }),\n    })\n\n    new IamRolePolicyAttachment(this, 'TeskExecutionRoleAttachment', {\n      policyArn:\n        'arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy',\n      role: ecsTaskExecutionRole.name,\n    })\n\n    const runnerLogGroup = new CloudwatchLogGroup(this, 'RunnerLogGroup', {\n      name: '/ecs/GHA',\n    })\n\n    const autoscalerLogGroup = new CloudwatchLogGroup(\n      this,\n      'AutoscalerLogGroup',\n      {\n        name: '/ecs/Autoscaler',\n      },\n    )\n\n    const autoscalerContainerDefinition: ContainerDefinition = {\n      name: 'autoscaler',\n      image: autoscalerImage.stringValue,\n      essential: true,\n      environment: [\n        {\n          name: 'PAT',\n          value: pat.value,\n        },\n        {\n          name: 'GITHUB_CONFIG_URL',\n          value: githubConfigUrl.value,\n        },\n        {\n          name: 'ECS_CLUSTER',\n          value: cluster.arn,\n        },\n        {\n          name: 'ECS_SUBNETS',\n          value: Fn.join(',', subnets.ids),\n        },\n        {\n          name: 'ECS_SECURITY_GROUPS',\n          value: Fn.join(',', securityGroups.ids),\n        },\n        {\n          name: 'SCALE_SET_NAME',\n          value: 'ecs-runner-set',\n        },\n      ],\n      logConfiguration: {\n        logDriver: 'awslogs',\n        options: {\n          'awslogs-group': autoscalerLogGroup.name,\n          'awslogs-region': region.name,\n          'awslogs-stream-prefix': 'ecs',\n        },\n      },\n    }\n\n    const runnerVolume: EcsTaskDefinitionVolume[] = []\n\n    const runnerContainerDefinition: ContainerDefinition = {\n      name: 'runner',\n      image: 'ghcr.io/actions/actions-runner:2.323.0',\n      essential: true,\n      entryPoint: ['/home/runner/run.sh'],\n      environment: [\n        {\n          name: 'ECS_CLUSTER_NAME',\n          value: cluster.name,\n        },\n        {\n          name: 'ACTIONS_RUNNER_POD_NAME',\n          value: 'gha-pod',\n        },\n        {\n          name: 'ECS_SUBNETS',\n          value: Fn.join(',', subnets.ids),\n        },\n        {\n          name: 'ECS_SECURITY_GROUPS',\n          value: Fn.join(',', securityGroups.ids),\n        },\n        {\n          name: 'ECS_TASK_ROLE',\n          value: runnerRole.arn,\n        },\n        {\n          name: 'ECS_EXECUTION_ROLE',\n          value: ecsTaskExecutionRole.arn,\n        },\n      ],\n      mountPoints: [],\n      logConfiguration: {\n        logDriver: 'awslogs',\n        options: {\n          'awslogs-group': runnerLogGroup.name,\n          'awslogs-region': region.name,\n          'awslogs-stream-prefix': 'ecs',\n        },\n      },\n    }\n\n    if (props.containerSupport) {\n      // EFS volume to allow sharing data between tasks\n      const efs = new EfsFileSystem(this, 'efs', {\n        throughputMode: 'elastic',\n        tags: {\n          Name: 'work',\n        },\n      })\n\n      const externalsEfs = new EfsFileSystem(this, 'externalsEfs', {\n        throughputMode: 'elastic',\n        tags: {\n          Name: 'externals',\n        },\n      })\n\n      // Each subnet in VPC are on different AZs, so creating mountpoint to each\n      const iterator = TerraformIterator.fromList(subnets.ids)\n\n      new EfsMountTarget(this, 'EfsMountTarget', {\n        forEach: iterator,\n        fileSystemId: efs.id,\n        subnetId: iterator.value,\n      })\n\n      new EfsMountTarget(this, 'ExternalsEfsMountTarget', {\n        forEach: iterator,\n        fileSystemId: externalsEfs.id,\n        subnetId: iterator.value,\n      })\n\n      const runnerVolumeName = 'work'\n      const externalsVolumeName = 'externals'\n\n      runnerVolume.push(\n        {\n          name: runnerVolumeName,\n          efsVolumeConfiguration: {\n            fileSystemId: efs.id,\n          },\n        },\n        // This doesn't work with same volume, as volume is initially empty so it can't map to it's \"externals\" directory\n        {\n          name: externalsVolumeName,\n          efsVolumeConfiguration: {\n            fileSystemId: externalsEfs.id,\n          },\n        },\n      )\n\n      runnerContainerDefinition.mountPoints?.push(\n        {\n          sourceVolume: runnerVolumeName,\n          containerPath: '/tmp/_work',\n        },\n        {\n          sourceVolume: externalsVolumeName,\n          containerPath: '/tmp/externals',\n        },\n      )\n\n      runnerContainerDefinition.environment?.push(\n        {\n          name: 'EFS_ID',\n          value: efs.id,\n        },\n        {\n          name: 'EXTERNALS_EFS_ID',\n          value: externalsEfs.id,\n        },\n        {\n          name: 'ACTIONS_RUNNER_REQUIRE_JOB_CONTAINER',\n          value: 'false',\n        },\n      )\n\n      runnerContainerDefinition.command = [\n        '/bin/sh',\n        '-c',\n        'export EXECID=$(cat /proc/sys/kernel/random/uuid) && sudo mkdir -p /tmp/_work/$EXECID && sudo chown runner:runner /tmp/_work/$EXECID && ln -s /tmp/_work/$EXECID _work && sudo chown runner:runner /tmp/externals && /home/runner/run.sh ; sudo rm -r /tmp/_work/$EXECID',\n      ]\n      runnerContainerDefinition.image = 'ghcr.io/hi-fi/actions-runner:ecs'\n    }\n\n    // TODO: Images through caching: https://docs.aws.amazon.com/AmazonECR/latest/userguide/pull-through-cache.html (requires authentication)\n    // TODO: Pass Execution role to job task: https://www.ernestchiang.com/en/posts/2021/using-amazon-ecs-exec/#1-grant-permissions-ecs-task-iam-role\n    // TODO: Pass Task role to job task\n    const runnerTaskDefinition = new EcsTaskDefinition(\n      this,\n      'RunnerTaskDefinition',\n      {\n        family: 'GHA',\n        taskRoleArn: runnerRole.arn,\n        executionRoleArn: ecsTaskExecutionRole.arn,\n        containerDefinitions: Fn.jsonencode([runnerContainerDefinition]),\n        cpu: '1024',\n        memory: '2048',\n        requiresCompatibilities: ['FARGATE'],\n        runtimePlatform: {\n          cpuArchitecture: 'X86_64',\n          operatingSystemFamily: 'LINUX',\n        },\n        networkMode: 'awsvpc',\n        volume: runnerVolume,\n      },\n    )\n\n    autoscalerContainerDefinition.environment?.push({\n      name: 'TASK_DEFINITION_ARN',\n      value: runnerTaskDefinition.arn,\n    })\n\n    const autoscalerTaskDefinition = new EcsTaskDefinition(\n      this,\n      'AutoscalerTaskDefinition',\n      {\n        family: 'Autoscaler',\n        taskRoleArn: autoscalerRole.arn,\n        executionRoleArn: ecsTaskExecutionRole.arn,\n        containerDefinitions: Fn.jsonencode([autoscalerContainerDefinition]),\n        cpu: '256',\n        memory: '512',\n        requiresCompatibilities: ['FARGATE'],\n        runtimePlatform: {\n          cpuArchitecture: 'X86_64',\n          operatingSystemFamily: 'LINUX',\n        },\n        networkMode: 'awsvpc',\n      },\n    )\n\n    const runnerPolicyDefinition = {\n      Version: '2012-10-17',\n      Statement: [\n        {\n          Sid: 'StartandMonitorTask',\n          Effect: 'Allow',\n          Action: [\n            'ecs:RunTask',\n            'ecs:TagResource',\n            'ecs:ListTaskDefinitions',\n            'ecs:ListTasks',\n            'ecs:StopTask',\n            'ecs:RegisterTaskDefinition',\n            'ecs:DescribeTaskDefinition',\n            'ecs:DeregisterTaskDefinition',\n            'ecs:DeleteTaskDefinitions',\n            'ecs:ExecuteCommand',\n            // Needed for waiting\n            'ecs:DescribeTasks',\n            'logs:GetLogEvents',\n            'iam:PassRole',\n            'logs:StartLiveTail',\n            'logs:CreateLogStream',\n          ],\n          Resource: [\n            `arn:aws:ecs:${region.name}:${identity.accountId}:task-definition/gha-pod-workflow:*`,\n            cluster.arn,\n            // Triggerer has to be allowed to pass both task and task execution role\n            ecsTaskExecutionRole.arn,\n            runnerRole.arn,\n            `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,\n            //TODO: reorder rights so that listing is only one with star\n            '*',\n          ],\n        },\n        {\n          Sid: 'GetVpcInfo',\n          Effect: 'Allow',\n          Action: ['ec2:DescribeSubnets', 'ec2:DescribeSecurityGroups'],\n          Resource: '*',\n        },\n        {\n          Sid: 'ExecCommands',\n          Effect: 'Allow',\n          Action: [\n            'ssmmessages:CreateControlChannel',\n            'ssmmessages:CreateDataChannel',\n            'ssmmessages:OpenControlChannel',\n            'ssmmessages:OpenDataChannel',\n          ],\n          Resource: '*',\n        },\n      ],\n    }\n\n    const runnerPolicy = new IamPolicy(this, 'RunnerPolicy', {\n      policy: Fn.jsonencode(runnerPolicyDefinition),\n    })\n\n    new IamRolePolicyAttachment(this, 'RunnerPolicyAttachment', {\n      policyArn: runnerPolicy.arn,\n      role: runnerRole.name,\n    })\n\n    const autoscalerPolicy = new IamPolicy(this, 'AutoscalerPolicy', {\n      policy: Fn.jsonencode({\n        Version: '2012-10-17',\n        Statement: [\n          {\n            Sid: 'StartandMonitorTask',\n            Effect: 'Allow',\n            Action: [\n              'ecs:RunTask',\n              // Needed for waiting\n              'ecs:DescribeTasks',\n              'logs:GetLogEvents',\n              'iam:PassRole',\n            ],\n            Resource: [\n              `${runnerTaskDefinition.arnWithoutRevision}:*`,\n              // Triggerer has to be allowed to pass both task and task execution role\n              ecsTaskExecutionRole.arn,\n              runnerRole.arn,\n              `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,\n              `${runnerLogGroup.arn}:log-stream:*`,\n            ],\n          },\n          {\n            Sid: 'GetVpcInfo',\n            Effect: 'Allow',\n            Action: ['ec2:DescribeSubnets', 'ec2:DescribeSecurityGroups'],\n            Resource: '*',\n          },\n        ],\n      }),\n    })\n    new IamRolePolicyAttachment(this, 'AutoscalerPolicyAttachment', {\n      policyArn: autoscalerPolicy.arn,\n      role: autoscalerRole.name,\n    })\n\n    new EcsService(this, 'AutoscalerService', {\n      cluster: cluster.arn,\n      name: 'autoscaler-service',\n      desiredCount: 1,\n      launchType: 'FARGATE',\n      taskDefinition: autoscalerTaskDefinition.arnWithoutRevision,\n      networkConfiguration: {\n        assignPublicIp: true,\n        subnets: subnets.ids,\n        securityGroups: securityGroups.ids,\n      },\n      lifecycle: {\n        ignoreChanges: ['desired_count'],\n      },\n    })\n  }\n}\n"]}