forked from johanmeijer/grott
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgrottdata.py
558 lines (484 loc) · 27.1 KB
/
grottdata.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
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
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
# grottdata.py processing data functions
# Updated: 2021-12-14
# Version 2.6.1g
#import time
from datetime import datetime, timedelta
#import pytz
import time
import sys
import struct
import textwrap
from itertools import cycle # to support "cycling" the iterator
import json, codecs
# requests
#import mqtt
import paho.mqtt.publish as publish
# Formats multi-line data
def format_multi_line(prefix, string, size=80):
size -= len(prefix)
if isinstance(string, bytes):
string = ''.join(r'\x{:02x}'.format(byte) for byte in string)
if size % 2:
size -= 1
return '\n'.join([prefix + line for line in textwrap.wrap(string, size)])
#decrypt data.
def decrypt(decdata) :
ndecdata = len(decdata)
# Create mask and convert to hexadecimal
mask = "Growatt"
hex_mask = ['{:02x}'.format(ord(x)) for x in mask]
nmask = len(hex_mask)
#start decrypt routine
unscrambled = list(decdata[0:8]) #take unscramble header
for i,j in zip(range(0,ndecdata-8),cycle(range(0,nmask))):
unscrambled = unscrambled + [decdata[i+8] ^ int(hex_mask[j],16)]
result_string = "".join("{:02x}".format(n) for n in unscrambled)
print("\t - " + "Growatt data decrypted V2")
return result_string
def str2bool(defstr):
if defstr in ("True", "true", "TRUE", "y", "Y", "yes", "YES", 1, "1") : defret = True
if defstr in ("False", "false", "FALSE", "n", "N", "no", "NO", 0, "0") : defret = False
if 'defret' in locals():
return(defret)
else : return()
def procdata(conf,data):
if conf.verbose:
print("\t - " + "Growatt original Data:")
print(format_multi_line("\t\t ", data))
header = "".join("{:02x}".format(n) for n in data[0:8])
ndata = len(data)
buffered = "nodetect" # set buffer detection to nodetect (for compat mode), wil in auto detection changed to no or yes
# automatic detect protocol (decryption and protocol) only if compat = False!
novalidrec = False
if conf.compat is False :
if conf.verbose :
print("\t - " + "Grott automatic protocol detection")
print("\t - " + "Grott data record length", ndata)
#print(header)
layout = "T" + header[6:8] + header[12:14] + header[14:16]
if ndata > 375: layout = layout + "X"
if conf.invtype != "default" :
layout = layout + conf.invtype.upper()
if header[14:16] == "50" : buffered = "yes"
else: buffered = "no"
if conf.verbose : print("\t - " + "layout : ", layout)
try:
# does record layout record exists?
test = conf.recorddict[layout]
except:
#try generic if generic record exist
if conf.verbose : print("\t - " + "no matching record layout found, try generic")
if header[14:16] in ("04","50") :
layout = layout.replace(header[12:16], "NNNN")
try:
# does generic record layout record exists?
test = conf.recorddict[layout]
except:
#no valid record fall back on old processing?
if conf.verbose : print("\t - " + "no matching record layout found, standard processing performed")
layout = "none"
novalidrec = True
else:
novalidrec = True
conf.layout = layout
if conf.verbose : print("\t - " + "Record layout used : ", layout)
#Decrypt
try:
#see if decrypt keyword is defined
conf.decrypt = str2bool(conf.recorddict[layout]["decrypt"]["value"])
except:
#if decrypt not defined, default is decypt
conf.decrypt = True
if conf.decrypt:
result_string = decrypt(data)
if conf.verbose : print("\t - " + "Grott Growatt data decrypted")
else:
#do not decrypt
result_string = data.hex()
if conf.verbose: print("\t - " + "Grott Growatt unencrypted data used")
if conf.verbose:
print("\t - " + 'Growatt plain data:')
print(format_multi_line("\t\t ", result_string))
#debug only: print(result_string)
# test position :
# print(result_string.find('0074' ))
# Test length if < 12 it is a data ack record, if novalidrec flag is true it is not a (recognized) data record
if ndata < 12 or novalidrec == True:
if conf.verbose : print("\t - " + "Grott data ack record or data record not defined no processing done")
return
# Inital flag to detect off real data is processed
dataprocessed = False
# define dictonary for key values.
definedkey = {}
if conf.compat is False:
# new method if compat = False (autoatic detection):
if conf.verbose:
print("\t - " + 'Growatt new layout processing')
print("\t\t - " + "decrypt : ",conf.decrypt)
print("\t\t - " + "offset : ", conf.offset)
print("\t\t - " + "record layout : ", layout)
print()
for keyword in conf.recorddict[layout].keys() :
if keyword not in ("decrypt","date") :
#try if keyword should be included
include=True
try:
#try if key type is specified
if conf.recorddict[layout][keyword]["incl"] == "no" :
include=False
except:
#no include statement keyword should be process, set include to prevent except errors
include = True
#process only keyword needs to be included (default):
if ((include) or (conf.includeall)):
try:
#try if key type is specified
keytype = conf.recorddict[layout][keyword]["type"]
except:
#if not default is num
keytype = "num"
if keytype == "text" :
definedkey[keyword] = result_string[conf.recorddict[layout][keyword]["value"]:conf.recorddict[layout][keyword]["value"]+(conf.recorddict[layout][keyword]["length"]*2)]
definedkey[keyword] = codecs.decode(definedkey[keyword], "hex").decode('utf-8')
#print(definedkey[keyword])
if keytype == "num" :
#else:
definedkey[keyword] = int(result_string[conf.recorddict[layout][keyword]["value"]:conf.recorddict[layout][keyword]["value"]+(conf.recorddict[layout][keyword]["length"]*2)],16)
if keytype == "numx" :
#process signed integer
keybytes = bytes.fromhex(result_string[conf.recorddict[layout][keyword]["value"]:conf.recorddict[layout][keyword]["value"]+(conf.recorddict[layout][keyword]["length"]*2)])
definedkey[keyword] = int.from_bytes(keybytes, byteorder='big', signed=True)
# test if pvserial was defined, if not take inverterid from config.
try:
test = definedkey["pvserial"]
except:
definedkey["pvserial"] = conf.inverterid
conf.recorddict[layout]["pvserial"] = {"value" : 0, "type" : "text"}
if conf.verbose : print("\t - pvserial not found used configuration defined invertid:", definedkey["pvserial"] )
# test if dateoffset is defined, if not take set to 0 (no futher date retrieval processing) .
try:
# test of date is specified in layout
dateoffset = int(conf.recorddict[layout]["date"]["value"])
except:
# no date specified, default no date specified
dateoffset = 0
#proces date value if specifed
if dateoffset > 0 and (conf.gtime != "server" or buffered == "yes"):
if conf.verbose: print("\t - " + 'Grott data record date/time processing started')
#date
pvyearI = int(result_string[dateoffset:dateoffset+2],16)
if pvyearI < 10 : pvyear = "200" + str(pvyearI)
else: pvyear = "20" + str(pvyearI)
pvmonthI = int(result_string[dateoffset+2:dateoffset+4],16)
if pvmonthI < 10 : pvmonth = "0" + str(pvmonthI)
else: pvmonth = str(pvmonthI)
pvdayI = int(result_string[dateoffset+4:dateoffset+6],16)
if pvdayI < 10 : pvday = "0" + str(pvdayI)
else: pvday = str(pvdayI)
#Time
pvhourI = int(result_string[dateoffset+6:dateoffset+8],16)
if pvhourI < 10 : pvhour = "0" + str(pvhourI)
else: pvhour = str(pvhourI)
pvminuteI = int(result_string[dateoffset+8:dateoffset+10],16)
if pvminuteI < 10 : pvminute = "0" + str(pvminuteI)
else: pvminute = str(pvminuteI)
pvsecondI = int(result_string[dateoffset+10:dateoffset+12],16)
if pvsecondI < 10 : pvsecond = "0" + str(pvsecondI)
else: pvsecond = str(pvsecondI)
# create date/time is format
pvdate = pvyear + "-" + pvmonth + "-" + pvday + "T" + pvhour + ":" + pvminute + ":" + pvsecond
# test if valid date/time in data record
try:
testdate = datetime.strptime(pvdate, "%Y-%m-%dT%H:%M:%S")
jsondate = pvdate
if conf.verbose : print("\t - date-time: ", jsondate)
timefromserver = False # Indicate of date/time is from server (used for buffered data)
except ValueError:
# Date could not be parsed - either the format is different or it's not a
# valid date
if conf.verbose : print("\t - " + "no or no valid time/date found, grott server time will be used (buffer records not sent!)")
timefromserver = True
jsondate = datetime.now().replace(microsecond=0).isoformat()
else:
if conf.verbose: print("\t - " + "Grott server date/time used")
jsondate = datetime.now().replace(microsecond=0).isoformat()
timefromserver = True
dataprocessed = True
else:
serialfound = False
if(result_string.find(conf.SN) > -1):
serialfound = True
if serialfound == True:
jsondate = datetime.now().replace(microsecond=0).isoformat()
timefromserver = True
if conf.verbose: print("\t - " + 'Growatt processing values for: ', bytearray.fromhex(conf.SN).decode())
#Retrieve values
snstart = result_string.find(conf.SN)
definedkey["pvserial"] = result_string[snstart:snstart+20]
definedkey["pvstatus"] = int(result_string[snstart+conf.offset*2+15*2:snstart+conf.offset*2+15*2+4],16)
#Only process value if pvstatus is oke (this is because unexpected pvstatus of 257)
if definedkey["pvstatus"] == 0 or definedkey["pvstatus"] == 1:
definedkey["pvpowerin"] = int(result_string[snstart+conf.offset*2+17*2:snstart+conf.offset*2+17*2+8],16)
definedkey["pv1voltage"] = int(result_string[snstart+conf.offset*2+21*2:snstart+conf.offset*2+21*2+4],16)
definedkey["pv1current"] = int(result_string[snstart+conf.offset*2+23*2:snstart+conf.offset*2+23*2+4],16)
definedkey["pv1watt"] = int(result_string[snstart+conf.offset*2+25*2:snstart+conf.offset*2+25*2+8],16)
definedkey["pv2voltage"] = int(result_string[snstart+conf.offset*2+29*2:snstart+conf.offset*2+29*2+4],16)
definedkey["pv2current"] = int(result_string[snstart+conf.offset*2+31*2:snstart+conf.offset*2+31*2+4],16)
definedkey["pv2watt"] = int(result_string[snstart+conf.offset*2+33*2:snstart+conf.offset*2+33*2+8],16)
definedkey["pvpowerout"] = int(result_string[snstart+conf.offset*2+37*2:snstart+conf.offset*2+37*2+8],16)
definedkey["pvfrequentie"] = int(result_string[snstart+conf.offset*2+41*2:snstart+conf.offset*2+41*2+4],16)
definedkey["pvgridvoltage"] = int(result_string[snstart+conf.offset*2+43*2:snstart+conf.offset*2+43*2+4],16)
definedkey["pvenergytoday"] = int(result_string[snstart+conf.offset*2+67*2:snstart+conf.offset*2+67*2+8],16)
definedkey["pvenergytotal"] = int(result_string[snstart+conf.offset*2+71*2:snstart+conf.offset*2+71*2+8],16)
definedkey["pvtemperature"] = int(result_string[snstart+conf.offset*2+79*2:snstart+conf.offset*2+79*2+4],16)
definedkey["pvipmtemperature"] = int(result_string[snstart+conf.offset*2+97*2:snstart+conf.offset*2+97*2+4],16)
dataprocessed = True
else:
if conf.verbose: print("\t - " + 'No valid monitor data, PV status: :', definedkey["pvstatus"])
else:
if conf.verbose: print("\t - "+ 'No Growatt data processed or SN not found:')
if conf.trace:
print("\t - "+ 'Growatt unprocessed Data:')
print(format_multi_line("\t\t - ", result_string))
if dataprocessed:
# only sendout data to MQTT if it is processed.
# Print values
if conf.verbose:
if conf.compat :
#print in compatibility mode
definedkey["pvserial"] = codecs.decode(definedkey["pvserial"], "hex").decode('utf-8')
print("\t - " + "Grott values retrieved:")
print("\t\t - " + "pvserial: ", definedkey["pvserial"])
print("\t\t - " + "pvstatus: ", definedkey["pvstatus"])
print("\t\t - " + "pvpowerin: ", definedkey["pvpowerin"]/10)
print("\t\t - " + "pvpowerout: ", definedkey["pvpowerout"]/10)
print("\t\t - " + "pvenergytoday: ", definedkey["pvenergytoday"]/10)
print("\t\t - " + "pvenergytotal: ", definedkey["pvenergytotal"]/10)
print("\t\t - " + "pv1watt: ", definedkey["pv1watt"]/10)
print("\t\t - " + "pv2watt: ", definedkey["pv2watt"]/10)
print("\t\t - " + "pvfrequentie: ", definedkey["pvfrequentie"]/100)
print("\t\t - " + "pvgridvoltage: ", definedkey["pvgridvoltage"]/10)
print("\t\t - " + "pv1voltage: ", definedkey["pv1voltage"]/10)
print("\t\t - " + "pv1current: ", definedkey["pv1current"]/10)
print("\t\t - " + "pv2voltage: ", definedkey["pv2voltage"]/10)
print("\t\t - " + "pv2current: ", definedkey["pv2current"]/10)
print("\t\t - " + "pvtemperature: ", definedkey["pvtemperature"]/10)
print("\t\t - " + "pvipmtemperature: ", definedkey["pvipmtemperature"]/10)
else:
#dynamic print
print("\t - " + "Grott values retrieved:")
for key in definedkey :
# test if there is an divide factor is specifed
try:
#print(keyword)
keydivide = conf.recorddict[layout][key]["divide"]
#print(keydivide)
except:
#print("error")
keydivide = 1
if type(definedkey[key]) != type(str()) and keydivide != 1 :
printkey = "{:.1f}".format(definedkey[key]/keydivide)
else :
printkey = definedkey[key]
print("\t\t - ",key.ljust(20) + " : ",printkey)
#create JSON message (first create obj dict and then convert to a JSON message)
# if record is a smart monitor record use datalogserial as device (to distinguish from solar record)
# and filter if invalid record (0 < voltage_l1 > 500 )
if header[14:16] != "20" :
jsonobj = {
"device" : definedkey["pvserial"],
"time" : jsondate,
"buffered" : buffered,
"values" : {}
}
else :
# filter if invalid 0120 record (0 < voltage_l1 > 500 )
if (definedkey["voltage_l1"]/10 > 500) or (definedkey["voltage_l1"]/10 < 0) :
print("\t - " + "Grott invalid 0120 record processing stopped")
return
jsonobj = {
"device" : definedkey["datalogserial"],
"time" : jsondate,
"buffered" : buffered,
"values" : {}
}
for key in definedkey :
if key != "pvserial" :
#if conf.recorddict[layout][key]["type"] == "num" :
# only add int values to the json object
if type(definedkey[key]) == type(1) :
jsonobj["values"][key] = definedkey[key]
jsonmsg = json.dumps(jsonobj)
if conf.verbose:
print("\t - " + "MQTT jsonmsg: ")
print(format_multi_line("\t\t\t ", jsonmsg))
#do net invalid records (e.g. buffered records with time from server) or buffered records if sendbuf = False
if (buffered == "yes") :
if (conf.sendbuf == False) or (timefromserver == True) :
if conf.verbose: print("\t - " + 'Buffered record not sent: sendbuf = False or invalid date/time format')
return
if conf.nomqtt != True:
try:
publish.single(conf.mqtttopic, payload=jsonmsg, qos=0, retain=False, hostname=conf.mqttip,port=conf.mqttport, client_id=conf.inverterid, keepalive=60, auth=conf.pubauth)
if conf.verbose: print("\t - " + 'MQTT message message sent')
except TimeoutError:
if conf.verbose: print("\t - " + 'MQTT connection time out error')
except ConnectionRefusedError:
if conf.verbose: print("\t - " + 'MQTT connection refused by target')
except BaseException as error:
if conf.verbose: print("\t - "+ 'MQTT send failed:', str(error))
else:
if conf.verbose: print("\t - " + 'No MQTT message sent, MQTT disabled')
# process pvoutput if enabled
if conf.pvoutput :
import requests
pvidfound = False
if conf.pvinverters == 1 :
pvssid = conf.pvsystemid[1]
pvidfound = True
else:
for pvnum, pvid in conf.pvinverterid.items():
if pvid == definedkey["pvserial"] :
print(pvid)
pvssid = conf.pvsystemid[pvnum]
pvidfound = True
if not pvidfound:
if conf.verbose : print("\t - " + "pvsystemid not found for inverter : ", definedkey["pvserial"])
return
if conf.verbose : print("\t - " + "Grott send data to PVOutput systemid: ", pvssid, "for inverter: ", definedkey["pvserial"])
pvheader = {
"X-Pvoutput-Apikey" : conf.pvapikey,
"X-Pvoutput-SystemId" : pvssid
}
pvodate = jsondate[:4] +jsondate[5:7] + jsondate[8:10]
# debug: pvodate = jsondate[:4] +jsondate[5:7] + "16"
pvotime = jsondate[11:16]
# debug: pvotime = "09:05"
# if record is a smart monitor record sent smart monitor data to PVOutput
if header[14:16] != "20" :
pvdata = {
"d" : pvodate,
"t" : pvotime,
"v1" : definedkey["pvenergytoday"]*100,
"v2" : definedkey["pvpowerout"]/10,
"v6" : definedkey["pvgridvoltage"]/10
}
#print(pvheader)
if conf.verbose : print("\t\t - ", pvheader)
if conf.verbose : print("\t\t - ", pvdata)
reqret = requests.post(conf.pvurl, data = pvdata, headers = pvheader)
if conf.verbose : print("\t - " + "Grott PVOutput response: ")
if conf.verbose : print("\t\t - ", reqret.text)
else:
# send smat monitor data c1 = 3 indiates v3 is lifetime energy (day wil be calculated), n=1 indicates is net data (import /export)
# value seprated because it is not allowed to sent combination at once
pvdata1 = {
"d" : pvodate,
"t" : pvotime,
"v3" : definedkey["pos_act_energy"]*100,
"c1" : 3,
"v6" : definedkey["voltage_l1"]/10
}
pvdata2 = {
"d" : pvodate,
"t" : pvotime,
"v4" : definedkey["pos_rev_act_power"]/10,
"v6" : definedkey["voltage_l1"]/10,
"n" : 1
}
#"v4" : definedkey["pos_act_power"]/10,
#print(pvheader)
if conf.verbose : print("\t\t - ", pvheader)
if conf.verbose : print("\t\t - ", pvdata1)
if conf.verbose : print("\t\t - ", pvdata2)
reqret = requests.post(conf.pvurl, data = pvdata1, headers = pvheader)
if conf.verbose : print("\t - " + "Grott PVOutput response SM1: ")
if conf.verbose : print("\t\t - ", reqret.text)
reqret = requests.post(conf.pvurl, data = pvdata2, headers = pvheader)
if conf.verbose : print("\t - " + "Grott PVOutput response SM2: ")
if conf.verbose : print("\t\t - ", reqret.text)
else:
if conf.verbose : print("\t - " + "Grott Send data to PVOutput disabled ")
# influxDB processing
if conf.influx:
if conf.verbose : print("\t - " + "Grott InfluxDB publihing started")
try:
import pytz
except:
if conf.verbose : print("\t - " + "Grott PYTZ Library not installed in Python, influx processing disabled")
conf.inlyx = False
return
try:
local = pytz.timezone(conf.tmzone)
except :
if conf.verbose :
if conf.tmzone == "local": print("\t - " + "Timezone local specified default timezone used")
else : print("\t - " + "Grott unknown timezone : ",conf.tmzone,", default timezone used")
conf.tmzone = "local"
local = int(time.timezone/3600)
#print(local)
if conf.tmzone == "local":
curtz = time.timezone
utc_dt = datetime.strptime (jsondate, "%Y-%m-%dT%H:%M:%S") + timedelta(seconds=curtz)
else :
naive = datetime.strptime (jsondate, "%Y-%m-%dT%H:%M:%S")
local_dt = local.localize(naive, is_dst=None)
utc_dt = local_dt.astimezone(pytz.utc)
ifdt = utc_dt.strftime ("%Y-%m-%dT%H:%M:%S")
if conf.verbose : print("\t - " + "Grott original time : ",jsondate,"adjusted UTC time for influx : ",ifdt)
# prepare influx jsonmsg dictionary
# if record is a smart monitor record use datalogserial as measurement (to distinguish from solar record)
if header[14:16] != "20" :
ifobj = {
"measurement" : definedkey["pvserial"],
"time" : ifdt,
"fields" : {}
}
else:
ifobj = {
"measurement" : definedkey["datalogserial"],
"time" : ifdt,
"fields" : {}
}
for key in definedkey :
if key != "date" :
ifobj["fields"][key] = definedkey[key]
#Create list for influx
ifjson = [ifobj]
print("\t - " + "Grott influxdb jsonmsg: ")
print(format_multi_line("\t\t\t ", str(ifjson)))
#if conf.verbose : print("\t - " + "Grott InfluxDB publihing started")
try:
if (conf.influx2):
if conf.verbose : print("\t - " + "Grott write to influxdb v2")
ifresult = conf.ifwrite_api.write(conf.ifbucket,conf.iforg,ifjson)
#print(ifresult)
else:
if conf.verbose : print("\t - " + "Grott write to influxdb v1")
ifresult = conf.influxclient.write_points(ifjson)
#except :
except Exception as e:
# if conf.verbose:
print("\t - " + "Grott InfluxDB error ")
print(e)
raise SystemExit("Grott Influxdb write error, grott will be stopped")
else:
if conf.verbose : print("\t - " + "Grott Send data to Influx disabled ")
if conf.extension :
if conf.verbose : print("\t - " + "Grott extension processing started : ", conf.extname)
import importlib
try:
module = importlib.import_module(conf.extname, package=None)
except :
if conf.verbose : print("\t - " + "Grott import extension failed:", conf.extname)
return
try:
ext_result = module.grottext(conf,result_string,jsonmsg)
if conf.verbose :
print("\t - " + "Grott extension processing ended:", ext_result)
except Exception as e:
print("\t - " + "Grott extension processing error:", repr(e))
import traceback
traceback.format_exc()
else:
if conf.verbose : print("\t - " + "Grott extension processing disabled ")