Resolve env vars from secret stores.
Project description
envresolve
Resolve environment variables from secret stores like Azure Key Vault.
Features
- Variable expansion: Expand
${VAR}and$VARsyntax in strings - Secret resolution: Fetch secrets from Azure Key Vault (more providers coming)
- Circular reference detection: Prevents infinite loops in variable chains
- Type-safe: Full mypy type checking support
Quick Start
Variable Expansion
Expand variables without connecting to external services:
from envresolve import expand_variables
env = {"VAULT": "corp-kv", "SECRET": "db-password"}
result = expand_variables("akv://${VAULT}/${SECRET}", env)
print(result) # akv://corp-kv/db-password
Default Values
Use default values for undefined or empty variables with ${VAR:-default} syntax:
from envresolve import expand_variables
# Variable is undefined - use default
result = expand_variables("${HOST:-localhost}", {})
print(result) # localhost
# Variable is empty - use default (Bash :- semantics)
result = expand_variables("${HOST:-localhost}", {"HOST": ""})
print(result) # localhost
# Variable is defined - ignore default
result = expand_variables("${HOST:-localhost}", {"HOST": "example.com"})
print(result) # example.com
# Defaults can contain nested variables
env = {"FALLBACK_HOST": "backup.example.com"}
result = expand_variables("${HOST:-${FALLBACK_HOST}}", env)
print(result) # backup.example.com
# Multiple defaults in one string
result = expand_variables("${PROTO:-https}://${HOST:-localhost}:${PORT:-443}", {})
print(result) # https://localhost:443
Load from .env File
Load environment variables from a .env file with automatic secret resolution:
import envresolve
# .env file content:
# VAULT_NAME=my-vault
# DATABASE_URL=akv://${VAULT_NAME}/db-url
# API_KEY=akv://${VAULT_NAME}/api-key
# Requires: pip install envresolve[azure]
# Requires: Azure authentication (az login, Managed Identity, etc.)
envresolve.register_azure_kv_provider()
# Load .env and resolve all secret URIs
# By default, searches for .env in current directory and exports to os.environ
resolved_vars = envresolve.load_env()
# Or specify explicit path and disable export
resolved_vars = envresolve.load_env(dotenv_path=".env", export=False)
Note: For complete python-dotenv compatibility, use load_dotenv() + resolve_os_environ():
from dotenv import load_dotenv
import envresolve
# Use python-dotenv's search behavior (from calling script location)
load_dotenv()
# Resolve secrets in os.environ
envresolve.register_azure_kv_provider()
envresolve.resolve_os_environ()
Direct Secret Resolution
Fetch individual secrets from Azure Key Vault:
import envresolve
# Requires: pip install envresolve[azure]
# Requires: Azure authentication (az login, Managed Identity, etc.)
try:
envresolve.register_azure_kv_provider()
secret_value = envresolve.resolve_secret("akv://corp-vault/db-password")
print(secret_value)
except envresolve.ProviderRegistrationError as e:
print(f"Azure SDK not available: {e}")
except envresolve.SecretResolutionError as e:
print(f"Failed to fetch secret: {e}")
Custom Provider Configuration
Inject custom provider instances for advanced scenarios (testing, custom credentials, etc.):
import envresolve
from envresolve.providers.azure_kv import AzureKVProvider
from azure.identity import ManagedIdentityCredential
# Create custom provider with specific credential
custom_provider = AzureKVProvider(
credential=ManagedIdentityCredential(client_id="your-client-id")
)
# Register the custom provider
envresolve.register_azure_kv_provider(provider=custom_provider)
# Now use envresolve as normal
secret = envresolve.resolve_secret("akv://vault/secret")
This is particularly useful for:
- Testing: Inject mock providers without patching internal implementation details
- Custom authentication: Use specific Azure credentials (service principal, managed identity, etc.)
- Provider configuration: Pre-configure providers with custom settings
Resolve Existing Environment Variables
Resolve secret URIs already set in os.environ (useful for containerized applications):
import os
import envresolve
# Environment variables set by container orchestrator or parent process
os.environ["API_KEY"] = "akv://prod-vault/api-key"
os.environ["DB_PASSWORD"] = "akv://prod-vault/db-password"
# Requires: pip install envresolve[azure]
envresolve.register_azure_kv_provider()
# Resolve all environment variables containing secret URIs
resolved = envresolve.resolve_os_environ()
# Resolve only specific keys
resolved = envresolve.resolve_os_environ(keys=["API_KEY"])
# Resolve variables with prefix and strip the prefix
# DEV_API_KEY -> API_KEY, DEV_DB_URL -> DB_URL
os.environ["DEV_API_KEY"] = "akv://dev-vault/api-key"
os.environ["DEV_DB_URL"] = "akv://dev-vault/db-url"
resolved = envresolve.resolve_os_environ(prefix="DEV_")
# Ignore specific variables (exact match)
os.environ["PS1"] = "${USER}@${HOST}$ " # Should not be expanded
os.environ["API_KEY"] = "akv://vault/api-key"
resolved = envresolve.resolve_os_environ(ignore_keys=["PS1"])
# Ignore variables by pattern (glob matching)
os.environ["PS1"] = "${USER}@${HOST}$ "
os.environ["PS2"] = "> "
os.environ["PROMPT"] = "${PWD}$ "
os.environ["API_KEY"] = "akv://vault/api-key"
resolved = envresolve.resolve_os_environ(ignore_patterns=["PS*", "PROMPT*"])
# Combine exact match and patterns
resolved = envresolve.resolve_os_environ(
ignore_keys=["SPECIFIC_VAR"],
ignore_patterns=["TEMP_*", "DEBUG_*"]
)
Error Handling
Resolution errors include context about which environment variable failed:
import envresolve
try:
envresolve.load_env(dotenv_path=".env", export=False)
except envresolve.EnvironmentVariableResolutionError as e:
print(f"Failed variable: {e.context_key}")
print(f"Cause: {e.original_error}")
Control error behavior:
# Skip variables with expansion errors
resolved = envresolve.load_env(stop_on_expansion_error=False)
# Skip variables with secret resolution errors
resolved = envresolve.load_env(stop_on_resolution_error=False)
Installation
# Basic installation (variable expansion only)
pip install envresolve
# With Azure Key Vault support
pip install envresolve[azure]
Documentation
Full documentation: https://osoekawaitlab.github.io/envresolve/
Development
Setup
This project uses uv for dependency management and nox for task automation:
# Install uv (if not already installed)
pip install uv
# Clone the repository
git clone https://github.com/osoekawaitlab/envresolve.git
cd envresolve
# Install dependencies and create virtual environment
uv sync --all-extras --all-groups
# Activate virtual environment
source .venv/bin/activate # On Unix/macOS
# .venv\Scripts\activate # On Windows
Running Tests
# Quick test during development
nox -s tests_unit # Unit tests only (fast)
nox -s tests_e2e # E2E tests with mocked Azure SDK
# Full test suite
nox -s tests # All tests with coverage report (HTML in htmlcov/)
# Test across Python versions
nox -s tests_all_versions # Test on Python 3.10-3.14
# Test without Azure SDK
nox -s tests_without_azure # For environments without Azure dependencies
Code Quality
# Run all quality checks
nox -s quality # Type checking (mypy) + linting (ruff)
# Individual checks
nox -s mypy # Type checking only
nox -s lint # Linting only
nox -s format_code # Auto-format code
# Run everything
nox -s check_all # Tests + quality checks
Live Azure Tests
Optional integration tests against real Azure Key Vault:
# One-time setup
cd infra/terraform
cp terraform.tfvars.example terraform.tfvars
# Edit terraform.tfvars with your Azure credentials
terraform init
terraform apply
# Before running tests
az login
source scripts/setup_live_tests.sh
# Run live tests
nox -s tests_live
See Live Azure Tests documentation for detailed setup instructions.
Build Documentation
# Build documentation
nox -s docs_build
# Serve documentation locally (with live reload)
mkdocs serve # Open http://localhost:8000
Contributing
See Contributing Guide for guidelines on:
- Code style and conventions
- Test-driven development workflow
- Creating issues and pull requests
- Architecture Decision Records (ADRs)
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 envresolve-0.1.11.tar.gz.
File metadata
- Download URL: envresolve-0.1.11.tar.gz
- Upload date:
- Size: 21.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ad4e9b5f2242df23a74ee9b569655d62014276077ded14ecc148cc4c97d80853
|
|
| MD5 |
bfa0d7f6885b237941849fc10d3ac386
|
|
| BLAKE2b-256 |
46aaf7ce4499461f5d48e00e6f96c88344fbdea49db3c9c473a553ede2168658
|
Provenance
The following attestation bundles were made for envresolve-0.1.11.tar.gz:
Publisher:
release.yml on osoekawaitlab/envresolve
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
envresolve-0.1.11.tar.gz -
Subject digest:
ad4e9b5f2242df23a74ee9b569655d62014276077ded14ecc148cc4c97d80853 - Sigstore transparency entry: 774095356
- Sigstore integration time:
-
Permalink:
osoekawaitlab/envresolve@59075f66dc06527fba4eb224c8ef618d5685dc1e -
Branch / Tag:
refs/tags/v0.1.11 - Owner: https://github.com/osoekawaitlab
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@59075f66dc06527fba4eb224c8ef618d5685dc1e -
Trigger Event:
push
-
Statement type:
File details
Details for the file envresolve-0.1.11-py3-none-any.whl.
File metadata
- Download URL: envresolve-0.1.11-py3-none-any.whl
- Upload date:
- Size: 21.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
15a30e84b63ca66b28a61ffdb8fdf274665219f5213c16d4b54369204448f7bd
|
|
| MD5 |
63bb12d1672612bd646aa0aa08971d7d
|
|
| BLAKE2b-256 |
3c45a98d18a62d1c79f7edf7ec16136b6082bb8d7529103fbea3a3bbdf445a7c
|
Provenance
The following attestation bundles were made for envresolve-0.1.11-py3-none-any.whl:
Publisher:
release.yml on osoekawaitlab/envresolve
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
envresolve-0.1.11-py3-none-any.whl -
Subject digest:
15a30e84b63ca66b28a61ffdb8fdf274665219f5213c16d4b54369204448f7bd - Sigstore transparency entry: 774095358
- Sigstore integration time:
-
Permalink:
osoekawaitlab/envresolve@59075f66dc06527fba4eb224c8ef618d5685dc1e -
Branch / Tag:
refs/tags/v0.1.11 - Owner: https://github.com/osoekawaitlab
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@59075f66dc06527fba4eb224c8ef618d5685dc1e -
Trigger Event:
push
-
Statement type: