Skip to main content

Typed settings based on attrs classes

Project description

Typed Settings

Load and merge settings from multiple different sources and present them in a structured, typed, and validated way!

Why?

There are many different config file formats and libraries. Many of them have a narrow scope, don't integrate well with other libs, or lack in typing support.

Typed Settings' goal is to enable you to load settings from any source (e.g., env vars, config files, vaults) and can convert values to anything you need.

You can extend Typed Settings to support config sources that aren't supported yet and its extensive documentation will help you on your way.

What can it be used for?

You can use Typed Settings in any context, e.g.:

  • server processes
  • containerized apps
  • command line applications
  • scripts and tools for scientific experiments and data analysis

What does it do?

  • It loads settings from multiple sources (e.g., env vars, config files, secret vaults) in a unified way and merges the loaded values. You can add loaders for sources we cannot imagine yet.

  • It can post-process loaded values. This allows value interpolation/templating or calling helpers that retrieve secrets from vaults. You can create and add any processors you can image if the built-in ones are not enough.

  • You can add a CLI on top to let users update the loaded settings via command line arguments. Click and argparse are currently supported.

  • Settings are cleanly structured and typed. The type annotations are used to convert the loaded settings to the proper types. This also includes higher level structures like dates, paths and various collections (lists, dicts, …). You can use attrs, dataclasses, or Pydantic to write settings classes.

    Types Settings uses the powerful and fast cattrs) by default and falls back to an internal converter if cattrs is not installed.

  • No mandatory requirements. Typed Settings works out-of-the box with dataclasses, argparse and its own converter.

The documentation contains a full list of all features.

Installation

Install and update using pip:

$ python -m pip install typed-settings

Typed Settings as no required dependencies (except for tomli on older Python versions). You can install dependencies for optional features via

$ python -m pip install typed-settings[<feature>,...]

Available features:

  • typed-settings[attrs]: Enable settings classes via attrs.
  • typed-settings[pydantic]: Enable settings classes via Pydantic.
  • typed-settings[cattrs]: Enable usage of the powerful and fast cattrs converter.
  • typed-settings[click]: Enable support for Click options.
  • typed-settings[option-groups]: Enable support for Click and Click option groups.
  • typed-settings[jinja]: Enable support for value interpolation with Jinja templates.
  • typed-settings[all]: Install all optional requirements.

Examples

Hello, World!, with env. vars.

This is a very simple example that demonstrates how you can load settings from environment variables.

# example.py
import attrs
import typed_settings as ts

@attrs.frozen
class Settings:
    option: str

settings = ts.load(cls=Settings, appname="example")
print(settings)
$ EXAMPLE_OPTION="Hello, World!" python example.py
Settings(option='Hello, World!')

Nested classes and config files

Settings classes can be nested. Config files define a different section for each class.

# example.py
import attrs
import click

import typed_settings as ts

@attrs.frozen
class Host:
    name: str
    port: int

@attrs.frozen
class Settings:
    host: Host
    endpoint: str
    retries: int = 3

settings = ts.load(
    cls=Settings, appname="example", config_files=["settings.toml"]
)
print(settings)
# settings.toml
[example]
endpoint = "/spam"

[example.host]
name = "example.com"
port = 443
$ python example.py
Settings(host=Host(name='example.com', port=443), endpoint='/spam', retries=3)

Configurable settings loaders

The first example used a convenience shortcut with pre-configured settings loaders. However, Typed Settings lets you explicitly configure which loaders are used and how they work:

# example.py
import attrs
import typed_settings as ts

@attrs.frozen
class Settings:
    option: str

settings = ts.load_settings(
    cls=Settings,
    loaders=[
        ts.FileLoader(
            files=[],
            env_var="EXAMPLE_SETTINGS",
            formats={
                "*.toml": ts.TomlFormat("example"),
            },
        ),
        ts.EnvLoader(prefix="EXAMPLE_"),
      ],
)
print(settings)
$ EXAMPLE_OPTION="Hello, World!" python example.py
Settings(option='Hello, World!')

In order to write your own loaders or support new file formats, you need to implement the Loader or FileFormat protocols.

You can also pass a custom cattrs converter to add support for additional Python types.

Command Line Interfaces

Typed Settings can generate a command line interfaces (CLI) based on your settings. These CLIs will load settings as described above and let users override the loades settings with command line argumments.

Typed Settings supports argparse and click.

Argparse

# example.py
import attrs
import typed_settings as ts

@attrs.frozen
class Settings:
    a_str: str = ts.option(default="default", help="A string")
    an_int: int = ts.option(default=3, help="An int")

@ts.cli(Settings, "example")
def main(settings):
    print(settings)

if __name__ == "__main__":
    main()
$ python example.py --help
usage: example.py [-h] [--a-str TEXT] [--an-int INT]

options:
  -h, --help    show this help message and exit

Settings:
  Settings options

  --a-str TEXT  A string [default: default]
  --an-int INT  An int [default: 3]
$ python example.py --a-str=spam --an-int=1
Settings(a_str='spam', an_int=1)

Click

# example.py
import attrs
import click
import typed_settings as ts

@attrs.frozen
class Settings:
    a_str: str = ts.option(default="default", help="A string")
    an_int: int = ts.option(default=3, help="An int")

@click.command()
@ts.click_options(Settings, "example")
def main(settings):
    print(settings)

if __name__ == "__main__":
    main()
$ python example.py --help
Usage: example.py [OPTIONS]

Options:
  --a-str TEXT      A string  [default: default]
  --an-int INTEGER  An int  [default: 3]
  --help            Show this message and exit.
$ python example.py --a-str=spam --an-int=1
Settings(a_str='spam', an_int=1)

Project Links

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

typed_settings-24.6.0.tar.gz (29.5 MB view details)

Uploaded Source

Built Distribution

typed_settings-24.6.0-py3-none-any.whl (58.4 kB view details)

Uploaded Python 3

File details

Details for the file typed_settings-24.6.0.tar.gz.

File metadata

  • Download URL: typed_settings-24.6.0.tar.gz
  • Upload date:
  • Size: 29.5 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.4.30

File hashes

Hashes for typed_settings-24.6.0.tar.gz
Algorithm Hash digest
SHA256 9a5595de33f80452a2038e018bc45508fffc238a8752f08f5c03dc8e7bc0d1e2
MD5 0f5e4d25b02878e11e8b5724f2ad4b06
BLAKE2b-256 c091f39cff1a6eb43ba5d8e7eb810405b0a7abed7b7033f7725a09796c625547

See more details on using hashes here.

File details

Details for the file typed_settings-24.6.0-py3-none-any.whl.

File metadata

File hashes

Hashes for typed_settings-24.6.0-py3-none-any.whl
Algorithm Hash digest
SHA256 22221598a821395700beaafd8eab0e1bc596b394767aa37e0926c5f74d0b6a62
MD5 70a4bdd9e9567b4922482edc0e467e5b
BLAKE2b-256 d665659c1bebd863edc9184715fd1cf2475926cead71d83535d980577bfa2439

See more details on using hashes here.

Supported by

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