d409d508a5
runCommandNoCC has been replaced by runCommand in nixpkgs |
2 years ago | |
---|---|---|
etc | 3 years ago | |
nix | 2 years ago | |
tests | 3 years ago | |
.gitignore | 3 years ago | |
LICENSE | 3 years ago | |
Makefile | 3 years ago | |
Procfile | 3 years ago | |
README.md | 3 years ago | |
default.nix | 3 years ago | |
epoll.go | 3 years ago | |
flake.lock | 3 years ago | |
flake.nix | 3 years ago | |
go.mod | 3 years ago | |
justfile | 3 years ago | |
main.go | 3 years ago | |
setup.cfg | 3 years ago | |
shell.nix | 3 years ago | |
systemd_sockets.go | 3 years ago | |
treefmt.toml | 3 years ago | |
watcher.go | 2 years ago |
README.md
systemd-vaultd - load vault credentials with systemd units
Mostly written in a train
- Jörg Thalheim
This project's goal is to simplify the loading of HashiCorp Vault secrets from systemd units.
Problem statement
Systemd has an option called LoadCredentials
that allows to provide
credentials to a service:
# myservice.service
[Service]
ExecStart=/usr/bin/myservice.sh
LoadCredential=foobar:/etc/myfoobarcredential.txt
In this case systemd will load credential the file
/etc/myfoobarcredential.txt
and provide it to the service at
$CREDENTIAL_PATH/foobar
.
It's handy because it bypasses file permission issues. /etc/myfoobarcredential.txt can be owned by root, and the unit run as a different or dynamic user.
While vault agent also supports writing these secrets, a major issue is that the consumer service may be started before vault agent was able to retrieve secrets from vault. In that case, systemd would fail to start the service.
The solution
In order to do so, I wrote a systemd-vaultd
service which acts as a proxy
between systemd and vault agent that is running on the machine. It provides a
unix socket that can be used in systemd services in the LoadCredential
option and then waits for vault agent to write these secrets at
/run/systemd-vaultd/<service_name>-<secret_name>
.
We take advantage that in addition to normal paths, systemd also supports loading credentials from unix sockets.
With systemd-vaultd
the service myservice.service
would look like this:
[Service]
ExecStart=/usr/bin/myservice.sh
LoadCredential=foobar:/run/systemd-vaultd/sock
vault agent is then expected to write secrets to /run/systemd-vaultd/
template {
contents = "{{ with secret \"secret/my-secret\" }}{{ .Data.data.foo }}{{ end }}"
destination = "/run/systemd-vaultd/secrets/myservice.service-foo"
}
When myservice
is started, systemd will open a connection to
systemd-vaultd
's socket. systemd-vaultd
then either serve the secrets
from /run/systemd-vaultd/secrets/myservice.service-foo
or it waits with
inotify on secret directory for vault agent to write the secret.
⋈
Installation
The installation requires a go
compiler and make
to be installed.
This command will install the systemd-vaultd
binary to
/usr/bin/systemd-vaultd
as well as installing a following systemd unit
files: systemd-vaultd.service
, systemd-vaultd.socket
:
make install
License
Copyright (c) 2022 Jörg Thalheim and contributors.
This project is free software, and may be redistributed under the terms specified in the LICENSE file.
About
This project is maintained by Numtide.
Need help or support? Contact us