2019-02-13 20:21:14 +00:00
|
|
|
"""Support for the Xiaomi IR Remote (Chuangmi IR)."""
|
2021-12-27 23:15:40 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2018-02-06 18:47:24 +00:00
|
|
|
import asyncio
|
2019-11-08 17:32:44 +00:00
|
|
|
from datetime import timedelta
|
2018-02-06 18:47:24 +00:00
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
|
2021-03-02 08:02:04 +00:00
|
|
|
from miio import ChuangmiIr, DeviceException
|
2018-02-06 18:47:24 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-01-13 07:45:30 +00:00
|
|
|
from homeassistant.components import persistent_notification
|
2018-02-06 18:47:24 +00:00
|
|
|
from homeassistant.components.remote import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_DELAY_SECS,
|
2019-11-08 17:32:44 +00:00
|
|
|
ATTR_NUM_REPEATS,
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_DELAY_SECS,
|
2019-11-08 17:32:44 +00:00
|
|
|
PLATFORM_SCHEMA,
|
2020-04-26 00:12:36 +00:00
|
|
|
RemoteEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_COMMAND,
|
2019-11-08 17:32:44 +00:00
|
|
|
CONF_HOST,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_TIMEOUT,
|
|
|
|
CONF_TOKEN,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-12-27 23:15:40 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2018-03-16 20:15:23 +00:00
|
|
|
from homeassistant.exceptions import PlatformNotReady
|
2020-05-19 22:23:18 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv, entity_platform
|
2021-12-27 23:15:40 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2018-02-06 18:47:24 +00:00
|
|
|
from homeassistant.util.dt import utcnow
|
|
|
|
|
2020-07-08 19:29:51 +00:00
|
|
|
from .const import SERVICE_LEARN, SERVICE_SET_REMOTE_LED_OFF, SERVICE_SET_REMOTE_LED_ON
|
2019-12-02 19:49:39 +00:00
|
|
|
|
2018-02-06 18:47:24 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_KEY = "remote.xiaomi_miio"
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_SLOT = "slot"
|
|
|
|
CONF_COMMANDS = "commands"
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
DEFAULT_TIMEOUT = 10
|
|
|
|
DEFAULT_SLOT = 1
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
COMMAND_SCHEMA = vol.Schema(
|
|
|
|
{vol.Required(CONF_COMMAND): vol.All(cv.ensure_list, [cv.string])}
|
|
|
|
)
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
2020-10-11 20:04:49 +00:00
|
|
|
vol.Optional(CONF_TIMEOUT, default=DEFAULT_TIMEOUT): cv.positive_int,
|
2019-07-31 19:25:30 +00:00
|
|
|
vol.Optional(CONF_SLOT, default=DEFAULT_SLOT): vol.All(
|
|
|
|
int, vol.Range(min=1, max=1000000)
|
|
|
|
),
|
|
|
|
vol.Required(CONF_TOKEN): vol.All(str, vol.Length(min=32, max=32)),
|
|
|
|
vol.Optional(CONF_COMMANDS, default={}): cv.schema_with_slug_keys(
|
|
|
|
COMMAND_SCHEMA
|
|
|
|
),
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-12-27 23:15:40 +00:00
|
|
|
async def async_setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2018-02-06 18:47:24 +00:00
|
|
|
"""Set up the Xiaomi IR Remote (Chuangmi IR) platform."""
|
2019-10-29 00:45:22 +00:00
|
|
|
host = config[CONF_HOST]
|
|
|
|
token = config[CONF_TOKEN]
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
# Create handler
|
|
|
|
_LOGGER.info("Initializing with host %s (token %s...)", host, token[:5])
|
2018-03-05 06:06:00 +00:00
|
|
|
|
|
|
|
# The Chuang Mi IR Remote Controller wants to be re-discovered every
|
|
|
|
# 5 minutes. As long as polling is disabled the device should be
|
|
|
|
# re-discovered (lazy_discover=False) in front of every command.
|
|
|
|
device = ChuangmiIr(host, token, lazy_discover=False)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
# Check that we can communicate with device.
|
|
|
|
try:
|
2019-10-29 00:44:26 +00:00
|
|
|
device_info = await hass.async_add_executor_job(device.info)
|
2018-03-16 20:15:23 +00:00
|
|
|
model = device_info.model
|
2019-09-03 19:15:31 +00:00
|
|
|
unique_id = f"{model}-{device_info.mac_address}"
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.info(
|
|
|
|
"%s %s %s detected",
|
|
|
|
model,
|
|
|
|
device_info.firmware_version,
|
|
|
|
device_info.hardware_version,
|
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
except DeviceException as ex:
|
2018-03-16 20:15:23 +00:00
|
|
|
_LOGGER.error("Device unavailable or token incorrect: %s", ex)
|
2020-08-28 11:50:32 +00:00
|
|
|
raise PlatformNotReady from ex
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2018-03-05 06:06:00 +00:00
|
|
|
if DATA_KEY not in hass.data:
|
|
|
|
hass.data[DATA_KEY] = {}
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2020-04-04 21:09:34 +00:00
|
|
|
friendly_name = config.get(CONF_NAME, f"xiaomi_miio_{host.replace('.', '_')}")
|
2018-02-06 18:47:24 +00:00
|
|
|
slot = config.get(CONF_SLOT)
|
|
|
|
timeout = config.get(CONF_TIMEOUT)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
xiaomi_miio_remote = XiaomiMiioRemote(
|
2020-04-04 21:09:34 +00:00
|
|
|
friendly_name, device, unique_id, slot, timeout, config.get(CONF_COMMANDS)
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2018-03-05 06:06:00 +00:00
|
|
|
hass.data[DATA_KEY][host] = xiaomi_miio_remote
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities([xiaomi_miio_remote])
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2020-05-19 22:23:18 +00:00
|
|
|
async def async_service_led_off_handler(entity, service):
|
|
|
|
"""Handle set_led_off command."""
|
|
|
|
await hass.async_add_executor_job(entity.device.set_indicator_led, False)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2020-05-19 22:23:18 +00:00
|
|
|
async def async_service_led_on_handler(entity, service):
|
|
|
|
"""Handle set_led_on command."""
|
|
|
|
await hass.async_add_executor_job(entity.device.set_indicator_led, True)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2020-05-19 22:23:18 +00:00
|
|
|
async def async_service_learn_handler(entity, service):
|
|
|
|
"""Handle a learn command."""
|
2018-02-06 18:47:24 +00:00
|
|
|
device = entity.device
|
|
|
|
|
|
|
|
slot = service.data.get(CONF_SLOT, entity.slot)
|
|
|
|
|
2018-11-07 08:03:35 +00:00
|
|
|
await hass.async_add_executor_job(device.learn, slot)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
timeout = service.data.get(CONF_TIMEOUT, entity.timeout)
|
|
|
|
|
|
|
|
_LOGGER.info("Press the key you want Home Assistant to learn")
|
|
|
|
start_time = utcnow()
|
|
|
|
while (utcnow() - start_time) < timedelta(seconds=timeout):
|
2019-07-31 19:25:30 +00:00
|
|
|
message = await hass.async_add_executor_job(device.read, slot)
|
2018-04-04 21:30:02 +00:00
|
|
|
_LOGGER.debug("Message received from device: '%s'", message)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if "code" in message and message["code"]:
|
|
|
|
log_msg = "Received command is: {}".format(message["code"])
|
2018-02-06 18:47:24 +00:00
|
|
|
_LOGGER.info(log_msg)
|
2022-01-13 07:45:30 +00:00
|
|
|
persistent_notification.async_create(
|
|
|
|
hass, log_msg, title="Xiaomi Miio Remote"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if "error" in message and message["error"]["message"] == "learn timeout":
|
2018-11-07 08:03:35 +00:00
|
|
|
await hass.async_add_executor_job(device.learn, slot)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2019-05-23 04:09:59 +00:00
|
|
|
await asyncio.sleep(1)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
_LOGGER.error("Timeout. No infrared command captured")
|
2022-01-13 07:45:30 +00:00
|
|
|
persistent_notification.async_create(
|
|
|
|
hass, "Timeout. No infrared command captured", title="Xiaomi Miio Remote"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2021-05-03 16:34:28 +00:00
|
|
|
platform = entity_platform.async_get_current_platform()
|
2020-05-19 22:23:18 +00:00
|
|
|
|
|
|
|
platform.async_register_entity_service(
|
|
|
|
SERVICE_LEARN,
|
|
|
|
{
|
2020-10-11 20:04:49 +00:00
|
|
|
vol.Optional(CONF_TIMEOUT, default=10): cv.positive_int,
|
2020-05-19 22:23:18 +00:00
|
|
|
vol.Optional(CONF_SLOT, default=1): vol.All(
|
|
|
|
int, vol.Range(min=1, max=1000000)
|
|
|
|
),
|
|
|
|
},
|
|
|
|
async_service_learn_handler,
|
|
|
|
)
|
|
|
|
platform.async_register_entity_service(
|
2020-08-27 11:56:20 +00:00
|
|
|
SERVICE_SET_REMOTE_LED_ON,
|
|
|
|
{},
|
|
|
|
async_service_led_on_handler,
|
2020-05-19 22:23:18 +00:00
|
|
|
)
|
|
|
|
platform.async_register_entity_service(
|
2020-08-27 11:56:20 +00:00
|
|
|
SERVICE_SET_REMOTE_LED_OFF,
|
|
|
|
{},
|
|
|
|
async_service_led_off_handler,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
|
2020-04-26 00:12:36 +00:00
|
|
|
class XiaomiMiioRemote(RemoteEntity):
|
2018-02-06 18:47:24 +00:00
|
|
|
"""Representation of a Xiaomi Miio Remote device."""
|
|
|
|
|
2020-01-13 12:50:01 +00:00
|
|
|
def __init__(self, friendly_name, device, unique_id, slot, timeout, commands):
|
2018-02-06 18:47:24 +00:00
|
|
|
"""Initialize the remote."""
|
|
|
|
self._name = friendly_name
|
|
|
|
self._device = device
|
2018-03-16 20:15:23 +00:00
|
|
|
self._unique_id = unique_id
|
2018-02-06 18:47:24 +00:00
|
|
|
self._slot = slot
|
|
|
|
self._timeout = timeout
|
|
|
|
self._state = False
|
|
|
|
self._commands = commands
|
|
|
|
|
2018-03-16 20:15:23 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return an unique ID."""
|
|
|
|
return self._unique_id
|
|
|
|
|
2018-02-06 18:47:24 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the remote."""
|
|
|
|
return self._name
|
|
|
|
|
2018-08-26 19:25:39 +00:00
|
|
|
@property
|
|
|
|
def device(self):
|
|
|
|
"""Return the remote object."""
|
|
|
|
return self._device
|
|
|
|
|
2018-02-06 18:47:24 +00:00
|
|
|
@property
|
|
|
|
def slot(self):
|
|
|
|
"""Return the slot to save learned command."""
|
|
|
|
return self._slot
|
|
|
|
|
|
|
|
@property
|
|
|
|
def timeout(self):
|
|
|
|
"""Return the timeout for learning command."""
|
|
|
|
return self._timeout
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return False if device is unreachable, else True."""
|
|
|
|
try:
|
2018-08-26 19:25:39 +00:00
|
|
|
self.device.info()
|
2018-02-06 18:47:24 +00:00
|
|
|
return True
|
|
|
|
except DeviceException:
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""We should not be polled for device up state."""
|
|
|
|
return False
|
|
|
|
|
2018-10-01 06:56:50 +00:00
|
|
|
async def async_turn_on(self, **kwargs):
|
2018-02-06 18:47:24 +00:00
|
|
|
"""Turn the device on."""
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Device does not support turn_on, "
|
2020-07-05 21:04:19 +00:00
|
|
|
"please use 'remote.send_command' to send commands"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
2018-10-01 06:56:50 +00:00
|
|
|
async def async_turn_off(self, **kwargs):
|
2018-02-06 18:47:24 +00:00
|
|
|
"""Turn the device off."""
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Device does not support turn_off, "
|
2020-07-05 21:04:19 +00:00
|
|
|
"please use 'remote.send_command' to send commands"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
def _send_command(self, payload):
|
|
|
|
"""Send a command."""
|
|
|
|
_LOGGER.debug("Sending payload: '%s'", payload)
|
|
|
|
try:
|
2018-08-26 19:25:39 +00:00
|
|
|
self.device.play(payload)
|
2018-02-06 18:47:24 +00:00
|
|
|
except DeviceException as ex:
|
|
|
|
_LOGGER.error(
|
2019-07-31 19:25:30 +00:00
|
|
|
"Transmit of IR command failed, %s, exception: %s", payload, ex
|
|
|
|
)
|
2018-02-06 18:47:24 +00:00
|
|
|
|
|
|
|
def send_command(self, command, **kwargs):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Send a command."""
|
2018-02-06 18:47:24 +00:00
|
|
|
num_repeats = kwargs.get(ATTR_NUM_REPEATS)
|
|
|
|
|
|
|
|
delay = kwargs.get(ATTR_DELAY_SECS, DEFAULT_DELAY_SECS)
|
|
|
|
|
|
|
|
for _ in range(num_repeats):
|
|
|
|
for payload in command:
|
|
|
|
if payload in self._commands:
|
|
|
|
for local_payload in self._commands[payload][CONF_COMMAND]:
|
|
|
|
self._send_command(local_payload)
|
|
|
|
else:
|
|
|
|
self._send_command(payload)
|
|
|
|
time.sleep(delay)
|