"""Support for the Torque OBD application.""" from __future__ import annotations import re import voluptuous as vol from homeassistant.components.http import HomeAssistantView from homeassistant.components.sensor import PLATFORM_SCHEMA, SensorEntity from homeassistant.const import CONF_EMAIL, CONF_NAME, DEGREE from homeassistant.core import HomeAssistant, callback import homeassistant.helpers.config_validation as cv from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType API_PATH = "/api/torque" DEFAULT_NAME = "vehicle" DOMAIN = "torque" ENTITY_NAME_FORMAT = "{0} {1}" SENSOR_EMAIL_FIELD = "eml" SENSOR_NAME_KEY = r"userFullName(\w+)" SENSOR_UNIT_KEY = r"userUnit(\w+)" SENSOR_VALUE_KEY = r"k(\w+)" NAME_KEY = re.compile(SENSOR_NAME_KEY) UNIT_KEY = re.compile(SENSOR_UNIT_KEY) VALUE_KEY = re.compile(SENSOR_VALUE_KEY) PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend( { vol.Required(CONF_EMAIL): cv.string, vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string, } ) def convert_pid(value): """Convert pid from hex string to integer.""" return int(value, 16) def setup_platform( hass: HomeAssistant, config: ConfigType, add_entities: AddEntitiesCallback, discovery_info: DiscoveryInfoType | None = None, ) -> None: """Set up the Torque platform.""" vehicle = config.get(CONF_NAME) email = config.get(CONF_EMAIL) sensors: dict[int, TorqueSensor] = {} hass.http.register_view( TorqueReceiveDataView(email, vehicle, sensors, add_entities) ) class TorqueReceiveDataView(HomeAssistantView): """Handle data from Torque requests.""" url = API_PATH name = "api:torque" def __init__(self, email, vehicle, sensors, add_entities): """Initialize a Torque view.""" self.email = email self.vehicle = vehicle self.sensors = sensors self.add_entities = add_entities @callback def get(self, request): """Handle Torque data request.""" hass = request.app["hass"] data = request.query if self.email is not None and self.email != data[SENSOR_EMAIL_FIELD]: return names = {} units = {} for key in data: is_name = NAME_KEY.match(key) is_unit = UNIT_KEY.match(key) is_value = VALUE_KEY.match(key) if is_name: pid = convert_pid(is_name.group(1)) names[pid] = data[key] elif is_unit: pid = convert_pid(is_unit.group(1)) temp_unit = data[key] if "\\xC2\\xB0" in temp_unit: temp_unit = temp_unit.replace("\\xC2\\xB0", DEGREE) units[pid] = temp_unit elif is_value: pid = convert_pid(is_value.group(1)) if pid in self.sensors: self.sensors[pid].async_on_update(data[key]) for pid, name in names.items(): if pid not in self.sensors: self.sensors[pid] = TorqueSensor( ENTITY_NAME_FORMAT.format(self.vehicle, name), units.get(pid) ) hass.async_add_job(self.add_entities, [self.sensors[pid]]) return "OK!" class TorqueSensor(SensorEntity): """Representation of a Torque sensor.""" def __init__(self, name, unit): """Initialize the sensor.""" self._name = name self._unit = unit self._state = None @property def name(self): """Return the name of the sensor.""" return self._name @property def native_unit_of_measurement(self): """Return the unit of measurement.""" return self._unit @property def native_value(self): """Return the state of the sensor.""" return self._state @property def icon(self): """Return the default icon of the sensor.""" return "mdi:car" @callback def async_on_update(self, value): """Receive an update.""" self._state = value self.async_write_ha_state()