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]
filters: dict[str, _FilterConfiguration]
[docs]
handlers: dict[str, _HandlerConfiguration]
[docs]
loggers: dict[str, _LoggerConfiguration]
[docs]
root: _RootLoggerConfiguration
[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 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