2014-04-14 16:26:48 +02:00
|
|
|
{ config, lib, pkgs, ... }:
|
2009-10-12 19:27:57 +02:00
|
|
|
|
2014-04-14 16:26:48 +02:00
|
|
|
with lib;
|
2009-07-16 23:08:32 +02:00
|
|
|
|
|
|
|
let
|
2012-04-01 12:54:10 +02:00
|
|
|
cfg = config.networking.wireless;
|
2015-12-29 11:21:38 +01:00
|
|
|
configFile = if cfg.networks != {} then pkgs.writeText "wpa_supplicant.conf" ''
|
|
|
|
${optionalString cfg.userControlled.enable ''
|
|
|
|
ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=${cfg.userControlled.group}
|
|
|
|
update_config=1''}
|
2016-02-09 16:53:11 +01:00
|
|
|
${concatStringsSep "\n" (mapAttrsToList (ssid: networkConfig: let
|
|
|
|
psk = if networkConfig.psk != null
|
|
|
|
then ''"${networkConfig.psk}"''
|
|
|
|
else networkConfig.pskRaw;
|
2017-01-07 13:19:32 +01:00
|
|
|
priority = networkConfig.priority;
|
2016-02-09 16:53:11 +01:00
|
|
|
in ''
|
2015-12-29 11:21:38 +01:00
|
|
|
network={
|
|
|
|
ssid="${ssid}"
|
2016-02-09 16:53:11 +01:00
|
|
|
${optionalString (psk != null) ''psk=${psk}''}
|
|
|
|
${optionalString (psk == null) ''key_mgmt=NONE''}
|
2017-01-07 13:19:32 +01:00
|
|
|
${optionalString (priority != null) ''priority=${toString priority}''}
|
2015-12-29 11:21:38 +01:00
|
|
|
}
|
|
|
|
'') cfg.networks)}
|
|
|
|
'' else "/etc/wpa_supplicant.conf";
|
2015-12-26 02:12:32 +01:00
|
|
|
in {
|
2009-07-16 23:08:32 +02:00
|
|
|
options = {
|
2012-04-01 12:54:10 +02:00
|
|
|
networking.wireless = {
|
2015-12-29 11:21:38 +01:00
|
|
|
enable = mkEnableOption "wpa_supplicant";
|
2012-04-01 12:54:10 +02:00
|
|
|
|
|
|
|
interfaces = mkOption {
|
2015-06-15 18:18:46 +02:00
|
|
|
type = types.listOf types.str;
|
2012-04-01 12:54:10 +02:00
|
|
|
default = [];
|
|
|
|
example = [ "wlan0" "wlan1" ];
|
|
|
|
description = ''
|
2015-12-29 11:21:38 +01:00
|
|
|
The interfaces <command>wpa_supplicant</command> will use. If empty, it will
|
2014-04-24 18:16:12 +02:00
|
|
|
automatically use all wireless interfaces.
|
2012-04-01 12:54:10 +02:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
driver = mkOption {
|
2013-10-30 11:02:04 +01:00
|
|
|
type = types.str;
|
2013-01-22 12:32:19 +01:00
|
|
|
default = "nl80211,wext";
|
2012-11-02 17:05:30 +01:00
|
|
|
description = "Force a specific wpa_supplicant driver.";
|
2012-04-01 12:54:10 +02:00
|
|
|
};
|
2009-12-09 21:30:40 +01:00
|
|
|
|
2015-12-29 11:21:38 +01:00
|
|
|
networks = mkOption {
|
|
|
|
type = types.attrsOf (types.submodule {
|
|
|
|
options = {
|
|
|
|
psk = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The network's pre-shared key in plaintext defaulting
|
|
|
|
to being a network without any authentication.
|
2016-01-06 04:57:25 +01:00
|
|
|
|
|
|
|
Be aware that these will be written to the nix store
|
|
|
|
in plaintext!
|
2016-02-09 16:53:11 +01:00
|
|
|
|
|
|
|
Mutually exclusive with <varname>pskRaw</varname>.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
pskRaw = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
The network's pre-shared key in hex defaulting
|
|
|
|
to being a network without any authentication.
|
|
|
|
|
|
|
|
Mutually exclusive with <varname>psk</varname>.
|
2015-12-29 11:21:38 +01:00
|
|
|
'';
|
|
|
|
};
|
2017-01-07 13:19:32 +01:00
|
|
|
priority = mkOption {
|
|
|
|
type = types.nullOr types.int;
|
|
|
|
default = null;
|
|
|
|
description = ''
|
|
|
|
By default, all networks will get same priority group (0). If some of the
|
|
|
|
networks are more desirable, this field can be used to change the order in
|
|
|
|
which wpa_supplicant goes through the networks when selecting a BSS. The
|
|
|
|
priority groups will be iterated in decreasing priority (i.e., the larger the
|
|
|
|
priority value, the sooner the network is matched against the scan results).
|
|
|
|
Within each priority group, networks will be selected based on security
|
|
|
|
policy, signal strength, etc.
|
|
|
|
'';
|
|
|
|
};
|
2015-12-29 11:21:38 +01:00
|
|
|
};
|
|
|
|
});
|
|
|
|
description = ''
|
|
|
|
The network definitions to automatically connect to when
|
|
|
|
<command>wpa_supplicant</command> is running. If this
|
|
|
|
parameter is left empty wpa_supplicant will use
|
|
|
|
/etc/wpa_supplicant.conf as the configuration file.
|
|
|
|
'';
|
|
|
|
default = {};
|
|
|
|
example = literalExample ''
|
2016-02-29 01:47:12 +01:00
|
|
|
{ echelon = {
|
|
|
|
psk = "abcdefgh";
|
|
|
|
};
|
|
|
|
"free.wifi" = {};
|
|
|
|
}
|
2015-12-29 11:21:38 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2012-04-01 12:54:10 +02:00
|
|
|
userControlled = {
|
|
|
|
enable = mkOption {
|
2013-10-29 13:04:26 +01:00
|
|
|
type = types.bool;
|
2012-04-01 12:54:10 +02:00
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Allow normal users to control wpa_supplicant through wpa_gui or wpa_cli.
|
2015-02-17 22:14:29 +01:00
|
|
|
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.
|
2012-04-01 12:54:10 +02:00
|
|
|
|
2015-12-29 11:21:38 +01:00
|
|
|
When using a declarative network specification you cannot persist any
|
|
|
|
settings via wpa_gui or wpa_cli.
|
2012-04-01 12:54:10 +02:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
group = mkOption {
|
2013-10-30 11:02:04 +01:00
|
|
|
type = types.str;
|
2012-04-01 12:54:10 +02:00
|
|
|
default = "wheel";
|
|
|
|
example = "network";
|
2012-11-02 17:05:30 +01:00
|
|
|
description = "Members of this group can control wpa_supplicant.";
|
2012-04-01 12:54:10 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2009-07-16 23:08:32 +02:00
|
|
|
};
|
|
|
|
|
2016-05-09 07:53:27 +02:00
|
|
|
config = mkIf cfg.enable {
|
|
|
|
assertions = flip mapAttrsToList cfg.networks (name: cfg: {
|
|
|
|
assertion = cfg.psk == null || cfg.pskRaw == null;
|
|
|
|
message = ''networking.wireless."${name}".psk and networking.wireless."${name}".pskRaw are mutually exclusive'';
|
|
|
|
});
|
|
|
|
|
|
|
|
environment.systemPackages = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
services.dbus.packages = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
# FIXME: start a separate wpa_supplicant instance per interface.
|
|
|
|
systemd.services.wpa_supplicant = let
|
|
|
|
ifaces = cfg.interfaces;
|
|
|
|
deviceUnit = interface: [ "sys-subsystem-net-devices-${interface}.device" ];
|
|
|
|
in {
|
|
|
|
description = "WPA Supplicant";
|
|
|
|
|
2016-10-19 00:09:49 +02:00
|
|
|
after = lib.concatMap deviceUnit ifaces;
|
|
|
|
before = [ "network.target" ];
|
|
|
|
wants = [ "network.target" ];
|
2016-05-09 07:53:27 +02:00
|
|
|
requires = lib.concatMap deviceUnit ifaces;
|
2016-10-19 00:09:49 +02:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
2017-09-27 23:20:08 +02:00
|
|
|
stopIfChanged = false;
|
2016-05-09 07:53:27 +02:00
|
|
|
|
|
|
|
path = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
script = ''
|
|
|
|
${if ifaces == [] then ''
|
|
|
|
for i in $(cd /sys/class/net && echo *); do
|
|
|
|
DEVTYPE=
|
|
|
|
source /sys/class/net/$i/uevent
|
|
|
|
if [ "$DEVTYPE" = "wlan" -o -e /sys/class/net/$i/wireless ]; then
|
|
|
|
ifaces="$ifaces''${ifaces:+ -N} -i$i"
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
'' else ''
|
|
|
|
ifaces="${concatStringsSep " -N " (map (i: "-i${i}") ifaces)}"
|
|
|
|
''}
|
|
|
|
exec wpa_supplicant -s -u -D${cfg.driver} -c ${configFile} $ifaces
|
2012-02-19 23:53:25 +01:00
|
|
|
'';
|
2016-05-09 07:53:27 +02:00
|
|
|
};
|
2012-02-19 23:53:25 +01:00
|
|
|
|
2016-05-09 07:53:27 +02:00
|
|
|
powerManagement.resumeCommands = ''
|
|
|
|
${config.systemd.package}/bin/systemctl try-restart wpa_supplicant
|
|
|
|
'';
|
|
|
|
|
|
|
|
# Restart wpa_supplicant when a wlan device appears or disappears.
|
|
|
|
services.udev.extraRules = ''
|
|
|
|
ACTION=="add|remove", SUBSYSTEM=="net", ENV{DEVTYPE}=="wlan", RUN+="${config.systemd.package}/bin/systemctl try-restart wpa_supplicant.service"
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
meta.maintainers = with lib.maintainers; [ globin ];
|
2009-07-16 23:08:32 +02:00
|
|
|
}
|