Skip to main content

Unifies configuration sources into a Configuration object that can be bound to complex types, or accessed directly for configuration data.

Project description

appsettings2

A python library that unifies configuration sources into a Configuration object that can be bound to complex types, or accessed directly for configuration data.

It can be installed from PyPI through the usual methods.

This README is a high-level overview of core features. For complete documentation please visit https://appsettings2.readthedocs.io/.

Quick Start

Using appsettings2 is straightforward:

  1. Construct a ConfigurationBuilder object.
  2. Add one or more ConfigurationProvider objects to it.
  3. Call build(...) to get a Configuration object populated with configuration data.
from appsettings2 import *
from appsettings2.providers import *

config = ConfigurationBuilder()\
    .addProvider(JsonConfigurationProvider(f'appsettings.json'))\
    .addProvider(JsonConfigurationProvider(f'appsettings.Development.json', required=False))\
    .addProvider(EnvironmentConfigurationProvider())\
    .build()

print(config)

In the above example two JSON Configuration Providers are added to the builder. The first will fail if "appsettings.json" is not found, the second will succeed whether or not the "appsettings.Development.json" exists (because required=False). This allows for an optional development configuration to exist on development workstations without requiring a code change.

A third Configuration Provider is also added, an EnvironmentConfigurationProvider, which allows configuration to be loaded from Environment Variables.

The order of providers determines precedence of configuration data. Providers added last will take precedence over providers added first. Values which are not provided by later registrations will not override values provided by earlier registrations.

Given the example above, this means that configuration data provided via Environment Variables will override configuration data provided via JSON files.

Accessing Configuration Data

There are multiple ways of accessing Configuration data:

  • Using dynamic attributes which represent your configuration data.
  • Using configuration keys by calling get(...) and set(...) methods.
  • A dictionary-like interface exposing configuration data via indexer syntax.
  • Binding the configuration data to a class/object you define.

Direct Access via Configuration object

Consider the following code which demonstrates the first three methods mentioned above. All three of these methods are equivalent and return the same underlying value.

# access `Configuration` attributes
value = configuration.ConnectionStrings.SampleDb
# access using `get(...)`
value = configuration.get('ConnectionStrings__SampleDb')
value = configuration.get('ConnectionStrings:SampleDb')
value = configuration.get('ConnectionStrings').get('SampleDb')
# access using indexer
value = configuration['ConnectionStrings__SampleDb']
value = configuration['ConnectionStrings:SampleDb']
value = configuration['ConnectionStrings']['SampleDb']

When accessing hierarchical data by "key" you can use a double-underscore delimiter or a colon delimiter, they are equivalent.

Devs and Ops from different walks will likely prefer one form over the other, so both are supported.

Additionally, keys are case-insensitive (attribute names are not.)

Binding Configuration to Objects

It's possible to bind configuration data to complex types. While some of the implementation is currently naive, it should work well for the vast majority of use cases. If you find your particular case does not work well please reach out to me and I will work with you to implement a sensible solution. Consider the following Python code:

json = """{
  "ConnectionStrings": {
    "SampleDb": "my_cxn_string"
  },
  "EnableSwagger": true,
  "MaxBatchSize": 100
}"""

class ConnStrs:
    """An ugly class name to demonstrate the class name does not matter."""
    SampleDB:str

class AppSettings:
    ConnectionStrings:ConnStrs
    EnableSwagger:bool
    MaxBatchSize:int

configuration = ConfigurationBuilder()\
    .addProvider(JsonConfigurationProvider(json=json))
    .build()

settings = AppSettings()
configuration.bind(settings)

print(settings.ConnectionStrings.SampleDB) # prints "my_cxn_string"

The resulting settings object will contain all of the configuration data properly typed according to type hints.

It is also possible to bind to a subset of a configuration, building upon the above, consider the following:

connectionStrings = configuration.bind(ConnStrs(), 'ConnectionStrings')
print(connectionStrings.SampleDB)

Lastly, a cautious eye may have noticed that the input configuration and class definition have a casing difference. SampleDb vs SampleDB -- by design binding is case-insensitive. This ensures that automation/configuration systems which can only communicate in upper-case can be used to populate complex objects which follow a strict naming convention without burdening devs/devops with extra work.

Accessing Configuration Dictionary-like

Configuration is dict-like and in most cases can be used as if it were a dict where strict type checks would not otherwise prevent it.

Transform Configuration to Dictionary

You can transform a Configuration instance into a dictionary (as a copy.) If you have some chunk of code that can consume a dictionary but can't consume a Python object (it happens) then you can get at a proper dictionary instance as follows:

configuration = builder.build()
d = configuration.toDictionary()
print(d)

Providers

Custom Provider Development

You can implement custom Configuration Providers by subclassing ConfigurationProvider and implementing populateConfiguration(...).

For a peek at the simplicity of provider implementation, this is ConfigurationProvider:

class ConfigurationProvider(abstract):

    @abstractmethod
    def populateConfiguration(self, configuration:Configuration) -> None:
        """The ConfigurationProvider will populate the provided Configuration instance."""
        pass

Essentially, you read your configuration source and write the configuration data into the specified Configuration object. Much of the complexity in dealing with hierarchy and allocation is encapsulated within the impl of Configuration. As a result, most providers are less than 20 lines of functional code.

Helpers

There are several helpers available which simplify loading configurations. Consider the following example:

import appsettings2

config = appsettings2.getConfiguration()

You can read more about getConfiguration in the docs.

Contact

You can reach me on Discord or open an Issue on Github.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

appsettings2-1.2.5.tar.gz (13.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

appsettings2-1.2.5-py3-none-any.whl (16.3 kB view details)

Uploaded Python 3

File details

Details for the file appsettings2-1.2.5.tar.gz.

File metadata

  • Download URL: appsettings2-1.2.5.tar.gz
  • Upload date:
  • Size: 13.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.3+

File hashes

Hashes for appsettings2-1.2.5.tar.gz
Algorithm Hash digest
SHA256 65bec253e28e6bd42f7e223224d6d78de581a2d17a9ae06a7eff1728f1eedea6
MD5 b34115dfc87518a289998355757f4179
BLAKE2b-256 6a7e4c0df27c996bd0e2bf87a3fc6ba4324b033c4dc5fdb6ce243bc9353e9aaf

See more details on using hashes here.

File details

Details for the file appsettings2-1.2.5-py3-none-any.whl.

File metadata

  • Download URL: appsettings2-1.2.5-py3-none-any.whl
  • Upload date:
  • Size: 16.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.3+

File hashes

Hashes for appsettings2-1.2.5-py3-none-any.whl
Algorithm Hash digest
SHA256 f757d9d61465b4a6a31c64e26b93bebca6ef9e74fa2e177bec7b4754f7d47592
MD5 cac56c42933de4ca0039fce18a1c0cf3
BLAKE2b-256 c036a77c57af422d948a870b60494771f020938b2422f5163af514b3ac81ba4c

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page