# flake8: noqa import os os.environ['FAKEUPLOAD'] = "1" from common.params import Params from common.realtime import sec_since_boot from selfdrive.manager import manager_init, manager_prepare, start_daemon_process from selfdrive.test.helpers import phone_only, with_processes, set_params_enabled import json import requests import signal import subprocess import time # must run first @phone_only def test_manager_prepare(): set_params_enabled() manager_init() manager_prepare() @phone_only @with_processes(['loggerd', 'logmessaged', 'tombstoned', 'proclogd', 'logcatd']) def test_logging(): print("LOGGING IS SET UP") time.sleep(1.0) @phone_only @with_processes(['camerad', 'modeld', 'dmonitoringmodeld']) def test_visiond(): print("VISIOND IS SET UP") time.sleep(5.0) @phone_only @with_processes(['sensord']) def test_sensord(): print("SENSORS ARE SET UP") time.sleep(1.0) @phone_only @with_processes(['ui']) def test_ui(): print("RUNNING UI") time.sleep(1.0) # will have one thing to upload if loggerd ran # TODO: assert it actually uploaded @phone_only @with_processes(['uploader']) def test_uploader(): print("UPLOADER") time.sleep(10.0) @phone_only def test_athena(): print("ATHENA") start = sec_since_boot() start_daemon_process("manage_athenad") params = Params() manage_athenad_pid = params.get("AthenadPid") assert manage_athenad_pid is not None try: os.kill(int(manage_athenad_pid), 0) # process is running except OSError: assert False, "manage_athenad is dead" def expect_athena_starts(timeout=30): now = time.time() athenad_pid = None while athenad_pid is None: try: athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip() return athenad_pid except subprocess.CalledProcessError: if time.time() - now > timeout: assert False, f"Athena did not start within {timeout} seconds" time.sleep(0.5) def athena_post(payload, max_retries=5, wait=5): tries = 0 while 1: try: resp = requests.post( "https://athena.comma.ai/" + params.get("DongleId", encoding="utf-8"), headers={ "Authorization": "JWT " + os.getenv("COMMA_JWT"), "Content-Type": "application/json" }, data=json.dumps(payload), timeout=30 ) resp_json = resp.json() if resp_json.get('error'): raise Exception(resp_json['error']) return resp_json except Exception as e: time.sleep(wait) tries += 1 if tries == max_retries: raise else: print(f'athena_post failed {e}. retrying...') def expect_athena_registers(test_t0): resp = athena_post({ "method": "echo", "params": ["hello"], "id": 0, "jsonrpc": "2.0" }, max_retries=12, wait=5) assert resp.get('result') == "hello", f'Athena failed to register ({resp})' last_pingtime = params.get("LastAthenaPingTime", encoding='utf8') assert last_pingtime, last_pingtime assert ((int(last_pingtime)/1e9) - test_t0) < (sec_since_boot() - test_t0) try: athenad_pid = expect_athena_starts() # kill athenad and ensure it is restarted (check_output will throw if it is not) os.kill(int(athenad_pid), signal.SIGINT) expect_athena_starts() if not os.getenv('COMMA_JWT'): print('WARNING: COMMA_JWT env not set, will not test requests to athena.comma.ai') return expect_athena_registers(start) print("ATHENA: getSimInfo") resp = athena_post({ "method": "getSimInfo", "id": 0, "jsonrpc": "2.0" }) assert resp.get('result'), resp assert 'sim_id' in resp['result'], resp['result'] print("ATHENA: takeSnapshot") resp = athena_post({ "method": "takeSnapshot", "id": 0, "jsonrpc": "2.0" }) assert resp.get('result'), resp assert resp['result']['jpegBack'], resp['result'] @with_processes(["thermald"]) def test_athena_thermal(): print("ATHENA: getMessage(thermal)") resp = athena_post({ "method": "getMessage", "params": {"service": "thermal", "timeout": 5000}, "id": 0, "jsonrpc": "2.0" }) assert resp.get('result'), resp assert resp['result']['thermal'], resp['result'] test_athena_thermal() finally: try: athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip() except subprocess.CalledProcessError: athenad_pid = None try: os.kill(int(manage_athenad_pid), signal.SIGINT) os.kill(int(athenad_pid), signal.SIGINT) except (OSError, TypeError): pass # TODO: re-enable when jenkins test has /data/pythonpath -> /data/openpilot # @phone_only # @with_apks() # def test_apks(): # print("APKS") # time.sleep(14.0)