504525a148
GitOrigin-RevId: bd645e8668ec6612439a9ee7e71f7eac4099d4f6
184 lines
5.3 KiB
Nix
184 lines
5.3 KiB
Nix
{ config, lib, pkgs, ... }:
|
|
|
|
with lib;
|
|
|
|
let
|
|
cfg = config.services.grocy;
|
|
in {
|
|
options.services.grocy = {
|
|
enable = mkEnableOption (lib.mdDoc "grocy");
|
|
|
|
package = mkPackageOption pkgs "grocy" { };
|
|
|
|
hostName = mkOption {
|
|
type = types.str;
|
|
description = lib.mdDoc ''
|
|
FQDN for the grocy instance.
|
|
'';
|
|
};
|
|
|
|
nginx.enableSSL = mkOption {
|
|
type = types.bool;
|
|
default = true;
|
|
description = lib.mdDoc ''
|
|
Whether or not to enable SSL (with ACME and let's encrypt)
|
|
for the grocy vhost.
|
|
'';
|
|
};
|
|
|
|
phpfpm.settings = mkOption {
|
|
type = with 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 = lib.mdDoc ''
|
|
Options for grocy's PHPFPM pool.
|
|
'';
|
|
};
|
|
|
|
dataDir = mkOption {
|
|
type = types.str;
|
|
default = "/var/lib/grocy";
|
|
description = lib.mdDoc ''
|
|
Home directory of the `grocy` user which contains
|
|
the application's state.
|
|
'';
|
|
};
|
|
|
|
settings = {
|
|
currency = mkOption {
|
|
type = types.str;
|
|
default = "USD";
|
|
example = "EUR";
|
|
description = lib.mdDoc ''
|
|
ISO 4217 code for the currency to display.
|
|
'';
|
|
};
|
|
|
|
culture = mkOption {
|
|
type = types.enum [ "de" "en" "da" "en_GB" "es" "fr" "hu" "it" "nl" "no" "pl" "pt_BR" "ru" "sk_SK" "sv_SE" "tr" ];
|
|
default = "en";
|
|
description = lib.mdDoc ''
|
|
Display language of the frontend.
|
|
'';
|
|
};
|
|
|
|
calendar = {
|
|
showWeekNumber = mkOption {
|
|
default = true;
|
|
type = types.bool;
|
|
description = lib.mdDoc ''
|
|
Show the number of the weeks in the calendar views.
|
|
'';
|
|
};
|
|
firstDayOfWeek = mkOption {
|
|
default = null;
|
|
type = types.nullOr (types.enum (range 0 6));
|
|
description = lib.mdDoc ''
|
|
Which day of the week (0=Sunday, 1=Monday etc.) should be the
|
|
first day.
|
|
'';
|
|
};
|
|
};
|
|
};
|
|
};
|
|
|
|
config = mkIf cfg.enable {
|
|
environment.etc."grocy/config.php".text = ''
|
|
<?php
|
|
Setting('CULTURE', '${cfg.settings.culture}');
|
|
Setting('CURRENCY', '${cfg.settings.currency}');
|
|
Setting('CALENDAR_FIRST_DAY_OF_WEEK', '${toString cfg.settings.calendar.firstDayOfWeek}');
|
|
Setting('CALENDAR_SHOW_WEEK_OF_YEAR', ${boolToString cfg.settings.calendar.showWeekNumber});
|
|
'';
|
|
|
|
users.users.grocy = {
|
|
isSystemUser = true;
|
|
createHome = true;
|
|
home = cfg.dataDir;
|
|
group = "nginx";
|
|
};
|
|
|
|
systemd.tmpfiles.rules = map (
|
|
dirName: "d '${cfg.dataDir}/${dirName}' - grocy nginx - -"
|
|
) [ "viewcache" "plugins" "settingoverrides" "storage" ];
|
|
|
|
services.phpfpm.pools.grocy = {
|
|
user = "grocy";
|
|
group = "nginx";
|
|
|
|
# PHP 8.1 and 8.2 are the only version which are supported/tested by upstream:
|
|
# https://github.com/grocy/grocy/blob/v4.0.2/README.md#platform-support
|
|
phpPackage = pkgs.php82;
|
|
|
|
inherit (cfg.phpfpm) settings;
|
|
|
|
phpEnv = {
|
|
GROCY_CONFIG_FILE = "/etc/grocy/config.php";
|
|
GROCY_DB_FILE = "${cfg.dataDir}/grocy.db";
|
|
GROCY_STORAGE_DIR = "${cfg.dataDir}/storage";
|
|
GROCY_PLUGIN_DIR = "${cfg.dataDir}/plugins";
|
|
GROCY_CACHE_DIR = "${cfg.dataDir}/viewcache";
|
|
};
|
|
};
|
|
|
|
# After an update of grocy, the viewcache needs to be deleted. Otherwise grocy will not work
|
|
# https://github.com/grocy/grocy#how-to-update
|
|
systemd.services.grocy-setup = {
|
|
wantedBy = [ "multi-user.target" ];
|
|
before = [ "phpfpm-grocy.service" ];
|
|
script = ''
|
|
rm -rf ${cfg.dataDir}/viewcache/*
|
|
'';
|
|
};
|
|
|
|
services.nginx = {
|
|
enable = true;
|
|
virtualHosts."${cfg.hostName}" = mkMerge [
|
|
{ root = "${cfg.package}/public";
|
|
locations."/".extraConfig = ''
|
|
rewrite ^ /index.php;
|
|
'';
|
|
locations."~ \\.php$".extraConfig = ''
|
|
fastcgi_split_path_info ^(.+\.php)(/.+)$;
|
|
fastcgi_pass unix:${config.services.phpfpm.pools.grocy.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;
|
|
'';
|
|
}
|
|
(mkIf cfg.nginx.enableSSL {
|
|
enableACME = true;
|
|
forceSSL = true;
|
|
})
|
|
];
|
|
};
|
|
};
|
|
|
|
meta = {
|
|
maintainers = with maintainers; [ n0emis ];
|
|
doc = ./grocy.md;
|
|
};
|
|
}
|