Commit 77520d39 authored by Victor Freire's avatar Victor Freire Committed by Anderson Torres
Browse files

nixos/legit: init

parent 3e5995d7
Loading
Loading
Loading
Loading
+1 −0
Original line number Diff line number Diff line
@@ -911,6 +911,7 @@
  ./services/networking/knot.nix
  ./services/networking/kresd.nix
  ./services/networking/lambdabot.nix
  ./services/networking/legit.nix
  ./services/networking/libreswan.nix
  ./services/networking/lldpd.nix
  ./services/networking/logmein-hamachi.nix
+182 −0
Original line number Diff line number Diff line
{ config, lib, pkgs, ... }:

let
  inherit (lib)
    literalExpression
    mkEnableOption
    mdDoc
    mkIf
    mkOption
    mkPackageOptionMD
    optionalAttrs
    optional
    types;

  cfg = config.services.legit;

  yaml = pkgs.formats.yaml { };
  configFile = yaml.generate "legit.yaml" cfg.settings;

  defaultStateDir = "/var/lib/legit";
  defaultStaticDir = "${cfg.settings.repo.scanPath}/static";
  defaultTemplatesDir = "${cfg.settings.repo.scanPath}/templates";
in
{
  options.services.legit = {
    enable = mkEnableOption (mdDoc "legit git web frontend");

    package = mkPackageOptionMD pkgs "legit-web" { };

    user = mkOption {
      type = types.str;
      default = "legit";
      description = mdDoc "User account under which legit runs.";
    };

    group = mkOption {
      type = types.str;
      default = "legit";
      description = mdDoc "Group account under which legit runs.";
    };

    settings = mkOption {
      default = { };
      description = mdDoc ''
        The primary legit configuration. See the
        [sample configuration](https://github.com/icyphox/legit/blob/master/config.yaml)
        for possible values.
      '';
      type = types.submodule {
        options.repo = {
          scanPath = mkOption {
            type = types.path;
            default = defaultStateDir;
            description = mdDoc "Directory where legit will scan for repositories.";
          };
          readme = mkOption {
            type = types.listOf types.str;
            default = [ ];
            description = mdDoc "Readme files to look for.";
          };
          mainBranch = mkOption {
            type = types.listOf types.str;
            default = [ "main" "master" ];
            description = mdDoc "Main branch to look for.";
          };
          ignore = mkOption {
            type = types.listOf types.str;
            default = [ ];
            description = mdDoc "Repositories to ignore.";
          };
        };
        options.dirs = {
          templates = mkOption {
            type = types.path;
            default = "${pkgs.legit-web}/lib/legit/templates";
            defaultText = literalExpression ''"''${pkgs.legit-web}/lib/legit/templates"'';
            description = mdDoc "Directories where template files are located.";
          };
          static = mkOption {
            type = types.path;
            default = "${pkgs.legit-web}/lib/legit/static";
            defaultText = literalExpression ''"''${pkgs.legit-web}/lib/legit/static"'';
            description = mdDoc "Directories where static files are located.";
          };
        };
        options.meta = {
          title = mkOption {
            type = types.str;
            default = "legit";
            description = mdDoc "Website title.";
          };
          description = mkOption {
            type = types.str;
            default = "git frontend";
            description = mdDoc "Website description.";
          };
        };
        options.server = {
          name = mkOption {
            type = types.str;
            default = "localhost";
            description = mdDoc "Server name.";
          };
          host = mkOption {
            type = types.str;
            default = "127.0.0.1";
            description = mdDoc "Host address.";
          };
          port = mkOption {
            type = types.port;
            default = 5555;
            description = mdDoc "Legit port.";
          };
        };
      };
    };
  };

  config = mkIf cfg.enable {
    users.groups = optionalAttrs (cfg.group == "legit") {
      "${cfg.group}" = { };
    };

    users.users = optionalAttrs (cfg.user == "legit") {
      "${cfg.user}" = {
        group = cfg.group;
        isSystemUser = true;
      };
    };

    systemd.services.legit = {
      description = "legit git frontend";

      after = [ "network.target" ];
      wantedBy = [ "multi-user.target" ];
      restartTriggers = [ configFile ];

      serviceConfig = {
        Type = "simple";
        User = cfg.user;
        Group = cfg.group;
        ExecStart = "${cfg.package}/bin/legit -config ${configFile}";
        Restart = "always";

        WorkingDirectory = cfg.settings.repo.scanPath;
        StateDirectory = [ ] ++
          optional (cfg.settings.repo.scanPath == defaultStateDir) "legit" ++
          optional (cfg.settings.dirs.static == defaultStaticDir) "legit/static" ++
          optional (cfg.settings.dirs.templates == defaultTemplatesDir) "legit/templates";

        # Hardening
        CapabilityBoundingSet = [ "" ];
        DeviceAllow = [ "" ];
        LockPersonality = true;
        MemoryDenyWriteExecute = true;
        NoNewPrivileges = true;
        PrivateDevices = true;
        PrivateTmp = true;
        PrivateUsers = true;
        ProcSubset = "pid";
        ProtectClock = true;
        ProtectControlGroups = true;
        ProtectHome = true;
        ProtectHostname = true;
        ProtectKernelLogs = true;
        ProtectKernelModules = true;
        ProtectKernelTunables = true;
        ProtectProc = "invisible";
        ProtectSystem = "strict";
        ReadWritePaths = cfg.settings.repo.scanPath;
        RemoveIPC = true;
        RestrictAddressFamilies = [ "AF_INET" "AF_INET6" ];
        RestrictNamespaces = true;
        RestrictRealtime = true;
        RestrictSUIDSGID = true;
        SystemCallArchitectures = "native";
        SystemCallFilter = [ "@system-service" "~@privileged" ];
        UMask = "0077";
      };
    };
  };
}