camerad: spectra stress test (#34716)
	
		
	
				
					
				
			* cam stress * lint fixes --------- Co-authored-by: Comma Device <device@comma.ai>pull/34665/head^2
							parent
							
								
									c8e598e647
								
							
						
					
					
						commit
						937e2f7eb2
					
				
				 3 changed files with 75 additions and 64 deletions
			
			
		| @ -1,81 +1,70 @@ | |||||||
| import pytest | import os | ||||||
| import time | import time | ||||||
|  | import pytest | ||||||
| import numpy as np | import numpy as np | ||||||
| from collections import defaultdict |  | ||||||
| 
 | 
 | ||||||
| import cereal.messaging as messaging | import cereal.messaging as messaging | ||||||
| from cereal import log |  | ||||||
| from cereal.services import SERVICE_LIST | from cereal.services import SERVICE_LIST | ||||||
| from openpilot.common.retry import retry |  | ||||||
| from openpilot.system.manager.process_config import managed_processes | from openpilot.system.manager.process_config import managed_processes | ||||||
|  | from openpilot.tools.lib.log_time_series import msgs_to_time_series | ||||||
| 
 | 
 | ||||||
| TEST_TIMESPAN = 10 | TEST_TIMESPAN = 10 | ||||||
| LAG_FRAME_TOLERANCE = {log.FrameData.ImageSensor.ar0231: 0.5,  # ARs use synced pulses for frame starts |  | ||||||
|                        log.FrameData.ImageSensor.ox03c10: 1.1, # OXs react to out-of-sync at next frame |  | ||||||
|                        log.FrameData.ImageSensor.os04c10: 1.1} |  | ||||||
| FRAME_DELTA_TOLERANCE = {log.FrameData.ImageSensor.ar0231: 1.0, |  | ||||||
|                          log.FrameData.ImageSensor.ox03c10: 1.0, |  | ||||||
|                          log.FrameData.ImageSensor.os04c10: 1.0} |  | ||||||
| 
 |  | ||||||
| CAMERAS = ('roadCameraState', 'driverCameraState', 'wideRoadCameraState') | CAMERAS = ('roadCameraState', 'driverCameraState', 'wideRoadCameraState') | ||||||
| 
 | 
 | ||||||
| @retry(attempts=3, delay=5.0) |  | ||||||
| def setup_camerad(cls): |  | ||||||
|   # run camerad and record logs |  | ||||||
|   managed_processes['camerad'].start() |  | ||||||
|   time.sleep(3) |  | ||||||
|   socks = {c: messaging.sub_sock(c, conflate=False, timeout=100) for c in CAMERAS} |  | ||||||
| 
 |  | ||||||
|   cls.logs = defaultdict(list) |  | ||||||
|   start_time = time.monotonic() |  | ||||||
|   while time.monotonic()- start_time < TEST_TIMESPAN: |  | ||||||
|     for cam, s in socks.items(): |  | ||||||
|       cls.logs[cam] += messaging.drain_sock(s) |  | ||||||
|     time.sleep(0.2) |  | ||||||
|   managed_processes['camerad'].stop() |  | ||||||
| 
 |  | ||||||
|   cls.log_by_frame_id = defaultdict(list) |  | ||||||
|   cls.sensor_type = None |  | ||||||
|   for cam, msgs in cls.logs.items(): |  | ||||||
|     if cls.sensor_type is None: |  | ||||||
|       cls.sensor_type = getattr(msgs[0], msgs[0].which()).sensor.raw |  | ||||||
|     expected_frames = SERVICE_LIST[cam].frequency * TEST_TIMESPAN |  | ||||||
|     assert expected_frames*0.95 < len(msgs) < expected_frames*1.05, f"unexpected frame count {cam}: {expected_frames=}, got {len(msgs)}" |  | ||||||
| 
 | 
 | ||||||
|     dts = np.abs(np.diff([getattr(m, m.which()).timestampSof/1e6 for m in msgs]) - 1000/SERVICE_LIST[cam].frequency) | def run_and_log(procs, services, duration): | ||||||
|     assert (dts < FRAME_DELTA_TOLERANCE[cls.sensor_type]).all(), f"{cam} dts(ms) out of spec: max diff {dts.max()}, 99 percentile {np.percentile(dts, 99)}" |   logs = [] | ||||||
|  | 
 | ||||||
|  |   try: | ||||||
|  |     for p in procs: | ||||||
|  |       managed_processes[p].start() | ||||||
|  |     socks = [messaging.sub_sock(s, conflate=False, timeout=100) for s in services] | ||||||
|  | 
 | ||||||
|  |     start_time = time.monotonic() | ||||||
|  |     while time.monotonic() - start_time < duration: | ||||||
|  |       for s in socks: | ||||||
|  |         logs.extend(messaging.drain_sock(s)) | ||||||
|  |     for p in procs: | ||||||
|  |       assert managed_processes[p].proc.is_alive() | ||||||
|  |   finally: | ||||||
|  |     for p in procs: | ||||||
|  |       managed_processes[p].stop() | ||||||
|  | 
 | ||||||
|  |   return logs | ||||||
| 
 | 
 | ||||||
|     for m in msgs: | @pytest.fixture(scope="module") | ||||||
|       cls.log_by_frame_id[getattr(m, m.which()).frameId].append(m) | def logs(): | ||||||
|  |   logs = run_and_log(["camerad", ], CAMERAS, TEST_TIMESPAN) | ||||||
|  |   ts = msgs_to_time_series(logs) | ||||||
| 
 | 
 | ||||||
|   # strip beginning and end |   for cam in CAMERAS: | ||||||
|   for _ in range(3): |     expected_frames = SERVICE_LIST[cam].frequency * TEST_TIMESPAN | ||||||
|     mn, mx = min(cls.log_by_frame_id.keys()), max(cls.log_by_frame_id.keys()) |     cnt = len(ts[cam]['t']) | ||||||
|     del cls.log_by_frame_id[mn] |     assert expected_frames*0.8 < cnt < expected_frames*1.2, f"unexpected frame count {cam}: {expected_frames=}, got {cnt}" | ||||||
|     del cls.log_by_frame_id[mx] | 
 | ||||||
|  |     dts = np.abs(np.diff([ts[cam]['timestampSof']/1e6]) - 1000/SERVICE_LIST[cam].frequency) | ||||||
|  |     assert (dts < 1.0).all(), f"{cam} dts(ms) out of spec: max diff {dts.max()}, 99 percentile {np.percentile(dts, 99)}" | ||||||
|  |   return ts | ||||||
| 
 | 
 | ||||||
| @pytest.mark.tici | @pytest.mark.tici | ||||||
| class TestCamerad: | class TestCamerad: | ||||||
|   @classmethod |   def test_frame_skips(self, logs): | ||||||
|   def setup_class(cls): |     for c in CAMERAS: | ||||||
|     setup_camerad(cls) |       assert set(np.diff(logs[c]['frameId'])) == {1, }, f"{c} has frame skips" | ||||||
| 
 | 
 | ||||||
|   def test_frame_skips(self): |   def test_frame_sync(self, logs): | ||||||
|     skips = {} |     n = range(len(logs['roadCameraState']['t'][:-10])) | ||||||
|     frame_ids = self.log_by_frame_id.keys() | 
 | ||||||
|     for frame_id in range(min(frame_ids), max(frame_ids)): |     frame_ids = {i: [logs[cam]['frameId'][i] for cam in CAMERAS] for i in n} | ||||||
|       seen_cams = [msg.which() for msg in self.log_by_frame_id[frame_id]] |     assert all(len(set(v)) == 1 for v in frame_ids.values()), "frame IDs not aligned" | ||||||
|       skip_cams = set(CAMERAS) - set(seen_cams) | 
 | ||||||
|       if len(skip_cams): |     frame_times = {i: [logs[cam]['timestampSof'][i] for cam in CAMERAS] for i in n} | ||||||
|         skips[frame_id] = skip_cams |     diffs = {i: (max(ts) - min(ts))/1e6 for i, ts in frame_times.items()} | ||||||
|     assert len(skips) == 0, f"Found frame skips, missing cameras for the following frames: {skips}" | 
 | ||||||
| 
 |     laggy_frames = {k: v for k, v in diffs.items() if v > 1.1} | ||||||
|   def test_frame_sync(self): |  | ||||||
|     frame_times = {frame_id: [getattr(m, m.which()).timestampSof for m in msgs] for frame_id, msgs in self.log_by_frame_id.items()} |  | ||||||
|     diffs = {frame_id: (max(ts) - min(ts))/1e6 for frame_id, ts in frame_times.items()} |  | ||||||
| 
 |  | ||||||
|     def get_desc(fid, diff): |  | ||||||
|       cam_times = [(m.which(), getattr(m, m.which()).timestampSof/1e6) for m in self.log_by_frame_id[fid]] |  | ||||||
|       return (diff, cam_times) |  | ||||||
|     laggy_frames = {k: get_desc(k, v) for k, v in diffs.items() if v > LAG_FRAME_TOLERANCE[self.sensor_type]} |  | ||||||
|     assert len(laggy_frames) == 0, f"Frames not synced properly: {laggy_frames=}" |     assert len(laggy_frames) == 0, f"Frames not synced properly: {laggy_frames=}" | ||||||
|  | 
 | ||||||
|  |   @pytest.mark.skip("TODO: enable this") | ||||||
|  |   def test_stress_test(self, logs): | ||||||
|  |     os.environ['SPECTRA_STRESS_TEST'] = '1' | ||||||
|  |     run_and_log(["camerad", ], CAMERAS, 5) | ||||||
|  | |||||||
					Loading…
					
					
				
		Reference in new issue