Source code for configconfig.metaclass

#!/usr/bin/env python3
#
#  metaclass.py
"""
Metaclass for configuration values.
"""
#
#  Copyright © 2020 Dominic Davis-Foster <dominic@davis-foster.co.uk>
#
#  Permission is hereby granted, free of charge, to any person obtaining a copy
#  of this software and associated documentation files (the "Software"), to deal
#  in the Software without restriction, including without limitation the rights
#  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
#  copies of the Software, and to permit persons to whom the Software is
#  furnished to do so, subject to the following conditions:
#
#  The above copyright notice and this permission notice shall be included in all
#  copies or substantial portions of the Software.
#
#  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
#  EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
#  MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
#  IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
#  DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
#  OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE
#  OR OTHER DEALINGS IN THE SOFTWARE.
#

# stdlib
from abc import abstractmethod
from typing import TYPE_CHECKING, Any, Callable, Dict, Mapping, Optional, Type, cast

# this package
from configconfig.utils import basic_schema, get_json_type

__all__ = ["ConfigVarMeta"]

if TYPE_CHECKING:
	# this package
	from configconfig.configvar import ConfigVar


[docs]class ConfigVarMeta(type): """ Metaclass for configuration values. """ dtype: Type rtype: Type required: bool default: Any validator: Callable category: str __name__: str def __new__(cls, name: str, bases, dct: Dict): # noqa: D102,MAN001 x = cast("ConfigVar", super().__new__(cls, name, bases, dct)) def get(name: str, default: Any) -> Any: return dct.get(name, getattr(x, name, default)) x.dtype = get("dtype", Any) if "rtype" in dct: x.rtype = dct["rtype"] elif getattr(x, "rtype", Any) != Any: pass else: x.rtype = x.dtype x.required = get("required", False) x.default = get("default", '') x.validator = get("validator", lambda y: y) # type: ignore[assignment] x.category = get("category", "other") x.__name__ = dct.get("name", dct.get("__name__", x.__name__)) return x
[docs] def get_schema_entry(cls, schema: Optional[Dict] = None) -> Dict[str, Any]: """ Returns the JSON schema entry for this configuration value. :param schema: :return: Dictionary representation of the JSON schema. """ if schema is None: schema = { **basic_schema, "properties": {}, "required": [], } dtype = get_json_type(cls.dtype) if dtype is NotImplemented: raise NotImplementedError(cls.__name__, cls.dtype) else: schema["properties"][cls.__name__] = dtype if cls.required: schema["required"].append(cls.__name__) for line in (cls.__doc__ or '').split("\n\n"): line = ' '.join([p.strip() for p in line.split('\n') if p.strip()]) if line: schema["properties"][cls.__name__]["description"] = line break return schema
@property def schema_entry(cls) -> Dict[str, Any]: # noqa: D102 return cls.get_schema_entry()
[docs] def __call__(cls, raw_config_vars: Dict[str, Any]) -> Any: # type: ignore[override] """ Alias for :meth:`ConfigVar.get <.ConfigVar.get>`. Returns the value of the :class:`~configconfig.configvar.ConfigVar`. :param raw_config_vars: Dictionary to obtain the value from. :rtype: See the :attr:`ConfigVar.rtype <.ConfigVar.rtype>` attribute. """ return cls.get(raw_config_vars)
@abstractmethod def get(cls, raw_config_vars: Mapping[str, Any]): # pragma: no cover # noqa: D102,MAN002 return NotImplemented
[docs] def __repr__(self) -> str: """ Return a string representation of the :class:`~.ConfigVarMeta` class. """ return f"<ConfigVar {self.__name__!r}>"