Configuration library
- Merge multiple configuration sources
- EDN file
- Environment variables
- Java system properties
- Static map
- Map inside a var or atom
- lambdaisland.cli command line flags
- ConfigProvider protocol, easy to plug in additional sources
- Unopinionated base API
- Support for a specific opinionated convention which is encouraged
- Environment-specific configuration (dev, prod, staging, etc)
- Provenance tracking (know where config values came from)
- Support for hooking up reloading
To use the latest release, add the following to your deps.edn
(Clojure CLI)
com.lambdaisland/config {:mvn/version "0.4.17"}
or add the following to your project.clj
(Leiningen)
[com.lambdaisland/config "0.4.17"]
lambdaisland/config
implements a pattern we've settled on through doing lots
of different Clojure projects, about how to handle configuration, in particular
the kind of things that differ between environments (dev, test, staging, prod),
and that you might want to set or override on multiple levels.
It is highly flexible in how you configure the sources that are checked, but has opinionated defaults, and allows plugging in custom "providers", for instance for checking a secret store like Hashicorp Vault or Google Secret Manager.
This is how we handle configuration, and the approach we encourage others to adopt. We'll get into the nitty gritty of the library and how to do other things with it down below.
When creating a new Clojure application, we pick a short symbolic name for it,
for this example we'll use app-name
. lambdaisland/config
refers to this as
the "prefix" because it's used to prefix various things.
We then create a base configuration in resources/app-name/config.edn
. We
generally use namespaced keywords in there, and try to have a sensible default
for every configuration key that is used, maybe with a comment about what it
does.
Then we create per-environment config files: resources/app-name/dev.edn
,
resources/app-name/prod.edn
, etc. We generally have at least dev, prod, test,
and likely also staging. Any environment-specific config that isn't a secret
should go in there and get checked in, so that running the app, either in dev or
prod or test mode, "just works", and new developers can get onboarded quickly.
Finally each person creates a config.local.edn
at the project root. This file
does not get checked in (we gitigore *.local.*
). Here developers can easily
make local overrides, or configure secrets.
For dev work this is usually all you need, but when it comes time to deploying
or releasing the story varies. If you are running in a cloud environment often
the easiest (sometimes the only) option you have to do per-instance
configuration is through environment variables. Or sometimes the thing you have
the most control over is the command line invocation, in which case Java system
properties are convenient. If you're releasing to the public it might make sense
to follow XDG conventions and read a ~/.config/app-name.edn
file.
lambdaisland/config
provides all of these mechanisms out of the box.
(def config
(config/create {:prefix "app-name"
:env :dev}))
(config/get config :http/port) ;;=> 8080
This will check, in order, until it's found a value:
- The
$APP_NAME__HTTP__PORT
environment variable - The
app-name.http.port
Java system property (System/getProperty
) config.local.edn
in the JVM's CWD$XDG_CONFIG_HOME/app-name.edn
/etc/app-name.edn
app-name/dev.edn
on the CLASSPATH (e.g. underresources
)app-name/config.edn
on the CLASSPATH
To know where a given setting came from, use config/source
(config/source config :http/port)
;;=> `"$HTTP__PORT environment variable"
You don't have to provide an :env
key, if not it'll be determined by the
APP_NAME__ENV
environment variable, or the app-name.env
system property. If
neither is set and CI=true
(an env var set by most CI providers) then env will
be test
. If none of these apply then the default is dev
.
If you build a Docker image you might want to bake in -Japp-name.env=prod
, so
it doesn't try to start in dev mode. You can still use the env var to override
this for instance in a staging environment.
lambdaisland/config
is based on the ConfigProvider
protocol.
(defprotocol ConfigProvider
(-value [this k])
(-source [this k])
(-reload [this]))
The result of config/create
is a three-element map. The "environment" name, a
sequence of config providers, and an atom which acts as a cache of values
already accessed.
{:env :prod
:providers [,,,<implement ConfigProvider protocol>,,,]
:values (atom {:http/port {:val 8080 :source "$HTTP__PORT environment variable}})
:env
can be explicitly passed in, otherwise we check the PREFIX__ENV
(e.g.
APP_NAME__ENV
) env var, or the prefix.env
System property (app-name.env
). If
neither is set and the CI
env var is true, then we default to :test
, if not
we fall back to :dev
.
create
can take a number of other options besides :env
and :prefix
.
:env-vars false
- Don't check environemnt variables:prefix-env true
- Include the prefix when checking environment variables, e.g.APP_NAME__HTTP__PORT
instead ofHTTP_PORT
:java-system-props false
- Don't check Java system properties:local-config false
- Don't checkconfig.local.edn
:xdg-config false
- Don't checkXDG_CONFIG_HOME
(default:~/.config
)
If you want a different precedence order, or want to inject your own
ConfigProvider
, then either don't use create
and construct your own config
map as you see fit, or do use create
, but subsequently update the :providers
list.
All EDN files are read with Aero, so reader
macros like #env
, #profile
, and #or
are available. We pass the app env
(prod
, dev
, etc) in as the Aero :profile
.
Apart from the options listed, you might also want to provide command line flags
to set specific options. For this you can use
com.lambdaisland/cli, and
lambdaisland.config.cli/add-provider
.
Here's an illustrative example. Note that this relies on the dynamic var
lambdaisland.cli/*opts*
which gets bound while the command handler is being
invoked. For simple use cases this is fine. In more complex (in particular
multi-threaded) scenarios you can pass a var or other derefable to
add-provider
explicitly.
(ns app-name
(:require
[lambdaisland.cli :as cli]
[lambdaisland.config :as config]
[lambdaisland.config.cli :as config-cli]))
(def cfg
(-> (config/create {:prefix "app-name"})
config-cli/add-provider))
(defn inspect-cmd
"Inspect the `opts` coming from lambdaisland/cli"
[opts]
(println (str "Port=" (config/get cfg :http/port) " read from " (config/source cfg :http/port)))
(prn opts))
(def cmdspec
{:name "app-name"
:doc "Illustrate cli+config usage"
:commands
["inspect" #'inspect-cmd]
:flags
["--port <port>" {:key :http/port
:doc "HTTP port to listen on"}]})
(defn -main [& argv]
(cli/dispatch* cmdspec argv))
In summary, the general idea is:
- Create a
resources/<prefix>/config.edn
file with your base config. Whenever adding a new config key it's a good idea to add a sensible default here - Create a
resources/<prefix>/<env>.edn
for each environment. This way you can check in sensibledev.edn
settings, and separateprod.edn
settings. - During development, create
config.local.edn
so you can easily change settings locally. Add*.local.*
to.gitignore
. - When deploying/running in specific settings, use whatever makes most sense in
that context to customize the deployment. If you're running through some cloud
or lambda provider env vars might be your main option. If you control the
clojure
orjava
command line invocation, then system props might be handy. If you want a file on the filesystem you can look at and tweak, then theXDG_CONFIG_HOME
convention is useful.
At the end of the boot process it can be a good idea to print/log
config/sources
or config/entries
(perhaps with
clojure.pprint/print-table
), so when you go in to debug things you have a
record of where various configuration items are coming from.
Thank you! config is made possible thanks to our generous backers. Become a backer on OpenCollective so that we can continue to make config better.
config is part of a growing collection of quality Clojure libraries created and maintained by the fine folks at Gaiwan.
Pay it forward by becoming a backer on our OpenCollective, so that we continue to enjoy a thriving Clojure ecosystem.
You can find an overview of all our different projects at lambdaisland/open-source.
We warmly welcome patches to config. Please keep in mind the following:
- adhere to the LambdaIsland Clojure Style Guide
- write patches that solve a problem
- start by stating the problem, then supply a minimal solution
*
- by contributing you agree to license your contributions as MPL 2.0
- don't break the contract with downstream consumers
**
- don't break the tests
We would very much appreciate it if you also
- update the CHANGELOG and README
- add tests for new functionality
We recommend opening an issue first, before opening a pull request. That way we can make sure we agree what the problem is, and discuss how best to solve it. This is especially true if you add new dependencies, or significantly increase the API surface. In cases like these we need to decide if these changes are in line with the project's goals.
*
This goes for features too, a feature needs to solve a problem. State the problem it solves first, only then move on to solving it.
**
Projects that have a version that starts with 0.
may still see breaking changes, although we also consider the level of community adoption. The more widespread a project is, the less likely we're willing to introduce breakage. See LambdaIsland-flavored Versioning for more info.
Copyright © 2024 Arne Brasseur and Contributors
Licensed under the term of the Mozilla Public License 2.0, see LICENSE.