forked from PX4/flight_review
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserve.py
executable file
·171 lines (145 loc) · 6.88 KB
/
serve.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
""" Script to run the bokeh server """
from __future__ import absolute_import
from __future__ import print_function
import argparse
import os
import sys
import errno
from bokeh.application import Application
from bokeh.server.server import Server
from bokeh.application.handlers import DirectoryHandler
from tornado.web import StaticFileHandler
# this is needed for the following imports
sys.path.append(os.path.join(os.path.dirname(os.path.realpath(__file__)), 'plot_app'))
from tornado.web import RedirectHandler
from tornado_handlers.download import DownloadHandler
from tornado_handlers.upload import UploadHandler
from tornado_handlers.browse import BrowseHandler, BrowseDataRetrievalHandler
from tornado_handlers.edit_entry import EditEntryHandler
from tornado_handlers.db_info_json import DBInfoHandler
from tornado_handlers.three_d import ThreeDHandler
from tornado_handlers.radio_controller import RadioControllerHandler
from tornado_handlers.error_labels import UpdateErrorLabelHandler
from helper import set_log_id_is_filename, print_cache_info
from config import debug_print_timing, get_overview_img_filepath
#pylint: disable=invalid-name
def _fixup_deprecated_host_args(arguments):
# --host is deprecated since bokeh 0.12.5. You might want to use
# --allow-websocket-origin instead
if arguments.host is not None and len(arguments.host) > 0:
if arguments.allow_websocket_origin is None:
arguments.allow_websocket_origin = []
arguments.allow_websocket_origin += arguments.host
arguments.allow_websocket_origin = list(set(arguments.allow_websocket_origin))
parser = argparse.ArgumentParser(description='Start bokeh Server')
parser.add_argument('-s', '--show', dest='show', action='store_true',
help='Open browser on startup')
parser.add_argument('--use-xheaders', action='store_true',
help="Prefer X-headers for IP/protocol information")
parser.add_argument('-f', '--file', metavar='file.ulg', action='store',
help='Directly show an ULog file, only for local use (implies -s)',
default=None)
parser.add_argument('--3d', dest='threed', action='store_true',
help='Open 3D page (only if --file is provided)')
parser.add_argument('--pid-analysis', dest='pid_analysis', action='store_true',
help='Open PID analysis page (only if --file is provided)')
parser.add_argument('--num-procs', dest='numprocs', type=int, action='store',
help="""Number of worker processes. Default to 1.
0 will autodetect number of cores""",
default=1)
parser.add_argument('--port', type=int, action='store',
help='Port to listen on', default=None)
parser.add_argument('--address', action='store',
help='Network address to listen to', default=None)
parser.add_argument('--host', action='append', type=str, metavar='HOST[:PORT]',
help="""Hosts whitelist, that must match the Host header in new
requests. It has the form <host>[:<port>]. If no port is specified, 80
is used. You should use the DNS name of the public endpoint here. \'*\'
matches all hosts (for testing only) (default=localhost)""",
default=None)
parser.add_argument('--allow-websocket-origin', action='append', type=str, metavar='HOST[:PORT]',
help="""Public hostnames which may connect to the Bokeh websocket""",
default=None)
args = parser.parse_args()
# This should remain here until --host is removed entirely
_fixup_deprecated_host_args(args)
applications = {}
main_path = os.path.join(os.path.dirname(os.path.realpath(__file__)), 'plot_app')
handler = DirectoryHandler(filename=main_path)
applications['/plot_app'] = Application(handler)
server_kwargs = {}
if args.port is not None: server_kwargs['port'] = args.port
if args.use_xheaders: server_kwargs['use_xheaders'] = args.use_xheaders
server_kwargs['num_procs'] = args.numprocs
if args.address is not None: server_kwargs['address'] = args.address
if args.host is not None: server_kwargs['host'] = args.host
if args.allow_websocket_origin is not None:
server_kwargs['allow_websocket_origin'] = args.allow_websocket_origin
server_kwargs['websocket_max_message_size'] = 100 * 1024 * 1024
# increase the maximum upload size (default is 100MB)
server_kwargs['http_server_kwargs'] = {'max_buffer_size': 300 * 1024 * 1024}
show_ulog_file = False
show_3d_page = False
show_pid_analysis_page = False
if args.file is not None:
ulog_file = os.path.abspath(args.file)
show_ulog_file = True
args.show = True
show_3d_page = args.threed
show_pid_analysis_page = args.pid_analysis
set_log_id_is_filename(show_ulog_file)
# additional request handlers
extra_patterns = [
(r'/upload', UploadHandler),
(r'/browse', BrowseHandler),
(r'/browse_data_retrieval', BrowseDataRetrievalHandler),
(r'/3d', ThreeDHandler),
(r'/radio_controller', RadioControllerHandler),
(r'/edit_entry', EditEntryHandler),
(r'/?', UploadHandler), #root should point to upload
(r'/download', DownloadHandler),
(r'/dbinfo', DBInfoHandler),
(r'/error_label', UpdateErrorLabelHandler),
(r"/stats", RedirectHandler, {"url": "/plot_app?stats=1"}),
(r'/overview_img/(.*)', StaticFileHandler, {'path': get_overview_img_filepath()}),
]
server = None
custom_port = 5006
while server is None:
try:
server = Server(applications, extra_patterns=extra_patterns, **server_kwargs)
except OSError as e:
# if we get a port bind error and running locally with '-f',
# automatically select another port (useful for opening multiple logs)
if e.errno == errno.EADDRINUSE and show_ulog_file:
custom_port += 1
server_kwargs['port'] = custom_port
else:
raise
if args.show:
# we have to defer opening in browser until we start up the server
def show_callback():
""" callback to open a browser window after server is fully initialized"""
if show_ulog_file:
if show_3d_page:
server.show('/3d?log='+ulog_file)
elif show_pid_analysis_page:
server.show('/plot_app?plots=pid_analysis&log='+ulog_file)
else:
server.show('/plot_app?log='+ulog_file)
else:
server.show('/upload')
server.io_loop.add_callback(show_callback)
if debug_print_timing():
def print_statistics():
""" print ulog cache info once per hour """
print_cache_info()
server.io_loop.call_later(60*60, print_statistics)
server.io_loop.call_later(60, print_statistics)
# run_until_shutdown has been added 0.12.4 and is the preferred start method
run_op = getattr(server, "run_until_shutdown", None)
if callable(run_op):
server.run_until_shutdown()
else:
server.start()