terraform-cdk-serverless-github-actions-runner-controller 0.0.2 → 0.0.4

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
@@ -11,6 +11,8 @@ const data_aws_subnets_1 = require("@cdktf/provider-aws/lib/data-aws-subnets");
11
11
  const ecs_cluster_1 = require("@cdktf/provider-aws/lib/ecs-cluster");
12
12
  const ecs_service_1 = require("@cdktf/provider-aws/lib/ecs-service");
13
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");
14
16
  const iam_policy_1 = require("@cdktf/provider-aws/lib/iam-policy");
15
17
  const iam_role_1 = require("@cdktf/provider-aws/lib/iam-role");
16
18
  const iam_role_policy_attachment_1 = require("@cdktf/provider-aws/lib/iam-role-policy-attachment");
@@ -18,62 +20,67 @@ const provider_1 = require("@cdktf/provider-aws/lib/provider");
18
20
  const cdktf_1 = require("cdktf");
19
21
  const constructs_1 = require("constructs");
20
22
  const variables_1 = require("./variables");
21
- const efs_file_system_1 = require("@cdktf/provider-aws/lib/efs-file-system");
22
- const efs_mount_target_1 = require("@cdktf/provider-aws/lib/efs-mount-target");
23
23
  class Aws extends constructs_1.Construct {
24
- constructor(scope, id) {
24
+ constructor(scope, id, props) {
25
25
  super(scope, id);
26
26
  new provider_1.AwsProvider(this, 'aws', {});
27
27
  const identity = new data_aws_caller_identity_1.DataAwsCallerIdentity(this, 'Identity', {});
28
28
  const region = new data_aws_region_1.DataAwsRegion(this, 'Region', {});
29
- const { pat, githubConfigUrl } = (0, variables_1.commonVariables)(this);
29
+ const { pat, githubConfigUrl, autoscalerImage } = (0, variables_1.commonVariables)(this);
30
30
  const cluster = new ecs_cluster_1.EcsCluster(this, 'Cluster', {
31
- name: 'gha-runner-cluster',
31
+ name: props.clusterName,
32
32
  });
33
- 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', {
34
40
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
35
- 'Version': '2012-10-17',
36
- 'Statement': [
41
+ Version: '2012-10-17',
42
+ Statement: [
37
43
  {
38
- 'Effect': 'Allow',
39
- 'Principal': {
40
- 'Service': 'ecs-tasks.amazonaws.com'
44
+ Effect: 'Allow',
45
+ Principal: {
46
+ Service: 'ecs-tasks.amazonaws.com',
41
47
  },
42
- 'Action': 'sts:AssumeRole'
43
- }
44
- ]
45
- })
48
+ Action: 'sts:AssumeRole',
49
+ },
50
+ ],
51
+ }),
46
52
  });
47
- const autoscalerRole = new iam_role_1.IamRole(this, 'AutoscalerRole', {
53
+ const runnerRole = new iam_role_1.IamRole(this, 'RunnerRole', {
48
54
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
49
- 'Version': '2012-10-17',
50
- 'Statement': [
55
+ Version: '2012-10-17',
56
+ Statement: [
51
57
  {
52
- 'Effect': 'Allow',
53
- 'Principal': {
54
- 'Service': 'ecs-tasks.amazonaws.com'
58
+ Effect: 'Allow',
59
+ Principal: {
60
+ Service: 'ecs-tasks.amazonaws.com',
55
61
  },
56
- 'Action': 'sts:AssumeRole'
57
- }
58
- ]
59
- })
62
+ Action: 'sts:AssumeRole',
63
+ },
64
+ ],
65
+ }),
60
66
  });
61
67
  const ecsTaskExecutionRole = new iam_role_1.IamRole(this, 'TaskExecutionRole', {
62
68
  assumeRolePolicy: cdktf_1.Fn.jsonencode({
63
- 'Version': '2012-10-17',
64
- 'Statement': [
69
+ Version: '2012-10-17',
70
+ Statement: [
65
71
  {
66
- 'Effect': 'Allow',
67
- 'Principal': {
68
- 'Service': 'ecs-tasks.amazonaws.com'
72
+ Effect: 'Allow',
73
+ Principal: {
74
+ Service: 'ecs-tasks.amazonaws.com',
69
75
  },
70
- 'Action': 'sts:AssumeRole'
71
- }
72
- ]
76
+ Action: 'sts:AssumeRole',
77
+ },
78
+ ],
73
79
  }),
74
- managedPolicyArns: [
75
- 'arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy'
76
- ]
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,
77
84
  });
