12-factor.net settings handler for Django
Project description
django-settings-env
12-factor.net settings handler for Django.
envex
Introduction
The primary functionality of this module is provided by the dependency envex
,
which provides settings via the OS environment, .env
files and, optionally,
HashiCorp vault (since envex
v2.0).
envex
provides a convenient type-smart interface for handling the OS environment, and therefore
configuration of any application using 12factor.net principals removing many environment-specific
variables and/or security-sensitive information from application code and source code repositories.
Settings may be sourced from .env files, directly from the environment, or from a HashiCorp vault.
By default, values set in the environment take priority over those set in .env files, and
those take priority of any corresponding values stored in Vault.
This can be changed by setting ENVEX_SOURCE
to a value such as file
, vault
, or any other value except env
.
Some features not supported by other dotenv handlers (python-dotenv, etc.) are available including expansion of template variables, which can enhance Don't Repeat Yourself.
Installation
pip install django-settings-env
...
poetry add django-settings-env
...
uv pip add django-settings-env
Usage
This module shouldn't need to be added to INSTALLED_APPS
as it isn't a Django app, but is an add-on module available
for import.
import django_settings_env
or
from django_settings_env import Env
Basic Usage
from django_settings_env import Env
...
env = Env()
By default, the env object will check both the operating system environment and any .env files in the project root for settings; this can be customised by:
- passing readenv=False to prevent reading from any .env files
- passing a search_path to specify the location of the .env file
- adding parents=True to search parent directories for the .env file
- adding env_file arg to override the name of the default
.env
These options are available via the envex
module.
Wherever an Env instance is available, the environment can be accessed with env["VAR_NAME"], or env("VAR_NAME"). The latter is a convenience method that will return the value of the variable or None if it is not set. The "call" syntax also has another advantage in that it can be used to set a default value if the variable is unset. When assigned to a variable of the same name as the variable in the current scope, the variable does not need to be specified, for example:
from django_settings_env import Env
env = Env()
...
DEBUG = env(default=False)
The value of DEBUG
is deferred until used or referenced.
The default
keyword is required to set the default, because the first positional argument is reserved for the variable
name.
Note that this functionality only works at the same scope level as the declaration of the variable: class, module (aka "global") or function. It will not work for cross-scope assignments (assigning a class variable from a method, for example). Explicitly specifying the variable name, however, will still work in this case.
Django Settings
django-settings-env
features to this base and specifically aims at bringing full 12-factor.net compliance to
Django settings.
It will typically avoid the need to separate local/development configuration settings from production settings, as
values are determined at runtime by the content of the environment, .env
files, or values obtained from a HashiCorp
vault.
By default, the DjangoEnv class can apply a given prefix (default is "DJANGO_") to environment variables names, but will
only be used in that form if the raw (no prefix) variable name is not already set in the environment.
To change the prefix including setting it to an empty string, pass the prefix= kwarg to Env()
, and many of the methods
can also accept a prefix=
keyword argument.
One key difference between envex
and django-settings-env
is that the latter will read .env files by default, and
will automatically search parent directories if one is not found where initially expected.
This default behaviour needs to be explicitly enabled in envex
.
Connection to Vault
Connecting to vault is optional, and handled by the envex
module.
The connection is determined by the presence of VAULT_ADDR in the environment.
It can't be set from a .env
as typically the connection to Vault determined when the env object is instantiated,
before it scans and reads any .env
files.
⚠️ If $VAULT_ADDR is set but the vault server is not running or is unavailable, there will be a considerable startup delay until the connection times out.
In addition, $VAULT_TOKEN is required to be set in the environment to authenticate with the vault server. Other environment variables may also be required, depending on the vault configuration.
Variable | Description |
---|---|
VAULT_ADDR | URL of the vault server |
VAULT_TOKEN | Token to authenticate with the vault server |
VAULT_CACERT | Path to the CA certificate for the vault server |
VAULT_SKIP_VERIFY | Skip verification of the vault server certificate |
VAULT_CLIENT_CERT | Path to the client certificate for the vault server |
VAULT_CLIENT_KEY | Path to the client key for the vault server |
VAULT_TIMEOUT | Timeout for the vault connection (in seconds) |
While setting the VAULT_TIMEOUT
item can reduce the startup delay if the vault server is not available, the vault
module retries multiple times before giving up, so the delay may still be considerable.
This parameter is provided to increase the connection timeout should the default of 5 seconds be insufficient to
successfully establish the connection.
Reducing it even further is not recommended.
VAULT_CACERT is useful when running vault with TLS is enabled (highly recommended), and the certificate is not
signed by a recognised Certificate Authority, i.e. self-signed or an internal CA.
Alternatively, VAULT_SKIP_VERIFY=true
in the environment will disable verification of the vault server certificate (*
not recommended*).
VAULT_CLIENT_CERT and VAULT_CLIENT_KEY are optional and are only required if the vault server requires client certificates. If used, both variables must be set and provide valid paths to the client certificate and key.
The vault store contains a single object containing all values in a dictionary format, and is cached by default. The cache is used to return individual values by key (same key as the environment variable) with the assumption that the vault secrets remain unchanged during the application runtime. Consequently, any changes to vault require an application restart, so it is wise to consider which items to put in vault and which to put in the environment. It is recommended to only place items in the vault that contain secrets or are otherwise sensitive.
django-settings-env API
This module provides a number of type-safe methods to help in retrieving values from the environment (including .env
files or vault).
The env.get()
method assumes a string, but other methods are available to handle other types, such
as env.int()
, env.bool()
, env.float()
, env.list()
etc.
All provide seamless conversion of the environment variable to the desired type, or return a default value if the
variable is not set.
The env()
call syntax also provides a type
parameter that can be used to specify the type of the variable to be
returned, which can be either a class, or the name of the class.
Only primitive types and list
(comma separated values) are supported.
All of this functionality is provided out of the box by envex
.
Django Specific Methods
Some django specific functionality is included in this module, such as URL parsers for:
Default Var | Parser |
---|---|
DATABASE_URL | env.database_url() |
CACHE_URL | env.cache_url() |
EMAIL_URL | env.email_url() |
SEARCH_URL | env.search_url() |
Each of which can be injected into django settings via the environment, typically from a .env file at the project root, or set from a variable in vault. Individual components of these URLs can also be set, but passing the URL provides a way of setting all the required components, including options.
The url specified includes a scheme that determines the "backend" class, engine, class or module that handles the
corresponding functionality as documented below.
This can be overridden using the backend=
parameter even if the scheme is not known by django-settings-env
.
URLs may include options, in the form of query options, i.e. ?option=value&option2=value2
etc. that are specific to
the engine or backend being used.
Options are usually case-sensitive, and must use the same case as expected by the backend.
database_url
Evaluate a URL in the form
schema://[username:[password]@]host_or_path[:port]/name[?...options]
Schemas:
Scheme | Database |
---|---|
postgres | Postgres (psycopg2 or psycopg) |
postgresql | Postgres (psycopg2 or psycopg) |
psql | Postgres (psycopg2 or psycopg) |
pgsql | Postgres (psycopg2 or psycopg) |
postgis | Postgres (psycopg2 or psycopg) + PostGIS |
mysql | MySql (mysqlclient) |
mysql2 | MySql (mysqlclient) |
mysql-connector | MySql (mysql-connector) |
mysqlgis | MySql (mysqlclient) using GIS extensions |
mssql | SqlServer (sql_server.pyodbc) |
oracle | Oracle (cx_Oracle) |
pyodbc | ODBC (pyodbc) |
redshift | Amazon Redshift |
spatialite | Sqlite with spatial extensions (spatialite) |
sqlite | Sqlite |
ldap | django-ldap |
Examples (snippets from settings.py)
from django_settings_env import Env
env = Env()
...
DATABASES = {
"default": env.database_url(),
"backup": env.database_url("DATABASE_BACKUP_URL"),
}
cache_url
Evaluate a URL in the form
schema://[username:[password]@]host_or_path[:port]/[name][?...options]
Schemas:
Scheme | Cache |
---|---|
dbcache | cache in database |
dummycache | dummy cache - "no cache" |
filecache | cache data in files |
locmemcache | cache in memory |
memcache | memcached (python-memcached) |
pymemcache | memcached (pymemcache) |
rediscache | redis |
redis | redis |
rediss | redis (ssl connection) |
email_url
Evaluate a URL in the form
schema://[username[@domain]:[password]@]host_or_path[:port]/[?...options]
Schemas:
Scheme | Service |
---|---|
smtp | smtp, no SSL |
smtps | smtp over SSL |
smtp+tls | smtp over SSL |
smtp+ssl | smtp over SSL |
consolemail | publish mail to console (dev) |
filemail | append email to file (dev) |
memorymail | store emails in memory |
dummymail | do-nothing email backend |
amazonses | Amazon Wimple Email Service |
amazon-ses | Amazon Wimple Email Service |
search_url
Evaluate a URL in the form
schema://[username:[password]@]host_or_path[:port]/[index]
Schemas:
Scheme | Engine |
---|---|
elasticsearch | elasticsearch (django-haystack) |
elasticsearch2 | elasticsearch2 (django-haystack) |
elasticsearch-dsl | elasticsearch-dsl |
solr | Apache solr (django-haystack) |
whoosh | Whoosh search engine (pure python, haystack) |
xapian | Xapian search engine (haystack) |
simple | Simple search engine (haystack) |
Note that django-haystack may require many additional settings not supported by this module.
elasticsearch-dsl
is recommended as a suitable replacement, and in general integrates well with Django's ORM,
providing the ability to easily relate ES documents+indexes to Django models.
The DSL version also supports more contemporary versions of Elasticsearch and is well maintained.
Django Class Settings
Support for the django-class-settings
module is dynamically added
to the env
handler, allowing a much simplified use withing a class_settings.Settings class, e.g.:
from django_settings_env import Env
# noinspection PyUnresolvedReferences
from class_settings import Settings
env = Env(prefix='DJANGO_') # redundant, this is the default
class MySettings(Settings):
MYSETTING = env()
This usage will look for 'MYSETTING' or 'DJANGO_MYSETTING' in the environment and lazily assign it to the MYSETTING value for the settings class.
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
Hashes for django_settings_env-5.0.1.tar.gz
Algorithm | Hash digest | |
---|---|---|
SHA256 | 5f7c26cdbbc7b76db9ae732ee76b7a71bfb35206ab5c2fb8944168bd7d1361f6 |
|
MD5 | ae29856ce66cec24a6c527470bc6a6ac |
|
BLAKE2b-256 | 2dff4acc680f1a4421e0a33b378817dfa6845a28cbf9cf1dda25ae07f332142c |
Hashes for django_settings_env-5.0.1-py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | 45f9ddd012f3ade287e6a10a2ed3b730302bea3481d7cf7bd69009829bfe80f0 |
|
MD5 | ca2821124eed8d6440162434bed38ea3 |
|
BLAKE2b-256 | f5996a31a3a8acab4c72a0960a65f4b2f7d94b1220df3f20d4cb041e8573fbe5 |