1 { config, lib, pkgs, ... }:
3 cfg = config.services.dockerRegistry;
5 blobCache = if cfg.enableRedisCache
11 log.fields.service = "registry";
13 cache.blobdescriptor = blobCache;
14 delete.enabled = cfg.enableDelete;
15 } // (lib.optionalAttrs (cfg.storagePath != null) { filesystem.rootdirectory = cfg.storagePath; });
17 addr = "${cfg.listenAddress}:${builtins.toString cfg.port}";
18 headers.X-Content-Type-Options = ["nosniff"];
20 health.storagedriver = {
27 registryConfig.redis = lib.mkIf cfg.enableRedisCache {
28 addr = "${cfg.redisUrl}";
29 password = "${cfg.redisPassword}";
33 writetimeout = "10ms";
41 configFile = cfg.configFile;
43 options.services.dockerRegistry = {
44 enable = lib.mkEnableOption "Docker Registry";
46 package = lib.mkPackageOption pkgs "docker-distribution" {
47 example = "gitlab-container-registry";
50 listenAddress = lib.mkOption {
51 description = "Docker registry host or ip to bind to.";
52 default = "127.0.0.1";
57 description = "Docker registry port to bind to.";
59 type = lib.types.port;
62 openFirewall = lib.mkOption {
63 type = lib.types.bool;
65 description = "Opens the port used by the firewall.";
68 storagePath = lib.mkOption {
69 type = lib.types.nullOr lib.types.path;
70 default = "/var/lib/docker-registry";
72 Docker registry storage path for the filesystem storage backend. Set to
73 null to configure another backend via extraConfig.
77 enableDelete = lib.mkOption {
78 type = lib.types.bool;
80 description = "Enable delete for manifests and blobs.";
83 enableRedisCache = lib.mkEnableOption "redis as blob cache";
85 redisUrl = lib.mkOption {
87 default = "localhost:6379";
88 description = "Set redis host and port.";
91 redisPassword = lib.mkOption {
94 description = "Set redis password.";
97 extraConfig = lib.mkOption {
99 Docker extra registry configuration.
101 example = lib.literalExpression ''
107 type = lib.types.attrs;
110 configFile = lib.mkOption {
111 default = pkgs.writeText "docker-registry-config.yml" (builtins.toJSON (lib.recursiveUpdate registryConfig cfg.extraConfig));
112 defaultText = lib.literalExpression ''pkgs.writeText "docker-registry-config.yml" "# my custom docker-registry-config.yml ..."'';
114 Path to CNCF distribution config file.
116 Setting this option will override any configuration applied by the extraConfig option.
118 type = lib.types.path;
121 enableGarbageCollect = lib.mkEnableOption "garbage collect";
123 garbageCollectDates = lib.mkOption {
125 type = lib.types.str;
127 Specification (in the format described by
128 {manpage}`systemd.time(7)`) of the time at
129 which the garbage collect will occur.
134 config = lib.mkIf cfg.enable {
135 systemd.services.docker-registry = {
136 description = "Docker Container Registry";
137 wantedBy = [ "multi-user.target" ];
138 after = [ "network.target" ];
140 ${cfg.package}/bin/registry serve ${configFile}
144 User = "docker-registry";
145 WorkingDirectory = cfg.storagePath;
146 AmbientCapabilities = lib.mkIf (cfg.port < 1024) "cap_net_bind_service";
150 systemd.services.docker-registry-garbage-collect = {
151 description = "Run Garbage Collection for docker registry";
153 restartIfChanged = false;
154 unitConfig.X-StopOnRemoval = false;
156 serviceConfig.Type = "oneshot";
159 ${cfg.package}/bin/registry garbage-collect ${configFile}
160 /run/current-system/systemd/bin/systemctl restart docker-registry.service
163 startAt = lib.optional cfg.enableGarbageCollect cfg.garbageCollectDates;
166 users.users.docker-registry =
167 (lib.optionalAttrs (cfg.storagePath != null) {
169 home = cfg.storagePath;
171 group = "docker-registry";
174 users.groups.docker-registry = {};
176 networking.firewall = lib.mkIf cfg.openFirewall {
177 allowedTCPPorts = [ cfg.port ];