206 lines
6.4 KiB
Nix
206 lines
6.4 KiB
Nix
{ pkgs, config, secret, name, tag, nixTag, ... }:
|
|
|
|
# Based on: https://git.clan.lol/clan/clan-infra/src/branch/main/modules/web01/gitea/actions-runner.nix
|
|
|
|
let
|
|
forgejoUrl = "https://git.kempkens.io";
|
|
|
|
storeDepsBins = [
|
|
pkgs.attic-client
|
|
pkgs.coreutils
|
|
pkgs.findutils
|
|
pkgs.gnugrep
|
|
pkgs.gawk
|
|
pkgs.git
|
|
pkgs.nixVersions.stable
|
|
pkgs.nix-update
|
|
pkgs.bash
|
|
pkgs.jq
|
|
pkgs.nodejs
|
|
];
|
|
|
|
storeDeps = pkgs.runCommand "store-deps" { } ''
|
|
mkdir -p $out/bin
|
|
for dir in ${toString storeDepsBins}; do
|
|
for bin in "$dir"/bin/*; do
|
|
ln -s "$bin" "$out/bin/$(basename "$bin")"
|
|
done
|
|
done
|
|
|
|
# Add SSL CA certs
|
|
mkdir -p $out/etc/ssl/certs
|
|
cp -a "${pkgs.cacert}/etc/ssl/certs/ca-bundle.crt" $out/etc/ssl/certs/ca-bundle.crt
|
|
'';
|
|
in
|
|
{
|
|
systemd.services = {
|
|
forgejo-runner-nix-image = {
|
|
wantedBy = [ "multi-user.target" ];
|
|
after = [ "podman.service" ];
|
|
requires = [ "podman.service" ];
|
|
path = [ config.virtualisation.podman.package pkgs.gnutar pkgs.shadow pkgs.getent ];
|
|
script = ''
|
|
set -eux -o pipefail
|
|
mkdir -p etc/nix
|
|
|
|
touch etc/passwd etc/group
|
|
groupid=$(cut -d: -f3 < <(getent group nix-ci-user))
|
|
userid=$(cut -d: -f3 < <(getent passwd nix-ci-user))
|
|
groupadd --prefix $(pwd) --gid "$groupid" nix-ci-user
|
|
emptypassword='$6$1ero.LwbisiU.h3D$GGmnmECbPotJoPQ5eoSTD6tTjKnSWZcjHoVTkxFLZP17W9hRi/XkmCiAMOfWruUwy8gMjINrBMNODc7cYEo4K.'
|
|
useradd --prefix $(pwd) -p "$emptypassword" -m -d /tmp -u "$userid" -g "$groupid" -G nix-ci-user nix-ci-user
|
|
|
|
cat <<NIX_CONFIG > etc/nix/nix.conf
|
|
accept-flake-config = true
|
|
experimental-features = nix-command flakes
|
|
substituters = https://attic.cache.daniel.sx/nifoc-ci?priority=1 https://nix-community.cachix.org?priority=2 https://cache.nixos.org/
|
|
trusted-public-keys = nifoc-ci:JpD9zqVQi8JuS7B8htPDOQZh08rhInMnGFS9RVhiuwk= nix-community.cachix.org-1:mB9FSh9qf2dCimDSUo8Zy7bkq5CX+/rkCWyvRCYg3Fs= cache.nixos.org-1:6NCHdD59X431o0gWypbMrAURkbJ16ZPMQFGspcDShjY=
|
|
NIX_CONFIG
|
|
|
|
cat <<NIX_NETRC > etc/nix/netrc
|
|
${secret.forgejo_runner.netrc}
|
|
NIX_NETRC
|
|
|
|
cat <<NSSWITCH > etc/nsswitch.conf
|
|
passwd: files mymachines systemd
|
|
group: files mymachines systemd
|
|
shadow: files
|
|
|
|
hosts: files mymachines dns myhostname
|
|
networks: files
|
|
|
|
ethers: files
|
|
services: files
|
|
protocols: files
|
|
rpc: files
|
|
NSSWITCH
|
|
|
|
tar -cv . | tar -tvf -
|
|
tar -cv . | podman import --change "LABEL io.kempkens.keepImage=true" - forgejo-runner-nix
|
|
'';
|
|
serviceConfig = {
|
|
RuntimeDirectory = "forgejo-runner-nix-image";
|
|
WorkingDirectory = "/run/forgejo-runner-nix-image";
|
|
Type = "oneshot";
|
|
RemainAfterExit = true;
|
|
};
|
|
};
|
|
|
|
gitea-runner-nix = {
|
|
after = [ "forgejo-runner-nix-image.service" ];
|
|
requires = [ "forgejo-runner-nix-image.service" ];
|
|
|
|
serviceConfig = {
|
|
# Hardening (may overlap with DynamicUser=)
|
|
# The following options are only for optimizing output of systemd-analyze
|
|
AmbientCapabilities = "";
|
|
CapabilityBoundingSet = "";
|
|
# ProtectClock= adds DeviceAllow=char-rtc r
|
|
DeviceAllow = "";
|
|
NoNewPrivileges = true;
|
|
PrivateDevices = true;
|
|
PrivateMounts = true;
|
|
PrivateTmp = true;
|
|
PrivateUsers = true;
|
|
ProtectClock = true;
|
|
ProtectControlGroups = true;
|
|
ProtectHome = true;
|
|
ProtectHostname = true;
|
|
ProtectKernelLogs = true;
|
|
ProtectKernelModules = true;
|
|
ProtectKernelTunables = true;
|
|
ProtectSystem = "strict";
|
|
RemoveIPC = true;
|
|
RestrictNamespaces = true;
|
|
RestrictRealtime = true;
|
|
RestrictSUIDSGID = true;
|
|
UMask = "0066";
|
|
ProtectProc = "invisible";
|
|
SystemCallFilter = [
|
|
"~@clock"
|
|
"~@cpu-emulation"
|
|
"~@module"
|
|
"~@mount"
|
|
"~@obsolete"
|
|
"~@raw-io"
|
|
"~@reboot"
|
|
"~@swap"
|
|
# needed by go?
|
|
#"~@resources"
|
|
"~@privileged"
|
|
"~capset"
|
|
"~setdomainname"
|
|
"~sethostname"
|
|
];
|
|
RestrictAddressFamilies = [ "AF_INET" "AF_INET6" "AF_UNIX" "AF_NETLINK" ];
|
|
|
|
# Needs network access
|
|
PrivateNetwork = false;
|
|
# Cannot be true due to Node
|
|
MemoryDenyWriteExecute = false;
|
|
|
|
# The more restrictive "pid" option makes `nix` commands in CI emit
|
|
# "GC Warning: Couldn't read /proc/stat"
|
|
# You may want to set this to "pid" if not using `nix` commands
|
|
ProcSubset = "all";
|
|
# Coverage programs for compiled code such as `cargo-tarpaulin` disable
|
|
# ASLR (address space layout randomization) which requires the
|
|
# `personality` syscall
|
|
# You may want to set this to `true` if not using coverage tooling on
|
|
# compiled code
|
|
LockPersonality = false;
|
|
|
|
# Note that this has some interactions with the User setting; so you may
|
|
# want to consult the systemd docs if using both.
|
|
DynamicUser = true;
|
|
};
|
|
};
|
|
};
|
|
|
|
users.users.nix-ci-user = {
|
|
group = "nix-ci-user";
|
|
description = "Used for running nix-based CI jobs";
|
|
home = "/var/empty";
|
|
isSystemUser = true;
|
|
};
|
|
users.groups.nix-ci-user = { };
|
|
|
|
services.gitea-actions-runner = {
|
|
package = pkgs.forgejo-actions-runner;
|
|
|
|
instances = {
|
|
act = {
|
|
enable = true;
|
|
url = forgejoUrl;
|
|
|
|
inherit name;
|
|
tokenFile = config.age.secrets.forgejo-actions-token.path;
|
|
|
|
labels = [
|
|
"${tag}:docker://ghcr.io/catthehacker/ubuntu:act-latest"
|
|
];
|
|
};
|
|
|
|
nix = {
|
|
enable = true;
|
|
url = forgejoUrl;
|
|
|
|
name = "${name}-nix";
|
|
tokenFile = config.age.secrets.forgejo-actions-token.path;
|
|
|
|
labels = [
|
|
"nix-${nixTag}:docker://forgejo-runner-nix"
|
|
];
|
|
|
|
settings = {
|
|
container.options = "-e NIX_BUILD_SHELL=/bin/bash -e PAGER=cat -e PATH=/bin -e SSL_CERT_FILE=/etc/ssl/certs/ca-bundle.crt --device /dev/kvm -v /nix:/nix -v ${storeDeps}/bin:/bin -v ${storeDeps}/etc/ssl:/etc/ssl --user nix-ci-user";
|
|
container.valid_volumes = [
|
|
"/nix"
|
|
"${storeDeps}/bin"
|
|
"${storeDeps}/etc/ssl"
|
|
];
|
|
};
|
|
};
|
|
};
|
|
};
|
|
}
|