2019-02-14 04:35:12 +00:00
|
|
|
"""KIRA interface to receive UDP packets from an IR-IP bridge."""
|
2017-05-13 04:12:47 +00:00
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
|
2019-10-21 08:24:06 +00:00
|
|
|
import pykira
|
2017-05-13 04:12:47 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
from voluptuous.error import Error as VoluptuousError
|
2018-01-21 06:35:38 +00:00
|
|
|
import yaml
|
2017-05-13 04:12:47 +00:00
|
|
|
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.const import (
|
2019-10-21 08:24:06 +00:00
|
|
|
CONF_CODE,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_DEVICE,
|
|
|
|
CONF_HOST,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PORT,
|
2021-02-11 10:04:11 +00:00
|
|
|
CONF_REPEAT,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_SENSORS,
|
|
|
|
CONF_TYPE,
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
STATE_UNKNOWN,
|
|
|
|
)
|
2017-05-13 04:12:47 +00:00
|
|
|
from homeassistant.helpers import discovery
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "kira"
|
2017-05-13 04:12:47 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DEFAULT_HOST = "0.0.0.0"
|
|
|
|
DEFAULT_PORT = 65432
|
|
|
|
|
|
|
|
CONF_REMOTES = "remotes"
|
|
|
|
CONF_SENSOR = "sensor"
|
|
|
|
CONF_REMOTE = "remote"
|
|
|
|
|
2019-09-03 19:14:00 +00:00
|
|
|
CODES_YAML = f"{DOMAIN}_codes.yaml"
|
2019-07-31 19:25:30 +00:00
|
|
|
|
|
|
|
CODE_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_CODE): cv.string,
|
|
|
|
vol.Optional(CONF_TYPE): cv.string,
|
|
|
|
vol.Optional(CONF_DEVICE): cv.string,
|
|
|
|
vol.Optional(CONF_REPEAT): cv.positive_int,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
SENSOR_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_NAME, default=DOMAIN): vol.Exclusive(cv.string, "sensors"),
|
|
|
|
vol.Optional(CONF_HOST, default=DEFAULT_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
REMOTE_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_NAME, default=DOMAIN): vol.Exclusive(cv.string, "remotes"),
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_SENSORS): [SENSOR_SCHEMA],
|
|
|
|
vol.Optional(CONF_REMOTES): [REMOTE_SCHEMA],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2017-05-13 04:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
def load_codes(path):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Load KIRA codes from specified file."""
|
2017-05-13 04:12:47 +00:00
|
|
|
codes = []
|
|
|
|
if os.path.exists(path):
|
2021-07-28 07:41:45 +00:00
|
|
|
with open(path, encoding="utf8") as code_file:
|
2019-06-14 22:30:47 +00:00
|
|
|
data = yaml.safe_load(code_file) or []
|
2017-05-13 04:12:47 +00:00
|
|
|
for code in data:
|
|
|
|
try:
|
|
|
|
codes.append(CODE_SCHEMA(code))
|
|
|
|
except VoluptuousError as exception:
|
|
|
|
# keep going
|
2018-01-21 06:35:38 +00:00
|
|
|
_LOGGER.warning("KIRA code invalid data: %s", exception)
|
2017-05-13 04:12:47 +00:00
|
|
|
else:
|
2021-07-28 07:41:45 +00:00
|
|
|
with open(path, "w", encoding="utf8") as code_file:
|
2019-07-31 19:25:30 +00:00
|
|
|
code_file.write("")
|
2017-05-13 04:12:47 +00:00
|
|
|
return codes
|
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up the KIRA component."""
|
2017-05-13 04:12:47 +00:00
|
|
|
sensors = config.get(DOMAIN, {}).get(CONF_SENSORS, [])
|
|
|
|
remotes = config.get(DOMAIN, {}).get(CONF_REMOTES, [])
|
|
|
|
# If no sensors or remotes were specified, add a sensor
|
2019-07-31 19:25:30 +00:00
|
|
|
if not (sensors or remotes):
|
2017-05-13 04:12:47 +00:00
|
|
|
sensors.append({})
|
|
|
|
|
|
|
|
codes = load_codes(hass.config.path(CODES_YAML))
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.data[DOMAIN] = {CONF_SENSOR: {}, CONF_REMOTE: {}}
|
2017-05-13 04:12:47 +00:00
|
|
|
|
|
|
|
def load_module(platform, idx, module_conf):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Set up the KIRA module and load platform."""
|
2017-05-13 04:12:47 +00:00
|
|
|
# note: module_name is not the HA device name. it's just a unique name
|
|
|
|
# to ensure the component and platform can share information
|
|
|
|
module_name = ("%s_%d" % (DOMAIN, idx)) if idx else DOMAIN
|
|
|
|
device_name = module_conf.get(CONF_NAME, DOMAIN)
|
|
|
|
port = module_conf.get(CONF_PORT, DEFAULT_PORT)
|
|
|
|
host = module_conf.get(CONF_HOST, DEFAULT_HOST)
|
|
|
|
|
|
|
|
if platform == CONF_SENSOR:
|
|
|
|
module = pykira.KiraReceiver(host, port)
|
|
|
|
module.start()
|
|
|
|
else:
|
|
|
|
module = pykira.KiraModule(host, port)
|
|
|
|
|
|
|
|
hass.data[DOMAIN][platform][module_name] = module
|
|
|
|
for code in codes:
|
2019-07-31 19:25:30 +00:00
|
|
|
code_tuple = (code.get(CONF_NAME), code.get(CONF_DEVICE, STATE_UNKNOWN))
|
2017-05-13 04:12:47 +00:00
|
|
|
module.registerCode(code_tuple, code.get(CONF_CODE))
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
discovery.load_platform(
|
|
|
|
hass, platform, DOMAIN, {"name": module_name, "device": device_name}, config
|
|
|
|
)
|
2017-05-13 04:12:47 +00:00
|
|
|
|
|
|
|
for idx, module_conf in enumerate(sensors):
|
|
|
|
load_module(CONF_SENSOR, idx, module_conf)
|
|
|
|
|
|
|
|
for idx, module_conf in enumerate(remotes):
|
|
|
|
load_module(CONF_REMOTE, idx, module_conf)
|
|
|
|
|
|
|
|
def _stop_kira(_event):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Stop the KIRA receiver."""
|
2017-05-13 04:12:47 +00:00
|
|
|
for receiver in hass.data[DOMAIN][CONF_SENSOR].values():
|
|
|
|
receiver.stop()
|
|
|
|
_LOGGER.info("Terminated receivers")
|
|
|
|
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, _stop_kira)
|
|
|
|
|
|
|
|
return True
|