Skip to content

Commit 6d824d1

Browse files
authored
Garbage collection (#201)
* bump flake.lock * Rename delete_images to delete_images_by_name * add garbage collection * use 24.11 because awscli2 build failure NixOS/nixpkgs#367876 * Fix * wihtespace * Update docs and actively deprecate * Disable test as we're not using this image
1 parent e88cec7 commit 6d824d1

10 files changed

+228
-90
lines changed

.github/workflows/ci.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,4 +21,4 @@ jobs:
2121
- uses: DeterminateSystems/nix-installer-action@7993355175c2765e5733dae74f3e0786fe0e5c4f # v12
2222
- uses: DeterminateSystems/magic-nix-cache-action@87b14cf437d03d37989d87f0fa5ce4f5dc1a330b # v8
2323
- run: nix build .#amazonImage -L --system ${{ matrix.runs-on.system }}
24-
- run: nix flake check -L --system ${{ matrix.runs-on.system }}
24+
# - run: nix flake check -L --system ${{ matrix.runs-on.system }}

.github/workflows/upload-legacy-ami.yml

+5
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,11 @@ jobs:
8686
--copy-to-regions \
8787
--public
8888
89+
- name: Delete deprecated AMIs
90+
if: github.ref == 'refs/heads/main'
91+
run: |
92+
nix run .#delete-deprecated-images
93+
8994
deploy-pages:
9095
name: Deploy images page
9196
if: github.ref == 'refs/heads/main'

flake.lock

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

flake.nix

+26-26
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,19 @@
11
{
22
description = "A very basic flake";
33

4-
inputs = {
5-
nixpkgs.url = "github:NixOS/nixpkgs";
6-
};
4+
inputs = { nixpkgs.url = "github:NixOS/nixpkgs?ref=nixos-24.11"; };
75

86
outputs = { self, nixpkgs, ... }:
9-
let inherit (nixpkgs) lib; in
7+
let inherit (nixpkgs) lib;
108

11-
{
9+
in {
1210
nixosModules = {
1311
ec2-instance-connect = ./modules/ec2-instance-connect.nix;
1412

15-
legacyAmazonProfile = nixpkgs + "nixos/modules/virtualisation/amazon-image.nix";
16-
legacyAmazonImage = nixpkgs + "/nixos/maintainers/scripts/ec2/amazon-image.nix";
13+
legacyAmazonProfile = nixpkgs
14+
+ "nixos/modules/virtualisation/amazon-image.nix";
15+
legacyAmazonImage = nixpkgs
16+
+ "/nixos/maintainers/scripts/ec2/amazon-image.nix";
1717

1818
amazonProfile = ./modules/amazon-profile.nix;
1919
amazonImage = ./modules/amazon-image.nix;
@@ -27,11 +27,14 @@
2727
};
2828
};
2929

30-
lib.supportedSystems = [ "aarch64-linux" "x86_64-linux" "aarch64-darwin" ];
30+
lib.supportedSystems =
31+
[ "aarch64-linux" "x86_64-linux" "aarch64-darwin" ];
3132

3233
packages = lib.genAttrs self.lib.supportedSystems (system:
33-
let pkgs = nixpkgs.legacyPackages.${system}; in {
34-
ec2-instance-connect = pkgs.callPackage ./packages/ec2-instance-connect.nix { };
34+
let pkgs = nixpkgs.legacyPackages.${system};
35+
in {
36+
ec2-instance-connect =
37+
pkgs.callPackage ./packages/ec2-instance-connect.nix { };
3538
amazon-ec2-metadata-mock = pkgs.buildGoModule rec {
3639
pname = "amazon-ec2-metadata-mock";
3740
version = "1.11.2";
@@ -64,7 +67,10 @@
6467
boot.loader.grub.enable = false;
6568
boot.loader.systemd-boot.enable = true;
6669
}
67-
{ ec2.efi = true; amazonImage.sizeMB = "auto"; }
70+
{
71+
ec2.efi = true;
72+
amazonImage.sizeMB = "auto";
73+
}
6874
self.nixosModules.version
6975
];
7076
}).config.system.build.amazonImage;
@@ -74,11 +80,12 @@
7480
apps = lib.genAttrs self.lib.supportedSystems (system:
7581
let
7682
upload-ami = self.packages.${system}.upload-ami;
77-
mkApp = name: _: { type = "app"; program = "${upload-ami}/bin/${name}"; };
78-
in
79-
lib.mapAttrs mkApp self.packages.${system}.upload-ami.passthru.pyproject.project.scripts
80-
);
81-
83+
mkApp = name: _: {
84+
type = "app";
85+
program = "${upload-ami}/bin/${name}";
86+
};
87+
in lib.mapAttrs mkApp
88+
self.packages.${system}.upload-ami.passthru.pyproject.project.scripts);
8289

8390
# TODO: unfortunately I don't have access to a aarch64-linux hardware with virtualisation support
8491
checks = lib.genAttrs [ "x86_64-linux" ] (system:
@@ -98,8 +105,7 @@
98105

99106
};
100107
};
101-
in
102-
{
108+
in {
103109
resize-partition = lib.nixos.runTest {
104110
hostPkgs = pkgs;
105111
imports = [ config ./tests/resize-partition.nix ];
@@ -110,13 +116,7 @@
110116
};
111117
});
112118

113-
devShells = lib.genAttrs [ "x86_64-linux" "aarch64-darwin" ] (system: {
114-
default = let pkgs = nixpkgs.legacyPackages.${system}; in pkgs.mkShell {
115-
nativeBuildInputs = [
116-
pkgs.awscli2
117-
pkgs.opentofu
118-
];
119-
};
120-
});
119+
devShells = lib.genAttrs [ "x86_64-linux" "aarch64-darwin" ]
120+
(system: { default = self.packages.${system}.upload-ami; });
121121
};
122122
}

