2019-01-11 02:20:35 +00:00
|
|
|
"""Bridge between emulated_roku and Home Assistant."""
|
|
|
|
import logging
|
|
|
|
|
2019-12-05 05:16:51 +00:00
|
|
|
from emulated_roku import EmulatedRokuCommandHandler, EmulatedRokuServer
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP
|
2019-01-11 02:20:35 +00:00
|
|
|
from homeassistant.core import CoreState, EventOrigin
|
|
|
|
|
2019-05-07 22:52:49 +00:00
|
|
|
LOGGER = logging.getLogger(__package__)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_ROKU_COMMAND = "roku_command"
|
2019-01-11 02:20:35 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_COMMAND_TYPE = "type"
|
|
|
|
ATTR_SOURCE_NAME = "source_name"
|
|
|
|
ATTR_KEY = "key"
|
|
|
|
ATTR_APP_ID = "app_id"
|
2019-01-11 02:20:35 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ROKU_COMMAND_KEYDOWN = "keydown"
|
|
|
|
ROKU_COMMAND_KEYUP = "keyup"
|
|
|
|
ROKU_COMMAND_KEYPRESS = "keypress"
|
|
|
|
ROKU_COMMAND_LAUNCH = "launch"
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EmulatedRoku:
|
|
|
|
"""Manages an emulated_roku server."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hass,
|
|
|
|
name,
|
|
|
|
host_ip,
|
|
|
|
listen_port,
|
|
|
|
advertise_ip,
|
|
|
|
advertise_port,
|
|
|
|
upnp_bind_multicast,
|
|
|
|
):
|
2019-01-11 02:20:35 +00:00
|
|
|
"""Initialize the properties."""
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
self.roku_usn = name
|
|
|
|
self.host_ip = host_ip
|
|
|
|
self.listen_port = listen_port
|
|
|
|
|
|
|
|
self.advertise_port = advertise_port
|
|
|
|
self.advertise_ip = advertise_ip
|
|
|
|
|
|
|
|
self.bind_multicast = upnp_bind_multicast
|
|
|
|
|
|
|
|
self._api_server = None
|
|
|
|
|
|
|
|
self._unsub_start_listener = None
|
|
|
|
self._unsub_stop_listener = None
|
|
|
|
|
|
|
|
async def setup(self):
|
|
|
|
"""Start the emulated_roku server."""
|
|
|
|
|
|
|
|
class EventCommandHandler(EmulatedRokuCommandHandler):
|
|
|
|
"""emulated_roku command handler to turn commands into events."""
|
|
|
|
|
|
|
|
def __init__(self, hass):
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
def on_keydown(self, roku_usn, key):
|
|
|
|
"""Handle keydown event."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_ROKU_COMMAND,
|
|
|
|
{
|
|
|
|
ATTR_SOURCE_NAME: roku_usn,
|
|
|
|
ATTR_COMMAND_TYPE: ROKU_COMMAND_KEYDOWN,
|
|
|
|
ATTR_KEY: key,
|
|
|
|
},
|
|
|
|
EventOrigin.local,
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
def on_keyup(self, roku_usn, key):
|
|
|
|
"""Handle keyup event."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_ROKU_COMMAND,
|
|
|
|
{
|
|
|
|
ATTR_SOURCE_NAME: roku_usn,
|
|
|
|
ATTR_COMMAND_TYPE: ROKU_COMMAND_KEYUP,
|
|
|
|
ATTR_KEY: key,
|
|
|
|
},
|
|
|
|
EventOrigin.local,
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
def on_keypress(self, roku_usn, key):
|
|
|
|
"""Handle keypress event."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_ROKU_COMMAND,
|
|
|
|
{
|
|
|
|
ATTR_SOURCE_NAME: roku_usn,
|
|
|
|
ATTR_COMMAND_TYPE: ROKU_COMMAND_KEYPRESS,
|
|
|
|
ATTR_KEY: key,
|
|
|
|
},
|
|
|
|
EventOrigin.local,
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
def launch(self, roku_usn, app_id):
|
|
|
|
"""Handle launch event."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_ROKU_COMMAND,
|
|
|
|
{
|
|
|
|
ATTR_SOURCE_NAME: roku_usn,
|
|
|
|
ATTR_COMMAND_TYPE: ROKU_COMMAND_LAUNCH,
|
|
|
|
ATTR_APP_ID: app_id,
|
|
|
|
},
|
|
|
|
EventOrigin.local,
|
|
|
|
)
|
|
|
|
|
|
|
|
LOGGER.debug(
|
2020-01-31 16:33:00 +00:00
|
|
|
"Initializing emulated_roku %s on %s:%s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.roku_usn,
|
|
|
|
self.host_ip,
|
|
|
|
self.listen_port,
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
handler = EventCommandHandler(self.hass)
|
|
|
|
|
|
|
|
self._api_server = EmulatedRokuServer(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.loop,
|
|
|
|
handler,
|
|
|
|
self.roku_usn,
|
|
|
|
self.host_ip,
|
|
|
|
self.listen_port,
|
2019-01-11 02:20:35 +00:00
|
|
|
advertise_ip=self.advertise_ip,
|
|
|
|
advertise_port=self.advertise_port,
|
2019-07-31 19:25:30 +00:00
|
|
|
bind_multicast=self.bind_multicast,
|
2019-01-11 02:20:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
async def emulated_roku_stop(event):
|
|
|
|
"""Wrap the call to emulated_roku.close."""
|
|
|
|
LOGGER.debug("Stopping emulated_roku %s", self.roku_usn)
|
|
|
|
self._unsub_stop_listener = None
|
|
|
|
await self._api_server.close()
|
|
|
|
|
|
|
|
async def emulated_roku_start(event):
|
|
|
|
"""Wrap the call to emulated_roku.start."""
|
|
|
|
try:
|
|
|
|
LOGGER.debug("Starting emulated_roku %s", self.roku_usn)
|
|
|
|
self._unsub_start_listener = None
|
|
|
|
await self._api_server.start()
|
|
|
|
except OSError:
|
2019-07-31 19:25:30 +00:00
|
|
|
LOGGER.exception(
|
|
|
|
"Failed to start Emulated Roku %s on %s:%s",
|
|
|
|
self.roku_usn,
|
|
|
|
self.host_ip,
|
|
|
|
self.listen_port,
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
# clean up inconsistent state on errors
|
|
|
|
await emulated_roku_stop(None)
|
|
|
|
else:
|
|
|
|
self._unsub_stop_listener = self.hass.bus.async_listen_once(
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_HOMEASSISTANT_STOP, emulated_roku_stop
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
# start immediately if already running
|
|
|
|
if self.hass.state == CoreState.running:
|
|
|
|
await emulated_roku_start(None)
|
|
|
|
else:
|
|
|
|
self._unsub_start_listener = self.hass.bus.async_listen_once(
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_HOMEASSISTANT_START, emulated_roku_start
|
|
|
|
)
|
2019-01-11 02:20:35 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
async def unload(self):
|
|
|
|
"""Unload the emulated_roku server."""
|
|
|
|
LOGGER.debug("Unloading emulated_roku %s", self.roku_usn)
|
|
|
|
|
|
|
|
if self._unsub_start_listener:
|
|
|
|
self._unsub_start_listener()
|
|
|
|
self._unsub_start_listener = None
|
|
|
|
|
|
|
|
if self._unsub_stop_listener:
|
|
|
|
self._unsub_stop_listener()
|
|
|
|
self._unsub_stop_listener = None
|
|
|
|
|
|
|
|
await self._api_server.close()
|
|
|
|
|
|
|
|
return True
|