esphome/esphomeyaml/components/wifi.py

190 lines
6.2 KiB
Python
Raw Normal View History

2018-04-07 01:23:03 +02:00
import voluptuous as vol
import esphomeyaml.config_validation as cv
from esphomeyaml.const import CONF_AP, CONF_CHANNEL, CONF_DNS1, CONF_DNS2, CONF_DOMAIN, \
2019-01-06 19:38:23 +01:00
CONF_GATEWAY, CONF_HOSTNAME, CONF_ID, CONF_MANUAL_IP, CONF_PASSWORD, CONF_POWER_SAVE_MODE, \
CONF_REBOOT_TIMEOUT, CONF_SSID, CONF_STATIC_IP, CONF_SUBNET, CONF_NETWORKS, CONF_BSSID, \
CONF_FAST_CONNECT
2019-01-06 19:38:23 +01:00
from esphomeyaml.core import CORE, HexInt
from esphomeyaml.cpp_generator import Pvariable, StructInitializer, add, variable, ArrayInitializer
from esphomeyaml.cpp_types import App, Component, esphomelib_ns, global_ns
IPAddress = global_ns.class_('IPAddress')
ManualIP = esphomelib_ns.struct('ManualIP')
WiFiComponent = esphomelib_ns.class_('WiFiComponent', Component)
WiFiAP = esphomelib_ns.struct('WiFiAP')
WiFiPowerSaveMode = esphomelib_ns.enum('WiFiPowerSaveMode')
WIFI_POWER_SAVE_MODES = {
'NONE': WiFiPowerSaveMode.WIFI_POWER_SAVE_NONE,
'LIGHT': WiFiPowerSaveMode.WIFI_POWER_SAVE_LIGHT,
'HIGH': WiFiPowerSaveMode.WIFI_POWER_SAVE_HIGH,
}
2018-04-07 01:23:03 +02:00
def validate_password(value):
value = cv.string(value)
if not value:
return value
if len(value) < 8:
raise vol.Invalid(u"WPA password must be at least 8 characters long")
if len(value) > 64:
raise vol.Invalid(u"WPA password must be at most 64 characters long")
return value
def validate_channel(value):
value = cv.positive_int(value)
if value < 1:
raise vol.Invalid("Minimum WiFi channel is 1")
if value > 14:
raise vol.Invalid("Maximum WiFi channel is 14")
return value
AP_MANUAL_IP_SCHEMA = vol.Schema({
vol.Required(CONF_STATIC_IP): cv.ipv4,
vol.Required(CONF_GATEWAY): cv.ipv4,
vol.Required(CONF_SUBNET): cv.ipv4,
})
STA_MANUAL_IP_SCHEMA = AP_MANUAL_IP_SCHEMA.extend({
vol.Inclusive(CONF_DNS1, 'dns'): cv.ipv4,
vol.Inclusive(CONF_DNS2, 'dns'): cv.ipv4,
})
WIFI_NETWORK_BASE = vol.Schema({
2019-01-06 19:38:23 +01:00
cv.GenerateID(): cv.declare_variable_id(WiFiAP),
vol.Optional(CONF_SSID): cv.ssid,
vol.Optional(CONF_PASSWORD): validate_password,
vol.Optional(CONF_CHANNEL): validate_channel,
2019-01-06 19:38:23 +01:00
vol.Optional(CONF_MANUAL_IP): STA_MANUAL_IP_SCHEMA,
})
2018-06-11 15:22:03 +02:00
WIFI_NETWORK_AP = WIFI_NETWORK_BASE.extend({
2019-01-06 19:38:23 +01:00
2018-06-11 15:22:03 +02:00
})
2018-06-11 15:22:03 +02:00
WIFI_NETWORK_STA = WIFI_NETWORK_BASE.extend({
2019-01-06 19:38:23 +01:00
vol.Optional(CONF_BSSID): cv.mac_address,
2018-06-11 15:22:03 +02:00
})
2018-06-11 16:13:46 +02:00
def validate(config):
if CONF_PASSWORD in config and CONF_SSID not in config:
raise vol.Invalid("Cannot have WiFi password without SSID!")
2019-01-06 19:38:23 +01:00
if CONF_SSID in config:
network = {CONF_SSID: config.pop(CONF_SSID)}
if CONF_PASSWORD in config:
network[CONF_PASSWORD] = config.pop(CONF_PASSWORD)
if CONF_NETWORKS in config:
raise vol.Invalid("You cannot use the 'ssid:' option together with 'networks:'. Please "
"copy your network into the 'networks:' key")
config[CONF_NETWORKS] = cv.ensure_list(WIFI_NETWORK_STA)(network)
if (CONF_NETWORKS not in config) and (CONF_AP not in config):
2018-06-11 16:13:46 +02:00
raise vol.Invalid("Please specify at least an SSID or an Access Point "
"to create.")
if config.get(CONF_FAST_CONNECT, False):
networks = config.get(CONF_NETWORKS, [])
if not networks:
raise vol.Invalid("At least one network required for fast_connect!")
if len(networks) != 1:
raise vol.Invalid("Fast connect can only be used with one network!")
return config
CONFIG_SCHEMA = vol.All(vol.Schema({
2018-06-02 22:22:20 +02:00
cv.GenerateID(): cv.declare_variable_id(WiFiComponent),
2019-01-06 19:38:23 +01:00
vol.Optional(CONF_NETWORKS): cv.ensure_list(WIFI_NETWORK_STA),
vol.Optional(CONF_SSID): cv.ssid,
vol.Optional(CONF_PASSWORD): validate_password,
2018-06-11 15:22:03 +02:00
vol.Optional(CONF_MANUAL_IP): STA_MANUAL_IP_SCHEMA,
2019-01-06 19:38:23 +01:00
vol.Optional(CONF_AP): WIFI_NETWORK_AP,
2018-04-07 01:23:03 +02:00
vol.Optional(CONF_HOSTNAME): cv.hostname,
2018-10-20 15:18:12 +02:00
vol.Optional(CONF_DOMAIN, default='.local'): cv.domain_name,
2018-06-12 21:18:04 +02:00
vol.Optional(CONF_REBOOT_TIMEOUT): cv.positive_time_period_milliseconds,
2019-01-06 19:38:23 +01:00
vol.Optional(CONF_POWER_SAVE_MODE): cv.one_of(*WIFI_POWER_SAVE_MODES, upper=True),
vol.Optional(CONF_FAST_CONNECT): cv.boolean,
2018-06-11 16:13:46 +02:00
}), validate)
2018-04-07 01:23:03 +02:00
def safe_ip(ip):
if ip is None:
return IPAddress(0, 0, 0, 0)
2018-04-07 01:23:03 +02:00
return IPAddress(*ip.args)
def manual_ip(config):
if config is None:
return None
return StructInitializer(
2018-05-20 12:41:52 +02:00
ManualIP,
('static_ip', safe_ip(config[CONF_STATIC_IP])),
('gateway', safe_ip(config[CONF_GATEWAY])),
('subnet', safe_ip(config[CONF_SUBNET])),
('dns1', safe_ip(config.get(CONF_DNS1))),
('dns2', safe_ip(config.get(CONF_DNS2))),
)
2019-01-06 19:38:23 +01:00
def wifi_network(config, static_ip):
ap = variable(config[CONF_ID], WiFiAP())
if CONF_SSID in config:
add(ap.set_ssid(config[CONF_SSID]))
if CONF_PASSWORD in config:
add(ap.set_password(config[CONF_PASSWORD]))
if CONF_BSSID in config:
bssid = [HexInt(i) for i in config[CONF_BSSID].parts]
add(ap.set_bssid(ArrayInitializer(*bssid, multiline=False)))
if CONF_CHANNEL in config:
add(ap.set_channel(config[CONF_CHANNEL]))
if static_ip is not None:
add(ap.set_manual_ip(manual_ip(static_ip)))
return ap
def get_upload_host(config):
if CONF_MANUAL_IP in config:
return str(config[CONF_MANUAL_IP][CONF_STATIC_IP])
hostname = config.get(CONF_HOSTNAME) or CORE.name
return hostname + config[CONF_DOMAIN]
2018-04-07 01:23:03 +02:00
def to_code(config):
2018-06-11 16:13:46 +02:00
rhs = App.init_wifi()
2018-06-02 22:22:20 +02:00
wifi = Pvariable(config[CONF_ID], rhs)
2019-01-06 19:38:23 +01:00
for network in config.get(CONF_NETWORKS, []):
add(wifi.add_sta(wifi_network(network, config.get(CONF_MANUAL_IP))))
if CONF_AP in config:
2019-01-06 19:38:23 +01:00
add(wifi.set_ap(wifi_network(config[CONF_AP], config.get(CONF_MANUAL_IP))))
2018-04-07 01:23:03 +02:00
if CONF_HOSTNAME in config:
add(wifi.set_hostname(config[CONF_HOSTNAME]))
2018-05-14 11:50:56 +02:00
2018-06-12 21:18:04 +02:00
if CONF_REBOOT_TIMEOUT in config:
add(wifi.set_reboot_timeout(config[CONF_REBOOT_TIMEOUT]))
if CONF_POWER_SAVE_MODE in config:
add(wifi.set_power_save_mode(WIFI_POWER_SAVE_MODES[config[CONF_POWER_SAVE_MODE]]))
if CONF_FAST_CONNECT in config:
add(wifi.set_fast_connect(config[CONF_FAST_CONNECT]))
2018-05-14 11:50:56 +02:00
def lib_deps(config):
2019-01-06 19:38:23 +01:00
if CORE.is_esp8266:
2018-05-14 11:50:56 +02:00
return 'ESP8266WiFi'
2019-01-06 19:38:23 +01:00
if CORE.is_esp32:
return None
raise NotImplementedError