Source code for server.core

from __future__ import annotations

import logging.config

from onegov.server.collection import ApplicationCollection
from webob.exc import HTTPNotFound, HTTPForbidden
from webob.request import BaseRequest
from urllib.parse import urlparse


from typing import cast, Any, Literal, TypedDict, TYPE_CHECKING
if TYPE_CHECKING:
    from _typeshed.wsgi import StartResponse, WSGIEnvironment
    from collections.abc import Callable, Iterable, Iterator
    from logging.config import (
        _DictConfigArgs,
        _FormatterConfiguration,
        _FilterConfiguration,
        _HandlerConfiguration,
        _LoggerConfiguration,
        _RootLoggerConfiguration)

    from .config import Config

    # FIXME: This is pretty gross, all because we allow to omit the version
[docs] class _OptionalDictConfigArgs(TypedDict, total=False):
[docs] version: Literal[1]
[docs] formatters: dict[str, _FormatterConfiguration]
[docs] filters: dict[str, _FilterConfiguration]
[docs] handlers: dict[str, _HandlerConfiguration]
[docs] loggers: dict[str, _LoggerConfiguration]
[docs] root: _RootLoggerConfiguration
[docs] incremental: bool
[docs] disable_existing_loggers: bool
[docs] local_hostnames = { '127.0.0.1', '::1', 'localhost' }
[docs] class Request(BaseRequest):
[docs] hostname_keys = ('HTTP_HOST', 'HTTP_X_VHM_HOST')
@property
[docs] def hostnames(self) -> Iterator[str]: """ Iterates through the hostnames of the request. """ for key in self.hostname_keys: if key in self.environ: hostname = urlparse(self.environ[key]).hostname if hostname is None: yield self.environ[key].split(':')[0] else: yield hostname
[docs] class Server: """ A WSGI application that hosts multiple WSGI applications in the same process. Not to be confused with Morepath's mounting functionality. The morepath applications hosted by this WSGI application are root applications, not mounted applications. See `Morepath's way of nesting applications <https://morepath.readthedocs.org/en/latest/app_reuse.html #nesting-applications>`_ Applications are hosted in two ways: 1. As static applications under a base path (`/app`) 2. As wildcard applications under a base path with wildcard (`/sites/*`) There is no further nesting and there is no way to run an application under `/`. The idea for this server is to run a number of WSGI applications that are relatively independent, but share a common framework. Though thought to be used with Morepath this module does not try to assume anything but a WSGI application. Since most of the time we *will* be running morepath applications, this server automatically configures morepath for the applications that depend on it. If morepath autoconfig is not desired, set ``configure_morepath`` to False. """ def __init__( self, config: Config, configure_morepath: bool = True, configure_logging: bool = True, post_mortem: bool = False, environ_overrides: dict[str, Any] | None = None, exception_hook: Callable[[WSGIEnvironment], Any] | None = None ):
[docs] self.applications = ApplicationCollection(config.applications)
[docs] self.wildcard_applications = { a.root for a in config.applications if not a.is_static }
if configure_logging: self.configure_logging(config.logging) if configure_morepath: self.configure_morepath()
[docs] self.post_mortem = post_mortem
[docs] self.environ_overrides = environ_overrides
[docs] self.exception_hook = exception_hook
[docs] def configure_logging( self, config: _OptionalDictConfigArgs | _DictConfigArgs ) -> None: """ Configures the python logging. :config: A dictionary that is understood by python's `logging.config.dictConfig` method. """ config = cast('_DictConfigArgs', config) config.setdefault('version', 1) logging.config.dictConfig(config)
[docs] def configure_morepath(self) -> None: """ Configures morepath automatically, if any application uses it. """ # morepath is only loaded if there's at lest one app depending on it if next(self.applications.morepath_applications(), None): import morepath morepath.autoscan()
[docs] def handle_request( self, environ: WSGIEnvironment, start_response: StartResponse ) -> Iterable[bytes]: if self.environ_overrides: environ.update(self.environ_overrides) request = Request(environ) path_fragments = request.path.split('/') # try to find the application that handles this path application_root = '/'.join(path_fragments[:2]) application = self.applications.get(application_root) if application is None: return HTTPNotFound()(environ, start_response) # give applications the ability to deal with exceptions try: # make sure the application accepts the given hostname for host in request.hostnames: if host not in local_hostnames: if not application.is_allowed_hostname(host): return HTTPForbidden()(environ, start_response) if application_root in self.wildcard_applications: base_path = '/'.join(path_fragments[:3]) application_id = ''.join(path_fragments[2:3]) # dealias the application id if application_id in application._aliases: application_id = application._aliases[application_id] else: base_path = application_root application_id = ''.join(path_fragments[1:2]) # happens if the root of a wildcard path is requested # ('/wildcard' from '/wildcard/*') - this is not allowed if not application_id: return HTTPNotFound()(environ, start_response) # dashes are not allowed in application ids and are automatically # replaced by underscores application_id = application_id.replace('-', '_') environ['PATH_INFO'] = environ['PATH_INFO'][len(base_path):] environ['SCRIPT_NAME'] = base_path application.set_application_base_path(base_path) if not application.is_allowed_application_id(application_id): return HTTPNotFound()(environ, start_response) application.set_application_id( f'{application.namespace}/{application_id}') return application(environ, start_response) except Exception as e: return application.handle_exception(e, environ, start_response)
[docs] def __call__( self, environ: WSGIEnvironment, start_response: StartResponse ) -> Iterable[bytes]: try: return self.handle_request(environ, start_response) except Exception: if self.exception_hook: self.exception_hook(environ) if self.post_mortem: import pdb; pdb.post_mortem() # noqa: E702 raise