mirror of
https://github.com/esphome/esphome.git
synced 2024-11-22 06:58:11 +01:00
Add growatt modbus sensor (#2922)
Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com>
This commit is contained in:
parent
1fb0a7109d
commit
66e0ff8392
5 changed files with 344 additions and 0 deletions
|
@ -65,6 +65,7 @@ esphome/components/globals/* @esphome/core
|
||||||
esphome/components/gpio/* @esphome/core
|
esphome/components/gpio/* @esphome/core
|
||||||
esphome/components/gps/* @coogle
|
esphome/components/gps/* @coogle
|
||||||
esphome/components/graph/* @synco
|
esphome/components/graph/* @synco
|
||||||
|
esphome/components/growatt_solar/* @leeuwte
|
||||||
esphome/components/havells_solar/* @sourabhjaiswal
|
esphome/components/havells_solar/* @sourabhjaiswal
|
||||||
esphome/components/hbridge/fan/* @WeekendWarrior
|
esphome/components/hbridge/fan/* @WeekendWarrior
|
||||||
esphome/components/hbridge/light/* @DotNetDann
|
esphome/components/hbridge/light/* @DotNetDann
|
||||||
|
|
0
esphome/components/growatt_solar/__init__.py
Normal file
0
esphome/components/growatt_solar/__init__.py
Normal file
69
esphome/components/growatt_solar/growatt_solar.cpp
Normal file
69
esphome/components/growatt_solar/growatt_solar.cpp
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
#include "growatt_solar.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace growatt_solar {
|
||||||
|
|
||||||
|
static const char *const TAG = "growatt_solar";
|
||||||
|
|
||||||
|
static const uint8_t MODBUS_CMD_READ_IN_REGISTERS = 0x04;
|
||||||
|
static const uint8_t MODBUS_REGISTER_COUNT = 33;
|
||||||
|
|
||||||
|
void GrowattSolar::update() { this->send(MODBUS_CMD_READ_IN_REGISTERS, 0, MODBUS_REGISTER_COUNT); }
|
||||||
|
|
||||||
|
void GrowattSolar::on_modbus_data(const std::vector<uint8_t> &data) {
|
||||||
|
auto publish_1_reg_sensor_state = [&](sensor::Sensor *sensor, size_t i, float unit) -> void {
|
||||||
|
if (sensor == nullptr)
|
||||||
|
return;
|
||||||
|
float value = encode_uint16(data[i * 2], data[i * 2 + 1]) * unit;
|
||||||
|
sensor->publish_state(value);
|
||||||
|
};
|
||||||
|
|
||||||
|
auto publish_2_reg_sensor_state = [&](sensor::Sensor *sensor, size_t reg1, size_t reg2, float unit) -> void {
|
||||||
|
float value = ((encode_uint16(data[reg1 * 2], data[reg1 * 2 + 1]) << 16) +
|
||||||
|
encode_uint16(data[reg2 * 2], data[reg2 * 2 + 1])) *
|
||||||
|
unit;
|
||||||
|
if (sensor != nullptr)
|
||||||
|
sensor->publish_state(value);
|
||||||
|
};
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->inverter_status_, 0, 1);
|
||||||
|
|
||||||
|
publish_2_reg_sensor_state(this->pv_active_power_sensor_, 1, 2, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->pvs_[0].voltage_sensor_, 3, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->pvs_[0].current_sensor_, 4, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->pvs_[0].active_power_sensor_, 5, 6, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->pvs_[1].voltage_sensor_, 7, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->pvs_[1].current_sensor_, 8, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->pvs_[1].active_power_sensor_, 9, 10, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_2_reg_sensor_state(this->grid_active_power_sensor_, 11, 12, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->grid_frequency_sensor_, 13, TWO_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->phases_[0].voltage_sensor_, 14, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->phases_[0].current_sensor_, 15, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->phases_[0].active_power_sensor_, 16, 17, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->phases_[1].voltage_sensor_, 18, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->phases_[1].current_sensor_, 19, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->phases_[1].active_power_sensor_, 20, 21, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->phases_[2].voltage_sensor_, 22, ONE_DEC_UNIT);
|
||||||
|
publish_1_reg_sensor_state(this->phases_[2].current_sensor_, 23, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->phases_[2].active_power_sensor_, 24, 25, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_2_reg_sensor_state(this->today_production_, 26, 27, ONE_DEC_UNIT);
|
||||||
|
publish_2_reg_sensor_state(this->total_energy_production_, 28, 29, ONE_DEC_UNIT);
|
||||||
|
|
||||||
|
publish_1_reg_sensor_state(this->inverter_module_temp_, 32, ONE_DEC_UNIT);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GrowattSolar::dump_config() {
|
||||||
|
ESP_LOGCONFIG(TAG, "GROWATT Solar:");
|
||||||
|
ESP_LOGCONFIG(TAG, " Address: 0x%02X", this->address_);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace growatt_solar
|
||||||
|
} // namespace esphome
|
73
esphome/components/growatt_solar/growatt_solar.h
Normal file
73
esphome/components/growatt_solar/growatt_solar.h
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/components/modbus/modbus.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace growatt_solar {
|
||||||
|
|
||||||
|
static const float TWO_DEC_UNIT = 0.01;
|
||||||
|
static const float ONE_DEC_UNIT = 0.1;
|
||||||
|
|
||||||
|
class GrowattSolar : public PollingComponent, public modbus::ModbusDevice {
|
||||||
|
public:
|
||||||
|
void update() override;
|
||||||
|
void on_modbus_data(const std::vector<uint8_t> &data) override;
|
||||||
|
void dump_config() override;
|
||||||
|
|
||||||
|
void set_inverter_status_sensor(sensor::Sensor *sensor) { this->inverter_status_ = sensor; }
|
||||||
|
|
||||||
|
void set_grid_frequency_sensor(sensor::Sensor *sensor) { this->grid_frequency_sensor_ = sensor; }
|
||||||
|
void set_grid_active_power_sensor(sensor::Sensor *sensor) { this->grid_active_power_sensor_ = sensor; }
|
||||||
|
void set_pv_active_power_sensor(sensor::Sensor *sensor) { this->pv_active_power_sensor_ = sensor; }
|
||||||
|
|
||||||
|
void set_today_production_sensor(sensor::Sensor *sensor) { this->today_production_ = sensor; }
|
||||||
|
void set_total_energy_production_sensor(sensor::Sensor *sensor) { this->total_energy_production_ = sensor; }
|
||||||
|
void set_inverter_module_temp_sensor(sensor::Sensor *sensor) { this->inverter_module_temp_ = sensor; }
|
||||||
|
|
||||||
|
void set_voltage_sensor(uint8_t phase, sensor::Sensor *voltage_sensor) {
|
||||||
|
this->phases_[phase].voltage_sensor_ = voltage_sensor;
|
||||||
|
}
|
||||||
|
void set_current_sensor(uint8_t phase, sensor::Sensor *current_sensor) {
|
||||||
|
this->phases_[phase].current_sensor_ = current_sensor;
|
||||||
|
}
|
||||||
|
void set_active_power_sensor(uint8_t phase, sensor::Sensor *active_power_sensor) {
|
||||||
|
this->phases_[phase].active_power_sensor_ = active_power_sensor;
|
||||||
|
}
|
||||||
|
void set_voltage_sensor_pv(uint8_t pv, sensor::Sensor *voltage_sensor) {
|
||||||
|
this->pvs_[pv].voltage_sensor_ = voltage_sensor;
|
||||||
|
}
|
||||||
|
void set_current_sensor_pv(uint8_t pv, sensor::Sensor *current_sensor) {
|
||||||
|
this->pvs_[pv].current_sensor_ = current_sensor;
|
||||||
|
}
|
||||||
|
void set_active_power_sensor_pv(uint8_t pv, sensor::Sensor *active_power_sensor) {
|
||||||
|
this->pvs_[pv].active_power_sensor_ = active_power_sensor;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
struct GrowattPhase {
|
||||||
|
sensor::Sensor *voltage_sensor_{nullptr};
|
||||||
|
sensor::Sensor *current_sensor_{nullptr};
|
||||||
|
sensor::Sensor *active_power_sensor_{nullptr};
|
||||||
|
} phases_[3];
|
||||||
|
struct GrowattPV {
|
||||||
|
sensor::Sensor *voltage_sensor_{nullptr};
|
||||||
|
sensor::Sensor *current_sensor_{nullptr};
|
||||||
|
sensor::Sensor *active_power_sensor_{nullptr};
|
||||||
|
} pvs_[2];
|
||||||
|
|
||||||
|
sensor::Sensor *inverter_status_{nullptr};
|
||||||
|
|
||||||
|
sensor::Sensor *grid_frequency_sensor_{nullptr};
|
||||||
|
sensor::Sensor *grid_active_power_sensor_{nullptr};
|
||||||
|
|
||||||
|
sensor::Sensor *pv_active_power_sensor_{nullptr};
|
||||||
|
|
||||||
|
sensor::Sensor *today_production_{nullptr};
|
||||||
|
sensor::Sensor *total_energy_production_{nullptr};
|
||||||
|
sensor::Sensor *inverter_module_temp_{nullptr};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace growatt_solar
|
||||||
|
} // namespace esphome
|
201
esphome/components/growatt_solar/sensor.py
Normal file
201
esphome/components/growatt_solar/sensor.py
Normal file
|
@ -0,0 +1,201 @@
|
||||||
|
import esphome.codegen as cg
|
||||||
|
import esphome.config_validation as cv
|
||||||
|
from esphome.components import sensor, modbus
|
||||||
|
from esphome.const import (
|
||||||
|
CONF_ACTIVE_POWER,
|
||||||
|
CONF_CURRENT,
|
||||||
|
CONF_FREQUENCY,
|
||||||
|
CONF_ID,
|
||||||
|
CONF_VOLTAGE,
|
||||||
|
DEVICE_CLASS_CURRENT,
|
||||||
|
DEVICE_CLASS_ENERGY,
|
||||||
|
DEVICE_CLASS_POWER,
|
||||||
|
DEVICE_CLASS_VOLTAGE,
|
||||||
|
ICON_CURRENT_AC,
|
||||||
|
STATE_CLASS_MEASUREMENT,
|
||||||
|
STATE_CLASS_TOTAL_INCREASING,
|
||||||
|
UNIT_AMPERE,
|
||||||
|
UNIT_CELSIUS,
|
||||||
|
UNIT_HERTZ,
|
||||||
|
UNIT_VOLT,
|
||||||
|
UNIT_WATT,
|
||||||
|
)
|
||||||
|
|
||||||
|
CONF_PHASE_A = "phase_a"
|
||||||
|
CONF_PHASE_B = "phase_b"
|
||||||
|
CONF_PHASE_C = "phase_c"
|
||||||
|
|
||||||
|
CONF_ENERGY_PRODUCTION_DAY = "energy_production_day"
|
||||||
|
CONF_TOTAL_ENERGY_PRODUCTION = "total_energy_production"
|
||||||
|
CONF_TOTAL_GENERATION_TIME = "total_generation_time"
|
||||||
|
CONF_TODAY_GENERATION_TIME = "today_generation_time"
|
||||||
|
CONF_PV1 = "pv1"
|
||||||
|
CONF_PV2 = "pv2"
|
||||||
|
UNIT_KILOWATT_HOURS = "kWh"
|
||||||
|
UNIT_HOURS = "h"
|
||||||
|
UNIT_KOHM = "kΩ"
|
||||||
|
UNIT_MILLIAMPERE = "mA"
|
||||||
|
|
||||||
|
CONF_INVERTER_STATUS = "inverter_status"
|
||||||
|
CONF_PV_ACTIVE_POWER = "pv_active_power"
|
||||||
|
CONF_INVERTER_MODULE_TEMP = "inverter_module_temp"
|
||||||
|
|
||||||
|
|
||||||
|
AUTO_LOAD = ["modbus"]
|
||||||
|
CODEOWNERS = ["@leeuwte"]
|
||||||
|
|
||||||
|
growatt_solar_ns = cg.esphome_ns.namespace("growatt_solar")
|
||||||
|
GrowattSolar = growatt_solar_ns.class_(
|
||||||
|
"GrowattSolar", cg.PollingComponent, modbus.ModbusDevice
|
||||||
|
)
|
||||||
|
|
||||||
|
PHASE_SENSORS = {
|
||||||
|
CONF_VOLTAGE: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_VOLT,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_VOLTAGE,
|
||||||
|
),
|
||||||
|
CONF_CURRENT: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_AMPERE,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_CURRENT,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
CONF_ACTIVE_POWER: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_WATT,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_POWER,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
PV_SENSORS = {
|
||||||
|
CONF_VOLTAGE: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_VOLT,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_VOLTAGE,
|
||||||
|
),
|
||||||
|
CONF_CURRENT: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_AMPERE,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_CURRENT,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
CONF_ACTIVE_POWER: sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_WATT,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_POWER,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
PHASE_SCHEMA = cv.Schema(
|
||||||
|
{cv.Optional(sensor): schema for sensor, schema in PHASE_SENSORS.items()}
|
||||||
|
)
|
||||||
|
PV_SCHEMA = cv.Schema(
|
||||||
|
{cv.Optional(sensor): schema for sensor, schema in PV_SENSORS.items()}
|
||||||
|
)
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = (
|
||||||
|
cv.Schema(
|
||||||
|
{
|
||||||
|
cv.GenerateID(): cv.declare_id(GrowattSolar),
|
||||||
|
cv.Optional(CONF_PHASE_A): PHASE_SCHEMA,
|
||||||
|
cv.Optional(CONF_PHASE_B): PHASE_SCHEMA,
|
||||||
|
cv.Optional(CONF_PHASE_C): PHASE_SCHEMA,
|
||||||
|
cv.Optional(CONF_PV1): PV_SCHEMA,
|
||||||
|
cv.Optional(CONF_PV2): PV_SCHEMA,
|
||||||
|
cv.Optional(CONF_INVERTER_STATUS): sensor.sensor_schema(),
|
||||||
|
cv.Optional(CONF_FREQUENCY): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_HERTZ,
|
||||||
|
icon=ICON_CURRENT_AC,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_ACTIVE_POWER): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_WATT,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_POWER,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_PV_ACTIVE_POWER): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_WATT,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_POWER,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_ENERGY_PRODUCTION_DAY): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_KILOWATT_HOURS,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_ENERGY,
|
||||||
|
state_class=STATE_CLASS_TOTAL_INCREASING,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_TOTAL_ENERGY_PRODUCTION): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_KILOWATT_HOURS,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_ENERGY,
|
||||||
|
state_class=STATE_CLASS_TOTAL_INCREASING,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_INVERTER_MODULE_TEMP): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_CELSIUS,
|
||||||
|
accuracy_decimals=1,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.extend(cv.polling_component_schema("10s"))
|
||||||
|
.extend(modbus.modbus_device_schema(0x01))
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def to_code(config):
|
||||||
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
|
await cg.register_component(var, config)
|
||||||
|
await modbus.register_modbus_device(var, config)
|
||||||
|
|
||||||
|
if CONF_INVERTER_STATUS in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_INVERTER_STATUS])
|
||||||
|
cg.add(var.set_inverter_status_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_FREQUENCY in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_FREQUENCY])
|
||||||
|
cg.add(var.set_grid_frequency_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_ACTIVE_POWER in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_ACTIVE_POWER])
|
||||||
|
cg.add(var.set_grid_active_power_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_PV_ACTIVE_POWER in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_PV_ACTIVE_POWER])
|
||||||
|
cg.add(var.set_pv_active_power_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_ENERGY_PRODUCTION_DAY in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_ENERGY_PRODUCTION_DAY])
|
||||||
|
cg.add(var.set_today_production_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_TOTAL_ENERGY_PRODUCTION in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_TOTAL_ENERGY_PRODUCTION])
|
||||||
|
cg.add(var.set_total_energy_production_sensor(sens))
|
||||||
|
|
||||||
|
if CONF_INVERTER_MODULE_TEMP in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_INVERTER_MODULE_TEMP])
|
||||||
|
cg.add(var.set_inverter_module_temp_sensor(sens))
|
||||||
|
|
||||||
|
for i, phase in enumerate([CONF_PHASE_A, CONF_PHASE_B, CONF_PHASE_C]):
|
||||||
|
if phase not in config:
|
||||||
|
continue
|
||||||
|
|
||||||
|
phase_config = config[phase]
|
||||||
|
for sensor_type in PHASE_SENSORS:
|
||||||
|
if sensor_type in phase_config:
|
||||||
|
sens = await sensor.new_sensor(phase_config[sensor_type])
|
||||||
|
cg.add(getattr(var, f"set_{sensor_type}_sensor")(i, sens))
|
||||||
|
|
||||||
|
for i, pv in enumerate([CONF_PV1, CONF_PV2]):
|
||||||
|
if pv not in config:
|
||||||
|
continue
|
||||||
|
|
||||||
|
pv_config = config[pv]
|
||||||
|
for sensor_type in pv_config:
|
||||||
|
if sensor_type in pv_config:
|
||||||
|
sens = await sensor.new_sensor(pv_config[sensor_type])
|
||||||
|
cg.add(getattr(var, f"set_{sensor_type}_sensor_pv")(i, sens))
|
Loading…
Reference in a new issue