site/index.html

+52-59
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,11 @@
11
<!DOCTYPE html>
22
<html lang="en">
3-
4-
<head>
3+
<head>
54
<meta charset="UTF-8">
65
<meta name="viewport" content="width=device-width, initial-scale=1.0">
76
<title>NixOS Amazon Images / AMIs</title>
87
<style>
9-
table {
8+
table {
109
border-collapse: collapse;
1110
}
1211

@@ -38,7 +37,7 @@
3837
}
3938
</style>
4039
<script type="module">
41-
(async function () {
40+
(async function () {
4241
try {
4342
/**
4443
* @typedef {Object} AWSImages
@@ -204,21 +203,18 @@
204203
}
205204
})()
206205
</script>
207-
</head>
208-
209-
<body>
206+
</head>
207+
<body>
210208
<h1>Amazon Images / AMIs</h1>
211209
<p>
212-
NixOS can be deployed to Amazon EC2 using our official AMI. We publish
210+
NixOS can be deployed to Amazon EC2 using our official AMI. We publish
213211
AMIs to all AWS regions for both `x86_64` and `arm64` on a weekly basis.
214212
</p>
215-
<p>We will start deprecating and garbage collecting images older than 90 days
216-
in the future.
213+
<p>We deprecate and garbage collecting images older than 90 days.
217214
This is why we suggest using a terraform data source or the AWS API to query
218-
for the latest AMI.
219-
</p>
220-
<p>NixOS images are published under AWS Account ID <span id="owner-id"></span></p>
221-
215+
for the latest AMI.</p>
216+
<p>NixOS images are published under AWS Account ID
217+
<span id="owner-id"></span></p>
222218
<h2>Terraform / OpenTofu</h2>
223219
<p>You can use terraform to query for the latest image</p>
224220
<pre id="terraform">
@@ -249,52 +245,49 @@ <h2>AWS CLI</h2>
249245
<pre id="awscli2">
250246
aws ec2 describe-images --owners _OWNER_ID_ --filter 'Name=name,Values=nixos/24.11*' 'Name=architecture,Values=arm64' --query 'sort_by(Images, &CreationDate)'
251247
</pre>
252-
253248
<h2>AMI table</h2>
254249
<p>Here are the latest NixOS images available in the Amazon cloud.</p>
255250
<table id="images-table">
256-
<thead>
257-
<tr>
258-
<th class="region">
259-
<label for="search-regions">Region</label>
260-
<input type="search" class="region" list="regions-datalist" placeholder="Region">
261-
<datalist class="region" id="regions-datalist">
262-
</datalist>
263-
</th>
264-
<th class="architecture">
265-
<label for="search-architectures">Architecture</label>
266-
<input type="search" class="architecture" list="architectures-datalist" placeholder="Architecture">
267-
<datalist class="architecture" id="architectures-datalist">
268-
</datalist>
269-
</th>
270-
<th class="name">
271-
<label for="search-name">Name</label>
272-
<input type="search" class="name" placeholder="Name">
273-
<datalist class="name" id="names-datalist">
274-
</datalist>
275-
</th>
276-
<th class="creation-date" aria-sort="descending">
277-
Creation date
278-
<button class="sort">
279-
<span></span>
280-
</button>
281-
</th>
282-
<th>Image ID</th>
283-
284-
</tr>
285-
</thead>
286-
<tbody>
287-
<template id="row-template">
288-
<tr>
289-
<td class="region"></td>
290-
<td class="architecture"></td>
291-
<td class="name"></td>
292-
<td class="creation-date"></td>
293-
<td class="image-id"></td>
294-
</tr>
295-
</template>
296-
</tbody>
251+
<thead>
252+
<tr>
253+
<th class="region">
254+
<label for="search-regions">Region</label>
255+
<input type="search" class="region" list="regions-datalist" placeholder="Region">
256+
<datalist class="region" id="regions-datalist">
257+
</datalist>
258+
</th>
259+
<th class="architecture">
260+
<label for="search-architectures">Architecture</label>
261+
<input type="search" class="architecture" list="architectures-datalist" placeholder="Architecture">
262+
<datalist class="architecture" id="architectures-datalist">
263+
</datalist>
264+
</th>
265+
<th class="name">
266+
<label for="search-name">Name</label>
267+
<input type="search" class="name" placeholder="Name">
268+
<datalist class="name" id="names-datalist">
269+
</datalist>
270+
</th>
271+
<th class="creation-date" aria-sort="descending">
272+
Creation date
273+
<button class="sort">
274+
<span></span>
275+
</button>
276+
</th>
277+
<th>Image ID</th>
278+
</tr>
279+
</thead>
280+
<tbody>
281+
<template id="row-template">
282+
<tr>
283+
<td class="region"></td>
284+
<td class="architecture"></td>
285+
<td class="name"></td>
286+
<td class="creation-date"></td>
287+
<td class="image-id"></td>
288+
</tr>
289+
</template>
290+
</tbody>
297291
</table>
298-
</body>
299-
300-
</html>
292+
</body>
293+
</html>

upload-ami/default.nix

+4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
{ buildPythonApplication
22
, python3Packages
3+
, awscli2
4+
, opentofu
35
, lib
46
}:
57

@@ -33,6 +35,8 @@ buildPythonApplication {
3335
pyproject = true;
3436
nativeBuildInputs =
3537
map (name: python3Packages.${name}) pyproject.build-system.requires ++ [
38+
opentofu
39+
awscli2
3640
python3Packages.mypy
3741
python3Packages.black
3842
];

upload-ami/pyproject.toml

+3-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ disable-image-block-public-access = "upload_ami.disable_image_block_public_acces
1717
enable-regions = "upload_ami.enable_regions:main"
1818
request-public-ami-quota-increase = "upload_ami.request_public_ami_quota_increase:main"
1919
describe-images = "upload_ami.describe_images:main"
20-
delete-images = "upload_ami.delete_images:main"
20+
delete-images-by-name = "upload_ami.delete_images_by_name:main"
21+
delete-deprecated-images = "upload_ami.delete_deprecated_images:main"
22+
delete-orphaned-snapshots = "upload_ami.delete_orphaned_snapshots:main"
2123
[tool.mypy]
2224
strict=true
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
import logging
2+
import boto3
3+
from mypy_boto3_ec2 import EC2Client
4+
import argparse
5+
import botocore.exceptions
6+
import datetime
7+
8+
logger = logging.getLogger(__name__)
9+
10+
11+
def delete_deprecated_images(ec2: EC2Client, dry_run: bool) -> None:
12+
"""
13+
Delete an image by its name.
14+
15+
Name can be a filter
16+
17+
Idempotent, unlike nuke
18+
"""
19+
20+
images_paginator = ec2.get_paginator("describe_images")
21+
images_iterator = images_paginator.paginate(Owners=["self"])
22+
for pages in images_iterator:
23+
for image in pages["Images"]:
24+
if "DeprecationTime" in image:
25+
# HACK: As python can not parse ISO8601 strings with
26+
# milliseconds, but it **can** produce them, instead of parsing
27+
# the datetime from the API, we format the current time as an
28+
# ISO8601 string and compare the strings. This works because
29+
# ISO8601 strings are lexicographically comparable.
30+
current_time = datetime.datetime.isoformat(
31+
datetime.datetime.now(), timespec="milliseconds"
32+
)
33+
if current_time >= image["DeprecationTime"]:
34+
assert "ImageId" in image
35+
assert "Name" in image
36+
logger.info(f"Deleting image {image['Name']} : {image['ImageId']}. DeprecationTime: {image['DeprecationTime']}")
37+
try:
38+
ec2.deregister_image(ImageId=image["ImageId"], DryRun=dry_run)
39+
except botocore.exceptions.ClientError as e:
40+
if "DryRunOperation" in str(e):
41+
logger.info(f"Would have deleted image {image['ImageId']}")
42+
else:
43+
raise
44+
assert "BlockDeviceMappings" in image
45+
assert "Ebs" in image["BlockDeviceMappings"][0]
46+
assert "SnapshotId" in image["BlockDeviceMappings"][0]["Ebs"]
47+
snapshot_id = image["BlockDeviceMappings"][0]["Ebs"]["SnapshotId"]
48+
logger.info(f"Deleting snapshot {snapshot_id}")
49+
try:
50+
ec2.delete_snapshot(SnapshotId=snapshot_id, DryRun=dry_run)
51+
except botocore.exceptions.ClientError as e:
52+
if "DryRunOperation" in str(e):
53+
logger.info(f"Would have deleted snapshot {snapshot_id}")
54+
else:
55+
raise
56+
57+
58+
def main() -> None:
59+
parser = argparse.ArgumentParser()
60+
parser.add_argument(
61+
"--dry-run",
62+
action="store_true",
63+
help="Do not actually delete anything, just log what would be deleted",
64+
)
65+
logging.basicConfig(level=logging.INFO)
66+
ec2: EC2Client = boto3.client("ec2")
67+
68+
args = parser.parse_args()
69+
regions = ec2.describe_regions()["Regions"]
70+
for region in regions:
71+
assert "RegionName" in region
72+
ec2r = boto3.client("ec2", region_name=region["RegionName"])
73+
logging.info(f"Checking region {region['RegionName']}")
74+
delete_deprecated_images(ec2r, args.dry_run)
75+
76+
77+
if __name__ == "__main__":
78+
main()

0 commit comments

Comments
 (0)