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.
		
		
		
		
		
			
		
			
				
					
					
						
							112 lines
						
					
					
						
							3.2 KiB
						
					
					
				
			
		
		
	
	
							112 lines
						
					
					
						
							3.2 KiB
						
					
					
				| #!/usr/bin/env python3
 | |
| import subprocess
 | |
| import threading
 | |
| import time
 | |
| import unittest
 | |
| from typing import Callable, cast, Optional
 | |
| from unittest.mock import MagicMock
 | |
| 
 | |
| from common.params import Params
 | |
| from common.timeout import Timeout
 | |
| from selfdrive.athena import athenad
 | |
| from selfdrive.manager.helpers import write_onroad_params
 | |
| from system.hardware import TICI
 | |
| 
 | |
| 
 | |
| def wifi_radio(on: bool) -> None:
 | |
|   if not TICI:
 | |
|     return
 | |
|   print(f"wifi {'on' if on else 'off'}")
 | |
|   subprocess.run(["nmcli", "radio", "wifi", "on" if on else "off"], check=True)
 | |
| 
 | |
| 
 | |
| class TestAthenadPing(unittest.TestCase):
 | |
|   params: Params
 | |
|   dongle_id: str
 | |
| 
 | |
|   athenad: threading.Thread
 | |
|   exit_event: threading.Event
 | |
| 
 | |
|   _create_connection: Callable
 | |
| 
 | |
|   def _get_ping_time(self) -> Optional[str]:
 | |
|     return cast(Optional[str], self.params.get("LastAthenaPingTime", encoding="utf-8"))
 | |
| 
 | |
|   def _clear_ping_time(self) -> None:
 | |
|     self.params.remove("LastAthenaPingTime")
 | |
| 
 | |
|   def _received_ping(self) -> bool:
 | |
|     return self._get_ping_time() is not None
 | |
| 
 | |
|   @classmethod
 | |
|   def setUpClass(cls) -> None:
 | |
|     cls.params = Params()
 | |
|     cls.dongle_id = cls.params.get("DongleId", encoding="utf-8")
 | |
|     cls._create_connection = athenad.create_connection
 | |
|     athenad.create_connection = MagicMock(wraps=cls._create_connection)
 | |
| 
 | |
|   @classmethod
 | |
|   def tearDownClass(cls) -> None:
 | |
|     wifi_radio(True)
 | |
|     athenad.create_connection = cls._create_connection
 | |
| 
 | |
|   def setUp(self) -> None:
 | |
|     wifi_radio(True)
 | |
|     self._clear_ping_time()
 | |
| 
 | |
|     self.exit_event = threading.Event()
 | |
|     self.athenad = threading.Thread(target=athenad.main, args=(self.exit_event,))
 | |
| 
 | |
|     athenad.create_connection.reset_mock()
 | |
| 
 | |
|   def tearDown(self) -> None:
 | |
|     if self.athenad.is_alive():
 | |
|       self.exit_event.set()
 | |
|       self.athenad.join()
 | |
| 
 | |
|   def assertTimeout(self, reconnect_time: float) -> None:
 | |
|     self.athenad.start()
 | |
| 
 | |
|     time.sleep(1)
 | |
|     athenad.create_connection.assert_called_once()
 | |
|     athenad.create_connection.reset_mock()
 | |
| 
 | |
|     # check normal behaviour
 | |
|     with self.subTest("Wi-Fi: receives ping"), Timeout(70, "no ping received"):
 | |
|       while not self._received_ping():
 | |
|         time.sleep(0.1)
 | |
|       print("ping received")
 | |
| 
 | |
|     athenad.create_connection.assert_not_called()
 | |
| 
 | |
|     # websocket should attempt reconnect after short time
 | |
|     with self.subTest("LTE: attempt reconnect"):
 | |
|       wifi_radio(False)
 | |
|       print("waiting for reconnect attempt")
 | |
|       start_time = time.monotonic()
 | |
|       with Timeout(reconnect_time, "no reconnect attempt"):
 | |
|         while not athenad.create_connection.called:
 | |
|           time.sleep(0.1)
 | |
|         print(f"reconnect attempt after {time.monotonic() - start_time:.2f}s")
 | |
| 
 | |
|     self._clear_ping_time()
 | |
| 
 | |
|     # check ping received after reconnect
 | |
|     with self.subTest("LTE: receives ping"), Timeout(70, "no ping received"):
 | |
|       while not self._received_ping():
 | |
|         time.sleep(0.1)
 | |
|       print("ping received")
 | |
| 
 | |
|   @unittest.skipIf(not TICI, "only run on desk")
 | |
|   def test_offroad(self) -> None:
 | |
|     write_onroad_params(False, self.params)
 | |
|     self.assertTimeout(100)  # expect approx 90s
 | |
| 
 | |
|   @unittest.skipIf(not TICI, "only run on desk")
 | |
|   def test_onroad(self) -> None:
 | |
|     write_onroad_params(True, self.params)
 | |
|     self.assertTimeout(30)  # expect 20-30s
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|   unittest.main()
 | |
| 
 |