Starlette encourages a strict separation of configuration from code, following [the twelve-factor pattern][twelve-factor]. Configuration should be stored in environment variables, or in a `.env` file that is not committed to source control. ```python title="main.py" from sqlalchemy import create_engine from starlette.applications import Starlette from starlette.config import Config from starlette.datastructures import CommaSeparatedStrings, Secret # Config will be read from environment variables and/or ".env" files. config = Config(".env") DEBUG = config('DEBUG', cast=bool, default=False) DATABASE_URL = config('DATABASE_URL') SECRET_KEY = config('SECRET_KEY', cast=Secret) ALLOWED_HOSTS = config('ALLOWED_HOSTS', cast=CommaSeparatedStrings) app = Starlette(debug=DEBUG) engine = create_engine(DATABASE_URL) ... ``` ```shell title=".env" # Don't commit this to source control. # Eg. Include ".env" in your `.gitignore` file. DEBUG=True DATABASE_URL=postgresql://user:password@localhost:5432/database SECRET_KEY=43n080musdfjt54t-09sdgr ALLOWED_HOSTS=127.0.0.1, localhost ``` ## Configuration precedence The order in which configuration values are read is: * From an environment variable. * From the `.env` file. * The default value given in `config`. If none of those match, then `config(...)` will raise an error. ## Secrets For sensitive keys, the `Secret` class is useful, since it helps minimize occasions where the value it holds could leak out into tracebacks or other code introspection. To get the value of a `Secret` instance, you must explicitly cast it to a string. You should only do this at the point at which the value is used. ```python >>> from myproject import settings >>> settings.SECRET_KEY Secret('**********') >>> str(settings.SECRET_KEY) '98n349$%8b8-7yjn0n8y93T$23r' ``` !!! tip You can use `DatabaseURL` from `databases` package [here](https://github.com/encode/databases/blob/ab5eb718a78a27afe18775754e9c0fa2ad9cd211/databases/core.py#L420) to store database URLs and avoid leaking them in the logs. ## CommaSeparatedStrings For holding multiple inside a single config key, the `CommaSeparatedStrings` type is useful. ```python >>> from myproject import settings >>> print(settings.ALLOWED_HOSTS) CommaSeparatedStrings(['127.0.0.1', 'localhost']) >>> print(list(settings.ALLOWED_HOSTS)) ['127.0.0.1', 'localhost'] >>> print(len(settings.ALLOWED_HOSTS)) 2 >>> print(settings.ALLOWED_HOSTS[0]) '127.0.0.1' ``` ## Reading or modifying the environment In some cases you might want to read or modify the environment variables programmatically. This is particularly useful in testing, where you may want to override particular keys in the environment. Rather than reading or writing from `os.environ`, you should use Starlette's `environ` instance. This instance is a mapping onto the standard `os.environ` that additionally protects you by raising an error if any environment variable is set *after* the point that it has already been read by the configuration. If you're using `pytest`, then you can setup any initial environment in `tests/conftest.py`. ```python title="tests/conftest.py" from starlette.config import environ environ['DEBUG'] = 'TRUE' ``` ## Reading prefixed environment variables You can namespace the environment variables by setting `env_prefix` argument. ```python title="myproject/settings.py" import os from starlette.config import Config os.environ['APP_DEBUG'] = 'yes' os.environ['ENVIRONMENT'] = 'dev' config = Config(env_prefix='APP_') DEBUG = config('DEBUG') # lookups APP_DEBUG, returns "yes" ENVIRONMENT = config('ENVIRONMENT') # lookups APP_ENVIRONMENT, raises KeyError as variable is not defined ``` ## A full example Structuring large applications can be complex. You need proper separation of configuration and code, database isolation during tests, separate test and production databases, etc... Here we'll take a look at a complete example, that demonstrates how we can start to structure an application. First, let's keep our settings, our database table definitions, and our application logic separated: ```python title="myproject/settings.py" from starlette.config import Config from starlette.datastructures import Secret config = Config(".env") DEBUG = config('DEBUG', cast=bool, default=False) SECRET_KEY = config('SECRET_KEY', cast=Secret) DATABASE_URL = config('DATABASE_URL') ``` ```python title="myproject/tables.py" import sqlalchemy # Database table definitions. metadata = sqlalchemy.MetaData() organisations = sqlalchemy.Table( ... ) ``` ```python title="myproject/app.py" from starlette.applications import Starlette from starlette.middleware import Middleware from starlette.middleware.sessions import SessionMiddleware from starlette.routing import Route from myproject import settings async def homepage(request): ... routes = [ Route("/", endpoint=homepage) ] middleware = [ Middleware( SessionMiddleware, secret_key=settings.SECRET_KEY, ) ] app = Starlette(debug=settings.DEBUG, routes=routes, middleware=middleware) ``` Now let's deal with our test configuration. We'd like to create a new test database every time the test suite runs, and drop it once the tests complete. We'd also like to ensure ```python title="tests/conftest.py" from starlette.config import environ from starlette.testclient import TestClient from sqlalchemy import create_engine from sqlalchemy_utils import create_database, database_exists, drop_database # This line would raise an error if we use it after 'settings' has been imported. environ['DEBUG'] = 'TRUE' from myproject import settings from myproject.app import app from myproject.tables import metadata @pytest.fixture(autouse=True, scope="session") def setup_test_database(): """ Create a clean test database every time the tests are run. """ url = settings.DATABASE_URL engine = create_engine(url) assert not database_exists(url), 'Test database already exists. Aborting tests.' create_database(url) # Create the test database. metadata.create_all(engine) # Create the tables. yield # Run the tests. drop_database(url) # Drop the test database. @pytest.fixture() def client(): """ Make a 'client' fixture available to test cases. """ # Our fixture is created within a context manager. This ensures that # application lifespan runs for every test case. with TestClient(app) as test_client: yield test_client ``` [twelve-factor]: https://12factor.net/config