-
Notifications
You must be signed in to change notification settings - Fork 31
Expand file tree
/
Copy pathweb.py
More file actions
144 lines (112 loc) · 4.79 KB
/
web.py
File metadata and controls
144 lines (112 loc) · 4.79 KB
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
import argparse
import logging
from logging import StreamHandler
from logging.handlers import TimedRotatingFileHandler
import multiprocessing
import random
import string
import sys
from threading import Thread
from time import sleep
from urllib import request
import webbrowser
from flask import Flask
import socketio
from werkzeug.middleware.dispatcher import DispatcherMiddleware
from utils.config import Config, static_path, get_data_path, initialize_paths_from_args
from utils.storage import Storage
from webapp.backend import Backend
from webapp.index import Index
def url_ok(url):
try:
request.urlopen(url=url)
return True
except Exception:
return False
def parse_cli(open_browser=True, webview=False):
parser = argparse.ArgumentParser()
parser.add_argument("port", nargs="?", type=int, default=5000, help="Port for web server to listen on")
parser.add_argument("--listen", type=str, default="127.0.0.1" if webview else "0.0.0.0",
help="Listen on address of specific interface (defaults to all interfaces)")
parser.add_argument("--on-receive", help="Call this program/script when new measurements are received")
parser.add_argument("--on-receive-interval", type=int, default=60, help="Interval for --on-receive (in seconds)")
parser.add_argument("--data-dir", type=str, help="Where to store configuration and user data files")
if webview:
parser.add_argument("--disable-gpu", action="store_true", default=False, help="Disable GPU rendering")
else:
parser.add_argument("--daemon", action="store_true", default=not open_browser, help="Do not launch web browser")
parser.add_argument("--prefix", default="/", help="If you want to reverse-proxy from path, like /rd-usb")
return parser.parse_args()
def run(args=None, embedded=False, webview=None):
if not args:
args = parse_cli()
if not embedded:
initialize_paths_from_args(args)
port = args.port
listen = args.listen
daemon = "daemon" in args and args.daemon
if "prefix" in args:
prefix = args.prefix
else:
prefix = "/"
if not prefix.startswith("/"):
prefix = "/" + prefix
if len(prefix) > 1 and prefix.endswith("/"):
prefix = prefix[0:-1]
app = Flask(__name__, static_folder=static_path)
app.config["embedded"] = embedded
app.config["webview"] = webview
app.config["app_prefix"] = prefix
app.register_blueprint(Index().register())
if prefix != "/":
def fallback(env, resp):
resp(b"200 OK", [(b"Content-Type", b"text/plain; charset=UTF-8")])
return [b"use '%s' instead" % prefix.encode("utf-8")]
app.config["APPLICATION_ROOT"] = prefix
app.wsgi_app = DispatcherMiddleware(fallback, {prefix: app.wsgi_app})
logger = logging.getLogger()
logger.setLevel(logging.INFO)
formatter = logging.Formatter("%(asctime)s - %(levelname)s - %(message)s")
console = StreamHandler()
console.setLevel(logging.DEBUG)
console.setFormatter(formatter)
logger.addHandler(console)
if not app.debug:
file = TimedRotatingFileHandler(get_data_path() + "/error.log", when="w0", backupCount=14)
file.setLevel(logging.ERROR)
file.setFormatter(formatter)
logger.addHandler(file)
try:
config = Config()
secret_key = config.read("secret_key")
if not secret_key:
secret_key = "".join(random.choice(string.ascii_uppercase + string.digits) for _ in range(16))
config.write("secret_key", secret_key)
app.secret_key = secret_key
Storage().init()
sockets = socketio.Server(async_mode="threading", cors_allowed_origins="*")
socketio_path = "socket.io"
if len(prefix) > 1:
socketio_path = prefix[1:] + "/" + socketio_path
app.wsgi_app = socketio.Middleware(sockets, app.wsgi_app, socketio_path=socketio_path)
sockets.register_namespace(Backend(args.on_receive, args.on_receive_interval))
if not embedded:
def open_in_browser():
logging.info("Application is starting...")
url = "http://127.0.0.1:%s" % port
while not url_ok(url):
sleep(0.5)
logging.info("Application is available at " + url)
if not app.debug and not daemon:
webbrowser.open(url)
Thread(target=open_in_browser, daemon=True).start()
app.run(host=listen, port=port, threaded=True, use_reloader=False)
except Exception as e:
logging.exception(e)
if __name__ == "__main__":
if len(sys.argv) > 1 and "fork" in sys.argv[1]:
multiprocessing.freeze_support()
exit(0)
if sys.platform.startswith("win"):
multiprocessing.freeze_support()
run()