Add support for IPP Printers to the CUPS integration (#24756)

* Add support for IPP Printers to the CUPS integration

* Fixed lint error

* Addressed comments, removed redundant check

* Simplified check, improved code readability
This commit is contained in:
Matte23 2019-06-26 01:13:08 +02:00 committed by Paulus Schoutsen
parent bd4f66fda3
commit 29311e6391

View File

@ -8,6 +8,7 @@ import voluptuous as vol
import homeassistant.helpers.config_validation as cv import homeassistant.helpers.config_validation as cv
from homeassistant.components.sensor import PLATFORM_SCHEMA from homeassistant.components.sensor import PLATFORM_SCHEMA
from homeassistant.const import CONF_HOST, CONF_PORT from homeassistant.const import CONF_HOST, CONF_PORT
from homeassistant.exceptions import PlatformNotReady
from homeassistant.helpers.entity import Entity from homeassistant.helpers.entity import Entity
_LOGGER = logging.getLogger(__name__) _LOGGER = logging.getLogger(__name__)
@ -23,9 +24,11 @@ ATTR_PRINTER_TYPE = 'printer_type'
ATTR_PRINTER_URI_SUPPORTED = 'printer_uri_supported' ATTR_PRINTER_URI_SUPPORTED = 'printer_uri_supported'
CONF_PRINTERS = 'printers' CONF_PRINTERS = 'printers'
CONF_IS_CUPS_SERVER = 'is_cups_server'
DEFAULT_HOST = '127.0.0.1' DEFAULT_HOST = '127.0.0.1'
DEFAULT_PORT = 631 DEFAULT_PORT = 631
DEFAULT_IS_CUPS_SERVER = True
ICON = 'mdi:printer' ICON = 'mdi:printer'
@ -39,6 +42,8 @@ PRINTER_STATES = {
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({ PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
vol.Required(CONF_PRINTERS): vol.All(cv.ensure_list, [cv.string]), vol.Required(CONF_PRINTERS): vol.All(cv.ensure_list, [cv.string]),
vol.Optional(CONF_IS_CUPS_SERVER,
default=DEFAULT_IS_CUPS_SERVER): cv.boolean,
vol.Optional(CONF_HOST, default=DEFAULT_HOST): cv.string, vol.Optional(CONF_HOST, default=DEFAULT_HOST): cv.string,
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port, vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
}) })
@ -49,21 +54,36 @@ def setup_platform(hass, config, add_entities, discovery_info=None):
host = config.get(CONF_HOST) host = config.get(CONF_HOST)
port = config.get(CONF_PORT) port = config.get(CONF_PORT)
printers = config.get(CONF_PRINTERS) printers = config.get(CONF_PRINTERS)
is_cups = config.get(CONF_IS_CUPS_SERVER)
try: if is_cups:
data = CupsData(host, port) data = CupsData(host, port, None)
data.update() data.update()
except RuntimeError: if data.available is False:
_LOGGER.error("Unable to connect to CUPS server: %s:%s", host, port) _LOGGER.error("Unable to connect to CUPS server: %s:%s",
return False host, port)
raise PlatformNotReady()
dev = []
for printer in printers:
if printer not in data.printers:
_LOGGER.error("Printer is not present: %s", printer)
continue
dev.append(CupsSensor(data, printer))
add_entities(dev, True)
return
data = CupsData(host, port, printers)
data.update()
if data.available is False:
_LOGGER.error("Unable to connect to IPP printer: %s:%s",
host, port)
raise PlatformNotReady()
dev = [] dev = []
for printer in printers: for printer in printers:
if printer in data.printers: dev.append(IPPSensor(data, printer))
dev.append(CupsSensor(data, printer))
else:
_LOGGER.error("Printer is not present: %s", printer)
continue
add_entities(dev, True) add_entities(dev, True)
@ -76,6 +96,7 @@ class CupsSensor(Entity):
self.data = data self.data = data
self._name = printer self._name = printer
self._printer = None self._printer = None
self._available = False
@property @property
def name(self): def name(self):
@ -85,12 +106,16 @@ class CupsSensor(Entity):
@property @property
def state(self): def state(self):
"""Return the state of the sensor.""" """Return the state of the sensor."""
if self._printer is not None: if self._printer is None:
try: return None
return next(v for k, v in PRINTER_STATES.items()
if self._printer['printer-state'] == k) key = self._printer['printer-state']
except StopIteration: return PRINTER_STATES.get(key, key)
return self._printer['printer-state']
@property
def available(self):
"""Return True if entity is available."""
return self._available
@property @property
def icon(self): def icon(self):
@ -100,41 +125,133 @@ class CupsSensor(Entity):
@property @property
def device_state_attributes(self): def device_state_attributes(self):
"""Return the state attributes of the sensor.""" """Return the state attributes of the sensor."""
if self._printer is not None: if self._printer is None:
return { return None
ATTR_DEVICE_URI: self._printer['device-uri'],
ATTR_PRINTER_INFO: self._printer['printer-info'], return {
ATTR_PRINTER_IS_SHARED: self._printer['printer-is-shared'], ATTR_DEVICE_URI: self._printer['device-uri'],
ATTR_PRINTER_LOCATION: self._printer['printer-location'], ATTR_PRINTER_INFO: self._printer['printer-info'],
ATTR_PRINTER_MODEL: self._printer['printer-make-and-model'], ATTR_PRINTER_IS_SHARED: self._printer['printer-is-shared'],
ATTR_PRINTER_STATE_MESSAGE: ATTR_PRINTER_LOCATION: self._printer['printer-location'],
self._printer['printer-state-message'], ATTR_PRINTER_MODEL: self._printer['printer-make-and-model'],
ATTR_PRINTER_STATE_REASON: ATTR_PRINTER_STATE_MESSAGE:
self._printer['printer-state-reasons'], self._printer['printer-state-message'],
ATTR_PRINTER_TYPE: self._printer['printer-type'], ATTR_PRINTER_STATE_REASON:
ATTR_PRINTER_URI_SUPPORTED: self._printer['printer-state-reasons'],
self._printer['printer-uri-supported'], ATTR_PRINTER_TYPE: self._printer['printer-type'],
} ATTR_PRINTER_URI_SUPPORTED:
self._printer['printer-uri-supported'],
}
def update(self): def update(self):
"""Get the latest data and updates the states.""" """Get the latest data and updates the states."""
self.data.update() self.data.update()
self._printer = self.data.printers.get(self._name) self._printer = self.data.printers.get(self._name)
self._available = self.data.available
class IPPSensor(Entity):
"""Implementation of the IPPSensor.
This sensor represents the status of the printer.
"""
def __init__(self, data, name):
"""Initialize the sensor."""
self.data = data
self._name = name
self._attributes = None
self._available = False
@property
def name(self):
"""Return the name of the sensor."""
return self._attributes['printer-make-and-model']
@property
def icon(self):
"""Return the icon to use in the frontend."""
return ICON
@property
def available(self):
"""Return True if entity is available."""
return self._available
@property
def state(self):
"""Return the state of the sensor."""
if self._attributes is None:
return None
key = self._attributes['printer-state']
return PRINTER_STATES.get(key, key)
@property
def device_state_attributes(self):
"""Return the state attributes of the sensor."""
if self._attributes is None:
return None
state_attributes = {}
if 'printer-info' in self._attributes:
state_attributes[ATTR_PRINTER_INFO] = \
self._attributes['printer-info']
if 'printer-location' in self._attributes:
state_attributes[ATTR_PRINTER_LOCATION] = \
self._attributes['printer-location']
if 'printer-state-message' in self._attributes:
state_attributes[ATTR_PRINTER_STATE_MESSAGE] = \
self._attributes['printer-state-message']
if 'printer-state-reasons' in self._attributes:
state_attributes[ATTR_PRINTER_STATE_REASON] = \
self._attributes['printer-state-reasons']
if 'printer-uri-supported' in self._attributes:
state_attributes[ATTR_PRINTER_URI_SUPPORTED] = \
self._attributes['printer-uri-supported']
return state_attributes
def update(self):
"""Fetch new state data for the sensor."""
self.data.update()
self._attributes = self.data.attributes.get(self._name)
self._available = self.data.available
# pylint: disable=no-name-in-module # pylint: disable=no-name-in-module
class CupsData: class CupsData:
"""Get the latest data from CUPS and update the state.""" """Get the latest data from CUPS and update the state."""
def __init__(self, host, port): def __init__(self, host, port, ipp_printers):
"""Initialize the data object.""" """Initialize the data object."""
self._host = host self._host = host
self._port = port self._port = port
self._ipp_printers = ipp_printers
self.is_cups = (ipp_printers is None)
self.printers = None self.printers = None
self.attributes = {}
self.available = False
def update(self): def update(self):
"""Get the latest data from CUPS.""" """Get the latest data from CUPS."""
cups = importlib.import_module('cups') cups = importlib.import_module('cups')
conn = cups.Connection(host=self._host, port=self._port) try:
self.printers = conn.getPrinters() conn = cups.Connection(host=self._host, port=self._port)
if self.is_cups:
self.printers = conn.getPrinters()
else:
for ipp_printer in self._ipp_printers:
self.attributes[ipp_printer] = conn.getPrinterAttributes(
uri="ipp://{}:{}/{}"
.format(self._host, self._port, ipp_printer))
self.available = True
except RuntimeError:
self.available = False