core/homeassistant/components/sensor/rest.py

139 lines
4.6 KiB
Python
Raw Normal View History

2015-09-13 20:41:16 +00:00
"""
Support for RESTful API sensors.
2015-09-13 20:41:16 +00:00
For more details about this platform, please refer to the documentation at
2015-11-09 12:12:18 +00:00
https://home-assistant.io/components/sensor.rest/
2015-09-13 20:41:16 +00:00
"""
import logging
2016-02-19 05:27:50 +00:00
import voluptuous as vol
import requests
from requests.auth import HTTPBasicAuth, HTTPDigestAuth
2015-09-13 20:41:16 +00:00
from homeassistant.components.sensor import PLATFORM_SCHEMA
from homeassistant.const import (
CONF_PAYLOAD, CONF_NAME, CONF_VALUE_TEMPLATE, CONF_METHOD, CONF_RESOURCE,
CONF_UNIT_OF_MEASUREMENT, STATE_UNKNOWN, CONF_VERIFY_SSL, CONF_USERNAME,
CONF_PASSWORD, CONF_AUTHENTICATION, HTTP_BASIC_AUTHENTICATION,
HTTP_DIGEST_AUTHENTICATION, CONF_HEADERS)
2015-09-13 20:41:16 +00:00
from homeassistant.helpers.entity import Entity
import homeassistant.helpers.config_validation as cv
2015-09-13 20:41:16 +00:00
2016-09-03 23:45:49 +00:00
_LOGGER = logging.getLogger(__name__)
DEFAULT_METHOD = 'GET'
DEFAULT_NAME = 'REST Sensor'
2016-09-03 23:45:49 +00:00
DEFAULT_VERIFY_SSL = True
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
vol.Required(CONF_RESOURCE): cv.url,
vol.Optional(CONF_AUTHENTICATION):
vol.In([HTTP_BASIC_AUTHENTICATION, HTTP_DIGEST_AUTHENTICATION]),
vol.Optional(CONF_HEADERS): {cv.string: cv.string},
vol.Optional(CONF_METHOD, default=DEFAULT_METHOD): vol.In(['POST', 'GET']),
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
vol.Optional(CONF_PASSWORD): cv.string,
vol.Optional(CONF_PAYLOAD): cv.string,
vol.Optional(CONF_UNIT_OF_MEASUREMENT): cv.string,
vol.Optional(CONF_USERNAME): cv.string,
vol.Optional(CONF_VALUE_TEMPLATE): cv.template,
2016-09-03 23:45:49 +00:00
vol.Optional(CONF_VERIFY_SSL, default=DEFAULT_VERIFY_SSL): cv.boolean,
})
2015-09-13 20:41:16 +00:00
def setup_platform(hass, config, add_devices, discovery_info=None):
"""Set up the RESTful sensor."""
name = config.get(CONF_NAME)
resource = config.get(CONF_RESOURCE)
method = config.get(CONF_METHOD)
payload = config.get(CONF_PAYLOAD)
2016-09-03 23:45:49 +00:00
verify_ssl = config.get(CONF_VERIFY_SSL)
username = config.get(CONF_USERNAME)
password = config.get(CONF_PASSWORD)
headers = config.get(CONF_HEADERS)
unit = config.get(CONF_UNIT_OF_MEASUREMENT)
value_template = config.get(CONF_VALUE_TEMPLATE)
if value_template is not None:
value_template.hass = hass
if username and password:
if config.get(CONF_AUTHENTICATION) == HTTP_DIGEST_AUTHENTICATION:
auth = HTTPDigestAuth(username, password)
else:
auth = HTTPBasicAuth(username, password)
else:
auth = None
rest = RestData(method, resource, auth, headers, payload, verify_ssl)
2016-01-02 21:29:33 +00:00
rest.update()
2016-01-02 21:29:33 +00:00
if rest.data is None:
_LOGGER.error("Unable to fetch REST data")
2016-01-02 21:29:33 +00:00
return False
2015-09-13 20:41:16 +00:00
add_devices([RestSensor(hass, rest, name, unit, value_template)])
2015-09-13 20:41:16 +00:00
class RestSensor(Entity):
2016-03-08 15:46:34 +00:00
"""Implementation of a REST sensor."""
2015-09-13 20:41:16 +00:00
2015-12-11 22:19:49 +00:00
def __init__(self, hass, rest, name, unit_of_measurement, value_template):
2016-09-03 23:45:49 +00:00
"""Initialize the REST sensor."""
2015-12-11 22:19:49 +00:00
self._hass = hass
2015-09-13 20:41:16 +00:00
self.rest = rest
self._name = name
2015-12-19 09:06:54 +00:00
self._state = STATE_UNKNOWN
2015-09-13 20:41:16 +00:00
self._unit_of_measurement = unit_of_measurement
2015-12-11 22:19:49 +00:00
self._value_template = value_template
2015-09-13 20:41:16 +00:00
self.update()
@property
def name(self):
2016-03-08 15:46:34 +00:00
"""Return the name of the sensor."""
2015-09-13 20:41:16 +00:00
return self._name
@property
def unit_of_measurement(self):
2016-03-08 15:46:34 +00:00
"""Return the unit the value is expressed in."""
2015-09-13 20:41:16 +00:00
return self._unit_of_measurement
@property
def state(self):
2016-03-08 15:46:34 +00:00
"""Return the state of the device."""
2015-09-13 20:41:16 +00:00
return self._state
def update(self):
2016-03-08 15:46:34 +00:00
"""Get the latest data from REST API and update the state."""
2015-09-13 20:41:16 +00:00
self.rest.update()
value = self.rest.data
2015-12-25 21:34:30 +00:00
if value is None:
value = STATE_UNKNOWN
elif self._value_template is not None:
value = self._value_template.render_with_possible_json_value(
value, STATE_UNKNOWN)
2015-12-25 21:34:30 +00:00
self._state = value
2015-09-13 20:41:16 +00:00
2016-01-02 21:29:33 +00:00
class RestData(object):
"""Class for handling the data retrieval."""
2015-09-13 20:41:16 +00:00
def __init__(self, method, resource, auth, headers, data, verify_ssl):
2016-03-08 15:46:34 +00:00
"""Initialize the data object."""
self._request = requests.Request(
method, resource, headers=headers, auth=auth, data=data).prepare()
self._verify_ssl = verify_ssl
2015-12-25 21:34:30 +00:00
self.data = None
def update(self):
2016-10-31 04:51:03 +00:00
"""Get the latest data from REST service with provided method."""
try:
2016-01-02 21:29:33 +00:00
with requests.Session() as sess:
response = sess.send(
self._request, timeout=10, verify=self._verify_ssl)
2015-12-11 22:19:49 +00:00
self.data = response.text
2016-01-02 21:29:33 +00:00
except requests.exceptions.RequestException:
_LOGGER.error("Error fetching data: %s", self._request)
2015-12-25 21:34:30 +00:00
self.data = None