133 lines
4.7 KiB
Python
133 lines
4.7 KiB
Python
"""
|
|
Support for monitoring OctoPrint sensors.
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
https://home-assistant.io/components/sensor.octoprint/
|
|
"""
|
|
import logging
|
|
|
|
import requests
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
|
from homeassistant.const import (
|
|
TEMP_CELSIUS, CONF_NAME, CONF_MONITORED_CONDITIONS)
|
|
from homeassistant.helpers.entity import Entity
|
|
from homeassistant.loader import get_component
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
DEPENDENCIES = ['octoprint']
|
|
|
|
DEFAULT_NAME = 'OctoPrint'
|
|
|
|
SENSOR_TYPES = {
|
|
# API Endpoint, Group, Key, unit
|
|
'Temperatures': ['printer', 'temperature', '*', TEMP_CELSIUS],
|
|
'Current State': ['printer', 'state', 'text', None],
|
|
'Job Percentage': ['job', 'progress', 'completion', '%'],
|
|
}
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
vol.Optional(CONF_MONITORED_CONDITIONS, default=SENSOR_TYPES):
|
|
vol.All(cv.ensure_list, [vol.In(SENSOR_TYPES)]),
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
})
|
|
|
|
|
|
# pylint: disable=unused-argument
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
|
"""Setup the available OctoPrint sensors."""
|
|
octoprint = get_component('octoprint')
|
|
name = config.get(CONF_NAME)
|
|
monitored_conditions = config.get(CONF_MONITORED_CONDITIONS)
|
|
|
|
devices = []
|
|
types = ["actual", "target"]
|
|
for octo_type in monitored_conditions:
|
|
if octo_type == "Temperatures":
|
|
for tool in octoprint.OCTOPRINT.get_tools():
|
|
for temp_type in types:
|
|
new_sensor = OctoPrintSensor(octoprint.OCTOPRINT,
|
|
temp_type,
|
|
temp_type,
|
|
name,
|
|
SENSOR_TYPES[octo_type][3],
|
|
SENSOR_TYPES[octo_type][0],
|
|
SENSOR_TYPES[octo_type][1],
|
|
tool)
|
|
devices.append(new_sensor)
|
|
else:
|
|
new_sensor = OctoPrintSensor(octoprint.OCTOPRINT,
|
|
octo_type,
|
|
SENSOR_TYPES[octo_type][2],
|
|
name,
|
|
SENSOR_TYPES[octo_type][3],
|
|
SENSOR_TYPES[octo_type][0],
|
|
SENSOR_TYPES[octo_type][1])
|
|
devices.append(new_sensor)
|
|
add_devices(devices)
|
|
|
|
|
|
class OctoPrintSensor(Entity):
|
|
"""Representation of an OctoPrint sensor."""
|
|
|
|
def __init__(self, api, condition, sensor_type, sensor_name, unit,
|
|
endpoint, group, tool=None):
|
|
"""Initialize a new OctoPrint sensor."""
|
|
self.sensor_name = sensor_name
|
|
if tool is None:
|
|
self._name = '{} {}'.format(sensor_name, condition)
|
|
else:
|
|
self._name = '{} {} {} {}'.format(
|
|
sensor_name, condition, tool, 'temp')
|
|
self.sensor_type = sensor_type
|
|
self.api = api
|
|
self._state = None
|
|
self._unit_of_measurement = unit
|
|
self.api_endpoint = endpoint
|
|
self.api_group = group
|
|
self.api_tool = tool
|
|
# Set initial state
|
|
self.update()
|
|
_LOGGER.debug("Created OctoPrint sensor %r", self)
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of the sensor."""
|
|
return self._name
|
|
|
|
@property
|
|
def state(self):
|
|
"""Return the state of the sensor."""
|
|
sensor_unit = self.unit_of_measurement
|
|
if sensor_unit == TEMP_CELSIUS or sensor_unit == "%":
|
|
# API sometimes returns null and not 0
|
|
if self._state is None:
|
|
self._state = 0
|
|
return round(self._state, 2)
|
|
else:
|
|
return self._state
|
|
|
|
@property
|
|
def unit_of_measurement(self):
|
|
"""Unit of measurement of this entity, if any."""
|
|
return self._unit_of_measurement
|
|
|
|
def update(self):
|
|
"""Update state of sensor."""
|
|
try:
|
|
self._state = self.api.update(self.sensor_type,
|
|
self.api_endpoint,
|
|
self.api_group,
|
|
self.api_tool)
|
|
except requests.exceptions.ConnectionError:
|
|
# Error calling the api, already logged in api.update()
|
|
return
|
|
|
|
if self._state is None:
|
|
_LOGGER.warning("Unable to locate value for %s", self.sensor_type)
|
|
return
|