core/homeassistant/components/device_tracker/owntracks.py

212 lines
7.2 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
from collections import defaultdict
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
2015-09-20 07:27:50 +00:00
DEPENDENCIES = ['mqtt']
REGIONS_ENTERED = defaultdict(list)
2016-01-29 11:49:44 +00:00
MOBILE_BEACONS_ACTIVE = defaultdict(list)
BEACON_DEV_ID = 'beacon'
2015-09-20 07:27:50 +00:00
LOCATION_TOPIC = 'owntracks/+/+'
EVENT_TOPIC = 'owntracks/+/+/event'
2015-09-20 07:27:50 +00:00
_LOGGER = logging.getLogger(__name__)
LOCK = threading.Lock()
CONF_MAX_GPS_ACCURACY = 'max_gps_accuracy'
2016-01-03 16:42:49 +00:00
2015-09-20 07:27:50 +00:00
def setup_scanner(hass, config, see):
2016-03-07 17:12:06 +00:00
"""Setup an OwnTracks tracker."""
max_gps_accuracy = config.get(CONF_MAX_GPS_ACCURACY)
def validate_payload(payload, data_type):
"""Validate 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
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
if max_gps_accuracy is not None and \
convert(data.get('acc'), float, 0.0) > max_gps_accuracy:
_LOGGER.debug('Skipping %s update because expected GPS '
'accuracy %s is not met: %s',
data_type, max_gps_accuracy, data)
return None
2016-05-19 15:16:43 +00:00
if convert(data.get('acc'), float, 1.0) == 0.0:
_LOGGER.debug('Skipping %s update because GPS accuracy'
'is zero',
data_type)
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
data = validate_payload(payload, '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
data = validate_payload(payload, '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
2016-05-19 15:04:55 +00:00
location = slugify(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(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)
2016-03-13 18:01:29 +00:00
_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(new_region))
2016-03-13 18:01:29 +00:00
_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
if not ('acc' in data and
max_gps_accuracy is not None and
data['acc'] > max_gps_accuracy):
see(**kwargs)
see_beacons(dev_id, kwargs)
else:
_LOGGER.info("Inaccurate GPS reported")
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
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
return True
2016-01-03 16:12:11 +00:00
2015-09-20 07:27:50 +00:00
def _parse_see_args(topic, data):
2016-03-07 17:12:06 +00:00
"""Parse the OwnTracks location parameters, into the format see expects."""
parts = topic.split('/')
dev_id = '{}_{}'.format(parts[1], parts[2])
host_name = parts[1]
kwargs = {
'dev_id': dev_id,
'host_name': host_name,
'gps': (data['lat'], data['lon'])
}
if 'acc' in data:
kwargs['gps_accuracy'] = data['acc']
if 'batt' in data:
kwargs['battery'] = data['batt']
return dev_id, kwargs
2016-03-13 18:01:29 +00:00
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