-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathflask_wrapper.py
executable file
·170 lines (141 loc) · 5.96 KB
/
flask_wrapper.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
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright 2020 by Murray Altheim. All rights reserved. This file is part of
# the Robot Operating System project and is released under the "Apache Licence,
# Version 2.0". Please see the LICENSE file included as part of this package.
#
# author: Murray Altheim
# created: 2019-12-23
# modified: 2020-04-09
#
import sys, logging, threading
from colorama import init, Fore, Style
init()
try:
from flask import Flask, jsonify, request, render_template, send_from_directory
except Exception as e:
sys.exit("This script requires the flask module\nInstall with: sudo pip3 install flask")
try:
from flask_restful import Api
except Exception as e:
# see: https://flask-restful.readthedocs.io/en/latest/
sys.exit("This script requires the flask-restful module\nInstall with: sudo pip3 install flask-restful")
#from flask_socketio import SocketIO, emit
from lib.logger import Level, Logger
from lib.message_factory import MessageFactory
from lib.event import Event
# RESTful Flask Service .......................................................
app = Flask(__name__)
api = Api(app)
#socketio = SocketIO(app)
# endpoints ...............................................
@app.route('/')
def home():
return render_template('index.html') # render a template
@app.route('/cancel')
def cancel():
global loop
if ( loop is True ):
loop = False
return jsonify(state='loop cancelled.')
else:
return jsonify(state='no active loop.')
@app.route('/speed')
def setSpeed():
port = request.args.get('port', type=int)
starboard = request.args.get('starboard', type=int)
port_speed = float( port )
starboard_speed = float( starboard )
print(Fore.GREEN + 'setSpeed: port: {:>5.2f}, starboard: {:>5.2f}'.format(port_speed, starboard_speed) + Style.RESET_ALL)
return jsonify([ { 'port_speed': port_speed }, { 'starboard_speed': starboard_speed } ])
@app.route('/event')
def setEvent():
global g_queue
event_label = request.args.get('id', type=str)
event = Event.from_str(event_label)
print( Fore.MAGENTA + Style.BRIGHT + 'ACTION: {}'.format(event.name) + Style.RESET_ALL)
response = g_queue.respond(event)
print( Fore.MAGENTA + Style.BRIGHT + 'RESPONSE mimetype: {}; json: {}'.format(response.mimetype, response.json) + Style.RESET_ALL)
return response
# return jsonify([ event_label ])
@app.route('/style/<path:path>')
def send_style(path):
return send_from_directory('style', path)
@app.route('/script/<path:path>')
def send_script(path):
return send_from_directory('script', path)
#@app.route('/doc')
#def doc_home():
# return render_template('apidocs/index.html') # render a template
@app.route('/apidocs/<path:path>')
def send_api_docs(path):
return send_from_directory('apidocs', path)
# see: https://stackoverflow.com/questions/31264826/start-a-flask-application-in-separate-thread
def shutdown_server(environ):
if not 'werkzeug.server.shutdown' in environ:
raise RuntimeError('Not running the development server')
environ['werkzeug.server.shutdown']()
# ..............................................................................
class MockMessageQueue(object):
def __init__(self, level=Level.INFO):
self._log = Logger('flask.wrapper', level)
self._message_factory = MessageFactory(None, level)
self._log.info('ready.')
# ......................................................
def respond(self, event):
'''
Responds to the Event by wrapping it in Message and adding it to the backing queue.
This is only used by FlaskWrapper.
'''
self._log.info('RESPOND to event {}.'.format(event.name))
_message = self._message_factory.get_message(event, None)
self.add(_message)
return jsonify( [ { 'id': _message.message_id }, { 'event': event.name }, { 'priority': event.priority } ] )
# ..........................................................................
def add(self, message):
'''
Add a new Message to the queue, then additionally to any consumers.
'''
# if self._queue.full():
# try:
# _dumped = self._queue.get()
# self._log.debug('dumping old message eid#{}/msg#{}: {}'.format(_dumped.eid, _dumped.number, _dumped.description))
# except Empty:
# pass
# self._queue.put(message);
self._log.info('added message id {} with event: {}'.format(message.message_id, message.event.description))
# # add to any consumers
# for consumer in self._consumers:
# consumer.add(message);
# ..............................................................................
class FlaskWrapperService(threading.Thread):
def __init__(self):
global g_queue
super().__init__()
self._log = Logger('flask.wrapper', Level.INFO)
g_queue = MockMessageQueue()
threading.Thread.__init__(self)
self._thread = None
self._log.debug('ready')
# ..........................................................................
def run(self):
global app
self._log.info('starting flask wrapper...')
# suppress console log messages (this doesn't actually work)
_flask_log = logging.getLogger('werkzeug')
_flask_log.setLevel(logging.ERROR)
_flask_log.disabled = True
# socketio.run(app)
self._thread = threading.Thread(target=app.run(host='0.0.0.0', port=8085, debug=False, use_reloader=False))
self._thread.start()
# app.run(host='0.0.0.0', port=8085, debug=False, use_reloader=False)
self._log.info('ended flask thread.')
# ..........................................................................
def close(self):
self._log.info('closing flask.')
if self._thread != None:
self._thread.join(timeout=1.0)
self._log.info('flask app thread joined.')
self._log.info('closed flask.')
#EOF