Skip to content

Commit 2fd1207

Browse files
committed
Updated comments in glass_server to make the code more usable
1 parent 620f1e2 commit 2fd1207

File tree

1 file changed

+36
-45
lines changed

1 file changed

+36
-45
lines changed

glass_server.py

+36-45
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,21 @@
33
from time import sleep
44
import random
55

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+
621

722
app = Flask(__name__)
823

@@ -14,41 +29,7 @@
1429
aq = AircraftRequests(sm, _time=10)
1530

1631
# 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
5233
request_location = [
5334
'ALTITUDE',
5435
'LATITUDE',
@@ -264,7 +245,8 @@
264245
'CABIN_NO_SMOKING_ALERT_SWITCH'
265246
]
266247

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
268250
def thousandify(x):
269251
return f"{x:,}"
270252

@@ -291,21 +273,24 @@ def get_dataset(data_type):
291273
if data_type == "trim": request_to_action = request_trim
292274
if data_type == "autopilot": request_to_action = request_autopilot
293275
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
295276

296277
return request_to_action
297278

298279

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
299282
@app.route('/ui')
300283
def output_ui_variables():
301284

302-
# Initialise dictionaru
285+
# Initialise dictionary
303286
ui_friendly_dictionary = {}
304287
ui_friendly_dictionary["STATUS"] = "success"
305288

306289
# Fuel
307290
fuel_percentage = (aq.get("FUEL_TOTAL_QUANTITY") / aq.get("FUEL_TOTAL_CAPACITY")) * 100
308291
ui_friendly_dictionary["FUEL_PERCENTAGE"] = round(fuel_percentage)
292+
293+
# Airspeed and altitude
309294
ui_friendly_dictionary["AIRSPEED_INDICATE"] = round(aq.get("AIRSPEED_INDICATED"))
310295
ui_friendly_dictionary["ALTITUDE"] = thousandify(round(aq.get("PLANE_ALTITUDE")))
311296

@@ -355,15 +340,20 @@ def output_ui_variables():
355340

356341
@app.route('/dataset/<dataset_name>/', methods=["GET"])
357342
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
359347
data_dictionary = get_dataset(dataset_name)
348+
360349
for datapoint_name in data_dictionary:
361350
dataset_map[datapoint_name] = aq.get(datapoint_name)
351+
362352
return jsonify(dataset_map)
363353

364354

355+
# This function actually does the work of getting an individual datapoint from the sim
365356
def get_datapoint(datapoint_name, index=None):
366-
# This function actually does the work of getting the datapoint
367357

368358
if index is not None and ':index' in datapoint_name:
369359
dp = aq.find(datapoint_name)
@@ -373,9 +363,9 @@ def get_datapoint(datapoint_name, index=None):
373363
return aq.get(datapoint_name)
374364

375365

366+
# This is the http endpoint wrapper for getting an individual datapoint
376367
@app.route('/datapoint/<datapoint_name>/get', methods=["GET"])
377368
def get_datapoint_endpoint(datapoint_name):
378-
# This is the http endpoint wrapper for getting a datapoint
379369

380370
ds = request.get_json() if request.is_json else request.form
381371
index = ds.get('index')
@@ -388,8 +378,8 @@ def get_datapoint_endpoint(datapoint_name):
388378
return jsonify(output)
389379

390380

381+
# This function actually does the work of setting an individual datapoint
391382
def set_datapoint(datapoint_name, index=None, value_to_use=None):
392-
# This function actually does the work of setting the datapoint
393383

394384
if index is not None and ':index' in datapoint_name:
395385
clas = aq.find(datapoint_name)
@@ -410,9 +400,9 @@ def set_datapoint(datapoint_name, index=None, value_to_use=None):
410400
return status
411401

412402

403+
# This is the http endpoint wrapper for setting a datapoint
413404
@app.route('/datapoint/<datapoint_name>/set', methods=["POST"])
414405
def set_datapoint_endpoint(datapoint_name):
415-
# This is the http endpoint wrapper for setting a datapoint
416406

417407
ds = request.get_json() if request.is_json else request.form
418408
index = ds.get('index')
@@ -423,8 +413,8 @@ def set_datapoint_endpoint(datapoint_name):
423413
return jsonify(status)
424414

425415

416+
# This function actually does the work of triggering an event
426417
def trigger_event(event_name, value_to_use = None):
427-
# This function actually does the work of triggering the event
428418

429419
EVENT_TO_TRIGGER = ae.find(event_name)
430420
if EVENT_TO_TRIGGER is not None:
@@ -440,9 +430,9 @@ def trigger_event(event_name, value_to_use = None):
440430
return status
441431

442432

433+
# This is the http endpoint wrapper for triggering an event
443434
@app.route('/event/<event_name>/trigger', methods=["POST"])
444435
def trigger_event_endpoint(event_name):
445-
# This is the http endpoint wrapper for triggering an event
446436

447437
ds = request.get_json() if request.is_json else request.form
448438
value_to_use = ds.get('value_to_use')
@@ -471,4 +461,5 @@ def custom_emergency(emergency_type):
471461
return text_to_return
472462

473463

464+
# Main loop to run the flask app
474465
app.run(host='0.0.0.0', port=5000, debug=True)

0 commit comments

Comments
 (0)