From 6c2da9e6f9b2a3d0d8cd612b33d2aeca0becaeb2 Mon Sep 17 00:00:00 2001 From: mdipierro Date: Sat, 17 Aug 2024 08:03:06 -0400 Subject: [PATCH] new server_adapters logic (#904) --- py4web/core.py | 34 +- py4web/server_adapters.py | 649 ------------------ py4web/server_adapters/__init__.py | 78 +++ py4web/server_adapters/adapter_gevent.py | 56 ++ .../adapter_gevent_websockets.py | 46 ++ .../adapter_gunicorn_gevent.py | 191 ++++++ py4web/server_adapters/adapter_rocket3.py | 27 + py4web/server_adapters/adapter_wsgiref.py | 109 +++ py4web/server_adapters/logging_utils.py | 89 +++ 9 files changed, 613 insertions(+), 666 deletions(-) delete mode 100644 py4web/server_adapters.py create mode 100644 py4web/server_adapters/__init__.py create mode 100644 py4web/server_adapters/adapter_gevent.py create mode 100644 py4web/server_adapters/adapter_gevent_websockets.py create mode 100644 py4web/server_adapters/adapter_gunicorn_gevent.py create mode 100644 py4web/server_adapters/adapter_rocket3.py create mode 100644 py4web/server_adapters/adapter_wsgiref.py create mode 100644 py4web/server_adapters/logging_utils.py diff --git a/py4web/core.py b/py4web/core.py index 254b1818..ca15f1a4 100644 --- a/py4web/core.py +++ b/py4web/core.py @@ -1650,7 +1650,7 @@ async def watch_folder(apps_folder): click.echo("--watch option has no effect in multi-process environment \n") return - if server_config["server"].startswith(("wsgiref", "waitress", "rocket")): + if server_adapters.blocking.get(server_config["server"]): # these servers block the main thread so we open a new thread for the file watcher threading.Thread( target=watch_folder_event_loop, args=(apps_folder,), daemon=True @@ -1709,7 +1709,7 @@ def start_server(kwargs): if not server_config["server"]: if server_config["platform"] == "windows" or number_workers < 2: - server_config["server"] = "rocketServer" + server_config["server"] = "rocket" else: if not gunicorn: logging.error("gunicorn not installed") @@ -1717,18 +1717,15 @@ def start_server(kwargs): server_config["server"] = "gunicorn" # Catch interrupts like Ctrl-C if needed - if server_config["server"] not in {"rocketServer", "wsgirefWsTwistedServer"}: - signal.signal( - signal.SIGINT, - lambda signal, frame: click.echo( - "KeyboardInterrupt (ID: {signal}) has been caught. Cleaning up..." - ) - and sys.exit(0), - ) + def stop_then_kill(sig, grame): + os.kill(os.getpid(), signal.SIGTERM) + time.sleep(2) + os.kill(os.getpid(), signal.SIGKILL) + + signal.signal(signal.SIGINT, stop_then_kill) - params["server"] = server_config["server"] - if params["server"] in server_adapters.__all__: - params["server"] = getattr(server_adapters, params["server"])() + adapter = server_adapters.available.get(server_config["server"]) + params["server"] = adapter or server_config["server"] if number_workers > 1: params["workers"] = number_workers if server_config["server"] == "gunicorn": @@ -2051,7 +2048,9 @@ def new_app(apps_folder, app_name, yes, scaffold_zip): help="No prompt, assume yes to questions", show_default=True, ) -@click.option("-H", "--host", default="127.0.0.1", help="Host listening IP", show_default=True) +@click.option( + "-H", "--host", default="127.0.0.1", help="Host listening IP", show_default=True +) @click.option( "-P", "--port", default=8000, type=int, help="Port number", show_default=True ) @@ -2089,10 +2088,11 @@ def new_app(apps_folder, app_name, yes, scaffold_zip): "--server", default="default", type=click.Choice( - ["default", "wsgiref", "tornado", "gevent", "waitress"] - + server_adapters.__all__ + ["default"] + list(server_adapters.available) + server_adapters.unavailable + ), + help="Web server to use (unavailable: {})".format( + ", ".join(server_adapters.unavailable) ), - help="Web server to use", show_default=False, ) @click.option( diff --git a/py4web/server_adapters.py b/py4web/server_adapters.py deleted file mode 100644 index 6fe5b7ec..00000000 --- a/py4web/server_adapters.py +++ /dev/null @@ -1,649 +0,0 @@ -import errno -import logging -import os -import socket -import ssl -import subprocess -import sys -import threading -from ast import literal_eval - -from ombott.server_adapters import ServerAdapter - -try: - from .utils.wsservers import wsservers_list -except ImportError: - wsservers_list = [] - -__all__ = [ - "gunicorn", - "gunicornGevent", - "gevent", - "geventWebSocketServer", - "geventWs", # short_name - "wsgirefThreadingServer", - "wsgiTh", # short_name - "rocketServer", -] + wsservers_list - -# ---------------------- utils ----------------------------------------------- - - -def get_log_file(out_banner=True): - """ - Returns the filename for logging or None - Assumes: - export PY4WEB_ERRORLOG=/tmp # export PY4WEB_ERRORLOG= - if PY4WEB_ERRORLOG is :stderr or :stdout returns None - if PY4WEB_ERRORLOG is a folder returns the name of a logfile in that dir - if PY4WEB_ERRORLOG is a filename it returns that filename - if the out_banner argument is true, it outputs the filename - """ - log_dir = os.environ.get("PY4WEB_ERRORLOG", None) - if log_dir and not log_dir.startswith(":"): - if os.path.isdir(log_dir): - log_file = os.path.join(log_dir, "server-py4web.log") - else: - log_file = log_dir - if out_banner: - print(f"log_file: {log_file}") - return log_file - return None - - -def check_level(level): - "Check the level is a valid loglevel" - # lib/python3.7/logging/__init__.py - # CRITICAL = 50 - # FATAL = CRITICAL - # ERROR = 40 - # WARNING = 30 - # WARN = WARNING - # INFO = 20 - # DEBUG = 10 - # NOTSET = 0 - - return ( - level - if level - in ( - logging.CRITICAL, - logging.ERROR, - logging.WARN, - logging.INFO, - logging.DEBUG, - logging.NOTSET, - ) - else logging.WARN - ) - - -def logging_conf(level=logging.WARN, logger_name=__name__, fmode="w"): - "Configures logging" - - log_file = get_log_file() - log_to = {} - - if log_file: - if sys.version_info >= (3, 9): - log_to["filename"] = log_file - log_to["filemode"] = fmode - log_to["encoding"] = "utf-8" - else: - h = logging.FileHandler(log_file, mode=fmode, encoding="utf-8") - log_to.update({"handlers": [h]}) - - short_msg = "%(message)s > %(threadName)s > %(asctime)s.%(msecs)03d" - # long_msg = short_msg + " > %(funcName)s > %(filename)s:%(lineno)d > %(levelname)s" - - time_msg = "%H:%M:%S" - # date_time_msg = '%Y-%m-%d %H:%M:%S' - - logging.basicConfig( - format=short_msg, - datefmt=time_msg, - level=check_level(level), - **log_to, - ) - - if logger_name is None: - return None - - log = logging.getLogger("SA:" + logger_name) - log.propagate = True - - return log - - -def get_workers(opts, default=10): - "Extracts the number of worker from opts or default" - return int(opts["workers"]) if opts.get("workers") else default - - -def check_port(host="127.0.0.1", port=8000): - "Check the specified port is available and print debug info" - - if host.startswith("unix:/"): - socket_path = host[5:] - if os.path.exists(socket_path): - if port == 0: - if ( - subprocess.run( - ["ls", "-alFi", "socket_path"], shell=False, check=False - ).returncode - != 0 - ): - sys.exit(f"can't run gunicorn: {socket_path} exists") - elif port == 1: - subprocess.run( - "ps -ef | head -1; ps -ef | grep py4web | grep -v grep", - shell=True, - check=False, - ) - subprocess.run(["ls", "-alFi", socket_path], shell=False, check=False) - subprocess.run(["lsof", "-w", socket_path], shell=False, check=False) - elif port == 8000: - pass - print(f"gunicorn listening at: {host}") - return - - s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) - try: - s.bind((host, int(port))) - except socket.error as e: - if e.errno == errno.EADDRINUSE: - subprocess.run( - f"command -v lsof >/dev/null 2>&1 && ps -ef | head -1; ps -ef |" - f" grep py4web | grep -v grep && lsof -nPi:{port}", - shell=True, - check=False, - ) - sys.exit(f"{host}:{port} is already in use") - else: - sys.exit(f"{e}\n{host}:{port} cannot be acessed") - s.close() - - -# ---------------------- servers ----------------------------------------------- - - -def gunicorn(): - "Builds and returns a guncorn server" - from gevent import local # pip install gevent gunicorn setproctitle - - if isinstance(threading.local(), local.local): - print("gunicorn: monkey.patch_all() applied") - - class GunicornServer(ServerAdapter): - "The gunicorn server adapter" - - def run(self, handler): - "runs the server" - try: - from gunicorn.app.base import Application - except ImportError as ex: - sys.exit(f"{ex}\nTry: pip install gunicorn gevent setproctitle") - - check_port(self.host, self.port) - - logger = None - - sa_bind = ( - self.host - if self.host.startswith("unix:/") - else f"{self.host}:{self.port}" - ) - - sa_config = { - "bind": sa_bind, # f"{self.host}:{self.port}", - "workers": get_workers(self.options), - "certfile": self.options.get("certfile", None), - "keyfile": self.options.get("keyfile", None), - "accesslog": None, - "errorlog": None, - "proc_name": "sa_py4web", # ps a | grep py4web - "config": "sa_config", - # ( 'sa_config', 'GUNICORN_', 'gunicorn.saenv', 'gunicorn.conf.py' ) - } - - if not self.quiet: - level = check_level(self.options["logging_level"]) - log_file = get_log_file() - - logger = logging_conf(level) - log_to = "-" if log_file is None else log_file - - sa_config.update( - { - "loglevel": logging.getLevelName(level), - "access_log_format": '%(h)s %(l)s %(u)s %(t)s "%(r)s" %(s)s %(b)s "%(f)s" "%(a)s"', - "accesslog": log_to, - "errorlog": log_to, - } - ) - - class GunicornApplication(Application): - "A gunicorn application" - - @staticmethod - def check_kv(kx, vx): - "convenience function" - if ( - kx - and vx - and ( - kx - not in ( - "bind", - "config", - ) - ) - ): - if vx.startswith("{") and vx.endswith("}"): - vx = literal_eval(vx) - if vx == "None": - vx = None - return kx, vx - return None, None - - def get_gunicorn_options( - self, - gu_default="gunicorn.conf.py", - env_file="gunicorn.saenv", - env_key="GUNICORN_", - ): - "Returns the default options" - if os.path.isfile(gu_default): - return {"use_python_config": gu_default, "config": gu_default} - - res_opts = {} - - if os.path.isfile(env_file): - try: - with open(env_file, "r") as f: - lines = f.read().splitlines() - for line in lines: - line = line.strip() - if not line or line.startswith(("#", "[")): - continue - for e in ("export ", env_key): - line = line.replace(e, "", 1) - k, v = None, None - try: - k, v = line.split("=", 1) - k, v = k.strip().lower(), v.strip() - except (ValueError, AttributeError): - continue - k, v = GunicornApplication.check_kv(k, v) - if k is None: - continue - res_opts[k] = v - - if res_opts: - res_opts["config"] = env_file - return res_opts - - except (IOError, OSError) as ex: - sys.exit(f"{ex}\nError: {env_file}") - - for k, v in os.environ.items(): - if k.startswith(env_key): - k = k.split("_", 1)[1].lower() - k, v = GunicornApplication.check_kv(k, v) - if k is None: - continue - res_opts[k] = v - - if res_opts: - res_opts["config"] = env_key - - return res_opts - - def load_config(self): - "Loads the config" - sa_config.update(self.get_gunicorn_options()) - logger and logger.debug(sa_config) - - for k, v in sa_config.items(): - if k not in self.cfg.settings: - continue - self.cfg.set(k, v) - - for e in ( - "use_python_config", - "usepy", - ): - if e in sa_config: - Application.load_config_from_file(self, sa_config[e]) - break - - def load(self): - return handler - - GunicornApplication().run() - - return GunicornServer - - -gunicornGevent = gunicorn - - -def gevent(): - "Returns a gevent server" - # gevent version 23.9.1 - from gevent import local, pywsgi # pip install gevent - - if not isinstance(threading.local(), local.local): - msg = "Ombott requires gevent.monkey.patch_all() (before import)" - raise RuntimeError(msg) - - # ./py4web.py run apps --watch=off -s gevent -L 20 - # - # ./py4web.py run apps -s gevent --watch=off --port=8443 --ssl_cert=cert.pem --ssl_key=key.pem -L 0 - # ./py4web.py run apps -s gevent --watch=off --host=192.168.1.161 --port=8443 --ssl_cert=server.pem -L 0 - - class GeventServer(ServerAdapter): - "Defines a gevent server" - - def run(self, handler): - "runs the server" - - logger = None # "default" - - if not self.quiet: - logger = logging_conf( - self.options["logging_level"], - "gevent", - ) - # logger.addHandler(logging.StreamHandler()) - - certfile = self.options.get("certfile", None) - - ssl_args = ( - dict( - certfile=certfile, - keyfile=self.options.get("keyfile", None), - ssl_version=ssl.PROTOCOL_SSLv23, - server_side=True, - do_handshake_on_connect=False, - ) - if certfile - else {} - ) - - server = pywsgi.WSGIServer( - (self.host, self.port), - handler, - log=logger, - error_log=logger, - **ssl_args, - ) - - server.serve_forever() - - return GeventServer - - -def geventWebSocketServer(): - "builds and returns a Gevent websocket server" - from gevent import pywsgi - # from geventwebsocket.handler import WebSocketHandler # pip install gevent-websocket - from gevent_ws import WebSocketHandler # pip install gevent gevent-ws - - # https://stackoverflow.com/questions/5312311/secure-websockets-with-self-signed-certificate - # https://pypi.org/project/gevent-ws/ - # ./py4web.py run apps -s geventWebSocketServer --watch=off --ssl_cert=server.pem -H 192.168.1.161 -P 9000 -L 10 - # vi apps/_websocket/templates/index.html set: ws, wss, host, port - # firefox http://localhost:8000/_websocket - # firefox https://192.168.1.161:9000/_websocket test wss - # curl --insecure -I -H 'Upgrade: websocket' \ - # -H "Sec-WebSocket-Key: `openssl rand -base64 16`" \ - # -H 'Sec-WebSocket-Version: 13' \ - # -sSv https://192.168.1.161:9000/ - - class GeventWebSocketServer(ServerAdapter): - "Class implementing a Gevent websocket server" - - def run(self, handler): - "Runs the server" - logger = None # "default" - - if not self.quiet: - logger = logging_conf( - self.options["logging_level"], - "gevent-ws", - ) - - certfile = self.options.get("certfile", None) - - ssl_args = ( - dict( - certfile=certfile, - keyfile=self.options.get("keyfile", None), - ) - if certfile - else {} - ) - - server = pywsgi.WSGIServer( - (self.host, self.port), - handler, - handler_class=WebSocketHandler, - log=logger, - error_log=logger, - **ssl_args, - ) - - server.serve_forever() - - return GeventWebSocketServer - - -geventWs = geventWebSocketServer - - -def wsgirefThreadingServer(): - "builds and returns a wsgiref threading server" - - # https://www.electricmonk.nl/log/2016/02/15/multithreaded-dev-web-server-for-the-python-bottle-web-framework/ - - from concurrent.futures import ThreadPoolExecutor # pip install futures - from socketserver import ThreadingMixIn - from wsgiref.simple_server import WSGIRequestHandler, WSGIServer, make_server - - class WSGIRefThreadingServer(ServerAdapter): - "Class implementing a WSGIRef server" - - def run(self, handler): - "runs the server" - - self.log = None - - if not self.quiet: - self.log = logging_conf( - self.options["logging_level"], - "wsgiref", - ) - - self_run = self # used in internal classes to access options and logger - - class PoolMixIn(ThreadingMixIn): - def process_request(self, request, client_address): - self.pool.submit( - self.process_request_thread, request, client_address - ) - - class ThreadingWSGIServer(PoolMixIn, WSGIServer): - daemon_threads = True - pool = ThreadPoolExecutor( - max_workers=get_workers(self.options, default=40) - ) - - class Server: - def __init__( - self, server_address=("127.0.0.1", 8000), handler_cls=None - ): - self.wsgi_app = None - self.listen, self.port = server_address - self.handler_cls = handler_cls - - def set_app(self, app): - self.wsgi_app = app - - def get_app(self): - return self.wsgi_app - - def serve_forever(self): - self.server = make_server( - self.listen, - self.port, - self.wsgi_app, - ThreadingWSGIServer, - self.handler_cls, - ) - - # openssl req -newkey rsa:4096 -new -x509 -keyout server.pem -out server.pem -days 365 -nodes - # openssl req -x509 -newkey rsa:4096 -nodes -out cert.pem -keyout key.pem -days 365 - # ./py4web.py run apps -s wsgirefThreadingServer --watch=off --port=8443 --ssl_cert=cert.pem --ssl_key=key.pem - # openssl s_client -showcerts -connect 127.0.0.1:8443 - - certfile = self_run.options.get("certfile", None) - - if certfile: - self.server.socket = ssl.wrap_socket( - self.server.socket, - certfile=certfile, - keyfile=self_run.options.get("keyfile", None), - ssl_version=ssl.PROTOCOL_SSLv23, - server_side=True, - do_handshake_on_connect=False, - ) - - self.server.serve_forever() - - class LogHandler(WSGIRequestHandler): - def address_string(self): # Prevent reverse DNS lookups please. - return self.client_address[0] - - def log_request(self, *args, **kw): - if not self_run.quiet: - WSGIRequestHandler.log_request(self, *args, **kw) - - def log_message(self, formatstr, *args): - if not self_run.quiet: # and ( not args[1] in ['200', '304']) : - msg = "%s - - [%s] %s" % ( - self.client_address[0], - self.log_date_time_string(), - formatstr % args, - ) - self_run.log.info(msg) - - # handler_cls = self.options.get("handler_class", LogHandler) - server_cls = Server - - if ":" in self.host: # Fix wsgiref for IPv6 addresses. - if getattr(server_cls, "address_family") == socket.AF_INET: - - class ServerClass(Server): - address_family = socket.AF_INET6 - - server_cls = ServerClass - - srv = make_server( - self.host, self.port, handler, server_cls, LogHandler - ) # handler_cls) - srv.serve_forever() - - return WSGIRefThreadingServer - - -wsgiTh = wsgirefThreadingServer - - -def rocketServer(): - "Builds and returns a rocket3 server" - - try: - from rocket3 import Rocket3 as Rocket - except ImportError: - from .rocket3 import Rocket3 as Rocket - - class RocketServer(ServerAdapter): - "Class implementing a rocket3 server" - - def run(self, handler): - "runs the server" - - if not self.quiet: - logging_conf( - self.options["logging_level"], - ) - - interface = ( - ( - self.host, - self.port, - self.options["keyfile"], - self.options["certfile"], - ) - if ( - self.options.get("certfile", None) - and self.options.get("keyfile", None) - ) - else (self.host, self.port) - ) - - server = Rocket(interface, "wsgi", dict(wsgi_app=handler)) - server.start() - - return RocketServer - - -""" -# how to write to server-adapters.log from controllers.py -# cp -a _scaffold test-salog - -import sys -import logging -from .common import logger -from .settings import APP_NAME -from threading import Lock - - -_srv_log=None -def log_info(mess, dbg=True, ): - def salog(pat='SA:'): - global _srv_log - if _srv_log and isinstance( _srv_log, logging.Logger ): - return _srv_log - hs= [e for e in logging.root.manager.loggerDict if e.startswith(pat) ] - if len(hs) == 0: - return logger - - sa_lock = Lock() - with sa_lock: - _srv_log = logging.getLogger(hs[0]) - - return _srv_log - - caller = f" > {APP_NAME} > {sys._getframe().f_back.f_code.co_name}" - dbg and salog().info(mess + caller) - -log_warn=log_info -log_debug=log_info - -log_warn('0'* 30 + ' ' +APP_NAME) - -@action("index") -@action.uses("index.html", auth, T) -def index(): - - log_warn('7'* 30 + ' ' +APP_NAME) - log_info('9'* 30 + ' ' +APP_NAME) - - user = auth.get_user() - message = T("Hello {first_name}").format(**user) if user else T("Hello") - actions = {"allowed_actions": auth.param.allowed_actions} - return dict(message=message, actions=actions) - -""" diff --git a/py4web/server_adapters/__init__.py b/py4web/server_adapters/__init__.py new file mode 100644 index 00000000..8b294ca7 --- /dev/null +++ b/py4web/server_adapters/__init__.py @@ -0,0 +1,78 @@ +__all__ = ["available", "unavailable", "blocking"] + +unavailable = [] +available = {} +blocking = {} + +# Web servers supported natively by ombott + +available["wsgiref"] = "wsgiref" +blocking["wsgiref"] = True + +try: + import tornado + + available["tornado"] = "tornado" + blocking["tornado"] = False +except ModuleNotFoundError: + unavailable.append("tornado") + + +try: + import waitress + + available["waitress"] = "waitress" + blocking["waitress"] = True +except ModuleNotFoundError: + unavailable.append("waitress") + +try: + import gunicorn + + available["gunicorn"] = "gunicorn" + blocking["gunicorn"] = False +except ModuleNotFoundError: + unavailable.append("gunicorn") + +# additional custom adaptrs + +try: + from .adapter_wsgiref import WSGIRefAdapter + + available["wsgiref+threaded"] = WSGIRefAdapter + blocking["wsgiref+threaded"] = True +except ModuleNotFoundError: + unavailable.append("wsgiref+threaded") + +try: + from .adapter_rocket3 import Rocket3Adapter + + available["rocket"] = Rocket3Adapter + blocking["rocket"] = True +except ModuleNotFoundError: + unavailable.append("rocket") + +try: + from .adapter_gevent import GeventAdapter + + available["gevent"] = GeventAdapter + blocking["gevent"] = False +except ModuleNotFoundError: + unavailable.append("gevent") + +try: + from .adapter_gunicorn_gevent import GunicornGeventAdapter + + available["gunicorn+gevent"] = GunicornGeventAdapter + blocking["gunicorn+gevent"] = False +except ModuleNotFoundError: + unavailable.append("gunicorn+gevent") + +try: + from .adapter_gevent_websockets import GeventWebsocketsAdapter + + available["gevent+websockets"] = GeventWebsocketsAdapter + blocking["gevent+websockets"] = False + +except ModuleNotFoundError: + unavailable.append("gevent+websockets") diff --git a/py4web/server_adapters/adapter_gevent.py b/py4web/server_adapters/adapter_gevent.py new file mode 100644 index 00000000..afc113a2 --- /dev/null +++ b/py4web/server_adapters/adapter_gevent.py @@ -0,0 +1,56 @@ +import ssl +import threading + +from gevent import local, pywsgi # pip install gevent +from ombott.server_adapters import ServerAdapter + +from .logging_utils import logging_conf + +# ./py4web.py run apps --watch=off -s gevent -L 20 +# +# ./py4web.py run apps -s gevent --watch=off --port=8443 --ssl_cert=cert.pem --ssl_key=key.pem -L 0 +# ./py4web.py run apps -s gevent --watch=off --host=192.168.1.161 --port=8443 --ssl_cert=server.pem -L 0 + + +class GeventAdapter(ServerAdapter): + "Defines a gevent server" + + def run(self, handler): + "runs the server" + + if not isinstance(threading.local(), local.local): + msg = "Ombott requires gevent.monkey.patch_all() (before import)" + raise RuntimeError(msg) + + logger = None # "default" + + if not self.quiet: + logger = logging_conf( + self.options["logging_level"], + "gevent", + ) + # logger.addHandler(logging.StreamHandler()) + + certfile = self.options.get("certfile", None) + + ssl_args = ( + dict( + certfile=certfile, + keyfile=self.options.get("keyfile", None), + ssl_version=ssl.PROTOCOL_SSLv23, + server_side=True, + do_handshake_on_connect=False, + ) + if certfile + else {} + ) + + server = pywsgi.WSGIServer( + (self.host, self.port), + handler, + log=logger, + error_log=logger, + **ssl_args, + ) + + server.serve_forever() diff --git a/py4web/server_adapters/adapter_gevent_websockets.py b/py4web/server_adapters/adapter_gevent_websockets.py new file mode 100644 index 00000000..a06f03ed --- /dev/null +++ b/py4web/server_adapters/adapter_gevent_websockets.py @@ -0,0 +1,46 @@ +# from geventwebsocket.handler import WebSocketHandler # pip install gevent-websocket +from gevent import pywsgi +from gevent_ws import WebSocketHandler # pip install gevent gevent-ws +from ombott.server_adapters import ServerAdapter + +from .logging_utils import logging_conf + +# https://stackoverflow.com/questions/5312311/secure-websockets-with-self-signed-certificate +# https://pypi.org/project/gevent-ws/ +# ./py4web.py run apps -s geventWebSocketServer --watch=off --ssl_cert=server.pem -H 192.168.1.161 -P 9000 -L 10 +# vi apps/_websocket/templates/index.html set: ws, wss, host, port +# firefox http://localhost:8000/_websocket +# firefox https://192.168.1.161:9000/_websocket test wss +# curl --insecure -I -H 'Upgrade: websocket' \ +# -H "Sec-WebSocket-Key: `openssl rand -base64 16`" \ +# -H 'Sec-WebSocket-Version: 13' \ +# -sSv https://192.168.1.161:9000/ + + +class GeventWebSocketAdapter(ServerAdapter): + "Class implementing a Gevent websocket server" + + def run(self, handler): + "Runs the server" + logger = None # "default" + + if not self.quiet: + logger = logging_conf( + self.options["logging_level"], + "gevent-ws", + ) + + args = dict( + handler_class=WebSocketHandler, + log=logger, + error_log=logger, + ) + + certfile = self.options.get("certfile") + keyfile = self.options.get("keyfile") + if certfile and keyfile: + args.update(certfile=certfile, keyfile=keyfile) + + server = pywsgi.WSGIServer((self.host, self.port), handler, **args) + + server.serve_forever() diff --git a/py4web/server_adapters/adapter_gunicorn_gevent.py b/py4web/server_adapters/adapter_gunicorn_gevent.py new file mode 100644 index 00000000..ddbd4016 --- /dev/null +++ b/py4web/server_adapters/adapter_gunicorn_gevent.py @@ -0,0 +1,191 @@ +import ast +import errno +import logging +import os +import socket +import subprocess +import sys +import threading + +from gevent import local # pip install gevent gunicorn setproctitle +from ombott.server_adapters import ServerAdapter + +from .logging_utils import check_level, get_log_file, logging_conf + + +def check_port(host="127.0.0.1", port=8000): + "Check the specified port is available and print debug info" + + if host.startswith("unix:/"): + socket_path = host[5:] + if os.path.exists(socket_path): + if port == 0: + if ( + subprocess.run( + ["ls", "-alFi", "socket_path"], shell=False, check=False + ).returncode + != 0 + ): + sys.exit(f"can't run gunicorn: {socket_path} exists") + elif port == 1: + subprocess.run( + "ps -ef | head -1; ps -ef | grep py4web | grep -v grep", + shell=True, + check=False, + ) + subprocess.run(["ls", "-alFi", socket_path], shell=False, check=False) + subprocess.run(["lsof", "-w", socket_path], shell=False, check=False) + elif port == 8000: + pass + print(f"gunicorn listening at: {host}") + return + + s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) + try: + s.bind((host, int(port))) + except socket.error as e: + if e.errno == errno.EADDRINUSE: + subprocess.run( + f"command -v lsof >/dev/null 2>&1 && ps -ef | head -1; ps -ef |" + f" grep py4web | grep -v grep && lsof -nPi:{port}", + shell=True, + check=False, + ) + sys.exit(f"{host}:{port} is already in use") + else: + sys.exit(f"{e}\n{host}:{port} cannot be acessed") + s.close() + + +def check_kv(kx, vx): + "convenience function" + if kx and vx and kx not in ("bind", "config"): + if vx.startswith("{") and vx.endswith("}"): + vx = ast.literal_eval(vx) + if vx == "None": + vx = None + return kx, vx + return None, None + + +def get_gunicorn_options( + gu_default="gunicorn.conf.py", + env_file="gunicorn.saenv", + env_key="GUNICORN_", +): + "Returns the default options" + if os.path.isfile(gu_default): + return {"use_python_config": gu_default, "config": gu_default} + + res_opts = {} + + if os.path.isfile(env_file): + with open(env_file, "r") as f: + lines = f.read().splitlines() + for line in lines: + line = line.strip() + if not line or line.startswith(("#", "[")): + continue + for e in ("export ", env_key): + line = line.replace(e, "", 1) + k, v = None, None + try: + k, v = line.split("=", 1) + k, v = k.strip().lower(), v.strip() + except (ValueError, AttributeError): + continue + k, v = check_kv(k, v) + if k is None: + continue + res_opts[k] = v + + if res_opts: + res_opts["config"] = env_file + return res_opts + + for k, v in os.environ.items(): + if k.startswith(env_key): + k = k.split("_", 1)[1].lower() + k, v = check_kv(k, v) + if k is None: + continue + res_opts[k] = v + + if res_opts: + res_opts["config"] = env_key + + return res_opts + + +class GunicornGeventAdapter(ServerAdapter): + "The gunicorn server adapter" + + def run(self, handler): + "runs the server" + + if isinstance(threading.local(), local.local): + print("gunicorn: monkey.patch_all() applied") + + try: + from gunicorn.app.base import Application + except ImportError as ex: + sys.exit(f"{ex}\nTry: pip install gunicorn gevent setproctitle") + + check_port(self.host, self.port) + + logger = None + + sa_bind = ( + self.host if self.host.startswith("unix:/") else f"{self.host}:{self.port}" + ) + + sa_config = { + "bind": sa_bind, # f"{self.host}:{self.port}", + "workers": self.options.get("workers", 10), + "certfile": self.options.get("certfile", None), + "keyfile": self.options.get("keyfile", None), + "accesslog": None, + "errorlog": None, + "proc_name": "sa_py4web", # ps a | grep py4web + "config": "sa_config", + # ( 'sa_config', 'GUNICORN_', 'gunicorn.saenv', 'gunicorn.conf.py' ) + } + + if not self.quiet: + level = check_level(self.options["logging_level"]) + log_file = get_log_file() + + logger = logging_conf(level) + log_to = "-" if log_file is None else log_file + + sa_config.update( + { + "loglevel": logging.getLevelName(level), + "access_log_format": '%(h)s %(l)s %(u)s %(t)s "%(r)s" %(s)s %(b)s "%(f)s" "%(a)s"', + "accesslog": log_to, + "errorlog": log_to, + } + ) + + class GunicornApplication(Application): + "A gunicorn application" + + def load_config(self): + "Loads the config" + sa_config.update(get_gunicorn_options()) + logger and logger.debug(sa_config) + + for k, v in sa_config.items(): + if k not in self.cfg.settings: + continue + self.cfg.set(k, v) + + for key in ("use_python_config", "usepy"): + if key in sa_config: + Application.load_config_from_file(self, sa_config[key]) + break + + def load(self): + return handler + + GunicornApplication().run() diff --git a/py4web/server_adapters/adapter_rocket3.py b/py4web/server_adapters/adapter_rocket3.py new file mode 100644 index 00000000..f9d5c7b1 --- /dev/null +++ b/py4web/server_adapters/adapter_rocket3.py @@ -0,0 +1,27 @@ +from ombott.server_adapters import ServerAdapter +from rocket3 import Rocket3 + +from .logging_utils import logging_conf + + +class Rocket3Adapter(ServerAdapter): + "Class implementing a rocket3 server" + + def run(self, handler): + "runs the server" + + if not self.quiet: + logging_conf( + self.options["logging_level"], + ) + + keyfile = self.options.get("keyfile") + certfile = self.options.get("certfile") + + if keyfile and certfile: + interface = (self.host, self.port, keyfile, certfile) + else: + interface = (self.host, self.port) + + server = Rocket3(interface, "wsgi", dict(wsgi_app=handler)) + server.start() diff --git a/py4web/server_adapters/adapter_wsgiref.py b/py4web/server_adapters/adapter_wsgiref.py new file mode 100644 index 00000000..19b7e204 --- /dev/null +++ b/py4web/server_adapters/adapter_wsgiref.py @@ -0,0 +1,109 @@ +"builds and returns a wsgiref threading server" +# https://www.electricmonk.nl/log/2016/02/15/multithreaded-dev-web-server-for-the-python-bottle-web-framework/ + +import socket +import ssl +from concurrent.futures import ThreadPoolExecutor # pip install futures +from socketserver import ThreadingMixIn +from wsgiref.simple_server import WSGIRequestHandler, WSGIServer, make_server + +from ombott.server_adapters import ServerAdapter + +from .logging_utils import logging_conf + + +class WSGIRefAdapter(ServerAdapter): + "Class implementing a WSGIRef server" + + def run(self, handler): + "runs the server" + + self.log = None + + if not self.quiet: + self.log = logging_conf( + self.options["logging_level"], + "wsgiref", + ) + + self_run = self # used in internal classes to access options and logger + + class LogHandler(WSGIRequestHandler): + def address_string(self): # Prevent reverse DNS lookups please. + return self.client_address[0] + + def log_request(self, *args, **kw): + if not self_run.quiet: + WSGIRequestHandler.log_request(self, *args, **kw) + + def log_message(self, formatstr, *args): + if not self_run.quiet: # and ( not args[1] in ['200', '304']) : + msg = "%s - - [%s] %s" % ( + self.client_address[0], + self.log_date_time_string(), + formatstr % args, + ) + self_run.log.info(msg) + + class PoolMixIn(ThreadingMixIn): + def process_request(self, request, client_address): + self.pool.submit(self.process_request_thread, request, client_address) + + class ThreadingWSGIServer(PoolMixIn, WSGIServer): + daemon_threads = True + pool = ThreadPoolExecutor(max_workers=self.options.get("workers", 40)) + + class Server: + def __init__(self, server_address=("127.0.0.1", 8000), handler_cls=None): + self.wsgi_app = None + self.listen, self.port = server_address + self.handler_cls = handler_cls + + def set_app(self, app): + self.wsgi_app = app + + def get_app(self): + return self.wsgi_app + + def serve_forever(self): + self.server = make_server( + self.listen, + self.port, + self.wsgi_app, + ThreadingWSGIServer, + self.handler_cls, + ) + + # openssl req -newkey rsa:4096 -new -x509 -keyout server.pem -out server.pem -days 365 -nodes + # openssl req -x509 -newkey rsa:4096 -nodes -out cert.pem -keyout key.pem -days 365 + # ./py4web.py run apps -s wsgirefThreadingServer --watch=off --port=8443 --ssl_cert=cert.pem --ssl_key=key.pem + # openssl s_client -showcerts -connect 127.0.0.1:8443 + + certfile = self_run.options.get("certfile", None) + + if certfile: + self.server.socket = ssl.wrap_socket( + self.server.socket, + certfile=certfile, + keyfile=self_run.options.get("keyfile", None), + ssl_version=ssl.PROTOCOL_SSLv23, + server_side=True, + do_handshake_on_connect=False, + ) + + self.server.serve_forever() + + server_cls = Server + + if ":" in self.host: # Fix wsgiref for IPv6 addresses. + if getattr(server_cls, "address_family") == socket.AF_INET: + + class ServerClass(Server): + address_family = socket.AF_INET6 + + server_cls = ServerClass + + srv = make_server( + self.host, self.port, handler, server_cls, LogHandler + ) # handler_cls) + srv.serve_forever() diff --git a/py4web/server_adapters/logging_utils.py b/py4web/server_adapters/logging_utils.py new file mode 100644 index 00000000..4dd4b8fa --- /dev/null +++ b/py4web/server_adapters/logging_utils.py @@ -0,0 +1,89 @@ +import logging +import os +import sys + + +def get_log_file(out_banner=True): + """ + Returns the filename for logging or None + Assumes: + export PY4WEB_ERRORLOG=/tmp # export PY4WEB_ERRORLOG= + if PY4WEB_ERRORLOG is :stderr or :stdout returns None + if PY4WEB_ERRORLOG is a folder returns the name of a logfile in that dir + if PY4WEB_ERRORLOG is a filename it returns that filename + if the out_banner argument is true, it outputs the filename + """ + log_dir = os.environ.get("PY4WEB_ERRORLOG", None) + if log_dir and not log_dir.startswith(":"): + if os.path.isdir(log_dir): + log_file = os.path.join(log_dir, "server-py4web.log") + else: + log_file = log_dir + if out_banner: + print(f"log_file: {log_file}") + return log_file + return None + + +def check_level(level): + "Check the level is a valid loglevel" + # lib/python3.7/logging/__init__.py + # CRITICAL = 50 + # FATAL = CRITICAL + # ERROR = 40 + # WARNING = 30 + # WARN = WARNING + # INFO = 20 + # DEBUG = 10 + # NOTSET = 0 + + return ( + level + if level + in ( + logging.CRITICAL, + logging.ERROR, + logging.WARN, + logging.INFO, + logging.DEBUG, + logging.NOTSET, + ) + else logging.WARN + ) + + +def logging_conf(level=logging.WARN, logger_name=__name__, fmode="w"): + "Configures logging" + + log_file = get_log_file() + log_to = {} + + if log_file: + if sys.version_info >= (3, 9): + log_to["filename"] = log_file + log_to["filemode"] = fmode + log_to["encoding"] = "utf-8" + else: + h = logging.FileHandler(log_file, mode=fmode, encoding="utf-8") + log_to.update({"handlers": [h]}) + + short_msg = "%(message)s > %(threadName)s > %(asctime)s.%(msecs)03d" + # long_msg = short_msg + " > %(funcName)s > %(filename)s:%(lineno)d > %(levelname)s" + + time_msg = "%H:%M:%S" + # date_time_msg = '%Y-%m-%d %H:%M:%S' + + logging.basicConfig( + format=short_msg, + datefmt=time_msg, + level=check_level(level), + **log_to, + ) + + if logger_name is None: + return None + + log = logging.getLogger("SA:" + logger_name) + log.propagate = True + + return log