#!/usr/bin/env python3 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								import  argparse 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  concurrent . futures 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  os 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								import  sys 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  collections  import  defaultdict 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  tqdm  import  tqdm 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  typing  import  Any 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . car . car_helpers  import  interface_names 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . tools . lib . openpilotci  import  get_url ,  upload_file 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								from  openpilot . selfdrive . test . process_replay . compare_logs  import  compare_logs ,  format_diff 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . selfdrive . test . process_replay . process_replay  import  CONFIGS ,  PROC_REPLAY_DIR ,  FAKEDATA ,  check_openpilot_enabled ,  replay_process 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . system . version  import  get_commit 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . tools . lib . filereader  import  FileReader 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . tools . lib . logreader  import  LogReader 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								from  openpilot . tools . lib . helpers  import  save_log 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
											 
										
											
												Live torque (#25456)
* wip torqued
* add basic logic
* setup in manager
* check sanity and publish msg
* add first order filter to outputs
* wire up controlsd, and update gains
* rename intercept to offset
* add cloudlog, live values are not updated
* fix bugs, do not reset points for now
* fix crashes
* rename to main
* fix bugs, works offline
* fix float in cereal bug
* add latacc filter
* randomly choose points, approx for iid
* add variable decay
* local param to capnp instead of dict
* verify works in replay
* use torqued output in controlsd
* use in controlsd; use points from past routes
* controlsd bugfix
* filter before updating gains, needs to be replaced
* save all points to ensure smooth transition across routes, revert friction factor to 1.5
* add filters to prevent noisy low-speed data points; improve fit sanity
* add engaged buffer
* revert lat_acc thresh
* use paramsd realtime process config
* make latacc-to-torque generic, and overrideable
* move freq to 4Hz, avoid storing in np.array, don't publish points in the message
* float instead of np
* remove constant while storing pts
* rename slope, offset to lat_accet_factor, offset
* resolve issues
* use camelcase in all capnp params
* use camelcase everywhere
* reduce latacc threshold or sanity, add car_sane todo, save points properly
* add and check tag
* write param to disk at end of route
* remove args
* rebase op, cereal
* save on exit
* restore default handler
* cpu usage check
* add to process replay
* handle reset better, reduce unnecessary computation
* always publish raw values - useful for debug
* regen routes
* update refs
* checks on cache restore
* check tuning vals too
* clean that up
* reduce cpu usage
* reduce cpu usage by 75%
* cleanup
* optimize further
* handle reset condition better, don't put points in init, use only in corolla
* bump cereal after rebasing
* update refs
* Update common/params.cc
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
* remove unnecessary checks
* Update RELEASES.md
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
											 
										 
										
											3 years ago 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								source_segments  =  [ 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " BODY " ,  " 937ccb7243511b65|2022-05-24--16-03-09--1 " ) ,         # COMMA.BODY 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " HYUNDAI " ,  " 02c45f73a2e5c6e9|2021-01-01--19-08-22--1 " ) ,      # HYUNDAI.SONATA 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " HYUNDAI2 " ,  " d545129f3ca90f28|2022-11-07--20-43-08--3 " ) ,     # HYUNDAI.KIA_EV6 (+ QCOM GPS) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " TOYOTA " ,  " 0982d79ebb0de295|2021-01-04--17-13-21--13 " ) ,      # TOYOTA.PRIUS 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " TOYOTA2 " ,  " 0982d79ebb0de295|2021-01-03--20-03-36--6 " ) ,      # TOYOTA.RAV4 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " TOYOTA3 " ,  " f7d7e3538cda1a2a|2021-08-16--08-55-34--6 " ) ,      # TOYOTA.COROLLA_TSS2 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " HONDA " ,  " eb140f119469d9ab|2021-06-12--10-46-24--27 " ) ,       # HONDA.CIVIC (NIDEC) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " HONDA2 " ,  " 7d2244f34d1bbcda|2021-06-25--12-25-37--26 " ) ,      # HONDA.ACCORD (BOSCH) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " CHRYSLER " ,  " 4deb27de11bee626|2021-02-20--11-28-55--8 " ) ,     # CHRYSLER.PACIFICA_2018_HYBRID 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " RAM " ,  " 17fc16d840fe9d21|2023-04-26--13-28-44--5 " ) ,          # CHRYSLER.RAM_1500 
  
						 
					
						
							
								
									
										
											 
										
											
												Live torque (#25456)
* wip torqued
* add basic logic
* setup in manager
* check sanity and publish msg
* add first order filter to outputs
* wire up controlsd, and update gains
* rename intercept to offset
* add cloudlog, live values are not updated
* fix bugs, do not reset points for now
* fix crashes
* rename to main
* fix bugs, works offline
* fix float in cereal bug
* add latacc filter
* randomly choose points, approx for iid
* add variable decay
* local param to capnp instead of dict
* verify works in replay
* use torqued output in controlsd
* use in controlsd; use points from past routes
* controlsd bugfix
* filter before updating gains, needs to be replaced
* save all points to ensure smooth transition across routes, revert friction factor to 1.5
* add filters to prevent noisy low-speed data points; improve fit sanity
* add engaged buffer
* revert lat_acc thresh
* use paramsd realtime process config
* make latacc-to-torque generic, and overrideable
* move freq to 4Hz, avoid storing in np.array, don't publish points in the message
* float instead of np
* remove constant while storing pts
* rename slope, offset to lat_accet_factor, offset
* resolve issues
* use camelcase in all capnp params
* use camelcase everywhere
* reduce latacc threshold or sanity, add car_sane todo, save points properly
* add and check tag
* write param to disk at end of route
* remove args
* rebase op, cereal
* save on exit
* restore default handler
* cpu usage check
* add to process replay
* handle reset better, reduce unnecessary computation
* always publish raw values - useful for debug
* regen routes
* update refs
* checks on cache restore
* check tuning vals too
* clean that up
* reduce cpu usage
* reduce cpu usage by 75%
* cleanup
* optimize further
* handle reset condition better, don't put points in init, use only in corolla
* bump cereal after rebasing
* update refs
* Update common/params.cc
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
* remove unnecessary checks
* Update RELEASES.md
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
											 
										 
										
											3 years ago 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " SUBARU " ,  " 341dccd5359e3c97|2022-09-12--10-35-33--3 " ) ,       # SUBARU.OUTBACK 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " GM " ,  " 0c58b6a25109da2b|2021-02-23--16-35-50--11 " ) ,          # GM.VOLT 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " GM2 " ,  " 376bf99325883932|2022-10-27--13-41-22--1 " ) ,          # GM.BOLT_EUV 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " NISSAN " ,  " 35336926920f3571|2021-02-12--18-38-48--46 " ) ,      # NISSAN.XTRAIL 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " VOLKSWAGEN " ,  " de9592456ad7d144|2021-06-29--11-00-15--6 " ) ,   # VOLKSWAGEN.GOLF 
  
						 
					
						
							
								
									
										
											 
										
											
												Live torque (#25456)
* wip torqued
* add basic logic
* setup in manager
* check sanity and publish msg
* add first order filter to outputs
* wire up controlsd, and update gains
* rename intercept to offset
* add cloudlog, live values are not updated
* fix bugs, do not reset points for now
* fix crashes
* rename to main
* fix bugs, works offline
* fix float in cereal bug
* add latacc filter
* randomly choose points, approx for iid
* add variable decay
* local param to capnp instead of dict
* verify works in replay
* use torqued output in controlsd
* use in controlsd; use points from past routes
* controlsd bugfix
* filter before updating gains, needs to be replaced
* save all points to ensure smooth transition across routes, revert friction factor to 1.5
* add filters to prevent noisy low-speed data points; improve fit sanity
* add engaged buffer
* revert lat_acc thresh
* use paramsd realtime process config
* make latacc-to-torque generic, and overrideable
* move freq to 4Hz, avoid storing in np.array, don't publish points in the message
* float instead of np
* remove constant while storing pts
* rename slope, offset to lat_accet_factor, offset
* resolve issues
* use camelcase in all capnp params
* use camelcase everywhere
* reduce latacc threshold or sanity, add car_sane todo, save points properly
* add and check tag
* write param to disk at end of route
* remove args
* rebase op, cereal
* save on exit
* restore default handler
* cpu usage check
* add to process replay
* handle reset better, reduce unnecessary computation
* always publish raw values - useful for debug
* regen routes
* update refs
* checks on cache restore
* check tuning vals too
* clean that up
* reduce cpu usage
* reduce cpu usage by 75%
* cleanup
* optimize further
* handle reset condition better, don't put points in init, use only in corolla
* bump cereal after rebasing
* update refs
* Update common/params.cc
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
* remove unnecessary checks
* Update RELEASES.md
Co-authored-by: Adeeb Shihadeh <adeebshihadeh@gmail.com>
											 
										 
										
											3 years ago 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " MAZDA " ,  " bd6a637565e91581|2021-10-30--15-14-53--4 " ) ,        # MAZDA.CX9_2021 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " FORD " ,  " 54827bf84c38b14f|2023-01-26--21-59-07--4 " ) ,         # FORD.BRONCO_SPORT_MK1 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  # Enable when port is tested and dashcamOnly is no longer set 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  #("TESLA", "bb50caf5f0945ab1|2021-06-19--17-20-18--3"),      # TESLA.AP2_MODELS 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  #("VOLKSWAGEN2", "3cfdec54aa035f3f|2022-07-19--23-45-10--2"),  # VOLKSWAGEN.PASSAT_NMS 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								] 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								segments  =  [ 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  ( " BODY " ,  " regen997DF2697CB|2023-10-30--23-14-29--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " HYUNDAI " ,  " regen2A9D2A8E0B4|2023-10-30--23-13-34--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " HYUNDAI2 " ,  " regen6CA24BC3035|2023-10-30--23-14-28--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " TOYOTA " ,  " regen5C019D76307|2023-10-30--23-13-31--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " TOYOTA2 " ,  " regen5DCADA88A96|2023-10-30--23-14-57--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " TOYOTA3 " ,  " regen7204CA3A498|2023-10-30--23-15-55--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " HONDA " ,  " regen048F8FA0B24|2023-10-30--23-15-53--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " HONDA2 " ,  " regen7D2D3F82D5B|2023-10-30--23-15-55--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " CHRYSLER " ,  " regen7125C42780C|2023-10-30--23-16-21--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " RAM " ,  " regen2731F3213D2|2023-10-30--23-18-11--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " SUBARU " ,  " regen86E4C1B4DDD|2023-10-30--23-18-14--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " GM " ,  " regenF6393D64745|2023-10-30--23-17-18--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " GM2 " ,  " regen220F830C05B|2023-10-30--23-18-39--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " NISSAN " ,  " regen4F671F7C435|2023-10-30--23-18-40--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " VOLKSWAGEN " ,  " regen8BDFE7307A0|2023-10-30--23-19-36--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " MAZDA " ,  " regen2E9F1A15FD5|2023-10-30--23-20-36--0 " ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ( " FORD " ,  " regen6D39E54606E|2023-10-30--23-20-54--0 " ) , 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								] 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								# dashcamOnly makes don't need to be tested until a full port is done 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								excluded_interfaces  =  [ " mock " ,  " tesla " ] 
 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								BASE_URL  =  " https://commadataci.blob.core.windows.net/openpilotci/ " 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								REF_COMMIT_FN  =  os . path . join ( PROC_REPLAY_DIR ,  " ref_commit " ) 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								EXCLUDED_PROCS  =  { " modeld " ,  " dmonitoringmodeld " } 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  run_test_process ( data ) : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  segment ,  cfg ,  args ,  cur_log_fn ,  ref_log_path ,  lr_dat  =  data 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  res  =  None 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  not  args . upload_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    lr  =  LogReader . from_bytes ( lr_dat ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    res ,  log_msgs  =  test_process ( cfg ,  lr ,  segment ,  ref_log_path ,  cur_log_fn ,  args . ignore_fields ,  args . ignore_msgs ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    # save logs so we can upload when updating refs 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    save_log ( cur_log_fn ,  log_msgs ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  args . update_refs  or  args . upload_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( f ' Uploading:  { os . path . basename ( cur_log_fn ) } ' ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    assert  os . path . exists ( cur_log_fn ) ,  f " Cannot find log to upload:  { cur_log_fn } " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    upload_file ( cur_log_fn ,  os . path . basename ( cur_log_fn ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    os . remove ( cur_log_fn ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  return  ( segment ,  cfg . proc_name ,  res ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  get_log_data ( segment ) : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  r ,  n  =  segment . rsplit ( " -- " ,  1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  with  FileReader ( get_url ( r ,  n ) )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  ( segment ,  f . read ( ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								def  test_process ( cfg ,  lr ,  segment ,  ref_log_path ,  new_log_path ,  ignore_fields = None ,  ignore_msgs = None ) : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ignore_fields  is  None : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ignore_fields  =  [ ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  ignore_msgs  is  None : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ignore_msgs  =  [ ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ref_log_msgs  =  list ( LogReader ( ref_log_path ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  try : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    log_msgs  =  replay_process ( cfg ,  lr ,  disable_progress = True ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  except  Exception  as  e : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    raise  Exception ( " failed on segment:  "  +  segment )  from  e 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  # check to make sure openpilot is engaged in the route 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  cfg . proc_name  ==  " controlsd " : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  not  check_openpilot_enabled ( log_msgs ) : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      # FIXME: these segments should work, but the replay enabling logic is too brittle 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  segment  not  in  ( " regen6CA24BC3035|2023-10-30--23-14-28--0 " ,  " regen7D2D3F82D5B|2023-10-30--23-15-55--0 " ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return  f " Route did not enable at all or for long enough:  { new_log_path } " ,  log_msgs 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  try : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  compare_logs ( ref_log_msgs ,  log_msgs ,  ignore_fields  +  cfg . ignore ,  ignore_msgs ,  cfg . tolerance ) ,  log_msgs 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  except  Exception  as  e : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    return  str ( e ) ,  log_msgs 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								if  __name__  ==  " __main__ " : 
 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  all_cars  =  { car  for  car ,  _  in  segments } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  all_procs  =  { cfg . proc_name  for  cfg  in  CONFIGS  if  cfg . proc_name  not  in  EXCLUDED_PROCS } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cpu_count  =  os . cpu_count ( )  or  1 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser  =  argparse . ArgumentParser ( description = " Regression test to identify changes in a process ' s output " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --whitelist-procs " ,  type = str ,  nargs = " * " ,  default = all_procs , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Whitelist given processes from the test (e.g. controlsd) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --whitelist-cars " ,  type = str ,  nargs = " * " ,  default = all_cars , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Whitelist given cars from the test (e.g. HONDA) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --blacklist-procs " ,  type = str ,  nargs = " * " ,  default = [ ] , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Blacklist given processes from the test (e.g. controlsd) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --blacklist-cars " ,  type = str ,  nargs = " * " ,  default = [ ] , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Blacklist given cars from the test (e.g. HONDA) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --ignore-fields " ,  type = str ,  nargs = " * " ,  default = [ ] , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Extra fields or msgs to ignore (e.g. carState.events) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --ignore-msgs " ,  type = str ,  nargs = " * " ,  default = [ ] , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Msgs to ignore (e.g. carEvents) " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --update-refs " ,  action = " store_true " , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Updates reference logs using current commit " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " --upload-only " ,  action = " store_true " , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Skips testing processes and uploads logs from previous test run " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  parser . add_argument ( " -j " ,  " --jobs " ,  type = int ,  default = max ( cpu_count  -  2 ,  1 ) , 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                      help = " Max amount of parallel jobs " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  args  =  parser . parse_args ( ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  tested_procs  =  set ( args . whitelist_procs )  -  set ( args . blacklist_procs ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  tested_cars  =  set ( args . whitelist_cars )  -  set ( args . blacklist_cars ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  tested_cars  =  { c . upper ( )  for  c  in  tested_cars } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  full_test  =  ( tested_procs  ==  all_procs )  and  ( tested_cars  ==  all_cars )  and  all ( len ( x )  ==  0  for  x  in  ( args . ignore_fields ,  args . ignore_msgs ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  upload  =  args . update_refs  or  args . upload_only 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  os . makedirs ( os . path . dirname ( FAKEDATA ) ,  exist_ok = True ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  upload : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    assert  full_test ,  " Need to run full test when updating refs " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  try : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    with  open ( REF_COMMIT_FN )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      ref_commit  =  f . read ( ) . strip ( ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  except  FileNotFoundError : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( " Couldn ' t find reference commit " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    sys . exit ( 1 ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  cur_commit  =  get_commit ( ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  if  not  cur_commit : 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								    raise  Exception ( " Couldn ' t get current commit " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  print ( f " ***** testing against commit  { ref_commit }  ***** " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  # check to make sure all car brands are tested 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  full_test : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    untested  =  ( set ( interface_names )  -  set ( excluded_interfaces ) )  -  { c . lower ( )  for  c  in  tested_cars } 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    assert  len ( untested )  ==  0 ,  f " Cars missing routes:  { str ( untested ) } " 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  log_paths :  defaultdict [ str ,  dict [ str ,  dict [ str ,  str ] ] ]  =  defaultdict ( lambda :  defaultdict ( dict ) ) 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								  with  concurrent . futures . ProcessPoolExecutor ( max_workers = args . jobs )  as  pool : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  not  args . upload_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      download_segments  =  [ seg  for  car ,  seg  in  segments  if  car  in  tested_cars ] 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      log_data :  dict [ str ,  LogReader ]  =  { } 
  
						 
					
						
							
								
							 
							
								
									
										 
								
							 
							
								 
							
							
								      p1  =  pool . map ( get_log_data ,  download_segments ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      for  segment ,  lr  in  tqdm ( p1 ,  desc = " Getting Logs " ,  total = len ( download_segments ) ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        log_data [ segment ]  =  lr 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    pool_args :  Any  =  [ ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  car_brand ,  segment  in  segments : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  car_brand  not  in  tested_cars : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      for  cfg  in  CONFIGS : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  cfg . proc_name  not  in  tested_procs : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          continue 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        cur_log_fn  =  os . path . join ( FAKEDATA ,  f " { segment } _ { cfg . proc_name } _ { cur_commit } .bz2 " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if  args . update_refs :   # reference logs will not exist if routes were just regenerated 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          ref_log_path  =  get_url ( * segment . rsplit ( " -- " ,  1 ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          ref_log_fn  =  os . path . join ( FAKEDATA ,  f " { segment } _ { cfg . proc_name } _ { ref_commit } .bz2 " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          ref_log_path  =  ref_log_fn  if  os . path . exists ( ref_log_fn )  else  BASE_URL  +  os . path . basename ( ref_log_fn ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        dat  =  None  if  args . upload_only  else  log_data [ segment ] 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        pool_args . append ( ( segment ,  cfg ,  args ,  cur_log_fn ,  ref_log_path ,  dat ) ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        log_paths [ segment ] [ cfg . proc_name ] [ ' ref ' ]  =  ref_log_path 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        log_paths [ segment ] [ cfg . proc_name ] [ ' new ' ]  =  cur_log_fn 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    results :  Any  =  defaultdict ( dict ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    p2  =  pool . map ( run_test_process ,  pool_args ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    for  ( segment ,  proc ,  result )  in  tqdm ( p2 ,  desc = " Running Tests " ,  total = len ( pool_args ) ) : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      if  not  args . upload_only : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        results [ segment ] [ proc ]  =  result 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  diff_short ,  diff_long ,  failed  =  format_diff ( results ,  log_paths ,  ref_commit ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  if  not  upload : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    with  open ( os . path . join ( PROC_REPLAY_DIR ,  " diff.txt " ) ,  " w " )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      f . write ( diff_long ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( diff_short ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    if  failed : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      print ( " TEST FAILED " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      print ( " \n \n To push the new reference logs for this commit run: " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      print ( " ./test_processes.py --upload-only " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      print ( " TEST SUCCEEDED " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  else : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    with  open ( REF_COMMIT_FN ,  " w " )  as  f : 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      f . write ( cur_commit ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    print ( f " \n \n Updated reference logs for commit:  { cur_commit } " ) 
  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  sys . exit ( int ( failed ) )