Use file secrets in nested pydantic-settings models instead of built-in SecretsSettingsSource
Project description
pydantic-file-secrets ๐
Use file secrets in nested pydantic-settings models instead of built-in SecretsSettingsSource
This project is inspired by discussions in Pydantic Settings repository and proposes solution to #30 and #154.
Features
- Plain or nested directory layout:
secrets/dir__keyorsecrets/dir/key - Respects
env_prefix,env_nested_delimiterand other config options - Implements config options
secrets_prefix,secrets_nested_delimiterand more to configure secrets and env vars independently - Drop-in replacement of standard
SecretsSettingsSource - Pure Python thin wrapper over standard
EnvSettingsSource - No third party dependencies except
pydantic-settings - Fully typed
- 100% test coverage
Installation
$ pip install pydantic-file-secrets
Motivation
Nested Pydantic config can contain nested models with secret entries, as well as secrets in top level config. In dockerized environment, these entries may be read from file system, e.g. /run/secrets when using Docker Secrets:
from pydantic import BaseModel, Secret
from pydantic_settings import BaseSettings, SettingsConfigDict
class DbSettings(BaseModel):
user: str
passwd: Secret[str] # secret in nested model
class Settings(BaseSettings):
app_key: Secret[str] # secret in root model
db: DbSettings
model_config = SettingsConfigDict(
secrets_dir='/run/secrets',
)
Usage
Plain secrets directory layout
๐ secrets
โโโ ๐ app_key
โโโ ๐ db__passwd
from pydantic import BaseModel, SecretStr
from pydantic_file_secrets import FileSecretsSettingsSource, SettingsConfigDict
from pydantic_settings import BaseSettings
from pydantic_settings.sources import PydanticBaseSettingsSource
class DbSettings(BaseModel):
passwd: SecretStr
class Settings(BaseSettings):
app_key: SecretStr
db: DbSettings
model_config = SettingsConfigDict(
secrets_dir='secrets',
secrets_nested_delimiter='__',
)
@classmethod
def settings_customise_sources(
cls,
settings_cls: type[BaseSettings],
init_settings: PydanticBaseSettingsSource,
env_settings: PydanticBaseSettingsSource,
dotenv_settings: PydanticBaseSettingsSource,
file_secret_settings: PydanticBaseSettingsSource,
) -> tuple[PydanticBaseSettingsSource, ...]:
return (
init_settings,
env_settings,
dotenv_settings,
FileSecretsSettingsSource(file_secret_settings),
)
>>> Settings().model_dump()
{'app_key': SecretStr('**********'), 'db': {'passwd': SecretStr('**********')}}
Nested secrets directory layout
Config option secrets_nested_delimiter overrides env_nested_delimiter for files. In particular, this allows to use nested directory layout along with environmemt variables for other non-secret settings:
๐ secrets
โโโ ๐ app_key
โโโ ๐ db
โโโ ๐ passwd
model_config = SettingsConfigDict(
secrets_dir='secrets',
secrets_nested_subdir=True,
)
>>> Settings().model_dump()
{'app_key': SecretStr('**********'), 'db': {'passwd': SecretStr('**********')}}
Multiple secrets_dir
๐ secrets
โโโ ๐ layer1
โ โโโ ๐ app_key
โโโ ๐ layer2
โโโ ๐ db__passwd
model_config = SettingsConfigDict(
secrets_dir=['secrets/layer1', 'secrets/layer2'],
secrets_nested_delimiter='__',
)
>>> Settings().model_dump()
{'app_key': SecretStr('**********'), 'db': {'passwd': SecretStr('**********')}}
Experimental syntactic sugar ๐งช
[!CAUTION] This syntax may change at any time. Pin current
pydantic-file-secretsversion if decided to use it.
Few important things to note:
@with_builtin_sourcesdecorator enablesNamedTupleargumentsrc: BuiltinSourcesencapsulating default builtins settings sourcesBaseSourcealias is shorter thanPydanticBaseSettingsSourceand is easier to use in type hintssettings_clswas removed fromsettings_customise_sourcessignature:clsseems to be sufficient
from pydantic import BaseModel, SecretStr
from pydantic_file_secrets import (
BaseSource,
BuiltinSources,
FileSecretsSettingsSource,
SettingsConfigDict,
with_builtin_sources,
)
from pydantic_settings import BaseSettings
class DbSettings(BaseModel):
passwd: SecretStr
class Settings(BaseSettings):
app_key: SecretStr
db: DbSettings
model_config = SettingsConfigDict(
secrets_dir='secrets',
secrets_nested_delimiter='__',
)
@classmethod
@with_builtin_sources
def settings_customise_sources(cls, src: BuiltinSources) -> tuple[BaseSource, ...]:
return (
src.init_settings,
src.env_settings,
src.dotenv_settings,
FileSecretsSettingsSource(src.file_secret_settings),
)
>>> Settings().model_dump()
{'app_key': SecretStr('**********'), 'db': {'passwd': SecretStr('**********')}}
Configuration options
secrets_dir
Path to secrets directory. Same as SecretsSettingsSource.secrets_dir if str or Path. If list, the last match wins. If secrets_dir is passed in both source constructor and model config, values are not merged (constructor takes priority).
secrets_dir_missing
If secrets_dir does not exist, original SecretsSettingsSource issues a warning. However, this may be undesirable, for example if we don't mount Docker Secrets in e.g. dev environment. Now you have a choice:
'ok'โ do nothing ifsecrets_dirdoes not exist'warn'(default) โ print warning, same asSecretsSettingsSource'error'โ raiseSettingsError
If multiple secrets_dir passed, the same secrets_dir_missing action applies to each of them.
secrets_dir_max_size
Limit the size of secrets_dir for security reasons, defaults to SECRETS_DIR_MAX_SIZE equal to 16 MiB.
FileSecretsSettingsSource is a thin wrapper around EnvSettingsSource, which loads all potential secrets on initialization. This could lead to MemoryError if we mount a large file under secrets_dir.
If multiple secrets_dir passed, the limit applies to each directory independently.
secrets_case_sensitive
Same as case_sensitive, but works for secrets only. If not specified, defaults to case_sensitive.
secrets_nested_delimiter
Same as env_nested_delimiter, but works for secrets only. If not specified, defaults to env_nested_delimiter. This option is used to implement nested secrets directory layout and allows to do even nastier things like /run/secrets/model/delim/nested1/delim/nested2.
secrets_nested_subdir
Boolean flag to turn on nested secrets directory mode, False by default. If True, sets secrets_nested_delimiter to os.sep. Raises SettingsError if secrets_nested_delimiter is already specified.
secrets_prefix
Secret path prefix, similar to env_prefix, but works for secrets only. Defaults to env_prefix if not specified. Works in both plain and nested directory modes, like '/run/secrets/prefix_model__nested' and '/run/secrets/prefix_model/nested'.
Not supported config options
Some config options that are declared in SecretsSettingsSource interface are actually not working and are not supported in FileSecretsSettingsSource:
env_ignore_emptyenv_parse_none_strenv_parse_enums
However, we make sure that the behaviour of FileSecretsSettingsSource matches SecretsSettingsSource to provide a drop-in replacement, although it is somewhat wierd (e.g. env_parse_enums is always True).
Testing
100% test coverage is provided for latest Python and pydantic-settings version. Tests are run for all minor pydantic-settings v2 versions and all minor Python 3 versions supported by them:
pyXYโ Python 3.{8,9,10,11,12,13}psXYโ pydantic-settings v2.{2,3,4,5,6,7,8}
| ps29 | ps28 | ps27 | ps26 | ps25 | ps24 | ps23 | ps22 | |
|---|---|---|---|---|---|---|---|---|
| py313 | โณ๏ธ | โณ๏ธ | โ | โ | โ | โ๏ธ | โณ๏ธ | โ๏ธ |
| py312 | โ | โ | โ | โ | โ | โ๏ธ | โ๏ธ | โ๏ธ |
| py311 | โ | โ | โ | โ | โ | โ๏ธ | โ๏ธ | โ๏ธ |
| py310 | โ | โ | โ | โ | โ | โ๏ธ | โ๏ธ | โ๏ธ |
| py39 | โ | โ | โ | โ | โ | โ๏ธ | โ๏ธ | โ๏ธ |
| py38 | โ | โ | โ | โ | โ | โ๏ธ | โ๏ธ | โ๏ธ |
- โณ๏ธ pytest and mypy passing, coverage report generated
- โ pytest and mypy passing
- โ๏ธ pytest passing, mypy not attempted
- โ tests failing or not attempted
History
- September 2024 โ Multiple secrets_dir feature was merged to pydantic-settings v2.5.0
Contributing
Pull requests, feature requests, and bug reports are welcome!
Authors
- Michael Makukha
See also
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
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pydantic_file_secrets-0.4.4.tar.gz.
File metadata
- Download URL: pydantic_file_secrets-0.4.4.tar.gz
- Upload date:
- Size: 100.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.6.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
16cf16efeed0eeae91f2f52e3aaa910344ffe6d4e2ce0f1499aacc48fb90a905
|
|
| MD5 |
dad07f4d372b3218f39ccb20ca3075a9
|
|
| BLAKE2b-256 |
d284888ca64f606c11cf4c6b60f882d4c24ab8d76162fbc49399e3ff1bd84dbc
|
File details
Details for the file pydantic_file_secrets-0.4.4-py3-none-any.whl.
File metadata
- Download URL: pydantic_file_secrets-0.4.4-py3-none-any.whl
- Upload date:
- Size: 10.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.6.6
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3c8ca6d1950a6c851ea93487f8602a6c26d18a801552efcb12b3012570ae5a1e
|
|
| MD5 |
a4bb525bdbd92dd520f94d1f10633928
|
|
| BLAKE2b-256 |
f773cdcc022b33a80eba6b4085e2cbf1f769a32486da33eab10e57dc7d7c85dd
|