Metadata-Version: 2.1
Name: flexigurator
Version: 0.5.1
Summary: Python Configuration solution.
Author: Thomas Bos
Author-email: thymer.bos217@gmail.com
Requires-Python: >=3.11,<4.0
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.11
Requires-Dist: confz (>=1.8.1,<2.0.0)
Requires-Dist: pydantic (>=1.10.8,<2.0.0)
Requires-Dist: pylint (>=2.17.4,<3.0.0)
Requires-Dist: pyyaml (>=6.0,<7.0)
Description-Content-Type: text/markdown

# Flexigurator

**Flexigurator** is a collection of useful tools when working with configuration.
Flexigurator builds on top of [ConfZ](https://confz.readthedocs.io/en/latest/) and [Pydantic](https://docs.pydantic.dev/latest/)
while adding some extra functionality to make your life easier when testing and deploying.


## Examples

### `config_patch`
When using ConfZ to define a configuration singleton using a `.yaml` file,

```yaml
# config.yaml
some_string: default
some_int: 42
```


```python
from confz import ConfZ, ConfZFileSource

class Config(ConfZ):
    some_string: str
    some_int: int

    CONFIG_SOURCES = ConfZFileSource("config.yaml")
```

configuration variables can be easily hotswapped using `patch_config`:

```python
from flexigurator import patch_config


data = dict(some_int=9001)


with patch_config(Config, data):
    Config().some_int     # 9001

    
Config().some_int     # 42
```

Importantly, the new data does not need to be complete.

### `ConfigVersions`
Allows for easy storing and on-demand loading of configuration versions.

```python
from pathlib import Path
from confz import ConfZFileSource
from flexigurator import ConfigVersions, DirectorySource

class Config(ConfZ):  # type: ignore
    a: int
    b: int

class Configs(ConfigVersions):
    CONFIG_CLASS = Config
    BASE = dict(a=1, b=2)  # Optional base version which is loaded before other versions
    test = ConfZFileSource("configs/test.yaml")
    folder = DirectorySource(Path("configs/versions"))

with Configs().version("test"):
    print(Config())  # Configuration from the ConfZFileSource is now loaded

with Configs().version("folder.version_1"):
    print(Config())  # Configuration from "configs/versions/version_1.yaml" is loaded
```


### `placeholder`
When having nested, optional `BaseModel`s in your `Config`,

```python
class Config(ConfZ):
    ui: UIConfig | None
    server: ServerConfig | None    
```

it would be nice if trying to load configuration automatically throws an exception if it is not configured.
The `placeholder` method provides such functionality, and it can be used as follows:

```python
from flexigurator import placeholder, patch_config


class Config(ConfZ):
    ui: UIConfig = placeholder(UIConfig)
    server: ServerConfig = placeholder(UIConfig)
    

Config().server.ip_address  # NotConfiguredException!
```

This removes the need for `None`-checking as exception handling is done behind the scenes.


### `ConfigForm`
`ConfigForm` allows for the easy creation of forms for `ConfZ` or `BaseModel` classes.
Forms are generated using [Json Editor](https://github.com/json-editor/json-editor) and are served using [FastAPI](https://fastapi.tiangolo.com).
Having instantiated `ConfigForm`,

```python
# form.py
from flexigurator import ConfigForm


app = ConfigForm(Config, "save/path", "templates/path")
```

The server can be started using (for instance) [Uvicorn](https://www.uvicorn.org):

```bash
uvicorn form:app
```


## Installation
Flexigurator is available on [PyPi](https://pypi.org/project/flexigurator/0.3.0/#description) and can be installed using pip:

```bash
pip install flexigurator
```

