-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstartGateway.py
executable file
·421 lines (324 loc) · 14.9 KB
/
startGateway.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
#! /usr/bin/python3
#########################################################
# Python3 Script that simulates OBDII, GPS and beacons #
# received data from car on a supposed trip. #
# MOTAM project: https://www.nics.uma.es/projects/motam #
# Created by Manuel Montenegro, Sep 25, 2019. #
#########################################################
import os
import time
import argparse
import threading
import sqlite3
import queue
import json
import socket
import ssl
import subprocess
import base64
from modules import in_Ble4Scanner
from modules import in_Ble5Scanner
from modules import in_InteractiveScanner
from modules import in_ObdGpsBeaconsTrip
from modules import in_ObdGpsInterface
# ==== Global variables ====
# Version of this script
scriptVersion = 3.7
# OBD II and GPS interfaces for real data
realObdGps = False
# OBD II and GPS data can be loaded from DB
obdGpsTrip = False
# Bluetooth beacons can be loaded from session DB
beaconsTrip = False
# OBD II and GPS is connected: read from hardware interface
realObdGps = False
# Bluetooth 4 beacons can be real (RPi Bluetooth interface)
ble4Beacons = False
# Bluetooth 5 beacons can be real (nRF52840 dongle)
ble5Beacons = False
# BLE interactive beacons generated from terminal in real time
interactiveBeacons = False
# obdGpsBeacons alternative saved session trip passed by arguments
obdGpsBeaconsDb = None
# Coordinates given by terminal of BLE interactive beacons
interactiveBeaconsCoord = None
# path of certificate and its key
certRoute = "certs/"
certPath = certRoute+"pasarela_normal.crt"
keyCertPath = certRoute+"pasarela_normal.key"
# path of CA certificate
caCertPath = certRoute+"cacert.crt"
# path of camera shots
shotsRoute = "shots/"
# Seconds after the beacon will be removed from list (beacon is not in range)
beaconThreshold = 1
# default ip and port assigned to the gateway in AVATAR-Gateway connection
gatewayIP = "192.168.0.1"
# default gateway port in AVATAR-Gateway connection
gatewayPort = 4443
# user logged on the system from AVATAR
user = None
# software timer for taking photos every x seconds with RPi camera modules
cameraTimer = None
# User ID and Group ID of Pi user for managging files
uid = 1000
gid = 1000
# ==== Main execution ====
def main():
ble4Thread = None
ble5Thread = None
bleInteractiveThread = None
receiveFromSocketThread = None
obdGpsBeaconsTripThread = None
obdGpsInterfaceThread = None
# threading event used for closing safely the threads when interrupt signal is received
threadStopEvent = threading.Event()
# queue where thread will put data (dictionary format) from sensors or car interfaces (or simulation)
dataQueue = queue.Queue()
# capture command line arguments
setUpArgParser()
if gatewayIP == "192.168.0.1":
subprocess.run(["wpa_cli", "-i", "p2p-dev-wlan0", "p2p_find"], capture_output=True)
subprocess.run(["wpa_cli", "-i", "p2p-dev-wlan0", "p2p_group_add"], capture_output=True)
subprocess.run("wpa_cli -i $(ip -br link | grep -Po 'p2p-wlan0-\\d+') wps_pin any 12345670", shell=True, capture_output=True)
# create SSL socket for communication with AVATAR
sock = createSslSocket ()
#user = "74943620Y"
if realObdGps:
obdGpsInterface = in_ObdGpsInterface.ObdGpsInterface (threadStopEvent, dataQueue)
obdGpsInterfaceThread = obdGpsInterface.run()
if obdGpsTrip:
obdGpsBeaconsTrip = in_ObdGpsBeaconsTrip.ObdGpsBeaconsTrip (threadStopEvent, dataQueue, beaconThreshold, beaconsTrip, obdGpsBeaconsDb)
obdGpsBeaconsTripThread = obdGpsBeaconsTrip.run()
if (ble4Beacons):
ble4Scanner = in_Ble4Scanner.Ble4Scanner (threadStopEvent, dataQueue, beaconThreshold)
ble4Thread = ble4Scanner.run()
if (ble5Beacons):
ble5Scanner = in_Ble5Scanner.Ble5Scanner (threadStopEvent, dataQueue, beaconThreshold)
ble5Thread = ble5Scanner.run()
if (interactiveBeacons):
bleInteractive = in_InteractiveScanner.InteractiveScanner (threadStopEvent, dataQueue, beaconThreshold, interactiveBeaconsCoord)
bleInteractiveThread = bleInteractive.run()
# make folder for user's profile photo and camera shots. Returns path for camera command
takePictureMakeDir ()
# start thread for taking shots of the driver every x seconds
takePictureStartTimer (threadStopEvent)
# start thread that parse and send by socket the collected data
sendDataToAvatarThread = threading.Thread(target=sendDataToAvatar, args=(threadStopEvent, dataQueue, sock))
sendDataToAvatarThread.daemon = True
sendDataToAvatarThread.start()
# start thread for reading data received from socket (like user image)
receiveFromSocketThread = threading.Thread(target=receiveFromSocket, args=(threadStopEvent, sock))
receiveFromSocketThread.daemon = True
receiveFromSocketThread.start()
try:
if obdGpsInterfaceThread != None:
obdGpsInterfaceThread.join()
if obdGpsBeaconsTripThread != None:
obdGpsBeaconsTripThread.join()
if ble4Thread != None:
ble4Thread.join()
if ble5Thread != None:
ble5Thread.join()
if bleInteractiveThread != None:
bleInteractiveThread.join()
if sendDataToAvatarThread != None:
sendDataToAvatarThread.join()
if receiveFromSocketThread != None:
receiveFromSocketThread.join()
# finishing the execution with Ctrl + c
except KeyboardInterrupt:
threadStopEvent.set()
# unknown exception from nobody knows where
except Exception as error:
print ("\r\nUnknown error\r\n ", error)
finally:
# stop taking pictures
if cameraTimer != None:
cameraTimer.cancel()
# Close Wifi-Direct connection
if gatewayIP == "192.168.0.1":
subprocess.run("wpa_cli -i p2p-dev-wlan0 p2p_group_remove $(ip -br link | grep -Po 'p2p-wlan0-\\d+')", shell=True, capture_output=True)
# ==== Functions ====
# create a SSL connection with AVATAR
def createSslSocket ( ):
global user
# TLS configuration for connection
try:
# create the SSL context
SSLcontext = ssl.create_default_context(ssl.Purpose.CLIENT_AUTH)
# server certificate and private key (and its key)
SSLcontext.load_cert_chain(certfile=certPath, keyfile=keyCertPath, password = '123456')
# Certificate Authority
SSLcontext.load_verify_locations(caCertPath)
# certificates are required from the other side of the socket connection
SSLcontext.verify_mode = ssl.CERT_REQUIRED
# create secure socket for data transmission
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# prevent "Address already in use" error
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
# asociate the secure socket with server address an port
sock.bind((gatewayIP, gatewayPort))
# put the secure socket in server mode and only accept 1 connection
sock.listen(1)
print ("Waiting connection...")
# socket accepts the connection
sockConnection, clientAddress = sock.accept()
# wrap socket with SSL layer
sslSockConnection = SSLcontext.wrap_socket(sockConnection, server_side=True)
print ("Connection accepted from", clientAddress[0])
user = sslSockConnection.getpeercert() ["subject"][0][0][1]
print ("Welcome: ", user)
return sslSockConnection
except ssl.SSLError as err:
print (err)
subprocess.run("wpa_cli -i p2p-dev-wlan0 p2p_group_remove $(ip -br link | grep -Po 'p2p-wlan0-\\d+')", shell=True, capture_output=True)
exit()
# if the other side close the socket...
except socket.error:
print("Connection Closed")
subprocess.run("wpa_cli -i p2p-dev-wlan0 p2p_group_remove $(ip -br link | grep -Po 'p2p-wlan0-\\d+')", shell=True, capture_output=True)
exit()
except KeyboardInterrupt:
subprocess.run("wpa_cli -i p2p-dev-wlan0 p2p_group_remove $(ip -br link | grep -Po 'p2p-wlan0-\\d+')", shell=True, capture_output=True)
exit()
# Receives an image (or other data) from AVATAR by socket and store it
def receiveFromSocket (threadStopEvent, sock):
# this will stop the execution with keyboard interrupt
while not threadStopEvent.is_set():
dataRead = sock.recv(1024)
idMessage = int(dataRead.decode("utf-8"))
# User profile photo
if idMessage == 0:
image = b''
dataRead = sock.recv(1024)
imageLength = int(dataRead.decode("utf-8"))
for _ in range (0, imageLength, 1024):
image += sock.recv(1024)
imgdata = base64.b64decode(image.decode("utf-8"))
global shotsRoute
global user
filePath = shotsRoute+user+"/_profile.jpg"
with open(filePath, 'wb') as f:
f.write(imgdata)
# Activate emergency mode
elif idMessage == 1:
dataRead = sock.recv(1024)
messageLength = int(dataRead.decode("utf-8"))
dataRead = sock.recv(1024)
activated = int(dataRead.decode("utf-8"))
print ("\r\nEnabled emergency mode:", activated)
# Activate slow vehicle mode
elif idMessage == 2:
print ("Enabling slow vehicle mode")
# dataRead = sock.recv(1024)
# messageLength = int(dataRead.decode("utf-8"))
# dataRead = sock.recv(1024)
# coordinates = dataRead.decode("utf-8")
# Activate crashed vehicle mode
elif idMessage == 3:
print ("\r\nEnabling crashed vehicle mode")
dataRead = sock.recv(1024)
messageLength = int(dataRead.decode("utf-8"))
dataRead = sock.recv(1024)
coordinates = dataRead.decode("utf-8")
# Make folder for saved camera shots of driver and profile photo from AVATAR
def takePictureMakeDir ():
global shotsRoute
global user
path = shotsRoute + user
if not os.path.exists(shotsRoute):
os.makedirs(shotsRoute, 0o755)
os.chown(shotsRoute, uid, gid)
if not os.path.exists(path):
os.makedirs(path, 0o755)
os.chown(path, uid, gid)
# starts a 30s timer for taking picture with RPI camera
def takePictureStartTimer (threadStopEvent):
global cameraTimer
global shotsRoute
global user
filePath = shotsRoute + user + "/%d.jpg"
if not threadStopEvent.is_set():
# take picture every 30 sec
cameraTimer = threading.Timer(30, takePictureStartTimer, [threadStopEvent])
cameraTimer.start()
# run raspistill and redirect output tu /dev/null in order to avoid error messages when camera is not connected
with open(os.devnull, 'w') as devnull:
subprocess.run(["raspistill", "-rot", "180", "-t", "500", "-ts", "-o", filePath, "-n"], stdout=devnull, stderr=devnull)
# Read units of data from sensor store queue and send it to AVATAR
def sendDataToAvatar ( threadStopEvent, dataQueue, sock ):
dict = None
# this will stop the execution with keyboard interrupt
while not threadStopEvent.is_set():
dataDict = dataQueue.get()
if 'sensors' in dataDict:
dict = dataDict
dict ["carInfo"]= {}
elif 'carInfo' in dataDict:
dict = dataDict
dict ["sensors"]= {}
try:
# send the data in JSON format
sock.sendall (json.dumps(dict).encode())
# if the other side close the socket...
except socket.error:
pass
# manage command line interface arguments
def setUpArgParser ( ):
global certPath
global keyCertPath
global caCertPath
global gatewayIP
global obdGpsTrip
global realObdGps
global beaconsTrip
global obdGpsBeaconsDb
global interactiveBeacons
global interactiveBeaconsCoord
global ble4Beacons
global ble5Beacons
# description of the script shown in command line
scriptDescription = 'This is the main script of MOTAM Gateway. By default, this starts a Wifi Direct connection'
# initiate the arguments parser
argParser = argparse.ArgumentParser(description = scriptDescription)
# command line arguments
argParser.add_argument("-l", "--loadCarTrip", help="Loads a specific session database. This will simulate only OBDII and GPS data. You can use default saved session trip database or indicate the name of another one placed in sessions folder: '--loadCarTrip' or '--loadCarTrip Session.db'", nargs='?', type=str, const='no_path')
argParser.add_argument("-L", "--loadBeaconsTrip", help="This arguments must be indicated together with '--loadCarTrip'. This activates the simulated beacons from session database.", action='store_true')
argParser.add_argument("-c", "--cert", help="Loads a specific gateway certificate. By default, the script loads certificate for normal vehicle. The certificate file must be on cetificates folder.")
argParser.add_argument("-C", "--ca", help="Loads a specific certificate of CA. By default, the script loads AVATAR CA. The certificate file must be on certificates folder.")
argParser.add_argument("-a", "--address", help="Disables Wifi-Direct and open socket on selected IP ", type=str)
argParser.add_argument("-v", "--version", help="Shows script version", action="store_true")
argParser.add_argument("-r", "--real_obd_gps", help="Data source: Uses OBDII USB interface and GPS receiver instead of simulating their values. It's neccesary to connect OBDII and GPS by USB.", action='store_true')
argParser.add_argument("-b", "--real_ble4", help="Data source: Uses Bluetooth 4 RPi receiver for capturing road beacons", action='store_true')
argParser.add_argument("-B", "--real_ble5", help="Data source: Uses nRF52840 dongle for capturing BLE5 beacons.", action='store_true')
argParser.add_argument("-i", "--interactive", help="Data source: Simulates BLE scanner in real time from terminal input. You can use only '--interactive' (default coordinates) '--interactive 36.778 -4.234' ", nargs='*', type=float)
args = argParser.parse_args ()
if args.loadCarTrip:
obdGpsTrip = True
if args.loadCarTrip != 'no_path':
obdGpsBeaconsDb = args.loadCarTrip
if args.loadBeaconsTrip:
beaconsTrip = True
if args.cert:
certPath = certRoute + args.cert
keyCertPath = certRoute + args.cert.split('.')[0]+'.key'
if args.ca:
caCertPath = certRoute+args.ca
if args.address:
gatewayIP = args.address
if args.real_obd_gps:
realObdGps = True
if args.real_ble4:
ble4Beacons = True
if args.real_ble5:
ble5Beacons = True
if args.interactive is not None and len(args.interactive) in (0,2):
interactiveBeacons = True
interactiveBeaconsCoord = args.interactive
if args.version:
print ("MOTAM Simulation script version: ", scriptVersion)
exit()
# start script
if __name__ == '__main__':
main()