You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
5.5 KiB
160 lines
5.5 KiB
#!/usr/bin/env python3
|
|
import os
|
|
import sys
|
|
import multiprocessing
|
|
import platform
|
|
import shutil
|
|
import subprocess
|
|
import tarfile
|
|
import tempfile
|
|
import requests
|
|
import argparse
|
|
|
|
from common.basedir import BASEDIR
|
|
from selfdrive.test.process_replay.compare_logs import save_log
|
|
from tools.lib.robust_logreader import RobustLogReader
|
|
from tools.lib.route import Route, SegmentName
|
|
from urllib.parse import urlparse, parse_qs
|
|
|
|
juggle_dir = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
DEMO_ROUTE = "4cf7a6ad03080c90|2021-09-29--13-46-36"
|
|
RELEASES_URL="https://github.com/commaai/PlotJuggler/releases/download/latest"
|
|
INSTALL_DIR = os.path.join(juggle_dir, "bin")
|
|
PLOTJUGGLER_BIN = os.path.join(juggle_dir, "bin/plotjuggler")
|
|
|
|
|
|
def install():
|
|
m = f"{platform.system()}-{platform.machine()}"
|
|
supported = ("Linux-x86_64", "Darwin-arm64", "Darwin-x86_64")
|
|
if m not in supported:
|
|
raise Exception(f"Unsupported platform: '{m}'. Supported platforms: {supported}")
|
|
|
|
if os.path.exists(INSTALL_DIR):
|
|
shutil.rmtree(INSTALL_DIR)
|
|
os.mkdir(INSTALL_DIR)
|
|
|
|
url = os.path.join(RELEASES_URL, m + ".tar.gz")
|
|
with requests.get(url, stream=True) as r, tempfile.NamedTemporaryFile() as tmp:
|
|
r.raise_for_status()
|
|
with open(tmp.name, 'wb') as tmpf:
|
|
for chunk in r.iter_content(chunk_size=1024*1024):
|
|
tmpf.write(chunk)
|
|
|
|
with tarfile.open(tmp.name) as tar:
|
|
tar.extractall(path=INSTALL_DIR)
|
|
|
|
|
|
def load_segment(segment_name):
|
|
if segment_name is None:
|
|
return []
|
|
|
|
try:
|
|
return list(RobustLogReader(segment_name))
|
|
except ValueError as e:
|
|
print(f"Error parsing {segment_name}: {e}")
|
|
return []
|
|
|
|
|
|
def start_juggler(fn=None, dbc=None, layout=None):
|
|
env = os.environ.copy()
|
|
env["BASEDIR"] = BASEDIR
|
|
env["PATH"] = f"{INSTALL_DIR}:{os.getenv('PATH', '')}"
|
|
if dbc:
|
|
env["DBC_NAME"] = dbc
|
|
|
|
extra_args = ""
|
|
if fn is not None:
|
|
extra_args += f" -d {fn}"
|
|
if layout is not None:
|
|
extra_args += f" -l {layout}"
|
|
|
|
cmd = f'{PLOTJUGGLER_BIN} --plugin_folders {INSTALL_DIR}{extra_args}'
|
|
subprocess.call(cmd, shell=True, env=env, cwd=juggle_dir)
|
|
|
|
|
|
def juggle_route(route_or_segment_name, segment_count, qlog, can, layout, dbc=None):
|
|
segment_start = 0
|
|
if 'cabana' in route_or_segment_name:
|
|
query = parse_qs(urlparse(route_or_segment_name).query)
|
|
route_or_segment_name = query["route"][0]
|
|
|
|
if route_or_segment_name.startswith(("http://", "https://")) or os.path.isfile(route_or_segment_name):
|
|
logs = [route_or_segment_name]
|
|
else:
|
|
route_or_segment_name = SegmentName(route_or_segment_name, allow_route_name=True)
|
|
segment_start = max(route_or_segment_name.segment_num, 0)
|
|
|
|
if route_or_segment_name.segment_num != -1 and segment_count is None:
|
|
segment_count = 1
|
|
|
|
r = Route(route_or_segment_name.route_name.canonical_name)
|
|
logs = r.qlog_paths() if qlog else r.log_paths()
|
|
|
|
segment_end = segment_start + segment_count if segment_count else None
|
|
logs = logs[segment_start:segment_end]
|
|
|
|
if None in logs:
|
|
ans = input(f"{logs.count(None)}/{len(logs)} of the rlogs in this segment are missing, would you like to fall back to the qlogs? (y/n) ")
|
|
if ans == 'y':
|
|
logs = r.qlog_paths()[segment_start:segment_end]
|
|
else:
|
|
print("Please try a different route or segment")
|
|
return
|
|
|
|
all_data = []
|
|
with multiprocessing.Pool(24) as pool:
|
|
for d in pool.map(load_segment, logs):
|
|
all_data += d
|
|
|
|
if not can:
|
|
all_data = [d for d in all_data if d.which() not in ['can', 'sendcan']]
|
|
|
|
# Infer DBC name from logs
|
|
if dbc is None:
|
|
for cp in [m for m in all_data if m.which() == 'carParams']:
|
|
try:
|
|
DBC = __import__(f"selfdrive.car.{cp.carParams.carName}.values", fromlist=['DBC']).DBC
|
|
dbc = DBC[cp.carParams.carFingerprint]['pt']
|
|
except Exception:
|
|
pass
|
|
break
|
|
|
|
with tempfile.NamedTemporaryFile(suffix='.rlog', dir=juggle_dir) as tmp:
|
|
save_log(tmp.name, all_data, compress=False)
|
|
del all_data
|
|
start_juggler(tmp.name, dbc, layout)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
parser = argparse.ArgumentParser(description="A helper to run PlotJuggler on openpilot routes",
|
|
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
|
|
|
|
parser.add_argument("--demo", action="store_true", help="Use the demo route instead of providing one")
|
|
parser.add_argument("--qlog", action="store_true", help="Use qlogs")
|
|
parser.add_argument("--can", action="store_true", help="Parse CAN data")
|
|
parser.add_argument("--stream", action="store_true", help="Start PlotJuggler in streaming mode")
|
|
parser.add_argument("--layout", nargs='?', help="Run PlotJuggler with a pre-defined layout")
|
|
parser.add_argument("--install", action="store_true", help="Install or update PlotJuggler + plugins")
|
|
parser.add_argument("--dbc", help="Set the DBC name to load for parsing CAN data. If not set, the DBC will be automatically inferred from the logs.")
|
|
parser.add_argument("route_or_segment_name", nargs='?', help="The route or segment name to plot (cabana share URL accepted)")
|
|
parser.add_argument("segment_count", type=int, nargs='?', help="The number of segments to plot")
|
|
|
|
if len(sys.argv) == 1:
|
|
parser.print_help()
|
|
sys.exit()
|
|
args = parser.parse_args()
|
|
|
|
if args.install:
|
|
install()
|
|
sys.exit()
|
|
|
|
if not os.path.exists(PLOTJUGGLER_BIN):
|
|
print("PlotJuggler is missing. Downloading...")
|
|
install()
|
|
|
|
if args.stream:
|
|
start_juggler(layout=args.layout)
|
|
else:
|
|
route_or_segment_name = DEMO_ROUTE if args.demo else args.route_or_segment_name.strip()
|
|
juggle_route(route_or_segment_name, args.segment_count, args.qlog, args.can, args.layout, args.dbc)
|
|
|