2017-03-08 15:46:41 +00:00
|
|
|
"""
|
|
|
|
Support for IP Webcam, an Android app that acts as a full-featured webcam.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/android_ip_webcam/
|
|
|
|
"""
|
|
|
|
import asyncio
|
2017-03-09 11:03:08 +00:00
|
|
|
import logging
|
2017-03-08 15:46:41 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_NAME, CONF_HOST, CONF_PORT, CONF_USERNAME, CONF_PASSWORD,
|
2017-03-09 11:00:50 +00:00
|
|
|
CONF_SENSORS, CONF_SWITCHES, CONF_TIMEOUT, CONF_SCAN_INTERVAL,
|
|
|
|
CONF_PLATFORM)
|
2017-03-08 15:46:41 +00:00
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
|
|
from homeassistant.helpers import discovery
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.helpers.dispatcher import (
|
|
|
|
async_dispatcher_send, async_dispatcher_connect)
|
|
|
|
from homeassistant.helpers.entity import Entity
|
|
|
|
from homeassistant.helpers.event import async_track_point_in_utc_time
|
|
|
|
from homeassistant.util.dt import utcnow
|
2017-03-09 11:00:50 +00:00
|
|
|
from homeassistant.components.camera.mjpeg import (
|
2017-03-09 14:10:39 +00:00
|
|
|
CONF_MJPEG_URL, CONF_STILL_IMAGE_URL)
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-04-05 04:40:19 +00:00
|
|
|
REQUIREMENTS = ['pydroid-ipcam==0.8']
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-03-09 11:03:08 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-03-24 20:42:00 +00:00
|
|
|
ATTR_AUD_CONNS = 'Audio Connections'
|
2017-03-09 11:00:50 +00:00
|
|
|
ATTR_HOST = 'host'
|
2017-03-08 15:46:41 +00:00
|
|
|
ATTR_VID_CONNS = 'Video Connections'
|
2017-03-24 20:42:00 +00:00
|
|
|
|
|
|
|
CONF_MOTION_SENSOR = 'motion_sensor'
|
|
|
|
|
|
|
|
DATA_IP_WEBCAM = 'android_ip_webcam'
|
|
|
|
DEFAULT_NAME = 'IP Webcam'
|
|
|
|
DEFAULT_PORT = 8080
|
|
|
|
DEFAULT_TIMEOUT = 10
|
|
|
|
DOMAIN = 'android_ip_webcam'
|
|
|
|
|
|
|
|
SCAN_INTERVAL = timedelta(seconds=10)
|
|
|
|
SIGNAL_UPDATE_DATA = 'android_ip_webcam_update'
|
2017-03-08 15:46:41 +00:00
|
|
|
|
|
|
|
KEY_MAP = {
|
|
|
|
'audio_connections': 'Audio Connections',
|
|
|
|
'adet_limit': 'Audio Trigger Limit',
|
|
|
|
'antibanding': 'Anti-banding',
|
|
|
|
'audio_only': 'Audio Only',
|
|
|
|
'battery_level': 'Battery Level',
|
|
|
|
'battery_temp': 'Battery Temperature',
|
|
|
|
'battery_voltage': 'Battery Voltage',
|
|
|
|
'coloreffect': 'Color Effect',
|
|
|
|
'exposure': 'Exposure Level',
|
|
|
|
'exposure_lock': 'Exposure Lock',
|
|
|
|
'ffc': 'Front-facing Camera',
|
|
|
|
'flashmode': 'Flash Mode',
|
|
|
|
'focus': 'Focus',
|
|
|
|
'focus_homing': 'Focus Homing',
|
|
|
|
'focus_region': 'Focus Region',
|
|
|
|
'focusmode': 'Focus Mode',
|
|
|
|
'gps_active': 'GPS Active',
|
|
|
|
'idle': 'Idle',
|
|
|
|
'ip_address': 'IPv4 Address',
|
|
|
|
'ipv6_address': 'IPv6 Address',
|
|
|
|
'ivideon_streaming': 'Ivideon Streaming',
|
|
|
|
'light': 'Light Level',
|
|
|
|
'mirror_flip': 'Mirror Flip',
|
|
|
|
'motion': 'Motion',
|
|
|
|
'motion_active': 'Motion Active',
|
|
|
|
'motion_detect': 'Motion Detection',
|
|
|
|
'motion_event': 'Motion Event',
|
|
|
|
'motion_limit': 'Motion Limit',
|
|
|
|
'night_vision': 'Night Vision',
|
|
|
|
'night_vision_average': 'Night Vision Average',
|
|
|
|
'night_vision_gain': 'Night Vision Gain',
|
|
|
|
'orientation': 'Orientation',
|
|
|
|
'overlay': 'Overlay',
|
|
|
|
'photo_size': 'Photo Size',
|
|
|
|
'pressure': 'Pressure',
|
|
|
|
'proximity': 'Proximity',
|
|
|
|
'quality': 'Quality',
|
|
|
|
'scenemode': 'Scene Mode',
|
|
|
|
'sound': 'Sound',
|
|
|
|
'sound_event': 'Sound Event',
|
|
|
|
'sound_timeout': 'Sound Timeout',
|
|
|
|
'torch': 'Torch',
|
|
|
|
'video_connections': 'Video Connections',
|
|
|
|
'video_chunk_len': 'Video Chunk Length',
|
|
|
|
'video_recording': 'Video Recording',
|
|
|
|
'video_size': 'Video Size',
|
|
|
|
'whitebalance': 'White Balance',
|
|
|
|
'whitebalance_lock': 'White Balance Lock',
|
|
|
|
'zoom': 'Zoom'
|
|
|
|
}
|
|
|
|
|
|
|
|
ICON_MAP = {
|
|
|
|
'audio_connections': 'mdi:speaker',
|
|
|
|
'battery_level': 'mdi:battery',
|
|
|
|
'battery_temp': 'mdi:thermometer',
|
|
|
|
'battery_voltage': 'mdi:battery-charging-100',
|
|
|
|
'exposure_lock': 'mdi:camera',
|
|
|
|
'ffc': 'mdi:camera-front-variant',
|
|
|
|
'focus': 'mdi:image-filter-center-focus',
|
|
|
|
'gps_active': 'mdi:crosshairs-gps',
|
|
|
|
'light': 'mdi:flashlight',
|
|
|
|
'motion': 'mdi:run',
|
|
|
|
'night_vision': 'mdi:weather-night',
|
|
|
|
'overlay': 'mdi:monitor',
|
|
|
|
'pressure': 'mdi:gauge',
|
|
|
|
'proximity': 'mdi:map-marker-radius',
|
|
|
|
'quality': 'mdi:quality-high',
|
|
|
|
'sound': 'mdi:speaker',
|
|
|
|
'sound_event': 'mdi:speaker',
|
|
|
|
'sound_timeout': 'mdi:speaker',
|
|
|
|
'torch': 'mdi:white-balance-sunny',
|
|
|
|
'video_chunk_len': 'mdi:video',
|
|
|
|
'video_connections': 'mdi:eye',
|
|
|
|
'video_recording': 'mdi:record-rec',
|
|
|
|
'whitebalance_lock': 'mdi:white-balance-auto'
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCHES = ['exposure_lock', 'ffc', 'focus', 'gps_active', 'night_vision',
|
|
|
|
'overlay', 'torch', 'whitebalance_lock', 'video_recording']
|
|
|
|
|
|
|
|
SENSORS = ['audio_connections', 'battery_level', 'battery_temp',
|
|
|
|
'battery_voltage', 'light', 'motion', 'pressure', 'proximity',
|
|
|
|
'sound', 'video_connections']
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.All(cv.ensure_list, [vol.Schema({
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
|
|
vol.Optional(CONF_TIMEOUT, default=DEFAULT_TIMEOUT): cv.positive_int,
|
|
|
|
vol.Optional(CONF_SCAN_INTERVAL, default=SCAN_INTERVAL):
|
|
|
|
cv.time_period,
|
|
|
|
vol.Inclusive(CONF_USERNAME, 'authentication'): cv.string,
|
|
|
|
vol.Inclusive(CONF_PASSWORD, 'authentication'): cv.string,
|
2018-02-17 09:29:14 +00:00
|
|
|
vol.Optional(CONF_SWITCHES):
|
2017-03-08 15:46:41 +00:00
|
|
|
vol.All(cv.ensure_list, [vol.In(SWITCHES)]),
|
2018-02-17 09:29:14 +00:00
|
|
|
vol.Optional(CONF_SENSORS):
|
2017-03-08 15:46:41 +00:00
|
|
|
vol.All(cv.ensure_list, [vol.In(SENSORS)]),
|
2018-02-17 09:29:14 +00:00
|
|
|
vol.Optional(CONF_MOTION_SENSOR): cv.boolean,
|
2017-03-08 15:46:41 +00:00
|
|
|
})])
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config):
|
2017-03-24 20:42:00 +00:00
|
|
|
"""Set up the IP Webcam component."""
|
2017-03-08 15:46:41 +00:00
|
|
|
from pydroid_ipcam import PyDroidIPCam
|
|
|
|
|
|
|
|
webcams = hass.data[DATA_IP_WEBCAM] = {}
|
|
|
|
websession = async_get_clientsession(hass)
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup_ipcamera(cam_config):
|
2017-03-24 20:42:00 +00:00
|
|
|
"""Set up an IP camera."""
|
2017-03-08 15:46:41 +00:00
|
|
|
host = cam_config[CONF_HOST]
|
|
|
|
username = cam_config.get(CONF_USERNAME)
|
|
|
|
password = cam_config.get(CONF_PASSWORD)
|
|
|
|
name = cam_config[CONF_NAME]
|
2017-03-09 00:00:57 +00:00
|
|
|
interval = cam_config[CONF_SCAN_INTERVAL]
|
2018-02-17 09:29:14 +00:00
|
|
|
switches = cam_config.get(CONF_SWITCHES)
|
|
|
|
sensors = cam_config.get(CONF_SENSORS)
|
|
|
|
motion = cam_config.get(CONF_MOTION_SENSOR)
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-03-24 20:42:00 +00:00
|
|
|
# Init ip webcam
|
2017-03-08 15:46:41 +00:00
|
|
|
cam = PyDroidIPCam(
|
|
|
|
hass.loop, websession, host, cam_config[CONF_PORT],
|
|
|
|
username=username, password=password,
|
|
|
|
timeout=cam_config[CONF_TIMEOUT]
|
|
|
|
)
|
|
|
|
|
2017-03-12 17:56:48 +00:00
|
|
|
if switches is None:
|
|
|
|
switches = [setting for setting in cam.enabled_settings
|
|
|
|
if setting in SWITCHES]
|
|
|
|
|
|
|
|
if sensors is None:
|
|
|
|
sensors = [sensor for sensor in cam.enabled_sensors
|
|
|
|
if sensor in SENSORS]
|
|
|
|
sensors.extend(['audio_connections', 'video_connections'])
|
|
|
|
|
|
|
|
if motion is None:
|
|
|
|
motion = 'motion_active' in cam.enabled_sensors
|
|
|
|
|
2017-03-08 15:46:41 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_update_data(now):
|
2017-03-24 20:42:00 +00:00
|
|
|
"""Update data from IP camera in SCAN_INTERVAL."""
|
2017-03-08 15:46:41 +00:00
|
|
|
yield from cam.update()
|
|
|
|
async_dispatcher_send(hass, SIGNAL_UPDATE_DATA, host)
|
|
|
|
|
|
|
|
async_track_point_in_utc_time(
|
2017-03-09 00:00:57 +00:00
|
|
|
hass, async_update_data, utcnow() + interval)
|
2017-03-08 15:46:41 +00:00
|
|
|
|
|
|
|
yield from async_update_data(None)
|
2017-03-09 11:00:50 +00:00
|
|
|
|
2017-03-24 20:42:00 +00:00
|
|
|
# Load platforms
|
2017-03-08 15:46:41 +00:00
|
|
|
webcams[host] = cam
|
|
|
|
|
|
|
|
mjpeg_camera = {
|
2017-03-09 14:10:39 +00:00
|
|
|
CONF_PLATFORM: 'mjpeg',
|
2017-03-09 11:00:50 +00:00
|
|
|
CONF_MJPEG_URL: cam.mjpeg_url,
|
|
|
|
CONF_STILL_IMAGE_URL: cam.image_url,
|
2017-03-08 15:46:41 +00:00
|
|
|
CONF_NAME: name,
|
|
|
|
}
|
|
|
|
if username and password:
|
|
|
|
mjpeg_camera.update({
|
|
|
|
CONF_USERNAME: username,
|
|
|
|
CONF_PASSWORD: password
|
|
|
|
})
|
|
|
|
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
|
|
hass, 'camera', 'mjpeg', mjpeg_camera, config))
|
|
|
|
|
2017-03-10 22:10:35 +00:00
|
|
|
if sensors:
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
|
|
hass, 'sensor', DOMAIN, {
|
|
|
|
CONF_NAME: name,
|
|
|
|
CONF_HOST: host,
|
|
|
|
CONF_SENSORS: sensors,
|
|
|
|
}, config))
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-03-10 22:10:35 +00:00
|
|
|
if switches:
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
|
|
hass, 'switch', DOMAIN, {
|
|
|
|
CONF_NAME: name,
|
|
|
|
CONF_HOST: host,
|
|
|
|
CONF_SWITCHES: switches,
|
|
|
|
}, config))
|
2017-03-08 15:46:41 +00:00
|
|
|
|
2017-03-09 11:00:50 +00:00
|
|
|
if motion:
|
|
|
|
hass.async_add_job(discovery.async_load_platform(
|
|
|
|
hass, 'binary_sensor', DOMAIN, {
|
|
|
|
CONF_HOST: host,
|
|
|
|
CONF_NAME: name,
|
|
|
|
}, config))
|
|
|
|
|
2017-03-08 15:46:41 +00:00
|
|
|
tasks = [async_setup_ipcamera(conf) for conf in config[DOMAIN]]
|
|
|
|
if tasks:
|
|
|
|
yield from asyncio.wait(tasks, loop=hass.loop)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class AndroidIPCamEntity(Entity):
|
|
|
|
"""The Android device running IP Webcam."""
|
|
|
|
|
|
|
|
def __init__(self, host, ipcam):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Initialize the data object."""
|
2017-03-08 15:46:41 +00:00
|
|
|
self._host = host
|
|
|
|
self._ipcam = ipcam
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_added_to_hass(self):
|
|
|
|
"""Register update dispatcher."""
|
|
|
|
@callback
|
|
|
|
def async_ipcam_update(host):
|
|
|
|
"""Update callback."""
|
|
|
|
if self._host != host:
|
|
|
|
return
|
2017-09-12 08:01:03 +00:00
|
|
|
self.async_schedule_update_ha_state(True)
|
2017-03-08 15:46:41 +00:00
|
|
|
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIGNAL_UPDATE_DATA, async_ipcam_update)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2017-03-09 00:00:57 +00:00
|
|
|
"""Return True if entity has to be polled for state."""
|
2017-03-08 15:46:41 +00:00
|
|
|
return False
|
|
|
|
|
2017-03-08 16:52:49 +00:00
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self._ipcam.available
|
|
|
|
|
2017-03-08 15:46:41 +00:00
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes."""
|
2017-03-09 11:03:08 +00:00
|
|
|
state_attr = {ATTR_HOST: self._host}
|
2017-03-08 16:57:22 +00:00
|
|
|
if self._ipcam.status_data is None:
|
2017-03-08 15:46:41 +00:00
|
|
|
return state_attr
|
|
|
|
|
|
|
|
state_attr[ATTR_VID_CONNS] = \
|
|
|
|
self._ipcam.status_data.get('video_connections')
|
|
|
|
state_attr[ATTR_AUD_CONNS] = \
|
|
|
|
self._ipcam.status_data.get('audio_connections')
|
|
|
|
|
|
|
|
return state_attr
|