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/.jsii +202 -3
- package/API.md +285 -0
- package/biome.jsonc +66 -0
- package/lib/index.d.ts +2 -0
- package/lib/index.js +3 -1
- package/lib/lib/aws.d.ts +16 -1
- package/lib/lib/aws.js +278 -279
- package/lib/lib/azure.d.ts +1 -1
- package/lib/lib/azure.js +105 -130
- package/lib/lib/gcp.d.ts +1 -1
- package/lib/lib/gcp.js +49 -47
- package/lib/lib/variables.d.ts +3 -2
- package/lib/lib/variables.js +12 -4
- package/lib/tfModules.js +40 -10
- package/modules/azure-container-apps/README.md +3 -2
- package/modules/azure-container-apps/cdk.tf.json +15 -11
- package/modules/elastic-container-service/README.md +26 -25
- package/modules/elastic-container-service/cdk.tf.json +65 -75
- package/modules/google-cloud-run/README.md +19 -18
- package/modules/google-cloud-run/cdk.tf.json +12 -8
- package/package.json +16 -69
- package/renovate.json5 +42 -0
- package/vitest.config.ts +28 -0
- package/lib/__tests__/index-test.d.ts +0 -1
- package/lib/__tests__/index-test.js +0 -26
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:
|
|
31
|
+
name: props.clusterName,
|
|
30
32
|
});
|
|
31
|
-
const
|
|
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
|
-
|
|
34
|
-
|
|
41
|
+
Version: '2012-10-17',
|
|
42
|
+
Statement: [
|
|
35
43
|
{
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
44
|
+
Effect: 'Allow',
|
|
45
|
+
Principal: {
|
|
46
|
+
Service: 'ecs-tasks.amazonaws.com',
|
|
39
47
|
},
|
|
40
|
-
|
|
41
|
-
}
|
|
42
|
-
]
|
|
43
|
-
})
|
|
48
|
+
Action: 'sts:AssumeRole',
|
|
49
|
+
},
|
|
50
|
+
],
|
|
51
|
+
}),
|
|
44
52
|
});
|
|
45
|
-
const
|
|
53
|
+
const runnerRole = new iam_role_1.IamRole(this, 'RunnerRole', {
|
|
46
54
|
assumeRolePolicy: cdktf_1.Fn.jsonencode({
|
|
47
|
-
|
|
48
|
-
|
|
55
|
+
Version: '2012-10-17',
|
|
56
|
+
Statement: [
|
|
49
57
|
{
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
58
|
+
Effect: 'Allow',
|
|
59
|
+
Principal: {
|
|
60
|
+
Service: 'ecs-tasks.amazonaws.com',
|
|
53
61
|
},
|
|
54
|
-
|
|
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
|
-
|
|
62
|
-
|
|
69
|
+
Version: '2012-10-17',
|
|
70
|
+
Statement: [
|
|
63
71
|
{
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
72
|
+
Effect: 'Allow',
|
|
73
|
+
Principal: {
|
|
74
|
+
Service: 'ecs-tasks.amazonaws.com',
|
|
67
75
|
},
|
|
68
|
-
|
|
69
|
-
}
|
|
70
|
-
]
|
|
76
|
+
Action: 'sts:AssumeRole',
|
|
77
|
+
},
|
|
78
|
+
],
|
|
71
79
|
}),
|
|
72
|
-
|
|
73
|
-
|
|
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
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
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:
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
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(
|
|
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
|
-
|
|
196
|
-
|
|
197
|
-
|
|
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
|
-
|
|
335
|
-
|
|
336
|
+
Version: '2012-10-17',
|
|
337
|
+
Statement: [
|
|
336
338
|
{
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
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
|
-
|
|
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
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
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
|
-
|
|
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"]}
|