# -*- coding: utf-8 -*- """A specialized map implementation to manage configuration and context information. Features: * Uses YAML configuration files * Use environment variables to override configs * Can pass a list of required parameters at initialization * Works with encrypted files * Accepts multiple config files * Can query information from consul * Does type coercion on strings * Composable configs * Ordered merge: downstream configs will override upstream * Reload from sources on SIGHUP Available source types: * Environment variables * Yaml file on file system * Yaml file in S3 * Consul * Hashicorp Vault Notes: * Keys are case-insensitive. """ from __future__ import absolute_import, division, print_function import inspect import logging import os import signal from .compat import string_types from .decorators import memoize, memoizemethod from .exceptions import AssignmentError, NotFoundError from .path import PackageFile from .type_coercion import listify from .type_coercion import typify log = logging.getLogger(__name__) @memoize def make_env_key(app_name, key): """Creates an environment key-equivalent for the given key""" key = key.replace('-', '_').replace(' ', '_') return str("_".join((x.upper() for x in (app_name, key)))) @memoize def reverse_env_key(app_name, key): app = app_name.upper() + '_' assert key.startswith(app), "{0} is not a(n) {1} environment key".format(key, app) return key[len(app):].lower() class Configuration(object): """A map implementation to manage configuration and context information. Values can be accessed (read, not assigned) as either a dict lookup (e.g. `config[key]`) or as an attribute (e.g. `config.key`). This class makes the foundational assumption of a yaml configuration file, as values in yaml are typed. This class allows overriding configuration keys with environment variables. Given an app name `foo` and a config parameter `bar: 15`, setting a `FOO_BAR` environment variable to `22` will override the value of `bar`. The type of `22` remains `int` because the underlying value of `15` is used to infer the type of the `FOO_BAR` environment variable. When an underlying parameter does not exist in a config file, the type is intelligently guessed. Args: app_name (str) config_sources (str or list, optional) required_parameters (iter, optional) Raises: InitializationError: on instantiation, when `required_parameters` are not found warns: on instantiation, when a given `config_file` cannot be read NotFoundError: when requesting a key that does not exist Examples: >>> for (key, value) in [('FOO_BAR', 22), ('FOO_BAZ', 'yes'), ('FOO_BANG', 'monkey')]: ... os.environ[key] = str(value) >>> context = Configuration('foo') >>> context.bar, context.bar.__class__ (22, <... 'int'>) >>> context['baz'], type(context['baz']) (True, <... 'bool'>) >>> context.bang, type(context.bang) ('monkey', <... 'str'>) """ def __init__(self, appname, config_sources=None, required_parameters=None, package=None): self.appname = appname self.package = package or inspect.getmodule(self).__package__ # Indicates if sources are being loaded for the first time or are being reloaded. self.__initial_load = True # The ordered list of sources from which to load key, value pairs. self.__sources = list() # The object that holds the actual key, value pairs after they're loaded from sources. # Keys are stored as all lower-case. Access by key is provided in a case-insensitive way. self._config_map = dict() self._registered_env_keys = set() # TODO: add explanation comments self._required_keys = set(listify(required_parameters)) if hasattr(signal, 'SIGHUP'): # Reload config on SIGHUP (UNIX only) self.__set_up_sighup_handler() self.__load_environment_keys() self.append_sources(config_sources) def append_sources(self, config_sources): force_reload = True for source in listify(config_sources): self.__append_source(source, force_reload) def append_required(self, required_parameters): self._required_keys.update(listify(required_parameters)) def __append_source(self, source, force_reload=False, _parent_source=None): source.parent_config = self self.__load_source(source, force_reload) source.parent_source = _parent_source self.__sources.append(source) def verify(self): self.__ensure_required_keys() return self def set_env(self, key, value): """Sets environment variables by prepending the app_name to `key`. Also registers the environment variable with the instance object preventing an otherwise-required call to `reload()`. """ os.environ[make_env_key(self.appname, key)] = str(value) # must coerce to string self._registered_env_keys.add(key) self._clear_memoization() def unset_env(self, key): """Removes an environment variable using the prepended app_name convention with `key`.""" os.environ.pop(make_env_key(self.appname, key), None) self._registered_env_keys.discard(key) self._clear_memoization() def _reload(self, force=False): """Reloads the configuration from the file and environment variables. Useful if using `os.environ` instead of this class' `set_env` method, or if the underlying configuration file is changed externally. """ self._config_map = dict() self._registered_env_keys = set() self.__reload_sources(force) self.__load_environment_keys() self.verify() self._clear_memoization() @memoizemethod # memoized for performance; always use self.set_env() instead of os.setenv() def __getitem__(self, key): key = key.lower() if key in self._registered_env_keys: from_env = os.getenv(make_env_key(self.appname, key)) from_sources = self._config_map.get(key) return typify(from_env, type(from_sources) if from_sources is not None else None) else: try: return self._config_map[key] except KeyError as e: raise NotFoundError(e) def __getattr__(self, key): return self[key] def get(self, key, default=None): try: return self[key] except KeyError: return default def __setitem__(self, *args): raise AssignmentError() def __iter__(self): for key in self._registered_env_keys | set(self._config_map.keys()): yield key def items(self): for key in self: yield key, self[key] def __load_source(self, source, force_reload=False): if force_reload and source.parent_source: # TODO: double-check case of reload without force reload for chained configs return items = source.dump(force_reload) if source.provides and not set(source.provides).issubset(items): raise NotImplementedError() # TODO: fix this additional_requirements = items.pop('additional_requirements', None) if isinstance(additional_requirements, string_types): additional_requirements = additional_requirements.split(',') self._required_keys |= set(listify(additional_requirements)) additional_sources = items.pop('additional_sources', None) self._config_map.update(items) if additional_sources: for src in additional_sources: class_name, kwargs = src.popitem() additional_source = globals()[class_name](**kwargs) self.__append_source(additional_source, force_reload, source) def __load_environment_keys(self): app_prefix = self.appname.upper() + '_' for env_key in os.environ: if env_key.startswith(app_prefix): self._registered_env_keys.add(reverse_env_key(self.appname, env_key)) # We don't actually add values to _config_map here. Rather, they're pulled # directly from os.environ at __getitem__ time. This allows for type casting # environment variables if possible. def __ensure_required_keys(self): available_keys = self._registered_env_keys | set(self._config_map.keys()) missing_keys = self._required_keys - available_keys if missing_keys: raise EnvironmentError("Required key(s) not found in environment\n" " or configuration sources.\n" " Missing Keys: {0}".format(list(missing_keys))) def _clear_memoization(self): self.__dict__.pop('_memoized_results', None) def __set_up_sighup_handler(self): def sighup_handler(signum, frame): if signum != signal.SIGHUP: return self._reload(True) if callable(self.__previous_sighup_handler): self.__previous_sighup_handler(signum, frame) self.__previous_sighup_handler = signal.getsignal(signal.SIGHUP) signal.signal(signal.SIGHUP, sighup_handler) class Source(object): _items = None _provides = None _parent_source = None @property def provides(self): return self._provides @property def items(self): return self.dump() def load(self): """Must return a key, value dict""" raise NotImplementedError() # pragma: no cover def dump(self, force_reload=False): if self._items is None or force_reload: self._items = self.load() return self._items @property def parent_config(self): return self._parent_config @parent_config.setter def parent_config(self, parent_config): self._parent_config = parent_config @property def parent_source(self): return self._parent_source @parent_source.setter def parent_source(self, parent_source): self._parent_source = parent_source class YamlSource(Source): def __init__(self, location, provides=None): self._location = location self._provides = provides if provides else None def load(self): with PackageFile(self._location, self.parent_config.package) as fh: import yaml contents = yaml.safe_load(fh) if self.provides is None: return contents else: return dict((key, contents[key]) for key in self.provides) class EnvironmentMappedSource(Source): """Load a full Source object given the value of an environment variable.""" def __init__(self, envvar, sourcemap): self._envvar = envvar self._sourcemap = sourcemap def load(self): mapped_source = self._sourcemap[self.parent_config[self._envvar]] mapped_source.parent_config = self.parent_config params = mapped_source.load() return params