#!/usr/bin/env python3.7
import json
import os
import io
import random
import re
import select
import subprocess
import socket
import time
import threading
import traceback
import base64
import requests
import queue
from functools import partial
from jsonrpc import JSONRPCResponseManager , dispatcher
from websocket import create_connection , WebSocketTimeoutException , ABNF
from selfdrive . loggerd . config import ROOT
import selfdrive . messaging as messaging
from common . api import Api
from common . params import Params
from selfdrive . services import service_list
from selfdrive . swaglog import cloudlog
from functools import reduce
ATHENA_HOST = os . getenv ( ' ATHENA_HOST ' , ' wss://athena.comma.ai ' )
HANDLER_THREADS = os . getenv ( ' HANDLER_THREADS ' , 4 )
LOCAL_PORT_WHITELIST = set ( [ 8022 ] )
dispatcher [ " echo " ] = lambda s : s
payload_queue = queue . Queue ( )
response_queue = queue . Queue ( )
def handle_long_poll ( ws ) :
end_event = threading . Event ( )
threads = [
threading . Thread ( target = ws_recv , args = ( ws , end_event ) ) ,
threading . Thread ( target = ws_send , args = ( ws , end_event ) )
] + [
threading . Thread ( target = jsonrpc_handler , args = ( end_event , ) )
for x in range ( HANDLER_THREADS )
]
for thread in threads :
thread . start ( )
try :
while not end_event . is_set ( ) :
time . sleep ( 0.1 )
except ( KeyboardInterrupt , SystemExit ) :
end_event . set ( )
raise
finally :
for i , thread in enumerate ( threads ) :
thread . join ( )
def jsonrpc_handler ( end_event ) :
dispatcher [ " startLocalProxy " ] = partial ( startLocalProxy , end_event )
while not end_event . is_set ( ) :
try :
data = payload_queue . get ( timeout = 1 )
response = JSONRPCResponseManager . handle ( data , dispatcher )
response_queue . put_nowait ( response )
except queue . Empty :
pass
except Exception as e :
cloudlog . exception ( " athena jsonrpc handler failed " )
traceback . print_exc ( )
response_queue . put_nowait ( json . dumps ( { " error " : str ( e ) } ) )
# security: user should be able to request any message from their car
# TODO: add service to, for example, start visiond and take a picture
@dispatcher . add_method
def getMessage ( service = None , timeout = 1000 ) :
if service is None or service not in service_list :
raise Exception ( " invalid service " )
socket = messaging . sub_sock ( service )
socket . setTimeout ( timeout )
ret = messaging . recv_one ( socket )
return ret . to_dict ( )
@dispatcher . add_method
def listDataDirectory ( ) :
files = [ os . path . relpath ( os . path . join ( dp , f ) , ROOT ) for dp , dn , fn in os . walk ( ROOT ) for f in fn ]
return files
@dispatcher . add_method
def uploadFileToUrl ( fn , url , headers ) :
if len ( fn ) == 0 or fn [ 0 ] == ' / ' or ' .. ' in fn :
return 500
with open ( os . path . join ( ROOT , fn ) , " rb " ) as f :
ret = requests . put ( url , data = f , headers = headers , timeout = 10 )
return ret . status_code
def startLocalProxy ( global_end_event , remote_ws_uri , local_port ) :
try :
if local_port not in LOCAL_PORT_WHITELIST :
raise Exception ( " Requested local port not whitelisted " )
params = Params ( )
dongle_id = params . get ( " DongleId " ) . decode ( ' utf8 ' )
identity_token = Api ( dongle_id ) . get_token ( )
ws = create_connection ( remote_ws_uri ,
cookie = " jwt= " + identity_token ,
enable_multithread = True )
ssock , csock = socket . socketpair ( )
local_sock = socket . socket ( socket . AF_INET , socket . SOCK_STREAM )
local_sock . connect ( ( ' 127.0.0.1 ' , local_port ) )
local_sock . setblocking ( 0 )
proxy_end_event = threading . Event ( )
threads = [
threading . Thread ( target = ws_proxy_recv , args = ( ws , local_sock , ssock , proxy_end_event , global_end_event ) ) ,
threading . Thread ( target = ws_proxy_send , args = ( ws , local_sock , csock , proxy_end_event ) )
]
for thread in threads :
thread . start ( )
return { " success " : 1 }
except Exception as e :
traceback . print_exc ( )
raise e
@dispatcher . add_method
def getPublicKey ( ) :
if not os . path . isfile ( ' /persist/comma/id_rsa.pub ' ) :
return None
with open ( ' /persist/comma/id_rsa.pub ' , ' r ' ) as f :
return f . read ( )
@dispatcher . add_method
def getSshAuthorizedKeys ( ) :
return Params ( ) . get ( " GithubSshKeys " , encoding = ' utf8 ' ) or ' '
@dispatcher . add_method
def getSimInfo ( ) :
sim_state = subprocess . check_output ( [ ' getprop ' , ' gsm.sim.state ' ] , encoding = ' utf8 ' ) . strip ( ) . split ( ' , ' ) # pylint: disable=unexpected-keyword-arg
network_type = subprocess . check_output ( [ ' getprop ' , ' gsm.network.type ' ] , encoding = ' utf8 ' ) . strip ( ) . split ( ' , ' ) # pylint: disable=unexpected-keyword-arg
mcc_mnc = subprocess . check_output ( [ ' getprop ' , ' gsm.sim.operator.numeric ' ] , encoding = ' utf8 ' ) . strip ( ) or None # pylint: disable=unexpected-keyword-arg
sim_id_aidl_out = subprocess . check_output ( [ ' service ' , ' call ' , ' iphonesubinfo ' , ' 11 ' ] , encoding = ' utf8 ' ) # pylint: disable=unexpected-keyword-arg
sim_id_aidl_lines = sim_id_aidl_out . split ( ' \n ' )
if len ( sim_id_aidl_lines ) > 3 :
sim_id_lines = sim_id_aidl_lines [ 1 : 4 ]
sim_id_fragments = [ re . search ( r " ' ([0-9 \ .]+) ' " , line ) . group ( 1 ) for line in sim_id_lines ]
sim_id = reduce ( lambda frag1 , frag2 : frag1 . replace ( ' . ' , ' ' ) + frag2 . replace ( ' . ' , ' ' ) , sim_id_fragments )
else :
sim_id = None
return {
' sim_id ' : sim_id ,
' mcc_mnc ' : mcc_mnc ,
' network_type ' : network_type ,
' sim_state ' : sim_state
}
@dispatcher . add_method
def takeSnapshot ( ) :
from selfdrive . visiond . snapshot . snapshot import snapshot , jpeg_write
ret = snapshot ( )
if ret is not None :
def b64jpeg ( x ) :
if x is not None :
f = io . BytesIO ( )
jpeg_write ( f , x )
return base64 . b64encode ( f . getvalue ( ) ) . decode ( " utf-8 " )
else :
return None
return { ' jpegBack ' : b64jpeg ( ret [ 0 ] ) ,
' jpegFront ' : b64jpeg ( ret [ 1 ] ) }
else :
raise Exception ( " not available while visiond is started " )
def ws_proxy_recv ( ws , local_sock , ssock , end_event , global_end_event ) :
while not ( end_event . is_set ( ) or global_end_event . is_set ( ) ) :
try :
data = ws . recv ( )
local_sock . sendall ( data )
except WebSocketTimeoutException :
pass
except Exception :
cloudlog . exception ( " athenad.ws_proxy_recv.exception " )
traceback . print_exc ( )
break
ssock . close ( )
end_event . set ( )
def ws_proxy_send ( ws , local_sock , signal_sock , end_event ) :
while not end_event . is_set ( ) :
try :
r , _ , _ = select . select ( ( local_sock , signal_sock ) , ( ) , ( ) )
if r :
if r [ 0 ] . fileno ( ) == signal_sock . fileno ( ) :
# got end signal from ws_proxy_recv
end_event . set ( )
break
data = local_sock . recv ( 4096 )
if not data :
# local_sock is dead
end_event . set ( )
break
ws . send ( data , ABNF . OPCODE_BINARY )
except Exception :
cloudlog . exception ( " athenad.ws_proxy_send.exception " )
traceback . print_exc ( )
end_event . set ( )
def ws_recv ( ws , end_event ) :
while not end_event . is_set ( ) :
try :
data = ws . recv ( )
payload_queue . put_nowait ( data )
except WebSocketTimeoutException :
pass
except Exception :
cloudlog . exception ( " athenad.ws_recv.exception " )
traceback . print_exc ( )
end_event . set ( )
def ws_send ( ws , end_event ) :
while not end_event . is_set ( ) :
try :
response = response_queue . get ( timeout = 1 )
ws . send ( response . json )
except queue . Empty :
pass
except Exception :
cloudlog . exception ( " athenad.ws_send.exception " )
traceback . print_exc ( )
end_event . set ( )
def backoff ( retries ) :
return random . randrange ( 0 , min ( 128 , int ( 2 * * retries ) ) )
def main ( gctx = None ) :
params = Params ( )
dongle_id = params . get ( " DongleId " ) . decode ( ' utf-8 ' )
ws_uri = ATHENA_HOST + " /ws/v2/ " + dongle_id
api = Api ( dongle_id )
conn_retries = 0
while 1 :
try :
ws = create_connection ( ws_uri ,
cookie = " jwt= " + api . get_token ( ) ,
enable_multithread = True )
cloudlog . event ( " athenad.main.connected_ws " , ws_uri = ws_uri )
ws . settimeout ( 1 )
conn_retries = 0
handle_long_poll ( ws )
except ( KeyboardInterrupt , SystemExit ) :
break
except Exception :
cloudlog . exception ( " athenad.main.exception " )
conn_retries + = 1
traceback . print_exc ( )
time . sleep ( backoff ( conn_retries ) )
if __name__ == " __main__ " :
main ( )