2020-04-24 23:36:52 +00:00
|
|
|
{ config, lib, utils, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.networking.supplicant;
|
|
|
|
|
|
|
|
# We must escape interfaces due to the systemd interpretation
|
|
|
|
subsystemDevice = interface:
|
|
|
|
"sys-subsystem-net-devices-${utils.escapeSystemdPath interface}.device";
|
|
|
|
|
|
|
|
serviceName = iface: "supplicant-${if (iface=="WLAN") then "wlan@" else (
|
|
|
|
if (iface=="LAN") then "lan@" else (
|
|
|
|
if (iface=="DBUS") then "dbus"
|
2022-12-17 10:02:37 +00:00
|
|
|
else (replaceStrings [" "] ["-"] iface)))}";
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
# TODO: Use proper privilege separation for wpa_supplicant
|
|
|
|
supplicantService = iface: suppl:
|
|
|
|
let
|
|
|
|
deps = (if (iface=="WLAN"||iface=="LAN") then ["sys-subsystem-net-devices-%i.device"] else (
|
|
|
|
if (iface=="DBUS") then ["dbus.service"]
|
|
|
|
else (map subsystemDevice (splitString " " iface))))
|
|
|
|
++ optional (suppl.bridge!="") (subsystemDevice suppl.bridge);
|
|
|
|
|
|
|
|
ifaceArg = concatStringsSep " -N " (map (i: "-i${i}") (splitString " " iface));
|
|
|
|
driverArg = optionalString (suppl.driver != null) "-D${suppl.driver}";
|
|
|
|
bridgeArg = optionalString (suppl.bridge!="") "-b${suppl.bridge}";
|
|
|
|
confFileArg = optionalString (suppl.configFile.path!=null) "-c${suppl.configFile.path}";
|
2022-12-17 10:02:37 +00:00
|
|
|
extraConfFile = pkgs.writeText "supplicant-extra-conf-${replaceStrings [" "] ["-"] iface}" ''
|
2020-04-24 23:36:52 +00:00
|
|
|
${optionalString suppl.userControlled.enable "ctrl_interface=DIR=${suppl.userControlled.socketDir} GROUP=${suppl.userControlled.group}"}
|
|
|
|
${optionalString suppl.configFile.writable "update_config=1"}
|
|
|
|
${suppl.extraConf}
|
|
|
|
'';
|
|
|
|
in
|
|
|
|
{ description = "Supplicant ${iface}${optionalString (iface=="WLAN"||iface=="LAN") " %I"}";
|
|
|
|
wantedBy = [ "multi-user.target" ] ++ deps;
|
|
|
|
wants = [ "network.target" ];
|
|
|
|
bindsTo = deps;
|
|
|
|
after = deps;
|
|
|
|
before = [ "network.target" ];
|
|
|
|
|
|
|
|
path = [ pkgs.coreutils ];
|
|
|
|
|
|
|
|
preStart = ''
|
2022-04-27 09:35:20 +00:00
|
|
|
${optionalString (suppl.configFile.path!=null && suppl.configFile.writable) ''
|
2021-05-20 23:08:51 +00:00
|
|
|
(umask 077 && touch -a "${suppl.configFile.path}")
|
2020-04-24 23:36:52 +00:00
|
|
|
''}
|
|
|
|
${optionalString suppl.userControlled.enable ''
|
2021-05-20 23:08:51 +00:00
|
|
|
install -dm770 -g "${suppl.userControlled.group}" "${suppl.userControlled.socketDir}"
|
2020-04-24 23:36:52 +00:00
|
|
|
''}
|
|
|
|
'';
|
|
|
|
|
|
|
|
serviceConfig.ExecStart = "${pkgs.wpa_supplicant}/bin/wpa_supplicant -s ${driverArg} ${confFileArg} -I${extraConfFile} ${bridgeArg} ${suppl.extraCmdArgs} ${if (iface=="WLAN"||iface=="LAN") then "-i%I" else (if (iface=="DBUS") then "-u" else ifaceArg)}";
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
|
|
networking.supplicant = mkOption {
|
|
|
|
type = with types; attrsOf (submodule {
|
|
|
|
options = {
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
configFile = {
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
path = mkOption {
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
default = null;
|
2021-10-06 13:57:05 +00:00
|
|
|
example = literalExpression "/etc/wpa_supplicant.conf";
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
External `wpa_supplicant.conf` configuration file.
|
|
|
|
The configuration options defined declaratively within `networking.supplicant` have
|
|
|
|
precedence over options defined in `configFile`.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
writable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
Whether the configuration file at `configFile.path` should be written to by
|
|
|
|
`wpa_supplicant`.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
extraConf = mkOption {
|
|
|
|
type = types.lines;
|
|
|
|
default = "";
|
|
|
|
example = ''
|
|
|
|
ap_scan=1
|
|
|
|
device_name=My-NixOS-Device
|
|
|
|
device_type=1-0050F204-1
|
|
|
|
driver_param=use_p2p_group_interface=1
|
|
|
|
disable_scan_offload=1
|
|
|
|
p2p_listen_reg_class=81
|
|
|
|
p2p_listen_channel=1
|
|
|
|
p2p_oper_reg_class=81
|
|
|
|
p2p_oper_channel=1
|
|
|
|
manufacturer=NixOS
|
|
|
|
model_name=NixOS_Unstable
|
|
|
|
model_number=2015
|
|
|
|
'';
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
Configuration options for `wpa_supplicant.conf`.
|
|
|
|
Options defined here have precedence over options in `configFile`.
|
|
|
|
NOTE: Do not write sensitive data into `extraConf` as it will
|
|
|
|
be world-readable in the `nix-store`. For sensitive information
|
|
|
|
use the `configFile` instead.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
extraCmdArgs = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
|
|
|
example = "-e/run/wpa_supplicant/entropy.bin";
|
|
|
|
description =
|
2022-08-12 12:06:08 +00:00
|
|
|
lib.mdDoc "Command line arguments to add when executing `wpa_supplicant`.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
driver = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = "nl80211,wext";
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc "Force a specific wpa_supplicant driver.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
bridge = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "";
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc "Name of the bridge interface that wpa_supplicant should listen at.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
userControlled = {
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
enable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Allow normal users to control wpa_supplicant through wpa_gui or wpa_cli.
|
|
|
|
This is useful for laptop users that switch networks a lot and don't want
|
|
|
|
to depend on a large package such as NetworkManager just to pick nearby
|
|
|
|
access points.
|
|
|
|
'';
|
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
socketDir = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "/run/wpa_supplicant";
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc "Directory of sockets for controlling wpa_supplicant.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
group = mkOption {
|
|
|
|
type = types.str;
|
|
|
|
default = "wheel";
|
|
|
|
example = "network";
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc "Members of this group can control wpa_supplicant.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
2020-08-20 17:08:02 +00:00
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
default = { };
|
|
|
|
|
2021-10-06 13:57:05 +00:00
|
|
|
example = literalExpression ''
|
2020-04-24 23:36:52 +00:00
|
|
|
{ "wlan0 wlan1" = {
|
|
|
|
configFile.path = "/etc/wpa_supplicant.conf";
|
|
|
|
userControlled.group = "network";
|
|
|
|
extraConf = '''
|
|
|
|
ap_scan=1
|
|
|
|
p2p_disabled=1
|
|
|
|
''';
|
|
|
|
extraCmdArgs = "-u -W";
|
|
|
|
bridge = "br0";
|
|
|
|
};
|
|
|
|
}
|
|
|
|
'';
|
|
|
|
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc ''
|
|
|
|
Interfaces for which to start {command}`wpa_supplicant`.
|
2020-04-24 23:36:52 +00:00
|
|
|
The supplicant is used to scan for and associate with wireless networks,
|
|
|
|
or to authenticate with 802.1x capable network switches.
|
|
|
|
|
|
|
|
The value of this option is an attribute set. Each attribute configures a
|
2022-08-12 12:06:08 +00:00
|
|
|
{command}`wpa_supplicant` service, where the attribute name specifies
|
|
|
|
the name of the interface that {command}`wpa_supplicant` operates on.
|
2020-04-24 23:36:52 +00:00
|
|
|
The attribute name can be a space separated list of interfaces.
|
2022-08-12 12:06:08 +00:00
|
|
|
The attribute names `WLAN`, `LAN` and `DBUS`
|
|
|
|
have a special meaning. `WLAN` and `LAN` are
|
|
|
|
configurations for universal {command}`wpa_supplicant` service that is
|
2020-04-24 23:36:52 +00:00
|
|
|
started for each WLAN interface or for each LAN interface, respectively.
|
2022-08-12 12:06:08 +00:00
|
|
|
`DBUS` defines a device-unrelated {command}`wpa_supplicant`
|
|
|
|
service that can be accessed through `D-Bus`.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf (cfg != {}) {
|
|
|
|
|
|
|
|
environment.systemPackages = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
services.dbus.packages = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
systemd.services = mapAttrs' (n: v: nameValuePair (serviceName n) (supplicantService n v)) cfg;
|
|
|
|
|
|
|
|
services.udev.packages = [
|
|
|
|
(pkgs.writeTextFile {
|
|
|
|
name = "99-zzz-60-supplicant.rules";
|
|
|
|
destination = "/etc/udev/rules.d/99-zzz-60-supplicant.rules";
|
|
|
|
text = ''
|
|
|
|
${flip (concatMapStringsSep "\n") (filter (n: n!="WLAN" && n!="LAN" && n!="DBUS") (attrNames cfg)) (iface:
|
|
|
|
flip (concatMapStringsSep "\n") (splitString " " iface) (i: ''
|
2022-12-17 10:02:37 +00:00
|
|
|
ACTION=="add", SUBSYSTEM=="net", ENV{INTERFACE}=="${i}", TAG+="systemd", ENV{SYSTEMD_WANTS}+="supplicant-${replaceStrings [" "] ["-"] iface}.service", TAG+="SUPPLICANT_ASSIGNED"''))}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
${optionalString (hasAttr "WLAN" cfg) ''
|
2022-05-18 14:49:53 +00:00
|
|
|
ACTION=="add", SUBSYSTEM=="net", ENV{DEVTYPE}=="wlan", TAG!="SUPPLICANT_ASSIGNED", TAG+="systemd", PROGRAM="/run/current-system/systemd/bin/systemd-escape -p %E{INTERFACE}", ENV{SYSTEMD_WANTS}+="supplicant-wlan@$result.service"
|
2020-04-24 23:36:52 +00:00
|
|
|
''}
|
|
|
|
${optionalString (hasAttr "LAN" cfg) ''
|
2022-05-18 14:49:53 +00:00
|
|
|
ACTION=="add", SUBSYSTEM=="net", ENV{DEVTYPE}=="lan", TAG!="SUPPLICANT_ASSIGNED", TAG+="systemd", PROGRAM="/run/current-system/systemd/bin/systemd-escape -p %E{INTERFACE}", ENV{SYSTEMD_WANTS}+="supplicant-lan@$result.service"
|
2020-04-24 23:36:52 +00:00
|
|
|
''}
|
|
|
|
'';
|
|
|
|
})];
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|