Source code for selfdrive.test.process_replay.compare_logs

#!/usr/bin/env python3
import sys
import math
import capnp
import numbers
import dictdiffer
from collections import Counter

from openpilot.tools.lib.logreader import LogReader

EPSILON = sys.float_info.epsilon


[docs] def remove_ignored_fields(msg, ignore): msg = msg.as_builder() for key in ignore: attr = msg keys = key.split(".") if msg.which() != keys[0] and len(keys) > 1: continue for k in keys[:-1]: # indexing into list if k.isdigit(): attr = attr[int(k)] else: attr = getattr(attr, k) v = getattr(attr, keys[-1]) if isinstance(v, bool): val = False elif isinstance(v, numbers.Number): val = 0 elif isinstance(v, (list, capnp.lib.capnp._DynamicListBuilder)): val = [] else: raise NotImplementedError(f"Unknown type: {type(v)}") setattr(attr, keys[-1], val) return msg
[docs] def compare_logs(log1, log2, ignore_fields=None, ignore_msgs=None, tolerance=None,): if ignore_fields is None: ignore_fields = [] if ignore_msgs is None: ignore_msgs = [] tolerance = EPSILON if tolerance is None else tolerance log1, log2 = ( [m for m in log if m.which() not in ignore_msgs] for log in (log1, log2) ) if len(log1) != len(log2): cnt1 = Counter(m.which() for m in log1) cnt2 = Counter(m.which() for m in log2) raise Exception(f"logs are not same length: {len(log1)} VS {len(log2)}\n\t\t{cnt1}\n\t\t{cnt2}") diff = [] for msg1, msg2 in zip(log1, log2, strict=True): if msg1.which() != msg2.which(): raise Exception("msgs not aligned between logs") msg1 = remove_ignored_fields(msg1, ignore_fields) msg2 = remove_ignored_fields(msg2, ignore_fields) if msg1.to_bytes() != msg2.to_bytes(): msg1_dict = msg1.as_reader().to_dict(verbose=True) msg2_dict = msg2.as_reader().to_dict(verbose=True) dd = dictdiffer.diff(msg1_dict, msg2_dict, ignore=ignore_fields) # Dictdiffer only supports relative tolerance, we also want to check for absolute # TODO: add this to dictdiffer def outside_tolerance(diff): try: if diff[0] == "change": a, b = diff[2] finite = math.isfinite(a) and math.isfinite(b) if finite and isinstance(a, numbers.Number) and isinstance(b, numbers.Number): return abs(a - b) > max(tolerance, tolerance * max(abs(a), abs(b))) except TypeError: pass return True dd = list(filter(outside_tolerance, dd)) diff.extend(dd) return diff
[docs] def format_process_diff(diff): diff_short, diff_long = "", "" if isinstance(diff, str): diff_short += f" {diff}\n" diff_long += f"\t{diff}\n" else: cnt: dict[str, int] = {} for d in diff: diff_long += f"\t{str(d)}\n" k = str(d[1]) cnt[k] = 1 if k not in cnt else cnt[k] + 1 for k, v in sorted(cnt.items()): diff_short += f" {k}: {v}\n" return diff_short, diff_long
[docs] def format_diff(results, log_paths, ref_commit): diff_short, diff_long = "", "" diff_long += f"***** tested against commit {ref_commit} *****\n" failed = False for segment, result in list(results.items()): diff_short += f"***** results for segment {segment} *****\n" diff_long += f"***** differences for segment {segment} *****\n" for proc, diff in list(result.items()): diff_long += f"*** process: {proc} ***\n" diff_long += f"\tref: {log_paths[segment][proc]['ref']}\n" diff_long += f"\tnew: {log_paths[segment][proc]['new']}\n\n" diff_short += f" {proc}\n" if isinstance(diff, str) or len(diff): diff_short += f" ref: {log_paths[segment][proc]['ref']}\n" diff_short += f" new: {log_paths[segment][proc]['new']}\n\n" failed = True proc_diff_short, proc_diff_long = format_process_diff(diff) diff_long += proc_diff_long diff_short += proc_diff_short return diff_short, diff_long, failed
if __name__ == "__main__": log1 = list(LogReader(sys.argv[1])) log2 = list(LogReader(sys.argv[2])) ignore_fields = sys.argv[3:] or ["logMonoTime", "controlsState.startMonoTime", "controlsState.cumLagMs"] results = {"segment": {"proc": compare_logs(log1, log2, ignore_fields)}} log_paths = {"segment": {"proc": {"ref": sys.argv[1], "new": sys.argv[2]}}} diff_short, diff_long, failed = format_diff(results, log_paths, None) print(diff_long) print(diff_short)