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;
|
2016-01-05 19:32:41 +01:00
|
|
|
configFile = "/etc/wpa_supplicant.conf";
|
|
|
|
in
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
###### interface
|
|
|
|
|
2009-07-16 23:08:32 +02:00
|
|
|
options = {
|
2012-04-01 12:54:10 +02:00
|
|
|
networking.wireless = {
|
2016-01-05 19:32:41 +01:00
|
|
|
enable = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
|
|
|
description = ''
|
|
|
|
Whether to start <command>wpa_supplicant</command> to scan for
|
|
|
|
and associate with wireless networks. Note: NixOS currently
|
|
|
|
does not manage <command>wpa_supplicant</command>'s
|
|
|
|
configuration file, <filename>${configFile}</filename>. You
|
|
|
|
should edit this file yourself to define wireless networks,
|
|
|
|
WPA keys and so on (see
|
|
|
|
<citerefentry><refentrytitle>wpa_supplicant.conf</refentrytitle>
|
|
|
|
<manvolnum>5</manvolnum></citerefentry>), or use
|
|
|
|
networking.wireless.userControlled.* to allow users to add entries
|
|
|
|
through <command>wpa_cli</command> and <command>wpa_gui</command>.
|
|
|
|
'';
|
|
|
|
};
|
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 = ''
|
2016-01-05 19:32:41 +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
|
|
|
|
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
|
|
|
|
2016-01-05 19:32:41 +01:00
|
|
|
When you want to use this, make sure ${configFile} doesn't exist.
|
|
|
|
It will be created for you.
|
|
|
|
|
|
|
|
Currently it is also necessary to explicitly specify networking.wireless.interfaces.
|
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-01-05 19:32:41 +01:00
|
|
|
###### implementation
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
|
|
|
|
environment.systemPackages = [ pkgs.wpa_supplicant ];
|
|
|
|
|
|
|
|
services.dbus.packages = [ pkgs.wpa_supplicant ];
|
2011-12-20 00:16:32 +01:00
|
|
|
|
2016-01-05 19:32:41 +01:00
|
|
|
# FIXME: start a separate wpa_supplicant instance per interface.
|
2015-12-26 02:09:00 +01:00
|
|
|
jobs.wpa_supplicant = let
|
|
|
|
ifaces = cfg.interfaces;
|
|
|
|
in { description = "WPA Supplicant";
|
2013-01-10 13:46:34 +01:00
|
|
|
|
|
|
|
wantedBy = [ "network.target" ];
|
2009-07-24 02:31:42 +02:00
|
|
|
|
2011-11-25 17:32:54 +01:00
|
|
|
path = [ pkgs.wpa_supplicant ];
|
|
|
|
|
2016-01-05 19:32:41 +01:00
|
|
|
preStart = ''
|
|
|
|
touch -a ${configFile}
|
|
|
|
chmod 600 ${configFile}
|
|
|
|
'' + optionalString cfg.userControlled.enable ''
|
|
|
|
if [ ! -s ${configFile} ]; then
|
|
|
|
echo "ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=${cfg.userControlled.group}" >> ${configFile}
|
|
|
|
echo "update_config=1" >> ${configFile}
|
|
|
|
fi
|
2015-12-26 02:12:32 +01:00
|
|
|
'';
|
2016-01-05 19:32:41 +01:00
|
|
|
|
|
|
|
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
|
|
|
|
'';
|
2009-07-16 23:08:32 +02:00
|
|
|
};
|
2012-06-29 11:53:08 +02:00
|
|
|
|
2016-01-05 19:32:41 +01:00
|
|
|
powerManagement.resumeCommands =
|
|
|
|
''
|
2013-01-16 13:17:57 +01:00
|
|
|
${config.systemd.package}/bin/systemctl try-restart wpa_supplicant
|
2012-02-19 23:53:25 +01:00
|
|
|
'';
|
|
|
|
|
2016-01-05 19:32:41 +01:00
|
|
|
assertions = [{ assertion = !cfg.userControlled.enable || cfg.interfaces != [];
|
|
|
|
message = "user controlled wpa_supplicant needs explicit networking.wireless.interfaces";}];
|
|
|
|
|
|
|
|
# Restart wpa_supplicant when a wlan device appears or disappears.
|
|
|
|
services.udev.extraRules =
|
|
|
|
''
|
2014-04-28 20:10:49 +02:00
|
|
|
ACTION=="add|remove", SUBSYSTEM=="net", ENV{DEVTYPE}=="wlan", RUN+="${config.systemd.package}/bin/systemctl try-restart wpa_supplicant.service"
|
|
|
|
'';
|
2016-01-05 19:32:41 +01:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2009-07-16 23:08:32 +02:00
|
|
|
}
|