Skip to content

Commit 4d6bd0c

Browse files
committed
feat: ECS fargate infra
1 parent 8f6411f commit 4d6bd0c

File tree

4 files changed

+317
-0
lines changed

4 files changed

+317
-0
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
.terraform

.terraform.lock.hcl

+25
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

main.tf

+209
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,209 @@
1+
terraform {
2+
required_version = ">= 1.0"
3+
required_providers {
4+
aws = {
5+
source = "hashicorp/aws"
6+
version = ">= 5.0"
7+
}
8+
}
9+
10+
backend "remote" {
11+
hostname = "app.terraform.io"
12+
organization = "wallet-connect"
13+
workspaces {
14+
name = "github-actions-runners"
15+
}
16+
}
17+
}
18+
19+
provider "aws" {
20+
region = "eu-central-1"
21+
22+
default_tags {
23+
tags = {
24+
Application = "github-actions-runners"
25+
}
26+
}
27+
}
28+
29+
variable "run_task" {
30+
type = bool
31+
default = false
32+
}
33+
34+
variable "cpu" {
35+
type = number
36+
default = 256
37+
}
38+
39+
variable "memory" {
40+
type = number
41+
default = 512
42+
}
43+
44+
variable "runner_token" {
45+
type = string
46+
default = null
47+
}
48+
49+
variable "repo_url" {
50+
type = string
51+
default = null
52+
}
53+
54+
variable "labels" {
55+
type = string
56+
}
57+
58+
variable "desired_count" {
59+
type = number
60+
}
61+
62+
resource "aws_vpc" "this" {
63+
cidr_block = "10.0.0.0/16"
64+
}
65+
66+
# Public subnet
67+
68+
resource "aws_subnet" "public" {
69+
vpc_id = aws_vpc.this.id
70+
cidr_block = "10.0.1.0/24"
71+
}
72+
73+
resource "aws_route_table" "public" {
74+
vpc_id = aws_vpc.this.id
75+
}
76+
77+
resource "aws_internet_gateway" "this" {
78+
vpc_id = aws_vpc.this.id
79+
}
80+
81+
resource "aws_route" "internet_gateway" {
82+
route_table_id = aws_route_table.public.id
83+
destination_cidr_block = "0.0.0.0/0"
84+
gateway_id = aws_internet_gateway.this.id
85+
}
86+
87+
resource "aws_route_table_association" "public" {
88+
subnet_id = aws_subnet.public.id
89+
route_table_id = aws_route_table.public.id
90+
}
91+
92+
# Private subnet
93+
94+
resource "aws_subnet" "private" {
95+
vpc_id = aws_vpc.this.id
96+
cidr_block = "10.0.2.0/24"
97+
}
98+
99+
resource "aws_route_table" "private" {
100+
vpc_id = aws_vpc.this.id
101+
}
102+
103+
resource "aws_eip" "this" {}
104+
105+
resource "aws_nat_gateway" "this" {
106+
allocation_id = aws_eip.this.id
107+
subnet_id = aws_subnet.public.id
108+
109+
depends_on = [aws_internet_gateway.this]
110+
}
111+
112+
resource "aws_route" "nat_gateway" {
113+
route_table_id = aws_route_table.private.id
114+
destination_cidr_block = "0.0.0.0/0"
115+
nat_gateway_id = aws_nat_gateway.this.id
116+
}
117+
118+
resource "aws_route_table_association" "private" {
119+
subnet_id = aws_subnet.private.id
120+
route_table_id = aws_route_table.private.id
121+
}
122+
123+
# ECS
124+
125+
data "aws_iam_policy_document" "assume_role" {
126+
statement {
127+
actions = ["sts:AssumeRole"]
128+
129+
principals {
130+
type = "Service"
131+
identifiers = ["ecs-tasks.amazonaws.com"]
132+
}
133+
}
134+
}
135+
136+
resource "aws_iam_role" "this" {
137+
name = "github-actions-runner"
138+
assume_role_policy = data.aws_iam_policy_document.assume_role.json
139+
}
140+
141+
resource "aws_iam_role_policy_attachment" "ecs_task_execution_role_policy" {
142+
role = aws_iam_role.this.name
143+
policy_arn = "arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy"
144+
}
145+
146+
resource "aws_iam_role_policy_attachment" "cloudwatch_write_policy" {
147+
role = aws_iam_role.this.name
148+
policy_arn = "arn:aws:iam::aws:policy/CloudWatchLogsFullAccess"
149+
}
150+
151+
resource "aws_ecs_cluster" "this" {
152+
name = "github-actions-runners"
153+
}
154+
155+
resource "aws_ecs_cluster_capacity_providers" "example" {
156+
cluster_name = aws_ecs_cluster.this.name
157+
158+
capacity_providers = ["FARGATE_SPOT"]
159+
160+
default_capacity_provider_strategy {
161+
capacity_provider = "FARGATE_SPOT"
162+
}
163+
}
164+
165+
resource "aws_cloudwatch_log_group" "this" {
166+
name = "github-action-runner-logs"
167+
retention_in_days = 1
168+
}
169+
170+
resource "aws_ecs_task_definition" "this" {
171+
family = "github-actions-runner"
172+
requires_compatibilities = ["FARGATE"]
173+
network_mode = "awsvpc"
174+
175+
container_definitions = jsonencode([{
176+
name = "github-actions-runner"
177+
image = "myoung34/github-runner:ubuntu-focal"
178+
essential = true
179+
180+
logConfiguration = {
181+
logDriver = "awslogs"
182+
options = {
183+
awslogs-group = aws_cloudwatch_log_group.this.name
184+
awslogs-region = "eu-central-1"
185+
awslogs-stream-prefix = "ecs"
186+
}
187+
}
188+
}])
189+
190+
cpu = 256
191+
memory = 512
192+
execution_role_arn = aws_iam_role.this.arn
193+
194+
runtime_platform {
195+
operating_system_family = "LINUX"
196+
cpu_architecture = "X86_64"
197+
}
198+
}
199+
200+
module "run_task" {
201+
count = var.run_task ? 1 : 0
202+
source = "./run_task"
203+
cpu = var.cpu
204+
memory = var.memory
205+
runner_token = var.runner_token
206+
repo_url = var.repo_url
207+
labels = var.labels
208+
desired_count = var.desired_count
209+
}

