{ config, lib, pkgs, ...}: with lib; let cfgs = config.services.cgit; settingType = with types; oneOf [ bool int str ]; repeatedSettingType = with types; oneOf [ settingType (listOf settingType) ]; genAttrs' = names: f: listToAttrs (map f names); regexEscape = let # taken from https://github.com/python/cpython/blob/05cb728d68a278d11466f9a6c8258d914135c96c/Lib/re.py#L251-L266 special = [ "(" ")" "[" "]" "{" "}" "?" "*" "+" "-" "|" "^" "$" "\\" "." "&" "~" "#" " " "\t" "\n" "\r" " " # \v / 0x0B " " # \f / 0x0C ]; in replaceStrings special (map (c: "\\${c}") special); stripLocation = cfg: removeSuffix "/" cfg.nginx.location; regexLocation = cfg: regexEscape (stripLocation cfg); mkFastcgiPass = name: cfg: '' ${if cfg.nginx.location == "/" then '' fastcgi_param PATH_INFO $uri; '' else '' fastcgi_split_path_info ^(${regexLocation cfg})(/.+)$; fastcgi_param PATH_INFO $fastcgi_path_info; '' }fastcgi_pass unix:${config.services.fcgiwrap.instances."cgit-${name}".socket.address}; ''; cgitrcLine = name: value: "${name}=${ if value == true then "1" else if value == false then "0" else toString value }"; # list value as multiple lines (for "readme" for example) cgitrcEntry = name: value: if isList value then map (cgitrcLine name) value else [ (cgitrcLine name value) ]; mkCgitrc = cfg: pkgs.writeText "cgitrc" '' # global settings ${concatStringsSep "\n" ( flatten (mapAttrsToList cgitrcEntry ({ virtual-root = cfg.nginx.location; } // cfg.settings) ) ) } ${optionalString (cfg.scanPath != null) (cgitrcLine "scan-path" cfg.scanPath)} # repository settings ${concatStrings ( mapAttrsToList (url: settings: '' ${cgitrcLine "repo.url" url} ${concatStringsSep "\n" ( mapAttrsToList (name: cgitrcLine "repo.${name}") settings ) } '') cfg.repos ) } # extra config ${cfg.extraConfig} ''; fcgiwrapUnitName = name: "fcgiwrap-cgit-${name}"; fcgiwrapRuntimeDir = name: "/run/${fcgiwrapUnitName name}"; gitProjectRoot = name: cfg: if cfg.scanPath != null then cfg.scanPath else "${fcgiwrapRuntimeDir name}/repos"; in { options = { services.cgit = mkOption { description = "Configure cgit instances."; default = {}; type = types.attrsOf (types.submodule ({ config, ... }: { options = { enable = mkEnableOption "cgit"; package = mkPackageOption pkgs "cgit" {}; nginx.virtualHost = mkOption { description = "VirtualHost to serve cgit on, defaults to the attribute name."; type = types.str; default = config._module.args.name; example = "git.example.com"; }; nginx.location = mkOption { description = "Location to serve cgit under."; type = types.str; default = "/"; example = "/git/"; }; repos = mkOption { description = "cgit repository settings, see cgitrc(5)"; type = with types; attrsOf (attrsOf settingType); default = {}; example = { blah = { path = "/var/lib/git/example"; desc = "An example repository"; }; }; }; scanPath = mkOption { description = "A path which will be scanned for repositories."; type = types.nullOr types.path; default = null; example = "/var/lib/git"; }; settings = mkOption { description = "cgit configuration, see cgitrc(5)"; type = types.attrsOf repeatedSettingType; default = {}; example = literalExpression '' { enable-follow-links = true; source-filter = "''${pkgs.cgit}/lib/cgit/filters/syntax-highlighting.py"; } ''; }; extraConfig = mkOption { description = "These lines go to the end of cgitrc verbatim."; type = types.lines; default = ""; }; user = mkOption { description = "User to run the cgit service as."; type = types.str; default = "cgit"; }; group = mkOption { description = "Group to run the cgit service as."; type = types.str; default = "cgit"; }; }; })); }; }; config = mkIf (any (cfg: cfg.enable) (attrValues cfgs)) { assertions = mapAttrsToList (vhost: cfg: { assertion = !cfg.enable || (cfg.scanPath == null) != (cfg.repos == {}); message = "Exactly one of services.cgit.${vhost}.scanPath or services.cgit.${vhost}.repos must be set."; }) cfgs; users = mkMerge (flip mapAttrsToList cfgs (_: cfg: { users.${cfg.user} = { isSystemUser = true; inherit (cfg) group; }; groups.${cfg.group} = { }; })); services.fcgiwrap.instances = flip mapAttrs' cfgs (name: cfg: nameValuePair "cgit-${name}" { process = { inherit (cfg) user group; }; socket = { inherit (config.services.nginx) user group; }; } ); systemd.services = flip mapAttrs' cfgs (name: cfg: nameValuePair (fcgiwrapUnitName name) (mkIf (cfg.repos != { }) { serviceConfig.RuntimeDirectory = fcgiwrapUnitName name; preStart = '' GIT_PROJECT_ROOT=${escapeShellArg (gitProjectRoot name cfg)} mkdir -p "$GIT_PROJECT_ROOT" cd "$GIT_PROJECT_ROOT" ${concatLines (flip mapAttrsToList cfg.repos (name: repo: '' ln -s ${escapeShellArg repo.path} ${escapeShellArg name} ''))} ''; } )); services.nginx.enable = true; services.nginx.virtualHosts = mkMerge (mapAttrsToList (name: cfg: { ${cfg.nginx.virtualHost} = { locations = ( genAttrs' [ "cgit.css" "cgit.png" "favicon.ico" "robots.txt" ] (fileName: nameValuePair "= ${stripLocation cfg}/${fileName}" { extraConfig = '' alias ${cfg.package}/cgit/${fileName}; ''; }) ) // { "~ ${regexLocation cfg}/.+/(info/refs|git-upload-pack)" = { fastcgiParams = rec { SCRIPT_FILENAME = "${pkgs.git}/libexec/git-core/git-http-backend"; GIT_HTTP_EXPORT_ALL = "1"; GIT_PROJECT_ROOT = gitProjectRoot name cfg; HOME = GIT_PROJECT_ROOT; }; extraConfig = mkFastcgiPass name cfg; }; "${stripLocation cfg}/" = { fastcgiParams = { SCRIPT_FILENAME = "${cfg.package}/cgit/cgit.cgi"; QUERY_STRING = "$args"; HTTP_HOST = "$server_name"; CGIT_CONFIG = mkCgitrc cfg; }; extraConfig = mkFastcgiPass name cfg; }; }; }; }) cfgs); }; }