core/homeassistant/components/device_tracker/owntracks.py

366 lines
13 KiB
Python
Raw Normal View History

2015-09-20 07:27:50 +00:00
"""
2016-03-07 17:12:06 +00:00
Support the OwnTracks platform.
2015-09-20 07:27:50 +00:00
2015-10-13 18:55:15 +00:00
For more details about this platform, please refer to the documentation at
2015-11-09 12:12:18 +00:00
https://home-assistant.io/components/device_tracker.owntracks/
2015-09-20 07:27:50 +00:00
"""
import json
2015-09-21 03:09:53 +00:00
import logging
import threading
2016-10-04 07:57:37 +00:00
import base64
from collections import defaultdict
2015-09-20 07:27:50 +00:00
import voluptuous as vol
import homeassistant.helpers.config_validation as cv
2015-09-20 07:27:50 +00:00
import homeassistant.components.mqtt as mqtt
from homeassistant.const import STATE_HOME
2016-05-19 15:04:55 +00:00
from homeassistant.util import convert, slugify
from homeassistant.components import zone as zone_comp
2016-09-07 01:35:10 +00:00
from homeassistant.components.device_tracker import PLATFORM_SCHEMA
2015-09-20 07:27:50 +00:00
2016-10-04 07:57:37 +00:00
REQUIREMENTS = ['libnacl==1.5.0']
2015-09-20 07:27:50 +00:00
2016-10-06 00:32:29 +00:00
_LOGGER = logging.getLogger(__name__)
2016-01-29 11:49:44 +00:00
BEACON_DEV_ID = 'beacon'
2016-10-06 00:32:29 +00:00
CONF_MAX_GPS_ACCURACY = 'max_gps_accuracy'
CONF_SECRET = 'secret'
CONF_WAYPOINT_IMPORT = 'waypoints'
CONF_WAYPOINT_WHITELIST = 'waypoint_whitelist'
2015-09-20 07:27:50 +00:00
2016-10-06 00:32:29 +00:00
DEPENDENCIES = ['mqtt']
EVENT_TOPIC = 'owntracks/+/+/event'
2016-10-06 00:32:29 +00:00
LOCATION_TOPIC = 'owntracks/+/+'
LOCK = threading.Lock()
2016-10-06 00:32:29 +00:00
MOBILE_BEACONS_ACTIVE = defaultdict(list)
REGIONS_ENTERED = defaultdict(list)
VALIDATE_LOCATION = 'location'
VALIDATE_TRANSITION = 'transition'
2016-08-31 02:46:01 +00:00
VALIDATE_WAYPOINTS = 'waypoints'
WAYPOINT_LAT_KEY = 'lat'
WAYPOINT_LON_KEY = 'lon'
2016-10-06 00:32:29 +00:00
WAYPOINT_TOPIC = 'owntracks/{}/{}/waypoint'
2016-09-07 01:35:10 +00:00
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
vol.Optional(CONF_MAX_GPS_ACCURACY): vol.Coerce(float),
vol.Optional(CONF_WAYPOINT_IMPORT, default=True): cv.boolean,
2016-10-04 07:57:37 +00:00
vol.Optional(CONF_WAYPOINT_WHITELIST): vol.All(
cv.ensure_list, [cv.string]),
vol.Optional(CONF_SECRET): vol.Any(
vol.Schema({vol.Optional(cv.string): cv.string}),
cv.string)
})
2016-10-04 07:57:37 +00:00
def get_cipher():
"""Return decryption function and length of key."""
from libnacl import crypto_secretbox_KEYBYTES as KEYLEN
from libnacl.secret import SecretBox
def decrypt(ciphertext, key):
"""Decrypt ciphertext using key."""
return SecretBox(key).decrypt(ciphertext)
return (KEYLEN, decrypt)
def setup_scanner(hass, config, see, discovery_info=None):
2016-10-06 00:32:29 +00:00
"""Set up an OwnTracks tracker."""
max_gps_accuracy = config.get(CONF_MAX_GPS_ACCURACY)
waypoint_import = config.get(CONF_WAYPOINT_IMPORT)
waypoint_whitelist = config.get(CONF_WAYPOINT_WHITELIST)
2016-10-04 07:57:37 +00:00
secret = config.get(CONF_SECRET)
def decrypt_payload(topic, ciphertext):
"""Decrypt encrypted payload."""
try:
keylen, decrypt = get_cipher()
except OSError:
_LOGGER.warning('Ignoring encrypted payload '
'because libsodium not installed.')
return None
if isinstance(secret, dict):
key = secret.get(topic)
else:
key = secret
if key is None:
_LOGGER.warning('Ignoring encrypted payload '
'because no decryption key known '
'for topic %s.', topic)
return None
key = key.encode("utf-8")
key = key[:keylen]
key = key.ljust(keylen, b'\0')
2016-10-04 07:57:37 +00:00
try:
ciphertext = base64.b64decode(ciphertext)
message = decrypt(ciphertext, key)
message = message.decode("utf-8")
_LOGGER.debug("Decrypted payload: %s", message)
return message
except ValueError:
_LOGGER.warning('Ignoring encrypted payload '
'because unable to decrypt using key '
'for topic %s.', topic)
return None
# pylint: disable=too-many-return-statements
2016-10-04 07:57:37 +00:00
def validate_payload(topic, payload, data_type):
2016-10-06 00:32:29 +00:00
"""Validate the OwnTracks payload."""
2015-09-20 18:46:01 +00:00
try:
data = json.loads(payload)
except ValueError:
# If invalid JSON
_LOGGER.error('Unable to parse payload as JSON: %s', payload)
return None
2016-10-04 07:57:37 +00:00
if isinstance(data, dict) and \
data.get('_type') == 'encrypted' and \
'data' in data:
plaintext_payload = decrypt_payload(topic, data['data'])
if plaintext_payload is None:
return None
else:
return validate_payload(topic, plaintext_payload, data_type)
if not isinstance(data, dict) or data.get('_type') != data_type:
_LOGGER.debug('Skipping %s update for following data '
'because of missing or malformatted data: %s',
data_type, data)
return None
2016-08-31 02:46:01 +00:00
if data_type == VALIDATE_TRANSITION or data_type == VALIDATE_WAYPOINTS:
return data
if max_gps_accuracy is not None and \
convert(data.get('acc'), float, 0.0) > max_gps_accuracy:
2016-11-04 01:42:22 +00:00
_LOGGER.info('Ignoring %s update because expected GPS '
'accuracy %s is not met: %s',
data_type, max_gps_accuracy, payload)
return None
2016-05-19 15:16:43 +00:00
if convert(data.get('acc'), float, 1.0) == 0.0:
_LOGGER.warning('Ignoring %s update because GPS accuracy '
2016-08-16 08:48:13 +00:00
'is zero: %s',
data_type, payload)
2016-05-19 15:16:43 +00:00
return None
return data
2015-09-21 03:09:53 +00:00
def owntracks_location_update(topic, payload, qos):
"""MQTT message received."""
# Docs on available data:
# http://owntracks.org/booklet/tech/json/#_typelocation
2016-10-04 07:57:37 +00:00
data = validate_payload(topic, payload, VALIDATE_LOCATION)
if not data:
2015-09-20 18:46:01 +00:00
return
2015-09-21 03:09:53 +00:00
dev_id, kwargs = _parse_see_args(topic, data)
# Block updates if we're in a region
with LOCK:
if REGIONS_ENTERED[dev_id]:
_LOGGER.debug(
"location update ignored - inside region %s",
REGIONS_ENTERED[-1])
return
2015-09-21 03:09:53 +00:00
see(**kwargs)
2016-01-29 11:49:44 +00:00
see_beacons(dev_id, kwargs)
2016-01-03 16:12:11 +00:00
def owntracks_event_update(topic, payload, qos):
2016-03-07 17:12:06 +00:00
"""MQTT event (geofences) received."""
# Docs on available data:
# http://owntracks.org/booklet/tech/json/#_typetransition
2016-10-04 07:57:37 +00:00
data = validate_payload(topic, payload, VALIDATE_TRANSITION)
if not data:
return
2015-09-20 07:27:50 +00:00
if data.get('desc') is None:
_LOGGER.error(
"Location missing from `Entering/Leaving` message - "
"please turn `Share` on in OwnTracks app")
return
# OwnTracks uses - at the start of a beacon zone
# to switch on 'hold mode' - ignore this
location = data['desc'].lstrip("-")
if location.lower() == 'home':
location = STATE_HOME
dev_id, kwargs = _parse_see_args(topic, data)
def enter_event():
"""Execute enter event."""
zone = hass.states.get("zone.{}".format(slugify(location)))
with LOCK:
if zone is None and data.get('t') == 'b':
# Not a HA zone, and a beacon so assume mobile
beacons = MOBILE_BEACONS_ACTIVE[dev_id]
if location not in beacons:
beacons.append(location)
_LOGGER.info("Added beacon %s", location)
2016-01-29 11:49:44 +00:00
else:
# Normal region
regions = REGIONS_ENTERED[dev_id]
if location not in regions:
regions.append(location)
_LOGGER.info("Enter region %s", location)
_set_gps_from_zone(kwargs, location, zone)
see(**kwargs)
2016-01-29 11:49:44 +00:00
see_beacons(dev_id, kwargs)
2016-01-03 16:12:11 +00:00
def leave_event():
"""Execute leave event."""
with LOCK:
regions = REGIONS_ENTERED[dev_id]
if location in regions:
regions.remove(location)
new_region = regions[-1] if regions else None
if new_region:
# Exit to previous region
zone = hass.states.get(
"zone.{}".format(slugify(new_region)))
_set_gps_from_zone(kwargs, new_region, zone)
_LOGGER.info("Exit to %s", new_region)
see(**kwargs)
see_beacons(dev_id, kwargs)
else:
_LOGGER.info("Exit to GPS")
# Check for GPS accuracy
valid_gps = True
if 'acc' in data:
if data['acc'] == 0.0:
valid_gps = False
2016-08-16 08:48:13 +00:00
_LOGGER.warning(
'Ignoring GPS in region exit because accuracy'
'is zero: %s',
payload)
if (max_gps_accuracy is not None and
data['acc'] > max_gps_accuracy):
valid_gps = False
2016-11-04 01:42:22 +00:00
_LOGGER.info(
2016-08-16 08:48:13 +00:00
'Ignoring GPS in region exit because expected '
'GPS accuracy %s is not met: %s',
max_gps_accuracy, payload)
if valid_gps:
see(**kwargs)
see_beacons(dev_id, kwargs)
2016-01-29 11:49:44 +00:00
beacons = MOBILE_BEACONS_ACTIVE[dev_id]
if location in beacons:
beacons.remove(location)
_LOGGER.info("Remove beacon %s", location)
if data['event'] == 'enter':
enter_event()
elif data['event'] == 'leave':
leave_event()
else:
_LOGGER.error(
2016-01-03 16:42:49 +00:00
'Misformatted mqtt msgs, _type=transition, event=%s',
data['event'])
return
def owntracks_waypoint_update(topic, payload, qos):
"""List of waypoints published by a user."""
# Docs on available data:
# http://owntracks.org/booklet/tech/json/#_typewaypoints
2016-10-04 07:57:37 +00:00
data = validate_payload(topic, payload, VALIDATE_WAYPOINTS)
if not data:
return
wayps = data['waypoints']
_LOGGER.info("Got %d waypoints from %s", len(wayps), topic)
for wayp in wayps:
name = wayp['desc']
pretty_name = parse_topic(topic, True)[1] + ' - ' + name
lat = wayp[WAYPOINT_LAT_KEY]
lon = wayp[WAYPOINT_LON_KEY]
rad = wayp['rad']
2016-09-24 07:04:03 +00:00
# check zone exists
entity_id = zone_comp.ENTITY_ID_FORMAT.format(slugify(pretty_name))
# Check if state already exists
if hass.states.get(entity_id) is not None:
continue
zone = zone_comp.Zone(hass, pretty_name, lat, lon, rad,
2016-09-24 07:04:03 +00:00
zone_comp.ICON_IMPORT, False)
zone.entity_id = entity_id
zone.update_ha_state()
2016-01-29 16:12:34 +00:00
def see_beacons(dev_id, kwargs_param):
2016-03-07 17:12:06 +00:00
"""Set active beacons to the current location."""
2016-01-29 16:12:34 +00:00
kwargs = kwargs_param.copy()
# the battery state applies to the tracking device, not the beacon
kwargs.pop('battery', None)
2016-01-29 11:49:44 +00:00
for beacon in MOBILE_BEACONS_ACTIVE[dev_id]:
2016-01-29 15:15:59 +00:00
kwargs['dev_id'] = "{}_{}".format(BEACON_DEV_ID, beacon)
kwargs['host_name'] = beacon
2016-01-29 11:49:44 +00:00
see(**kwargs)
mqtt.subscribe(hass, LOCATION_TOPIC, owntracks_location_update, 1)
mqtt.subscribe(hass, EVENT_TOPIC, owntracks_event_update, 1)
2016-01-03 16:12:11 +00:00
if waypoint_import:
if waypoint_whitelist is None:
mqtt.subscribe(hass, WAYPOINT_TOPIC.format('+', '+'),
owntracks_waypoint_update, 1)
else:
for whitelist_user in waypoint_whitelist:
mqtt.subscribe(hass, WAYPOINT_TOPIC.format(whitelist_user,
'+'),
owntracks_waypoint_update, 1)
return True
2016-01-03 16:12:11 +00:00
2015-09-20 07:27:50 +00:00
def parse_topic(topic, pretty=False):
"""Parse an MQTT topic owntracks/user/dev, return (user, dev) tuple."""
parts = topic.split('/')
dev_id_format = ''
if pretty:
dev_id_format = '{} {}'
else:
dev_id_format = '{}_{}'
dev_id = slugify(dev_id_format.format(parts[1], parts[2]))
host_name = parts[1]
return (host_name, dev_id)
def _parse_see_args(topic, data):
"""Parse the OwnTracks location parameters, into the format see expects."""
(host_name, dev_id) = parse_topic(topic, False)
kwargs = {
'dev_id': dev_id,
'host_name': host_name,
'gps': (data[WAYPOINT_LAT_KEY], data[WAYPOINT_LON_KEY])
}
if 'acc' in data:
kwargs['gps_accuracy'] = data['acc']
if 'batt' in data:
kwargs['battery'] = data['batt']
return dev_id, kwargs
def _set_gps_from_zone(kwargs, location, zone):
2016-03-07 17:12:06 +00:00
"""Set the see parameters from the zone parameters."""
if zone is not None:
kwargs['gps'] = (
zone.attributes['latitude'],
zone.attributes['longitude'])
kwargs['gps_accuracy'] = zone.attributes['radius']
2016-03-13 18:01:29 +00:00
kwargs['location_name'] = location
return kwargs