Source code for server.config

from __future__ import annotations

import yaml

from onegov.server import errors
from onegov.server.utils import load_class


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from _typeshed import StrOrBytesPath
    from typing import Self

    from .application import Application


[docs] class Config: """ Represents the configuration of the server. """ def __init__(self, configuration: dict[str, Any]): """ Load the given configuration, which is a list of dictionaries. Each dictionary is loaded by the :class:`ApplicationConfig` class. See :class:`ApplicationConfig` for a documentation of all configuration values. """
[docs] self.applications = [ ApplicationConfig(a) for a in configuration.get('applications', ()) ]
[docs] self.mail_queues = configuration.get('mail_queues', {})
[docs] self.logging = configuration.get('logging', {})
all_namespaces = [a.namespace for a in self.applications] unique_namespaces = set(all_namespaces) if len(unique_namespaces) != len(all_namespaces): raise errors.ApplicationConflictError( 'Not all namespaces are unique') @classmethod
[docs] def from_yaml_file(cls, yaml_file: StrOrBytesPath) -> Self: """ Load the given yaml file and return a new Configuration instance with the configuration values found in the yaml file. """ with open(yaml_file) as fp: return cls(yaml.safe_load(fp))
@classmethod
[docs] def from_yaml_string(cls, yaml_string: str | bytes) -> Self: """ Load the given yaml string and return a new Configuration instance with the configuration values found in the yaml string. """ return cls(yaml.safe_load(yaml_string))
[docs] class ApplicationConfig: """ Represents an application config entry loaded from a dictionary like this:: { 'path': '/application', 'application': 'my.module.StaticApp', 'namespace': 'my-namespace' 'configuration': { 'my': 'custom', 'config': 'values' } } It may contain the following keys: :path: The path of the application is the prefix under which the application is running. For each path onegov.server loads and caches an application. Applications are basically WSGI containers that are run independently in the same process. See :class:`~.application.Application` for more. There are two types of applications, static applications without a wildcard (`*`) in their path and wildcard applications *with* a wildcard in their path. Static applications always have the same application_id (`ns/static`, if the path is `/static` and the namespace is `ns`). Wildcard applications have the application_id set to the wildcard part of the path: (`/wildcard/*` can result in an applicaiton_id of `ns/blog` if `/wildcard/blog` is opened and the namespace is `ns`). See also: :meth:`~.application.Application.set_application_id`. Nested paths are not supported. `/static` works and `/wildcard/*` works, but not `/static/site` or `/wildcard/site/*`. :application: The application class or string to an application class that inherits from :class:`~.application.Application`. If `application` is a string, the class it points to is loaded immediately. :namespace: Each application has a namespace that must be unique. It is used make the application_id unique. Dashes in the namespace are replaced by underscores. :configuration: A dictionary that is passed to the application once it is initialized. See :meth:`.application.Application.configure_application`. """
[docs] __slots__ = ('_cfg',)
def __init__(self, configuration: dict[str, Any]):
[docs] self._cfg = configuration
assert self.path != '/' assert self.path.count('*') <= 1 assert self.path.count('/') <= 2 assert self.path.startswith('/') assert '/' not in self.namespace @property
[docs] def path(self) -> str: return self._cfg['path'].rstrip('/')
@property
[docs] def namespace(self) -> str: return self._cfg['namespace'].replace('-', '_')
@property
[docs] def application_class(self) -> type[Application]: application_class = load_class(self._cfg['application']) if application_class is None: raise errors.ApplicationConfigError( 'The application class could not be found.') return application_class
@property
[docs] def configuration(self) -> dict[str, Any]: return self._cfg.get('configuration', {})
@property
[docs] def root(self) -> str: """ The path without the wildcard such that `/app` and `/app/*` produce the same root (`/app`). """ return self.path.rstrip('/*')
@property
[docs] def is_static(self) -> bool: """ True if the application is static (has no '*' in the path). """ return '*' not in self.path