forked from MyHomeMyData/E3onCAN
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathE3onCANcollect.py
340 lines (297 loc) · 12.7 KB
/
E3onCANcollect.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
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
"""
Copyright 2023 MyHomeMyData
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
import can # https://pypi.org/project/python-can/
import argparse
import paho.mqtt.client as paho
import importlib
import datetime
import Open3Edatapoints
from Open3Edatapoints import *
import E3onCANdatapointsE380
from E3onCANdatapointsE380 import *
import E3onCANdatapointsE3100CB
from E3onCANdatapointsE3100CB import *
import Open3Ecodecs
from Open3Ecodecs import *
import E3onCANcodecs
from E3onCANcodecs import *
tsNextDecoding = {}
def decodeData(device, canid, ts, did, databytes):
def mqttdump(topic, obj, set_retain):
if (type(obj)==dict):
for k, itm in obj.items():
mqttdump(topic+'/'+str(k),itm, set_retain)
elif (type(obj)==list):
for k in range(len(obj)):
mqttdump(topic+'/'+str(k),obj[k], set_retain)
else:
ret = client_mqtt.publish(topic, str(obj), retain=set_retain)
didStr = str(did)
tsNow = int(datetime.datetime.now().timestamp()*1000)
if not didStr in tsNextDecoding:
tsNextDecoding[didStr] = 0
if tsGap == 0 or tsNow >= tsNextDecoding[didStr]:
tsNextDecoding[didStr] = tsNow + tsGap
if did in dataIdentifiers:
try:
topicPf = '' # clear topic Prefix
didNAME = dataIdentifiers[did].id
values = dataIdentifiers[did].decode(databytes)
except Exception as e:
# Exception while decoding
topicPf = "$" # set topic Prefix
values = {
"Error": "Exception: "+str(e),
"Did": did,
"Raw": databytes.hex()
}
else:
# No codec available for this did
topicPf = "$" # set topic Prefix
didNAME = "Unknown"
values = {
"Error": "No codec found for did",
"Did": did,
"Raw": databytes.hex()
}
if (args.mqtt != None):
topicStr = topicPf + mqttformatstring.format(
device = device,
didName = didNAME,
didNumber = did
)
set_retain = (retainall == True) or (did in retaindids)
if (args.json == True):
# Send one JSON message
ret = client_mqtt.publish(mqttParamas[2] + "/" + topicStr, json.dumps(values))
else:
# Split down to scalar types
mqttdump(mqttParamas[2] + "/" + topicStr, values, set_retain)
if (args.verbose == True):
print(str(did)+' '+didNAME+': '+json.dumps(values))
else:
if (args.verbose == True):
if ts > 0:
dt_str = str(datetime.datetime.fromtimestamp(ts))+' '
else:
dt_str = ''
print(dt_str+str(did)+' '+didNAME+': '+json.dumps(values))
else:
print(didNAME+': '+json.dumps(values))
def evalMessages(bus, device, args):
data = {
"len" : 0,
"timestamp" : 0,
"databytes" : bytearray(),
"did" : 0,
"collecting": False,
"D0expected": 0x21,
}
for msg in bus:
id = msg.arbitration_id
if args.dev == 'e380':
# e380 sends 8 bytes of data w/o any protocol
# use CAN id as did
did = id
if (dids == None) or (str(did) in dids):
decodeData(device,id,msg.timestamp,did,msg.data)
elif args.dev == 'e3100cb' and len(msg.data) == 8:
# e3100cb sends 8 bytes of data w/o any protocol
# did = CAN id plus databyte 3
D3str = '00'+str(msg.data[3])
D3str = D3str[-2:]
did = str(id)+'.'+D3str
if (dids == None) or (did in dids):
decodeData(device,id,msg.timestamp,did,msg.data[4:]) # Ignore bytes 0 to 3
else:
if data["collecting"]:
if msg.data[0] == data["D0expected"]:
# append next part of data
data["databytes"] += msg.data[1:]
data["D0expected"] += 1
if data["D0expected"] > 0x2f:
data["D0expected"] = 0x20
else:
# no more data
if ((dids == None) or (str(data["did"]) in dids)) and (len(data["databytes"]) >= data["len"]):
decodeData(device, id, data["timestamp"], data["did"],data["databytes"][0:data["len"]])
data["collecting"] = False
if not data["collecting"] and (msg.dlc > 4) and (msg.data[0] == 0x21) and (msg.data[3] in range(0xb0,0xc0)):
data["did"] = msg.data[1]+256*msg.data[2]
if data["did"] > 0 and data["did"] < 10000:
data["timestamp"] = msg.timestamp
D3 = msg.data[3]
if D3 in [0xb1,0xb2,0xb3,0xb4]:
# Single Frame B1,B2,B3,B4
data["len"] = D3-0xb0
data["databytes"] = msg.data[4:]
if ((dids == None) or (str(data["did"]) in dids)):
decodeData(device, id, data["timestamp"], data["did"],data["databytes"][0:data["len"]])
if D3 == 0xb0:
# Multi Frame B0
data["D0expected"] = msg.data[0]+1
if msg.data[4]==0xc1:
data["len"] = msg.data[5]
data["databytes"] = msg.data[6:]
else:
data["len"] = msg.data[4]
data["databytes"] = msg.data[5:]
data["collecting"] = True
if D3 in range(0xb5,0xc0):
# Multi Frame B5 .. BF
data["D0expected"] = msg.data[0]+1
data["len"] = D3-0xb0
data["databytes"] = msg.data[4:]
data["collecting"] = True
parser = argparse.ArgumentParser()
parser.add_argument("-c", "--can", type=str, help="use can device, e.g. can0")
parser.add_argument("-f", "--file", type=str, help="use candump as input, e.g. candump_vx3")
parser.add_argument("-dev", "--dev", type=str, help="device type --dev vcal or --dev vx3 or --dev vair or --dev vdens or --dev e380 or --dev e3100cb")
parser.add_argument("-canid", "--canid", type=str, help="CAN id to listen to --canid 0x451, overrides CAN id selected by device")
parser.add_argument("-r", "--read", type=str, help="read did, e.g. 0x173,0x174")
parser.add_argument("-raw", "--raw", action='store_true', help="return raw data for all dids")
parser.add_argument("-g", "--gap", type=str, help="minimum time gap (seconds) between decoding of specific dids. Default: immediate decoding.")
parser.add_argument("-m", "--mqtt", type=str, help="publish to server, e.g. 192.168.0.1:1883:topicname")
parser.add_argument("-mfstr", "--mqttformatstring", type=str, help="mqtt formatstring e.g. {didNumber}_{didName}")
parser.add_argument("-muser", "--mqttuser", type=str, help="mqtt username")
parser.add_argument("-mpass", "--mqttpass", type=str, help="mqtt password")
parser.add_argument("-j", "--json", action='store_true', help="send JSON structure")
parser.add_argument("-retain", "--retain", type=str, help="set retained flag for dids, e.g. 1834,1836")
parser.add_argument("-retainall", "--retainall", action='store_true', help="set retained flag for all dids")
parser.add_argument("-v", "--verbose", action='store_true', help="verbose info")
args = parser.parse_args()
Open3Ecodecs.flag_rawmode = args.raw
E3onCANcodecs.flag_rawmode = args.raw
if(args.dev != None):
device = args.dev
else:
device = 'vx3'
print('No device specified. Using '+device+' as default.')
Open3Ecodecs.flag_dev = device
if not device in ['vx3','vair','vcal','vdens','e380','e3100cb']:
print('Unknown device '+device+'. Aborting.')
exit(0)
if (device in ['e380','e3100cb']) and (args.canid != None):
print('Specification of CAN ids not allowed for energy meter. Aborting.')
exit(0)
if device == 'e380':
dataIdentifiers = dataIdentifiersE380
elif device == 'e3100cb':
dataIdentifiers = dataIdentifiersE3100CB
else:
# load datapoints for selected device
module_name = "Open3Edatapoints" + device.capitalize()
didmoduledev = importlib.import_module(module_name)
dataIdentifiersDev = didmoduledev.dataIdentifiers["dids"]
# load general datapoints table from Open3Edatapoints.py
dataIdentifiers = dataIdentifiers["dids"]
# overlay device dids over general table
lstpops = []
for itm in dataIdentifiers:
if not (itm in dataIdentifiersDev):
lstpops.append(itm)
elif not (dataIdentifiersDev[itm] is None): # None means 'no change', nothing special
dataIdentifiers[itm] = dataIdentifiersDev[itm]
# remove dids not existing with the device
for itm in lstpops:
dataIdentifiers.pop(itm)
# probably useless but to indicate that it's not required anymore
dataIdentifiersDev = None
didmoduledev = None
devCANid = {
"vcal" : list([0x693]),
"vx3" : list([0x451]),
"vair" : list([0x451]),
"vdens": list([0x451]),
"e380" : list(dataIdentifiersE380.keys()),
"e3100cb" : list([0x569])
}
if (args.canid != None):
try:
CANid = list([eval(args.canid)])
except:
CANid = devCANid[device]
print('WARNING: Could not evaluate given canid, using default value: '+json.dumps(CANid))
else:
CANid = devCANid[device]
if (args.read != None):
dids_str=args.read.split(",")
dids = []
for did in dids_str:
if (args.dev == 'e380') and (eval(did) in CANid):
dids.append(str(eval(did)))
elif args.dev == 'e3100cb':
dids.append(did)
else:
dids.append(str(eval(did)))
else:
dids = None
if (dids == []):
print('No valid dids specified. Aborting.')
exit(0)
if args.dev == 'e380':
filters = []
for id in CANid:
if dids == None or str(id) in dids:
filters.append({"can_id": id, "can_mask": 0x7FF, "extended": False})
elif args.dev == 'e3100cb':
filters = []
for id in CANid:
filters.append({"can_id": id, "can_mask": 0x7FF, "extended": False})
else:
filters = [{"can_id": CANid[0], "can_mask": 0x7FF, "extended": False}]
if(args.can != None):
channel = args.can
args.file = None # Avoid contradicting channels
elif (args.file != None):
channel = args.file
else:
channel = 'can0'
if args.gap != None:
tsGap = int(eval(args.gap) * 1000)
else:
tsGap = 0
retainall = args.retainall
retaindids = []
if (args.retain != None):
retaindids_str=args.retain.split(",")
for did in retaindids_str:
retaindids.append(eval(did))
client_mqtt = None
mqttParamas = None
mqttformatstring = "{didName}"
if(args.mqtt != None):
mqttParamas = args.mqtt.split(":")
if(args.mqttformatstring != None):
mqttformatstring = args.mqttformatstring
client_mqtt = paho.Client(paho.CallbackAPIVersion.VERSION2, "E3onCANclient.py")
if((args.mqttuser != None) and (args.mqttpass != None)):
client_mqtt.username_pw_set(args.mqttuser , password=args.mqttpass)
client_mqtt.connect(mqttParamas[0], int(mqttParamas[1]))
client_mqtt.reconnect_delay_set(min_delay=1, max_delay=30)
client_mqtt.loop_start()
print("Wait for dids and publish to mqtt...")
try:
if (args.file == None):
with can.Bus(interface='socketcan',
channel=channel,
receive_own_messages=True, can_filters=filters) as bus:
# iterate over received messages
evalMessages(bus, device, args)
else:
import candump2msgbus as dump
mydump = dump.candump2msgBus(args.file, CANid)
evalMessages(mydump.file2messages(), device, args)
except (KeyboardInterrupt, InterruptedError):
# got <STRG-C> or SIGINT (<kill -s SIGINT pid>)
pass