2023-03-24 00:07:29 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
with lib;
|
|
|
|
let
|
|
|
|
cfg = config.services.keyd;
|
|
|
|
|
2023-07-15 17:15:38 +00:00
|
|
|
keyboardOptions = { ... }: {
|
|
|
|
options = {
|
2023-03-24 00:07:29 +00:00
|
|
|
ids = mkOption {
|
2023-08-10 07:59:29 +00:00
|
|
|
type = types.listOf types.str;
|
2023-03-24 00:07:29 +00:00
|
|
|
default = [ "*" ];
|
|
|
|
example = [ "*" "-0123:0456" ];
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Device identifiers, as shown by {manpage}`keyd(1)`.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
settings = mkOption {
|
2023-10-09 19:29:22 +00:00
|
|
|
type = (pkgs.formats.ini { }).type;
|
2023-03-24 00:07:29 +00:00
|
|
|
default = { };
|
|
|
|
example = {
|
|
|
|
main = {
|
|
|
|
capslock = "overload(control, esc)";
|
|
|
|
rightalt = "layer(rightalt)";
|
|
|
|
};
|
|
|
|
|
|
|
|
rightalt = {
|
|
|
|
j = "down";
|
|
|
|
k = "up";
|
|
|
|
h = "left";
|
|
|
|
l = "right";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
description = lib.mdDoc ''
|
2023-07-15 17:15:38 +00:00
|
|
|
Configuration, except `ids` section, that is written to {file}`/etc/keyd/<keyboard>.conf`.
|
|
|
|
Appropriate names can be used to write non-alpha keys, for example "equal" instead of "=" sign (see <https://github.com/NixOS/nixpkgs/issues/236622>).
|
2023-03-24 00:07:29 +00:00
|
|
|
See <https://github.com/rvaiya/keyd> how to configure.
|
|
|
|
'';
|
|
|
|
};
|
2023-10-09 19:29:22 +00:00
|
|
|
|
|
|
|
extraConfig = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
[control+shift]
|
|
|
|
h = left
|
|
|
|
'';
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Extra configuration that is appended to the end of the file.
|
|
|
|
**Do not** write `ids` section here, use a separate option for it.
|
|
|
|
You can use this option to define compound layers that must always be defined after the layer they are comprised.
|
|
|
|
'';
|
|
|
|
};
|
2023-03-24 00:07:29 +00:00
|
|
|
};
|
|
|
|
};
|
2023-07-15 17:15:38 +00:00
|
|
|
in
|
|
|
|
{
|
|
|
|
imports = [
|
|
|
|
(mkRemovedOptionModule [ "services" "keyd" "ids" ]
|
|
|
|
''Use keyboards.<filename>.ids instead. If you don't need a multi-file configuration, just add keyboards.default before the ids. See https://github.com/NixOS/nixpkgs/pull/243271.'')
|
|
|
|
(mkRemovedOptionModule [ "services" "keyd" "settings" ]
|
|
|
|
''Use keyboards.<filename>.settings instead. If you don't need a multi-file configuration, just add keyboards.default before the settings. See https://github.com/NixOS/nixpkgs/pull/243271.'')
|
|
|
|
];
|
|
|
|
|
|
|
|
options.services.keyd = {
|
|
|
|
enable = mkEnableOption (lib.mdDoc "keyd, a key remapping daemon");
|
|
|
|
|
|
|
|
keyboards = mkOption {
|
|
|
|
type = types.attrsOf (types.submodule keyboardOptions);
|
|
|
|
default = { };
|
|
|
|
example = literalExpression ''
|
|
|
|
{
|
|
|
|
default = {
|
|
|
|
ids = [ "*" ];
|
|
|
|
settings = {
|
|
|
|
main = {
|
|
|
|
capslock = "overload(control, esc)";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
externalKeyboard = {
|
|
|
|
ids = [ "1ea7:0907" ];
|
|
|
|
settings = {
|
|
|
|
main = {
|
|
|
|
esc = capslock;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
description = mdDoc ''
|
|
|
|
Configuration for one or more device IDs. Corresponding files in the /etc/keyd/ directory are created according to the name of the keys (like `default` or `externalKeyboard`).
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
2023-03-24 00:07:29 +00:00
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
2023-07-15 17:15:38 +00:00
|
|
|
# Creates separate files in the `/etc/keyd/` directory for each key in the dictionary
|
|
|
|
environment.etc = mapAttrs'
|
|
|
|
(name: options:
|
|
|
|
nameValuePair "keyd/${name}.conf" {
|
2023-10-09 19:29:22 +00:00
|
|
|
text = ''
|
|
|
|
[ids]
|
|
|
|
${concatStringsSep "\n" options.ids}
|
|
|
|
|
|
|
|
${generators.toINI {} options.settings}
|
|
|
|
${options.extraConfig}
|
2023-07-15 17:15:38 +00:00
|
|
|
'';
|
|
|
|
})
|
|
|
|
cfg.keyboards;
|
2023-03-24 00:07:29 +00:00
|
|
|
|
|
|
|
hardware.uinput.enable = lib.mkDefault true;
|
|
|
|
|
|
|
|
systemd.services.keyd = {
|
|
|
|
description = "Keyd remapping daemon";
|
|
|
|
documentation = [ "man:keyd(1)" ];
|
|
|
|
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
|
2023-07-15 17:15:38 +00:00
|
|
|
restartTriggers = mapAttrsToList
|
|
|
|
(name: options:
|
|
|
|
config.environment.etc."keyd/${name}.conf".source
|
|
|
|
)
|
|
|
|
cfg.keyboards;
|
2023-03-24 00:07:29 +00:00
|
|
|
|
|
|
|
# this is configurable in 2.4.2, later versions seem to remove this option.
|
|
|
|
# post-2.4.2 may need to set makeFlags in the derivation:
|
|
|
|
#
|
|
|
|
# makeFlags = [ "SOCKET_PATH/run/keyd/keyd.socket" ];
|
|
|
|
environment.KEYD_SOCKET = "/run/keyd/keyd.sock";
|
|
|
|
|
|
|
|
serviceConfig = {
|
|
|
|
ExecStart = "${pkgs.keyd}/bin/keyd";
|
|
|
|
Restart = "always";
|
|
|
|
|
2023-05-24 13:37:59 +00:00
|
|
|
# TODO investigate why it doesn't work propeprly with DynamicUser
|
|
|
|
# See issue: https://github.com/NixOS/nixpkgs/issues/226346
|
|
|
|
# DynamicUser = true;
|
2023-03-24 00:07:29 +00:00
|
|
|
SupplementaryGroups = [
|
|
|
|
config.users.groups.input.name
|
|
|
|
config.users.groups.uinput.name
|
|
|
|
];
|
|
|
|
|
|
|
|
RuntimeDirectory = "keyd";
|
|
|
|
|
|
|
|
# Hardening
|
|
|
|
CapabilityBoundingSet = "";
|
|
|
|
DeviceAllow = [
|
|
|
|
"char-input rw"
|
|
|
|
"/dev/uinput rw"
|
|
|
|
];
|
|
|
|
ProtectClock = true;
|
|
|
|
PrivateNetwork = true;
|
|
|
|
ProtectHome = true;
|
|
|
|
ProtectHostname = true;
|
|
|
|
PrivateUsers = true;
|
|
|
|
PrivateMounts = true;
|
2023-05-24 13:37:59 +00:00
|
|
|
PrivateTmp = true;
|
2023-03-24 00:07:29 +00:00
|
|
|
RestrictNamespaces = true;
|
|
|
|
ProtectKernelLogs = true;
|
|
|
|
ProtectKernelModules = true;
|
|
|
|
ProtectKernelTunables = true;
|
|
|
|
ProtectControlGroups = true;
|
|
|
|
MemoryDenyWriteExecute = true;
|
|
|
|
RestrictRealtime = true;
|
|
|
|
LockPersonality = true;
|
2023-05-24 13:37:59 +00:00
|
|
|
ProtectProc = "invisible";
|
|
|
|
SystemCallFilter = [
|
|
|
|
"@system-service"
|
|
|
|
"~@privileged"
|
|
|
|
"~@resources"
|
|
|
|
];
|
|
|
|
RestrictAddressFamilies = [ "AF_UNIX" ];
|
|
|
|
RestrictSUIDSGID = true;
|
|
|
|
IPAddressDeny = [ "any" ];
|
|
|
|
NoNewPrivileges = true;
|
|
|
|
ProtectSystem = "strict";
|
|
|
|
ProcSubset = "pid";
|
2023-03-24 00:07:29 +00:00
|
|
|
UMask = "0077";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|