2016-10-15 03:56:40 +00:00
|
|
|
"""
|
2016-10-29 20:10:42 +00:00
|
|
|
Support for ZoneMinder.
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/zoneminder/
|
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
from urllib.parse import urljoin
|
|
|
|
|
|
|
|
import requests
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2016-10-27 13:54:03 +00:00
|
|
|
CONF_PATH, CONF_HOST, CONF_SSL, CONF_PASSWORD, CONF_USERNAME)
|
2016-10-29 20:10:42 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-02-21 06:17:11 +00:00
|
|
|
CONF_PATH_ZMS = 'path_zms'
|
2016-10-29 20:10:42 +00:00
|
|
|
DEFAULT_PATH = '/zm/'
|
2017-02-21 06:17:11 +00:00
|
|
|
DEFAULT_PATH_ZMS = '/zm/cgi-bin/nph-zms'
|
2016-10-29 20:10:42 +00:00
|
|
|
DEFAULT_SSL = False
|
|
|
|
DEFAULT_TIMEOUT = 10
|
2016-10-15 03:56:40 +00:00
|
|
|
DOMAIN = 'zoneminder'
|
|
|
|
|
2016-10-29 20:10:42 +00:00
|
|
|
LOGIN_RETRIES = 2
|
|
|
|
|
|
|
|
ZM = {}
|
|
|
|
|
2016-10-15 03:56:40 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
2016-10-29 20:10:42 +00:00
|
|
|
vol.Optional(CONF_SSL, default=DEFAULT_SSL): cv.boolean,
|
|
|
|
vol.Optional(CONF_PATH, default=DEFAULT_PATH): cv.string,
|
2017-02-21 06:17:11 +00:00
|
|
|
# This should match PATH_ZMS in ZoneMinder settings.
|
|
|
|
vol.Optional(CONF_PATH_ZMS, default=DEFAULT_PATH_ZMS): cv.string,
|
2016-10-15 03:56:40 +00:00
|
|
|
vol.Optional(CONF_USERNAME): cv.string,
|
|
|
|
vol.Optional(CONF_PASSWORD): cv.string
|
|
|
|
})
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
2016-10-29 20:10:42 +00:00
|
|
|
"""Set up the ZoneMinder component."""
|
2016-10-15 03:56:40 +00:00
|
|
|
global ZM
|
|
|
|
ZM = {}
|
|
|
|
|
|
|
|
conf = config[DOMAIN]
|
2016-10-27 13:54:03 +00:00
|
|
|
if conf[CONF_SSL]:
|
2016-10-29 20:10:42 +00:00
|
|
|
schema = 'https'
|
2016-10-27 13:54:03 +00:00
|
|
|
else:
|
2016-10-29 20:10:42 +00:00
|
|
|
schema = 'http'
|
2016-10-27 13:54:03 +00:00
|
|
|
|
2017-02-21 06:17:11 +00:00
|
|
|
server_origin = '{}://{}'.format(schema, conf[CONF_HOST])
|
|
|
|
url = urljoin(server_origin, conf[CONF_PATH])
|
2016-10-15 03:56:40 +00:00
|
|
|
username = conf.get(CONF_USERNAME, None)
|
|
|
|
password = conf.get(CONF_PASSWORD, None)
|
|
|
|
|
2017-02-21 06:17:11 +00:00
|
|
|
ZM['server_origin'] = server_origin
|
2016-10-15 03:56:40 +00:00
|
|
|
ZM['url'] = url
|
|
|
|
ZM['username'] = username
|
|
|
|
ZM['password'] = password
|
2017-02-21 06:17:11 +00:00
|
|
|
ZM['path_zms'] = conf.get(CONF_PATH_ZMS)
|
|
|
|
|
|
|
|
hass.data[DOMAIN] = ZM
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
return login()
|
|
|
|
|
|
|
|
|
|
|
|
def login():
|
2016-10-29 20:10:42 +00:00
|
|
|
"""Login to the ZoneMinder API."""
|
|
|
|
_LOGGER.debug("Attempting to login to ZoneMinder")
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
login_post = {'view': 'console', 'action': 'login'}
|
|
|
|
if ZM['username']:
|
|
|
|
login_post['username'] = ZM['username']
|
|
|
|
if ZM['password']:
|
|
|
|
login_post['password'] = ZM['password']
|
|
|
|
|
|
|
|
req = requests.post(ZM['url'] + '/index.php', data=login_post)
|
|
|
|
ZM['cookies'] = req.cookies
|
|
|
|
|
2016-10-29 20:10:42 +00:00
|
|
|
# Login calls returns a 200 response on both failure and success.
|
2016-10-15 03:56:40 +00:00
|
|
|
# The only way to tell if you logged in correctly is to issue an api call.
|
|
|
|
req = requests.get(
|
2016-10-29 20:10:42 +00:00
|
|
|
ZM['url'] + 'api/host/getVersion.json', cookies=ZM['cookies'],
|
|
|
|
timeout=DEFAULT_TIMEOUT)
|
2016-10-15 03:56:40 +00:00
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
if not req.ok:
|
2016-10-15 03:56:40 +00:00
|
|
|
_LOGGER.error("Connection error logging into ZoneMinder")
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
def _zm_request(method, api_url, data=None):
|
|
|
|
"""Perform a Zoneminder request."""
|
2016-10-29 20:10:42 +00:00
|
|
|
# Since the API uses sessions that expire, sometimes we need to re-auth
|
|
|
|
# if the call fails.
|
2016-10-15 03:56:40 +00:00
|
|
|
for _ in range(LOGIN_RETRIES):
|
2017-02-10 01:57:19 +00:00
|
|
|
req = requests.request(
|
|
|
|
method, urljoin(ZM['url'], api_url), data=data,
|
|
|
|
cookies=ZM['cookies'], timeout=DEFAULT_TIMEOUT)
|
2016-10-15 03:56:40 +00:00
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
if not req.ok:
|
2016-10-15 03:56:40 +00:00
|
|
|
login()
|
|
|
|
else:
|
|
|
|
break
|
2017-02-10 01:57:19 +00:00
|
|
|
|
2016-10-15 03:56:40 +00:00
|
|
|
else:
|
2017-05-21 09:11:33 +00:00
|
|
|
_LOGGER.error("Unable to get API response from ZoneMinder")
|
2016-10-15 03:56:40 +00:00
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
try:
|
|
|
|
return req.json()
|
|
|
|
except ValueError:
|
|
|
|
_LOGGER.exception('JSON decode exception caught while attempting to '
|
|
|
|
'decode "%s"', req.text)
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
def get_state(api_url):
|
|
|
|
"""Get a state from the ZoneMinder API service."""
|
|
|
|
return _zm_request('get', api_url)
|
2016-10-15 03:56:40 +00:00
|
|
|
|
|
|
|
|
2017-02-10 01:57:19 +00:00
|
|
|
def change_state(api_url, post_data):
|
|
|
|
"""Update a state using the Zoneminder API."""
|
|
|
|
return _zm_request('post', api_url, data=post_data)
|