513 lines
15 KiB
Python
513 lines
15 KiB
Python
"""Read status of growatt inverters."""
|
|
import datetime
|
|
import json
|
|
import logging
|
|
import re
|
|
|
|
import growattServer
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA, SensorEntity
|
|
from homeassistant.const import (
|
|
CONF_NAME,
|
|
CONF_PASSWORD,
|
|
CONF_USERNAME,
|
|
CURRENCY_EURO,
|
|
DEVICE_CLASS_BATTERY,
|
|
DEVICE_CLASS_CURRENT,
|
|
DEVICE_CLASS_ENERGY,
|
|
DEVICE_CLASS_POWER,
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
DEVICE_CLASS_VOLTAGE,
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
ENERGY_KILO_WATT_HOUR,
|
|
FREQUENCY_HERTZ,
|
|
PERCENTAGE,
|
|
POWER_WATT,
|
|
TEMP_CELSIUS,
|
|
VOLT,
|
|
)
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.util import Throttle
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
CONF_PLANT_ID = "plant_id"
|
|
DEFAULT_PLANT_ID = "0"
|
|
DEFAULT_NAME = "Growatt"
|
|
SCAN_INTERVAL = datetime.timedelta(minutes=5)
|
|
|
|
# Sensor type order is: Sensor name, Unit of measurement, api data name, additional options
|
|
|
|
TOTAL_SENSOR_TYPES = {
|
|
"total_money_today": ("Total money today", CURRENCY_EURO, "plantMoneyText", {}),
|
|
"total_money_total": ("Money lifetime", CURRENCY_EURO, "totalMoneyText", {}),
|
|
"total_energy_today": (
|
|
"Energy Today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"todayEnergy",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"total_output_power": (
|
|
"Output Power",
|
|
POWER_WATT,
|
|
"invTodayPpv",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
"total_energy_output": (
|
|
"Lifetime energy output",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"totalEnergy",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"total_maximum_output": (
|
|
"Maximum power",
|
|
POWER_WATT,
|
|
"nominalPower",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
}
|
|
|
|
INVERTER_SENSOR_TYPES = {
|
|
"inverter_energy_today": (
|
|
"Energy today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"powerToday",
|
|
{"round": 1, "device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"inverter_energy_total": (
|
|
"Lifetime energy output",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"powerTotal",
|
|
{"round": 1, "device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"inverter_voltage_input_1": (
|
|
"Input 1 voltage",
|
|
VOLT,
|
|
"vpv1",
|
|
{"round": 2, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"inverter_amperage_input_1": (
|
|
"Input 1 Amperage",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"ipv1",
|
|
{"round": 1, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"inverter_wattage_input_1": (
|
|
"Input 1 Wattage",
|
|
POWER_WATT,
|
|
"ppv1",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_voltage_input_2": (
|
|
"Input 2 voltage",
|
|
VOLT,
|
|
"vpv2",
|
|
{"round": 1, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"inverter_amperage_input_2": (
|
|
"Input 2 Amperage",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"ipv2",
|
|
{"round": 1, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"inverter_wattage_input_2": (
|
|
"Input 2 Wattage",
|
|
POWER_WATT,
|
|
"ppv2",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_voltage_input_3": (
|
|
"Input 3 voltage",
|
|
VOLT,
|
|
"vpv3",
|
|
{"round": 1, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"inverter_amperage_input_3": (
|
|
"Input 3 Amperage",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"ipv3",
|
|
{"round": 1, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"inverter_wattage_input_3": (
|
|
"Input 3 Wattage",
|
|
POWER_WATT,
|
|
"ppv3",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_internal_wattage": (
|
|
"Internal wattage",
|
|
POWER_WATT,
|
|
"ppv",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_reactive_voltage": (
|
|
"Reactive voltage",
|
|
VOLT,
|
|
"vacr",
|
|
{"round": 1, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"inverter_inverter_reactive_amperage": (
|
|
"Reactive amperage",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"iacr",
|
|
{"round": 1, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"inverter_frequency": ("AC frequency", FREQUENCY_HERTZ, "fac", {"round": 1}),
|
|
"inverter_current_wattage": (
|
|
"Output power",
|
|
POWER_WATT,
|
|
"pac",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_current_reactive_wattage": (
|
|
"Reactive wattage",
|
|
POWER_WATT,
|
|
"pacr",
|
|
{"device_class": DEVICE_CLASS_POWER, "round": 1},
|
|
),
|
|
"inverter_ipm_temperature": (
|
|
"Intelligent Power Management temperature",
|
|
TEMP_CELSIUS,
|
|
"ipmTemperature",
|
|
{"device_class": DEVICE_CLASS_TEMPERATURE, "round": 1},
|
|
),
|
|
"inverter_temperature": (
|
|
"Temperature",
|
|
TEMP_CELSIUS,
|
|
"temperature",
|
|
{"device_class": DEVICE_CLASS_TEMPERATURE, "round": 1},
|
|
),
|
|
}
|
|
|
|
STORAGE_SENSOR_TYPES = {
|
|
"storage_storage_production_today": (
|
|
"Storage production today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eBatDisChargeToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_storage_production_lifetime": (
|
|
"Lifetime Storage production",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eBatDisChargeTotal",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_grid_discharge_today": (
|
|
"Grid discharged today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eacDisChargeToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_load_consumption_today": (
|
|
"Load consumption today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eopDischrToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_load_consumption_lifetime": (
|
|
"Lifetime load consumption",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eopDischrTotal",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_grid_charged_today": (
|
|
"Grid charged today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eacChargeToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_charge_storage_lifetime": (
|
|
"Lifetime storaged charged",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eChargeTotal",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_solar_production": (
|
|
"Solar power production",
|
|
POWER_WATT,
|
|
"ppv",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
"storage_battery_percentage": (
|
|
"Battery percentage",
|
|
PERCENTAGE,
|
|
"capacity",
|
|
{"device_class": DEVICE_CLASS_BATTERY},
|
|
),
|
|
"storage_power_flow": (
|
|
"Storage charging/ discharging(-ve)",
|
|
POWER_WATT,
|
|
"pCharge",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
"storage_load_consumption_solar_storage": (
|
|
"Load consumption(Solar + Storage)",
|
|
"VA",
|
|
"rateVA",
|
|
{},
|
|
),
|
|
"storage_charge_today": (
|
|
"Charge today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eChargeToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_import_from_grid": (
|
|
"Import from grid",
|
|
POWER_WATT,
|
|
"pAcInPut",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
"storage_import_from_grid_today": (
|
|
"Import from grid today",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eToUserToday",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_import_from_grid_total": (
|
|
"Import from grid total",
|
|
ENERGY_KILO_WATT_HOUR,
|
|
"eToUserTotal",
|
|
{"device_class": DEVICE_CLASS_ENERGY},
|
|
),
|
|
"storage_load_consumption": (
|
|
"Load consumption",
|
|
POWER_WATT,
|
|
"outPutPower",
|
|
{"device_class": DEVICE_CLASS_POWER},
|
|
),
|
|
"storage_grid_voltage": (
|
|
"AC input voltage",
|
|
VOLT,
|
|
"vGrid",
|
|
{"round": 2, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"storage_pv_charging_voltage": (
|
|
"PV charging voltage",
|
|
VOLT,
|
|
"vpv",
|
|
{"round": 2, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"storage_ac_input_frequency_out": (
|
|
"AC input frequency",
|
|
FREQUENCY_HERTZ,
|
|
"freqOutPut",
|
|
{"round": 2},
|
|
),
|
|
"storage_output_voltage": (
|
|
"Output voltage",
|
|
VOLT,
|
|
"outPutVolt",
|
|
{"round": 2, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"storage_ac_output_frequency": (
|
|
"Ac output frequency",
|
|
FREQUENCY_HERTZ,
|
|
"freqGrid",
|
|
{"round": 2},
|
|
),
|
|
"storage_current_PV": (
|
|
"Solar charge current",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"iAcCharge",
|
|
{"round": 2, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"storage_current_1": (
|
|
"Solar current to storage",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"iChargePV1",
|
|
{"round": 2, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"storage_grid_amperage_input": (
|
|
"Grid charge current",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"chgCurr",
|
|
{"round": 2, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"storage_grid_out_current": (
|
|
"Grid out current",
|
|
ELECTRICAL_CURRENT_AMPERE,
|
|
"outPutCurrent",
|
|
{"round": 2, "device_class": DEVICE_CLASS_CURRENT},
|
|
),
|
|
"storage_battery_voltage": (
|
|
"Battery voltage",
|
|
VOLT,
|
|
"vBat",
|
|
{"round": 2, "device_class": DEVICE_CLASS_VOLTAGE},
|
|
),
|
|
"storage_load_percentage": (
|
|
"Load percentage",
|
|
PERCENTAGE,
|
|
"loadPercent",
|
|
{"device_class": DEVICE_CLASS_BATTERY, "round": 2},
|
|
),
|
|
}
|
|
|
|
SENSOR_TYPES = {**TOTAL_SENSOR_TYPES, **INVERTER_SENSOR_TYPES, **STORAGE_SENSOR_TYPES}
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
vol.Optional(CONF_PLANT_ID, default=DEFAULT_PLANT_ID): cv.string,
|
|
vol.Required(CONF_USERNAME): cv.string,
|
|
vol.Required(CONF_PASSWORD): cv.string,
|
|
}
|
|
)
|
|
|
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
|
"""Set up the Growatt sensor."""
|
|
username = config[CONF_USERNAME]
|
|
password = config[CONF_PASSWORD]
|
|
plant_id = config[CONF_PLANT_ID]
|
|
name = config[CONF_NAME]
|
|
|
|
api = growattServer.GrowattApi()
|
|
|
|
# Log in to api and fetch first plant if no plant id is defined.
|
|
login_response = api.login(username, password)
|
|
if not login_response["success"] and login_response["errCode"] == "102":
|
|
_LOGGER.error("Username or Password may be incorrect!")
|
|
return
|
|
user_id = login_response["userId"]
|
|
if plant_id == DEFAULT_PLANT_ID:
|
|
plant_info = api.plant_list(user_id)
|
|
plant_id = plant_info["data"][0]["plantId"]
|
|
|
|
# Get a list of devices for specified plant to add sensors for.
|
|
devices = api.device_list(plant_id)
|
|
entities = []
|
|
probe = GrowattData(api, username, password, plant_id, "total")
|
|
for sensor in TOTAL_SENSOR_TYPES:
|
|
entities.append(
|
|
GrowattInverter(probe, f"{name} Total", sensor, f"{plant_id}-{sensor}")
|
|
)
|
|
|
|
# Add sensors for each device in the specified plant.
|
|
for device in devices:
|
|
probe = GrowattData(
|
|
api, username, password, device["deviceSn"], device["deviceType"]
|
|
)
|
|
sensors = []
|
|
if device["deviceType"] == "inverter":
|
|
sensors = INVERTER_SENSOR_TYPES
|
|
elif device["deviceType"] == "storage":
|
|
probe.plant_id = plant_id
|
|
sensors = STORAGE_SENSOR_TYPES
|
|
else:
|
|
_LOGGER.debug(
|
|
"Device type %s was found but is not supported right now",
|
|
device["deviceType"],
|
|
)
|
|
|
|
for sensor in sensors:
|
|
entities.append(
|
|
GrowattInverter(
|
|
probe,
|
|
f"{device['deviceAilas']}",
|
|
sensor,
|
|
f"{device['deviceSn']}-{sensor}",
|
|
)
|
|
)
|
|
|
|
add_entities(entities, True)
|
|
|
|
|
|
class GrowattInverter(SensorEntity):
|
|
"""Representation of a Growatt Sensor."""
|
|
|
|
def __init__(self, probe, name, sensor, unique_id):
|
|
"""Initialize a PVOutput sensor."""
|
|
self.sensor = sensor
|
|
self.probe = probe
|
|
self._name = name
|
|
self._state = None
|
|
self._unique_id = unique_id
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of the sensor."""
|
|
return f"{self._name} {SENSOR_TYPES[self.sensor][0]}"
|
|
|
|
@property
|
|
def unique_id(self):
|
|
"""Return the unique id of the sensor."""
|
|
return self._unique_id
|
|
|
|
@property
|
|
def icon(self):
|
|
"""Return the icon of the sensor."""
|
|
return "mdi:solar-power"
|
|
|
|
@property
|
|
def state(self):
|
|
"""Return the state of the sensor."""
|
|
result = self.probe.get_data(SENSOR_TYPES[self.sensor][2])
|
|
round_to = SENSOR_TYPES[self.sensor][3].get("round")
|
|
if round_to is not None:
|
|
result = round(result, round_to)
|
|
return result
|
|
|
|
@property
|
|
def device_class(self):
|
|
"""Return the device class of the sensor."""
|
|
return SENSOR_TYPES[self.sensor][3].get("device_class")
|
|
|
|
@property
|
|
def unit_of_measurement(self):
|
|
"""Return the unit of measurement of this entity, if any."""
|
|
return SENSOR_TYPES[self.sensor][1]
|
|
|
|
def update(self):
|
|
"""Get the latest data from the Growat API and updates the state."""
|
|
self.probe.update()
|
|
|
|
|
|
class GrowattData:
|
|
"""The class for handling data retrieval."""
|
|
|
|
def __init__(self, api, username, password, device_id, growatt_type):
|
|
"""Initialize the probe."""
|
|
|
|
self.growatt_type = growatt_type
|
|
self.api = api
|
|
self.device_id = device_id
|
|
self.plant_id = None
|
|
self.data = {}
|
|
self.username = username
|
|
self.password = password
|
|
|
|
@Throttle(SCAN_INTERVAL)
|
|
def update(self):
|
|
"""Update probe data."""
|
|
self.api.login(self.username, self.password)
|
|
_LOGGER.debug("Updating data for %s", self.device_id)
|
|
try:
|
|
if self.growatt_type == "total":
|
|
total_info = self.api.plant_info(self.device_id)
|
|
del total_info["deviceList"]
|
|
# PlantMoneyText comes in as "3.1/€" remove anything that isn't part of the number
|
|
total_info["plantMoneyText"] = re.sub(
|
|
r"[^\d.,]", "", total_info["plantMoneyText"]
|
|
)
|
|
self.data = total_info
|
|
elif self.growatt_type == "inverter":
|
|
inverter_info = self.api.inverter_detail(self.device_id)
|
|
self.data = inverter_info
|
|
elif self.growatt_type == "storage":
|
|
storage_info_detail = self.api.storage_params(self.device_id)[
|
|
"storageDetailBean"
|
|
]
|
|
storage_energy_overview = self.api.storage_energy_overview(
|
|
self.plant_id, self.device_id
|
|
)
|
|
self.data = {**storage_info_detail, **storage_energy_overview}
|
|
except json.decoder.JSONDecodeError:
|
|
_LOGGER.error("Unable to fetch data from Growatt server")
|
|
|
|
def get_data(self, variable):
|
|
"""Get the data."""
|
|
return self.data.get(variable)
|