2020-08-10 16:39:47 +02:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.mirakurun;
|
|
|
|
mirakurun = pkgs.mirakurun;
|
|
|
|
username = config.users.users.mirakurun.name;
|
|
|
|
groupname = config.users.users.mirakurun.group;
|
|
|
|
settingsFmt = pkgs.formats.yaml {};
|
2021-05-12 19:57:49 +02:00
|
|
|
|
|
|
|
polkitRule = pkgs.writeTextDir "share/polkit-1/rules.d/10-mirakurun.rules" ''
|
|
|
|
polkit.addRule(function (action, subject) {
|
|
|
|
if (
|
|
|
|
(action.id == "org.debian.pcsc-lite.access_pcsc" ||
|
|
|
|
action.id == "org.debian.pcsc-lite.access_card") &&
|
|
|
|
subject.user == "${username}"
|
|
|
|
) {
|
|
|
|
return polkit.Result.YES;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
'';
|
2020-08-10 16:39:47 +02:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options = {
|
|
|
|
services.mirakurun = {
|
2021-11-26 01:16:07 +01:00
|
|
|
enable = mkEnableOption "the Mirakurun DVR Tuner Server";
|
2020-08-10 16:39:47 +02:00
|
|
|
|
|
|
|
port = mkOption {
|
|
|
|
type = with types; nullOr port;
|
|
|
|
default = 40772;
|
|
|
|
description = ''
|
2020-08-24 15:07:08 +02:00
|
|
|
Port to listen on. If <literal>null</literal>, it won't listen on
|
|
|
|
any port.
|
2020-08-10 16:39:47 +02:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
openFirewall = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Open ports in the firewall for Mirakurun.
|
2020-08-24 15:07:08 +02:00
|
|
|
|
|
|
|
<warning>
|
|
|
|
<para>
|
|
|
|
Exposing Mirakurun to the open internet is generally advised
|
|
|
|
against. Only use it inside a trusted local network, or
|
|
|
|
consider putting it behind a VPN if you want remote access.
|
|
|
|
</para>
|
|
|
|
</warning>
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
unixSocket = mkOption {
|
|
|
|
type = with types; nullOr path;
|
|
|
|
default = "/var/run/mirakurun/mirakurun.sock";
|
|
|
|
description = ''
|
|
|
|
Path to unix socket to listen on. If <literal>null</literal>, it
|
|
|
|
won't listen on any unix sockets.
|
2020-08-10 16:39:47 +02:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2021-05-12 19:57:49 +02:00
|
|
|
allowSmartCardAccess = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = true;
|
|
|
|
description = ''
|
|
|
|
Install polkit rules to allow Mirakurun to access smart card readers
|
|
|
|
which is commonly used along with tuner devices.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2020-08-10 16:39:47 +02:00
|
|
|
serverSettings = mkOption {
|
|
|
|
type = settingsFmt.type;
|
|
|
|
default = {};
|
2021-10-03 18:06:03 +02:00
|
|
|
example = literalExpression ''
|
2020-08-10 16:39:47 +02:00
|
|
|
{
|
|
|
|
highWaterMark = 25165824;
|
|
|
|
overflowTimeLimit = 30000;
|
|
|
|
};
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Options for server.yml.
|
|
|
|
|
|
|
|
Documentation:
|
|
|
|
<link xlink:href="https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md"/>
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
tunerSettings = mkOption {
|
|
|
|
type = with types; nullOr settingsFmt.type;
|
|
|
|
default = null;
|
2021-10-03 18:06:03 +02:00
|
|
|
example = literalExpression ''
|
2020-08-10 16:39:47 +02:00
|
|
|
[
|
|
|
|
{
|
|
|
|
name = "tuner-name";
|
|
|
|
types = [ "GR" "BS" "CS" "SKY" ];
|
|
|
|
dvbDevicePath = "/dev/dvb/adapterX/dvrX";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Options which are added to tuners.yml. If none is specified, it will
|
|
|
|
automatically be generated at runtime.
|
|
|
|
|
|
|
|
Documentation:
|
|
|
|
<link xlink:href="https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md"/>
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
channelSettings = mkOption {
|
|
|
|
type = with types; nullOr settingsFmt.type;
|
|
|
|
default = null;
|
2021-10-03 18:06:03 +02:00
|
|
|
example = literalExpression ''
|
2020-08-10 16:39:47 +02:00
|
|
|
[
|
|
|
|
{
|
|
|
|
name = "channel";
|
|
|
|
types = "GR";
|
|
|
|
channel = "0";
|
|
|
|
}
|
|
|
|
];
|
|
|
|
'';
|
|
|
|
description = ''
|
|
|
|
Options which are added to channels.yml. If none is specified, it
|
|
|
|
will automatically be generated at runtime.
|
|
|
|
|
|
|
|
Documentation:
|
|
|
|
<link xlink:href="https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md"/>
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2021-05-12 19:57:49 +02:00
|
|
|
environment.systemPackages = [ mirakurun ] ++ optional cfg.allowSmartCardAccess polkitRule;
|
2020-08-10 16:39:47 +02:00
|
|
|
environment.etc = {
|
|
|
|
"mirakurun/server.yml".source = settingsFmt.generate "server.yml" cfg.serverSettings;
|
|
|
|
"mirakurun/tuners.yml" = mkIf (cfg.tunerSettings != null) {
|
|
|
|
source = settingsFmt.generate "tuners.yml" cfg.tunerSettings;
|
|
|
|
mode = "0644";
|
|
|
|
user = username;
|
|
|
|
group = groupname;
|
|
|
|
};
|
|
|
|
"mirakurun/channels.yml" = mkIf (cfg.channelSettings != null) {
|
|
|
|
source = settingsFmt.generate "channels.yml" cfg.channelSettings;
|
|
|
|
mode = "0644";
|
|
|
|
user = username;
|
|
|
|
group = groupname;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
networking.firewall = mkIf cfg.openFirewall {
|
|
|
|
allowedTCPPorts = mkIf (cfg.port != null) [ cfg.port ];
|
|
|
|
};
|
|
|
|
|
|
|
|
users.users.mirakurun = {
|
|
|
|
description = "Mirakurun user";
|
|
|
|
group = "video";
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
services.mirakurun.serverSettings = {
|
|
|
|
logLevel = mkDefault 2;
|
2020-08-24 15:07:08 +02:00
|
|
|
path = mkIf (cfg.unixSocket != null) cfg.unixSocket;
|
|
|
|
port = mkIf (cfg.port != null) cfg.port;
|
2020-08-10 16:39:47 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
systemd.tmpfiles.rules = [
|
|
|
|
"d '/etc/mirakurun' - ${username} ${groupname} - -"
|
|
|
|
];
|
|
|
|
|
|
|
|
systemd.services.mirakurun = {
|
|
|
|
description = mirakurun.meta.description;
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "network.target" ];
|
|
|
|
serviceConfig = {
|
2021-09-05 17:15:31 +02:00
|
|
|
ExecStart = "${mirakurun}/bin/mirakurun-start";
|
2020-08-10 16:39:47 +02:00
|
|
|
User = username;
|
|
|
|
Group = groupname;
|
|
|
|
RuntimeDirectory="mirakurun";
|
|
|
|
StateDirectory="mirakurun";
|
|
|
|
Nice = -10;
|
|
|
|
IOSchedulingClass = "realtime";
|
|
|
|
IOSchedulingPriority = 7;
|
|
|
|
};
|
|
|
|
|
|
|
|
environment = {
|
|
|
|
SERVER_CONFIG_PATH = "/etc/mirakurun/server.yml";
|
|
|
|
TUNERS_CONFIG_PATH = "/etc/mirakurun/tuners.yml";
|
|
|
|
CHANNELS_CONFIG_PATH = "/etc/mirakurun/channels.yml";
|
|
|
|
SERVICES_DB_PATH = "/var/lib/mirakurun/services.json";
|
|
|
|
PROGRAMS_DB_PATH = "/var/lib/mirakurun/programs.json";
|
|
|
|
NODE_ENV = "production";
|
|
|
|
};
|
|
|
|
|
|
|
|
restartTriggers = let
|
|
|
|
getconf = target: config.environment.etc."mirakurun/${target}.yml".source;
|
|
|
|
targets = [
|
|
|
|
"server"
|
|
|
|
] ++ optional (cfg.tunerSettings != null) "tuners"
|
|
|
|
++ optional (cfg.channelSettings != null) "channels";
|
|
|
|
in (map getconf targets);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|