Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/shimmur/env_var_provider

Elixir Config Provider to allow configuration dynamically from environment variables 🏔
https://github.com/shimmur/env_var_provider

config elixir provider

Last synced: about 2 months ago
JSON representation

Elixir Config Provider to allow configuration dynamically from environment variables 🏔

Awesome Lists containing this project

README

        

EnvVarProvider
==============

[![CI](https://github.com/Shimmur/env_var_provider/actions/workflows/main.yml/badge.svg)](https://github.com/Shimmur/env_var_provider/actions/workflows/main.yml)
[![Coverage Status](https://coveralls.io/repos/github/Shimmur/env_var_provider/badge.svg)](https://coveralls.io/github/Shimmur/env_var_provider)

Config provider that allows _dynamic_ runtime configuration of Elixir releases
from system environment variables, including _type conversion_ to non-string
types. Great for running Elixir applications on modern infrastructure like
Docker, Kubernetes, Mesos.

Installation
------------

Add `env_var_provider` to your list of dependencies in `mix.exs`:

```elixir
def deps do
[
{:env_var_provider, "~> 0.5.3", organization: "community"}
]
end
```

Configuration
-------------

You can use the `EnvVar.Provider` config provider as it implements the
`Config.Provider` behaviour. In your releases configuration (usually in
`mix.exs`), use it like this:

```elixir
defp releases do
env_map = %{
erlcass: %{
cluster_options: %{
credentials: %{type: {:tuple, :string}, default: "user,pass"},
contact_points: %{type: :string, default: "127.0.0.1"},
port: %{type: :integer, default: "9042"}
}
},
simpler: %{
service_name: %{type: :string}
}
}

[
my_app: [
# ...,
config_providers: [
{EnvVar.Provider, enforce: true, prefix: "", env_map: env_map}
]
]
]
end
```

The above config would source env vars like this:

* `MY_RELEASE_ERLCASS_CLUSTER_OPTIONS_CREDENTIALS`
* `MY_RELEASE_ERLCASS_CLUSTER_OPTIONS_CONTACT_POINTS`
* `MY_RELEASE_ERLCASS_CLUSTER_OPTIONS_PORT`
* `MY_RELEASE_SIMPLER_SERVICE_NAME`

Note that the `*_ERLCASS_*` keys are assumed to be values in a Keyword list
in this format. Simpler configuration doesn't require the extra layer and
can be one layer flatter like in `:simpler` above.

**You may need to call the provider multiple times for different applications
and you can just specify it repeatedly.**

`prefix` is a string that will be capitalized and prepended to all
environment variables we look at. e.g. `prefix: "beowulf"` translates
into environment variables starting with `BEOWULF_`. This is used
to namespace our variables to prevent conflicts.

`env_map` follows the following format:

```elixir
env_map = %{
heorot: %{
location: %{type: :string, default: "land of the Geats"},
},
mycluster: %{
server_count: %{type: :integer},
name: %{type: :string, default: "grendel"},
settings: %{type: {:list, :string}, default: "swarthy,hairy"},
keys: %{type: {:tuple, :float}, default: "1.1,2.3,3.4"}
}
}
```

Type Conversion
---------------

Type conversion allows you to source settings from the environment that
are not strings. It even supports a limited set of complex types like
`List`s and `Tuple`s.

Type conversion uses the defined types to handle the destination
conversion.

Supported types:
* `:string`
* `:integer`
* `:float`
* `:boolean` - 0, 1, 'true', 'false' supported
* `{:tuple, }` - Complex type, where the second field is
one of the simple types above. Currently items in the tuple
must all be of the same type. A 3rd argument can be passed
to specify the field separator in the env var. Defaults to
comma.
* `{:list, }` - Complex type, following the same rules as
Tuples above.

Default Values
--------------
You have two choices when dealing with default values:

1. Leave them off and it will default to what is in the Application
config already.
2. Set them in the release configuration. This can provide better
visibility in one place as to what the fallback will be if the
variable is not provided.

If you supply default values, they will overwrite any existing values in the
config for this environment.

### Enforcement

For further safety, you may set `enforce: true` in the Keyword list to
configure the provider. This behavior is the equivalent of setting `required:
true` on each of the items. This prevents the provider from ever falling back
to values in the configuration. This is the safest way to guarantee that a
value, either in the environment, or a default, was set.