run_task/main.tf

+82
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
variable "cpu" {
2+
type = number
3+
}
4+
5+
variable "memory" {
6+
type = number
7+
}
8+
9+
variable "runner_token" {
10+
type = string
11+
}
12+
13+
variable "repo_url" {
14+
type = string
15+
}
16+
17+
variable "labels" {
18+
type = string
19+
}
20+
21+
variable "desired_count" {
22+
type = number
23+
}
24+
25+
data "aws_ecs_cluster" "this" {
26+
cluster_name = "github-actions-runners"
27+
}
28+
29+
data "aws_subnet" "this" {
30+
cidr_block = "10.0.2.0/24"
31+
filter {
32+
name = "tag:Application"
33+
values = ["github-actions-runners"]
34+
}
35+
}
36+
37+
data "aws_ecs_task_execution" "this" {
38+
cluster = data.aws_ecs_cluster.this.id
39+
task_definition = "github-actions-runner"
40+
desired_count = var.desired_count
41+
42+
network_configuration {
43+
subnets = [data.aws_subnet.this.id]
44+
}
45+
46+
overrides {
47+
cpu = var.cpu
48+
memory = var.memory
49+
50+
container_overrides {
51+
name = "github-actions-runner"
52+
cpu = var.cpu
53+
memory = var.memory
54+
55+
environment {
56+
key = "RUNNER_NAME_PREFIX"
57+
value = "aws-ecs-fargate-${var.cpu}cpu-${var.memory}mem"
58+
}
59+
environment {
60+
key = "RUNNER_TOKEN"
61+
value = var.runner_token
62+
}
63+
environment {
64+
key = "REPO_URL"
65+
value = var.repo_url
66+
}
67+
environment {
68+
key = "LABELS"
69+
value = var.labels
70+
}
71+
environment {
72+
key = "EPHEMERAL"
73+
value = true
74+
}
75+
environment {
76+
key = "START_DOCKER_SERVICE"
77+
value = true
78+
}
79+
}
80+
}
81+
}
82+

0 commit comments

Comments
 (0)