-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathiem3155-8266_modbus_esphome.yaml
493 lines (439 loc) · 14.9 KB
/
iem3155-8266_modbus_esphome.yaml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
# ***************************************
# * ESPHome Modbus config for *
# * Schneider iEM3155 Energy Meter *
# * 2023-09-08 *
# * v1.2 - Various bug fixes: *
# * - Initial register values fix *
# * - daily_export_base from RTC *
# * memory on reboots (not PC) *
# * - Various minor fixes *
# ***************************************
#
# My iEM3155 config:
#
# Serial: 19200, EVEN, 1 (default)
# Slave address: 0x01
# Ensure that Com.Protection in the device is DISABLED !!
# Com.Protected is ENABLED by default !!
# Otherwise partial reset of daily energy import registers will be ignored
# by the device (id: daily_energy_import_total + 3 x daily_energy_import_lX)
#
# The devide also offers some additional interesting features that
# might be useful for others. Tariff/tariff rates and overload alarm.
# There's also both a digital input and output port that can be utilized
# as well.
# Check the iEM3155 user manual/technical datasheet for further info:
# https://download.schneider-electric.com/files?p_Doc_Ref=DOCA0005EN&p_enDocType=User+guide&p_File_Name=DOCA0005EN-13.pdf
#############################################
# Substitutions section contains all the #
# user/device specific configurable details #
#############################################
substitutions:
devicename: iem3155-8266
friendly_name: iEM3155
device_description: Schneider iEM3155 Energy Meter
platform: ESP8266
board: d1_mini
wifi: !secret wifi_ssid_tp
password: !secret wifi_password_tp
ip: !secret static_ip_iem3155
subnet: 255.255.255.0
gateway: !secret gateway
ha_api_key: !secret api_key
ap_password: !secret wifi_ap_password
ota_password: !secret iem3155_ota_password
uart_tx: "5" #D1
uart_rx: "4" #D2
HA_solar_daily_yield_sensor: sensor.solivia_solar_daily_yield
###############################################
# Main YAML block #
# You should not need to edit below this line #
###############################################
esphome:
name: ${devicename}
friendly_name: ${friendly_name}
comment: ${device_description}
platform: ${platform}
board: ${board}
# If daily_export_base value has not been retained on reboot (e.g. device power cycle)
# then wait for 20 seconds after api connection and set value to current total export value
on_boot:
priority: -100
then:
- wait_until:
api.connected:
- delay: 20s
- lambda: |-
if (id(daily_export_base) == -1) {
id(daily_export_base) = id(total_energy_export_total).state;
}
# Fix to prevent initial extreme random register value (six figures) being reported as power_total value on reboots/power cycles
# Bug in modbus module or... ?
# Initial interval is set to 'never' for modbus controller id(iem3155_freq)
- lambda:
id(iem3155_freq).set_update_interval(10000);
id(iem3155_freq).call_setup();
globals:
# Enable variable restore from RTC memory (ESP8266 only)
# Daily export base variable will not survive a power cycle, only reboots
- id: daily_export_base
type: float
restore_value: yes
initial_value: '-1'
uart:
# TX gpio5 (D1) and RX gpio4 (D2)
id: mod_bus
tx_pin: ${uart_tx}
rx_pin: ${uart_rx}
baud_rate: 19200
parity: EVEN
stop_bits: 1
# Optional debug parameter if TX/TX UART hex data is needed for debugging
#debug:
# Enable logging
# Set baud_rate to 0 if you're using hw UART, in order to disable logging via UART pins
logger:
baud_rate: 0
# level: VERBOSE
# Enable Home Assistant API
#key: "c1lAMMlJXPg9+kFJbmc9T5FJ2N1A9K8OUhLmfzG/pww="
api:
encryption:
key: ${ha_api_key}
ota:
password: ${ota_password}
wifi:
ssid: ${wifi}
password: ${password}
fast_connect: true
manual_ip:
static_ip: ${ip}
gateway: ${gateway}
subnet: ${subnet}
# Enable fallback hotspot (captive portal) in case wifi connection fails
ap:
ssid: ${friendly_name} Fallback Hotspot
password: ${ap_password}
captive_portal:
#web_server:
# port: 80
time:
- platform: homeassistant
id: esptime
- platform: sntp
on_time:
# Reset all four daily import counters at midnight (minus 1 second)
# This will also force the command_result id to be updated
# Validation check could be performed on command_result sensor
# 0 = valid operation (reset all partial registers - ok)
# 3000 = invalid operation (reset all partial registers - nok) )
- seconds: 59
minutes: 59
hours: 23
days_of_week: MON-SUN
then:
- component.update: iem3155_daily
# Reset daily export base value at midnight
# The iEM3155 unfortunately doesn't provide a partial export value register
# Instead value is calculated using the total export register and export base value is reset at midnight
- seconds: 00
minutes: 00
hours: 00
days_of_week: MON-SUN
then:
# Set daily_yield sensor to zero at midnight and reset daily_export_base for next day
# Can't force an update from HA at 00:00:00 -> 'Old' daily_yield value could potentially result in updating daily_solar_consumption_total with wrong value
# Force update both template id sensors that are calculates values (not modbus sensors)
# Force update remaining modbus sensors (60 seconds interval)
- lambda: |-
id(daily_yield).publish_state(0);
id(daily_export_base) = id(total_energy_export_total).state;
id(daily_energy_export_total).update();
id(daily_solar_consumption_total).update();
- component.update: iem3155
modbus:
id: modbus1
modbus_controller:
- id: iem3155
address: 0x1
modbus_id: modbus1
# Update all, but Total Active, energy sensors every minute
update_interval: 60s
- id: iem3155_freq
address: 0x1
modbus_id: modbus1
# Update Total Active Energy sensor frequently (10 seconds interval)
# Initial update_interval set to 'never' to prevent invalid register values pushed to HA on device reboots/power cycles
# update_interval will be set to 10 sec.
update_interval: never
- id: iem3155_daily
address: 0x1
modbus_id: modbus1
# No interval sensor update !
# Only custom_command sensor is using this id (once daily at midnight)
update_interval: never
binary_sensor:
# Optional sensor - Device status sensor
- platform: status
name: "Node Status"
id: system_status
- platform: template
name: "Digital input control mode"
id: digital_input_bool
icon: mdi:toggle-switch-outline
# This binary sensor returns 'true' if device partial energy readings
# can be reset via digital input pins (12-40v dc pulse)
# Alternative to the modbus custom_command
lambda: |-
if (id(digital_input).state == 5) return true; else return false;
sensor:
# Optional sensors - Device uptime / Wifi strength db / WiFi strength in percent
- platform: uptime
name: "Uptime sensor"
- platform: wifi_signal
id: iem3155_wifi_strength_db
name: "WiFi strength db"
update_interval: 60s
entity_category: "diagnostic"
- platform: copy # Reports the WiFi signal strength in %
source_id: iem3155_wifi_strength_db
name: "WiFi strength pct."
filters:
- lambda: return min(max(2 * (x + 100.0), 0.0), 100.0);
unit_of_measurement: "%"
entity_category: "diagnostic"
# Home Assistant solar inverter daily yield sensor value
# Sensor only needed for optional calculated sensor (daily_solar_consumption_total)
- platform: homeassistant
id: daily_yield
entity_id: ${HA_solar_daily_yield_sensor}
internal: true
filters:
- multiply: 0.001
- platform: template
id: daily_energy_export_total
name: "Daily Energy Export Total"
icon: mdi:transmission-tower-import
device_class: energy
unit_of_measurement: "kWh"
accuracy_decimals: 3
# Check for initial export base variable value (-1)
# Do not calculate daily export value if initial value is set. Instead return 0
update_interval: 60s
lambda: |-
if (id(daily_export_base) == -1) {
return 0;
} else {
return (id(total_energy_export_total).state - id(daily_export_base));
}
# Optional daily own solar consumption sensor
# Value is calculated using solar_daily_yield sensor from HA
- platform: template
id: daily_solar_consumption_total
name: "Daily Solar Consumption Total"
icon: mdi:solar-power-variant
device_class: energy
unit_of_measurement: "kWh"
accuracy_decimals: 3
update_interval: 60s
lambda: |-
return (id(daily_yield).state - id(daily_energy_export_total).state);
# Sensors updated every 60 seconds
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Voltage avg. All Phases"
id: voltage_avg
register_type: holding
address: 3035
icon: mdi:current-ac
unit_of_measurement: "V"
value_type: FP32
accuracy_decimals: 0
- platform: modbus_controller
modbus_controller_id: iem3155
name: "L1 Active Power"
id: power_l1
register_type: holding
address: 3053
unit_of_measurement: "W"
device_class: energy
value_type: FP32
filters:
- multiply: 1000
accuracy_decimals: 0
- platform: modbus_controller
modbus_controller_id: iem3155
name: "L2 Active Power"
id: power_l2
register_type: holding
address: 3055
unit_of_measurement: "W"
device_class: energy
value_type: FP32
filters:
- multiply: 1000
accuracy_decimals: 0
- platform: modbus_controller
modbus_controller_id: iem3155
name: "L3 Active Power"
id: power_l3
register_type: holding
address: 3057
unit_of_measurement: "W"
device_class: energy
value_type: FP32
filters:
- multiply: 1000
accuracy_decimals: 0
# Sensor updated every 10 seconds
- platform: modbus_controller
modbus_controller_id: iem3155_freq
name: "Total Active Power"
id: power_total
register_type: holding
address: 3059
unit_of_measurement: "W"
device_class: energy
value_type: FP32
filters:
- multiply: 1000
accuracy_decimals: 0
# Sensors updated every 60 seconds
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Power Factor"
id: power_factor
register_type: holding
address: 3083
icon: mdi:power-settings
value_type: FP32
accuracy_decimals: 2
## Possible float values from: -2 to +1
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Frequency"
id: frequency
register_type: holding
address: 3109
icon: mdi:sine-wave
unit_of_measurement: "Hz"
value_type: FP32
accuracy_decimals: 2
- platform: modbus_controller
modbus_controller_id: iem3155
id: digital_input
# Don't expose register to HA. Binary digital_input_bool is exposed instead
internal: true
# This register returns 5 if device partial energy readings
# can be reset via digital input pins (12-40v dc pulse)
# Alternative to the custom_command
register_type: holding
address: 7273
value_type: U_WORD
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Energy Import Total"
id: total_energy_import_total
register_type: holding
address: 45099
icon: mdi:transmission-tower-export
unit_of_measurement: "kWh"
device_class: energy
# Set state_class in order for HA to use sensor in the Energy component
state_class: total_increasing
value_type: FP32
accuracy_decimals: 3
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Energy Export Total"
id: total_energy_export_total
register_type: holding
address: 45101
icon: mdi:transmission-tower-import
unit_of_measurement: "kWh"
device_class: energy
# Set state_class in order for HA to use sensor in the Energy component
state_class: total_increasing
value_type: FP32
accuracy_decimals: 3
# Partial energy register used here as daily energy import sensor
# Will be reset at midnight by custom_command
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Daily Energy Import Total"
id: daily_energy_import_total
register_type: holding
address: 45107
icon: mdi:lightning-bolt-circle
unit_of_measurement: "kWh"
device_class: energy
# Set state_class in order for HA to use sensor in the Energy component
state_class: total_increasing
value_type: FP32
accuracy_decimals: 3
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Daily Energy Import Phase 1"
id: daily_energy_import_l1
register_type: holding
address: 45111
icon: mdi:lightning-bolt-circle
unit_of_measurement: "kWh"
value_type: FP32
accuracy_decimals: 3
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Daily Energy Import Phase 2"
id: daily_energy_import_l2
register_type: holding
address: 45113
icon: mdi:lightning-bolt-circle
unit_of_measurement: "kWh"
value_type: FP32
accuracy_decimals: 3
- platform: modbus_controller
modbus_controller_id: iem3155
name: "Daily Energy Import Phase 3"
id: daily_energy_import_l3
register_type: holding
address: 45115
icon: mdi:lightning-bolt-circle
unit_of_measurement: "kWh"
value_type: FP32
accuracy_decimals: 3
# Sensors updated at midnight
- platform: modbus_controller
modbus_controller_id: iem3155_daily
name: "Command result"
id: command_result
# Command Result codes:
# 0 = Valid Operation
# 3000 = Invalid Command
# 3001 = Invalid Parameter
# 3002 = Invalid Number of Parameters
# 3007 = Operation Not Performed
# This register returns 0 if daily import registers has been
# reset correctly at midnight. Returns 3000 if device has
# Com.Protection enabled
# Sensor can be checked in HA to ensure partial counters
# has been reset correctly
register_type: holding
address: 5375
icon: mdi:restart
value_type: U_WORD
- platform: modbus_controller
modbus_controller_id: iem3155_daily
name: "Reset command"
id: reset_partial
internal: true
# custom_command to reset all partial energy readings just before midnight
# Sensor value can't be used as result value. That has to be checked via command_result sensor
# device slave address: 0x01
# modbus command 16: 0x10
# commmand register 5250 decimal: 0x1481
# number of registers to write: 0x0002 (write both command and command parameters registers)
# number of bytes: 0x04 (2 parameters x 2 bytes)
# command 2020 in register 5250: 0x07E4
# command parameters in register 5252: 0x000 (this command has no parameters)
custom_command: [ 0x01, 0x10, 0x14, 0x81, 0x00, 0x02, 0x04, 0x07, 0xE4, 0x00, 0x00 ]
value_type: FP32