2020-11-28 19:31:55 +00:00
|
|
|
{ config, pkgs, lib, ... }:
|
|
|
|
let
|
|
|
|
cfg = config.services.etebase-server;
|
|
|
|
|
2021-03-28 19:41:00 +01:00
|
|
|
iniFmt = pkgs.formats.ini {};
|
2020-11-28 19:31:55 +00:00
|
|
|
|
2021-03-28 19:41:00 +01:00
|
|
|
configIni = iniFmt.generate "etebase-server.ini" cfg.settings;
|
2020-11-28 19:31:55 +00:00
|
|
|
|
|
|
|
defaultUser = "etebase-server";
|
|
|
|
in
|
|
|
|
{
|
2021-03-28 19:41:00 +01:00
|
|
|
imports = [
|
2024-08-24 21:05:43 +01:00
|
|
|
(lib.mkRemovedOptionModule
|
2021-03-28 19:41:00 +01:00
|
|
|
[ "services" "etebase-server" "customIni" ]
|
|
|
|
"Set the option `services.etebase-server.settings' instead.")
|
2024-08-24 21:05:43 +01:00
|
|
|
(lib.mkRemovedOptionModule
|
2021-03-28 19:41:00 +01:00
|
|
|
[ "services" "etebase-server" "database" ]
|
|
|
|
"Set the option `services.etebase-server.settings.database' instead.")
|
2024-08-24 21:05:43 +01:00
|
|
|
(lib.mkRenamedOptionModule
|
2021-03-28 19:41:00 +01:00
|
|
|
[ "services" "etebase-server" "secretFile" ]
|
|
|
|
[ "services" "etebase-server" "settings" "secret_file" ])
|
2024-08-24 21:05:43 +01:00
|
|
|
(lib.mkRenamedOptionModule
|
2021-03-28 19:41:00 +01:00
|
|
|
[ "services" "etebase-server" "host" ]
|
|
|
|
[ "services" "etebase-server" "settings" "allowed_hosts" "allowed_host1" ])
|
|
|
|
];
|
|
|
|
|
2020-11-28 19:31:55 +00:00
|
|
|
options = {
|
|
|
|
services.etebase-server = {
|
2024-08-24 21:05:43 +01:00
|
|
|
enable = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = false;
|
|
|
|
example = true;
|
|
|
|
description = ''
|
|
|
|
Whether to enable the Etebase server.
|
|
|
|
|
2021-03-28 19:41:00 +01:00
|
|
|
Once enabled you need to create an admin user by invoking the
|
|
|
|
shell command `etebase-server createsuperuser` with
|
|
|
|
the user specified by the `user` option or a superuser.
|
2020-11-28 19:31:55 +00:00
|
|
|
Then you can login and create accounts on your-etebase-server.com/admin
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
package = lib.mkOption {
|
|
|
|
type = lib.types.package;
|
2024-07-11 23:35:53 +01:00
|
|
|
default = pkgs.etebase-server;
|
2024-08-24 21:05:43 +01:00
|
|
|
defaultText = lib.literalExpression "pkgs.python3.pkgs.etebase-server";
|
2024-01-23 22:08:51 +00:00
|
|
|
description = "etebase-server package to use.";
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
dataDir = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = "/var/lib/etebase-server";
|
|
|
|
description = "Directory to store the Etebase server data.";
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
port = lib.mkOption {
|
|
|
|
type = with lib.types; nullOr port;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = 8001;
|
|
|
|
description = "Port to listen on.";
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
openFirewall = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Whether to open ports in the firewall for the server.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
unixSocket = lib.mkOption {
|
|
|
|
type = with lib.types; nullOr str;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = null;
|
|
|
|
description = "The path to the socket to bind to.";
|
|
|
|
example = "/run/etebase-server/etebase-server.sock";
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
settings = lib.mkOption {
|
2021-03-28 19:41:00 +01:00
|
|
|
type = lib.types.submodule {
|
|
|
|
freeformType = iniFmt.type;
|
|
|
|
|
|
|
|
options = {
|
|
|
|
global = {
|
2024-08-24 21:05:43 +01:00
|
|
|
debug = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
2021-03-28 19:41:00 +01:00
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Whether to set django's DEBUG flag.
|
|
|
|
'';
|
|
|
|
};
|
2024-08-24 21:05:43 +01:00
|
|
|
secret_file = lib.mkOption {
|
|
|
|
type = with lib.types; nullOr str;
|
2021-03-28 19:41:00 +01:00
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The path to a file containing the secret
|
|
|
|
used as django's SECRET_KEY.
|
|
|
|
'';
|
|
|
|
};
|
2024-08-24 21:05:43 +01:00
|
|
|
static_root = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2021-03-28 22:50:18 +01:00
|
|
|
default = "${cfg.dataDir}/static";
|
2024-08-24 21:05:43 +01:00
|
|
|
defaultText = lib.literalExpression ''"''${config.services.etebase-server.dataDir}/static"'';
|
2021-03-28 22:50:18 +01:00
|
|
|
description = "The directory for static files.";
|
|
|
|
};
|
2024-08-24 21:05:43 +01:00
|
|
|
media_root = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2021-03-28 19:41:00 +01:00
|
|
|
default = "${cfg.dataDir}/media";
|
2024-08-24 21:05:43 +01:00
|
|
|
defaultText = lib.literalExpression ''"''${config.services.etebase-server.dataDir}/media"'';
|
2021-03-28 19:41:00 +01:00
|
|
|
description = "The media directory.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
allowed_hosts = {
|
2024-08-24 21:05:43 +01:00
|
|
|
allowed_host1 = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2021-03-28 19:41:00 +01:00
|
|
|
default = "0.0.0.0";
|
|
|
|
example = "localhost";
|
|
|
|
description = ''
|
|
|
|
The main host that is allowed access.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
database = {
|
2024-08-24 21:05:43 +01:00
|
|
|
engine = lib.mkOption {
|
|
|
|
type = lib.types.enum [ "django.db.backends.sqlite3" "django.db.backends.postgresql" ];
|
2021-03-28 19:41:00 +01:00
|
|
|
default = "django.db.backends.sqlite3";
|
|
|
|
description = "The database engine to use.";
|
|
|
|
};
|
2024-08-24 21:05:43 +01:00
|
|
|
name = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2021-03-28 19:41:00 +01:00
|
|
|
default = "${cfg.dataDir}/db.sqlite3";
|
2024-08-24 21:05:43 +01:00
|
|
|
defaultText = lib.literalExpression ''"''${config.services.etebase-server.dataDir}/db.sqlite3"'';
|
2021-03-28 19:41:00 +01:00
|
|
|
description = "The database name.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2020-11-28 19:31:55 +00:00
|
|
|
};
|
2021-03-28 19:41:00 +01:00
|
|
|
default = {};
|
2020-11-28 19:31:55 +00:00
|
|
|
description = ''
|
2022-08-13 04:15:06 +01:00
|
|
|
Configuration for `etebase-server`. Refer to
|
2021-03-28 19:41:00 +01:00
|
|
|
<https://github.com/etesync/server/blob/master/etebase-server.ini.example>
|
|
|
|
and <https://github.com/etesync/server/wiki>
|
|
|
|
for details on supported values.
|
2020-11-28 19:31:55 +00:00
|
|
|
'';
|
2021-03-28 19:41:00 +01:00
|
|
|
example = {
|
|
|
|
global = {
|
|
|
|
debug = true;
|
|
|
|
media_root = "/path/to/media";
|
|
|
|
};
|
|
|
|
allowed_hosts = {
|
|
|
|
allowed_host2 = "localhost";
|
|
|
|
};
|
|
|
|
};
|
2020-11-28 19:31:55 +00:00
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
user = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
2020-11-28 19:31:55 +00:00
|
|
|
default = defaultUser;
|
|
|
|
description = "User under which Etebase server runs.";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
config = lib.mkIf cfg.enable {
|
2020-11-28 19:31:55 +00:00
|
|
|
|
|
|
|
environment.systemPackages = with pkgs; [
|
|
|
|
(runCommand "etebase-server" {
|
2022-09-24 22:28:38 +01:00
|
|
|
nativeBuildInputs = [ makeWrapper ];
|
2020-11-28 19:31:55 +00:00
|
|
|
} ''
|
2024-03-01 21:55:27 +00:00
|
|
|
makeWrapper ${cfg.package}/bin/etebase-server \
|
2020-11-28 19:31:55 +00:00
|
|
|
$out/bin/etebase-server \
|
2024-08-24 21:05:43 +01:00
|
|
|
--chdir ${lib.escapeShellArg cfg.dataDir} \
|
2020-11-28 19:31:55 +00:00
|
|
|
--prefix ETEBASE_EASY_CONFIG_PATH : "${configIni}"
|
|
|
|
'')
|
|
|
|
];
|
|
|
|
|
|
|
|
systemd.tmpfiles.rules = [
|
|
|
|
"d '${cfg.dataDir}' - ${cfg.user} ${config.users.users.${cfg.user}.group} - -"
|
2024-03-19 08:59:57 +00:00
|
|
|
] ++ lib.optionals (cfg.unixSocket != null) [
|
2024-03-15 15:52:21 +00:00
|
|
|
"d '${builtins.dirOf cfg.unixSocket}' - ${cfg.user} ${config.users.users.${cfg.user}.group} - -"
|
2020-11-28 19:31:55 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
systemd.services.etebase-server = {
|
|
|
|
description = "An Etebase (EteSync 2.0) server";
|
|
|
|
after = [ "network.target" "systemd-tmpfiles-setup.service" ];
|
2024-03-01 21:55:27 +00:00
|
|
|
path = [ cfg.package ];
|
2020-11-28 19:31:55 +00:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
serviceConfig = {
|
|
|
|
User = cfg.user;
|
|
|
|
Restart = "always";
|
|
|
|
WorkingDirectory = cfg.dataDir;
|
|
|
|
};
|
|
|
|
environment = {
|
2021-03-28 22:50:18 +01:00
|
|
|
ETEBASE_EASY_CONFIG_PATH = configIni;
|
2024-03-01 21:55:27 +00:00
|
|
|
PYTHONPATH = cfg.package.pythonPath;
|
2020-11-28 19:31:55 +00:00
|
|
|
};
|
|
|
|
preStart = ''
|
|
|
|
# Auto-migrate on first run or if the package has changed
|
|
|
|
versionFile="${cfg.dataDir}/src-version"
|
2024-01-23 22:08:51 +00:00
|
|
|
if [[ $(cat "$versionFile" 2>/dev/null) != ${cfg.package} ]]; then
|
2022-11-08 06:54:14 +00:00
|
|
|
etebase-server migrate --no-input
|
|
|
|
etebase-server collectstatic --no-input --clear
|
2024-01-23 22:08:51 +00:00
|
|
|
echo ${cfg.package} > "$versionFile"
|
2020-11-28 19:31:55 +00:00
|
|
|
fi
|
|
|
|
'';
|
|
|
|
script =
|
|
|
|
let
|
2024-03-01 21:55:27 +00:00
|
|
|
python = cfg.package.python;
|
2020-11-28 19:31:55 +00:00
|
|
|
networking = if cfg.unixSocket != null
|
2024-03-01 21:55:27 +00:00
|
|
|
then "--uds ${cfg.unixSocket}"
|
|
|
|
else "--host 0.0.0.0 --port ${toString cfg.port}";
|
2020-11-28 19:31:55 +00:00
|
|
|
in ''
|
2024-03-01 21:55:27 +00:00
|
|
|
${python.pkgs.uvicorn}/bin/uvicorn ${networking} \
|
|
|
|
--app-dir ${cfg.package}/${cfg.package.python.sitePackages} \
|
2020-11-28 19:31:55 +00:00
|
|
|
etebase_server.asgi:application
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
users = lib.optionalAttrs (cfg.user == defaultUser) {
|
2020-11-28 19:31:55 +00:00
|
|
|
users.${defaultUser} = {
|
2021-04-15 19:45:08 +01:00
|
|
|
isSystemUser = true;
|
2020-11-28 19:31:55 +00:00
|
|
|
group = defaultUser;
|
|
|
|
home = cfg.dataDir;
|
|
|
|
};
|
|
|
|
|
|
|
|
groups.${defaultUser} = {};
|
|
|
|
};
|
|
|
|
|
2024-08-24 21:05:43 +01:00
|
|
|
networking.firewall = lib.mkIf cfg.openFirewall {
|
2020-11-28 19:31:55 +00:00
|
|
|
allowedTCPPorts = [ cfg.port ];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|