python3Packages.orjson: Disable failing tests on 32 bit
[NixPkgs.git] / nixos / modules / services / misc / klipper.nix
bloba2158e9461bcecca05f764411b18a92e9d0bd7c0
1 { config, lib, pkgs, ... }:
2 with lib;
3 let
4   cfg = config.services.klipper;
5   format = pkgs.formats.ini {
6     # https://github.com/NixOS/nixpkgs/pull/121613#issuecomment-885241996
7     listToValue = l:
8       if builtins.length l == 1 then generators.mkValueStringDefault { } (head l)
9       else lib.concatMapStrings (s: "\n  ${generators.mkValueStringDefault {} s}") l;
10     mkKeyValue = generators.mkKeyValueDefault { } ":";
11   };
14   ##### interface
15   options = {
16     services.klipper = {
17       enable = mkEnableOption (lib.mdDoc "Klipper, the 3D printer firmware");
19       package = mkOption {
20         type = types.package;
21         default = pkgs.klipper;
22         defaultText = literalExpression "pkgs.klipper";
23         description = lib.mdDoc "The Klipper package.";
24       };
26       inputTTY = mkOption {
27         type = types.path;
28         default = "/run/klipper/tty";
29         description = lib.mdDoc "Path of the virtual printer symlink to create.";
30       };
32       apiSocket = mkOption {
33         type = types.nullOr types.path;
34         default = "/run/klipper/api";
35         description = lib.mdDoc "Path of the API socket to create.";
36       };
38       mutableConfig = mkOption {
39         type = types.bool;
40         default = false;
41         example = true;
42         description = lib.mdDoc ''
43           Whether to copy the config to a mutable directory instead of using the one directly from the nix store.
44           This will only copy the config if the file at `services.klipper.mutableConfigPath` doesn't exist.
45         '';
46       };
48       mutableConfigFolder = mkOption {
49         type = types.path;
50         default = "/var/lib/klipper";
51         description = lib.mdDoc "Path to mutable Klipper config file.";
52       };
54       configFile = mkOption {
55         type = types.nullOr types.path;
56         default = null;
57         description = lib.mdDoc ''
58           Path to default Klipper config.
59         '';
60       };
62       octoprintIntegration = mkOption {
63         type = types.bool;
64         default = false;
65         description = lib.mdDoc "Allows Octoprint to control Klipper.";
66       };
68       user = mkOption {
69         type = types.nullOr types.str;
70         default = null;
71         description = lib.mdDoc ''
72           User account under which Klipper runs.
74           If null is specified (default), a temporary user will be created by systemd.
75         '';
76       };
78       group = mkOption {
79         type = types.nullOr types.str;
80         default = null;
81         description = lib.mdDoc ''
82           Group account under which Klipper runs.
84           If null is specified (default), a temporary user will be created by systemd.
85         '';
86       };
88       settings = mkOption {
89         type = types.nullOr format.type;
90         default = null;
91         description = lib.mdDoc ''
92           Configuration for Klipper. See the [documentation](https://www.klipper3d.org/Overview.html#configuration-and-tuning-guides)
93           for supported values.
94         '';
95       };
97       firmwares = mkOption {
98         description = lib.mdDoc "Firmwares klipper should manage";
99         default = { };
100         type = with types; attrsOf
101           (submodule {
102             options = {
103               enable = mkEnableOption (lib.mdDoc ''
104                 building of firmware and addition of klipper-flash tools for manual flashing.
105                 This will add `klipper-flash-$mcu` scripts to your environment which can be called to flash the firmware.
106               '');
107               serial = mkOption {
108                 type = types.nullOr path;
109                 description = lib.mdDoc "Path to serial port this printer is connected to. Leave `null` to derive it from `service.klipper.settings`.";
110               };
111               configFile = mkOption {
112                 type = path;
113                 description = lib.mdDoc "Path to firmware config which is generated using `klipper-genconf`";
114               };
115             };
116           });
117       };
118     };
119   };
121   ##### implementation
122   config = mkIf cfg.enable {
123     assertions = [
124       {
125         assertion = cfg.octoprintIntegration -> config.services.octoprint.enable;
126         message = "Option services.klipper.octoprintIntegration requires Octoprint to be enabled on this system. Please enable services.octoprint to use it.";
127       }
128       {
129         assertion = cfg.user != null -> cfg.group != null;
130         message = "Option services.klipper.group is not set when services.klipper.user is specified.";
131       }
132       {
133         assertion = cfg.settings != null -> foldl (a: b: a && b) true (mapAttrsToList (mcu: _: mcu != null -> (hasAttrByPath [ "${mcu}" "serial" ] cfg.settings)) cfg.firmwares);
134         message = "Option services.klipper.settings.$mcu.serial must be set when settings.klipper.firmware.$mcu is specified";
135       }
136       {
137         assertion = (cfg.configFile != null) != (cfg.settings != null);
138         message = "You need to either specify services.klipper.settings or services.klipper.defaultConfig.";
139       }
140     ];
142     environment.etc = mkIf (!cfg.mutableConfig) {
143       "klipper.cfg".source = if cfg.settings != null then format.generate "klipper.cfg" cfg.settings else cfg.configFile;
144     };
146     services.klipper = mkIf cfg.octoprintIntegration {
147       user = config.services.octoprint.user;
148       group = config.services.octoprint.group;
149     };
151     systemd.services.klipper =
152       let
153         klippyArgs = "--input-tty=${cfg.inputTTY}"
154           + optionalString (cfg.apiSocket != null) " --api-server=${cfg.apiSocket}";
155         printerConfigPath =
156           if cfg.mutableConfig
157           then cfg.mutableConfigFolder + "/printer.cfg"
158           else "/etc/klipper.cfg";
159         printerConfigFile =
160           if cfg.settings != null
161           then format.generate "klipper.cfg" cfg.settings
162           else cfg.configFile;
163       in
164       {
165         description = "Klipper 3D Printer Firmware";
166         wantedBy = [ "multi-user.target" ];
167         after = [ "network.target" ];
168         preStart = ''
169           mkdir -p ${cfg.mutableConfigFolder}
170           ${lib.optionalString (cfg.mutableConfig) ''
171             [ -e ${printerConfigPath} ] || {
172               cp ${printerConfigFile} ${printerConfigPath}
173               chmod +w ${printerConfigPath}
174             }
175           ''}
176           mkdir -p ${cfg.mutableConfigFolder}/gcodes
177         '';
179         serviceConfig = {
180           ExecStart = "${cfg.package}/lib/klipper/klippy.py ${klippyArgs} ${printerConfigPath}";
181           RuntimeDirectory = "klipper";
182           StateDirectory = "klipper";
183           SupplementaryGroups = [ "dialout" ];
184           WorkingDirectory = "${cfg.package}/lib";
185           OOMScoreAdjust = "-999";
186           CPUSchedulingPolicy = "rr";
187           CPUSchedulingPriority = 99;
188           IOSchedulingClass = "realtime";
189           IOSchedulingPriority = 0;
190           UMask = "0002";
191         } // (if cfg.user != null then {
192           Group = cfg.group;
193           User = cfg.user;
194         } else {
195           DynamicUser = true;
196           User = "klipper";
197         });
198       };
200     environment.systemPackages =
201       with pkgs;
202       let
203         default = a: b: if a != null then a else b;
204         firmwares = filterAttrs (n: v: v!= null) (mapAttrs
205           (mcu: { enable, configFile, serial }: if enable then pkgs.klipper-firmware.override {
206             mcu = lib.strings.sanitizeDerivationName mcu;
207             firmwareConfig = configFile;
208           } else null)
209           cfg.firmwares);
210         firmwareFlasher = mapAttrsToList
211           (mcu: firmware: pkgs.klipper-flash.override {
212             mcu = lib.strings.sanitizeDerivationName mcu;
213             klipper-firmware = firmware;
214             flashDevice = default cfg.firmwares."${mcu}".serial cfg.settings."${mcu}".serial;
215             firmwareConfig = cfg.firmwares."${mcu}".configFile;
216           })
217           firmwares;
218       in
219       [ klipper-genconf ] ++ firmwareFlasher ++ attrValues firmwares;
220   };
221   meta.maintainers = [
222     maintainers.cab404
223   ];