2020-12-18 18:49:50 +01:00
# agenix - [age](https://github.com/FiloSottile/age)-encrypted secrets for NixOS
2020-09-03 21:03:01 +02:00
2020-12-18 18:49:50 +01:00
`agenix` is a commandline tool for managing secrets encrypted with your existing SSH keys. This project also includes the NixOS module `age` for adding encrypted secrets into the Nix store and decrypting them.
2020-09-03 21:03:01 +02:00
2021-12-29 19:13:26 +01:00
## Contents
* [Problem and solution ](#problem-and-solution )
* [Features ](#features )
* [Installation ](#installation )
2023-01-29 23:08:17 +01:00
* [niv ](#install-via-niv )
2021-12-29 19:13:26 +01:00
* [nix-channel ](#install-via-nix-channel )
* [fetchTarball ](#install-via-fetchtarball )
* [flakes ](#install-via-flakes )
* [Tutorial ](#tutorial )
2022-09-03 20:46:45 +02:00
* [Reference ](#reference )
2022-09-03 23:40:00 +02:00
* [`age` module reference ](#age-module-reference )
* [agenix CLI reference ](#agenix-cli-reference )
2022-03-01 04:34:22 +01:00
* [Community and Support ](#community-and-support )
2021-12-29 19:13:26 +01:00
* [Threat model/Warnings ](#threat-modelwarnings )
2023-01-29 17:27:20 +01:00
* [Contributing ](#contributing )
2021-12-29 19:13:26 +01:00
* [Acknowledgements ](#acknowledgements )
2020-12-18 18:49:50 +01:00
## Problem and solution
All files in the Nix store are readable by any system user, so it is not a suitable place for including cleartext secrets. Many existing tools (like NixOps deployment.keys) deploy secrets separately from `nixos-rebuild` , making deployment, caching, and auditing more difficult. Out-of-band secret management is also less reproducible.
`agenix` solves these issues by using your pre-existing SSH key infrastructure and `age` to encrypt secrets into the Nix store. Secrets are decrypted using an SSH host private key during NixOS system activation.
2020-09-03 22:16:44 +02:00
## Features
2020-09-03 21:03:01 +02:00
* Secrets are encrypted with SSH keys
2020-09-03 22:18:21 +02:00
* system public keys via `ssh-keyscan`
* can use public keys available on GitHub for users (for example, https://github.com/ryantm.keys)
2020-09-03 21:03:01 +02:00
* No GPG
* Very little code, so it should be easy for you to audit
2020-09-04 06:12:02 +02:00
* Encrypted secrets are stored in the Nix store, so a separate distribution mechanism is not necessary
2020-09-03 21:03:01 +02:00
2020-12-18 19:09:17 +01:00
## Notices
2020-12-19 00:40:34 +01:00
* Password-protected ssh keys: since the underlying tool age/rage do not support ssh-agent, password-protected ssh keys do not work well. For example, if you need to rekey 20 secrets you will have to enter your password 20 times.
2020-12-18 19:09:17 +01:00
2020-09-03 22:16:44 +02:00
## Installation
2020-09-03 21:03:01 +02:00
2023-01-29 23:08:17 +01:00
< details >
< summary >
2021-12-29 19:13:26 +01:00
### Install via [niv](https://github.com/nmattia/niv)
2020-09-03 21:03:01 +02:00
2023-01-29 23:08:17 +01:00
< / summary >
2020-09-03 21:03:01 +02:00
First add it to niv:
2020-12-18 18:49:50 +01:00
```ShellSession
2020-09-03 21:03:01 +02:00
$ niv add ryantm/agenix
```
2021-12-29 19:13:26 +01:00
#### Install module via niv
2020-09-03 22:16:44 +02:00
2021-12-29 19:13:26 +01:00
Then add the following to your `configuration.nix` in the `imports` list:
2020-09-03 21:03:01 +02:00
```nix
{
2021-12-01 00:08:57 +01:00
imports = [ "${(import ./nix/sources.nix).agenix}/modules/age.nix" ];
2020-09-03 21:03:01 +02:00
}
```
2021-12-29 19:13:26 +01:00
#### Install CLI via niv
To install the `agenix` binary:
```nix
{
2021-12-29 19:20:00 +01:00
environment.systemPackages = [ (pkgs.callPackage "${(import ./nix/sources.nix).agenix}/pkgs/agenix.nix" {}) ];
2021-12-29 19:13:26 +01:00
}
```
2023-01-29 23:08:17 +01:00
< / details >
< details >
< summary >
2021-12-29 19:13:26 +01:00
### Install via nix-channel
2020-09-03 21:03:01 +02:00
2023-01-29 23:08:17 +01:00
< / summary >
2021-12-29 19:13:26 +01:00
As root run:
2020-09-03 21:03:01 +02:00
2020-12-18 18:49:50 +01:00
```ShellSession
2021-12-29 19:13:26 +01:00
$ sudo nix-channel --add https://github.com/ryantm/agenix/archive/main.tar.gz agenix
$ sudo nix-channel --update
2020-09-03 21:03:01 +02:00
```
2021-12-29 19:13:26 +01:00
#### Install module via nix-channel
Then add the following to your `configuration.nix` in the `imports` list:
2020-09-03 21:03:01 +02:00
```nix
{
2021-08-01 13:26:50 +02:00
imports = [ < agenix / modules / age . nix > ];
2020-09-03 21:03:01 +02:00
}
```
2021-12-29 19:13:26 +01:00
#### Install CLI via nix-channel
2021-10-16 18:04:16 +02:00
2021-12-29 19:13:26 +01:00
To install the `agenix` binary:
2021-10-16 18:04:16 +02:00
```nix
{
2021-12-29 19:13:26 +01:00
environment.systemPackages = [ (pkgs.callPackage < agenix / pkgs / agenix . nix > {}) ];
2021-10-16 18:04:16 +02:00
}
```
2023-01-29 23:08:17 +01:00
< / details >
< details >
< summary >
2021-12-29 19:13:26 +01:00
### Install via fetchTarball
2023-01-29 23:08:17 +01:00
< / summary >
2021-12-29 19:13:26 +01:00
#### Install module via fetchTarball
2020-09-03 21:03:01 +02:00
2021-12-29 19:13:26 +01:00
Add the following to your configuration.nix:
2020-09-03 21:03:01 +02:00
2020-09-03 22:16:44 +02:00
```nix
2020-09-03 21:03:01 +02:00
{
2021-11-21 02:30:45 +01:00
imports = [ "${builtins.fetchTarball "https://github.com/ryantm/agenix/archive/main.tar.gz"}/modules/age.nix" ];
2020-09-03 21:03:01 +02:00
}
```
or with pinning:
```nix
{
imports = let
# replace this with an actual commit id or tag
commit = "298b235f664f925b433614dc33380f0662adfc3f";
in [
"${builtins.fetchTarball {
url = "https://github.com/ryantm/agenix/archive/${commit}.tar.gz";
2023-01-29 17:02:57 +01:00
# update hash from nix build output
sha256 = "";
2021-12-01 00:08:57 +01:00
}}/modules/age.nix"
2020-09-03 21:03:01 +02:00
];
}
```
2021-12-29 19:13:26 +01:00
#### Install CLI via fetchTarball
To install the `agenix` binary:
```nix
{
environment.systemPackages = [ (pkgs.callPackage "${builtins.fetchTarball "https://github.com/ryantm/agenix/archive/main.tar.gz"}/pkgs/agenix.nix" {}) ];
}
```
2023-01-29 23:08:17 +01:00
< / details >
< details >
< summary >
2021-12-29 19:13:26 +01:00
### Install via Flakes
2020-09-03 22:16:44 +02:00
2023-01-29 23:08:17 +01:00
< / summary >
2021-12-29 19:13:26 +01:00
#### Install module via Flakes
2020-09-03 21:03:01 +02:00
2020-09-03 22:16:44 +02:00
```nix
2020-09-03 21:03:01 +02:00
{
inputs.agenix.url = "github:ryantm/agenix";
# optional, not necessary for the module
#inputs .agenix.inputs.nixpkgs.follows = "nixpkgs";
outputs = { self, nixpkgs, agenix }: {
# change `yourhostname` to your actual hostname
nixosConfigurations.yourhostname = nixpkgs.lib.nixosSystem {
# change to your system:
system = "x86_64-linux";
modules = [
./configuration.nix
2023-01-29 17:02:57 +01:00
agenix.nixosModules.default
2020-09-03 21:03:01 +02:00
];
};
};
}
```
2021-12-29 19:13:26 +01:00
#### Install CLI via Flakes
2020-09-03 22:16:44 +02:00
2020-09-04 16:13:03 +02:00
You don't need to install it,
2020-09-03 22:16:44 +02:00
2020-12-18 18:49:50 +01:00
```ShellSession
2020-09-03 22:16:44 +02:00
nix run github:ryantm/agenix -- --help
```
2020-09-04 16:13:03 +02:00
but, if you want to (change the system based on your system):
2020-09-04 06:12:02 +02:00
```nix
{
2023-01-29 17:02:57 +01:00
environment.systemPackages = [ agenix.packages.x86_64-linux.default ];
2020-09-04 06:12:02 +02:00
}
```
2023-01-29 23:08:17 +01:00
< / details >
2020-09-03 22:16:44 +02:00
## Tutorial
2021-05-13 05:37:06 +02:00
1. The system you want to deploy secrets to should already exist and
have `sshd` running on it so that it has generated SSH host keys in
`/etc/ssh/` .
2022-09-25 21:57:30 +02:00
2. Make a directory to store secrets and `secrets.nix` file for listing secrets and their public keys (This file is **not** imported into your NixOS configuration. It is only used for the `agenix` CLI.):
2020-09-03 22:16:44 +02:00
2020-12-18 18:49:50 +01:00
```ShellSession
2020-09-03 22:16:44 +02:00
$ mkdir secrets
2020-12-18 20:37:23 +01:00
$ cd secrets
2020-09-04 00:18:20 +02:00
$ touch secrets.nix
2020-09-03 22:16:44 +02:00
```
2021-05-13 05:37:06 +02:00
3. Add public keys to `secrets.nix` file (hint: use `ssh-keyscan` or GitHub (for example, https://github.com/ryantm.keys)):
2020-09-04 00:18:20 +02:00
```nix
let
user1 = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIL0idNvgGiucWgup/mP78zyC23uFjYq0evcWdjGQUaBH";
2020-12-18 18:49:50 +01:00
user2 = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAILI6jSq53F/3hEmSs+oq9L4TwOo1PrDMAgcA1uo1CCV/";
users = [ user1 user2 ];
2020-09-04 00:18:20 +02:00
system1 = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIPJDyIr/FSz1cJdcoW69R+NrWzwGK/+3gJpqD1t8L2zE";
2020-12-18 18:49:50 +01:00
system2 = "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIKzxQgondgEYcLpcPdJLrTdNgZ2gznOHCAxMdaceTUT1";
systems = [ system1 system2 ];
2020-09-04 00:18:20 +02:00
in
{
2020-09-04 06:13:10 +02:00
"secret1.age".publicKeys = [ user1 system1 ];
2020-12-18 18:49:50 +01:00
"secret2.age".publicKeys = users ++ systems;
2020-09-04 00:18:20 +02:00
}
2020-09-03 22:16:44 +02:00
```
2021-05-13 05:37:06 +02:00
4. Edit secret files (these instructions assume your SSH private key is in ~/.ssh/):
2020-12-18 18:49:50 +01:00
```ShellSession
2020-09-03 22:16:44 +02:00
$ agenix -e secret1.age
```
2021-05-13 05:37:06 +02:00
5. Add secret to a NixOS module config:
2020-09-03 22:16:44 +02:00
```nix
2022-09-03 23:47:04 +02:00
{
age.secrets.secret1.file = ../secrets/secret1.age;
}
```
6. Use the secret in your config:
```nix
2022-09-03 23:47:54 +02:00
{
users.users.user1 = {
isNormalUser = true;
passwordFile = config.age.secrets.secret1.path;
};
}
2020-09-03 22:16:44 +02:00
```
2022-09-03 23:47:04 +02:00
7. NixOS rebuild or use your deployment tool like usual.
2020-09-03 22:16:44 +02:00
2022-09-03 23:48:36 +02:00
The secret will be decrypted to the value of `config.age.secrets.secret1.path` (`/run/agenix/secret1` by default).
2021-04-08 20:47:48 +02:00
2022-09-03 20:46:45 +02:00
## Reference
### `age` module reference
#### `age.secrets`
2022-09-03 23:40:00 +02:00
`age.secrets` attrset of secrets. You always need to use this
configuration option. Defaults to `{}` .
2022-09-03 20:46:45 +02:00
#### `age.secrets.<name>.file`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.file` is the path to the encrypted `.age` for this
secret. This is the only required secret option.
2022-09-03 20:46:45 +02:00
Example:
```nix
{
age.secrets.monitrc.file = ../secrets/monitrc.age;
}
```
2022-09-03 23:40:00 +02:00
#### `age.secrets.<name>.path`
`age.secrets.<name>.path` is the path where the secret is decrypted
to. Defaults to `/run/agenix/<name>` (`config.age.secretsDir/< name > `).
Example defining a different path:
```nix
{
age.secrets.monitrc = {
file = ../secrets/monitrc.age;
path = "/etc/monitrc";
};
}
```
For many services, you do not need to set this. Instead, refer to the
decryption path in your configuration with
`config.age.secrets.<name>.path` .
Example referring to path:
```nix
{
users.users.ryantm = {
isNormalUser = true;
passwordFile = config.age.secrets.passwordfile-ryantm.path;
};
}
```
##### builtins.readFile anti-pattern
```nix
{
# Do not do this!
config.password = builtins.readFile config.age.secrets.secret1.path;
}
```
This can cause the cleartext to be placed into the world-readable Nix
store. Instead, have your services read the cleartext path at runtime.
2022-09-03 20:46:45 +02:00
#### `age.secrets.<name>.mode`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.mode` is permissions mode of the decrypted secret
in a format understood by chmod. Usually, you only need to use this in
combination with `age.secrets.<name>.owner` and
`age.secrets.<name>.group`
2022-09-03 20:46:45 +02:00
Example:
```nix
{
age.secrets.nginx-htpasswd = {
file = ../secrets/nginx.htpasswd.age;
mode = "770";
owner = "nginx";
group = "nginx";
};
}
```
#### `age.secrets.<name>.owner`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.owner` is the username of the decrypted file's
owner. Usually, you only need to use this in combination with
`age.secrets.<name>.mode` and `age.secrets.<name>.group`
2022-09-03 20:46:45 +02:00
Example:
```nix
{
age.secrets.nginx-htpasswd = {
file = ../secrets/nginx.htpasswd.age;
mode = "770";
owner = "nginx";
group = "nginx";
};
}
```
#### `age.secrets.<name>.group`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.group` is the name of the decrypted file's
group. Usually, you only need to use this in combination with
`age.secrets.<name>.owner` and `age.secrets.<name>.mode`
2022-09-03 20:46:45 +02:00
Example:
```nix
{
age.secrets.nginx-htpasswd = {
file = ../secrets/nginx.htpasswd.age;
mode = "770";
owner = "nginx";
group = "nginx";
};
}
```
#### `age.secrets.<name>.symlink`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.symlink` is a boolean. If true (the default),
secrets are symlinked to `age.secrets.<name>.path` . If false, secerts
are copied to `age.secrets.<name>.path` . Usually, you want to keep
this as true, because it secure cleanup of secrets no longer
used. (The symlink will still be there, but it will be broken.) If
false, you are responsible for cleaning up your own secrets after you
stop using them.
2022-09-03 20:46:45 +02:00
2022-09-03 23:40:00 +02:00
Some programs do not like following symlinks (for example Java
programs like Elasticsearch).
2022-09-03 20:46:45 +02:00
Example:
```nix
{
age.secrets."elasticsearch.conf" = {
file = ../secrets/elasticsearch.conf.age;
symlink = false;
};
}
```
#### `age.secrets.<name>.name`
2022-09-03 23:40:00 +02:00
`age.secrets.<name>.name` is the string of the name of the file after
it is decrypted. Defaults to the `<name>` in the attrpath, but can be
set separately if you want the file name to be different from the
attribute name part.
2022-09-03 20:46:45 +02:00
Example of a secret with a name different from its attrpath:
```nix
{
age.secrets.monit = {
name = "monitrc";
file = ../secrets/monitrc.age;
};
}
```
#### `age.ageBin`
2022-09-03 23:40:00 +02:00
`age.ageBin` the string of the path to the `age` binary. Usually, you
don't need to change this. Defaults to `rage/bin/rage` .
2022-09-03 20:46:45 +02:00
Overriding `age.ageBin` example:
```nix
2022-09-03 20:50:45 +02:00
{pkgs, ...}:{
2022-09-03 20:46:45 +02:00
age.ageBin = "${pkgs.age}/bin/age";
}
```
#### `age.identityPaths`
2022-09-03 23:40:00 +02:00
`age.identityPaths` is a list of paths to recipient keys to try to use
to decrypt the secrets. All of the file paths must be present, but
only one needs to be able to decrypt the secret. Usually, you don't
need to change this. By default, this is the `rsa` and `ed25519` keys
in `config.services.openssh.hostKeys` .
2022-09-03 20:46:45 +02:00
Overriding `age.identityPaths` example:
```nix
{
age.identityPaths = [ "/var/lib/persistent/ssh_host_ed25519_key" ];
}
```
#### `age.secretsDir`
2022-09-03 23:40:00 +02:00
`age.secretsDir` is the directory where secrets are symlinked to by
default.Usually, you don't need to change this. Defaults to
`/run/agenix` .
2022-09-03 20:46:45 +02:00
Overriding `age.secretsDir` example:
```nix
{
age.secretsDir = "/run/keys";
}
```
#### `age.secretsMountPoint`
2022-09-03 23:40:00 +02:00
`age.secretsMountPoint` is the directory where the secret generations
are created before they are symlinked. Usually, you don't need to
change this. Defaults to `/run/agenix.d` .
2022-09-03 20:46:45 +02:00
Overriding `age.secretsMountPoint` example:
```nix
{
age.secretsMountPoint = "/run/secret-generations";
}
```
2022-09-03 23:40:00 +02:00
### agenix CLI reference
2022-03-01 04:34:22 +01:00
2022-09-03 23:40:00 +02:00
```
agenix -e FILE [-i PRIVATE_KEY]
agenix -r [-i PRIVATE_KEY]
2022-03-01 04:34:22 +01:00
2022-09-03 23:40:00 +02:00
options:
-h, --help show help
-e, --edit FILE edits FILE using $EDITOR
-r, --rekey re-encrypts all secrets with specified recipients
-i, --identity identity to use when decrypting
-v, --verbose verbose output
FILE an age-encrypted file
PRIVATE_KEY a path to a private SSH key used to decrypt file
EDITOR environment variable of editor to use when editing FILE
RULES environment variable with path to Nix file specifying recipient public keys.
Defaults to './secrets.nix'
```
#### Rekeying
2020-09-03 22:16:44 +02:00
2020-09-04 00:18:20 +02:00
If you change the public keys in `secrets.nix` , you should rekey your
2020-09-03 22:16:44 +02:00
secrets:
2020-12-18 18:49:50 +01:00
```ShellSession
2020-09-03 22:16:44 +02:00
$ agenix --rekey
```
To rekey a secret, you have to be able to decrypt it. Because of
randomness in `age` 's encryption algorithms, the files always change
2020-12-18 18:49:50 +01:00
when rekeyed, even if the identities do not. (This eventually could be
improved upon by reading the identities from the age file.)
2020-09-03 22:16:44 +02:00
2022-09-03 23:40:00 +02:00
#### Overriding age binary
2021-12-06 00:18:47 +01:00
2022-09-03 23:40:00 +02:00
The agenix CLI uses `rage` by default as its age implemenation, you
can use the reference implementation `age` with Flakes like this:
2021-12-06 00:18:47 +01:00
```nix
2022-09-03 23:42:11 +02:00
{pkgs,agenix,...}:{
2021-12-06 00:18:47 +01:00
environment.systemPackages = [
2023-01-29 17:02:57 +01:00
(agenix.packages.x86_64-linux.default.override { ageBin = "${pkgs.age}/bin/age"; })
2021-12-06 00:18:47 +01:00
];
}
```
2022-09-03 23:40:00 +02:00
## Community and Support
Support and development discussion is available here on GitHub and
also through [Matrix ](https://matrix.to/#/#agenix:nixos.org ).
2020-09-03 22:16:44 +02:00
## Threat model/Warnings
2022-02-02 22:53:46 +01:00
This project has not been audited by a security professional.
2020-09-03 22:16:44 +02:00
People unfamiliar with `age` might be surprised that secrets are not
authenticated. This means that every attacker that has write access to
2020-12-18 18:49:50 +01:00
the secret files can modify secrets because public keys are exposed.
2020-09-03 22:16:44 +02:00
This seems like not a problem on the first glance because changing the
2020-12-18 18:49:50 +01:00
configuration itself could expose secrets easily. However, reviewing
configuration changes is easier than reviewing random secrets (for
example, 4096-bit rsa keys). This would be solved by having a message
2020-09-03 22:16:44 +02:00
authentication code (MAC) like other implementations like GPG or
[sops ](https://github.com/Mic92/sops-nix ) have, however this was left
out for simplicity in `age` .
2023-01-29 17:27:20 +01:00
## Contributing
* The main branch is protected against direct pushes
* All changes must go through GitHub PR review and get at least one approval
* PR titles and commit messages should be prefixed with at least one of these categories:
* contrib - things that make the project development better
* doc - documentation
* feature - new features
* fix - bug fixes
* Please update or make integration tests for new features
* Use `nix fmt` to format nix code
2023-02-19 03:37:43 +01:00
### Tests
You can run the tests with
```ShellSession
nix flake check
```
You can run the integration tests in interactive mode like this:
```ShellSession
nix run .#checks.x86_64-linux.integration.driverInteractive
```
After it starts, enter `run_tests()` to run the tests.
2020-09-03 22:16:44 +02:00
## Acknowledgements
2020-09-03 21:03:01 +02:00
2020-12-18 18:49:50 +01:00
This project is based off of [sops-nix ](https://github.com/Mic92/sops-nix ) created Mic92. Thank you to Mic92 for inspiration and advice.