1 { config, lib, pkgs, ... }:
6 cfg = config.services.mirakurun;
7 mirakurun = pkgs.mirakurun;
8 username = config.users.users.mirakurun.name;
9 groupname = config.users.users.mirakurun.group;
10 settingsFmt = pkgs.formats.yaml {};
12 polkitRule = pkgs.writeTextDir "share/polkit-1/rules.d/10-mirakurun.rules" ''
13 polkit.addRule(function (action, subject) {
15 (action.id == "org.debian.pcsc-lite.access_pcsc" ||
16 action.id == "org.debian.pcsc-lite.access_card") &&
17 subject.user == "${username}"
19 return polkit.Result.YES;
26 services.mirakurun = {
27 enable = mkEnableOption (lib.mdDoc "the Mirakurun DVR Tuner Server");
30 type = with types; nullOr port;
32 description = lib.mdDoc ''
33 Port to listen on. If `null`, it won't listen on
38 openFirewall = mkOption {
41 description = lib.mdDoc ''
42 Open ports in the firewall for Mirakurun.
45 Exposing Mirakurun to the open internet is generally advised
46 against. Only use it inside a trusted local network, or
47 consider putting it behind a VPN if you want remote access.
52 unixSocket = mkOption {
53 type = with types; nullOr path;
54 default = "/var/run/mirakurun/mirakurun.sock";
55 description = lib.mdDoc ''
56 Path to unix socket to listen on. If `null`, it
57 won't listen on any unix sockets.
61 allowSmartCardAccess = mkOption {
64 description = lib.mdDoc ''
65 Install polkit rules to allow Mirakurun to access smart card readers
66 which is commonly used along with tuner devices.
70 serverSettings = mkOption {
71 type = settingsFmt.type;
73 example = literalExpression ''
75 highWaterMark = 25165824;
76 overflowTimeLimit = 30000;
79 description = lib.mdDoc ''
80 Options for server.yml.
83 <https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md>
87 tunerSettings = mkOption {
88 type = with types; nullOr settingsFmt.type;
90 example = literalExpression ''
94 types = [ "GR" "BS" "CS" "SKY" ];
95 dvbDevicePath = "/dev/dvb/adapterX/dvrX";
99 description = lib.mdDoc ''
100 Options which are added to tuners.yml. If none is specified, it will
101 automatically be generated at runtime.
104 <https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md>
108 channelSettings = mkOption {
109 type = with types; nullOr settingsFmt.type;
111 example = literalExpression ''
120 description = lib.mdDoc ''
121 Options which are added to channels.yml. If none is specified, it
122 will automatically be generated at runtime.
125 <https://github.com/Chinachu/Mirakurun/blob/master/doc/Configuration.md>
131 config = mkIf cfg.enable {
132 environment.systemPackages = [ mirakurun ] ++ optional cfg.allowSmartCardAccess polkitRule;
134 "mirakurun/server.yml".source = settingsFmt.generate "server.yml" cfg.serverSettings;
135 "mirakurun/tuners.yml" = mkIf (cfg.tunerSettings != null) {
136 source = settingsFmt.generate "tuners.yml" cfg.tunerSettings;
141 "mirakurun/channels.yml" = mkIf (cfg.channelSettings != null) {
142 source = settingsFmt.generate "channels.yml" cfg.channelSettings;
149 networking.firewall = mkIf cfg.openFirewall {
150 allowedTCPPorts = mkIf (cfg.port != null) [ cfg.port ];
153 users.users.mirakurun = {
154 description = "Mirakurun user";
159 services.mirakurun.serverSettings = {
160 logLevel = mkDefault 2;
161 path = mkIf (cfg.unixSocket != null) cfg.unixSocket;
162 port = mkIf (cfg.port != null) cfg.port;
165 systemd.tmpfiles.rules = [
166 "d '/etc/mirakurun' - ${username} ${groupname} - -"
169 systemd.services.mirakurun = {
170 description = mirakurun.meta.description;
171 wantedBy = [ "multi-user.target" ];
172 after = [ "network.target" ];
174 ExecStart = "${mirakurun}/bin/mirakurun-start";
177 RuntimeDirectory="mirakurun";
178 StateDirectory="mirakurun";
180 IOSchedulingClass = "realtime";
181 IOSchedulingPriority = 7;
185 SERVER_CONFIG_PATH = "/etc/mirakurun/server.yml";
186 TUNERS_CONFIG_PATH = "/etc/mirakurun/tuners.yml";
187 CHANNELS_CONFIG_PATH = "/etc/mirakurun/channels.yml";
188 SERVICES_DB_PATH = "/var/lib/mirakurun/services.json";
189 PROGRAMS_DB_PATH = "/var/lib/mirakurun/programs.json";
190 LOGO_DATA_DIR_PATH = "/var/lib/mirakurun/logos";
191 NODE_ENV = "production";
194 restartTriggers = let
195 getconf = target: config.environment.etc."mirakurun/${target}.yml".source;
198 ] ++ optional (cfg.tunerSettings != null) "tuners"
199 ++ optional (cfg.channelSettings != null) "channels";
200 in (map getconf targets);