-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathse-modbus-tcp-read.py
executable file
·115 lines (96 loc) · 4.26 KB
/
se-modbus-tcp-read.py
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
#!/usr/bin/env python3
import datetime
import json
import socket
import struct
import sys
import time
SUNSPEC_IDENTIFIER = 0x53756e53 # "SunS"
SUNSPEC_END_MODEL = 0xffff
host, port = sys.argv[1].split(':')
socket.setdefaulttimeout(10)
s = socket.socket()
s.connect((host, int(port)))
def modbus_read_holding(s, base, count):
assert base >= 40001
s.send(struct.pack('>HHHBBHH', 1, 0, 6, 0, 3, base - 40001, count))
r = s.recv(1024)
txn, proto, length, unit, function, n = struct.unpack('>HHHBBB', r[:9])
assert txn == 1
assert proto == 0
assert unit == 0
assert function == 3
assert length == n + 3
return r[9:9 + n]
def sunspec_enumerate_models(s):
cursor = 40001
suns_identifier = modbus_read_holding(s, cursor, 2)
assert struct.unpack('>L', suns_identifier) == (SUNSPEC_IDENTIFIER,)
cursor += 2
# Enumerate the models.
models = []
while True:
model_id, = struct.unpack('>H', modbus_read_holding(s, cursor, 1))
if model_id == SUNSPEC_END_MODEL:
# I'm not convinced the End Model will always have a length.
break
model_header = modbus_read_holding(s, cursor, 2)
model_id, model_l = struct.unpack('>HH', model_header)
models.append((model_id, cursor, model_l))
cursor += 2 + model_l
return models
models = sunspec_enumerate_models(s)
common_model, = [model for model in models if model[0] == 1]
common_model_data = modbus_read_holding(
s, common_model[1], common_model[2] + 2)
(C_SunSpec_DID, C_SunSpec_Length, C_Manufacturer, C_Model,
_, C_Version, C_SerialNumber, C_DeviceAddress) = struct.unpack(
'>HH32s32s16s16s32sH', common_model_data)
C_Manufacturer = C_Manufacturer.partition(b'\0')[0].decode('UTF-8')
C_Model = C_Model.partition(b'\0')[0].decode('UTF-8')
C_Version = C_Version.partition(b'\0')[0].decode('UTF-8')
C_SerialNumber = C_SerialNumber.partition(b'\0')[0].decode('UTF-8')
inverter_model, = [model for model in models if model[0] == 101]
while True:
res = modbus_read_holding(s, inverter_model[1], inverter_model[2] + 2)
res = res[4:]
(I_AC_Current, I_AC_CurrentA, I_AC_CurrentB, I_AC_CurrentC,
I_AC_Current_SF, I_AC_VoltageAB, I_AC_VoltageBC, I_AC_VoltageCA,
I_AC_VoltageAN, I_AC_VoltageBN, I_AC_VoltageCN, I_AC_Voltage_SF,
I_AC_Power, I_AC_Power_SF, I_AC_Frequency, I_AC_Frequency_SF,
I_AC_VA, I_AC_VA_SF, I_AC_VAR, I_AC_VAR_SF, I_AC_PF, I_AC_PF_SF,
I_AC_Energy_WH, I_AC_Energy_WH_SF, I_DC_Current, I_DC_Current_SF,
I_DC_Voltage, I_DC_Voltage_SF, I_DC_Power, I_DC_Power_SF, _,
I_Temp_Sink, _, _, I_Temp_SF, I_Status, I_Status_Vendor, _, _,
I_Event_1_Vendor, _, _, I_Event_4_Vendor) = struct.unpack(
'>HHHHhHHHHHHhhhHhhhhhhhLHHhHhhhhhhhhHHLLLLLL', res)
d = {
"C_Manufacturer": C_Manufacturer, "C_Model": C_Model,
"C_Version": C_Version, "C_SerialNumber": C_SerialNumber,
"C_DeviceAddress": C_DeviceAddress,
"I_AC_Current": I_AC_Current, "I_AC_CurrentA": I_AC_CurrentA,
"I_AC_CurrentB": I_AC_CurrentB, "I_AC_CurrentC": I_AC_CurrentC,
"I_AC_Current_SF": I_AC_Current_SF,
"I_AC_VoltageAB": I_AC_VoltageAB, "I_AC_VoltageBC": I_AC_VoltageBC,
"I_AC_VoltageCA": I_AC_VoltageCA, "I_AC_VoltageAN": I_AC_VoltageAN,
"I_AC_VoltageBN": I_AC_VoltageBN, "I_AC_VoltageCN": I_AC_VoltageCN,
"I_AC_Voltage_SF": I_AC_Voltage_SF,
"I_AC_Power": I_AC_Power, "I_AC_Power_SF": I_AC_Power_SF,
"I_AC_Frequency": I_AC_Frequency,
"I_AC_Frequency_SF": I_AC_Frequency_SF,
"I_AC_VA": I_AC_VA, "I_AC_VA_SF": I_AC_VA_SF,
"I_AC_VAR": I_AC_VAR, "I_AC_VAR_SF": I_AC_VAR_SF,
"I_AC_PF": I_AC_PF, "I_AC_PF_SF": I_AC_PF_SF,
"I_AC_Energy_WH": I_AC_Energy_WH,
"I_AC_Energy_WH_SF": I_AC_Energy_WH_SF,
"I_DC_Current": I_DC_Current, "I_DC_Current_SF": I_DC_Current_SF,
"I_DC_Voltage": I_DC_Voltage, "I_DC_Voltage_SF": I_DC_Voltage_SF,
"I_Temp_Sink": I_Temp_Sink, "I_Temp_SF": I_Temp_SF,
"I_Status": I_Status, "I_Status_Vendor": I_Status_Vendor,
"I_Event_1_Vendor": I_Event_1_Vendor,
"I_Event_4_Vendor": I_Event_4_Vendor,
}
d['timestamp'] = datetime.datetime.utcnow().isoformat()
print(json.dumps(d))
sys.stdout.flush()
time.sleep(10)