78
85
  const runnerLogGroup = new cloudwatch_log_group_1.CloudwatchLogGroup(this, 'RunnerLogGroup', {
79
86
  name: '/ecs/GHA',
@@ -81,99 +88,152 @@ class Aws extends constructs_1.Construct {
81
88
  const autoscalerLogGroup = new cloudwatch_log_group_1.CloudwatchLogGroup(this, 'AutoscalerLogGroup', {
82
89
  name: '/ecs/Autoscaler',
83
90
  });
84
- const subnets = new data_aws_subnets_1.DataAwsSubnets(this, 'Subnets', {});
85
- const securityGroups = new data_aws_security_groups_1.DataAwsSecurityGroups(this, 'SecurityGroups');
86
- // EFS volume to allow sharing data between tasks
87
- const efs = new efs_file_system_1.EfsFileSystem(this, 'efs', {
88
- throughputMode: 'elastic',
89
- tags: {
90
- Name: 'work'
91
- }
92
- });
93
- const externalsEfs = new efs_file_system_1.EfsFileSystem(this, 'externalsEfs', {
94
- throughputMode: 'elastic',
95
- tags: {
96
- Name: 'externals'
97
- }
98
- });
99
- // Each subnet in VPC are on different AZs, so creating mountpoint to each
100
- const iterator = cdktf_1.TerraformIterator.fromList(subnets.ids);
101
- new efs_mount_target_1.EfsMountTarget(this, 'EfsMountTarget', {
102
- forEach: iterator,
103
- fileSystemId: efs.id,
104
- subnetId: iterator.value
105
- });
106
- new efs_mount_target_1.EfsMountTarget(this, 'ExternalsEfsMountTarget', {
107
- forEach: iterator,
108
- fileSystemId: externalsEfs.id,
109
- subnetId: iterator.value
110
- });
111
- const runnerVolumeName = 'work';
112
- const externalsVolumeName = 'externals';
113
- 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
114
207
  {
115
- name: 'runner',
116
- image: 'ghcr.io/hi-fi/actions-runner:ecs',
117
- 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'],
118
- essential: true,
119
- environment: [
120
- {
121
- name: 'EFS_ID',
122
- value: efs.id
123
- },
124
- {
125
- name: 'EXTERNALS_EFS_ID',
126
- value: externalsEfs.id
127
- },
128
- {
129
- name: 'ECS_CLUSTER_NAME',
130
- value: cluster.name
131
- },
132
- {
133
- name: 'ACTIONS_RUNNER_POD_NAME',
134
- value: 'gha-pod'
135
- },
136
- {
137
- name: 'ACTIONS_RUNNER_REQUIRE_JOB_CONTAINER',
138
- value: 'false'
139
- },
140
- {
141
- name: 'ECS_SUBNETS',
142
- value: cdktf_1.Fn.join(',', subnets.ids)
143
- },
144
- {
145
- name: 'ECS_SECURITY_GROUPS',
146
- value: cdktf_1.Fn.join(',', securityGroups.ids)
147
- },
148
- {
149
- name: 'ECS_TASK_ROLE',
150
- value: runnerRole.arn
151
- },
152
- {
153
- name: 'ECS_EXECUTION_ROLE',
154
- value: ecsTaskExecutionRole.arn
155
- }
156
- ],
157
- mountPoints: [
158
- {
159
- sourceVolume: runnerVolumeName,
160
- containerPath: '/tmp/_work',
161
- },
162
- {
163
- sourceVolume: externalsVolumeName,
164
- containerPath: '/tmp/externals',
165
- }
166
- ],
167
- logConfiguration: {
168
- logDriver: 'awslogs',
169
- options: {
170
- "awslogs-group": runnerLogGroup.name,
171
- "awslogs-region": region.name,
172
- "awslogs-stream-prefix": "ecs",
173
- }
174
- }
175
- }
176
- ];
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
+ }
177
237
  // TODO: Images through caching: https://docs.aws.amazon.com/AmazonECR/latest/userguide/pull-through-cache.html (requires authentication)
178
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
179
239
  // TODO: Pass Task role to job task
@@ -181,194 +241,131 @@ class Aws extends constructs_1.Construct {
181
241
  family: 'GHA',
182
242
  taskRoleArn: runnerRole.arn,
183
243
  executionRoleArn: ecsTaskExecutionRole.arn,
184
- containerDefinitions: cdktf_1.Fn.jsonencode(runnerContainerDefinitions),
244
+ containerDefinitions: cdktf_1.Fn.jsonencode([runnerContainerDefinition]),
185
245
  cpu: '1024',
186
246
  memory: '2048',
187
- requiresCompatibilities: [
188
- 'FARGATE'
189
- ],
247
+ requiresCompatibilities: ['FARGATE'],
190
248
  runtimePlatform: {
191
249
  cpuArchitecture: 'X86_64',
192
- operatingSystemFamily: 'LINUX'
250
+ operatingSystemFamily: 'LINUX',
193
251
  },
194
252
  networkMode: 'awsvpc',
195
- volume: [
196
- {
197
- name: runnerVolumeName,
198
- efsVolumeConfiguration: {
199
- fileSystemId: efs.id,
200
- },
201
- },
202
- // This doesn't work with same volume, as volume is initially empty so it can't map to it's "externals" directory
203
- {
204
- name: externalsVolumeName,
205
- efsVolumeConfiguration: {
206
- fileSystemId: externalsEfs.id,
207
- }
208
- }
209
- ]
253
+ volume: runnerVolume,
254
+ });
255
+ autoscalerContainerDefinition.environment?.push({
256
+ name: 'TASK_DEFINITION_ARN',
257
+ value: runnerTaskDefinition.arn,
210
258
  });
