-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathtrident-metadata.py
executable file
·177 lines (151 loc) · 5.35 KB
/
trident-metadata.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
#!/usr/bin/env python3
#
# (C) Ketil Froyn <[email protected]> 2018
#
# This is a tool that tries to fetch timestamps and telemetry
# data from the OpenROV Trident's database files.
#
import re
import sys
import os
import subprocess
import json
import time
import getopt
import traceback
def usage():
print("Usage: trident-metadata.py <options> <path.to.cblite2/db.forest.0>")
print(" -a: dump all, no output time")
print(" -f: specify strftime format")
print(" -h: display this message, then exit")
print(" -s: output standard Trident date string")
print(" -t: output time")
print(" -T: don't output time")
print(" -e: output seconds since epoch")
print(" -l: use local time")
print(" -u: use UTC/gmtime (default)")
sys.exit(1)
def get_documents(dump):
for result in re.finditer('''.*?\nDoc ID:\s([^\n]+).*?\n\s+Body:\s\(hex\)\s*\n(.*?)\n\n''', dump, flags=re.DOTALL|re.MULTILINE):
docid = result.group(1)
raw_body = result.group(2)
if docid in ["privateUUID", "publicUUID"]:
continue
# strip leading space on each line
body = re.sub('''^\s+''', '', raw_body, flags=re.MULTILINE)
# strip trailing space and interpreted bytes on each line
body = re.sub('''\s{3}.*$''', '', body, flags=re.MULTILINE)
# split to individual bytes and decode bytes to binary string
raw = re.sub("\s*([a-f0-9]{2})\s*", (lambda x: chr(int(x.group(1), 16))), body)
# find the json string in the binary data
result = re.search('''(\{[\w\s\"\{\}\[\]:\.\?,\-_]{10,}\})''', raw)
if not result:
print("Couldn't recognize json in body data:", raw_body)
sys.exit(1)
j = result.group(1)
# parse json and fetch timestamp
try:
data = json.loads(j)
except json.decoder.JSONDecodeError:
print("Got error while parsing json:", j)
sys.exit(1)
try:
docid = docid.split(":")[1]
except IndexError:
pass
yield((docid, data))
def print_time(data, localtime, strftime_format):
sec = data['rx_ts']['sec']
timetuple = None
if localtime:
timetuple = time.localtime(sec)
else:
timetuple = time.gmtime(sec)
ts_str = time.strftime(strftime_format, timetuple)
print(ts_str)
def main():
strftime_format = "%Y%m%d-%H%M%SZ"
localtime = False
dump_all = False
output_time = True
opts, args = getopt.getopt(sys.argv[1:], 'ahstTeluf:')
for o, a in opts:
if o == '-f':
# specify strftime format
strftime_format = a
elif o == '-a':
dump_all = True
output_time = False
elif o == '-h':
# display help, then exit
usage()
elif o == '-s':
# output standard Trident date string
strftime_format = "Trident-%b-%d-%H%M%S"
elif o == '-t':
output_time = True
elif o == '-T':
output_time = False
elif o == '-e':
# Output seconds since epoch
strftime_format = "%s"
localtime = True # otherwise it'll count from the wrong base
elif o == '-l':
# use localtime (default is UTC / gmtime)
localtime = True
elif o == '-u':
# use UTC / gmtime (default is UTC / gmtime)
localtime = False
dbdump_cmd=['''forestdb_dump''', '''--hex-body''']
try:
dbfile=args[0]
except IndexError:
usage()
dump = ""
try:
command = dbdump_cmd + [dbfile]
if dump_all:
dump = subprocess.check_output(command).decode('utf-8')
else:
got_doc = False
proc = subprocess.Popen(command, bufsize=1, stdout=subprocess.PIPE)
for line in proc.stdout:
line = line.decode('utf-8')
if not got_doc:
if re.match('^Doc ID:', line):
got_doc = True
dump = "\n" # To match regex...
if got_doc:
dump = dump + line
if re.match('^\s*$', line):
break
proc.kill()
proc.wait()
except FileNotFoundError:
print("forestdb_dump not available. Compile from https://github.com/couchbase/forestdb.git")
print("and make sure the executable is in your PATH.")
sys.exit(1)
except Exception:
print("Error running forestdb_dump, bailing out:", command, file=sys.stderr)
traceback.print_exc(e)
sys.exit(1)
data = None
for i, d in get_documents(dump):
# print("Got document:", i, d)
# if i != d["topic"]:
# print("ID different from topic:", i, d["topic"])
if output_time:
print_time(d, localtime, strftime_format)
output_time = False
if not dump_all:
break
if d["topic"] in ["FlightCameraH264Message", "FlightVideoScreenshotMessage"]:
continue
else:
if d["data"] != "":
print("{2}.{3} {0} {1}".format(d["topic"], d["data"], d["rx_ts"]["sec"], d["rx_ts"]["nanosec"]))
if __name__ == '__main__':
try:
main()
except KeyboardInterrupt:
print("Interrupted by user, exiting...", file=sys.stderr)