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