2016-05-04 01:35:11 +00:00
|
|
|
"""
|
|
|
|
Support for monitoring OctoPrint 3D printers.
|
|
|
|
|
2016-05-07 01:03:18 +00:00
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/octoprint/
|
2016-05-04 01:35:11 +00:00
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
import time
|
|
|
|
|
2016-09-02 10:26:23 +00:00
|
|
|
import requests
|
|
|
|
import voluptuous as vol
|
2017-11-04 19:04:05 +00:00
|
|
|
from aiohttp.hdrs import CONTENT_TYPE
|
2016-05-04 01:35:11 +00:00
|
|
|
|
2016-09-02 10:26:23 +00:00
|
|
|
from homeassistant.const import CONF_API_KEY, CONF_HOST, CONTENT_TYPE_JSON
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2016-09-02 10:26:23 +00:00
|
|
|
DOMAIN = 'octoprint'
|
2017-08-22 13:37:06 +00:00
|
|
|
CONF_NUMBER_OF_TOOLS = 'number_of_tools'
|
|
|
|
CONF_BED = 'bed'
|
2016-09-02 10:26:23 +00:00
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Required(CONF_API_KEY): cv.string,
|
|
|
|
vol.Required(CONF_HOST): cv.string,
|
2017-08-22 13:37:06 +00:00
|
|
|
vol.Optional(CONF_NUMBER_OF_TOOLS, default=0): cv.positive_int,
|
|
|
|
vol.Optional(CONF_BED, default=False): cv.boolean
|
2016-09-02 10:26:23 +00:00
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the OctoPrint component."""
|
2016-09-02 10:26:23 +00:00
|
|
|
base_url = 'http://{}/api/'.format(config[DOMAIN][CONF_HOST])
|
2016-05-04 01:35:11 +00:00
|
|
|
api_key = config[DOMAIN][CONF_API_KEY]
|
2017-08-22 13:37:06 +00:00
|
|
|
number_of_tools = config[DOMAIN][CONF_NUMBER_OF_TOOLS]
|
|
|
|
bed = config[DOMAIN][CONF_BED]
|
2016-05-04 01:35:11 +00:00
|
|
|
|
2017-07-07 05:39:11 +00:00
|
|
|
hass.data[DOMAIN] = {"api": None}
|
|
|
|
|
2016-05-04 01:35:11 +00:00
|
|
|
try:
|
2017-08-22 13:37:06 +00:00
|
|
|
octoprint_api = OctoPrintAPI(base_url, api_key, bed, number_of_tools)
|
2017-07-07 05:39:11 +00:00
|
|
|
hass.data[DOMAIN]["api"] = octoprint_api
|
|
|
|
octoprint_api.get('printer')
|
|
|
|
octoprint_api.get('job')
|
2016-05-04 01:35:11 +00:00
|
|
|
except requests.exceptions.RequestException as conn_err:
|
|
|
|
_LOGGER.error("Error setting up OctoPrint API: %r", conn_err)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class OctoPrintAPI(object):
|
2016-05-07 01:03:18 +00:00
|
|
|
"""Simple JSON wrapper for OctoPrint's API."""
|
2016-05-04 01:35:11 +00:00
|
|
|
|
2017-08-22 13:37:06 +00:00
|
|
|
def __init__(self, api_url, key, bed, number_of_tools):
|
2016-05-04 01:35:11 +00:00
|
|
|
"""Initialize OctoPrint API and set headers needed later."""
|
|
|
|
self.api_url = api_url
|
2017-11-04 19:04:05 +00:00
|
|
|
self.headers = {
|
|
|
|
CONTENT_TYPE: CONTENT_TYPE_JSON,
|
|
|
|
'X-Api-Key': key,
|
|
|
|
}
|
2016-05-04 01:35:11 +00:00
|
|
|
self.printer_last_reading = [{}, None]
|
|
|
|
self.job_last_reading = [{}, None]
|
2017-07-07 05:39:11 +00:00
|
|
|
self.job_available = False
|
|
|
|
self.printer_available = False
|
|
|
|
self.available = False
|
|
|
|
self.printer_error_logged = False
|
|
|
|
self.job_error_logged = False
|
2017-08-22 13:37:06 +00:00
|
|
|
self.bed = bed
|
|
|
|
self.number_of_tools = number_of_tools
|
|
|
|
_LOGGER.error(str(bed) + " " + str(number_of_tools))
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
def get_tools(self):
|
2017-08-22 13:37:06 +00:00
|
|
|
"""Get the list of tools that temperature is monitored on."""
|
|
|
|
tools = []
|
|
|
|
if self.number_of_tools > 0:
|
|
|
|
for tool_number in range(0, self.number_of_tools):
|
|
|
|
tools.append("tool" + str(tool_number))
|
|
|
|
if self.bed:
|
|
|
|
tools.append('bed')
|
|
|
|
if not self.bed and self.number_of_tools == 0:
|
|
|
|
temps = self.printer_last_reading[0].get('temperature')
|
|
|
|
if temps is not None:
|
|
|
|
tools = temps.keys()
|
|
|
|
return tools
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
def get(self, endpoint):
|
|
|
|
"""Send a get request, and return the response as a dict."""
|
2017-07-07 05:39:11 +00:00
|
|
|
# Only query the API at most every 30 seconds
|
2016-05-04 01:35:11 +00:00
|
|
|
now = time.time()
|
|
|
|
if endpoint == "job":
|
|
|
|
last_time = self.job_last_reading[1]
|
|
|
|
if last_time is not None:
|
|
|
|
if now - last_time < 30.0:
|
|
|
|
return self.job_last_reading[0]
|
|
|
|
elif endpoint == "printer":
|
|
|
|
last_time = self.printer_last_reading[1]
|
|
|
|
if last_time is not None:
|
|
|
|
if now - last_time < 30.0:
|
|
|
|
return self.printer_last_reading[0]
|
2017-07-07 05:39:11 +00:00
|
|
|
|
2016-05-04 01:35:11 +00:00
|
|
|
url = self.api_url + endpoint
|
|
|
|
try:
|
2017-04-30 05:04:49 +00:00
|
|
|
response = requests.get(
|
2017-07-07 05:39:11 +00:00
|
|
|
url, headers=self.headers, timeout=9)
|
2016-05-04 01:35:11 +00:00
|
|
|
response.raise_for_status()
|
|
|
|
if endpoint == "job":
|
|
|
|
self.job_last_reading[0] = response.json()
|
|
|
|
self.job_last_reading[1] = time.time()
|
2017-07-07 05:39:11 +00:00
|
|
|
self.job_available = True
|
2016-05-04 01:35:11 +00:00
|
|
|
elif endpoint == "printer":
|
|
|
|
self.printer_last_reading[0] = response.json()
|
|
|
|
self.printer_last_reading[1] = time.time()
|
2017-07-07 05:39:11 +00:00
|
|
|
self.printer_available = True
|
|
|
|
self.available = self.printer_available and self.job_available
|
|
|
|
if self.available:
|
|
|
|
self.job_error_logged = False
|
|
|
|
self.printer_error_logged = False
|
2016-05-04 01:35:11 +00:00
|
|
|
return response.json()
|
2017-05-04 20:13:09 +00:00
|
|
|
except (requests.exceptions.ConnectionError,
|
2017-07-07 05:39:11 +00:00
|
|
|
requests.exceptions.HTTPError,
|
|
|
|
requests.exceptions.ReadTimeout) as conn_exc:
|
|
|
|
log_string = "Failed to update OctoPrint status. " + \
|
|
|
|
" Error: %s" % (conn_exc)
|
|
|
|
# Only log the first failure
|
|
|
|
if endpoint == "job":
|
|
|
|
log_string = "Endpoint: job " + log_string
|
|
|
|
if not self.job_error_logged:
|
|
|
|
_LOGGER.error(log_string)
|
|
|
|
self.job_error_logged = True
|
|
|
|
self.job_available = False
|
|
|
|
elif endpoint == "printer":
|
|
|
|
log_string = "Endpoint: printer " + log_string
|
|
|
|
if not self.printer_error_logged:
|
|
|
|
_LOGGER.error(log_string)
|
|
|
|
self.printer_error_logged = True
|
|
|
|
self.printer_available = False
|
|
|
|
self.available = False
|
|
|
|
return None
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
def update(self, sensor_type, end_point, group, tool=None):
|
|
|
|
"""Return the value for sensor_type from the provided endpoint."""
|
2017-05-04 20:13:09 +00:00
|
|
|
response = self.get(end_point)
|
|
|
|
if response is not None:
|
|
|
|
return get_value_from_json(response, sensor_type, group, tool)
|
2017-07-07 05:39:11 +00:00
|
|
|
return response
|
2016-05-04 01:35:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=unused-variable
|
|
|
|
def get_value_from_json(json_dict, sensor_type, group, tool):
|
2016-05-07 01:03:18 +00:00
|
|
|
"""Return the value for sensor_type from the JSON."""
|
2017-07-06 03:02:16 +00:00
|
|
|
if group not in json_dict:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if sensor_type in json_dict[group]:
|
|
|
|
if sensor_type == "target" and json_dict[sensor_type] is None:
|
|
|
|
return 0
|
|
|
|
return json_dict[group][sensor_type]
|
|
|
|
|
|
|
|
elif tool is not None:
|
|
|
|
if sensor_type in json_dict[group][tool]:
|
|
|
|
return json_dict[group][tool][sensor_type]
|
2017-07-07 05:39:11 +00:00
|
|
|
|
|
|
|
return None
|