211
259
  const autoscalerTaskDefinition = new ecs_task_definition_1.EcsTaskDefinition(this, 'AutoscalerTaskDefinition', {
212
260
  family: 'Autoscaler',
213
261
  taskRoleArn: autoscalerRole.arn,
214
262
  executionRoleArn: ecsTaskExecutionRole.arn,
215
- containerDefinitions: cdktf_1.Fn.jsonencode([
216
- {
217
- name: 'autoscaler',
218
- image: 'ghcr.io/hi-fi/gha-runners-on-managed-env:test',
219
- essential: true,
220
- environment: [
221
- {
222
- name: 'PAT',
223
- value: pat.value
224
- },
225
- {
226
- name: 'GITHUB_CONFIG_URL',
227
- value: githubConfigUrl.value
228
- },
229
- {
230
- name: 'TASK_DEFINITION_ARN',
231
- value: runnerTaskDefinition.arn
232
- },
233
- {
234
- name: 'ECS_CLUSTER',
235
- value: cluster.arn
236
- },
237
- {
238
- name: 'ECS_SUBNETS',
239
- value: cdktf_1.Fn.join(',', subnets.ids)
240
- },
241
- {
242
- name: 'ECS_SECURITY_GROUPS',
243
- value: cdktf_1.Fn.join(',', securityGroups.ids)
244
- },
245
- {
246
- name: 'SCALE_SET_NAME',
247
- value: 'ecs-runner-set'
248
- },
249
- ],
250
- logConfiguration: {
251
- logDriver: 'awslogs',
252
- options: {
253
- "awslogs-group": autoscalerLogGroup.name,
254
- "awslogs-region": region.name,
255
- "awslogs-stream-prefix": "ecs",
256
- }
257
- }
258
- }
259
- ]),
263
+ containerDefinitions: cdktf_1.Fn.jsonencode([autoscalerContainerDefinition]),
260
264
  cpu: '256',
261
265
  memory: '512',
262
- requiresCompatibilities: [
263
- 'FARGATE'
264
- ],
266
+ requiresCompatibilities: ['FARGATE'],
265
267
  runtimePlatform: {
266
268
  cpuArchitecture: 'X86_64',
267
- operatingSystemFamily: 'LINUX'
269
+ operatingSystemFamily: 'LINUX',
268
270
  },
269
271
  networkMode: 'awsvpc',
270
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
+ };
271
327
  const runnerPolicy = new iam_policy_1.IamPolicy(this, 'RunnerPolicy', {
272
- policy: cdktf_1.Fn.jsonencode({
273
- 'Version': '2012-10-17',
274
- 'Statement': [
275
- {
276
- 'Sid': 'StartandMonitorTask',
277
- 'Effect': 'Allow',
278
- 'Action': [
279
- 'ecs:RunTask',
280
- 'ecs:TagResource',
281
- 'ecs:ListTaskDefinitions',
282
- 'ecs:ListTasks',
283
- 'ecs:StopTask',
284
- 'ecs:RegisterTaskDefinition',
285
- 'ecs:DescribeTaskDefinition',
286
- 'ecs:DeregisterTaskDefinition',
287
- 'ecs:DeleteTaskDefinitions',
288
- 'ecs:ExecuteCommand',
289
- // Needed for waiting
290
- 'ecs:DescribeTasks',
291
- 'logs:GetLogEvents',
292
- 'iam:PassRole',
293
- 'logs:StartLiveTail',
294
- 'logs:CreateLogStream',
295
- ],
296
- 'Resource': [
297
- `arn:aws:ecs:${region.name}:${identity.accountId}:task-definition/gha-pod-workflow:*`,
298
- cluster.arn,
299
- // Triggerer has to be allowed to pass both task and task execution role
300
- ecsTaskExecutionRole.arn,
301
- runnerRole.arn,
302
- `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,
303
- //TODO: reorder rights so that listing is only one with star
304
- '*'
305
- ]
306
- },
307
- {
308
- 'Sid': 'GetVpcInfo',
309
- 'Effect': 'Allow',
310
- 'Action': [
311
- 'ec2:DescribeSubnets',
312
- 'ec2:DescribeSecurityGroups'
313
- ],
314
- 'Resource': '*'
315
- },
316
- {
317
- 'Sid': 'ExecCommands',
318
- 'Effect': 'Allow',
319
- 'Action': [
320
- 'ssmmessages:CreateControlChannel',
321
- 'ssmmessages:CreateDataChannel',
322
- 'ssmmessages:OpenControlChannel',
323
- 'ssmmessages:OpenDataChannel'
324
- ],
325
- 'Resource': '*'
326
- }
327
- ]
328
- })
328
+ policy: cdktf_1.Fn.jsonencode(runnerPolicyDefinition),
329
329
  });
330
330
  new iam_role_policy_attachment_1.IamRolePolicyAttachment(this, 'RunnerPolicyAttachment', {
331
331
  policyArn: runnerPolicy.arn,
332
- role: runnerRole.name
332
+ role: runnerRole.name,
333
333
  });
334
334
  const autoscalerPolicy = new iam_policy_1.IamPolicy(this, 'AutoscalerPolicy', {
335
335
  policy: cdktf_1.Fn.jsonencode({
336
- 'Version': '2012-10-17',
337
- 'Statement': [
336
+ Version: '2012-10-17',
337
+ Statement: [
338
338
  {
339
- 'Sid': 'StartandMonitorTask',
340
- 'Effect': 'Allow',
341
- 'Action': [
339
+ Sid: 'StartandMonitorTask',
340
+ Effect: 'Allow',
341
+ Action: [
342
342
  'ecs:RunTask',
343
343
  // Needed for waiting
344
344
  'ecs:DescribeTasks',
345
345
  'logs:GetLogEvents',
346
346
  'iam:PassRole',
347
347
  ],
348
- 'Resource': [
348
+ Resource: [
349
349
  `${runnerTaskDefinition.arnWithoutRevision}:*`,
350
350
  // Triggerer has to be allowed to pass both task and task execution role
351
351
  ecsTaskExecutionRole.arn,
352
352
  runnerRole.arn,
353
353
  `arn:aws:ecs:${region.name}:${identity.accountId}:task/${cluster.name}/*`,
354
354
  `${runnerLogGroup.arn}:log-stream:*`,
355
- ]
355
+ ],
356
356
  },
357
357
  {
358
- 'Sid': 'GetVpcInfo',
359
- 'Effect': 'Allow',
360
- 'Action': [
361
- 'ec2:DescribeSubnets',
362
- 'ec2:DescribeSecurityGroups'
363
- ],
364
- 'Resource': '*'
365
- }
366
- ]
367
- })
358
+ Sid: 'GetVpcInfo',
359
+ Effect: 'Allow',
360
+ Action: ['ec2:DescribeSubnets', 'ec2:DescribeSecurityGroups'],
361
+ Resource: '*',
362
+ },
363
+ ],
364
+ }),
368
365
  });
369
366
  new iam_role_policy_attachment_1.IamRolePolicyAttachment(this, 'AutoscalerPolicyAttachment', {
370
367
  policyArn: autoscalerPolicy.arn,
371
- role: autoscalerRole.name
368
+ role: autoscalerRole.name,
372
369
  });
373
370
  new ecs_service_1.EcsService(this, 'AutoscalerService', {
374
371
  cluster: cluster.arn,
@@ -379,17 +376,15 @@ class Aws extends constructs_1.Construct {
379
376
  networkConfiguration: {
380
377
  assignPublicIp: true,
381
378
  subnets: subnets.ids,
382
- securityGroups: securityGroups.ids
379
+ securityGroups: securityGroups.ids,
383
380
  },
384
381
  lifecycle: {
385
- ignoreChanges: [
386
- 'desired_count'
387
- ]
388
- }
382
+ ignoreChanges: ['desired_count'],
383
+ },
389
384
  });
390
385
  }
391
386
  }
392
387
  exports.Aws = Aws;
393
388
  _a = JSII_RTTI_SYMBOL_1;
394
- Aws[_a] = { fqn: "terraform-cdk-serverless-github-actions-runner-controller.Aws", version: "0.0.2" };
395
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["../../src/lib/aws.ts"],"names":[],"mappings":";;;;;AAAA,uFAAkF;AAClF,+FAAyF;AACzF,6EAAwE;AACxE,+FAAyF;AACzF,+EAA0E;AAC1E,qEAAiE;AACjE,qEAAiE;AACjE,qFAAgF;AAChF,mEAA+D;AAC/D,+DAA2D;AAC3D,mGAA6F;AAC7F,+DAA+D;AAC/D,iCAA8C;AAC9C,2CAAuC;AACvC,2CAA8C;AAC9C,6EAAwE;AAExE,+EAA0E;AAE1E,MAAa,GAAI,SAAQ,sBAAS;IAC9B,YAAY,KAAgB,EAAE,EAAU;QACpC,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,sBAAW,CAAC,IAAI,EAAE,KAAK,EAAE,EAE5B,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,IAAI,gDAAqB,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CAAC,CAAC;QAEjE,MAAM,MAAM,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAA;QAEpD,MAAM,EAAE,GAAG,EAAE,eAAe,EAAE,GAAG,IAAA,2BAAe,EAAC,IAAI,CAAC,CAAC;QAEvD,MAAM,OAAO,GAAG,IAAI,wBAAU,CAAC,IAAI,EAAE,SAAS,EAAE;YAC5C,IAAI,EAAE,oBAAoB;SAC7B,CAAC,CAAC;QAEH,MAAM,UAAU,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,YAAY,EAAE;YAC/C,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC5B,SAAS,EAAE,YAAY;gBACvB,WAAW,EAAE;oBACT;wBACI,QAAQ,EAAE,OAAO;wBACjB,WAAW,EAAE;4BACT,SAAS,EAAE,yBAAyB;yBACvC;wBACD,QAAQ,EAAE,gBAAgB;qBAC7B;iBACJ;aACJ,CAAC;SACL,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACvD,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC5B,SAAS,EAAE,YAAY;gBACvB,WAAW,EAAE;oBACT;wBACI,QAAQ,EAAE,OAAO;wBACjB,WAAW,EAAE;4BACT,SAAS,EAAE,yBAAyB;yBACvC;wBACD,QAAQ,EAAE,gBAAgB;qBAC7B;iBACJ;aACJ,CAAC;SACL,CAAC,CAAA;QAEF,MAAM,oBAAoB,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAChE,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC5B,SAAS,EAAE,YAAY;gBACvB,WAAW,EAAE;oBACT;wBACI,QAAQ,EAAE,OAAO;wBACjB,WAAW,EAAE;4BACT,SAAS,EAAE,yBAAyB;yBACvC;wBACD,QAAQ,EAAE,gBAAgB;qBAC7B;iBACJ;aACJ,CAAC;YACF,iBAAiB,EAAE;gBACf,uEAAuE;aAC1E;SACJ,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,yCAAkB,CAAC,IAAI,EAAE,gBAAgB,EAAE;YAClE,IAAI,EAAE,UAAU;SACnB,CAAC,CAAC;QAEH,MAAM,kBAAkB,GAAG,IAAI,yCAAkB,CAAC,IAAI,EAAE,oBAAoB,EAAE;YAC1E,IAAI,EAAE,iBAAiB;SAC1B,CAAC,CAAC;QAGH,MAAM,OAAO,GAAG,IAAI,iCAAc,CAAC,IAAI,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC;QAExD,MAAM,cAAc,GAAG,IAAI,gDAAqB,CAAC,IAAI,EAAE,gBAAgB,CAAC,CAAC;QAEzE,iDAAiD;QACjD,MAAM,GAAG,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,KAAK,EAAE;YACvC,cAAc,EAAE,SAAS;YACzB,IAAI,EAAE;gBACF,IAAI,EAAE,MAAM;aACf;SACJ,CAAC,CAAA;QAEF,MAAM,YAAY,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,cAAc,EAAE;YACzD,cAAc,EAAE,SAAS;YACzB,IAAI,EAAE;gBACF,IAAI,EAAE,WAAW;aACpB;SACJ,CAAC,CAAA;QAEF,0EAA0E;QAC1E,MAAM,QAAQ,GAAG,yBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAA;QAExD,IAAI,iCAAc,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACvC,OAAO,EAAE,QAAQ;YACjB,YAAY,EAAE,GAAG,CAAC,EAAE;YACpB,QAAQ,EAAE,QAAQ,CAAC,KAAK;SAC3B,CAAC,CAAC;QAEH,IAAI,iCAAc,CAAC,IAAI,EAAE,yBAAyB,EAAE;YAChD,OAAO,EAAE,QAAQ;YACjB,YAAY,EAAE,YAAY,CAAC,EAAE;YAC7B,QAAQ,EAAE,QAAQ,CAAC,KAAK;SAC3B,CAAC,CAAC;QAEH,MAAM,gBAAgB,GAAG,MAAM,CAAC;QAChC,MAAM,mBAAmB,GAAG,WAAW,CAAC;QACxC,MAAM,0BAA0B,GAA0B;YACtD;gBACA,IAAI,EAAE,QAAQ;gBACd,KAAK,EAAE,kCAAkC;gBACzC,OAAO,EAAE,CAAC,SAAS,EAAE,IAAI,EAAE,0QAA0Q,CAAC;gBACtS,SAAS,EAAE,IAAI;gBACf,WAAW,EAAE;oBACT;wBACI,IAAI,EAAE,QAAQ;wBACd,KAAK,EAAE,GAAG,CAAC,EAAE;qBAChB;oBACD;wBACI,IAAI,EAAE,kBAAkB;wBACxB,KAAK,EAAE,YAAY,CAAC,EAAE;qBACzB;oBACD;wBACI,IAAI,EAAE,kBAAkB;wBACxB,KAAK,EAAE,OAAO,CAAC,IAAI;qBACtB;oBACD;wBACI,IAAI,EAAE,yBAAyB;wBAC/B,KAAK,EAAE,SAAS;qBACnB;oBACD;wBACI,IAAI,EAAE,sCAAsC;wBAC5C,KAAK,EAAE,OAAO;qBACjB;oBACD;wBACI,IAAI,EAAE,aAAa;wBACnB,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC;qBACnC;oBACD;wBACI,IAAI,EAAE,qBAAqB;wBAC3B,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC;qBAC1C;oBACD;wBACI,IAAI,EAAE,eAAe;wBACrB,KAAK,EAAE,UAAU,CAAC,GAAG;qBACxB;oBACD;wBACI,IAAI,EAAE,oBAAoB;wBAC1B,KAAK,EAAE,oBAAoB,CAAC,GAAG;qBAClC;iBACJ;gBACD,WAAW,EAAE;oBACT;wBACI,YAAY,EAAE,gBAAgB;wBAC9B,aAAa,EAAE,YAAY;qBAC9B;oBACD;wBACI,YAAY,EAAE,mBAAmB;wBACjC,aAAa,EAAE,gBAAgB;qBAClC;iBACJ;gBACD,gBAAgB,EAAE;oBACd,SAAS,EAAE,SAAS;oBACpB,OAAO,EAAE;wBACL,eAAe,EAAE,cAAc,CAAC,IAAI;wBACpC,gBAAgB,EAAE,MAAM,CAAC,IAAI;wBAC7B,uBAAuB,EAAE,KAAK;qBACjC;iBACJ;aACJ;SAAC,CAAA;QACF,yIAAyI;QACzI,iJAAiJ;QACjJ,mCAAmC;QACnC,MAAM,oBAAoB,GAAG,IAAI,uCAAiB,CAAC,IAAI,EAAE,sBAAsB,EAAE;YAC7E,MAAM,EAAE,KAAK;YACb,WAAW,EAAE,UAAU,CAAC,GAAG;YAC3B,gBAAgB,EAAE,oBAAoB,CAAC,GAAG;YAC1C,oBAAoB,EAAE,UAAE,CAAC,UAAU,CAAC,0BAA0B,CAAC;YAC/D,GAAG,EAAE,MAAM;YACX,MAAM,EAAE,MAAM;YACd,uBAAuB,EAAE;gBACrB,SAAS;aACZ;YACD,eAAe,EAAE;gBACb,eAAe,EAAE,QAAQ;gBACzB,qBAAqB,EAAE,OAAO;aACjC;YACD,WAAW,EAAE,QAAQ;YACrB,MAAM,EAAE;gBACJ;oBACI,IAAI,EAAE,gBAAgB;oBACtB,sBAAsB,EAAE;wBACpB,YAAY,EAAE,GAAG,CAAC,EAAE;qBACvB;iBACJ;gBACD,iHAAiH;gBACjH;oBACI,IAAI,EAAE,mBAAmB;oBACzB,sBAAsB,EAAE;wBACpB,YAAY,EAAE,YAAY,CAAC,EAAE;qBAChC;iBACJ;aACJ;SACJ,CAAC,CAAA;QAEF,MAAM,wBAAwB,GAAG,IAAI,uCAAiB,CAAC,IAAI,EAAE,0BAA0B,EAAE;YACrF,MAAM,EAAE,YAAY;YACpB,WAAW,EAAE,cAAc,CAAC,GAAG;YAC/B,gBAAgB,EAAE,oBAAoB,CAAC,GAAG;YAC1C,oBAAoB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAChC;oBACI,IAAI,EAAE,YAAY;oBAClB,KAAK,EAAE,+CAA+C;oBACtD,SAAS,EAAE,IAAI;oBACf,WAAW,EAAE;wBACT;4BACI,IAAI,EAAE,KAAK;4BACX,KAAK,EAAE,GAAG,CAAC,KAAK;yBACnB;wBACD;4BACI,IAAI,EAAE,mBAAmB;4BACzB,KAAK,EAAE,eAAe,CAAC,KAAK;yBAC/B;wBACD;4BACI,IAAI,EAAE,qBAAqB;4BAC3B,KAAK,EAAE,oBAAoB,CAAC,GAAG;yBAClC;wBACD;4BACI,IAAI,EAAE,aAAa;4BACnB,KAAK,EAAE,OAAO,CAAC,GAAG;yBACrB;wBACD;4BACI,IAAI,EAAE,aAAa;4BACnB,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC;yBACnC;wBACD;4BACI,IAAI,EAAE,qBAAqB;4BAC3B,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC;yBAC1C;wBACD;4BACI,IAAI,EAAE,gBAAgB;4BACtB,KAAK,EAAE,gBAAgB;yBAC1B;qBACJ;oBACD,gBAAgB,EAAE;wBACd,SAAS,EAAE,SAAS;wBACpB,OAAO,EAAE;4BACL,eAAe,EAAE,kBAAkB,CAAC,IAAI;4BACxC,gBAAgB,EAAE,MAAM,CAAC,IAAI;4BAC7B,uBAAuB,EAAE,KAAK;yBACjC;qBACJ;iBACJ;aACJ,CAAC;YACF,GAAG,EAAE,KAAK;YACV,MAAM,EAAE,KAAK;YACb,uBAAuB,EAAE;gBACrB,SAAS;aACZ;YACD,eAAe,EAAE;gBACb,eAAe,EAAE,QAAQ;gBACzB,qBAAqB,EAAE,OAAO;aACjC;YACD,WAAW,EAAE,QAAQ;SACxB,CAAC,CAAA;QAEF,MAAM,YAAY,GAAG,IAAI,sBAAS,CAAC,IAAI,EAAE,cAAc,EAAE;YACrD,MAAM,EAAE,UAAE,CAAC,UAAU,CAAC;gBAClB,SAAS,EAAE,YAAY;gBACvB,WAAW,EAAE;oBACT;wBACI,KAAK,EAAE,qBAAqB;wBAC5B,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE;4BACN,aAAa;4BACb,iBAAiB;4BACjB,yBAAyB;4BACzB,eAAe;4BACf,cAAc;4BACd,4BAA4B;4BAC5B,4BAA4B;4BAC5B,8BAA8B;4BAC9B,2BAA2B;4BAC3B,oBAAoB;4BACpB,qBAAqB;4BACrB,mBAAmB;4BACnB,mBAAmB;4BACnB,cAAc;4BACd,oBAAoB;4BACpB,sBAAsB;yBACzB;wBACD,UAAU,EAAE;4BACR,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,qCAAqC;4BACrF,OAAO,CAAC,GAAG;4BACX,wEAAwE;4BACxE,oBAAoB,CAAC,GAAG;4BACxB,UAAU,CAAC,GAAG;4BACd,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,SAAS,OAAO,CAAC,IAAI,IAAI;4BACzE,4DAA4D;4BAC5D,GAAG;yBACN;qBACJ;oBACD;wBACI,KAAK,EAAE,YAAY;wBACnB,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE;4BACN,qBAAqB;4BACrB,4BAA4B;yBAC/B;wBACD,UAAU,EAAE,GAAG;qBAClB;oBACD;wBACI,KAAK,EAAE,cAAc;wBACrB,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE;4BACN,kCAAkC;4BAClC,+BAA+B;4BAC/B,gCAAgC;4BAChC,6BAA6B;yBAChC;wBACD,UAAU,EAAE,GAAG;qBAClB;iBACJ;aACJ,CAEA;SACJ,CAAC,CAAA;QAEF,IAAI,oDAAuB,CAAC,IAAI,EAAE,wBAAwB,EAAE;YACxD,SAAS,EAAE,YAAY,CAAC,GAAG;YAC3B,IAAI,EAAE,UAAU,CAAC,IAAI;SACxB,CAAC,CAAA;QAEF,MAAM,gBAAgB,GAAG,IAAI,sBAAS,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC7D,MAAM,EAAE,UAAE,CAAC,UAAU,CAAC;gBAClB,SAAS,EAAE,YAAY;gBACvB,WAAW,EAAE;oBACT;wBACI,KAAK,EAAE,qBAAqB;wBAC5B,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE;4BACN,aAAa;4BACb,qBAAqB;4BACrB,mBAAmB;4BACnB,mBAAmB;4BACnB,cAAc;yBACjB;wBACD,UAAU,EAAE;4BACR,GAAG,oBAAoB,CAAC,kBAAkB,IAAI;4BAC9C,wEAAwE;4BACxE,oBAAoB,CAAC,GAAG;4BACxB,UAAU,CAAC,GAAG;4BACd,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,SAAS,OAAO,CAAC,IAAI,IAAI;4BACzE,GAAG,cAAc,CAAC,GAAG,eAAe;yBACvC;qBACJ;oBACD;wBACI,KAAK,EAAE,YAAY;wBACnB,QAAQ,EAAE,OAAO;wBACjB,QAAQ,EAAE;4BACN,qBAAqB;4BACrB,4BAA4B;yBAC/B;wBACD,UAAU,EAAE,GAAG;qBAClB;iBACJ;aACJ,CAEA;SACJ,CAAC,CAAA;QACF,IAAI,oDAAuB,CAAC,IAAI,EAAE,4BAA4B,EAAE;YAC5D,SAAS,EAAE,gBAAgB,CAAC,GAAG;YAC/B,IAAI,EAAE,cAAc,CAAC,IAAI;SAC5B,CAAC,CAAA;QAEF,IAAI,wBAAU,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACtC,OAAO,EAAE,OAAO,CAAC,GAAG;YACpB,IAAI,EAAE,oBAAoB;YAC1B,YAAY,EAAE,CAAC;YACf,UAAU,EAAE,SAAS;YACrB,cAAc,EAAE,wBAAwB,CAAC,kBAAkB;YAC3D,oBAAoB,EAAE;gBAClB,cAAc,EAAE,IAAI;gBACpB,OAAO,EAAE,OAAO,CAAC,GAAG;gBACpB,cAAc,EAAE,cAAc,CAAC,GAAG;aACrC;YACD,SAAS,EAAE;gBACP,aAAa,EAAE;oBACX,eAAe;iBAClB;aACJ;SACJ,CAAC,CAAA;IACN,CAAC;;AA5YL,kBA6YC","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"]}
389
+ Aws[_a] = { fqn: "terraform-cdk-serverless-github-actions-runner-controller.Aws", version: "0.0.4" };
390
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["../../src/lib/aws.ts"],"names":[],"mappings":";;;;;AACA,uFAAiF;AACjF,+FAAwF;AACxF,6EAAuE;AACvE,+FAGyD;AACzD,+EAGiD;AACjD,qEAAgE;AAChE,qEAAgE;AAChE,qFAGoD;AACpD,6EAAuE;AACvE,+EAAyE;AACzE,mEAA8D;AAC9D,+DAA0D;AAC1D,mGAA4F;AAC5F,+DAA8D;AAC9D,iCAA0D;AAC1D,2CAAsC;AACtC,2CAA6C;AAiB7C,MAAa,GAAI,SAAQ,sBAAS;IAChC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAe;QACvD,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhB,IAAI,sBAAW,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC,CAAA;QAEhC,MAAM,QAAQ,GAAG,IAAI,gDAAqB,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CAAC,CAAA;QAEhE,MAAM,MAAM,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAA;QAEpD,MAAM,EAAE,GAAG,EAAE,eAAe,EAAE,eAAe,EAAE,GAAG,IAAA,2BAAe,EAAC,IAAI,CAAC,CAAA;QAEvE,MAAM,OAAO,GAAG,IAAI,wBAAU,CAAC,IAAI,EAAE,SAAS,EAAE;YAC9C,IAAI,EAAE,KAAK,CAAC,WAAW;SACxB,CAAC,CAAA;QAEF,MAAM,OAAO,GAAG,IAAI,iCAAc,CAAC,IAAI,EAAE,SAAS,EAAE;YAClD,MAAM,EAAE,KAAK,CAAC,aAAa;SAC5B,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,gDAAqB,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACvE,MAAM,EAAE,KAAK,CAAC,oBAAoB;SACnC,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACzD,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC9B,OAAO,EAAE,YAAY;gBACrB,SAAS,EAAE;oBACT;wBACE,MAAM,EAAE,OAAO;wBACf,SAAS,EAAE;4BACT,OAAO,EAAE,yBAAyB;yBACnC;wBACD,MAAM,EAAE,gBAAgB;qBACzB;iBACF;aACF,CAAC;SACH,CAAC,CAAA;QAEF,MAAM,UAAU,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,YAAY,EAAE;YACjD,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC9B,OAAO,EAAE,YAAY;gBACrB,SAAS,EAAE;oBACT;wBACE,MAAM,EAAE,OAAO;wBACf,SAAS,EAAE;4BACT,OAAO,EAAE,yBAAyB;yBACnC;wBACD,MAAM,EAAE,gBAAgB;qBACzB;iBACF;aACF,CAAC;SACH,CAAC,CAAA;QAEF,MAAM,oBAAoB,GAAG,IAAI,kBAAO,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAClE,gBAAgB,EAAE,UAAE,CAAC,UAAU,CAAC;gBAC9B,OAAO,EAAE,YAAY;gBACrB,SAAS,EAAE;oBACT;wBACE,MAAM,EAAE,OAAO;wBACf,SAAS,EAAE;4BACT,OAAO,EAAE,yBAAyB;yBACnC;wBACD,MAAM,EAAE,gBAAgB;qBACzB;iBACF;aACF,CAAC;SACH,CAAC,CAAA;QAEF,IAAI,oDAAuB,CAAC,IAAI,EAAE,6BAA6B,EAAE;YAC/D,SAAS,EACP,uEAAuE;YACzE,IAAI,EAAE,oBAAoB,CAAC,IAAI;SAChC,CAAC,CAAA;QAEF,MAAM,cAAc,GAAG,IAAI,yCAAkB,CAAC,IAAI,EAAE,gBAAgB,EAAE;YACpE,IAAI,EAAE,UAAU;SACjB,CAAC,CAAA;QAEF,MAAM,kBAAkB,GAAG,IAAI,yCAAkB,CAC/C,IAAI,EACJ,oBAAoB,EACpB;YACE,IAAI,EAAE,iBAAiB;SACxB,CACF,CAAA;QAED,MAAM,6BAA6B,GAAwB;YACzD,IAAI,EAAE,YAAY;YAClB,KAAK,EAAE,eAAe,CAAC,WAAW;YAClC,SAAS,EAAE,IAAI;YACf,WAAW,EAAE;gBACX;oBACE,IAAI,EAAE,KAAK;oBACX,KAAK,EAAE,GAAG,CAAC,KAAK;iBACjB;gBACD;oBACE,IAAI,EAAE,mBAAmB;oBACzB,KAAK,EAAE,eAAe,CAAC,KAAK;iBAC7B;gBACD;oBACE,IAAI,EAAE,aAAa;oBACnB,KAAK,EAAE,OAAO,CAAC,GAAG;iBACnB;gBACD;oBACE,IAAI,EAAE,aAAa;oBACnB,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC;iBACjC;gBACD;oBACE,IAAI,EAAE,qBAAqB;oBAC3B,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC;iBACxC;gBACD;oBACE,IAAI,EAAE,gBAAgB;oBACtB,KAAK,EAAE,gBAAgB;iBACxB;aACF;YACD,gBAAgB,EAAE;gBAChB,SAAS,EAAE,SAAS;gBACpB,OAAO,EAAE;oBACP,eAAe,EAAE,kBAAkB,CAAC,IAAI;oBACxC,gBAAgB,EAAE,MAAM,CAAC,IAAI;oBAC7B,uBAAuB,EAAE,KAAK;iBAC/B;aACF;SACF,CAAA;QAED,MAAM,YAAY,GAA8B,EAAE,CAAA;QAElD,MAAM,yBAAyB,GAAwB;YACrD,IAAI,EAAE,QAAQ;YACd,KAAK,EAAE,wCAAwC;YAC/C,SAAS,EAAE,IAAI;YACf,UAAU,EAAE,CAAC,qBAAqB,CAAC;YACnC,WAAW,EAAE;gBACX;oBACE,IAAI,EAAE,kBAAkB;oBACxB,KAAK,EAAE,OAAO,CAAC,IAAI;iBACpB;gBACD;oBACE,IAAI,EAAE,yBAAyB;oBAC/B,KAAK,EAAE,SAAS;iBACjB;gBACD;oBACE,IAAI,EAAE,aAAa;oBACnB,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC;iBACjC;gBACD;oBACE,IAAI,EAAE,qBAAqB;oBAC3B,KAAK,EAAE,UAAE,CAAC,IAAI,CAAC,GAAG,EAAE,cAAc,CAAC,GAAG,CAAC;iBACxC;gBACD;oBACE,IAAI,EAAE,eAAe;oBACrB,KAAK,EAAE,UAAU,CAAC,GAAG;iBACtB;gBACD;oBACE,IAAI,EAAE,oBAAoB;oBAC1B,KAAK,EAAE,oBAAoB,CAAC,GAAG;iBAChC;aACF;YACD,WAAW,EAAE,EAAE;YACf,gBAAgB,EAAE;gBAChB,SAAS,EAAE,SAAS;gBACpB,OAAO,EAAE;oBACP,eAAe,EAAE,cAAc,CAAC,IAAI;oBACpC,gBAAgB,EAAE,MAAM,CAAC,IAAI;oBAC7B,uBAAuB,EAAE,KAAK;iBAC/B;aACF;SACF,CAAA;QAED,IAAI,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAC3B,iDAAiD;YACjD,MAAM,GAAG,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,KAAK,EAAE;gBACzC,cAAc,EAAE,SAAS;gBACzB,IAAI,EAAE;oBACJ,IAAI,EAAE,MAAM;iBACb;aACF,CAAC,CAAA;YAEF,MAAM,YAAY,GAAG,IAAI,+BAAa,CAAC,IAAI,EAAE,cAAc,EAAE;gBAC3D,cAAc,EAAE,SAAS;gBACzB,IAAI,EAAE;oBACJ,IAAI,EAAE,WAAW;iBAClB;aACF,CAAC,CAAA;YAEF,0EAA0E;YAC1E,MAAM,QAAQ,GAAG,yBAAiB,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAA;YAExD,IAAI,iCAAc,CAAC,IAAI,EAAE,gBAAgB,EAAE;gBACzC,OAAO,EAAE,QAAQ;gBACjB,YAAY,EAAE,GAAG,CAAC,EAAE;gBACpB,QAAQ,EAAE,QAAQ,CAAC,KAAK;aACzB,CAAC,CAAA;YAEF,IAAI,iCAAc,CAAC,IAAI,EAAE,yBAAyB,EAAE;gBAClD,OAAO,EAAE,QAAQ;gBACjB,YAAY,EAAE,YAAY,CAAC,EAAE;gBAC7B,QAAQ,EAAE,QAAQ,CAAC,KAAK;aACzB,CAAC,CAAA;YAEF,MAAM,gBAAgB,GAAG,MAAM,CAAA;YAC/B,MAAM,mBAAmB,GAAG,WAAW,CAAA;YAEvC,YAAY,CAAC,IAAI,CACf;gBACE,IAAI,EAAE,gBAAgB;gBACtB,sBAAsB,EAAE;oBACtB,YAAY,EAAE,GAAG,CAAC,EAAE;iBACrB;aACF;YACD,iHAAiH;YACjH;gBACE,IAAI,EAAE,mBAAmB;gBACzB,sBAAsB,EAAE;oBACtB,YAAY,EAAE,YAAY,CAAC,EAAE;iBAC9B;aACF,CACF,CAAA;YAED,yBAAyB,CAAC,WAAW,EAAE,IAAI,CACzC;gBACE,YAAY,EAAE,gBAAgB;gBAC9B,aAAa,EAAE,YAAY;aAC5B,EACD;gBACE,YAAY,EAAE,mBAAmB;gBACjC,aAAa,EAAE,gBAAgB;aAChC,CACF,CAAA;YAED,yBAAyB,CAAC,WAAW,EAAE,IAAI,CACzC;gBACE,IAAI,EAAE,QAAQ;gBACd,KAAK,EAAE,GAAG,CAAC,EAAE;aACd,EACD;gBACE,IAAI,EAAE,kBAAkB;gBACxB,KAAK,EAAE,YAAY,CAAC,EAAE;aACvB,EACD;gBACE,IAAI,EAAE,sCAAsC;gBAC5C,KAAK,EAAE,OAAO;aACf,CACF,CAAA;YAED,yBAAyB,CAAC,OAAO,GAAG;gBAClC,SAAS;gBACT,IAAI;gBACJ,0QAA0Q;aAC3Q,CAAA;YACD,yBAAyB,CAAC,KAAK,GAAG,kCAAkC,CAAA;QACtE,CAAC;QAED,yIAAyI;QACzI,iJAAiJ;QACjJ,mCAAmC;QACnC,MAAM,oBAAoB,GAAG,IAAI,uCAAiB,CAChD,IAAI,EACJ,sBAAsB,EACtB;YACE,MAAM,EAAE,KAAK;YACb,WAAW,EAAE,UAAU,CAAC,GAAG;YAC3B,gBAAgB,EAAE,oBAAoB,CAAC,GAAG;YAC1C,oBAAoB,EAAE,UAAE,CAAC,UAAU,CAAC,CAAC,yBAAyB,CAAC,CAAC;YAChE,GAAG,EAAE,MAAM;YACX,MAAM,EAAE,MAAM;YACd,uBAAuB,EAAE,CAAC,SAAS,CAAC;YACpC,eAAe,EAAE;gBACf,eAAe,EAAE,QAAQ;gBACzB,qBAAqB,EAAE,OAAO;aAC/B;YACD,WAAW,EAAE,QAAQ;YACrB,MAAM,EAAE,YAAY;SACrB,CACF,CAAA;QAED,6BAA6B,CAAC,WAAW,EAAE,IAAI,CAAC;YAC9C,IAAI,EAAE,qBAAqB;YAC3B,KAAK,EAAE,oBAAoB,CAAC,GAAG;SAChC,CAAC,CAAA;QAEF,MAAM,wBAAwB,GAAG,IAAI,uCAAiB,CACpD,IAAI,EACJ,0BAA0B,EAC1B;YACE,MAAM,EAAE,YAAY;YACpB,WAAW,EAAE,cAAc,CAAC,GAAG;YAC/B,gBAAgB,EAAE,oBAAoB,CAAC,GAAG;YAC1C,oBAAoB,EAAE,UAAE,CAAC,UAAU,CAAC,CAAC,6BAA6B,CAAC,CAAC;YACpE,GAAG,EAAE,KAAK;YACV,MAAM,EAAE,KAAK;YACb,uBAAuB,EAAE,CAAC,SAAS,CAAC;YACpC,eAAe,EAAE;gBACf,eAAe,EAAE,QAAQ;gBACzB,qBAAqB,EAAE,OAAO;aAC/B;YACD,WAAW,EAAE,QAAQ;SACtB,CACF,CAAA;QAED,MAAM,sBAAsB,GAAG;YAC7B,OAAO,EAAE,YAAY;YACrB,SAAS,EAAE;gBACT;oBACE,GAAG,EAAE,qBAAqB;oBAC1B,MAAM,EAAE,OAAO;oBACf,MAAM,EAAE;wBACN,aAAa;wBACb,iBAAiB;wBACjB,yBAAyB;wBACzB,eAAe;wBACf,cAAc;wBACd,4BAA4B;wBAC5B,4BAA4B;wBAC5B,8BAA8B;wBAC9B,2BAA2B;wBAC3B,oBAAoB;wBACpB,qBAAqB;wBACrB,mBAAmB;wBACnB,mBAAmB;wBACnB,cAAc;wBACd,oBAAoB;wBACpB,sBAAsB;qBACvB;oBACD,QAAQ,EAAE;wBACR,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,qCAAqC;wBACrF,OAAO,CAAC,GAAG;wBACX,wEAAwE;wBACxE,oBAAoB,CAAC,GAAG;wBACxB,UAAU,CAAC,GAAG;wBACd,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,SAAS,OAAO,CAAC,IAAI,IAAI;wBACzE,4DAA4D;wBAC5D,GAAG;qBACJ;iBACF;gBACD;oBACE,GAAG,EAAE,YAAY;oBACjB,MAAM,EAAE,OAAO;oBACf,MAAM,EAAE,CAAC,qBAAqB,EAAE,4BAA4B,CAAC;oBAC7D,QAAQ,EAAE,GAAG;iBACd;gBACD;oBACE,GAAG,EAAE,cAAc;oBACnB,MAAM,EAAE,OAAO;oBACf,MAAM,EAAE;wBACN,kCAAkC;wBAClC,+BAA+B;wBAC/B,gCAAgC;wBAChC,6BAA6B;qBAC9B;oBACD,QAAQ,EAAE,GAAG;iBACd;aACF;SACF,CAAA;QAED,MAAM,YAAY,GAAG,IAAI,sBAAS,CAAC,IAAI,EAAE,cAAc,EAAE;YACvD,MAAM,EAAE,UAAE,CAAC,UAAU,CAAC,sBAAsB,CAAC;SAC9C,CAAC,CAAA;QAEF,IAAI,oDAAuB,CAAC,IAAI,EAAE,wBAAwB,EAAE;YAC1D,SAAS,EAAE,YAAY,CAAC,GAAG;YAC3B,IAAI,EAAE,UAAU,CAAC,IAAI;SACtB,CAAC,CAAA;QAEF,MAAM,gBAAgB,GAAG,IAAI,sBAAS,CAAC,IAAI,EAAE,kBAAkB,EAAE;YAC/D,MAAM,EAAE,UAAE,CAAC,UAAU,CAAC;gBACpB,OAAO,EAAE,YAAY;gBACrB,SAAS,EAAE;oBACT;wBACE,GAAG,EAAE,qBAAqB;wBAC1B,MAAM,EAAE,OAAO;wBACf,MAAM,EAAE;4BACN,aAAa;4BACb,qBAAqB;4BACrB,mBAAmB;4BACnB,mBAAmB;4BACnB,cAAc;yBACf;wBACD,QAAQ,EAAE;4BACR,GAAG,oBAAoB,CAAC,kBAAkB,IAAI;4BAC9C,wEAAwE;4BACxE,oBAAoB,CAAC,GAAG;4BACxB,UAAU,CAAC,GAAG;4BACd,eAAe,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,SAAS,SAAS,OAAO,CAAC,IAAI,IAAI;4BACzE,GAAG,cAAc,CAAC,GAAG,eAAe;yBACrC;qBACF;oBACD;wBACE,GAAG,EAAE,YAAY;wBACjB,MAAM,EAAE,OAAO;wBACf,MAAM,EAAE,CAAC,qBAAqB,EAAE,4BAA4B,CAAC;wBAC7D,QAAQ,EAAE,GAAG;qBACd;iBACF;aACF,CAAC;SACH,CAAC,CAAA;QACF,IAAI,oDAAuB,CAAC,IAAI,EAAE,4BAA4B,EAAE;YAC9D,SAAS,EAAE,gBAAgB,CAAC,GAAG;YAC/B,IAAI,EAAE,cAAc,CAAC,IAAI;SAC1B,CAAC,CAAA;QAEF,IAAI,wBAAU,CAAC,IAAI,EAAE,mBAAmB,EAAE;YACxC,OAAO,EAAE,OAAO,CAAC,GAAG;YACpB,IAAI,EAAE,oBAAoB;YAC1B,YAAY,EAAE,CAAC;YACf,UAAU,EAAE,SAAS;YACrB,cAAc,EAAE,wBAAwB,CAAC,kBAAkB;YAC3D,oBAAoB,EAAE;gBACpB,cAAc,EAAE,IAAI;gBACpB,OAAO,EAAE,OAAO,CAAC,GAAG;gBACpB,cAAc,EAAE,cAAc,CAAC,GAAG;aACnC;YACD,SAAS,EAAE;gBACT,aAAa,EAAE,CAAC,eAAe,CAAC;aACjC;SACF,CAAC,CAAA;IACJ,CAAC;;AAlaH,kBAmaC","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"]}