@cloudsnorkel/cdk-github-runners 0.5.8 → 0.6.1
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 +2847 -1100
- package/API.md +1290 -144
- package/README.md +11 -11
- package/lib/index.d.ts +6 -2
- package/lib/index.js +9 -2
- package/lib/lambdas/delete-ami/index.js +130 -0
- package/lib/lambdas/status/index.js +11 -1
- package/lib/lambdas/update-lambda/index.js +165 -107
- package/lib/providers/codebuild.d.ts +6 -4
- package/lib/providers/codebuild.js +20 -3
- package/lib/providers/common.d.ts +137 -9
- package/lib/providers/common.js +53 -4
- package/lib/providers/ec2.d.ts +106 -0
- package/lib/providers/ec2.js +256 -0
- package/lib/providers/fargate.d.ts +5 -3
- package/lib/providers/fargate.js +26 -5
- package/lib/providers/image-builders/ami.d.ts +131 -0
- package/lib/providers/image-builders/ami.js +274 -0
- package/lib/providers/image-builders/codebuild.js +3 -2
- package/lib/providers/image-builders/common.d.ts +196 -0
- package/lib/providers/image-builders/common.js +288 -0
- package/lib/providers/image-builders/container.d.ts +6 -100
- package/lib/providers/image-builders/container.js +41 -304
- package/lib/providers/image-builders/linux-components.d.ts +15 -0
- package/lib/providers/image-builders/linux-components.js +156 -0
- package/lib/providers/image-builders/static.js +3 -2
- package/lib/providers/image-builders/windows-components.d.ts +14 -0
- package/lib/providers/image-builders/windows-components.js +119 -0
- package/lib/providers/lambda.d.ts +5 -3
- package/lib/providers/lambda.js +20 -3
- package/lib/runner.js +8 -18
- package/lib/secrets.js +1 -1
- package/package.json +9 -9
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import * as cdk from 'aws-cdk-lib';
|
|
2
|
+
import { aws_ec2 as ec2, aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
|
|
3
|
+
import { Construct } from 'constructs';
|
|
4
|
+
import { BaseProvider, IAmiBuilder, IRunnerProvider, IRunnerProviderStatus, RunnerProviderProps, RunnerRuntimeParameters } from './common';
|
|
5
|
+
/**
|
|
6
|
+
* Properties for {@link Ec2Runner} construct.
|
|
7
|
+
*/
|
|
8
|
+
export interface Ec2RunnerProps extends RunnerProviderProps {
|
|
9
|
+
/**
|
|
10
|
+
* AMI builder that creates AMIs with GitHub runner pre-configured. On Linux, a user named `runner` is expected to exist with access to Docker.
|
|
11
|
+
*
|
|
12
|
+
* @default AMI builder for Ubuntu Linux
|
|
13
|
+
*/
|
|
14
|
+
readonly amiBuilder?: IAmiBuilder;
|
|
15
|
+
/**
|
|
16
|
+
* GitHub Actions labels used for this provider.
|
|
17
|
+
*
|
|
18
|
+
* These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for
|
|
19
|
+
* based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the
|
|
20
|
+
* job's labels, this provider will be chosen and spawn a new runner.
|
|
21
|
+
*
|
|
22
|
+
* @default ['ec2']
|
|
23
|
+
*/
|
|
24
|
+
readonly labels?: string[];
|
|
25
|
+
/**
|
|
26
|
+
* Instance type for launched runner instances.
|
|
27
|
+
*
|
|
28
|
+
* @default m5.large
|
|
29
|
+
*/
|
|
30
|
+
readonly instanceType?: ec2.InstanceType;
|
|
31
|
+
/**
|
|
32
|
+
* Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.
|
|
33
|
+
*
|
|
34
|
+
* @default 30GB
|
|
35
|
+
*/
|
|
36
|
+
readonly storageSize?: cdk.Size;
|
|
37
|
+
/**
|
|
38
|
+
* Security Group to assign to launched runner instances.
|
|
39
|
+
*
|
|
40
|
+
* @default account's default security group
|
|
41
|
+
*/
|
|
42
|
+
readonly securityGroup?: ec2.ISecurityGroup;
|
|
43
|
+
/**
|
|
44
|
+
* Subnet where the runner instances will be launched.
|
|
45
|
+
*
|
|
46
|
+
* @default default subnet of account's default VPC
|
|
47
|
+
*/
|
|
48
|
+
readonly subnet?: ec2.ISubnet;
|
|
49
|
+
/**
|
|
50
|
+
* Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.
|
|
51
|
+
*
|
|
52
|
+
* @default false
|
|
53
|
+
*/
|
|
54
|
+
readonly spot?: boolean;
|
|
55
|
+
/**
|
|
56
|
+
* Set a maximum price for spot instances.
|
|
57
|
+
*
|
|
58
|
+
* @default no max price (you will pay current spot price)
|
|
59
|
+
*/
|
|
60
|
+
readonly spotMaxPrice?: string;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* GitHub Actions runner provider using EC2 to execute jobs.
|
|
64
|
+
*
|
|
65
|
+
* This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
|
|
66
|
+
*/
|
|
67
|
+
export declare class Ec2Runner extends BaseProvider implements IRunnerProvider {
|
|
68
|
+
/**
|
|
69
|
+
* Labels associated with this provider.
|
|
70
|
+
*/
|
|
71
|
+
readonly labels: string[];
|
|
72
|
+
/**
|
|
73
|
+
* VPC subnet used for hosting launched instances.
|
|
74
|
+
*/
|
|
75
|
+
readonly subnet?: ec2.ISubnet;
|
|
76
|
+
/**
|
|
77
|
+
* Security group attached to launched instances.
|
|
78
|
+
*/
|
|
79
|
+
readonly securityGroup?: ec2.ISecurityGroup;
|
|
80
|
+
/**
|
|
81
|
+
* Grant principal used to add permissions to the runner role.
|
|
82
|
+
*/
|
|
83
|
+
readonly grantPrincipal: iam.IPrincipal;
|
|
84
|
+
private readonly ami;
|
|
85
|
+
private readonly logGroup;
|
|
86
|
+
private readonly role;
|
|
87
|
+
private readonly instanceType;
|
|
88
|
+
private readonly storageSize;
|
|
89
|
+
private readonly spot;
|
|
90
|
+
private readonly spotMaxPrice;
|
|
91
|
+
constructor(scope: Construct, id: string, props: Ec2RunnerProps);
|
|
92
|
+
/**
|
|
93
|
+
* Generate step function task(s) to start a new runner.
|
|
94
|
+
*
|
|
95
|
+
* Called by GithubRunners and shouldn't be called manually.
|
|
96
|
+
*
|
|
97
|
+
* @param parameters workflow job details
|
|
98
|
+
*/
|
|
99
|
+
getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;
|
|
100
|
+
grantStateMachine(stateMachineRole: iam.IGrantable): void;
|
|
101
|
+
status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus;
|
|
102
|
+
/**
|
|
103
|
+
* The network connections associated with this resource.
|
|
104
|
+
*/
|
|
105
|
+
get connections(): ec2.Connections;
|
|
106
|
+
}
|
|
@@ -0,0 +1,256 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.Ec2Runner = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk = require("aws-cdk-lib");
|
|
7
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
9
|
+
const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
|
|
10
|
+
const common_1 = require("./common");
|
|
11
|
+
const ami_1 = require("./image-builders/ami");
|
|
12
|
+
// this script is specifically made so `poweroff` is absolutely always called
|
|
13
|
+
// each `{}` is a variable coming from `params` below
|
|
14
|
+
const linuxUserDataTemplate = `#!/bin/bash -x
|
|
15
|
+
TASK_TOKEN="{}"
|
|
16
|
+
heartbeat () {
|
|
17
|
+
while true; do
|
|
18
|
+
aws stepfunctions send-task-heartbeat --task-token "$TASK_TOKEN"
|
|
19
|
+
sleep 60
|
|
20
|
+
done
|
|
21
|
+
}
|
|
22
|
+
setup_logs () {
|
|
23
|
+
cat <<EOF > /tmp/log.conf || exit 1
|
|
24
|
+
{
|
|
25
|
+
"logs": {
|
|
26
|
+
"log_stream_name": "unknown",
|
|
27
|
+
"logs_collected": {
|
|
28
|
+
"files": {
|
|
29
|
+
"collect_list": [
|
|
30
|
+
{
|
|
31
|
+
"file_path": "/var/log/runner.log",
|
|
32
|
+
"log_group_name": "{}",
|
|
33
|
+
"log_stream_name": "{}",
|
|
34
|
+
"timezone": "UTC"
|
|
35
|
+
}
|
|
36
|
+
]
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
EOF
|
|
42
|
+
/opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2
|
|
43
|
+
}
|
|
44
|
+
action () {
|
|
45
|
+
sudo -Hu runner /home/runner/config.sh --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" {} --name "{}" || exit 1
|
|
46
|
+
sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2
|
|
47
|
+
}
|
|
48
|
+
heartbeat &
|
|
49
|
+
if setup_logs && action | tee /var/log/runner.log 2>&1; then
|
|
50
|
+
aws stepfunctions send-task-success --task-token "$TASK_TOKEN" --task-output '{"ok": true}'
|
|
51
|
+
else
|
|
52
|
+
aws stepfunctions send-task-failure --task-token "$TASK_TOKEN"
|
|
53
|
+
fi
|
|
54
|
+
poweroff
|
|
55
|
+
`.replace(/{/g, '\\{').replace(/}/g, '\\}').replace(/\\{\\}/g, '{}');
|
|
56
|
+
// this script is specifically made so `poweroff` is absolutely always called
|
|
57
|
+
// each `{}` is a variable coming from `params` below and their order should match the linux script
|
|
58
|
+
const windowsUserDataTemplate = `<powershell>
|
|
59
|
+
$TASK_TOKEN = "{}"
|
|
60
|
+
Start-Job -ScriptBlock {
|
|
61
|
+
while (1) {
|
|
62
|
+
aws stepfunctions send-task-heartbeat --task-token "$using:TASK_TOKEN"
|
|
63
|
+
sleep 60
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
function setup_logs () {
|
|
67
|
+
echo '{
|
|
68
|
+
"logs": {
|
|
69
|
+
"log_stream_name": "unknown",
|
|
70
|
+
"logs_collected": {
|
|
71
|
+
"files": {
|
|
72
|
+
"collect_list": [
|
|
73
|
+
{
|
|
74
|
+
"file_path": "/actions/runner.log",
|
|
75
|
+
"log_group_name": "{}",
|
|
76
|
+
"log_stream_name": "{}",
|
|
77
|
+
"timezone": "UTC"
|
|
78
|
+
}
|
|
79
|
+
]
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
}' | Out-File -Encoding ASCII $Env:TEMP/log.conf
|
|
84
|
+
& "C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf
|
|
85
|
+
}
|
|
86
|
+
function action () {
|
|
87
|
+
cd /actions
|
|
88
|
+
./config.cmd --unattended --url "https://{}/{}/{}" --token "{}" --ephemeral --work _work --labels "{}" {} --name "{}" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log
|
|
89
|
+
if ($LASTEXITCODE -ne 0) { return 1 }
|
|
90
|
+
./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log
|
|
91
|
+
if ($LASTEXITCODE -ne 0) { return 2 }
|
|
92
|
+
return 0
|
|
93
|
+
}
|
|
94
|
+
setup_logs
|
|
95
|
+
$r = action
|
|
96
|
+
if ($r -eq 0) {
|
|
97
|
+
aws stepfunctions send-task-success --task-token "$TASK_TOKEN" --task-output '{ }'
|
|
98
|
+
} else {
|
|
99
|
+
aws stepfunctions send-task-failure --task-token "$TASK_TOKEN"
|
|
100
|
+
}
|
|
101
|
+
Stop-Computer -ComputerName localhost -Force
|
|
102
|
+
</powershell>
|
|
103
|
+
`.replace(/{/g, '\\{').replace(/}/g, '\\}').replace(/\\{\\}/g, '{}');
|
|
104
|
+
/**
|
|
105
|
+
* GitHub Actions runner provider using EC2 to execute jobs.
|
|
106
|
+
*
|
|
107
|
+
* This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.
|
|
108
|
+
*/
|
|
109
|
+
class Ec2Runner extends common_1.BaseProvider {
|
|
110
|
+
constructor(scope, id, props) {
|
|
111
|
+
super(scope, id);
|
|
112
|
+
this.labels = props.labels ?? ['ec2'];
|
|
113
|
+
this.securityGroup = props.securityGroup;
|
|
114
|
+
this.subnet = props.subnet;
|
|
115
|
+
this.instanceType = props.instanceType ?? aws_cdk_lib_1.aws_ec2.InstanceType.of(aws_cdk_lib_1.aws_ec2.InstanceClass.M5, aws_cdk_lib_1.aws_ec2.InstanceSize.LARGE);
|
|
116
|
+
this.storageSize = props.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows
|
|
117
|
+
this.spot = props.spot ?? false;
|
|
118
|
+
this.spotMaxPrice = props.spotMaxPrice;
|
|
119
|
+
const amiBuilder = props.amiBuilder ?? new ami_1.AmiBuilder(this, 'Image Builder');
|
|
120
|
+
this.ami = amiBuilder.bind();
|
|
121
|
+
if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {
|
|
122
|
+
throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);
|
|
123
|
+
}
|
|
124
|
+
this.grantPrincipal = this.role = new aws_cdk_lib_1.aws_iam.Role(this, 'Role', {
|
|
125
|
+
assumedBy: new aws_cdk_lib_1.aws_iam.ServicePrincipal('ec2.amazonaws.com'),
|
|
126
|
+
managedPolicies: [
|
|
127
|
+
aws_cdk_lib_1.aws_iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),
|
|
128
|
+
],
|
|
129
|
+
});
|
|
130
|
+
this.grantPrincipal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
131
|
+
actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],
|
|
132
|
+
resources: ['*'],
|
|
133
|
+
}));
|
|
134
|
+
this.logGroup = new aws_cdk_lib_1.aws_logs.LogGroup(this, 'Logs', {
|
|
135
|
+
retention: props.logRetention ?? aws_logs_1.RetentionDays.ONE_MONTH,
|
|
136
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
137
|
+
});
|
|
138
|
+
this.logGroup.grantWrite(this);
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Generate step function task(s) to start a new runner.
|
|
142
|
+
*
|
|
143
|
+
* Called by GithubRunners and shouldn't be called manually.
|
|
144
|
+
*
|
|
145
|
+
* @param parameters workflow job details
|
|
146
|
+
*/
|
|
147
|
+
getStepFunctionTask(parameters) {
|
|
148
|
+
// we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax
|
|
149
|
+
const params = [
|
|
150
|
+
aws_cdk_lib_1.aws_stepfunctions.JsonPath.taskToken,
|
|
151
|
+
this.logGroup.logGroupName,
|
|
152
|
+
parameters.runnerNamePath,
|
|
153
|
+
parameters.githubDomainPath,
|
|
154
|
+
parameters.ownerPath,
|
|
155
|
+
parameters.repoPath,
|
|
156
|
+
parameters.runnerTokenPath,
|
|
157
|
+
this.labels.join(','),
|
|
158
|
+
this.ami.runnerVersion.is(common_1.RunnerVersion.latest()) ? '' : '--disableupdate',
|
|
159
|
+
parameters.runnerNamePath,
|
|
160
|
+
];
|
|
161
|
+
const passUserData = new aws_cdk_lib_1.aws_stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {
|
|
162
|
+
parameters: {
|
|
163
|
+
userdataTemplate: this.ami.os.is(common_1.Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,
|
|
164
|
+
},
|
|
165
|
+
resultPath: aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.ec2'),
|
|
166
|
+
});
|
|
167
|
+
// we can't use fleets because they don't let us override user data, security groups or even disk size
|
|
168
|
+
// we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated
|
|
169
|
+
const run = new aws_cdk_lib_1.aws_stepfunctions_tasks.CallAwsService(this, this.labels.join(', '), {
|
|
170
|
+
integrationPattern: aws_stepfunctions_1.IntegrationPattern.WAIT_FOR_TASK_TOKEN,
|
|
171
|
+
service: 'ec2',
|
|
172
|
+
action: 'runInstances',
|
|
173
|
+
heartbeat: aws_cdk_lib_1.Duration.minutes(5),
|
|
174
|
+
parameters: {
|
|
175
|
+
LaunchTemplate: {
|
|
176
|
+
LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,
|
|
177
|
+
},
|
|
178
|
+
MinCount: 1,
|
|
179
|
+
MaxCount: 1,
|
|
180
|
+
InstanceType: this.instanceType.toString(),
|
|
181
|
+
UserData: aws_cdk_lib_1.aws_stepfunctions.JsonPath.base64Encode(aws_cdk_lib_1.aws_stepfunctions.JsonPath.format(aws_cdk_lib_1.aws_stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'), ...params)),
|
|
182
|
+
InstanceInitiatedShutdownBehavior: aws_cdk_lib_1.aws_ec2.InstanceInitiatedShutdownBehavior.TERMINATE,
|
|
183
|
+
IamInstanceProfile: {
|
|
184
|
+
Arn: new aws_cdk_lib_1.aws_iam.CfnInstanceProfile(this, 'Instance Profile', {
|
|
185
|
+
roles: [this.role.roleName],
|
|
186
|
+
}).attrArn,
|
|
187
|
+
},
|
|
188
|
+
MetadataOptions: {
|
|
189
|
+
HttpTokens: 'required',
|
|
190
|
+
},
|
|
191
|
+
SecurityGroupIds: this.securityGroup ? [this.securityGroup.securityGroupId] : undefined,
|
|
192
|
+
SubnetId: this.subnet?.subnetId,
|
|
193
|
+
BlockDeviceMappings: [{
|
|
194
|
+
DeviceName: '/dev/sda1',
|
|
195
|
+
Ebs: {
|
|
196
|
+
DeleteOnTermination: true,
|
|
197
|
+
VolumeSize: this.storageSize.toGibibytes(),
|
|
198
|
+
},
|
|
199
|
+
}],
|
|
200
|
+
InstanceMarketOptions: this.spot ? {
|
|
201
|
+
MarketType: 'spot',
|
|
202
|
+
SpotOptions: {
|
|
203
|
+
MaxPrice: this.spotMaxPrice,
|
|
204
|
+
SpotInstanceType: 'one-time',
|
|
205
|
+
},
|
|
206
|
+
} : undefined,
|
|
207
|
+
},
|
|
208
|
+
iamResources: ['*'],
|
|
209
|
+
});
|
|
210
|
+
return passUserData.next(run);
|
|
211
|
+
}
|
|
212
|
+
grantStateMachine(stateMachineRole) {
|
|
213
|
+
stateMachineRole.grantPrincipal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
214
|
+
actions: ['iam:PassRole'],
|
|
215
|
+
resources: [this.role.roleArn],
|
|
216
|
+
conditions: {
|
|
217
|
+
StringEquals: {
|
|
218
|
+
'iam:PassedToService': 'ec2.amazonaws.com',
|
|
219
|
+
},
|
|
220
|
+
},
|
|
221
|
+
}));
|
|
222
|
+
stateMachineRole.grantPrincipal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
223
|
+
actions: ['ec2:createTags'],
|
|
224
|
+
resources: [aws_cdk_lib_1.Stack.of(this).formatArn({
|
|
225
|
+
service: 'ec2',
|
|
226
|
+
resource: '*',
|
|
227
|
+
})],
|
|
228
|
+
}));
|
|
229
|
+
}
|
|
230
|
+
status(statusFunctionRole) {
|
|
231
|
+
statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new aws_cdk_lib_1.aws_iam.PolicyStatement({
|
|
232
|
+
actions: ['ec2:DescribeLaunchTemplateVersions'],
|
|
233
|
+
resources: ['*'],
|
|
234
|
+
}));
|
|
235
|
+
return {
|
|
236
|
+
type: this.constructor.name,
|
|
237
|
+
labels: this.labels,
|
|
238
|
+
securityGroup: this.securityGroup?.securityGroupId,
|
|
239
|
+
roleArn: this.role.roleArn,
|
|
240
|
+
ami: {
|
|
241
|
+
launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',
|
|
242
|
+
amiBuilderLogGroup: this.ami.logGroup?.logGroupName,
|
|
243
|
+
},
|
|
244
|
+
};
|
|
245
|
+
}
|
|
246
|
+
/**
|
|
247
|
+
* The network connections associated with this resource.
|
|
248
|
+
*/
|
|
249
|
+
get connections() {
|
|
250
|
+
return this.securityGroup?.connections ?? new aws_cdk_lib_1.aws_ec2.Connections();
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
exports.Ec2Runner = Ec2Runner;
|
|
254
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
255
|
+
Ec2Runner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.Ec2Runner", version: "0.6.1" };
|
|
256
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ec2.js","sourceRoot":"","sources":["../../src/providers/ec2.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport {\n  aws_ec2 as ec2,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  Duration,\n  RemovalPolicy,\n  Stack,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  BaseProvider,\n  IAmiBuilder,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerAmi,\n  RunnerProviderProps,\n  RunnerRuntimeParameters,\n  RunnerVersion,\n} from './common';\nimport { AmiBuilder } from './image-builders/ami';\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below\nconst linuxUserDataTemplate = `#!/bin/bash -x\nTASK_TOKEN=\"{}\"\nheartbeat () {\n  while true; do\n    aws stepfunctions send-task-heartbeat --task-token \"$TASK_TOKEN\"\n    sleep 60\n  done\n}\nsetup_logs () {\n  cat <<EOF > /tmp/log.conf || exit 1\n  {\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n          \"collect_list\": [\n            {\n              \"file_path\": \"/var/log/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }\nEOF\n  /opt/aws/amazon-cloudwatch-agent/bin/amazon-cloudwatch-agent-ctl -a fetch-config -m ec2 -s -c file:/tmp/log.conf || exit 2\n}\naction () {\n  sudo -Hu runner /home/runner/config.sh --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" {} --name \"{}\" || exit 1\n  sudo --preserve-env=AWS_REGION -Hu runner /home/runner/run.sh || exit 2\n}\nheartbeat &\nif setup_logs && action | tee /var/log/runner.log 2>&1; then\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{\"ok\": true}'\nelse\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\nfi\npoweroff\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n// this script is specifically made so `poweroff` is absolutely always called\n// each `{}` is a variable coming from `params` below and their order should match the linux script\nconst windowsUserDataTemplate = `<powershell>\n$TASK_TOKEN = \"{}\"\nStart-Job -ScriptBlock {\n  while (1) {\n    aws stepfunctions send-task-heartbeat --task-token \"$using:TASK_TOKEN\"\n    sleep 60\n  }\n}\nfunction setup_logs () {\n  echo '{\n    \"logs\": {\n      \"log_stream_name\": \"unknown\",\n      \"logs_collected\": {\n        \"files\": {\n         \"collect_list\": [\n            {\n              \"file_path\": \"/actions/runner.log\",\n              \"log_group_name\": \"{}\",\n              \"log_stream_name\": \"{}\",\n              \"timezone\": \"UTC\"\n            }\n          ]\n        }\n      }\n    }\n  }' | Out-File -Encoding ASCII $Env:TEMP/log.conf\n  & \"C:/Program Files/Amazon/AmazonCloudWatchAgent/amazon-cloudwatch-agent-ctl.ps1\" -a fetch-config -m ec2 -s -c file:$Env:TEMP/log.conf\n}\nfunction action () {\n  cd /actions\n  ./config.cmd --unattended --url \"https://{}/{}/{}\" --token \"{}\" --ephemeral --work _work --labels \"{}\" {} --name \"{}\" 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 1 }\n  ./run.cmd 2>&1 | Out-File -Encoding ASCII -Append /actions/runner.log\n  if ($LASTEXITCODE -ne 0) { return 2 }\n  return 0\n}\nsetup_logs\n$r = action\nif ($r -eq 0) {\n  aws stepfunctions send-task-success --task-token \"$TASK_TOKEN\" --task-output '{ }'\n} else {\n  aws stepfunctions send-task-failure --task-token \"$TASK_TOKEN\"\n}\nStop-Computer -ComputerName localhost -Force\n</powershell>\n`.replace(/{/g, '\\\\{').replace(/}/g, '\\\\}').replace(/\\\\{\\\\}/g, '{}');\n\n\n/**\n * Properties for {@link Ec2Runner} construct.\n */\nexport interface Ec2RunnerProps extends RunnerProviderProps {\n  /**\n   * AMI builder that creates AMIs with GitHub runner pre-configured. On Linux, a user named `runner` is expected to exist with access to Docker.\n   *\n   * @default AMI builder for Ubuntu Linux\n   */\n  readonly amiBuilder?: IAmiBuilder;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['ec2']\n   */\n  readonly labels?: string[];\n\n  /**\n   * Instance type for launched runner instances.\n   *\n   * @default m5.large\n   */\n  readonly instanceType?: ec2.InstanceType;\n\n  /**\n   * Size of volume available for launched runner instances. This modifies the boot volume size and doesn't add any additional volumes.\n   *\n   * @default 30GB\n   */\n  readonly storageSize?: cdk.Size;\n\n  /**\n   * Security Group to assign to launched runner instances.\n   *\n   * @default account's default security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Subnet where the runner instances will be launched.\n   *\n   * @default default subnet of account's default VPC\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * Use spot instances to save money. Spot instances are cheaper but not always available and can be stopped prematurely.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n\n  /**\n   * Set a maximum price for spot instances.\n   *\n   * @default no max price (you will pay current spot price)\n   */\n  readonly spotMaxPrice?: string;\n}\n\n/**\n * GitHub Actions runner provider using EC2 to execute jobs.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class Ec2Runner extends BaseProvider implements IRunnerProvider {\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC subnet used for hosting launched instances.\n   */\n  readonly subnet?: ec2.ISubnet;\n\n  /**\n   * Security group attached to launched instances.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  private readonly ami: RunnerAmi;\n  private readonly logGroup: logs.LogGroup;\n  private readonly role: iam.Role;\n  private readonly instanceType: ec2.InstanceType;\n  private readonly storageSize: cdk.Size;\n  private readonly spot: boolean;\n  private readonly spotMaxPrice: string | undefined;\n\n  constructor(scope: Construct, id: string, props: Ec2RunnerProps) {\n    super(scope, id);\n\n    this.labels = props.labels ?? ['ec2'];\n    this.securityGroup = props.securityGroup;\n    this.subnet = props.subnet;\n    this.instanceType = props.instanceType ?? ec2.InstanceType.of(ec2.InstanceClass.M5, ec2.InstanceSize.LARGE);\n    this.storageSize = props.storageSize ?? cdk.Size.gibibytes(30); // 30 is the minimum for Windows\n    this.spot = props.spot ?? false;\n    this.spotMaxPrice = props.spotMaxPrice;\n\n    const amiBuilder = props.amiBuilder ?? new AmiBuilder(this, 'Image Builder');\n    this.ami = amiBuilder.bind();\n\n    if (!this.ami.architecture.instanceTypeMatch(this.instanceType)) {\n      throw new Error(`AMI architecture (${this.ami.architecture.name}) doesn't match runner instance type (${this.instanceType} / ${this.instanceType.architecture})`);\n    }\n\n    this.grantPrincipal = this.role = new iam.Role(this, 'Role', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonSSMManagedInstanceCore'),\n      ],\n    });\n    this.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['states:SendTaskFailure', 'states:SendTaskSuccess', 'states:SendTaskHeartbeat'],\n      resources: ['*'], // no support for stateMachine.stateMachineArn :(\n    }));\n\n    this.logGroup = new logs.LogGroup(\n      this,\n      'Logs',\n      {\n        retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n        removalPolicy: RemovalPolicy.DESTROY,\n      },\n    );\n    this.logGroup.grantWrite(this);\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    // we need to build user data in two steps because passing the template as the first parameter to stepfunctions.JsonPath.format fails on syntax\n\n    const params = [\n      stepfunctions.JsonPath.taskToken,\n      this.logGroup.logGroupName,\n      parameters.runnerNamePath,\n      parameters.githubDomainPath,\n      parameters.ownerPath,\n      parameters.repoPath,\n      parameters.runnerTokenPath,\n      this.labels.join(','),\n      this.ami.runnerVersion.is(RunnerVersion.latest()) ? '' : '--disableupdate',\n      parameters.runnerNamePath,\n    ];\n\n    const passUserData = new stepfunctions.Pass(this, `${this.labels.join(', ')} data`, {\n      parameters: {\n        userdataTemplate: this.ami.os.is(Os.WINDOWS) ? windowsUserDataTemplate : linuxUserDataTemplate,\n      },\n      resultPath: stepfunctions.JsonPath.stringAt('$.ec2'),\n    });\n\n    // we can't use fleets because they don't let us override user data, security groups or even disk size\n    // we can't use requestSpotInstances because it doesn't support launch templates, and it's deprecated\n\n    const run = new stepfunctions_tasks.CallAwsService(this, this.labels.join(', '), {\n      integrationPattern: IntegrationPattern.WAIT_FOR_TASK_TOKEN,\n      service: 'ec2',\n      action: 'runInstances',\n      heartbeat: Duration.minutes(5),\n      parameters: {\n        LaunchTemplate: {\n          LaunchTemplateId: this.ami.launchTemplate.launchTemplateId,\n        },\n        MinCount: 1,\n        MaxCount: 1,\n        InstanceType: this.instanceType.toString(),\n        UserData: stepfunctions.JsonPath.base64Encode(\n          stepfunctions.JsonPath.format(\n            stepfunctions.JsonPath.stringAt('$.ec2.userdataTemplate'),\n            ...params,\n          ),\n        ),\n        InstanceInitiatedShutdownBehavior: ec2.InstanceInitiatedShutdownBehavior.TERMINATE,\n        IamInstanceProfile: {\n          Arn: new iam.CfnInstanceProfile(this, 'Instance Profile', {\n            roles: [this.role.roleName],\n          }).attrArn,\n        },\n        MetadataOptions: {\n          HttpTokens: 'required',\n        },\n        SecurityGroupIds: this.securityGroup ? [this.securityGroup.securityGroupId] : undefined,\n        SubnetId: this.subnet?.subnetId,\n        BlockDeviceMappings: [{\n          DeviceName: '/dev/sda1',\n          Ebs: {\n            DeleteOnTermination: true,\n            VolumeSize: this.storageSize.toGibibytes(),\n          },\n        }],\n        InstanceMarketOptions: this.spot ? {\n          MarketType: 'spot',\n          SpotOptions: {\n            MaxPrice: this.spotMaxPrice,\n            SpotInstanceType: 'one-time',\n          },\n        } : undefined,\n      },\n      iamResources: ['*'],\n    });\n\n    return passUserData.next(run);\n  }\n\n  grantStateMachine(stateMachineRole: iam.IGrantable) {\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['iam:PassRole'],\n      resources: [this.role.roleArn],\n      conditions: {\n        StringEquals: {\n          'iam:PassedToService': 'ec2.amazonaws.com',\n        },\n      },\n    }));\n\n    stateMachineRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:createTags'],\n      resources: [Stack.of(this).formatArn({\n        service: 'ec2',\n        resource: '*',\n      })],\n    }));\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    statusFunctionRole.grantPrincipal.addToPrincipalPolicy(new iam.PolicyStatement({\n      actions: ['ec2:DescribeLaunchTemplateVersions'],\n      resources: ['*'],\n    }));\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      securityGroup: this.securityGroup?.securityGroupId,\n      roleArn: this.role.roleArn,\n      ami: {\n        launchTemplate: this.ami.launchTemplate.launchTemplateId || 'unknown',\n        amiBuilderLogGroup: this.ami.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  /**\n   * The network connections associated with this resource.\n   */\n  public get connections(): ec2.Connections {\n    return this.securityGroup?.connections ?? new ec2.Connections();\n  }\n}\n"]}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { aws_ec2 as ec2, aws_ecs as ecs, aws_iam as iam, aws_stepfunctions as stepfunctions } from 'aws-cdk-lib';
|
|
2
2
|
import { Construct } from 'constructs';
|
|
3
|
-
import { BaseProvider, IImageBuilder, IRunnerProvider, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';
|
|
3
|
+
import { BaseProvider, IImageBuilder, IRunnerProvider, IRunnerProviderStatus, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';
|
|
4
4
|
/**
|
|
5
5
|
* Properties for FargateRunner.
|
|
6
6
|
*/
|
|
@@ -114,7 +114,7 @@ export interface FargateRunnerProps extends RunnerProviderProps {
|
|
|
114
114
|
readonly spot?: boolean;
|
|
115
115
|
}
|
|
116
116
|
/**
|
|
117
|
-
* GitHub Actions runner provider using Fargate to execute
|
|
117
|
+
* GitHub Actions runner provider using Fargate to execute jobs.
|
|
118
118
|
*
|
|
119
119
|
* Creates a task definition with a single container that gets started for each job.
|
|
120
120
|
*
|
|
@@ -182,7 +182,7 @@ export declare class FargateRunner extends BaseProvider implements IRunnerProvid
|
|
|
182
182
|
*/
|
|
183
183
|
readonly spot: boolean;
|
|
184
184
|
/**
|
|
185
|
-
* Docker image
|
|
185
|
+
* Docker image loaded with GitHub Actions Runner and its prerequisites. The image is built by an image builder and is specific to Fargate tasks.
|
|
186
186
|
*/
|
|
187
187
|
readonly image: RunnerImage;
|
|
188
188
|
constructor(scope: Construct, id: string, props: FargateRunnerProps);
|
|
@@ -194,5 +194,7 @@ export declare class FargateRunner extends BaseProvider implements IRunnerProvid
|
|
|
194
194
|
* @param parameters workflow job details
|
|
195
195
|
*/
|
|
196
196
|
getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable;
|
|
197
|
+
grantStateMachine(_: iam.IGrantable): void;
|
|
198
|
+
status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus;
|
|
197
199
|
private runCommand;
|
|
198
200
|
}
|
package/lib/providers/fargate.js
CHANGED
|
@@ -36,7 +36,7 @@ class EcsFargateLaunchTarget {
|
|
|
36
36
|
}
|
|
37
37
|
}
|
|
38
38
|
/**
|
|
39
|
-
* GitHub Actions runner provider using Fargate to execute
|
|
39
|
+
* GitHub Actions runner provider using Fargate to execute jobs.
|
|
40
40
|
*
|
|
41
41
|
* Creates a task definition with a single container that gets started for each job.
|
|
42
42
|
*
|
|
@@ -157,17 +157,38 @@ class FargateRunner extends common_1.BaseProvider {
|
|
|
157
157
|
],
|
|
158
158
|
});
|
|
159
159
|
}
|
|
160
|
+
grantStateMachine(_) {
|
|
161
|
+
}
|
|
162
|
+
status(statusFunctionRole) {
|
|
163
|
+
this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');
|
|
164
|
+
return {
|
|
165
|
+
type: this.constructor.name,
|
|
166
|
+
labels: this.labels,
|
|
167
|
+
vpcArn: this.vpc?.vpcArn,
|
|
168
|
+
securityGroup: this.securityGroup?.securityGroupId,
|
|
169
|
+
roleArn: this.task.taskRole.roleArn,
|
|
170
|
+
image: {
|
|
171
|
+
imageRepository: this.image.imageRepository.repositoryUri,
|
|
172
|
+
imageTag: this.image.imageTag,
|
|
173
|
+
imageBuilderLogGroup: this.image.logGroup?.logGroupName,
|
|
174
|
+
},
|
|
175
|
+
};
|
|
176
|
+
}
|
|
160
177
|
runCommand() {
|
|
178
|
+
let runnerFlags = '';
|
|
179
|
+
if (this.image.runnerVersion.is(common_1.RunnerVersion.latest())) {
|
|
180
|
+
runnerFlags = '--disableupdate';
|
|
181
|
+
}
|
|
161
182
|
if (this.image.os.is(common_1.Os.LINUX)) {
|
|
162
183
|
return [
|
|
163
184
|
'sh', '-c',
|
|
164
|
-
|
|
185
|
+
`./config.sh --unattended --url "https://$GITHUB_DOMAIN/$OWNER/$REPO" --token "$RUNNER_TOKEN" --ephemeral --work _work --labels "$RUNNER_LABEL" ${runnerFlags} --name "$RUNNER_NAME" && ./run.sh`,
|
|
165
186
|
];
|
|
166
187
|
}
|
|
167
188
|
else if (this.image.os.is(common_1.Os.WINDOWS)) {
|
|
168
189
|
return [
|
|
169
190
|
'powershell', '-Command',
|
|
170
|
-
|
|
191
|
+
`cd \\actions ; ./config.cmd --unattended --url "https://\${Env:GITHUB_DOMAIN}/\${Env:OWNER}/\${Env:REPO}" --token "\${Env:RUNNER_TOKEN}" --ephemeral --work _work --labels "\${Env:RUNNER_LABEL}" ${runnerFlags} --name "\${Env:RUNNER_NAME}" ; ./run.cmd`,
|
|
171
192
|
];
|
|
172
193
|
}
|
|
173
194
|
else {
|
|
@@ -177,7 +198,7 @@ class FargateRunner extends common_1.BaseProvider {
|
|
|
177
198
|
}
|
|
178
199
|
exports.FargateRunner = FargateRunner;
|
|
179
200
|
_a = JSII_RTTI_SYMBOL_1;
|
|
180
|
-
FargateRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunner", version: "0.
|
|
201
|
+
FargateRunner[_a] = { fqn: "@cloudsnorkel/cdk-github-runners.FargateRunner", version: "0.6.1" };
|
|
181
202
|
/**
|
|
182
203
|
* Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.
|
|
183
204
|
*
|
|
@@ -194,4 +215,4 @@ FargateRunner.LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images',
|
|
|
194
215
|
* * `EXTRA_PACKAGES` can be used to install additional packages.
|
|
195
216
|
*/
|
|
196
217
|
FargateRunner.LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'fargate', 'linux-arm64');
|
|
197
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate.js","sourceRoot":"","sources":["../../src/providers/fargate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_ec2 as ec2,\n  aws_ecs as ecs,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport { Architecture, BaseProvider, IImageBuilder, IRunnerProvider, Os, RunnerImage, RunnerProviderProps, RunnerRuntimeParameters } from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n/**\n * Properties for FargateRunner.\n */\nexport interface FargateRunnerProps extends RunnerProviderProps {\n  /**\n   * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist.\n   *\n   * @default image builder with `FargateRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile\n   */\n  readonly imageBuilder?: IImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default undefined\n   * @deprecated use {@link labels} instead\n   */\n  readonly label?: string;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['fargate']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets to run the runners in.\n   *\n   * @default Fargate default\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security Group to assign to the task.\n   *\n   * @default a new security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Existing Fargate cluster to use.\n   *\n   * @default a new cluster\n   */\n  readonly cluster?: ecs.Cluster;\n\n  /**\n   * Assign public IP to the runner task.\n   *\n   * Make sure the task will have access to GitHub. A public IP might be required unless you have NAT gateway.\n   *\n   * @default true\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The number of cpu units used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values,\n   * which determines your range of valid values for the memory parameter:\n   *\n   * 256 (.25 vCPU) - Available memory values: 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB)\n   *\n   * 512 (.5 vCPU) - Available memory values: 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB)\n   *\n   * 1024 (1 vCPU) - Available memory values: 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB)\n   *\n   * 2048 (2 vCPU) - Available memory values: Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB)\n   *\n   * 4096 (4 vCPU) - Available memory values: Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB)\n   *\n   * @default 1024\n   */\n  readonly cpu?: number;\n\n  /**\n   * The amount (in MiB) of memory used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values, which determines your range of valid values for the cpu parameter:\n   *\n   * 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB) - Available cpu values: 256 (.25 vCPU)\n   *\n   * 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB) - Available cpu values: 512 (.5 vCPU)\n   *\n   * 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB) - Available cpu values: 1024 (1 vCPU)\n   *\n   * Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB) - Available cpu values: 2048 (2 vCPU)\n   *\n   * Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB) - Available cpu values: 4096 (4 vCPU)\n   *\n   * @default 2048\n   */\n  readonly memoryLimitMiB?: number;\n\n  /**\n   * The amount (in GiB) of ephemeral storage to be allocated to the task. The maximum supported value is 200 GiB.\n   *\n   * NOTE: This parameter is only supported for tasks hosted on AWS Fargate using platform version 1.4.0 or later.\n   *\n   * @default 20\n   */\n  readonly ephemeralStorageGiB?: number;\n\n  /**\n   * Use Fargate spot capacity provider to save money.\n   *\n   * * Runners may fail to start due to missing capacity.\n   * * Runners might be stopped prematurely with spot pricing.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n}\n\n/**\n * Properties for EcsFargateLaunchTarget.\n */\ninterface EcsFargateLaunchTargetProps {\n  readonly spot: boolean;\n  readonly enableExecute: boolean;\n}\n\n/**\n * Our special launch target that can use spot instances and set EnableExecuteCommand.\n */\nclass EcsFargateLaunchTarget implements stepfunctions_tasks.IEcsLaunchTarget {\n  constructor(readonly props: EcsFargateLaunchTargetProps) {}\n\n  /**\n   * Called when the Fargate launch type configured on RunTask\n   */\n  public bind(_task: stepfunctions_tasks.EcsRunTask,\n    launchTargetOptions: stepfunctions_tasks.LaunchTargetBindOptions): stepfunctions_tasks.EcsLaunchTargetConfig {\n    if (!launchTargetOptions.taskDefinition.isFargateCompatible) {\n      throw new Error('Supplied TaskDefinition is not compatible with Fargate');\n    }\n\n    return {\n      parameters: {\n        EnableExecuteCommand: this.props.enableExecute,\n        CapacityProviderStrategy: [\n          {\n            CapacityProvider: this.props.spot ? 'FARGATE_SPOT' : 'FARGATE',\n          },\n        ],\n      },\n    };\n  }\n}\n\n/**\n * GitHub Actions runner provider using Fargate to execute the actions.\n *\n * Creates a task definition with a single container that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class FargateRunner extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'fargate', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'fargate', 'linux-arm64');\n\n  /**\n   * Cluster hosting the task hosting the runner.\n   */\n  readonly cluster: ecs.Cluster;\n\n  /**\n   * Fargate task hosting the runner.\n   */\n  readonly task: ecs.FargateTaskDefinition;\n\n  /**\n   * Container definition hosting the runner.\n   */\n  readonly container: ecs.ContainerDefinition;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the runner task.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets used for hosting the runner task.\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Whether runner task will have a public IP.\n   */\n  readonly assignPublicIp: boolean;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * The network connections associated with this resource.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Use spot pricing for Fargate tasks.\n   */\n  readonly spot: boolean;\n\n  /**\n   * Docker image used to start a new Fargate task.\n   */\n  readonly image: RunnerImage;\n\n  constructor(scope: Construct, id: string, props: FargateRunnerProps) {\n    super(scope, id);\n\n    this.labels = this.labelsFromProperties('fargate', props.label, props.labels);\n    this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });\n    this.subnetSelection = props.subnetSelection;\n    this.securityGroup = props.securityGroup ?? new ec2.SecurityGroup(this, 'security group', { vpc: this.vpc });\n    this.connections = this.securityGroup.connections;\n    this.assignPublicIp = props.assignPublicIp ?? true;\n    this.cluster = props.cluster ? props.cluster : new ecs.Cluster(\n      this,\n      'cluster',\n      {\n        vpc: this.vpc,\n        enableFargateCapacityProviders: true,\n      },\n    );\n    this.spot = props.spot ?? false;\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: FargateRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    let arch: ecs.CpuArchitecture;\n    if (image.architecture.is(Architecture.ARM64)) {\n      arch = ecs.CpuArchitecture.ARM64;\n    } else if (image.architecture.is(Architecture.X86_64)) {\n      arch = ecs.CpuArchitecture.X86_64;\n    } else {\n      throw new Error(`${image.architecture.name} is not supported on Fargate`);\n    }\n\n    let os: ecs.OperatingSystemFamily;\n    if (image.os.is(Os.LINUX)) {\n      os = ecs.OperatingSystemFamily.LINUX;\n    } else if (image.os.is(Os.WINDOWS)) {\n      os = ecs.OperatingSystemFamily.WINDOWS_SERVER_2019_CORE;\n      if (props.ephemeralStorageGiB) {\n        throw new Error('Ephemeral storage is not supported on Fargate Windows');\n      }\n    } else {\n      throw new Error(`${image.os.name} is not supported on Fargate`);\n    }\n\n    this.task = new ecs.FargateTaskDefinition(\n      this,\n      'task',\n      {\n        cpu: props.cpu ?? 1024,\n        memoryLimitMiB: props.memoryLimitMiB ?? 2048,\n        ephemeralStorageGiB: props.ephemeralStorageGiB ?? !image.os.is(Os.WINDOWS) ? 25 : undefined,\n        runtimePlatform: {\n          operatingSystemFamily: os,\n          cpuArchitecture: arch,\n        },\n      },\n    );\n    this.container = this.task.addContainer(\n      'runner',\n      {\n        image: ecs.AssetImage.fromEcrRepository(image.imageRepository, image.imageTag),\n        logging: ecs.AwsLogDriver.awsLogs({\n          logGroup: new logs.LogGroup(this, 'logs', {\n            retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n            removalPolicy: RemovalPolicy.DESTROY,\n          }),\n          streamPrefix: 'runner',\n        }),\n        command: this.runCommand(),\n      },\n    );\n\n    this.grantPrincipal = new iam.UnknownPrincipal({ resource: this.task.taskRole });\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    return new stepfunctions_tasks.EcsRunTask(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        taskDefinition: this.task,\n        cluster: this.cluster,\n        launchTarget: new EcsFargateLaunchTarget({\n          spot: this.spot,\n          enableExecute: this.image.os.is(Os.LINUX),\n        }),\n        subnets: this.subnetSelection,\n        assignPublicIp: this.assignPublicIp,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        containerOverrides: [\n          {\n            containerDefinition: this.container,\n            environment: [\n              {\n                name: 'RUNNER_TOKEN',\n                value: parameters.runnerTokenPath,\n              },\n              {\n                name: 'RUNNER_NAME',\n                value: parameters.runnerNamePath,\n              },\n              {\n                name: 'RUNNER_LABEL',\n                value: this.labels.join(','),\n              },\n              {\n                name: 'GITHUB_DOMAIN',\n                value: parameters.githubDomainPath,\n              },\n              {\n                name: 'OWNER',\n                value: parameters.ownerPath,\n              },\n              {\n                name: 'REPO',\n                value: parameters.repoPath,\n              },\n            ],\n          },\n        ],\n      },\n    );\n  }\n\n  private runCommand(): string[] {\n    if (this.image.os.is(Os.LINUX)) {\n      return [\n        'sh', '-c',\n        'if [ \"${RUNNER_VERSION}\" = \"latest\" ]; then RUNNER_FLAGS=\"\"; else RUNNER_FLAGS=\"--disableupdate\"; fi && ./config.sh --unattended --url \"https://${GITHUB_DOMAIN}/${OWNER}/${REPO}\" --token \"${RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${RUNNER_LABEL}\" ${RUNNER_FLAGS} --name \"${RUNNER_NAME}\" && ./run.sh',\n      ];\n    } else if (this.image.os.is(Os.WINDOWS)) {\n      return [\n        'powershell', '-Command',\n        'if (${Env:RUNNER_VERSION} -eq \"latest\") { $RunnerFlags = \"\" } else { $RunnerFlags = \"--disableupdate\" } ; cd \\\\actions ; ./config.cmd --unattended --url \"https://${Env:GITHUB_DOMAIN}/${Env:OWNER}/${Env:REPO}\" --token \"${Env:RUNNER_TOKEN}\" --ephemeral --work _work --labels \"${Env:RUNNER_LABEL}\" ${RunnerFlags} --name \"${Env:RUNNER_NAME}\" ; ./run.cmd',\n      ];\n    } else {\n      throw new Error(`Fargate runner doesn't support ${this.image.os.name}`);\n    }\n  }\n}\n"]}
|
|
218
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate.js","sourceRoot":"","sources":["../../src/providers/fargate.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport {\n  aws_ec2 as ec2,\n  aws_ecs as ecs,\n  aws_iam as iam,\n  aws_logs as logs,\n  aws_stepfunctions as stepfunctions,\n  aws_stepfunctions_tasks as stepfunctions_tasks,\n  RemovalPolicy,\n} from 'aws-cdk-lib';\nimport { RetentionDays } from 'aws-cdk-lib/aws-logs';\nimport { IntegrationPattern } from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\nimport {\n  Architecture,\n  BaseProvider,\n  IImageBuilder,\n  IRunnerProvider,\n  IRunnerProviderStatus,\n  Os,\n  RunnerImage,\n  RunnerProviderProps,\n  RunnerRuntimeParameters, RunnerVersion,\n} from './common';\nimport { CodeBuildImageBuilder } from './image-builders/codebuild';\n\n/**\n * Properties for FargateRunner.\n */\nexport interface FargateRunnerProps extends RunnerProviderProps {\n  /**\n   * Provider running an image to run inside CodeBuild with GitHub runner pre-configured. A user named `runner` is expected to exist.\n   *\n   * @default image builder with `FargateRunner.LINUX_X64_DOCKERFILE_PATH` as Dockerfile\n   */\n  readonly imageBuilder?: IImageBuilder;\n\n  /**\n   * GitHub Actions label used for this provider.\n   *\n   * @default undefined\n   * @deprecated use {@link labels} instead\n   */\n  readonly label?: string;\n\n  /**\n   * GitHub Actions labels used for this provider.\n   *\n   * These labels are used to identify which provider should spawn a new on-demand runner. Every job sends a webhook with the labels it's looking for\n   * based on runs-on. We match the labels from the webhook with the labels specified here. If all the labels specified here are present in the\n   * job's labels, this provider will be chosen and spawn a new runner.\n   *\n   * @default ['fargate']\n   */\n  readonly labels?: string[];\n\n  /**\n   * VPC to launch the runners in.\n   *\n   * @default default account VPC\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets to run the runners in.\n   *\n   * @default Fargate default\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security Group to assign to the task.\n   *\n   * @default a new security group\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Existing Fargate cluster to use.\n   *\n   * @default a new cluster\n   */\n  readonly cluster?: ecs.Cluster;\n\n  /**\n   * Assign public IP to the runner task.\n   *\n   * Make sure the task will have access to GitHub. A public IP might be required unless you have NAT gateway.\n   *\n   * @default true\n   */\n  readonly assignPublicIp?: boolean;\n\n  /**\n   * The number of cpu units used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values,\n   * which determines your range of valid values for the memory parameter:\n   *\n   * 256 (.25 vCPU) - Available memory values: 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB)\n   *\n   * 512 (.5 vCPU) - Available memory values: 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB)\n   *\n   * 1024 (1 vCPU) - Available memory values: 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB)\n   *\n   * 2048 (2 vCPU) - Available memory values: Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB)\n   *\n   * 4096 (4 vCPU) - Available memory values: Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB)\n   *\n   * @default 1024\n   */\n  readonly cpu?: number;\n\n  /**\n   * The amount (in MiB) of memory used by the task. For tasks using the Fargate launch type,\n   * this field is required and you must use one of the following values, which determines your range of valid values for the cpu parameter:\n   *\n   * 512 (0.5 GB), 1024 (1 GB), 2048 (2 GB) - Available cpu values: 256 (.25 vCPU)\n   *\n   * 1024 (1 GB), 2048 (2 GB), 3072 (3 GB), 4096 (4 GB) - Available cpu values: 512 (.5 vCPU)\n   *\n   * 2048 (2 GB), 3072 (3 GB), 4096 (4 GB), 5120 (5 GB), 6144 (6 GB), 7168 (7 GB), 8192 (8 GB) - Available cpu values: 1024 (1 vCPU)\n   *\n   * Between 4096 (4 GB) and 16384 (16 GB) in increments of 1024 (1 GB) - Available cpu values: 2048 (2 vCPU)\n   *\n   * Between 8192 (8 GB) and 30720 (30 GB) in increments of 1024 (1 GB) - Available cpu values: 4096 (4 vCPU)\n   *\n   * @default 2048\n   */\n  readonly memoryLimitMiB?: number;\n\n  /**\n   * The amount (in GiB) of ephemeral storage to be allocated to the task. The maximum supported value is 200 GiB.\n   *\n   * NOTE: This parameter is only supported for tasks hosted on AWS Fargate using platform version 1.4.0 or later.\n   *\n   * @default 20\n   */\n  readonly ephemeralStorageGiB?: number;\n\n  /**\n   * Use Fargate spot capacity provider to save money.\n   *\n   * * Runners may fail to start due to missing capacity.\n   * * Runners might be stopped prematurely with spot pricing.\n   *\n   * @default false\n   */\n  readonly spot?: boolean;\n}\n\n/**\n * Properties for EcsFargateLaunchTarget.\n */\ninterface EcsFargateLaunchTargetProps {\n  readonly spot: boolean;\n  readonly enableExecute: boolean;\n}\n\n/**\n * Our special launch target that can use spot instances and set EnableExecuteCommand.\n */\nclass EcsFargateLaunchTarget implements stepfunctions_tasks.IEcsLaunchTarget {\n  constructor(readonly props: EcsFargateLaunchTargetProps) {}\n\n  /**\n   * Called when the Fargate launch type configured on RunTask\n   */\n  public bind(_task: stepfunctions_tasks.EcsRunTask,\n    launchTargetOptions: stepfunctions_tasks.LaunchTargetBindOptions): stepfunctions_tasks.EcsLaunchTargetConfig {\n    if (!launchTargetOptions.taskDefinition.isFargateCompatible) {\n      throw new Error('Supplied TaskDefinition is not compatible with Fargate');\n    }\n\n    return {\n      parameters: {\n        EnableExecuteCommand: this.props.enableExecute,\n        CapacityProviderStrategy: [\n          {\n            CapacityProvider: this.props.spot ? 'FARGATE_SPOT' : 'FARGATE',\n          },\n        ],\n      },\n    };\n  }\n}\n\n/**\n * GitHub Actions runner provider using Fargate to execute jobs.\n *\n * Creates a task definition with a single container that gets started for each job.\n *\n * This construct is not meant to be used by itself. It should be passed in the providers property for GitHubRunners.\n */\nexport class FargateRunner extends BaseProvider implements IRunnerProvider {\n  /**\n   * Path to Dockerfile for Linux x64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_X64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'fargate', 'linux-x64');\n\n  /**\n   * Path to Dockerfile for Linux ARM64 with all the requirement for Fargate runner. Use this Dockerfile unless you need to customize it further than allowed by hooks.\n   *\n   * Available build arguments that can be set in the image builder:\n   * * `BASE_IMAGE` sets the `FROM` line. This should be an Ubuntu compatible image.\n   * * `EXTRA_PACKAGES` can be used to install additional packages.\n   */\n  public static readonly LINUX_ARM64_DOCKERFILE_PATH = path.join(__dirname, 'docker-images', 'fargate', 'linux-arm64');\n\n  /**\n   * Cluster hosting the task hosting the runner.\n   */\n  readonly cluster: ecs.Cluster;\n\n  /**\n   * Fargate task hosting the runner.\n   */\n  readonly task: ecs.FargateTaskDefinition;\n\n  /**\n   * Container definition hosting the runner.\n   */\n  readonly container: ecs.ContainerDefinition;\n\n  /**\n   * Labels associated with this provider.\n   */\n  readonly labels: string[];\n\n  /**\n   * VPC used for hosting the runner task.\n   */\n  readonly vpc?: ec2.IVpc;\n\n  /**\n   * Subnets used for hosting the runner task.\n   */\n  readonly subnetSelection?: ec2.SubnetSelection;\n\n  /**\n   * Security group attached to the task.\n   */\n  readonly securityGroup?: ec2.ISecurityGroup;\n\n  /**\n   * Whether runner task will have a public IP.\n   */\n  readonly assignPublicIp: boolean;\n\n  /**\n   * Grant principal used to add permissions to the runner role.\n   */\n  readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * The network connections associated with this resource.\n   */\n  readonly connections: ec2.Connections;\n\n  /**\n   * Use spot pricing for Fargate tasks.\n   */\n  readonly spot: boolean;\n\n  /**\n   * Docker image loaded with GitHub Actions Runner and its prerequisites. The image is built by an image builder and is specific to Fargate tasks.\n   */\n  readonly image: RunnerImage;\n\n  constructor(scope: Construct, id: string, props: FargateRunnerProps) {\n    super(scope, id);\n\n    this.labels = this.labelsFromProperties('fargate', props.label, props.labels);\n    this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, 'default vpc', { isDefault: true });\n    this.subnetSelection = props.subnetSelection;\n    this.securityGroup = props.securityGroup ?? new ec2.SecurityGroup(this, 'security group', { vpc: this.vpc });\n    this.connections = this.securityGroup.connections;\n    this.assignPublicIp = props.assignPublicIp ?? true;\n    this.cluster = props.cluster ? props.cluster : new ecs.Cluster(\n      this,\n      'cluster',\n      {\n        vpc: this.vpc,\n        enableFargateCapacityProviders: true,\n      },\n    );\n    this.spot = props.spot ?? false;\n\n    const imageBuilder = props.imageBuilder ?? new CodeBuildImageBuilder(this, 'Image Builder', {\n      dockerfilePath: FargateRunner.LINUX_X64_DOCKERFILE_PATH,\n    });\n    const image = this.image = imageBuilder.bind();\n\n    let arch: ecs.CpuArchitecture;\n    if (image.architecture.is(Architecture.ARM64)) {\n      arch = ecs.CpuArchitecture.ARM64;\n    } else if (image.architecture.is(Architecture.X86_64)) {\n      arch = ecs.CpuArchitecture.X86_64;\n    } else {\n      throw new Error(`${image.architecture.name} is not supported on Fargate`);\n    }\n\n    let os: ecs.OperatingSystemFamily;\n    if (image.os.is(Os.LINUX)) {\n      os = ecs.OperatingSystemFamily.LINUX;\n    } else if (image.os.is(Os.WINDOWS)) {\n      os = ecs.OperatingSystemFamily.WINDOWS_SERVER_2019_CORE;\n      if (props.ephemeralStorageGiB) {\n        throw new Error('Ephemeral storage is not supported on Fargate Windows');\n      }\n    } else {\n      throw new Error(`${image.os.name} is not supported on Fargate`);\n    }\n\n    this.task = new ecs.FargateTaskDefinition(\n      this,\n      'task',\n      {\n        cpu: props.cpu ?? 1024,\n        memoryLimitMiB: props.memoryLimitMiB ?? 2048,\n        ephemeralStorageGiB: props.ephemeralStorageGiB ?? !image.os.is(Os.WINDOWS) ? 25 : undefined,\n        runtimePlatform: {\n          operatingSystemFamily: os,\n          cpuArchitecture: arch,\n        },\n      },\n    );\n    this.container = this.task.addContainer(\n      'runner',\n      {\n        image: ecs.AssetImage.fromEcrRepository(image.imageRepository, image.imageTag),\n        logging: ecs.AwsLogDriver.awsLogs({\n          logGroup: new logs.LogGroup(this, 'logs', {\n            retention: props.logRetention ?? RetentionDays.ONE_MONTH,\n            removalPolicy: RemovalPolicy.DESTROY,\n          }),\n          streamPrefix: 'runner',\n        }),\n        command: this.runCommand(),\n      },\n    );\n\n    this.grantPrincipal = new iam.UnknownPrincipal({ resource: this.task.taskRole });\n  }\n\n  /**\n   * Generate step function task(s) to start a new runner.\n   *\n   * Called by GithubRunners and shouldn't be called manually.\n   *\n   * @param parameters workflow job details\n   */\n  getStepFunctionTask(parameters: RunnerRuntimeParameters): stepfunctions.IChainable {\n    return new stepfunctions_tasks.EcsRunTask(\n      this,\n      this.labels.join(', '),\n      {\n        integrationPattern: IntegrationPattern.RUN_JOB, // sync\n        taskDefinition: this.task,\n        cluster: this.cluster,\n        launchTarget: new EcsFargateLaunchTarget({\n          spot: this.spot,\n          enableExecute: this.image.os.is(Os.LINUX),\n        }),\n        subnets: this.subnetSelection,\n        assignPublicIp: this.assignPublicIp,\n        securityGroups: this.securityGroup ? [this.securityGroup] : undefined,\n        containerOverrides: [\n          {\n            containerDefinition: this.container,\n            environment: [\n              {\n                name: 'RUNNER_TOKEN',\n                value: parameters.runnerTokenPath,\n              },\n              {\n                name: 'RUNNER_NAME',\n                value: parameters.runnerNamePath,\n              },\n              {\n                name: 'RUNNER_LABEL',\n                value: this.labels.join(','),\n              },\n              {\n                name: 'GITHUB_DOMAIN',\n                value: parameters.githubDomainPath,\n              },\n              {\n                name: 'OWNER',\n                value: parameters.ownerPath,\n              },\n              {\n                name: 'REPO',\n                value: parameters.repoPath,\n              },\n            ],\n          },\n        ],\n      },\n    );\n  }\n\n  grantStateMachine(_: iam.IGrantable) {\n  }\n\n  status(statusFunctionRole: iam.IGrantable): IRunnerProviderStatus {\n    this.image.imageRepository.grant(statusFunctionRole, 'ecr:DescribeImages');\n\n    return {\n      type: this.constructor.name,\n      labels: this.labels,\n      vpcArn: this.vpc?.vpcArn,\n      securityGroup: this.securityGroup?.securityGroupId,\n      roleArn: this.task.taskRole.roleArn,\n      image: {\n        imageRepository: this.image.imageRepository.repositoryUri,\n        imageTag: this.image.imageTag,\n        imageBuilderLogGroup: this.image.logGroup?.logGroupName,\n      },\n    };\n  }\n\n  private runCommand(): string[] {\n    let runnerFlags = '';\n    if (this.image.runnerVersion.is(RunnerVersion.latest())) {\n      runnerFlags = '--disableupdate';\n    }\n\n    if (this.image.os.is(Os.LINUX)) {\n      return [\n        'sh', '-c',\n        `./config.sh --unattended --url \"https://$GITHUB_DOMAIN/$OWNER/$REPO\" --token \"$RUNNER_TOKEN\" --ephemeral --work _work --labels \"$RUNNER_LABEL\" ${runnerFlags} --name \"$RUNNER_NAME\" && ./run.sh`,\n      ];\n    } else if (this.image.os.is(Os.WINDOWS)) {\n      return [\n        'powershell', '-Command',\n        `cd \\\\actions ; ./config.cmd --unattended --url \"https://\\${Env:GITHUB_DOMAIN}/\\${Env:OWNER}/\\${Env:REPO}\" --token \"\\${Env:RUNNER_TOKEN}\" --ephemeral --work _work --labels \"\\${Env:RUNNER_LABEL}\" ${runnerFlags} --name \"\\${Env:RUNNER_NAME}\" ; ./run.cmd`,\n      ];\n    } else {\n      throw new Error(`Fargate runner doesn't support ${this.image.os.name}`);\n    }\n  }\n}\n"]}
|