You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
338 lines
9.7 KiB
338 lines
9.7 KiB
10 months ago
|
"""Inspecting module"""
|
||
|
from __future__ import annotations
|
||
|
|
||
|
import json
|
||
|
import sys
|
||
|
from functools import partial
|
||
|
from pathlib import PosixPath
|
||
|
from typing import Any
|
||
|
from typing import Callable
|
||
|
from typing import Literal
|
||
|
from typing import Optional
|
||
|
from typing import Protocol
|
||
|
from typing import TextIO
|
||
|
from typing import TYPE_CHECKING
|
||
|
from typing import Union
|
||
|
|
||
|
from dynaconf.loaders.base import SourceMetadata
|
||
|
from dynaconf.utils.boxing import DynaBox
|
||
|
from dynaconf.utils.functional import empty
|
||
|
from dynaconf.vendor.box.box_list import BoxList
|
||
|
from dynaconf.vendor.ruamel.yaml import YAML
|
||
|
|
||
|
if TYPE_CHECKING: # pragma: no cover
|
||
|
from dynaconf.base import LazySettings, Settings
|
||
|
|
||
|
|
||
|
# Dumpers config
|
||
|
|
||
|
json_pretty = partial(json.dump, indent=2)
|
||
|
json_compact = json.dump
|
||
|
|
||
|
builtin_dumpers = {
|
||
|
"yaml": YAML().dump,
|
||
|
"json": json_pretty,
|
||
|
"json-compact": json_compact,
|
||
|
}
|
||
|
|
||
|
DumperPreset = Union[Literal["yaml"], Literal["json"], Literal["json-compact"]]
|
||
|
|
||
|
|
||
|
class DumperType(Protocol):
|
||
|
def __call__(
|
||
|
self, data: dict, text_stream: TextIO
|
||
|
) -> None: # pragma: no cover
|
||
|
...
|
||
|
|
||
|
|
||
|
class ReportBuilderType(Protocol):
|
||
|
def __call__(
|
||
|
self,
|
||
|
*,
|
||
|
key: str | None,
|
||
|
env: str | None,
|
||
|
new_first: bool | None,
|
||
|
include_internal: bool | None,
|
||
|
history_limit: int | None,
|
||
|
current: Any,
|
||
|
history: list[dict] | None,
|
||
|
) -> dict: # pragma: no cover
|
||
|
...
|
||
|
|
||
|
|
||
|
class KeyNotFoundError(Exception):
|
||
|
pass
|
||
|
|
||
|
|
||
|
class EnvNotFoundError(Exception):
|
||
|
pass
|
||
|
|
||
|
|
||
|
class OutputFormatError(Exception):
|
||
|
pass
|
||
|
|
||
|
|
||
|
def inspect_settings(
|
||
|
settings: Settings | LazySettings,
|
||
|
key: str | None = None,
|
||
|
env: str | None = None,
|
||
|
*,
|
||
|
new_first: bool = True,
|
||
|
history_limit: int | None = None,
|
||
|
include_internal: bool = False,
|
||
|
to_file: str | PosixPath | None = None,
|
||
|
print_report: bool = False,
|
||
|
dumper: DumperPreset | DumperType | None = None,
|
||
|
report_builder: ReportBuilderType | None = None,
|
||
|
):
|
||
|
"""
|
||
|
Print and return the loading history of a settings object.
|
||
|
|
||
|
Optional arguments must be provided as kwargs.
|
||
|
|
||
|
:param settings: A Dynaconf instance
|
||
|
:param key: String dotted path. E.g "path.to.key"
|
||
|
:param env: Filter by this env
|
||
|
|
||
|
:param new_first: If True, uses newest to oldest loading order
|
||
|
:param history_limit: Limits how many entries are shown
|
||
|
:param include_internal: If True, include internal loaders (e.g. defaults).
|
||
|
This has effect only if key is not provided.
|
||
|
:param to_file: If specified, write to this filename
|
||
|
:param print_report: If true, prints the dumped report to stdout
|
||
|
:param dumper: Accepts preset strings (e.g. "yaml", "json") or custom
|
||
|
dumper callable ``(dict, TextIO) -> None``. Defaults to "yaml"
|
||
|
:param report_builder: if provided, it is used to generate the report
|
||
|
|
||
|
:return: Dict with a dict containing report data
|
||
|
:rtype: dict
|
||
|
"""
|
||
|
# choose dumper and report builder
|
||
|
if dumper is None:
|
||
|
_dumper = builtin_dumpers["yaml"]
|
||
|
elif isinstance(dumper, str):
|
||
|
_dumper = builtin_dumpers.get(dumper)
|
||
|
if _dumper is None:
|
||
|
raise OutputFormatError(
|
||
|
f"The desired format is not available: {dumper!r}"
|
||
|
)
|
||
|
else:
|
||
|
_dumper = dumper
|
||
|
|
||
|
_report_builder = report_builder or _default_report_builder
|
||
|
|
||
|
# get history and apply optional arguments
|
||
|
original_settings = settings
|
||
|
|
||
|
env_filter = None # type: ignore
|
||
|
if env:
|
||
|
settings = settings.from_env(env)
|
||
|
registered_envs = {
|
||
|
src_meta.env for src_meta in settings._loaded_by_loaders.keys()
|
||
|
}
|
||
|
if env.lower() not in registered_envs:
|
||
|
raise EnvNotFoundError(f"The requested env is not valid: {env!r}")
|
||
|
|
||
|
def env_filter(src: SourceMetadata) -> bool: # noqa: F811
|
||
|
return src.env.lower() == env.lower()
|
||
|
|
||
|
history = get_history(
|
||
|
original_settings,
|
||
|
key=key,
|
||
|
filter_callable=env_filter,
|
||
|
include_internal=include_internal,
|
||
|
)
|
||
|
|
||
|
if new_first:
|
||
|
history.reverse()
|
||
|
|
||
|
if history_limit:
|
||
|
history = history[:history_limit]
|
||
|
|
||
|
if key:
|
||
|
current_value = settings.get(key)
|
||
|
else:
|
||
|
current_value = settings.as_dict()
|
||
|
|
||
|
# format output
|
||
|
dict_report = _report_builder(
|
||
|
history=history,
|
||
|
current=current_value,
|
||
|
key=key,
|
||
|
env=env,
|
||
|
new_first=new_first,
|
||
|
history_limit=history_limit,
|
||
|
include_internal=include_internal,
|
||
|
)
|
||
|
|
||
|
dict_report["current"] = _ensure_serializable(dict_report["current"])
|
||
|
|
||
|
# write to stdout AND/OR to file AND return
|
||
|
if to_file is not None:
|
||
|
_encoding = settings.get("ENCODER_FOR_DYNACONF")
|
||
|
with open(to_file, "w", encoding=_encoding) as file:
|
||
|
_dumper(dict_report, file)
|
||
|
|
||
|
if print_report is True:
|
||
|
_dumper(dict_report, sys.stdout)
|
||
|
|
||
|
return dict_report
|
||
|
|
||
|
|
||
|
def _default_report_builder(**kwargs) -> dict:
|
||
|
"""
|
||
|
Default inspect report builder.
|
||
|
|
||
|
Accept the kwargs passed inside `inspect_settings` and returns a dict with
|
||
|
{header, current, history} as top-level keys.
|
||
|
"""
|
||
|
return {
|
||
|
"header": {
|
||
|
"env_filter": str(kwargs.get("env")),
|
||
|
"key_filter": str(kwargs.get("key")),
|
||
|
"new_first": str(kwargs.get("new_first")),
|
||
|
"history_limit": str(kwargs.get("history_limit")),
|
||
|
"include_internal": str(kwargs.get("include_internal")),
|
||
|
},
|
||
|
"current": kwargs.get("current"),
|
||
|
"history": kwargs.get("history"),
|
||
|
}
|
||
|
|
||
|
|
||
|
def get_history(
|
||
|
obj: Settings | LazySettings,
|
||
|
key: str | None = None,
|
||
|
*,
|
||
|
filter_callable: Callable[[SourceMetadata], bool] | None = None,
|
||
|
include_internal: bool = False,
|
||
|
history_limit: int | None = None,
|
||
|
) -> list[dict]:
|
||
|
"""
|
||
|
Gets data from `settings.loaded_by_loaders` in order of loading with
|
||
|
optional filtering options.
|
||
|
|
||
|
Returns a list of dict in new-first order, where the dict contains the
|
||
|
data and it's source metadata.
|
||
|
|
||
|
:param obj: Setting object which contain the data
|
||
|
:param key: Key path to desired key. Use all if not provided
|
||
|
:param filter_callable: Takes SourceMetadata and returns a boolean
|
||
|
:param include_internal: If True, include internal loaders (e.g. defaults).
|
||
|
This has effect only if key is not provided.
|
||
|
history_limit: limits how many entries are shown
|
||
|
|
||
|
Example:
|
||
|
>>> settings = Dynaconf(...)
|
||
|
>>> _get_history(settings)
|
||
|
[
|
||
|
{
|
||
|
"loader": "yaml"
|
||
|
"identifier": "path/to/file.yml"
|
||
|
"env": "default"
|
||
|
"data": {"foo": 123, "spam": "eggs"}
|
||
|
},
|
||
|
...
|
||
|
]
|
||
|
"""
|
||
|
if filter_callable is None:
|
||
|
filter_callable = lambda x: True # noqa
|
||
|
|
||
|
sep = obj.get("NESTED_SEPARATOR_FOR_DYNACONF", "__")
|
||
|
|
||
|
# trigger key based hooks
|
||
|
if key:
|
||
|
obj.get(key) # noqa
|
||
|
|
||
|
internal_identifiers = ["default_settings", "_root_path"]
|
||
|
result = []
|
||
|
for source_metadata, data in obj._loaded_by_loaders.items():
|
||
|
# filter by source_metadata
|
||
|
if filter_callable(source_metadata) is False:
|
||
|
continue
|
||
|
|
||
|
# filter by internal identifiers
|
||
|
if (
|
||
|
not key
|
||
|
and include_internal is False
|
||
|
and source_metadata.identifier in internal_identifiers
|
||
|
):
|
||
|
continue # skip: internal loaders
|
||
|
|
||
|
# filter by key path
|
||
|
try:
|
||
|
data = _get_data_by_key(data, key, sep=sep) if key else data
|
||
|
except KeyError:
|
||
|
continue # skip: source doesn't contain the requested key
|
||
|
|
||
|
# Normalize output
|
||
|
data = _ensure_serializable(data)
|
||
|
result.append({**source_metadata._asdict(), "value": data})
|
||
|
|
||
|
if key and not result:
|
||
|
# Key may be set in obj but history not tracked
|
||
|
if (data := obj.get(key, empty)) is not empty:
|
||
|
generic_source_metadata = SourceMetadata(
|
||
|
loader="undefined",
|
||
|
identifier="undefined",
|
||
|
)
|
||
|
data = _ensure_serializable(data)
|
||
|
result.append({**generic_source_metadata._asdict(), "value": data})
|
||
|
|
||
|
# Raise if still not found
|
||
|
if key and not result:
|
||
|
raise KeyNotFoundError(f"The requested key was not found: {key!r}")
|
||
|
|
||
|
return result
|
||
|
|
||
|
|
||
|
def _ensure_serializable(data: BoxList | DynaBox) -> dict | list:
|
||
|
"""
|
||
|
Converts box dict or list types to regular python dict or list
|
||
|
Bypasses other values.
|
||
|
{
|
||
|
"foo": [1,2,3, {"a": "A", "b": "B"}],
|
||
|
"bar": {"a": "A", "b": [1,2,3]},
|
||
|
}
|
||
|
"""
|
||
|
if isinstance(data, (BoxList, list)):
|
||
|
return [_ensure_serializable(v) for v in data]
|
||
|
elif isinstance(data, (DynaBox, dict)):
|
||
|
return {
|
||
|
k: _ensure_serializable(v) for k, v in data.items() # type: ignore
|
||
|
}
|
||
|
else:
|
||
|
return data if isinstance(data, (int, bool, float)) else str(data)
|
||
|
|
||
|
|
||
|
def _get_data_by_key(
|
||
|
data: dict,
|
||
|
key_dotted_path: str,
|
||
|
default: Any = None,
|
||
|
sep="__",
|
||
|
):
|
||
|
"""
|
||
|
Returns value found in data[key] using dot-path str (e.g, "path.to.key").
|
||
|
Raises KeyError if not found
|
||
|
"""
|
||
|
if not isinstance(data, DynaBox):
|
||
|
data = DynaBox(data) # DynaBox can handle insensitive keys
|
||
|
|
||
|
if sep in key_dotted_path:
|
||
|
key_dotted_path = key_dotted_path.replace(sep, ".")
|
||
|
|
||
|
def traverse_data(data, path):
|
||
|
# transform `a.b.c` in successive calls to `data['a']['b']['c']`
|
||
|
path = path.split(".")
|
||
|
root_key, nested_keys = path[0], path[1:]
|
||
|
result = data[root_key]
|
||
|
for key in nested_keys:
|
||
|
result = result[key]
|
||
|
return result
|
||
|
|
||
|
try:
|
||
|
return traverse_data(data, key_dotted_path)
|
||
|
except KeyError:
|
||
|
if not default:
|
||
|
raise KeyError(f"Path not found in data: {key_dotted_path!r}")
|
||
|
return default
|