mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-30 22:53:59 +00:00 
			
		
		
		
	Add growatt modbus sensor (#2922)
Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com>
This commit is contained in:
		| @@ -65,6 +65,7 @@ esphome/components/globals/* @esphome/core | ||||
| esphome/components/gpio/* @esphome/core | ||||
| esphome/components/gps/* @coogle | ||||
| esphome/components/graph/* @synco | ||||
| esphome/components/growatt_solar/* @leeuwte | ||||
| esphome/components/havells_solar/* @sourabhjaiswal | ||||
| esphome/components/hbridge/fan/* @WeekendWarrior | ||||
| 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)) | ||||
		Reference in New Issue
	
	Block a user