From 2465c7840e10fde35fe2b8ba40dadd1f756bd894 Mon Sep 17 00:00:00 2001 From: Zexin Yuan Date: Tue, 19 Nov 2024 18:20:07 +0800 Subject: [PATCH] kanboard: init at 1.2.42 --- .../manual/release-notes/rl-2505.section.md | 2 + nixos/modules/module-list.nix | 1 + nixos/modules/services/web-apps/kanboard.nix | 157 ++++++++++++++++++ nixos/tests/all-tests.nix | 1 + nixos/tests/web-apps/kanboard.nix | 28 ++++ pkgs/by-name/ka/kanboard/package.nix | 44 +++++ 6 files changed, 233 insertions(+) create mode 100644 nixos/modules/services/web-apps/kanboard.nix create mode 100644 nixos/tests/web-apps/kanboard.nix create mode 100644 pkgs/by-name/ka/kanboard/package.nix diff --git a/nixos/doc/manual/release-notes/rl-2505.section.md b/nixos/doc/manual/release-notes/rl-2505.section.md index 83ad39e636df47..2a9649268dc0de 100644 --- a/nixos/doc/manual/release-notes/rl-2505.section.md +++ b/nixos/doc/manual/release-notes/rl-2505.section.md @@ -18,6 +18,8 @@ - [agorakit](https://github.com/agorakit/agorakit), an organization tool for citizens' collectives. Available with [services.agorakit](#opt-services.agorakit.enable). +- [KanBoard](https://github.com/kanboard/kanboard), a project management tool that focuses on the Kanban methodology. Available as [services.kanboard](#opt-services.kanboard.enable). + ## Backward Incompatibilities {#sec-release-25.05-incompatibilities} diff --git a/nixos/modules/module-list.nix b/nixos/modules/module-list.nix index 8551e1b3406c5a..ff0ce4aec0fca6 100644 --- a/nixos/modules/module-list.nix +++ b/nixos/modules/module-list.nix @@ -1466,6 +1466,7 @@ ./services/web-apps/jirafeau.nix ./services/web-apps/jitsi-meet.nix ./services/web-apps/kasmweb/default.nix + ./services/web-apps/kanboard.nix ./services/web-apps/kavita.nix ./services/web-apps/keycloak.nix ./services/web-apps/kimai.nix diff --git a/nixos/modules/services/web-apps/kanboard.nix b/nixos/modules/services/web-apps/kanboard.nix new file mode 100644 index 00000000000000..acbfa444e297ce --- /dev/null +++ b/nixos/modules/services/web-apps/kanboard.nix @@ -0,0 +1,157 @@ +{ + config, + lib, + pkgs, + ... +}: + +let + cfg = config.services.kanboard; + + toStringAttrs = lib.mapAttrs (lib.const toString); +in +{ + meta.maintainers = with lib.maintainers; [ yzx9 ]; + + options.services.kanboard = { + enable = lib.mkEnableOption "Kanboard"; + + package = lib.mkPackageOption pkgs "kanboard" { }; + + dataDir = lib.mkOption { + type = lib.types.str; + default = "/var/lib/kanboard"; + description = "Default data folder for Kanboard."; + example = "/mnt/kanboard"; + }; + + user = lib.mkOption { + type = lib.types.str; + default = "kanboard"; + description = "User under which Kanboard runs."; + }; + + settings = lib.mkOption { + type = + with lib.types; + attrsOf (oneOf [ + str + int + bool + ]); + + default = { }; + + description = '' + Customize the default settings, refer to + for details on supported values. + ''; + }; + + # Nginx + domain = lib.mkOption { + type = lib.types.str; + default = "kanboard"; + description = "FQDN for the Kanboard instance."; + example = "kanboard.example.org"; + }; + nginx = lib.mkOption { + type = lib.types.nullOr ( + lib.types.submodule (import ../web-servers/nginx/vhost-options.nix { inherit config lib; }) + ); + default = { }; + description = '' + With this option, you can customize an NGINX virtual host which already + has sensible defaults for Kanboard. Set to `{ }` if you do not need any + customization for the virtual host. If enabled, then by default, the + {option}`serverName` is `''${domain}`. If this is set to null (the + default), no NGINX virtual host will be configured. + ''; + example = lib.literalExpression '' + { + enableACME = true; + forceHttps = true; + } + ''; + }; + + phpfpm.settings = lib.mkOption { + type = + with lib.types; + attrsOf (oneOf [ + int + str + bool + ]); + default = { + "pm" = "dynamic"; + "php_admin_value[error_log]" = "stderr"; + "php_admin_flag[log_errors]" = true; + "listen.owner" = "nginx"; + "catch_workers_output" = true; + "pm.max_children" = "32"; + "pm.start_servers" = "2"; + "pm.min_spare_servers" = "2"; + "pm.max_spare_servers" = "4"; + "pm.max_requests" = "500"; + }; + + description = '' + Options for kanboard's PHPFPM pool. + ''; + }; + }; + + config = lib.mkIf cfg.enable { + users.users.${cfg.user} = lib.mapAttrs (_: lib.mkDefault) { + isSystemUser = true; + group = config.services.nginx.group; + home = cfg.dataDir; + createHome = true; + }; + + services.phpfpm.pools.kanboard = { + user = cfg.user; + group = config.services.nginx.group; + + inherit (cfg.phpfpm) settings; + + phpEnv = lib.mkMerge [ + { DATA_DIR = cfg.dataDir; } + (toStringAttrs cfg.settings) + ]; + }; + + services.nginx = lib.mkIf (cfg.nginx != null) { + enable = lib.mkDefault true; + virtualHosts."${cfg.domain}" = lib.mkMerge [ + (lib.mapAttrs (_: lib.mkDefault) { + root = "${cfg.package}/share/kanboard"; + locations."/".extraConfig = '' + rewrite ^ /index.php; + ''; + locations."~ \\.php$".extraConfig = '' + fastcgi_split_path_info ^(.+\.php)(/.+)$; + fastcgi_pass unix:${config.services.phpfpm.pools.kanboard.socket}; + include ${config.services.nginx.package}/conf/fastcgi.conf; + include ${config.services.nginx.package}/conf/fastcgi_params; + ''; + locations."~ \\.(js|css|ttf|woff2?|png|jpe?g|svg)$".extraConfig = '' + add_header Cache-Control "public, max-age=15778463"; + add_header X-Content-Type-Options nosniff; + add_header X-XSS-Protection "1; mode=block"; + add_header X-Robots-Tag none; + add_header X-Download-Options noopen; + add_header X-Permitted-Cross-Domain-Policies none; + add_header Referrer-Policy no-referrer; + access_log off; + ''; + extraConfig = '' + try_files $uri /index.php; + ''; + }) + cfg.nginx + ]; + }; + }; +} diff --git a/nixos/tests/all-tests.nix b/nixos/tests/all-tests.nix index 6f941cc254ad46..1e751bf7008937 100644 --- a/nixos/tests/all-tests.nix +++ b/nixos/tests/all-tests.nix @@ -498,6 +498,7 @@ in { jotta-cli = handleTest ./jotta-cli.nix {}; k3s = handleTest ./k3s {}; kafka = handleTest ./kafka.nix {}; + kanboard = handleTest ./web-apps/kanboard.nix {}; kanidm = handleTest ./kanidm.nix {}; kanidm-provisioning = handleTest ./kanidm-provisioning.nix {}; karma = handleTest ./karma.nix {}; diff --git a/nixos/tests/web-apps/kanboard.nix b/nixos/tests/web-apps/kanboard.nix new file mode 100644 index 00000000000000..a8f8a64f2f51ed --- /dev/null +++ b/nixos/tests/web-apps/kanboard.nix @@ -0,0 +1,28 @@ +import ../make-test-python.nix ( + { pkgs, ... }: + + { + name = "kanboard"; + meta.maintainers = with pkgs.lib.maintainers; [ yzx9 ]; + + nodes = { + machine = + { ... }: + + { + services.kanboard = { + enable = true; + }; + }; + }; + + testScript = '' + start_all() + machine.wait_for_unit("nginx.service") + machine.wait_for_unit("phpfpm-kanboard.service") + machine.wait_for_open_port(80) + + machine.succeed("curl -k --fail http://localhost", timeout=10) + ''; + } +) diff --git a/pkgs/by-name/ka/kanboard/package.nix b/pkgs/by-name/ka/kanboard/package.nix new file mode 100644 index 00000000000000..df4ff66c0b0eb4 --- /dev/null +++ b/pkgs/by-name/ka/kanboard/package.nix @@ -0,0 +1,44 @@ +{ + lib, + stdenvNoCC, + fetchFromGitHub, + nixosTests, + nix-update-script, +}: + +stdenvNoCC.mkDerivation (attrs: { + pname = "kanboard"; + version = "1.2.42"; + + src = fetchFromGitHub { + owner = "kanboard"; + repo = "kanboard"; + rev = "refs/tags/v${attrs.version}"; + hash = "sha256-/Unxl9Vh9pEWjO89sSviGGPFzUwxdb1mbOfpTFTyRL0="; + }; + + dontBuild = true; + + installPhase = '' + runHook preInstall + + mkdir -p $out/share/kanboard + cp -rv . $out/share/kanboard + + runHook postInstall + ''; + + passthru = { + updateScript = nix-update-script { }; + tests = { + basic-frunctionality = nixosTests.kanboard; + }; + }; + + meta = { + description = "Kanban project management software"; + homepage = "https://kanboard.org"; + license = lib.licenses.mit; + maintainers = with lib.maintainers; [ yzx9 ]; + }; +})