161 lines
5.0 KiB
Python
161 lines
5.0 KiB
Python
"""Code to handle a Firmata board."""
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Mapping
|
|
import logging
|
|
from typing import Literal, Union
|
|
|
|
from pymata_express.pymata_express import PymataExpress
|
|
from pymata_express.pymata_express_serial import serial
|
|
|
|
from homeassistant.const import (
|
|
CONF_BINARY_SENSORS,
|
|
CONF_LIGHTS,
|
|
CONF_NAME,
|
|
CONF_SENSORS,
|
|
CONF_SWITCHES,
|
|
)
|
|
|
|
from .const import (
|
|
CONF_ARDUINO_INSTANCE_ID,
|
|
CONF_ARDUINO_WAIT,
|
|
CONF_SAMPLING_INTERVAL,
|
|
CONF_SERIAL_BAUD_RATE,
|
|
CONF_SERIAL_PORT,
|
|
CONF_SLEEP_TUNE,
|
|
PIN_TYPE_ANALOG,
|
|
PIN_TYPE_DIGITAL,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
FirmataPinType = Union[int, str]
|
|
|
|
|
|
class FirmataBoard:
|
|
"""Manages a single Firmata board."""
|
|
|
|
def __init__(self, config: Mapping) -> None:
|
|
"""Initialize the board."""
|
|
self.config = config
|
|
self.api: PymataExpress = None
|
|
self.firmware_version: str | None = None
|
|
self.protocol_version = None
|
|
self.name = self.config[CONF_NAME]
|
|
self.switches = []
|
|
self.lights = []
|
|
self.binary_sensors = []
|
|
self.sensors = []
|
|
self.used_pins: list[FirmataPinType] = []
|
|
|
|
if CONF_SWITCHES in self.config:
|
|
self.switches = self.config[CONF_SWITCHES]
|
|
if CONF_LIGHTS in self.config:
|
|
self.lights = self.config[CONF_LIGHTS]
|
|
if CONF_BINARY_SENSORS in self.config:
|
|
self.binary_sensors = self.config[CONF_BINARY_SENSORS]
|
|
if CONF_SENSORS in self.config:
|
|
self.sensors = self.config[CONF_SENSORS]
|
|
|
|
async def async_setup(self, tries=0) -> bool:
|
|
"""Set up a Firmata instance."""
|
|
try:
|
|
_LOGGER.debug("Connecting to Firmata %s", self.name)
|
|
self.api = await get_board(self.config)
|
|
except RuntimeError as err:
|
|
_LOGGER.error("Error connecting to PyMata board %s: %s", self.name, err)
|
|
return False
|
|
except serial.serialutil.SerialTimeoutException as err:
|
|
_LOGGER.error(
|
|
"Timeout writing to serial port for PyMata board %s: %s", self.name, err
|
|
)
|
|
return False
|
|
except serial.serialutil.SerialException as err:
|
|
_LOGGER.error(
|
|
"Error connecting to serial port for PyMata board %s: %s",
|
|
self.name,
|
|
err,
|
|
)
|
|
return False
|
|
|
|
self.firmware_version = await self.api.get_firmware_version()
|
|
if not self.firmware_version:
|
|
_LOGGER.error(
|
|
"Error retrieving firmware version from Firmata board %s", self.name
|
|
)
|
|
return False
|
|
|
|
if CONF_SAMPLING_INTERVAL in self.config:
|
|
try:
|
|
await self.api.set_sampling_interval(
|
|
self.config[CONF_SAMPLING_INTERVAL]
|
|
)
|
|
except RuntimeError as err:
|
|
_LOGGER.error(
|
|
"Error setting sampling interval for PyMata board %s: %s",
|
|
self.name,
|
|
err,
|
|
)
|
|
return False
|
|
|
|
_LOGGER.debug("Firmata connection successful for %s", self.name)
|
|
return True
|
|
|
|
async def async_reset(self) -> bool:
|
|
"""Reset the board to default state."""
|
|
_LOGGER.debug("Shutting down board %s", self.name)
|
|
# If the board was never setup, continue.
|
|
if self.api is None:
|
|
return True
|
|
|
|
await self.api.shutdown()
|
|
self.api = None
|
|
|
|
return True
|
|
|
|
def mark_pin_used(self, pin: FirmataPinType) -> bool:
|
|
"""Test if a pin is used already on the board or mark as used."""
|
|
if pin in self.used_pins:
|
|
return False
|
|
self.used_pins.append(pin)
|
|
return True
|
|
|
|
def get_pin_type(self, pin: FirmataPinType) -> tuple[Literal[0, 1], int]:
|
|
"""Return the type and Firmata location of a pin on the board."""
|
|
pin_type: Literal[0, 1]
|
|
firmata_pin: int
|
|
if isinstance(pin, str):
|
|
pin_type = PIN_TYPE_ANALOG
|
|
firmata_pin = int(pin[1:])
|
|
firmata_pin += self.api.first_analog_pin
|
|
else:
|
|
pin_type = PIN_TYPE_DIGITAL
|
|
firmata_pin = pin
|
|
return (pin_type, firmata_pin)
|
|
|
|
|
|
async def get_board(data: Mapping) -> PymataExpress:
|
|
"""Create a Pymata board object."""
|
|
board_data = {}
|
|
|
|
if CONF_SERIAL_PORT in data:
|
|
board_data["com_port"] = data[CONF_SERIAL_PORT]
|
|
if CONF_SERIAL_BAUD_RATE in data:
|
|
board_data["baud_rate"] = data[CONF_SERIAL_BAUD_RATE]
|
|
if CONF_ARDUINO_INSTANCE_ID in data:
|
|
board_data["arduino_instance_id"] = data[CONF_ARDUINO_INSTANCE_ID]
|
|
|
|
if CONF_ARDUINO_WAIT in data:
|
|
board_data["arduino_wait"] = data[CONF_ARDUINO_WAIT]
|
|
if CONF_SLEEP_TUNE in data:
|
|
board_data["sleep_tune"] = data[CONF_SLEEP_TUNE]
|
|
|
|
board_data["autostart"] = False
|
|
board_data["shutdown_on_exception"] = True
|
|
board_data["close_loop_on_shutdown"] = False
|
|
|
|
board = PymataExpress(**board_data)
|
|
|
|
await board.start_aio()
|
|
return board
|