py-microservice/config.py

125 lines
3.1 KiB
Python
Raw Normal View History

2023-10-04 19:03:20 +00:00
"""
Application configuration
values can be configured in the OS environment via KEY_NAME=...
or in a toml file using
[section]
foo = ...
bar_baz = ...
2023-10-04 20:25:21 +00:00
In the latter case, the section and name are uppercased and joined with `_`,
2023-10-04 19:03:20 +00:00
then inserted into the OS environment (in this case, resulting in
SECTION_FOO=... and SECTION_BAR_BAZ=... ).
The OS environment takes precedence - if a key is found in both env and config,
the config setting will be ignored.
2023-10-06 15:08:25 +00:00
Author(s): anatol.ulrich@basebox.io
Copyright (c) 2023 basebox GmbH. All rights reserved.
2023-10-04 19:03:20 +00:00
"""
2023-10-04 20:25:21 +00:00
# stdlib imports
2023-10-04 19:03:20 +00:00
from os import environ
from functools import wraps
2023-10-04 20:25:21 +00:00
# dependencies imports
try:
import tomllib
except ImportError:
import tomli as tomllib
2023-10-04 20:25:21 +00:00
import logging
# app imports
2023-10-04 19:03:20 +00:00
import consts
logger = logging.getLogger(consts.LOG_ROOT)
def load(file):
"""
2023-10-04 20:25:21 +00:00
Load configuration from toml and add to environment.
Skip anything already present there.
Configuration key names are converted according to the module documentation.
2023-10-04 19:03:20 +00:00
"""
if file is None:
logger.debug('no config file provided.')
else:
logger.debug('loading config')
2023-10-04 20:25:21 +00:00
with open(file, 'rb') as handle:
config = tomllib.load(handle)
for section, vals in config.items():
for key, val in vals.items():
effective_key = f'{section.upper()}_{key.upper()}'
if effective_key in environ:
logger.warning(
'%s already exists in os environment, skipping', effective_key)
else:
environ[effective_key] = str(val)
logger.debug('config/raw: %s = %s', effective_key,
environ[effective_key])
2023-10-04 19:03:20 +00:00
def env(key, default=None):
"""
retrieve a key from the OS environment.
throws `KeyError` when the key is not found and no default fallback is provided.
"""
if default is None:
return environ[key]
return environ.get(key, default=default)
def logged(f):
"""
2023-10-04 20:25:21 +00:00
logging wrapper for resolved configuration values
2023-10-04 19:03:20 +00:00
"""
@wraps(f)
def decorated(*args, **kwargs):
res = f(*args, **kwargs)
2023-10-04 20:25:21 +00:00
logger.debug('config/resolved: %s = %s', args[0], res)
2023-10-04 19:03:20 +00:00
return res
return decorated
@logged
def int_val(key, default=None):
"""
retrieve configuration value as int
"""
return int(env(key, default=default))
def strtobool(val):
2023-10-04 20:25:21 +00:00
"""Convert a string representation of truth to `True` or `False`.
2023-10-04 19:03:20 +00:00
True values are 'y', 'yes', 't', 'true', 'on', and '1'; false values
are 'n', 'no', 'f', 'false', 'off', and '0'. Raises ValueError if
'val' is anything else.
"""
val = val.lower()
if val in ("y", "yes", "t", "true", "on", "1"):
return True
elif val in ("n", "no", "f", "false", "off", "0"):
return False
else:
raise ValueError(f'Invalid bool string: {val}')
@logged
def bool_val(key, default=None):
"""
retrieve configuration value as bool
"""
return strtobool(env(key, default=default))
@logged
def str_val(key, default=None):
"""
retrieve configuration value as bool
"""
return env(key, default=default)