env-select

license crates.io version

Easily switch between predefined values for arbitrary environment variables Features include (but are not limited to):

Simple Example

```toml

.env-select.toml

[apps.server] dev = {SERVICE1 = "dev", SERVICE2 = "also-dev"} prd = {SERVICE1 = "prd", SERVICE2 = "also-prd"} ```

```sh

es server | source ❯ === dev === SERVICE1=dev SERVICE2=also-dev

=== prd === SERVICE1=prd SERVICE2=also-prd

echo $SERVICE1 $SERVICE2 dev also-dev ```

(See the disclaimer for why piping to source is needed)

Concepts

env-select operates with a few different building blocks:

Value

A value is a means of deriving a string for the shell. Typically this is just a literal string: "abc", but it can also be a command that will be evaluated to a string at runtime.

sh dev # Literal $(echo prd) # Command

Variable Mapping

A key and a value. Variables can either be selected independently (via the vars key in the config) or be part of a profile with other variables.

sh SERVICE1=dev

Profile

A profile is a set of variable mappings.

sh SERVICE1=dev SERVICE2=also-dev

Application

An application is a group. "Application" in this case is a synonym for "use case" or "purpose". Each profile in an application accomplishes different versions of the same goal.

```sh

dev

SERVICE1=dev SERVICE2=also-dev

prd

SERVICE1=prd SERVICE2=also-prd ```

Usage

First, define .env-select.toml. This is where you'll specify possible options for each variable. Here's an example:

```toml [vars] TEST_VARIABLE = ["abc", "def"]

[apps.server] dev = {SERVICE1 = "dev", SERVICE2 = "also-dev"} prd = {SERVICE1 = "prd", SERVICE2 = "also-prd"}

[apps.db] dev = {DATABASE = "dev", DBUSER = "root", DBPASSWORD = "badpw"} stg = {DATABASE = "stg", DBUSER = "root", DBPASSWORD = "goodpw"} prd = {DATABASE = "prd", DBUSER = "root", DBPASSWORD = "greatpw"} ```

Now, you can easily switch between the defined values with es.

Select a single variable

We can select between multiple values for a single variable, in this case TEST_VARIABLE. This is a shorthand for defining an application with multiple single-variable profiles.

```sh

es TESTVARIABLE | source TESTVARIABLE=abc ❯ TESTVARIABLE=def echo $TESTVARIABLE def ```

Select a set of variables

In the config above, we've already predefined an application called server, which consists of two profiles, dev and prd. We can select between those profiles by providing the application name.

```sh

es server | source ❯ === dev === SERVICE1=dev SERVICE2=also-dev

=== prd === SERVICE1=prd SERVICE2=also-prd

echo $SERVICE1 $SERVICE2 dev also-dev ```

If you know the name of the profile you want to select, you can also skip the prompt by providing it directly to the command:

```sh

es server dev | source echo $SERVICE1 $SERVICE2 dev also-dev ```

Configuration

Configuration is defined in TOML. There are two main tables in the config, each defined by a fixed key:

TODO add command usage

Let's see this in action:

```toml

Single variables

[vars] TESTVARIABLE = ["abc", "def"] OTHERVARIABLE = ["potato", "tomato"]

Applications

[apps.server] dev = {SERVICE1 = "dev", SERVICE2 = "also-dev"} prd = {SERVICE1 = "prd", SERVICE2 = "also-prd"}

This application has no profiles, but is still valid to configure

[apps.empty]

These profiles are big, so we can use full table syntax instead of the

inline syntax. This is purely stylistic; you can make your inline

tables as big as your heart desires. See https://toml.io/en/v1.0.0#table

[apps.big.prof1] VAR1 = "yes" VAR2 = "yes" VAR3 = "no" VAR4 = "no" VAR5 = "yes"

[apps.big.prof2] VAR1 = "no" VAR2 = "no" VAR3 = "no" VAR4 = "yes" VAR5 = "no" ```

Dynamic Values

You can define variables whose values are provided dynamically, by specific a command to execute rather than a static value. This allows you to provide values that can change over time, or secrets that you don't want appearing in the file. For example:

toml [apps.db] dev = {DATABASE = "dev", DB_USER = "root", DB_PASSWORD = {command = "cat password.txt"}}

When the dev profile is selected for the db app, the DB_PASSWORD value will be loaded from the file password.txt.

Note that the command evaluation is done by your shell, not by env-select. This means you can use aliases and functions defined in your shell as commands.

Disjoint Profiles

Profiles within an app can define differing sets of variables, like so:

toml [apps.db] dev = {DATABASE = "dev", DB_USER = "root"} stg = {DATABASE = "stg", DB_USER = "root", DB_PASSWORD = "goodpw"} prd = {DATABASE = "prd", DB_USER = "root", DB_PASSWORD = "greatpw"}

The dev profile excludes the DB_PASSWORD variable. Beware though, whenever switch to the dev profile, it will simply not output a value for DB_PASSWORD. That means if you're switch from another profile, DB_PASSWORD will retain its old value! For this reason, it's generally best to define the same set of values for every profile in an app, and just use empty values as appropriate.

Cascading configs

On every execution, env-select will scan the current directory for a file called .env-select.toml and parse it for a config. In addition to that, it will walk up the directory tree and check each ancestor directory tree for the same file. If multiple files are found, the results will be merged together, with lower config files having higher precedence. For example, if we execute es TEST_VARIABLE in ~/code/:

```toml

~/code/.env-select.toml

[vars] TEST_VARIABLE = ["abc", "def"]

[apps.server] dev = {SERVICE1 = "secret-dev-server", SERVICE2 = "another-secret-dev-server"} ```

```toml

~/.env-select.toml

[vars] TESTVARIABLE = ["ghi"] OTHERVARIABLE = ["potato", "tomato"]

[apps.server] dev = {SERVICE1 = "dev", SERVICE2 = "also-dev"} prd = {SERVICE1 = "prd", SERVICE2 = "also-prd"} ```

then our resulting config, at execution time, will look like:

```toml

Note: this config never exists in the file system, only in memory during program execution

[vars]

Variable lists get appended together

TESTVARIABLE = ["abc", "def", "ghi"] OTHERVARIABLE = ["potato", "tomato"]

[apps.server]

From ~/code/.env-select.toml (higher precedence)

dev = {SERVICE1 = "secret-dev-server", SERVICE2 = "another-secret-dev-server"}

From ~/.env-select.toml (no value in ~/code/.env-select.toml)

prd = {SERVICE1 = "prd", SERVICE2 = "also-prd"} ```

To see where env-select is loading configs from, and how they are being merged together, run the command with the --verbose (or -v) flag.

Shell Support

env-select supports the following shells:

If you use a different shell and would like support for it, please open an issue and I'll see what I can do!

source Disclaimer

env-select runs as a subprocess to your shell (as all commands do), meaning it cannot modify your shell environment. To get around this, env-select will simply output shell commands that you can then pipe to source to modify your session.

If you think piping stuff to source is dangerous and sPoOky, you're right. But consider the fact that at this point, you've already downloaded and executed a mystery binary on your machine. You should've already done your due diligence.

Alternatively, you can run the command without piping to source, and it will simply print out the commands to modify your environment, which you can copy-paste to run manually.

Bugs/Feedback

If you find a bug or have a feature request, please open an issue on GitHub.