core/homeassistant/components/lirc/__init__.py

79 lines
2.3 KiB
Python
Raw Normal View History

"""Support for LIRC devices."""
# pylint: disable=no-member, import-error
import logging
import threading
import time
import lirc
import voluptuous as vol
from homeassistant.const import EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP
_LOGGER = logging.getLogger(__name__)
2019-07-31 19:25:30 +00:00
BUTTON_NAME = "button_name"
2019-07-31 19:25:30 +00:00
DOMAIN = "lirc"
2019-07-31 19:25:30 +00:00
EVENT_IR_COMMAND_RECEIVED = "ir_command_received"
2019-07-31 19:25:30 +00:00
ICON = "mdi:remote"
2019-07-31 19:25:30 +00:00
CONFIG_SCHEMA = vol.Schema({DOMAIN: vol.Schema({})}, extra=vol.ALLOW_EXTRA)
2016-05-24 04:36:48 +00:00
def setup(hass, config):
"""Set up the LIRC capability."""
# blocking=True gives unexpected behavior (multiple responses for 1 press)
# also by not blocking, we allow hass to shut down the thread gracefully
# on exit.
2019-07-31 19:25:30 +00:00
lirc.init("home-assistant", blocking=False)
lirc_interface = LircInterface(hass)
2016-05-26 05:26:00 +00:00
def _start_lirc(_event):
lirc_interface.start()
def _stop_lirc(_event):
lirc_interface.stopped.set()
2016-05-26 05:26:00 +00:00
hass.bus.listen_once(EVENT_HOMEASSISTANT_START, _start_lirc)
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, _stop_lirc)
return True
class LircInterface(threading.Thread):
"""
This interfaces with the lirc daemon to read IR commands.
When using lirc in blocking mode, sometimes repeated commands get produced
in the next read of a command so we use a thread here to just wait
around until a non-empty response is obtained from lirc.
"""
def __init__(self, hass):
"""Construct a LIRC interface object."""
threading.Thread.__init__(self)
2016-05-26 05:26:00 +00:00
self.daemon = True
self.stopped = threading.Event()
self.hass = hass
def run(self):
"""Run the loop of the LIRC interface thread."""
_LOGGER.debug("LIRC interface thread started")
while not self.stopped.isSet():
try:
code = lirc.nextcode() # list; empty if no buttons pressed
except lirc.NextCodeError:
_LOGGER.warning("Error reading next code from LIRC")
code = None
# interpret result from python-lirc
if code:
code = code[0]
_LOGGER.info("Got new LIRC code %s", code)
2019-07-31 19:25:30 +00:00
self.hass.bus.fire(EVENT_IR_COMMAND_RECEIVED, {BUTTON_NAME: code})
else:
2016-05-26 05:26:00 +00:00
time.sleep(0.2)
lirc.deinit()
2019-07-31 19:25:30 +00:00
_LOGGER.debug("LIRC interface thread stopped")