2019-04-03 15:40:03 +00:00
|
|
|
"""Tracking for bluetooth devices."""
|
2016-04-19 15:18:46 +00:00
|
|
|
import logging
|
|
|
|
|
2016-09-04 17:10:20 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-04-19 15:18:46 +00:00
|
|
|
from homeassistant.helpers.event import track_point_in_utc_time
|
|
|
|
from homeassistant.components.device_tracker import (
|
2016-09-04 17:10:20 +00:00
|
|
|
YAML_DEVICES, CONF_TRACK_NEW, CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL,
|
2018-08-27 07:08:23 +00:00
|
|
|
load_config, PLATFORM_SCHEMA, DEFAULT_TRACK_NEW, SOURCE_TYPE_BLUETOOTH,
|
|
|
|
DOMAIN)
|
2016-04-19 15:18:46 +00:00
|
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
BT_PREFIX = 'BT_'
|
|
|
|
|
2018-03-31 21:22:54 +00:00
|
|
|
CONF_REQUEST_RSSI = 'request_rssi'
|
|
|
|
|
2018-12-25 17:29:44 +00:00
|
|
|
CONF_DEVICE_ID = "device_id"
|
|
|
|
|
|
|
|
DEFAULT_DEVICE_ID = -1
|
|
|
|
|
2016-09-04 17:10:20 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
2018-03-31 21:22:54 +00:00
|
|
|
vol.Optional(CONF_TRACK_NEW): cv.boolean,
|
2018-12-25 17:29:44 +00:00
|
|
|
vol.Optional(CONF_REQUEST_RSSI): cv.boolean,
|
|
|
|
vol.Optional(CONF_DEVICE_ID, default=DEFAULT_DEVICE_ID):
|
|
|
|
vol.All(vol.Coerce(int), vol.Range(min=-1))
|
2016-09-04 17:10:20 +00:00
|
|
|
})
|
|
|
|
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2017-02-07 19:47:11 +00:00
|
|
|
def setup_scanner(hass, config, see, discovery_info=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the Bluetooth Scanner."""
|
2016-04-19 15:18:46 +00:00
|
|
|
# pylint: disable=import-error
|
|
|
|
import bluetooth
|
2018-03-31 21:22:54 +00:00
|
|
|
from bt_proximity import BluetoothRSSI
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2018-03-31 21:22:54 +00:00
|
|
|
def see_device(mac, name, rssi=None):
|
2016-04-19 15:18:46 +00:00
|
|
|
"""Mark a device as seen."""
|
2018-03-31 21:22:54 +00:00
|
|
|
attributes = {}
|
|
|
|
if rssi is not None:
|
|
|
|
attributes['rssi'] = rssi
|
2018-04-15 11:59:10 +00:00
|
|
|
see(mac="{}{}".format(BT_PREFIX, mac), host_name=name,
|
2018-03-31 21:22:54 +00:00
|
|
|
attributes=attributes, source_type=SOURCE_TYPE_BLUETOOTH)
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2018-12-25 17:29:44 +00:00
|
|
|
device_id = config.get(CONF_DEVICE_ID)
|
|
|
|
|
2016-04-19 15:18:46 +00:00
|
|
|
def discover_devices():
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Discover Bluetooth devices."""
|
|
|
|
result = bluetooth.discover_devices(
|
|
|
|
duration=8, lookup_names=True, flush_cache=True,
|
2018-12-25 17:29:44 +00:00
|
|
|
lookup_class=False, device_id=device_id)
|
2018-02-11 17:20:28 +00:00
|
|
|
_LOGGER.debug("Bluetooth devices discovered = %d", len(result))
|
2016-04-19 15:18:46 +00:00
|
|
|
return result
|
|
|
|
|
|
|
|
yaml_path = hass.config.path(YAML_DEVICES)
|
|
|
|
devs_to_track = []
|
|
|
|
devs_donot_track = []
|
|
|
|
|
|
|
|
# Load all known devices.
|
|
|
|
# We just need the devices so set consider_home and home range
|
|
|
|
# to 0
|
2016-08-30 16:22:52 +00:00
|
|
|
for device in load_config(yaml_path, hass, 0):
|
2017-04-30 05:04:49 +00:00
|
|
|
# Check if device is a valid bluetooth device
|
2016-04-19 15:18:46 +00:00
|
|
|
if device.mac and device.mac[:3].upper() == BT_PREFIX:
|
|
|
|
if device.track:
|
|
|
|
devs_to_track.append(device.mac[3:])
|
|
|
|
else:
|
|
|
|
devs_donot_track.append(device.mac[3:])
|
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
# If track new devices is true discover new devices on startup.
|
2016-09-06 17:51:36 +00:00
|
|
|
track_new = config.get(CONF_TRACK_NEW, DEFAULT_TRACK_NEW)
|
2016-04-19 15:18:46 +00:00
|
|
|
if track_new:
|
|
|
|
for dev in discover_devices():
|
|
|
|
if dev[0] not in devs_to_track and \
|
2018-03-31 21:22:54 +00:00
|
|
|
dev[0] not in devs_donot_track:
|
2016-04-19 15:18:46 +00:00
|
|
|
devs_to_track.append(dev[0])
|
2018-03-31 21:22:54 +00:00
|
|
|
see_device(dev[0], dev[1])
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2016-09-04 17:10:20 +00:00
|
|
|
interval = config.get(CONF_SCAN_INTERVAL, DEFAULT_SCAN_INTERVAL)
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2018-03-31 21:22:54 +00:00
|
|
|
request_rssi = config.get(CONF_REQUEST_RSSI, False)
|
|
|
|
|
2018-09-13 05:52:31 +00:00
|
|
|
def update_bluetooth(_):
|
2018-08-27 07:08:23 +00:00
|
|
|
"""Update Bluetooth and set timer for the next update."""
|
|
|
|
update_bluetooth_once()
|
|
|
|
track_point_in_utc_time(
|
|
|
|
hass, update_bluetooth, dt_util.utcnow() + interval)
|
|
|
|
|
|
|
|
def update_bluetooth_once():
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Lookup Bluetooth device and update status."""
|
2016-04-19 15:18:46 +00:00
|
|
|
try:
|
2016-09-06 17:51:36 +00:00
|
|
|
if track_new:
|
|
|
|
for dev in discover_devices():
|
|
|
|
if dev[0] not in devs_to_track and \
|
2018-03-31 21:22:54 +00:00
|
|
|
dev[0] not in devs_donot_track:
|
2016-09-06 17:51:36 +00:00
|
|
|
devs_to_track.append(dev[0])
|
2016-04-19 15:18:46 +00:00
|
|
|
for mac in devs_to_track:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug("Scanning %s", mac)
|
2016-04-19 15:18:46 +00:00
|
|
|
result = bluetooth.lookup_name(mac, timeout=5)
|
2018-03-31 21:22:54 +00:00
|
|
|
rssi = None
|
|
|
|
if request_rssi:
|
|
|
|
rssi = BluetoothRSSI(mac).request_rssi()
|
|
|
|
if result is None:
|
2016-04-19 15:18:46 +00:00
|
|
|
# Could not lookup device name
|
|
|
|
continue
|
2018-03-31 21:22:54 +00:00
|
|
|
see_device(mac, result, rssi)
|
2016-04-19 15:18:46 +00:00
|
|
|
except bluetooth.BluetoothError:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.exception("Error looking up Bluetooth device")
|
2016-04-19 15:18:46 +00:00
|
|
|
|
2018-08-27 07:08:23 +00:00
|
|
|
def handle_update_bluetooth(call):
|
|
|
|
"""Update bluetooth devices on demand."""
|
|
|
|
update_bluetooth_once()
|
|
|
|
|
2018-09-13 05:52:31 +00:00
|
|
|
update_bluetooth(dt_util.utcnow())
|
2018-08-27 07:08:23 +00:00
|
|
|
|
|
|
|
hass.services.register(
|
|
|
|
DOMAIN, "bluetooth_tracker_update", handle_update_bluetooth)
|
2016-04-19 15:18:46 +00:00
|
|
|
|
|
|
|
return True
|