3
3
from time import sleep
4
4
import random
5
5
6
+ #
7
+ # glass_server.py is an example web app which demonstrates how data can be read and set in the simulator
8
+ #
9
+ # When run this code will start an http server running on http://localhost:5000/ which can be accessed. It includes both
10
+ # an HTML/JS front end which can be accessed through a browser and the ability to read/write datapoints and datasets
11
+ # via API requests using JSON
12
+ #
13
+ # The server runs using Flask: https://flask.palletsprojects.com/en/1.1.x/
14
+ #
15
+ # This is intended to be a demonstration of the Python-SimConnect library rather than a fully fledged implementation.
16
+ # This code has been forked into more fully worked projects including:
17
+ # - MSFS 2020 Cockpit Companion: https://msfs2020.cc/
18
+ # - MSFS Mobile Companion App: https://github.com/mracko/MSFS-Mobile-Companion-App
19
+ #
20
+
6
21
7
22
app = Flask (__name__ )
8
23
14
29
aq = AircraftRequests (sm , _time = 10 )
15
30
16
31
# Create request holders
17
-
18
- # Note: I have commented out request_ui as I don't think it makes sense to replicate the ui interface through JSON given the /ui endpoint returns a duplicate of this anyway
19
- # I have not deleted it yet as it's handy to have this list of helpful variables here
20
- #
21
- #request_ui = [
22
- # 'PLANE_ALTITUDE',
23
- # 'PLANE_LATITUDE',
24
- # 'PLANE_LONGITUDE',
25
- # 'AIRSPEED_INDICATED',
26
- # 'MAGNETIC_COMPASS', # Compass reading
27
- # 'VERTICAL_SPEED', # Vertical speed indication
28
- # 'FLAPS_HANDLE_PERCENT', # Percent flap handle extended
29
- # 'FUEL_TOTAL_QUANTITY', # Current quantity in volume
30
- # 'FUEL_TOTAL_CAPACITY', # Total capacity of the aircraft
31
- # 'GEAR_HANDLE_POSITION', # True if gear handle is applied
32
- # 'AUTOPILOT_MASTER',
33
- # 'AUTOPILOT_NAV_SELECTED',
34
- # 'AUTOPILOT_WING_LEVELER',
35
- # 'AUTOPILOT_HEADING_LOCK',
36
- # 'AUTOPILOT_HEADING_LOCK_DIR',
37
- # 'AUTOPILOT_ALTITUDE_LOCK',
38
- # 'AUTOPILOT_ALTITUDE_LOCK_VAR',
39
- # 'AUTOPILOT_ATTITUDE_HOLD',
40
- # 'AUTOPILOT_GLIDESLOPE_HOLD',
41
- # 'AUTOPILOT_PITCH_HOLD_REF',
42
- # 'AUTOPILOT_APPROACH_HOLD',
43
- # 'AUTOPILOT_BACKCOURSE_HOLD',
44
- # 'AUTOPILOT_VERTICAL_HOLD',
45
- # 'AUTOPILOT_VERTICAL_HOLD_VAR',
46
- # 'AUTOPILOT_PITCH_HOLD',
47
- # 'AUTOPILOT_FLIGHT_DIRECTOR_ACTIVE',
48
- # 'AUTOPILOT_AIRSPEED_HOLD',
49
- # 'AUTOPILOT_AIRSPEED_HOLD_VAR'
50
- #]
51
-
32
+ # These are groups of datapoints which it is convenient to call as a group because they fulfill a specific function
52
33
request_location = [
53
34
'ALTITUDE' ,
54
35
'LATITUDE' ,
264
245
'CABIN_NO_SMOKING_ALERT_SWITCH'
265
246
]
266
247
267
-
248
+ # This is a helper function which just adds a comma in the right place for readability,
249
+ # for instance converting 30000 to 30,000
268
250
def thousandify (x ):
269
251
return f"{ x :,} "
270
252
@@ -291,21 +273,24 @@ def get_dataset(data_type):
291
273
if data_type == "trim" : request_to_action = request_trim
292
274
if data_type == "autopilot" : request_to_action = request_autopilot
293
275
if data_type == 'cabin' : request_to_action = request_cabin
294
- #if data_type == "ui": request_to_action = request_ui # see comment above as to why I've removed this
295
276
296
277
return request_to_action
297
278
298
279
280
+ # In addition to the datapoints which can be pulled individually or as groups via JSON, the UI endpoint returns JSON
281
+ # with the datapoints which the HTML / JS uses in a friendly format
299
282
@app .route ('/ui' )
300
283
def output_ui_variables ():
301
284
302
- # Initialise dictionaru
285
+ # Initialise dictionary
303
286
ui_friendly_dictionary = {}
304
287
ui_friendly_dictionary ["STATUS" ] = "success"
305
288
306
289
# Fuel
307
290
fuel_percentage = (aq .get ("FUEL_TOTAL_QUANTITY" ) / aq .get ("FUEL_TOTAL_CAPACITY" )) * 100
308
291
ui_friendly_dictionary ["FUEL_PERCENTAGE" ] = round (fuel_percentage )
292
+
293
+ # Airspeed and altitude
309
294
ui_friendly_dictionary ["AIRSPEED_INDICATE" ] = round (aq .get ("AIRSPEED_INDICATED" ))
310
295
ui_friendly_dictionary ["ALTITUDE" ] = thousandify (round (aq .get ("PLANE_ALTITUDE" )))
311
296
@@ -355,15 +340,20 @@ def output_ui_variables():
355
340
356
341
@app .route ('/dataset/<dataset_name>/' , methods = ["GET" ])
357
342
def output_json_dataset (dataset_name ):
358
- dataset_map = {} #I have renamed map to dataset_map as map is used elsewhere
343
+ dataset_map = {}
344
+
345
+ # This uses get_dataset() to pull in a bunch of different datapoint names into a dictionary which means they can
346
+ # then be requested from the sim
359
347
data_dictionary = get_dataset (dataset_name )
348
+
360
349
for datapoint_name in data_dictionary :
361
350
dataset_map [datapoint_name ] = aq .get (datapoint_name )
351
+
362
352
return jsonify (dataset_map )
363
353
364
354
355
+ # This function actually does the work of getting an individual datapoint from the sim
365
356
def get_datapoint (datapoint_name , index = None ):
366
- # This function actually does the work of getting the datapoint
367
357
368
358
if index is not None and ':index' in datapoint_name :
369
359
dp = aq .find (datapoint_name )
@@ -373,9 +363,9 @@ def get_datapoint(datapoint_name, index=None):
373
363
return aq .get (datapoint_name )
374
364
375
365
366
+ # This is the http endpoint wrapper for getting an individual datapoint
376
367
@app .route ('/datapoint/<datapoint_name>/get' , methods = ["GET" ])
377
368
def get_datapoint_endpoint (datapoint_name ):
378
- # This is the http endpoint wrapper for getting a datapoint
379
369
380
370
ds = request .get_json () if request .is_json else request .form
381
371
index = ds .get ('index' )
@@ -388,8 +378,8 @@ def get_datapoint_endpoint(datapoint_name):
388
378
return jsonify (output )
389
379
390
380
381
+ # This function actually does the work of setting an individual datapoint
391
382
def set_datapoint (datapoint_name , index = None , value_to_use = None ):
392
- # This function actually does the work of setting the datapoint
393
383
394
384
if index is not None and ':index' in datapoint_name :
395
385
clas = aq .find (datapoint_name )
@@ -410,9 +400,9 @@ def set_datapoint(datapoint_name, index=None, value_to_use=None):
410
400
return status
411
401
412
402
403
+ # This is the http endpoint wrapper for setting a datapoint
413
404
@app .route ('/datapoint/<datapoint_name>/set' , methods = ["POST" ])
414
405
def set_datapoint_endpoint (datapoint_name ):
415
- # This is the http endpoint wrapper for setting a datapoint
416
406
417
407
ds = request .get_json () if request .is_json else request .form
418
408
index = ds .get ('index' )
@@ -423,8 +413,8 @@ def set_datapoint_endpoint(datapoint_name):
423
413
return jsonify (status )
424
414
425
415
416
+ # This function actually does the work of triggering an event
426
417
def trigger_event (event_name , value_to_use = None ):
427
- # This function actually does the work of triggering the event
428
418
429
419
EVENT_TO_TRIGGER = ae .find (event_name )
430
420
if EVENT_TO_TRIGGER is not None :
@@ -440,9 +430,9 @@ def trigger_event(event_name, value_to_use = None):
440
430
return status
441
431
442
432
433
+ # This is the http endpoint wrapper for triggering an event
443
434
@app .route ('/event/<event_name>/trigger' , methods = ["POST" ])
444
435
def trigger_event_endpoint (event_name ):
445
- # This is the http endpoint wrapper for triggering an event
446
436
447
437
ds = request .get_json () if request .is_json else request .form
448
438
value_to_use = ds .get ('value_to_use' )
@@ -471,4 +461,5 @@ def custom_emergency(emergency_type):
471
461
return text_to_return
472
462
473
463
464
+ # Main loop to run the flask app
474
465
app .run (host = '0.0.0.0' , port = 5000 , debug = True )
0 commit comments