342 lines
11 KiB
Nix
342 lines
11 KiB
Nix
{ config, lib, pkgs, ... }:
|
|
|
|
with lib;
|
|
|
|
let
|
|
cfg = config.services.redmine;
|
|
|
|
bundle = "${cfg.package}/share/redmine/bin/bundle";
|
|
|
|
databaseYml = pkgs.writeText "database.yml" ''
|
|
production:
|
|
adapter: ${cfg.database.type}
|
|
database: ${cfg.database.name}
|
|
host: ${cfg.database.host}
|
|
port: ${toString cfg.database.port}
|
|
username: ${cfg.database.user}
|
|
password: #dbpass#
|
|
${optionalString (cfg.database.socket != null) "socket: ${cfg.database.socket}"}
|
|
'';
|
|
|
|
configurationYml = pkgs.writeText "configuration.yml" ''
|
|
default:
|
|
scm_subversion_command: ${pkgs.subversion}/bin/svn
|
|
scm_mercurial_command: ${pkgs.mercurial}/bin/hg
|
|
scm_git_command: ${pkgs.gitAndTools.git}/bin/git
|
|
scm_cvs_command: ${pkgs.cvs}/bin/cvs
|
|
scm_bazaar_command: ${pkgs.bazaar}/bin/bzr
|
|
scm_darcs_command: ${pkgs.darcs}/bin/darcs
|
|
|
|
${cfg.extraConfig}
|
|
'';
|
|
|
|
unpackTheme = unpack "theme";
|
|
unpackPlugin = unpack "plugin";
|
|
unpack = id: (name: source:
|
|
pkgs.stdenv.mkDerivation {
|
|
name = "redmine-${id}-${name}";
|
|
buildInputs = [ pkgs.unzip ];
|
|
buildCommand = ''
|
|
mkdir -p $out
|
|
cd $out
|
|
unpackFile ${source}
|
|
'';
|
|
});
|
|
|
|
in
|
|
|
|
{
|
|
options = {
|
|
services.redmine = {
|
|
enable = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = "Enable the Redmine service.";
|
|
};
|
|
|
|
package = mkOption {
|
|
type = types.package;
|
|
default = pkgs.redmine;
|
|
defaultText = "pkgs.redmine";
|
|
description = "Which Redmine package to use.";
|
|
example = "pkgs.redmine.override { ruby = pkgs.ruby_2_3; }";
|
|
};
|
|
|
|
user = mkOption {
|
|
type = types.str;
|
|
default = "redmine";
|
|
description = "User under which Redmine is ran.";
|
|
};
|
|
|
|
group = mkOption {
|
|
type = types.str;
|
|
default = "redmine";
|
|
description = "Group under which Redmine is ran.";
|
|
};
|
|
|
|
port = mkOption {
|
|
type = types.int;
|
|
default = 3000;
|
|
description = "Port on which Redmine is ran.";
|
|
};
|
|
|
|
stateDir = mkOption {
|
|
type = types.str;
|
|
default = "/var/lib/redmine";
|
|
description = "The state directory, logs and plugins are stored here.";
|
|
};
|
|
|
|
extraConfig = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
description = ''
|
|
Extra configuration in configuration.yml.
|
|
|
|
See https://guides.rubyonrails.org/action_mailer_basics.html#action-mailer-configuration
|
|
'';
|
|
example = literalExample ''
|
|
email_delivery:
|
|
delivery_method: smtp
|
|
smtp_settings:
|
|
address: mail.example.com
|
|
port: 25
|
|
'';
|
|
};
|
|
|
|
themes = mkOption {
|
|
type = types.attrsOf types.path;
|
|
default = {};
|
|
description = "Set of themes.";
|
|
example = literalExample ''
|
|
{
|
|
dkuk-redmine_alex_skin = builtins.fetchurl {
|
|
url = https://bitbucket.org/dkuk/redmine_alex_skin/get/1842ef675ef3.zip;
|
|
sha256 = "0hrin9lzyi50k4w2bd2b30vrf1i4fi1c0gyas5801wn8i7kpm9yl";
|
|
};
|
|
}
|
|
'';
|
|
};
|
|
|
|
plugins = mkOption {
|
|
type = types.attrsOf types.path;
|
|
default = {};
|
|
description = "Set of plugins.";
|
|
example = literalExample ''
|
|
{
|
|
redmine_env_auth = builtins.fetchurl {
|
|
url = https://github.com/Intera/redmine_env_auth/archive/0.6.zip;
|
|
sha256 = "0yyr1yjd8gvvh832wdc8m3xfnhhxzk2pk3gm2psg5w9jdvd6skak";
|
|
};
|
|
}
|
|
'';
|
|
};
|
|
|
|
database = {
|
|
type = mkOption {
|
|
type = types.enum [ "mysql2" "postgresql" ];
|
|
example = "postgresql";
|
|
default = "mysql2";
|
|
description = "Database engine to use.";
|
|
};
|
|
|
|
host = mkOption {
|
|
type = types.str;
|
|
default = (if cfg.database.socket != null then "localhost" else "127.0.0.1");
|
|
description = "Database host address.";
|
|
};
|
|
|
|
port = mkOption {
|
|
type = types.int;
|
|
default = 3306;
|
|
description = "Database host port.";
|
|
};
|
|
|
|
name = mkOption {
|
|
type = types.str;
|
|
default = "redmine";
|
|
description = "Database name.";
|
|
};
|
|
|
|
user = mkOption {
|
|
type = types.str;
|
|
default = "redmine";
|
|
description = "Database user.";
|
|
};
|
|
|
|
password = mkOption {
|
|
type = types.str;
|
|
default = "";
|
|
description = ''
|
|
The password corresponding to <option>database.user</option>.
|
|
Warning: this is stored in cleartext in the Nix store!
|
|
Use <option>database.passwordFile</option> instead.
|
|
'';
|
|
};
|
|
|
|
passwordFile = mkOption {
|
|
type = types.nullOr types.path;
|
|
default = null;
|
|
example = "/run/keys/redmine-dbpassword";
|
|
description = ''
|
|
A file containing the password corresponding to
|
|
<option>database.user</option>.
|
|
'';
|
|
};
|
|
|
|
socket = mkOption {
|
|
type = types.nullOr types.path;
|
|
default = null;
|
|
example = "/run/mysqld/mysqld.sock";
|
|
description = "Path to the unix socket file to use for authentication.";
|
|
};
|
|
};
|
|
};
|
|
};
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
assertions = [
|
|
{ assertion = cfg.database.passwordFile != null || cfg.database.password != "" || cfg.database.socket != null;
|
|
message = "one of services.redmine.database.socket, services.redmine.database.passwordFile, or services.redmine.database.password must be set";
|
|
}
|
|
{ assertion = cfg.database.socket != null -> (cfg.database.type == "mysql2");
|
|
message = "Socket authentication is only available for the mysql2 database type";
|
|
}
|
|
];
|
|
|
|
environment.systemPackages = [ cfg.package ];
|
|
|
|
systemd.services.redmine = {
|
|
after = [ "network.target" (if cfg.database.type == "mysql2" then "mysql.service" else "postgresql.service") ];
|
|
wantedBy = [ "multi-user.target" ];
|
|
environment.HOME = "${cfg.package}/share/redmine";
|
|
environment.RAILS_ENV = "production";
|
|
environment.RAILS_CACHE = "${cfg.stateDir}/cache";
|
|
environment.REDMINE_LANG = "en";
|
|
environment.SCHEMA = "${cfg.stateDir}/cache/schema.db";
|
|
path = with pkgs; [
|
|
imagemagickBig
|
|
bazaar
|
|
cvs
|
|
darcs
|
|
gitAndTools.git
|
|
mercurial
|
|
subversion
|
|
];
|
|
preStart = ''
|
|
# ensure cache directory exists for db:migrate command
|
|
mkdir -p "${cfg.stateDir}/cache"
|
|
|
|
# create the basic directory layout the redmine package expects
|
|
mkdir -p /run/redmine/public
|
|
|
|
for i in config files log plugins tmp; do
|
|
mkdir -p "${cfg.stateDir}/$i"
|
|
ln -fs "${cfg.stateDir}/$i" /run/redmine/
|
|
done
|
|
|
|
for i in plugin_assets themes; do
|
|
mkdir -p "${cfg.stateDir}/public/$i"
|
|
ln -fs "${cfg.stateDir}/public/$i" /run/redmine/public/
|
|
done
|
|
|
|
|
|
# start with a fresh config directory
|
|
# the config directory is copied instead of linked as some mutable data is stored in there
|
|
rm -rf "${cfg.stateDir}/config/"*
|
|
cp -r ${cfg.package}/share/redmine/config.dist/* "${cfg.stateDir}/config/"
|
|
|
|
# link in the application configuration
|
|
ln -fs ${configurationYml} "${cfg.stateDir}/config/configuration.yml"
|
|
|
|
|
|
# link in all user specified themes
|
|
rm -rf "${cfg.stateDir}/public/themes/"*
|
|
for theme in ${concatStringsSep " " (mapAttrsToList unpackTheme cfg.themes)}; do
|
|
ln -fs $theme/* "${cfg.stateDir}/public/themes"
|
|
done
|
|
|
|
# link in redmine provided themes
|
|
ln -sf ${cfg.package}/share/redmine/public/themes.dist/* "${cfg.stateDir}/public/themes/"
|
|
|
|
|
|
# link in all user specified plugins
|
|
rm -rf "${cfg.stateDir}/plugins/"*
|
|
for plugin in ${concatStringsSep " " (mapAttrsToList unpackPlugin cfg.plugins)}; do
|
|
ln -fs $plugin/* "${cfg.stateDir}/plugins/''${plugin##*-redmine-plugin-}"
|
|
done
|
|
|
|
|
|
# ensure correct permissions for most files
|
|
chmod -R ug+rwX,o-rwx+x "${cfg.stateDir}/"
|
|
|
|
|
|
# handle database.passwordFile & permissions
|
|
DBPASS=$(head -n1 ${cfg.database.passwordFile})
|
|
cp -f ${databaseYml} "${cfg.stateDir}/config/database.yml"
|
|
sed -e "s,#dbpass#,$DBPASS,g" -i "${cfg.stateDir}/config/database.yml"
|
|
chmod 440 "${cfg.stateDir}/config/database.yml"
|
|
|
|
|
|
# generate a secret token if required
|
|
if ! test -e "${cfg.stateDir}/config/initializers/secret_token.rb"; then
|
|
${bundle} exec rake generate_secret_token
|
|
chmod 440 "${cfg.stateDir}/config/initializers/secret_token.rb"
|
|
fi
|
|
|
|
|
|
# ensure everything is owned by ${cfg.user}
|
|
chown -R ${cfg.user}:${cfg.group} "${cfg.stateDir}"
|
|
|
|
|
|
# execute redmine required commands prior to starting the application
|
|
# NOTE: su required in case using mysql socket authentication
|
|
/run/wrappers/bin/su -s ${pkgs.bash}/bin/bash -m -l redmine -c '${bundle} exec rake db:migrate'
|
|
/run/wrappers/bin/su -s ${pkgs.bash}/bin/bash -m -l redmine -c '${bundle} exec rake redmine:load_default_data'
|
|
|
|
|
|
# log files don't exist until after first command has been executed
|
|
# correct ownership of files generated by calling exec rake ...
|
|
chown -R ${cfg.user}:${cfg.group} "${cfg.stateDir}/log"
|
|
'';
|
|
|
|
serviceConfig = {
|
|
PermissionsStartOnly = true; # preStart must be run as root
|
|
Type = "simple";
|
|
User = cfg.user;
|
|
Group = cfg.group;
|
|
TimeoutSec = "300";
|
|
WorkingDirectory = "${cfg.package}/share/redmine";
|
|
ExecStart="${bundle} exec rails server webrick -e production -p ${toString cfg.port} -P '${cfg.stateDir}/redmine.pid'";
|
|
};
|
|
|
|
};
|
|
|
|
users.users = optionalAttrs (cfg.user == "redmine") (singleton
|
|
{ name = "redmine";
|
|
group = cfg.group;
|
|
home = cfg.stateDir;
|
|
createHome = true;
|
|
uid = config.ids.uids.redmine;
|
|
});
|
|
|
|
users.groups = optionalAttrs (cfg.group == "redmine") (singleton
|
|
{ name = "redmine";
|
|
gid = config.ids.gids.redmine;
|
|
});
|
|
|
|
warnings = optional (cfg.database.password != "")
|
|
''config.services.redmine.database.password will be stored as plaintext
|
|
in the Nix store. Use database.passwordFile instead.'';
|
|
|
|
# Create database passwordFile default when password is configured.
|
|
services.redmine.database.passwordFile =
|
|
(mkDefault (toString (pkgs.writeTextFile {
|
|
name = "redmine-database-password";
|
|
text = cfg.database.password;
|
|
})));
|
|
|
|
};
|
|
|
|
}
|