/dotenvy

An Elixir port of the Ruby dotenv package

Primary LanguageElixirApache License 2.0Apache-2.0

Dotenvy

Module Version Hex Docs Total Download License Last Updated

Dotenvy is an Elixir port of the original dotenv Ruby gem, compatible with mix and releases. It is designed to help the development of applications following the principles of the 12-factor app and its recommendation to store configuration in the environment.

Installation

Add dotenvy to your list of dependencies in mix.exs:

def deps do
  [
    {:dotenvy, "~> 0.8.0"}
  ]
end

It has no dependencies.

Usage

Dotenvy is designed to help configure your application at runtime, and one of the most effective places to do that is inside config/runtime.exs (available since Elixir v1.11).

The Dotenvy.source/2 function can accept a single file or a list of files. When combined with Config.config_env/0 it is easy to load up environment-specifc config, e.g.

source(["#{config_env()}.env", "#{config_env()}.override.env", System.get_env()])

By default, the listed files do not need to exist -- the function only needs to know where to look. This makes it easy to commit default values while still leaving the door open to developers to override values via their own configuration files.

You control if and how existing system env vars are handled: usually they should take precedence over values defined in .env files, so the System.get_env() should be included as the final input supplied to source/2.

Unlike other packages, Dotenvy has no opinions about the names or locations of your dotenv config files, you just need to pass their paths to Dotenvy.source/2 or Dotenvy.source!/2.

For a simple example, we can load a single file:

# config/runtime.exs
import Config
import Dotenvy

source!([".env", System.get_env()])

config :myapp, MyApp.Repo,
    database: env!("DATABASE", :string!),
    username: env!("USERNAME", :string),
    password: env!("PASSWORD", :string),
    hostname: env!("HOSTNAME", :string!),
    pool_size: env!("POOL_SIZE", :integer),
    adapter: env!("ADAPTER", :module, Ecto.Adapters.Postgres),
    pool: env!("POOL", :module?)

And then define your variables in the file(s) to be sourced. Dotenvy has no opinions about what you name your files; .env is merely a convention.

# .env
DATABASE=myapp_dev
USERNAME=myuser
PASSWORD=mypassword
HOSTNAME=localhost
POOL_SIZE=10
POOL=

When you set up your application configuration in this way, you are creating a contract with the environment: Dotenvy.env!/2 will raise if the required variables have not been set or if the values cannot be properly transformed. This is an approach that works equally well for your day-to-day development and for mix releases.

Read the configuration strategies for more detailed examples of how to configure your app.

Refer to the "dotenv" (.env) file format for more examples and features of the supported syntax.

See the Dotenvy module documentation on its functions.

Note for Mix Tasks

If you have authored your own Mix tasks, you must ensure that they load the application configuration in a way that is compatible with the runtime config. A good way to do this is to include Mix.Task.run("app.config") inside the run/1 implementation, e.g.

def run(_args) do
  Mix.Task.run("app.config")
  # ...
end

If you are dealing with third-party mix tasks that fail to properly load configuration, you may need to manually call mix app.config before running them, e.g.

mix do app.config other.task

Defining a task alias in mix.exs is another way to accomplish this:

# mix.exs
defp aliases do
    [
      "other.task": ["app.config", "other.task"]
    ]

Upgrading from v0.5.0 or before

Starting with Dotenvy v0.6.0, the precedence of system env variables over parsed .env files is not defined; the :overwrite? and :vars options are no longer supported in Dotenvy.source/2 and Dotenvy.source!/2. Instead, the source functions now accept file paths OR maps: this makes the question of variable precedence something that must be explicitly listed. The source functions act more like Map.merge/2, accumulating values, always giving precedence to the righthand source.

Most users upgrading from v0.5.0 will wish to include System.get_env() as the final input to source/2.

# in dotenvy 0.5.0 or before:
source(["#{config_env()}.env", "#{config_env()}.override.env"])

# should be changed to the following in dotenvy 0.6.0:
source(["#{config_env()}.env", "#{config_env()}.override.env", System.get_env()])

If you are relying on variable interpolation in your .env files, you may also need to include System.get_env() (or an equivalent subset) before you list your .env files. This is necessary to make values available to the file parser.

# in dotenvy 0.5.0 or before:
source(["#{config_env()}.env", "#{config_env()}.override.env"])

# should be changed to the following in dotenvy 0.6.0:
source([System.get_env(), "#{config_env()}.env", "#{config_env()}.override.env", System.get_env()])

The change in syntax introduced in v0.6.0 favors a declarative list of sources over opaquely inferred inputs. This also opens the door for compatibility with other value sources, e.g. secure parameter stores.


Image Attribution: "dot" by Stepan Voevodin from the Noun Project