Heare.io Configuration Utilities
Project description
heare-config
Configuration library used by projects under heare.io
Usage
heare-config allows developers to declare typed configuration using a code-as-schema syntax. The Setting class will infer the type of the property from the default parser.
Basic SettingsDefinition
from heare.config import SettingsDefinition, Setting
class MyConfig(SettingsDefinition):
foo = Setting(str, default="bazinga")
bar = Setting(float, default=1.0)
config: MyConfig = MyConfig.load()
config.foo.get() # "bazinga"
config.bar.get() # 1.0
The MyConfig.load()
will create an instance of MyConfig with GettableConfig objects, populated accordingly.
ListSettings
The ListSetting
is a version of Setting
that yields results as a list. Usage varies slightly between command line, environment variables, and config files.
from heare.config import SettingsDefinition, ListSetting, SettingAliases
class MyListConfig(SettingsDefinition):
numbers = ListSetting(int, default=[], aliases=SettingAliases(
flag='number'
))
config: MyListConfig = MyListConfig.load()
config.numbers.get() # []
Default Invocation
The settings for a definition can be specified in three ways: command line flags, environment variable, and config files, with conventions matching each format to the SettingsDefinition. By default, each setting property name is scoped by its definition class name, but will also have a short-name version for convenience, with formats relevant to the configuration source.
Command Line Flags
Command-line flags address config by a fully qualified flag name of the format <class name>.<property name>
,
a simple flag of the format <property name>
, or a short flag of the form <first char of property name>
.
# command line flags
$ ./main.py --foo FOO --bar 10.0
# fully qualified command line flag
$ ./main.py --MyConfig.foo FOO --MyConfig.bar 10.0
# command line short flags
$ ./main.py -f FOO -b 10.0
Multiple Values with ListSettings
Command-line flags can be specified multiple times. With a standard setting, the last specified flag will override any previous values. With a ListSetting, a list will be created with order matching the command line invocation.
$ ./main.py --number 1 --number 2
# fully qualified command line flag
$ ./main.py --MyListConfig.number 1 --MyListConfig.number 2
# command line short flags
$ ./main.py -f FOO -b 10.0
Note: It is invalid to mix formats of command line flags in a single invocation. The following example will yield a runtime error.
$ ./main.py --MyListConfig.number 1 --number 2 # foo == [1, 2]
Note: As the parsers share a common utility class, it is technically possible to merge multiple ListSettings together by use of multiple csv values for a flag. This is not considered a best practice, and may be deprecated in a future refactoring.
$ ./main.py --number 1,2,3 --number 4,5,6 # foo = [1,2,3,4,5,6]
Environment Variables
Environment variables address config by converting component names to upper snake_case, and joining parts with a double underscore __
.
# environment variables
$ MY_CONFIG__FOO="value" MY_CONFIG__BAR="10.0" ./main.py
$ FOO="value" BAR="10.0" ./main.py
Note: At this time, quotations and other escape characters are not supported.
Multiple Values with ListSettings
Environment Variables allow for multiple values to be specified for a single property using comma-separated values.
# environment variables
$ MY_LIST_CONFIG__NUMBERS="1,2,3" ./main.py
$ NUMBERS="1,2,3" ./main.py
Note: At this time, quotations and other escape characters are not supported.
Note: It is invalid to mix formats of environment variables in a single invocation. Values will be set based on precedence
.
Config Files
Config files address config with sections for the config class name, and matching property value names within the sections. Config file mappings do not support any aliases.
[MyConfig]
foo = "value"
bar = 10.0
Multiple Values with ListSettings
Config Files allow for multiple values to be specified for a single property using comma-separated values.
Note: At this time, quotations and other escape characters are not supported.
Collisions across Multiple Configuration Files
When multiple configuration files are specified and the files contain colliding section/properties, values will match the last specified file.
Type Enforcement
Type enforcement is handled when transforming
Precedence
If a configuration value is specified in multiple ways, the value in SettingsDefinition classes will be determined by precedence. There are two layers of precedence: precedence of settings sources (CLI, Environment, and Config Files), and within a settings source (when a property can be set multiple times).
Precedence of Settings Sources
The loader will check each settings source in the following order, and stop when first discovered.
- CLI Arguments
- Environment Variables
- Config Files
- Default from Setting
Precedence Within Settings Sources
Different settings sources will behave differently, and again differently based on the type of Setting
being used (singleton vs list).
Command Line Arguments for Settings and ListSettings
A Setting
specified via the CLI will take the last value specified at the command line.
$ ./main.py --foo bar --foo baz # MyConfig.foo == "baz"
A ListSetting
specified via the CLI will collect values specified at the command line, with the notable exception that mixed formats are
not allowed.
Environment Variables for Settings and ListSettings
Environment variables cannot contain multiple values within a single shell session. The most recent assignment of the variable specifies the value. It is considered invalid to specify multiple forms of the same environment variable, as it is potentially ambiguous.
$ FOO="bar" MY_CONFIG__FOO="baz" ./main.py # Raises an error, for both Setting and ListSetting
Config Files for Settings and ListSettings
Config files will be merged in the order specified. When section collisions occur, the values from the last file will override individual properties within the section.
# file1.ini
[MyConfig]
foo = bar
bar = 1.0
# file2.ini
[MyConfig]
bar = 2.0
# MyConfig.foo == bar
# MyConfig.bar == 2.0
Custom Aliases
The default aliases for each format can be optionally overloaded, to help when migrating existing applications.
Example Definition
from heare.config import SettingsDefinition, Setting, SettingAliases
class MyAliasedConfig(SettingsDefinition):
bar = Setting(str, aliases=SettingAliases(
flag='BAR',
short_flag='B',
env_variable='NOTBAR'
))
config: MyAliasedConfig = MyAliasedConfig.load()
Command Line Flags
$ ./main.py --MyAliasedConfig.BAR "value"
$ ./main.py --BAR "value"
$ ./main.py -B "value"
Environment Variables
Environment variables address config by converting component names to upper snake_case, and joining parts with a double underscore __
.
$ MY_CONFIG__FOO="value" ./main.py
$ FOO="value" ./main.py
$ MY_ALIASED_CONFIG__NOTBAR="value" ./main.py
$ NOTBAR="value" ./main.py
Using Multiple SettingsDefinitions
It's possible to declare multiple SettingsDefinitions
classes within a program. SettingsDefinition.load()
can be invoked on multiple classes,
and SettingsDefinition provides a convenience mechanism for discovering all SettingsDefinitions
.
from typing import Dict
from heare.config import SettingsDefinition
all_settings:Dict[type, SettingsDefinition] = {
definition: definition.load() for definition in SettingsDefinition.discover()
}
Naming Collisions with Multiple SettingsDefinitions
Property reuse is encouraged, but ambiguity is discouraged. As noted above, it is illegal to specify multiple formats of a Setting in a single invocation. Across settings sources, precedence handles this cleanly, however within a single source there is the potential for ambiguity.
The following example demonstrates two conflicting and potentially ambiguous configurations.
from heare.config import SettingsDefinition, Setting, ListSetting
class MyFirstConfig(SettingsDefinition):
foo = ListSetting(str)
class MySecondConfig(SettingsDefinition):
foo = Setting(str)
$ ./main.py --MyFirstConfig.foo=bar --foo=baz
In the above scenario, it's not clear whether the --foo
setting should be a member of MyFirstConfig.foo
or not. While this will more clearly fail
when MyFirstConfig.foo
and MySecondConfig.foo
are of different types, the ambiguity here allows for unexpected surprises. Instead, only unambiguous
invocations are allowed: where sharing is implicit and consistent, or distinct values are explicit.
$ ./main.py --foo=bar # valid!
# MyFirstConfig.foo = ["bar"]
# MySecondConfig.foo = "bar"
$ ./main.py --MyFirstConfig.foo=bar --MySecondConfig.foo=baz # valid!
# MyFirstConfig.foo = ["bar"]
# MySecondConfig.foo = "baz"
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file heare_config-1.0.7.tar.gz
.
File metadata
- Download URL: heare_config-1.0.7.tar.gz
- Upload date:
- Size: 14.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.1 CPython/3.9.19
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 4e98f9aa10b0363e854a710f85866646d52f0f8ae911e533641bc310b8295c34 |
|
MD5 | 331f5307a575542ed73bb33c81bc1a02 |
|
BLAKE2b-256 | d8eeb535ff530fa87e8b45f22c8c646a943bc65036c246d36f0fd64f357b4fb8 |
File details
Details for the file heare_config-1.0.7-py3-none-any.whl
.
File metadata
- Download URL: heare_config-1.0.7-py3-none-any.whl
- Upload date:
- Size: 11.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.1.1 CPython/3.9.19
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 8d73b4d3668837da9d810ab94d561cbc11b4a92751616ac8d966c9b13032cad9 |
|
MD5 | 4e97f43bd569149b189b877bc70bf09b |
|
BLAKE2b-256 | 392445c4635566a3a931657b1759637c30445968e11fd9a653bd8e136fa2cd94 |