forked from PX4/flight_review
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserve.py
executable file
·233 lines (187 loc) · 8.25 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
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
#! /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
import runpy
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
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.browse2 import Browse2Handler, Browse2DataRetrievalHandler
from tornado_handlers.top import TopHandler
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
import px4tools
import numpy as np
import math
import io
import os
import sys
import errno
from plotting import DataPlot
from functools import lru_cache
from os.path import dirname, join
from bokeh.io import output_file, show
from bokeh.models.widgets import FileInput
from bokeh.models.widgets import Paragraph
from bokeh.models import CheckboxGroup
from bokeh.models import RadioButtonGroup
from bokeh.models import Range1d
from bokeh.server.server import Server
from bokeh.themes import Theme
from bokeh.application.handlers import DirectoryHandler
import time
import copy
from bokeh.models import Div
import pandas as pd
import argparse
from bokeh.layouts import column, row
from bokeh.models import ColumnDataSource, PreText, Select
from bokeh.plotting import figure
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('--thiel', dest='thiel', action='store_true',
help='Open Thiel 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)
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
show_thiel = False
if args.thiel:
show_thiel = True
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
applications = {}
if args.show:
thiel_path = os.path.join(os.path.dirname(os.path.realpath(__file__)), 'thiel_app')
handler = DirectoryHandler(filename=thiel_path)
sys.path.append(os.path.join(os.path.dirname(os.path.realpath(__file__)), 'thiel_app'))
applications['/plot_app'] = Application(handler)
applications['/thiel_app'] = Application(handler)
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'/browse2', Browse2Handler),
(r'/browse2_data_retrieval', Browse2DataRetrievalHandler),
(r'/3d', ThreeDHandler),
(r'/radio_controller', RadioControllerHandler),
(r'/edit_entry', EditEntryHandler),
(r'/?', TopHandler),
(r'/download', DownloadHandler),
(r'/dbinfo', DBInfoHandler),
(r'/top', TopHandler),
(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)
elif show_thiel:
print("showing Thiel app")
server.show('/thiel_app')
else:
server.show('/top')
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()