mirror of
https://github.com/home-assistant/core.git
synced 2025-07-19 19:27:45 +00:00
add motion sensor / rewrite ffmpeg binary sensor (#2969)
🐬
* use const flags
This commit is contained in:
parent
d2bb61ad9e
commit
9ab2ac766e
@ -5,18 +5,27 @@ For more details about this platform, please refer to the documentation at
|
|||||||
https://home-assistant.io/components/binary_sensor.ffmpeg/
|
https://home-assistant.io/components/binary_sensor.ffmpeg/
|
||||||
"""
|
"""
|
||||||
import logging
|
import logging
|
||||||
|
from os import path
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
from homeassistant.components.binary_sensor import (BinarySensorDevice,
|
from homeassistant.components.binary_sensor import (BinarySensorDevice,
|
||||||
PLATFORM_SCHEMA)
|
PLATFORM_SCHEMA, DOMAIN)
|
||||||
from homeassistant.const import EVENT_HOMEASSISTANT_STOP, CONF_NAME
|
from homeassistant.config import load_yaml_config_file
|
||||||
|
from homeassistant.const import (EVENT_HOMEASSISTANT_STOP, CONF_NAME,
|
||||||
|
ATTR_ENTITY_ID)
|
||||||
|
|
||||||
REQUIREMENTS = ["ha-ffmpeg==0.7"]
|
REQUIREMENTS = ["ha-ffmpeg==0.8"]
|
||||||
|
|
||||||
|
SERVICE_RESTART = 'ffmpeg_restart'
|
||||||
|
|
||||||
|
FFMPEG_SENSOR_NOISE = 'noise'
|
||||||
|
FFMPEG_SENSOR_MOTION = 'motion'
|
||||||
|
|
||||||
MAP_FFMPEG_BIN = [
|
MAP_FFMPEG_BIN = [
|
||||||
'noise'
|
FFMPEG_SENSOR_NOISE,
|
||||||
|
FFMPEG_SENSOR_MOTION
|
||||||
]
|
]
|
||||||
|
|
||||||
CONF_TOOL = 'tool'
|
CONF_TOOL = 'tool'
|
||||||
@ -27,44 +36,140 @@ CONF_OUTPUT = 'output'
|
|||||||
CONF_PEAK = 'peak'
|
CONF_PEAK = 'peak'
|
||||||
CONF_DURATION = 'duration'
|
CONF_DURATION = 'duration'
|
||||||
CONF_RESET = 'reset'
|
CONF_RESET = 'reset'
|
||||||
|
CONF_CHANGES = 'changes'
|
||||||
|
CONF_REPEAT = 'repeat'
|
||||||
|
CONF_REPEAT_TIME = 'repeat_time'
|
||||||
|
|
||||||
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||||
vol.Required(CONF_TOOL): vol.In(MAP_FFMPEG_BIN),
|
vol.Required(CONF_TOOL): vol.In(MAP_FFMPEG_BIN),
|
||||||
vol.Required(CONF_INPUT): cv.string,
|
vol.Required(CONF_INPUT): cv.string,
|
||||||
vol.Optional(CONF_NAME, default="FFmpeg"): cv.string,
|
|
||||||
vol.Optional(CONF_FFMPEG_BIN, default="ffmpeg"): cv.string,
|
vol.Optional(CONF_FFMPEG_BIN, default="ffmpeg"): cv.string,
|
||||||
|
vol.Optional(CONF_NAME, default="FFmpeg"): cv.string,
|
||||||
vol.Optional(CONF_EXTRA_ARGUMENTS): cv.string,
|
vol.Optional(CONF_EXTRA_ARGUMENTS): cv.string,
|
||||||
vol.Optional(CONF_OUTPUT): cv.string,
|
vol.Optional(CONF_OUTPUT): cv.string,
|
||||||
vol.Optional(CONF_PEAK, default=-30): vol.Coerce(int),
|
vol.Optional(CONF_PEAK, default=-30): vol.Coerce(int),
|
||||||
vol.Optional(CONF_DURATION, default=1):
|
vol.Optional(CONF_DURATION, default=1):
|
||||||
vol.All(vol.Coerce(int), vol.Range(min=1)),
|
vol.All(vol.Coerce(int), vol.Range(min=1)),
|
||||||
vol.Optional(CONF_RESET, default=2):
|
vol.Optional(CONF_RESET, default=10):
|
||||||
vol.All(vol.Coerce(int), vol.Range(min=1)),
|
vol.All(vol.Coerce(int), vol.Range(min=1)),
|
||||||
|
vol.Optional(CONF_CHANGES, default=10):
|
||||||
|
vol.All(vol.Coerce(float), vol.Range(min=0, max=99)),
|
||||||
|
vol.Optional(CONF_REPEAT, default=0):
|
||||||
|
vol.All(vol.Coerce(int), vol.Range(min=0)),
|
||||||
|
vol.Optional(CONF_REPEAT_TIME, default=0):
|
||||||
|
vol.All(vol.Coerce(int), vol.Range(min=0)),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
SERVICE_RESTART_SCHEMA = vol.Schema({
|
||||||
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids,
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
# list of all ffmpeg sensors
|
||||||
|
DEVICES = []
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def setup_platform(hass, config, add_entities, discovery_info=None):
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||||
"""Create the binary sensor."""
|
"""Create the binary sensor."""
|
||||||
if config.get(CONF_TOOL) == "noise":
|
from haffmpeg import SensorNoise, SensorMotion
|
||||||
entity = FFmpegNoise(config)
|
|
||||||
|
if config.get(CONF_TOOL) == FFMPEG_SENSOR_NOISE:
|
||||||
|
entity = FFmpegNoise(SensorNoise, config)
|
||||||
|
else:
|
||||||
|
entity = FFmpegMotion(SensorMotion, config)
|
||||||
|
|
||||||
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, entity.shutdown_ffmpeg)
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, entity.shutdown_ffmpeg)
|
||||||
|
|
||||||
|
# add to system
|
||||||
add_entities([entity])
|
add_entities([entity])
|
||||||
|
DEVICES.append(entity)
|
||||||
|
|
||||||
|
# exists service?
|
||||||
|
if hass.services.has_service(DOMAIN, SERVICE_RESTART):
|
||||||
|
return True
|
||||||
|
|
||||||
|
descriptions = load_yaml_config_file(
|
||||||
|
path.join(path.dirname(__file__), 'services.yaml'))
|
||||||
|
|
||||||
|
# register service
|
||||||
|
def _service_handle_restart(service):
|
||||||
|
"""Handle service binary_sensor.ffmpeg_restart."""
|
||||||
|
entity_ids = service.data.get('entity_id')
|
||||||
|
|
||||||
|
if entity_ids:
|
||||||
|
_devices = [device for device in DEVICES
|
||||||
|
if device.entity_id in entity_ids]
|
||||||
|
else:
|
||||||
|
_devices = DEVICES
|
||||||
|
|
||||||
|
for device in _devices:
|
||||||
|
device.reset_ffmpeg()
|
||||||
|
|
||||||
|
hass.services.register(DOMAIN, SERVICE_RESTART,
|
||||||
|
_service_handle_restart,
|
||||||
|
descriptions.get(SERVICE_RESTART),
|
||||||
|
schema=SERVICE_RESTART_SCHEMA)
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
class FFmpegNoise(BinarySensorDevice):
|
class FFmpegBinarySensor(BinarySensorDevice):
|
||||||
"""A binary sensor which use ffmpeg for noise detection."""
|
"""A binary sensor which use ffmpeg for noise detection."""
|
||||||
|
|
||||||
def __init__(self, config):
|
def __init__(self, ffobj, config):
|
||||||
"""Constructor for binary sensor noise detection."""
|
"""Constructor for binary sensor noise detection."""
|
||||||
from haffmpeg import SensorNoise
|
|
||||||
|
|
||||||
self._state = False
|
self._state = False
|
||||||
|
self._config = config
|
||||||
self._name = config.get(CONF_NAME)
|
self._name = config.get(CONF_NAME)
|
||||||
self._ffmpeg = SensorNoise(config.get(CONF_FFMPEG_BIN), self._callback)
|
self._ffmpeg = ffobj(config.get(CONF_FFMPEG_BIN), self._callback)
|
||||||
|
|
||||||
|
self._start_ffmpeg(config)
|
||||||
|
|
||||||
|
def _callback(self, state):
|
||||||
|
"""HA-FFmpeg callback for noise detection."""
|
||||||
|
self._state = state
|
||||||
|
self.update_ha_state()
|
||||||
|
|
||||||
|
def _start_ffmpeg(self, config):
|
||||||
|
"""Start a FFmpeg instance."""
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def shutdown_ffmpeg(self, event):
|
||||||
|
"""For STOP event to shutdown ffmpeg."""
|
||||||
|
self._ffmpeg.close()
|
||||||
|
|
||||||
|
def reset_ffmpeg(self):
|
||||||
|
"""Restart ffmpeg with new config."""
|
||||||
|
self._ffmpeg.close()
|
||||||
|
self._start_ffmpeg(self._config)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_on(self):
|
||||||
|
"""True if the binary sensor is on."""
|
||||||
|
return self._state
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""Return True if entity has to be polled for state."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the entity."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self):
|
||||||
|
"""Return True if entity is available."""
|
||||||
|
return self._ffmpeg.is_running
|
||||||
|
|
||||||
|
|
||||||
|
class FFmpegNoise(FFmpegBinarySensor):
|
||||||
|
"""A binary sensor which use ffmpeg for noise detection."""
|
||||||
|
|
||||||
|
def _start_ffmpeg(self, config):
|
||||||
|
"""Start a FFmpeg instance."""
|
||||||
# init config
|
# init config
|
||||||
self._ffmpeg.set_options(
|
self._ffmpeg.set_options(
|
||||||
time_duration=config.get(CONF_DURATION),
|
time_duration=config.get(CONF_DURATION),
|
||||||
@ -79,31 +184,32 @@ class FFmpegNoise(BinarySensorDevice):
|
|||||||
extra_cmd=config.get(CONF_EXTRA_ARGUMENTS),
|
extra_cmd=config.get(CONF_EXTRA_ARGUMENTS),
|
||||||
)
|
)
|
||||||
|
|
||||||
def _callback(self, state):
|
|
||||||
"""HA-FFmpeg callback for noise detection."""
|
|
||||||
self._state = state
|
|
||||||
self.update_ha_state()
|
|
||||||
|
|
||||||
def shutdown_ffmpeg(self, event):
|
|
||||||
"""For STOP event to shutdown ffmpeg."""
|
|
||||||
self._ffmpeg.close()
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_on(self):
|
|
||||||
"""True if the binary sensor is on."""
|
|
||||||
return self._state
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def sensor_class(self):
|
def sensor_class(self):
|
||||||
"""Return the class of this sensor, from SENSOR_CLASSES."""
|
"""Return the class of this sensor, from SENSOR_CLASSES."""
|
||||||
return "sound"
|
return "sound"
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
class FFmpegMotion(FFmpegBinarySensor):
|
||||||
"""Return True if entity has to be polled for state."""
|
"""A binary sensor which use ffmpeg for noise detection."""
|
||||||
return False
|
|
||||||
|
def _start_ffmpeg(self, config):
|
||||||
|
"""Start a FFmpeg instance."""
|
||||||
|
# init config
|
||||||
|
self._ffmpeg.set_options(
|
||||||
|
time_reset=config.get(CONF_RESET),
|
||||||
|
time_repeat=config.get(CONF_REPEAT_TIME),
|
||||||
|
repeat=config.get(CONF_REPEAT),
|
||||||
|
changes=config.get(CONF_CHANGES),
|
||||||
|
)
|
||||||
|
|
||||||
|
# run
|
||||||
|
self._ffmpeg.open_sensor(
|
||||||
|
input_source=config.get(CONF_INPUT),
|
||||||
|
extra_cmd=config.get(CONF_EXTRA_ARGUMENTS),
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def name(self):
|
def sensor_class(self):
|
||||||
"""Return the name of the entity."""
|
"""Return the class of this sensor, from SENSOR_CLASSES."""
|
||||||
return self._name
|
return "motion"
|
||||||
|
9
homeassistant/components/binary_sensor/services.yaml
Normal file
9
homeassistant/components/binary_sensor/services.yaml
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
# Describes the format for available binary_sensor services
|
||||||
|
|
||||||
|
ffmpeg_restart:
|
||||||
|
description: Send a restart command to a ffmpeg based sensor (party mode).
|
||||||
|
|
||||||
|
fields:
|
||||||
|
entity_id:
|
||||||
|
description: Name(s) of entites that will restart. Platform dependent.
|
||||||
|
example: 'binary_sensor.ffmpeg_noise'
|
@ -14,7 +14,7 @@ from homeassistant.components.camera.mjpeg import extract_image_from_mjpeg
|
|||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
from homeassistant.const import CONF_NAME, CONF_PLATFORM
|
from homeassistant.const import CONF_NAME, CONF_PLATFORM
|
||||||
|
|
||||||
REQUIREMENTS = ["ha-ffmpeg==0.7"]
|
REQUIREMENTS = ["ha-ffmpeg==0.8"]
|
||||||
|
|
||||||
CONF_INPUT = 'input'
|
CONF_INPUT = 'input'
|
||||||
CONF_FFMPEG_BIN = 'ffmpeg_bin'
|
CONF_FFMPEG_BIN = 'ffmpeg_bin'
|
||||||
|
@ -104,7 +104,7 @@ gps3==0.33.2
|
|||||||
|
|
||||||
# homeassistant.components.binary_sensor.ffmpeg
|
# homeassistant.components.binary_sensor.ffmpeg
|
||||||
# homeassistant.components.camera.ffmpeg
|
# homeassistant.components.camera.ffmpeg
|
||||||
ha-ffmpeg==0.7
|
ha-ffmpeg==0.8
|
||||||
|
|
||||||
# homeassistant.components.mqtt.server
|
# homeassistant.components.mqtt.server
|
||||||
hbmqtt==0.7.1
|
hbmqtt==0.7.1
|
||||||
|
Loading…
x
Reference in New Issue
Block a user