2016-09-15 01:20:49 +00:00
|
|
|
"""
|
|
|
|
Component that connects to a Nuimo device over Bluetooth LE.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/nuimo_controller/
|
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
import threading
|
|
|
|
import time
|
2017-04-30 05:04:49 +00:00
|
|
|
|
2016-09-15 01:20:49 +00:00
|
|
|
import voluptuous as vol
|
2017-04-30 05:04:49 +00:00
|
|
|
|
2016-09-15 01:20:49 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.const import (CONF_MAC, CONF_NAME, EVENT_HOMEASSISTANT_STOP)
|
|
|
|
|
|
|
|
REQUIREMENTS = [
|
|
|
|
'--only-binary=all ' # avoid compilation of gattlib
|
2017-05-02 16:15:02 +00:00
|
|
|
'https://github.com/getSenic/nuimo-linux-python'
|
2016-12-03 18:18:00 +00:00
|
|
|
'/archive/29fc42987f74d8090d0e2382e8f248ff5990b8c9.zip'
|
2016-09-15 01:20:49 +00:00
|
|
|
'#nuimo==1.0.0']
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DOMAIN = 'nuimo_controller'
|
|
|
|
EVENT_NUIMO = 'nuimo_input'
|
|
|
|
|
|
|
|
DEFAULT_NAME = 'None'
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Optional(CONF_MAC): cv.string,
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
SERVICE_NUIMO = 'led_matrix'
|
|
|
|
DEFAULT_INTERVAL = 2.0
|
|
|
|
|
|
|
|
SERVICE_NUIMO_SCHEMA = vol.Schema({
|
|
|
|
vol.Required('matrix'): cv.string,
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional('interval', default=DEFAULT_INTERVAL): float
|
|
|
|
})
|
|
|
|
|
|
|
|
DEFAULT_ADAPTER = 'hci0'
|
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the Nuimo component."""
|
2016-09-15 01:20:49 +00:00
|
|
|
conf = config[DOMAIN]
|
|
|
|
mac = conf.get(CONF_MAC)
|
|
|
|
name = conf.get(CONF_NAME)
|
|
|
|
NuimoThread(hass, mac, name).start()
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
class NuimoLogger(object):
|
2016-09-15 01:20:49 +00:00
|
|
|
"""Handle Nuimo Controller event callbacks."""
|
|
|
|
|
|
|
|
def __init__(self, hass, name):
|
|
|
|
"""Initialize Logger object."""
|
|
|
|
self._hass = hass
|
|
|
|
self._name = name
|
|
|
|
|
|
|
|
def received_gesture_event(self, event):
|
|
|
|
"""Input Event received."""
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug("Received event: name=%s, gesture_id=%s,value=%s",
|
2016-09-15 01:20:49 +00:00
|
|
|
event.name, event.gesture, event.value)
|
|
|
|
self._hass.bus.fire(EVENT_NUIMO,
|
|
|
|
{'type': event.name, 'value': event.value,
|
|
|
|
'name': self._name})
|
|
|
|
|
|
|
|
|
|
|
|
class NuimoThread(threading.Thread):
|
|
|
|
"""Manage one Nuimo controller."""
|
|
|
|
|
|
|
|
def __init__(self, hass, mac, name):
|
|
|
|
"""Initialize thread object."""
|
|
|
|
super(NuimoThread, self).__init__()
|
|
|
|
self._hass = hass
|
|
|
|
self._mac = mac
|
|
|
|
self._name = name
|
|
|
|
self._hass_is_running = True
|
|
|
|
self._nuimo = None
|
2016-10-18 02:38:41 +00:00
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, self.stop)
|
2016-09-15 01:20:49 +00:00
|
|
|
|
|
|
|
def run(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the connection or be idle."""
|
2016-09-15 01:20:49 +00:00
|
|
|
while self._hass_is_running:
|
|
|
|
if not self._nuimo or not self._nuimo.is_connected():
|
|
|
|
self._attach()
|
|
|
|
self._connect()
|
|
|
|
else:
|
|
|
|
time.sleep(1)
|
|
|
|
|
|
|
|
if self._nuimo:
|
|
|
|
self._nuimo.disconnect()
|
|
|
|
self._nuimo = None
|
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=unused-argument
|
|
|
|
def stop(self, event):
|
2016-09-15 01:20:49 +00:00
|
|
|
"""Terminate Thread by unsetting flag."""
|
|
|
|
_LOGGER.debug('Stopping thread for Nuimo %s', self._mac)
|
|
|
|
self._hass_is_running = False
|
|
|
|
|
|
|
|
def _attach(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Create a Nuimo object from MAC address or discovery."""
|
2016-09-15 01:20:49 +00:00
|
|
|
# pylint: disable=import-error
|
|
|
|
from nuimo import NuimoController, NuimoDiscoveryManager
|
|
|
|
|
|
|
|
if self._nuimo:
|
|
|
|
self._nuimo.disconnect()
|
|
|
|
self._nuimo = None
|
|
|
|
|
|
|
|
if self._mac:
|
|
|
|
self._nuimo = NuimoController(self._mac)
|
|
|
|
else:
|
|
|
|
nuimo_manager = NuimoDiscoveryManager(
|
|
|
|
bluetooth_adapter=DEFAULT_ADAPTER, delegate=DiscoveryLogger())
|
|
|
|
nuimo_manager.start_discovery()
|
|
|
|
# Were any Nuimos found?
|
|
|
|
if not nuimo_manager.nuimos:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug("No Nuimo devices detected")
|
2016-09-15 01:20:49 +00:00
|
|
|
return
|
|
|
|
# Take the first Nuimo found.
|
|
|
|
self._nuimo = nuimo_manager.nuimos[0]
|
|
|
|
self._mac = self._nuimo.addr
|
|
|
|
|
|
|
|
def _connect(self):
|
|
|
|
"""Build up connection and set event delegator and service."""
|
|
|
|
if not self._nuimo:
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
self._nuimo.connect()
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug("Connected to %s", self._mac)
|
2016-09-15 01:20:49 +00:00
|
|
|
except RuntimeError as error:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.error("Could not connect to %s: %s", self._mac, error)
|
2016-09-15 01:20:49 +00:00
|
|
|
time.sleep(1)
|
|
|
|
return
|
|
|
|
|
|
|
|
nuimo_event_delegate = NuimoLogger(self._hass, self._name)
|
|
|
|
self._nuimo.set_delegate(nuimo_event_delegate)
|
|
|
|
|
|
|
|
def handle_write_matrix(call):
|
|
|
|
"""Handle led matrix service."""
|
|
|
|
matrix = call.data.get('matrix', None)
|
|
|
|
name = call.data.get(CONF_NAME, DEFAULT_NAME)
|
|
|
|
interval = call.data.get('interval', DEFAULT_INTERVAL)
|
|
|
|
if self._name == name and matrix:
|
|
|
|
self._nuimo.write_matrix(matrix, interval)
|
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
self._hass.services.register(
|
|
|
|
DOMAIN, SERVICE_NUIMO, handle_write_matrix,
|
|
|
|
schema=SERVICE_NUIMO_SCHEMA)
|
2016-09-15 01:20:49 +00:00
|
|
|
|
|
|
|
self._nuimo.write_matrix(HOMEASSIST_LOGO, 2.0)
|
|
|
|
|
|
|
|
|
|
|
|
# must be 9x9 matrix
|
|
|
|
HOMEASSIST_LOGO = (
|
|
|
|
" . " +
|
|
|
|
" ... " +
|
|
|
|
" ..... " +
|
|
|
|
" ....... " +
|
|
|
|
"..... ..." +
|
|
|
|
" ....... " +
|
|
|
|
" .. .... " +
|
|
|
|
" .. .... " +
|
|
|
|
".........")
|
|
|
|
|
|
|
|
|
|
|
|
class DiscoveryLogger(object):
|
|
|
|
"""Handle Nuimo Discovery callbacks."""
|
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=no-self-use
|
|
|
|
def discovery_started(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Discovery started."""
|
|
|
|
_LOGGER.info("Started discovery")
|
2016-09-15 01:20:49 +00:00
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=no-self-use
|
|
|
|
def discovery_finished(self):
|
2016-09-15 01:20:49 +00:00
|
|
|
"""Discovery finished."""
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.info("Finished discovery")
|
2016-09-15 01:20:49 +00:00
|
|
|
|
2016-10-30 21:18:53 +00:00
|
|
|
# pylint: disable=no-self-use
|
|
|
|
def controller_added(self, nuimo):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return that a controller was found."""
|
|
|
|
_LOGGER.info("Added Nuimo: %s", nuimo)
|