2017-05-12 15:51:54 +00:00
|
|
|
"""
|
|
|
|
Support for Axis devices.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/axis/
|
|
|
|
"""
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2017-10-25 07:04:30 +00:00
|
|
|
from homeassistant.components.discovery import SERVICE_AXIS
|
2017-06-24 07:14:57 +00:00
|
|
|
from homeassistant.config import load_yaml_config_file
|
2017-05-12 15:51:54 +00:00
|
|
|
from homeassistant.const import (ATTR_LOCATION, ATTR_TRIPPED,
|
2017-10-25 07:04:30 +00:00
|
|
|
CONF_EVENT, CONF_HOST, CONF_INCLUDE,
|
|
|
|
CONF_NAME, CONF_PASSWORD, CONF_PORT,
|
|
|
|
CONF_TRIGGER_TIME, CONF_USERNAME,
|
|
|
|
EVENT_HOMEASSISTANT_STOP)
|
2017-05-12 15:51:54 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
|
|
|
from homeassistant.helpers import discovery
|
2017-10-25 07:04:30 +00:00
|
|
|
from homeassistant.helpers.dispatcher import dispatcher_send
|
2017-05-12 15:51:54 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2017-11-20 03:47:55 +00:00
|
|
|
from homeassistant.util.json import load_json, save_json
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
|
2017-10-25 07:04:30 +00:00
|
|
|
REQUIREMENTS = ['axis==14']
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DOMAIN = 'axis'
|
|
|
|
CONFIG_FILE = 'axis.conf'
|
|
|
|
|
|
|
|
AXIS_DEVICES = {}
|
|
|
|
|
|
|
|
EVENT_TYPES = ['motion', 'vmd3', 'pir', 'sound',
|
|
|
|
'daynight', 'tampering', 'input']
|
|
|
|
|
|
|
|
PLATFORMS = ['camera']
|
|
|
|
|
|
|
|
AXIS_INCLUDE = EVENT_TYPES + PLATFORMS
|
|
|
|
|
|
|
|
AXIS_DEFAULT_HOST = '192.168.0.90'
|
|
|
|
AXIS_DEFAULT_USERNAME = 'root'
|
|
|
|
AXIS_DEFAULT_PASSWORD = 'pass'
|
|
|
|
|
|
|
|
DEVICE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_INCLUDE):
|
|
|
|
vol.All(cv.ensure_list, [vol.In(AXIS_INCLUDE)]),
|
|
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_HOST, default=AXIS_DEFAULT_HOST): cv.string,
|
|
|
|
vol.Optional(CONF_USERNAME, default=AXIS_DEFAULT_USERNAME): cv.string,
|
|
|
|
vol.Optional(CONF_PASSWORD, default=AXIS_DEFAULT_PASSWORD): cv.string,
|
|
|
|
vol.Optional(CONF_TRIGGER_TIME, default=0): cv.positive_int,
|
2017-09-19 08:09:47 +00:00
|
|
|
vol.Optional(CONF_PORT, default=80): cv.positive_int,
|
2017-05-12 15:51:54 +00:00
|
|
|
vol.Optional(ATTR_LOCATION, default=''): cv.string,
|
|
|
|
})
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
cv.slug: DEVICE_SCHEMA,
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
2017-06-24 07:14:57 +00:00
|
|
|
SERVICE_VAPIX_CALL = 'vapix_call'
|
|
|
|
SERVICE_VAPIX_CALL_RESPONSE = 'vapix_call_response'
|
|
|
|
SERVICE_CGI = 'cgi'
|
|
|
|
SERVICE_ACTION = 'action'
|
|
|
|
SERVICE_PARAM = 'param'
|
|
|
|
SERVICE_DEFAULT_CGI = 'param.cgi'
|
|
|
|
SERVICE_DEFAULT_ACTION = 'update'
|
|
|
|
|
|
|
|
SERVICE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(SERVICE_PARAM): cv.string,
|
|
|
|
vol.Optional(SERVICE_CGI, default=SERVICE_DEFAULT_CGI): cv.string,
|
|
|
|
vol.Optional(SERVICE_ACTION, default=SERVICE_DEFAULT_ACTION): cv.string,
|
|
|
|
})
|
|
|
|
|
2017-05-12 15:51:54 +00:00
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
def request_configuration(hass, config, name, host, serialnumber):
|
2017-05-12 15:51:54 +00:00
|
|
|
"""Request configuration steps from the user."""
|
2017-07-16 19:39:38 +00:00
|
|
|
configurator = hass.components.configurator
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
def configuration_callback(callback_data):
|
|
|
|
"""Called when config is submitted."""
|
|
|
|
if CONF_INCLUDE not in callback_data:
|
|
|
|
configurator.notify_errors(request_id,
|
|
|
|
"Functionality mandatory.")
|
|
|
|
return False
|
2017-10-25 07:04:30 +00:00
|
|
|
|
2017-05-12 15:51:54 +00:00
|
|
|
callback_data[CONF_INCLUDE] = callback_data[CONF_INCLUDE].split()
|
|
|
|
callback_data[CONF_HOST] = host
|
2017-10-25 07:04:30 +00:00
|
|
|
|
2017-05-12 15:51:54 +00:00
|
|
|
if CONF_NAME not in callback_data:
|
|
|
|
callback_data[CONF_NAME] = name
|
2017-10-25 07:04:30 +00:00
|
|
|
|
2017-05-12 15:51:54 +00:00
|
|
|
try:
|
2017-09-19 08:09:47 +00:00
|
|
|
device_config = DEVICE_SCHEMA(callback_data)
|
2017-05-12 15:51:54 +00:00
|
|
|
except vol.Invalid:
|
|
|
|
configurator.notify_errors(request_id,
|
|
|
|
"Bad input, please check spelling.")
|
|
|
|
return False
|
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
if setup_device(hass, config, device_config):
|
2017-11-20 03:47:55 +00:00
|
|
|
config_file = load_json(hass.config.path(CONFIG_FILE))
|
2017-09-19 08:09:47 +00:00
|
|
|
config_file[serialnumber] = dict(device_config)
|
2017-11-20 03:47:55 +00:00
|
|
|
save_json(hass.config.path(CONFIG_FILE), config_file)
|
2017-05-12 15:51:54 +00:00
|
|
|
configurator.request_done(request_id)
|
|
|
|
else:
|
|
|
|
configurator.notify_errors(request_id,
|
|
|
|
"Failed to register, please try again.")
|
|
|
|
return False
|
|
|
|
|
|
|
|
title = '{} ({})'.format(name, host)
|
|
|
|
request_id = configurator.request_config(
|
2017-08-14 05:37:50 +00:00
|
|
|
title, configuration_callback,
|
2017-05-12 15:51:54 +00:00
|
|
|
description='Functionality: ' + str(AXIS_INCLUDE),
|
|
|
|
entity_picture="/static/images/logo_axis.png",
|
|
|
|
link_name='Axis platform documentation',
|
|
|
|
link_url='https://home-assistant.io/components/axis/',
|
|
|
|
submit_caption="Confirm",
|
|
|
|
fields=[
|
|
|
|
{'id': CONF_NAME,
|
|
|
|
'name': "Device name",
|
|
|
|
'type': 'text'},
|
|
|
|
{'id': CONF_USERNAME,
|
|
|
|
'name': "User name",
|
|
|
|
'type': 'text'},
|
|
|
|
{'id': CONF_PASSWORD,
|
|
|
|
'name': 'Password',
|
|
|
|
'type': 'password'},
|
|
|
|
{'id': CONF_INCLUDE,
|
|
|
|
'name': "Device functionality (space separated list)",
|
|
|
|
'type': 'text'},
|
|
|
|
{'id': ATTR_LOCATION,
|
|
|
|
'name': "Physical location of device (optional)",
|
|
|
|
'type': 'text'},
|
2017-09-19 08:09:47 +00:00
|
|
|
{'id': CONF_PORT,
|
|
|
|
'name': "HTTP port (default=80)",
|
|
|
|
'type': 'number'},
|
2017-05-12 15:51:54 +00:00
|
|
|
{'id': CONF_TRIGGER_TIME,
|
|
|
|
'name': "Sensor update interval (optional)",
|
|
|
|
'type': 'number'},
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
def setup(hass, config):
|
2017-05-12 15:51:54 +00:00
|
|
|
"""Common setup for Axis devices."""
|
|
|
|
def _shutdown(call): # pylint: disable=unused-argument
|
2017-10-25 07:04:30 +00:00
|
|
|
"""Stop the event stream on shutdown."""
|
2017-05-12 15:51:54 +00:00
|
|
|
for serialnumber, device in AXIS_DEVICES.items():
|
2017-10-25 07:04:30 +00:00
|
|
|
_LOGGER.info("Stopping event stream for %s.", serialnumber)
|
|
|
|
device.stop()
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, _shutdown)
|
|
|
|
|
|
|
|
def axis_device_discovered(service, discovery_info):
|
|
|
|
"""Called when axis devices has been found."""
|
2017-06-24 07:14:57 +00:00
|
|
|
host = discovery_info[CONF_HOST]
|
2017-05-12 15:51:54 +00:00
|
|
|
name = discovery_info['hostname']
|
|
|
|
serialnumber = discovery_info['properties']['macaddress']
|
|
|
|
|
|
|
|
if serialnumber not in AXIS_DEVICES:
|
2017-11-20 03:47:55 +00:00
|
|
|
config_file = load_json(hass.config.path(CONFIG_FILE))
|
2017-05-12 15:51:54 +00:00
|
|
|
if serialnumber in config_file:
|
2017-10-25 07:04:30 +00:00
|
|
|
# Device config previously saved to file
|
2017-05-12 15:51:54 +00:00
|
|
|
try:
|
2017-09-19 08:09:47 +00:00
|
|
|
device_config = DEVICE_SCHEMA(config_file[serialnumber])
|
|
|
|
device_config[CONF_HOST] = host
|
2017-05-12 15:51:54 +00:00
|
|
|
except vol.Invalid as err:
|
|
|
|
_LOGGER.error("Bad data from %s. %s", CONFIG_FILE, err)
|
|
|
|
return False
|
2017-09-19 08:09:47 +00:00
|
|
|
if not setup_device(hass, config, device_config):
|
|
|
|
_LOGGER.error("Couldn\'t set up %s",
|
|
|
|
device_config[CONF_NAME])
|
2017-05-12 15:51:54 +00:00
|
|
|
else:
|
2017-06-24 07:14:57 +00:00
|
|
|
# New device, create configuration request for UI
|
2017-09-19 08:09:47 +00:00
|
|
|
request_configuration(hass, config, name, host, serialnumber)
|
2017-06-24 07:14:57 +00:00
|
|
|
else:
|
|
|
|
# Device already registered, but on a different IP
|
|
|
|
device = AXIS_DEVICES[serialnumber]
|
2017-10-25 07:04:30 +00:00
|
|
|
device.config.host = host
|
|
|
|
dispatcher_send(hass, DOMAIN + '_' + device.name + '_new_ip', host)
|
2017-06-24 07:14:57 +00:00
|
|
|
|
|
|
|
# Register discovery service
|
2017-05-12 15:51:54 +00:00
|
|
|
discovery.listen(hass, SERVICE_AXIS, axis_device_discovered)
|
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
if DOMAIN in config:
|
|
|
|
for device in config[DOMAIN]:
|
|
|
|
device_config = config[DOMAIN][device]
|
|
|
|
if CONF_NAME not in device_config:
|
|
|
|
device_config[CONF_NAME] = device
|
|
|
|
if not setup_device(hass, config, device_config):
|
|
|
|
_LOGGER.error("Couldn\'t set up %s", device_config[CONF_NAME])
|
2017-06-24 07:14:57 +00:00
|
|
|
|
|
|
|
# Services to communicate with device.
|
|
|
|
descriptions = load_yaml_config_file(
|
|
|
|
os.path.join(os.path.dirname(__file__), 'services.yaml'))
|
|
|
|
|
|
|
|
def vapix_service(call):
|
|
|
|
"""Service to send a message."""
|
|
|
|
for _, device in AXIS_DEVICES.items():
|
|
|
|
if device.name == call.data[CONF_NAME]:
|
2017-10-25 07:04:30 +00:00
|
|
|
response = device.vapix.do_request(
|
|
|
|
call.data[SERVICE_CGI],
|
|
|
|
call.data[SERVICE_ACTION],
|
|
|
|
call.data[SERVICE_PARAM])
|
|
|
|
hass.bus.fire(SERVICE_VAPIX_CALL_RESPONSE, response)
|
2017-06-24 07:14:57 +00:00
|
|
|
return True
|
|
|
|
_LOGGER.info("Couldn\'t find device %s", call.data[CONF_NAME])
|
|
|
|
return False
|
|
|
|
|
|
|
|
# Register service with Home Assistant.
|
|
|
|
hass.services.register(DOMAIN,
|
|
|
|
SERVICE_VAPIX_CALL,
|
|
|
|
vapix_service,
|
|
|
|
descriptions[DOMAIN][SERVICE_VAPIX_CALL],
|
|
|
|
schema=SERVICE_SCHEMA)
|
2017-05-12 15:51:54 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
def setup_device(hass, config, device_config):
|
2017-05-12 15:51:54 +00:00
|
|
|
"""Set up device."""
|
|
|
|
from axis import AxisDevice
|
|
|
|
|
2017-10-25 07:04:30 +00:00
|
|
|
def signal_callback(action, event):
|
|
|
|
"""Callback to configure events when initialized on event stream."""
|
|
|
|
if action == 'add':
|
|
|
|
event_config = {
|
|
|
|
CONF_EVENT: event,
|
|
|
|
CONF_NAME: device_config[CONF_NAME],
|
|
|
|
ATTR_LOCATION: device_config[ATTR_LOCATION],
|
|
|
|
CONF_TRIGGER_TIME: device_config[CONF_TRIGGER_TIME]
|
|
|
|
}
|
|
|
|
component = event.event_platform
|
|
|
|
discovery.load_platform(hass,
|
|
|
|
component,
|
|
|
|
DOMAIN,
|
|
|
|
event_config,
|
|
|
|
config)
|
|
|
|
|
|
|
|
event_types = list(filter(lambda x: x in device_config[CONF_INCLUDE],
|
|
|
|
EVENT_TYPES))
|
|
|
|
device_config['events'] = event_types
|
|
|
|
device_config['signal'] = signal_callback
|
|
|
|
device = AxisDevice(hass.loop, **device_config)
|
|
|
|
device.name = device_config[CONF_NAME]
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
if device.serial_number is None:
|
|
|
|
# If there is no serial number a connection could not be made
|
2017-09-19 08:09:47 +00:00
|
|
|
_LOGGER.error("Couldn\'t connect to %s", device_config[CONF_HOST])
|
2017-05-12 15:51:54 +00:00
|
|
|
return False
|
|
|
|
|
2017-09-19 08:09:47 +00:00
|
|
|
for component in device_config[CONF_INCLUDE]:
|
2017-10-25 07:04:30 +00:00
|
|
|
if component == 'camera':
|
2017-09-19 08:09:47 +00:00
|
|
|
camera_config = {
|
|
|
|
CONF_NAME: device_config[CONF_NAME],
|
2017-10-25 07:04:30 +00:00
|
|
|
CONF_HOST: device_config[CONF_HOST],
|
2017-09-19 08:09:47 +00:00
|
|
|
CONF_PORT: device_config[CONF_PORT],
|
|
|
|
CONF_USERNAME: device_config[CONF_USERNAME],
|
|
|
|
CONF_PASSWORD: device_config[CONF_PASSWORD]
|
|
|
|
}
|
|
|
|
discovery.load_platform(hass,
|
|
|
|
component,
|
|
|
|
DOMAIN,
|
|
|
|
camera_config,
|
|
|
|
config)
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
AXIS_DEVICES[device.serial_number] = device
|
2017-11-11 20:30:18 +00:00
|
|
|
if event_types:
|
|
|
|
hass.add_job(device.start)
|
2017-05-12 15:51:54 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class AxisDeviceEvent(Entity):
|
|
|
|
"""Representation of a Axis device event."""
|
|
|
|
|
2017-10-25 07:04:30 +00:00
|
|
|
def __init__(self, event_config):
|
2017-05-12 15:51:54 +00:00
|
|
|
"""Initialize the event."""
|
2017-10-25 07:04:30 +00:00
|
|
|
self.axis_event = event_config[CONF_EVENT]
|
|
|
|
self._name = '{}_{}_{}'.format(event_config[CONF_NAME],
|
|
|
|
self.axis_event.event_type,
|
2017-05-12 15:51:54 +00:00
|
|
|
self.axis_event.id)
|
2017-10-25 07:04:30 +00:00
|
|
|
self.location = event_config[ATTR_LOCATION]
|
2017-05-12 15:51:54 +00:00
|
|
|
self.axis_event.callback = self._update_callback
|
|
|
|
|
|
|
|
def _update_callback(self):
|
|
|
|
"""Update the sensor's state, if needed."""
|
|
|
|
self.update()
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the event."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_class(self):
|
|
|
|
"""Return the class of the event."""
|
2017-10-25 07:04:30 +00:00
|
|
|
return self.axis_event.event_class
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""No polling needed."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the state attributes of the event."""
|
|
|
|
attr = {}
|
|
|
|
|
|
|
|
tripped = self.axis_event.is_tripped
|
|
|
|
attr[ATTR_TRIPPED] = 'True' if tripped else 'False'
|
|
|
|
|
2017-10-25 07:04:30 +00:00
|
|
|
attr[ATTR_LOCATION] = self.location
|
2017-05-12 15:51:54 +00:00
|
|
|
|
|
|
|
return attr
|