power: rework configuration

Devices are now configured using "prefix" sections.   The pin configuration now more closely mimics that of Klipper's configuration so as to reduce confusion.

Signed-off-by:  Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
Arksine 2020-11-14 08:24:54 -05:00
parent 1975b875a5
commit 48266e0bd6
1 changed files with 46 additions and 27 deletions

View File

@ -6,6 +6,7 @@
import logging import logging
import os import os
import asyncio
from tornado.ioloop import IOLoop from tornado.ioloop import IOLoop
from tornado import gen from tornado import gen
@ -29,22 +30,17 @@ class PrinterPower:
self.current_dev = None self.current_dev = None
self.devices = {} self.devices = {}
dev_names = config.get('devices') prefix_sections = config.get_prefix_sections("power")
dev_names = [d.strip() for d in dev_names.split(',') if d.strip()] logging.info(f"Power plugin loading devices: f{prefix_sections}")
logging.info("Power plugin loading devices: " + str(dev_names))
devices = {} devices = {}
for dev in dev_names: for section in prefix_sections:
devices[dev] = GpioDevice(dev, config) dev = GpioDevice(config[section])
devices[dev.name] = dev
ioloop = IOLoop.current() ioloop = IOLoop.current()
ioloop.spawn_callback(self.initialize_devices, devices) ioloop.spawn_callback(self.initialize_devices, devices)
async def _handle_list_devices(self, web_request): async def _handle_list_devices(self, web_request):
output = {"devices": []} output = {"devices": list(self.devices.keys())}
for dev in self.devices:
output['devices'].append({
"name": self.devices[dev].name,
"id": dev
})
return output return output
async def _handle_power_request(self, web_request): async def _handle_power_request(self, web_request):
@ -70,7 +66,6 @@ class PrinterPower:
async def _power_dev(self, dev, req): async def _power_dev(self, dev, req):
if dev not in self.devices: if dev not in self.devices:
return False return False
if req in ["on", "off"]: if req in ["on", "off"]:
await self.devices[dev].power(req) await self.devices[dev].power(req)
@ -80,8 +75,9 @@ class PrinterPower:
}) })
elif req != "status": elif req != "status":
raise self.server.error("Unsupported power request") raise self.server.error("Unsupported power request")
ret = self.devices[dev].refresh_status()
await self.devices[dev].refresh_status() if asyncio.iscoroutine(ret):
await ret
return True return True
async def initialize_devices(self, devices): async def initialize_devices(self, devices):
@ -109,9 +105,12 @@ class PrinterPower:
ioloop = IOLoop.current() ioloop = IOLoop.current()
ioloop.spawn_callback(self._power_dev, device, status) ioloop.spawn_callback(self._power_dev, device, status)
async def add_device(self, dev, device): async def add_device(self, name, device):
if name in self.devices:
raise self.server.error(
f"Device [{name}] already configured")
await device.initialize() await device.initialize()
self.devices[dev] = device self.devices[name] = device
class GPIO: class GPIO:
@ -189,24 +188,44 @@ class GPIO:
class GpioDevice: class GpioDevice:
def __init__(self, dev, config): def __init__(self, config):
self.name = config.get(dev + "_name", dev) name_parts = config.get_name().split(maxsplit=1)
if len(name_parts) != 2:
raise config.error(f"Invalid Section Name: {config.get_name()}")
self.name = name_parts[1]
self.status = None self.status = None
self.pin = config.getint(dev + "_pin") cfg_pin = pin = config.get("pin")
self.active_low = int(config.getboolean(dev + "_active_low", False)) self.invert = False
if pin[0] == "!":
pin = pin[1:]
self.invert = True
self.consumer = "gpiochip0"
pin_parts = pin.split("/")
self.pin = ""
if len(pin_parts) == 2:
self.consumer, self.pin = pin_parts
elif len(pin_parts) == 1:
self.pin = pin_parts[0]
# Verify pin
if not self.consumer.startswith("gpiochip") or \
not self.consumer[-1].isdigit() or \
not self.pin.startswith("gpio") or \
not self.pin[4:].isdigit():
raise config.error(
f"Invalid Power Pin configuration: {cfg_pin}")
self.pin = int(self.pin[4:])
async def initialize(self): async def initialize(self):
logging.debug(f"Attempting to configure pin GPIO{self.pin}") await GPIO.setup_pin(self.pin, int(self.invert))
await GPIO.setup_pin(self.pin, self.active_low) self.refresh_status()
await self.refresh_status()
async def refresh_status(self): def refresh_status(self):
self.status = GPIO.is_pin_on(self.pin) self.status = GPIO.is_pin_on(self.pin)
async def power(self, status): async def power(self, status):
await GPIO.verify_pin(self.pin, self.active_low) await GPIO.verify_pin(self.pin, int(self.invert))
GPIO.set_pin_value(self.pin, int(status == "on")) GPIO.set_pin_value(self.pin, int(status == "on"))
# The power plugin has multiple configuration sections
def load_plugin(config): def load_plugin_multi(config):
return PrinterPower(config) return PrinterPower(config)