2016-09-15 12:35:40 +00:00
|
|
|
"""
|
2017-04-30 05:04:49 +00:00
|
|
|
Component that will help set the FFmpeg component.
|
2016-09-15 12:35:40 +00:00
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/ffmpeg/
|
|
|
|
"""
|
2016-10-24 06:48:01 +00:00
|
|
|
import asyncio
|
2016-09-15 12:35:40 +00:00
|
|
|
import logging
|
2017-01-31 15:48:03 +00:00
|
|
|
import os
|
2016-09-15 12:35:40 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_ENTITY_ID, EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STOP)
|
|
|
|
from homeassistant.config import load_yaml_config_file
|
2017-02-26 22:31:46 +00:00
|
|
|
from homeassistant.helpers.dispatcher import (
|
|
|
|
async_dispatcher_send, async_dispatcher_connect)
|
2016-09-15 12:35:40 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2017-01-31 15:48:03 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2016-09-15 12:35:40 +00:00
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
REQUIREMENTS = ['ha-ffmpeg==1.5']
|
|
|
|
|
2016-09-15 12:35:40 +00:00
|
|
|
DOMAIN = 'ffmpeg'
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
SERVICE_START = 'start'
|
|
|
|
SERVICE_STOP = 'stop'
|
|
|
|
SERVICE_RESTART = 'restart'
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
SIGNAL_FFMPEG_START = 'ffmpeg.start'
|
|
|
|
SIGNAL_FFMPEG_STOP = 'ffmpeg.stop'
|
|
|
|
SIGNAL_FFMPEG_RESTART = 'ffmpeg.restart'
|
|
|
|
|
2017-01-21 05:56:22 +00:00
|
|
|
DATA_FFMPEG = 'ffmpeg'
|
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
CONF_INITIAL_STATE = 'initial_state'
|
2016-09-15 12:35:40 +00:00
|
|
|
CONF_INPUT = 'input'
|
|
|
|
CONF_FFMPEG_BIN = 'ffmpeg_bin'
|
|
|
|
CONF_EXTRA_ARGUMENTS = 'extra_arguments'
|
|
|
|
CONF_OUTPUT = 'output'
|
|
|
|
CONF_RUN_TEST = 'run_test'
|
|
|
|
|
|
|
|
DEFAULT_BINARY = 'ffmpeg'
|
|
|
|
DEFAULT_RUN_TEST = True
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Optional(CONF_FFMPEG_BIN, default=DEFAULT_BINARY): cv.string,
|
|
|
|
vol.Optional(CONF_RUN_TEST, default=DEFAULT_RUN_TEST): cv.boolean,
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
SERVICE_FFMPEG_SCHEMA = vol.Schema({
|
|
|
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@callback
|
|
|
|
def async_start(hass, entity_id=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Start a FFmpeg process on entity."""
|
2017-01-31 15:48:03 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
2017-02-26 22:31:46 +00:00
|
|
|
hass.async_add_job(hass.services.async_call(DOMAIN, SERVICE_START, data))
|
2017-01-31 15:48:03 +00:00
|
|
|
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@callback
|
|
|
|
def async_stop(hass, entity_id=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Stop a FFmpeg process on entity."""
|
2017-01-31 15:48:03 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
2017-02-26 22:31:46 +00:00
|
|
|
hass.async_add_job(hass.services.async_call(DOMAIN, SERVICE_STOP, data))
|
2017-01-31 15:48:03 +00:00
|
|
|
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@callback
|
|
|
|
def async_restart(hass, entity_id=None):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Restart a FFmpeg process on entity."""
|
2017-01-31 15:48:03 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
2017-02-26 22:31:46 +00:00
|
|
|
hass.async_add_job(hass.services.async_call(DOMAIN, SERVICE_RESTART, data))
|
2017-01-31 15:48:03 +00:00
|
|
|
|
2016-09-15 12:35:40 +00:00
|
|
|
|
2017-01-21 05:56:22 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the FFmpeg component."""
|
2017-01-21 05:56:22 +00:00
|
|
|
conf = config.get(DOMAIN, {})
|
2016-09-15 12:35:40 +00:00
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
manager = FFmpegManager(
|
2017-01-21 05:56:22 +00:00
|
|
|
hass,
|
|
|
|
conf.get(CONF_FFMPEG_BIN, DEFAULT_BINARY),
|
|
|
|
conf.get(CONF_RUN_TEST, DEFAULT_RUN_TEST)
|
|
|
|
)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2017-05-26 15:28:07 +00:00
|
|
|
descriptions = yield from hass.async_add_job(
|
|
|
|
load_yaml_config_file,
|
2017-01-31 15:48:03 +00:00
|
|
|
os.path.join(os.path.dirname(__file__), 'services.yaml'))
|
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
# Register service
|
2017-01-31 15:48:03 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_service_handle(service):
|
|
|
|
"""Handle service ffmpeg process."""
|
|
|
|
entity_ids = service.data.get(ATTR_ENTITY_ID)
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
if service.service == SERVICE_START:
|
|
|
|
async_dispatcher_send(hass, SIGNAL_FFMPEG_START, entity_ids)
|
|
|
|
elif service.service == SERVICE_STOP:
|
|
|
|
async_dispatcher_send(hass, SIGNAL_FFMPEG_STOP, entity_ids)
|
2017-01-31 15:48:03 +00:00
|
|
|
else:
|
2017-02-26 22:31:46 +00:00
|
|
|
async_dispatcher_send(hass, SIGNAL_FFMPEG_RESTART, entity_ids)
|
2017-02-08 22:18:23 +00:00
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_START, async_service_handle,
|
|
|
|
descriptions[DOMAIN].get(SERVICE_START), schema=SERVICE_FFMPEG_SCHEMA)
|
|
|
|
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_STOP, async_service_handle,
|
|
|
|
descriptions[DOMAIN].get(SERVICE_STOP), schema=SERVICE_FFMPEG_SCHEMA)
|
|
|
|
|
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_RESTART, async_service_handle,
|
|
|
|
descriptions[DOMAIN].get(SERVICE_RESTART),
|
|
|
|
schema=SERVICE_FFMPEG_SCHEMA)
|
|
|
|
|
|
|
|
hass.data[DATA_FFMPEG] = manager
|
2017-01-21 05:56:22 +00:00
|
|
|
return True
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
|
2017-01-21 05:56:22 +00:00
|
|
|
class FFmpegManager(object):
|
|
|
|
"""Helper for ha-ffmpeg."""
|
|
|
|
|
|
|
|
def __init__(self, hass, ffmpeg_bin, run_test):
|
|
|
|
"""Initialize helper."""
|
|
|
|
self.hass = hass
|
|
|
|
self._cache = {}
|
|
|
|
self._bin = ffmpeg_bin
|
|
|
|
self._run_test = run_test
|
|
|
|
|
|
|
|
@property
|
|
|
|
def binary(self):
|
|
|
|
"""Return ffmpeg binary from config."""
|
|
|
|
return self._bin
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_run_test(self, input_source):
|
|
|
|
"""Run test on this input. TRUE is deactivate or run correct.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
|
|
|
from haffmpeg import Test
|
|
|
|
|
|
|
|
if self._run_test:
|
|
|
|
# if in cache
|
|
|
|
if input_source in self._cache:
|
|
|
|
return self._cache[input_source]
|
|
|
|
|
|
|
|
# run test
|
|
|
|
ffmpeg_test = Test(self.binary, loop=self.hass.loop)
|
|
|
|
success = yield from ffmpeg_test.run_test(input_source)
|
|
|
|
if not success:
|
|
|
|
_LOGGER.error("FFmpeg '%s' test fails!", input_source)
|
|
|
|
self._cache[input_source] = False
|
|
|
|
return False
|
|
|
|
self._cache[input_source] = True
|
|
|
|
return True
|
2017-01-31 15:48:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class FFmpegBase(Entity):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Interface object for FFmpeg."""
|
2017-01-31 15:48:03 +00:00
|
|
|
|
|
|
|
def __init__(self, initial_state=True):
|
|
|
|
"""Initialize ffmpeg base object."""
|
|
|
|
self.ffmpeg = None
|
|
|
|
self.initial_state = initial_state
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_added_to_hass(self):
|
|
|
|
"""Register dispatcher & events.
|
|
|
|
|
|
|
|
This method is a coroutine.
|
|
|
|
"""
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIGNAL_FFMPEG_START, self._async_start_ffmpeg)
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIGNAL_FFMPEG_STOP, self._async_stop_ffmpeg)
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIGNAL_FFMPEG_RESTART, self._async_restart_ffmpeg)
|
|
|
|
|
|
|
|
# register start/stop
|
|
|
|
self._async_register_events()
|
|
|
|
|
2017-01-31 15:48:03 +00:00
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self.ffmpeg.is_running
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""Return True if entity has to be polled for state."""
|
|
|
|
return False
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def _async_start_ffmpeg(self, entity_ids):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Start a FFmpeg process.
|
2017-01-31 15:48:03 +00:00
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
This method is a coroutine.
|
2017-01-31 15:48:03 +00:00
|
|
|
"""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def _async_stop_ffmpeg(self, entity_ids):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Stop a FFmpeg process.
|
2017-01-31 15:48:03 +00:00
|
|
|
|
2017-02-26 22:31:46 +00:00
|
|
|
This method is a coroutine.
|
2017-01-31 15:48:03 +00:00
|
|
|
"""
|
2017-02-26 22:31:46 +00:00
|
|
|
if entity_ids is None or self.entity_id in entity_ids:
|
|
|
|
yield from self.ffmpeg.close()
|
2017-01-31 15:48:03 +00:00
|
|
|
|
|
|
|
@asyncio.coroutine
|
2017-02-26 22:31:46 +00:00
|
|
|
def _async_restart_ffmpeg(self, entity_ids):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Stop a FFmpeg process.
|
2017-02-26 22:31:46 +00:00
|
|
|
|
|
|
|
This method is a coroutine.
|
|
|
|
"""
|
|
|
|
if entity_ids is None or self.entity_id in entity_ids:
|
|
|
|
yield from self._async_stop_ffmpeg(None)
|
|
|
|
yield from self._async_start_ffmpeg(None)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_register_events(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Register a FFmpeg process/device."""
|
2017-02-26 22:31:46 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_shutdown_handle(event):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Stop FFmpeg process."""
|
2017-02-26 22:31:46 +00:00
|
|
|
yield from self._async_stop_ffmpeg(None)
|
|
|
|
|
|
|
|
self.hass.bus.async_listen_once(
|
|
|
|
EVENT_HOMEASSISTANT_STOP, async_shutdown_handle)
|
|
|
|
|
|
|
|
# start on startup
|
|
|
|
if not self.initial_state:
|
|
|
|
return
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_start_handle(event):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Start FFmpeg process."""
|
2017-02-26 22:31:46 +00:00
|
|
|
yield from self._async_start_ffmpeg(None)
|
|
|
|
self.hass.async_add_job(self.async_update_ha_state())
|
|
|
|
|
|
|
|
self.hass.bus.async_listen_once(
|
|
|
|
EVENT_HOMEASSISTANT_START, async_start_handle)
|