2014-04-14 15:26:48 +01:00
|
|
|
{ config, lib, pkgs, ... }:
|
2014-04-01 05:31:09 +01:00
|
|
|
|
2014-04-14 15:26:48 +01:00
|
|
|
with lib;
|
2014-04-01 05:31:09 +01:00
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.murmur;
|
2018-04-11 15:41:42 +01:00
|
|
|
forking = cfg.logFile != null;
|
2014-04-01 05:31:09 +01:00
|
|
|
configFile = pkgs.writeText "murmurd.ini" ''
|
|
|
|
database=/var/lib/murmur/murmur.sqlite
|
|
|
|
dbDriver=QSQLITE
|
|
|
|
|
|
|
|
autobanAttempts=${toString cfg.autobanAttempts}
|
|
|
|
autobanTimeframe=${toString cfg.autobanTimeframe}
|
|
|
|
autobanTime=${toString cfg.autobanTime}
|
|
|
|
|
2018-04-11 15:41:42 +01:00
|
|
|
logfile=${optionalString (cfg.logFile != null) cfg.logFile}
|
|
|
|
${optionalString forking "pidfile=/run/murmur/murmurd.pid"}
|
2014-04-01 05:31:09 +01:00
|
|
|
|
2016-09-14 00:24:25 +01:00
|
|
|
welcometext="${cfg.welcometext}"
|
2014-04-01 05:31:09 +01:00
|
|
|
port=${toString cfg.port}
|
|
|
|
|
|
|
|
${if cfg.hostName == "" then "" else "host="+cfg.hostName}
|
|
|
|
${if cfg.password == "" then "" else "serverpassword="+cfg.password}
|
|
|
|
|
|
|
|
bandwidth=${toString cfg.bandwidth}
|
|
|
|
users=${toString cfg.users}
|
|
|
|
|
|
|
|
textmessagelength=${toString cfg.textMsgLength}
|
|
|
|
imagemessagelength=${toString cfg.imgMsgLength}
|
2017-04-11 17:08:51 +01:00
|
|
|
allowhtml=${boolToString cfg.allowHtml}
|
2014-04-01 05:31:09 +01:00
|
|
|
logdays=${toString cfg.logDays}
|
2017-04-14 10:05:42 +01:00
|
|
|
bonjour=${boolToString cfg.bonjour}
|
|
|
|
sendversion=${boolToString cfg.sendVersion}
|
2014-04-01 05:31:09 +01:00
|
|
|
|
|
|
|
${if cfg.registerName == "" then "" else "registerName="+cfg.registerName}
|
|
|
|
${if cfg.registerPassword == "" then "" else "registerPassword="+cfg.registerPassword}
|
|
|
|
${if cfg.registerUrl == "" then "" else "registerUrl="+cfg.registerUrl}
|
|
|
|
${if cfg.registerHostname == "" then "" else "registerHostname="+cfg.registerHostname}
|
|
|
|
|
2017-04-14 10:05:42 +01:00
|
|
|
certrequired=${boolToString cfg.clientCertRequired}
|
2014-04-01 05:31:09 +01:00
|
|
|
${if cfg.sslCert == "" then "" else "sslCert="+cfg.sslCert}
|
|
|
|
${if cfg.sslKey == "" then "" else "sslKey="+cfg.sslKey}
|
2015-12-03 01:36:19 +00:00
|
|
|
${if cfg.sslCa == "" then "" else "sslCA="+cfg.sslCa}
|
2017-04-11 17:08:51 +01:00
|
|
|
|
2015-12-03 01:36:19 +00:00
|
|
|
${cfg.extraConfig}
|
2014-04-01 05:31:09 +01:00
|
|
|
'';
|
|
|
|
in
|
|
|
|
{
|
2019-12-10 01:51:19 +00:00
|
|
|
imports = [
|
|
|
|
(mkRenamedOptionModule [ "services" "murmur" "welcome" ] [ "services" "murmur" "welcometext" ])
|
|
|
|
(mkRemovedOptionModule [ "services" "murmur" "pidfile" ] "Hardcoded to /run/murmur/murmurd.pid now")
|
|
|
|
];
|
|
|
|
|
2014-04-01 05:31:09 +01:00
|
|
|
options = {
|
|
|
|
services.murmur = {
|
|
|
|
enable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "If enabled, start the Murmur Mumble server.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
2022-07-24 09:27:48 +01:00
|
|
|
openFirewall = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2022-07-24 09:27:48 +01:00
|
|
|
Open ports in the firewall for the Murmur Mumble server.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-04-01 05:31:09 +01:00
|
|
|
autobanAttempts = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 10;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Number of attempts a client is allowed to make in
|
2022-07-28 22:19:15 +01:00
|
|
|
`autobanTimeframe` seconds, before being
|
|
|
|
banned for `autobanTime`.
|
2014-04-01 05:31:09 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
autobanTimeframe = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 120;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Timeframe in which a client can connect without being banned
|
|
|
|
for repeated attempts (in seconds).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
autobanTime = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 300;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "The amount of time an IP ban lasts (in seconds).";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
2018-04-11 15:41:42 +01:00
|
|
|
logFile = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
|
|
|
example = "/var/log/murmur/murmurd.log";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Path to the log file for Murmur daemon. Empty means log to journald.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
2016-09-14 00:24:25 +01:00
|
|
|
welcometext = mkOption {
|
2014-04-01 05:31:09 +01:00
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Welcome message for connected clients.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
port = mkOption {
|
2021-06-18 16:30:33 +01:00
|
|
|
type = types.port;
|
2014-04-01 05:31:09 +01:00
|
|
|
default = 64738;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Ports to bind to (UDP and TCP).";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
hostName = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Host to bind to. Defaults binding on all addresses.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
2021-01-10 16:50:20 +00:00
|
|
|
package = mkOption {
|
|
|
|
type = types.package;
|
|
|
|
default = pkgs.murmur;
|
2021-10-03 17:06:03 +01:00
|
|
|
defaultText = literalExpression "pkgs.murmur";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Overridable attribute of the murmur package to use.";
|
2021-01-10 16:50:20 +00:00
|
|
|
};
|
|
|
|
|
2014-04-01 05:31:09 +01:00
|
|
|
password = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Required password to join server, if specified.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
bandwidth = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 72000;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Maximum bandwidth (in bits per second) that clients may send
|
|
|
|
speech at.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
users = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 100;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Maximum number of concurrent clients allowed.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
textMsgLength = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 5000;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Max length of text messages. Set 0 for no limit.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
imgMsgLength = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 131072;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Max length of image messages. Set 0 for no limit.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
allowHtml = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Allow HTML in client messages, comments, and channel
|
|
|
|
descriptions.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
logDays = mkOption {
|
|
|
|
type = types.int;
|
|
|
|
default = 31;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
How long to store RPC logs for in the database. Set 0 to
|
|
|
|
keep logs forever, or -1 to disable DB logging.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
bonjour = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Enable Bonjour auto-discovery, which allows clients over
|
|
|
|
your LAN to automatically discover Murmur servers.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
sendVersion = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Send Murmur version in UDP response.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
registerName = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Public server registration name, and also the name of the
|
|
|
|
Root channel. Even if you don't publicly register your
|
|
|
|
server, you probably still want to set this.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
registerPassword = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
Public server registry password, used authenticate your
|
|
|
|
server to the registry to prevent impersonation; required for
|
|
|
|
subsequent registry updates.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
registerUrl = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "URL website for your server.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
registerHostname = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-08-14 08:03:30 +01:00
|
|
|
description = lib.mdDoc ''
|
2014-04-01 05:31:09 +01:00
|
|
|
DNS hostname where your server can be reached. This is only
|
|
|
|
needed if you want your server to be accessed by its
|
|
|
|
hostname and not IP - but the name *must* resolve on the
|
|
|
|
internet properly.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
clientCertRequired = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Require clients to authenticate via certificates.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
sslCert = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Path to your SSL certificate.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
sslKey = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Path to your SSL key.";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
2015-12-03 01:36:19 +00:00
|
|
|
|
|
|
|
sslCa = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Path to your SSL CA certificate.";
|
2015-12-03 01:36:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
extraConfig = mkOption {
|
2016-10-23 18:33:41 +01:00
|
|
|
type = types.lines;
|
2015-12-03 01:36:19 +00:00
|
|
|
default = "";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Extra configuration to put into murmur.ini.";
|
2015-12-03 01:36:19 +00:00
|
|
|
};
|
2020-10-07 10:27:05 +01:00
|
|
|
|
|
|
|
environmentFile = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
|
|
|
example = "/var/lib/murmur/murmurd.env";
|
2022-08-30 13:08:50 +01:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
Environment file as defined in {manpage}`systemd.exec(5)`.
|
2020-10-07 10:27:05 +01:00
|
|
|
|
|
|
|
Secrets may be passed to the service without adding them to the world-readable
|
|
|
|
Nix store, by specifying placeholder variables as the option value in Nix and
|
|
|
|
setting these variables accordingly in the environment file.
|
|
|
|
|
2022-08-30 13:08:50 +01:00
|
|
|
```
|
2020-10-07 10:27:05 +01:00
|
|
|
# snippet of murmur-related config
|
|
|
|
services.murmur.password = "$MURMURD_PASSWORD";
|
2022-08-30 13:08:50 +01:00
|
|
|
```
|
2020-10-07 10:27:05 +01:00
|
|
|
|
2022-08-30 13:08:50 +01:00
|
|
|
```
|
2020-10-07 10:27:05 +01:00
|
|
|
# content of the environment file
|
|
|
|
MURMURD_PASSWORD=verysecretpassword
|
2022-08-30 13:08:50 +01:00
|
|
|
```
|
2020-10-07 10:27:05 +01:00
|
|
|
|
|
|
|
Note that this file needs to be available on the host on which
|
2022-08-30 13:08:50 +01:00
|
|
|
`murmur` is running.
|
2020-10-07 10:27:05 +01:00
|
|
|
'';
|
|
|
|
};
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2018-06-30 00:58:35 +01:00
|
|
|
users.users.murmur = {
|
2014-04-01 05:31:09 +01:00
|
|
|
description = "Murmur Service user";
|
|
|
|
home = "/var/lib/murmur";
|
|
|
|
createHome = true;
|
|
|
|
uid = config.ids.uids.murmur;
|
2020-10-29 09:32:04 +00:00
|
|
|
group = "murmur";
|
|
|
|
};
|
|
|
|
users.groups.murmur = {
|
|
|
|
gid = config.ids.gids.murmur;
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
|
2022-07-24 09:27:48 +01:00
|
|
|
networking.firewall = mkIf cfg.openFirewall {
|
|
|
|
allowedTCPPorts = [ cfg.port ];
|
|
|
|
allowedUDPPorts = [ cfg.port ];
|
|
|
|
};
|
|
|
|
|
2014-04-01 05:31:09 +01:00
|
|
|
systemd.services.murmur = {
|
|
|
|
description = "Murmur Chat Service";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
2022-02-01 07:07:51 +00:00
|
|
|
after = [ "network-online.target" ];
|
2020-10-07 10:27:05 +01:00
|
|
|
preStart = ''
|
|
|
|
${pkgs.envsubst}/bin/envsubst \
|
|
|
|
-o /run/murmur/murmurd.ini \
|
|
|
|
-i ${configFile}
|
|
|
|
'';
|
2014-04-01 05:31:09 +01:00
|
|
|
|
|
|
|
serviceConfig = {
|
2018-04-11 15:41:42 +01:00
|
|
|
# murmurd doesn't fork when logging to the console.
|
2020-10-07 10:27:05 +01:00
|
|
|
Type = if forking then "forking" else "simple";
|
|
|
|
PIDFile = mkIf forking "/run/murmur/murmurd.pid";
|
|
|
|
EnvironmentFile = mkIf (cfg.environmentFile != null) cfg.environmentFile;
|
2021-10-01 13:39:21 +01:00
|
|
|
ExecStart = "${cfg.package}/bin/mumble-server -ini /run/murmur/murmurd.ini";
|
2020-10-07 10:27:05 +01:00
|
|
|
Restart = "always";
|
|
|
|
RuntimeDirectory = "murmur";
|
|
|
|
RuntimeDirectoryMode = "0700";
|
|
|
|
User = "murmur";
|
2020-10-29 09:32:04 +00:00
|
|
|
Group = "murmur";
|
2014-04-01 05:31:09 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|