Skip to content

fgouteroux/acme-manager

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

50 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

acme_manager

ACME Manager is a tool designed to create, manage, and deploy ACME certificates on servers. It handles automatic renewal, monitors expiration dates, and ensures seamless deployment for applications or proxies.

Acme Manager

Features

  • Certificate Management: Automatically renew certificates 30 days before expiration.
  • Cluster Mode: Operates in a cluster using the Memberlist protocol with automatic leader election.
  • Vault Storage: Stores certificates securely in Vault.
  • DNS/HTTP Challenges: Supports both challenge methods for domain validation.
  • Metrics and Monitoring: Provides application prometheus metrics and a web UI for certificate management.
  • Automatic refresh: Configuration file are periodically refreshed without any service restart.
  • Client Local Certificate: Ensure that client local certificate deployed are always up-to-date.
  • Client Local Cmd Run: Run a custom command once certificate have been created/updated/renewed/deployed on the client.

How It Works

  1. ACME Manager creates certificates using ACME.
  2. Certificates are stored securely in Vault.
  3. The application monitors expiration dates and renews certificates as needed. (by default 30d before expiration)

Usage

usage: acme_manager [<flags>]


Flags:
  -h, --[no-]help                Show context-sensitive help (also try --help-long and --help-man).
      --server.listen-address=":8989"  
                                 server listen address
      --server.tls-cert-file=SERVER.TLS-CERT-FILE  
                                 server tls certificate file
      --server.tls-key-file=SERVER.TLS-KEY-FILE  
                                 server tls key file
      --server.tls-client-ca-file=SERVER.TLS-CLIENT-CA-FILE  
                                 Root certificate authority used to verify client certificates
      --server.http-read-timeout=300  
                                 Read timeout for entire HTTP request, including headers and body
      --server.http-read-header-timeout=10  
                                 Read timeout for HTTP request headers
      --config-path="config.yml"  
                                 Config path
      --env-config-path=".env"   Environment vars config path
      --check-renewal-interval=30m  
                                 Time interval to check if certificate renewal needed
      --check-config-interval=30s  
                                 Time interval to check if config file changes
      --check-token-interval=1m  Time interval to check if tokens expired
      --ring.instance-id=RING.INSTANCE-ID  
                                 Instance ID to register in the ring.
      --ring.instance-addr=RING.INSTANCE-ADDR  
                                 IP address to advertise in the ring. Default is auto-detected.
      --ring.instance-port=7946  Port to advertise in the ring.
      --ring.instance-interface-names=RING.INSTANCE-INTERFACE-NAMES  
                                 List of network interface names to look up when finding the instance IP address.
      --ring.join-members=RING.JOIN-MEMBERS  
                                 Other cluster members to join.
      --[no-]client              Enables client mode.
      --client.manager-url="http://localhost:8989/api/v1"  
                                 Client manager URL ($ACME_MANAGER_URL)
      --client.manager-token=CLIENT.MANAGER-TOKEN  
                                 Client manager token ($ACME_MANAGER_TOKEN)
      --client.tls-ca-file=CLIENT.TLS-CA-FILE  
                                 Client manager tls ca certificate file
      --client.tls-cert-file=CLIENT.TLS-CERT-FILE  
                                 Client manager tls certificate file
      --client.tls-key-file=CLIENT.TLS-KEY-FILE  
                                 Client manager tls key file
      --[no-]client.tls-skip-verify  
                                 Client manager tls skip verify
      --client.config-path="client-config.yml"  
                                 Client config path
      --client.check-config-interval=1m  
                                 Time interval to check if client config file changes and to update local certificate file
      --log.level=info           Only log messages with the given severity or above. One of: [debug, info, warn, error]
      --log.format=logfmt        Output format of log messages. One of: [logfmt, json]
      --[no-]version             Show application version.

Cluster Mode

Acme Manager run in cluster mode with the memberlist protocol.

Memberlist

One instance of the ring is elected to be the leader and this is the only one which will make request to acme servers, store certificate in vault and store non-sensitive data in the key value store of the ring.

If the leader instance goes down, another one will be elected and will start to manage certificates.

Env File

Acme Manager load environment variables from .env file. It's use to configure the dns challenge as lego library need it.

Config file

Any valid acme issuers could be added in issuer block.

Private keys must exists for each given issuer in rootpath_account, here:

  • /tmp/accounts/sectigo/private_key.pem
  • /tmp/accounts/letsencrypt/private_key.pem
common:
  api_key_hash: 123abc456def
  rootpath_account: /tmp/accounts
  rootpath_certificate: /tmp/certificates

issuer:
  sectigo:
    ca_dir_url: https://acme.sectigo.com/v2/OV
    eab: true
    kid: kid_value
    hmac: hmac_value
  letsencrypt:
    ca_dir_url: https://acme-staging-v02.api.letsencrypt.org/directory

storage:
  vault:
    role_id: "role_id_value"
    secret_id: "secret_id_value"
    url: " https://vault.example.com"
    secret_engine: "myengine"
    secret_prefix: "acme"
    certificate_prefix: "certificates"
    token_prefix: "tokens"
    mount_path: "login/approle"

Required Common parameters:

  • api_key_hash (string): the api key hash used to manage tokens, required when api mode is enabled.
  • rootpath_account (string): path to find issuer private keys and account file
  • rootpath_certificate (string): path to temporary store certificate file before storing in vault.

Optional Common parameters:

  • cert_days_renewal (int): Number of days before certificate should be renewed (default: 30).

Optional Issuer parameters:

  • eab (bool): Use External Account Binding for account registration. Requires kid and hmac.
  • kid (string): Key identifier from External CA. Used for External Account Binding
  • hmac (string): MAC key from External CA. Should be in Base64 URL Encoding without padding format. Used for External Account Binding.
  • http_challenge (string): http challenge name to use for domain validation
  • dns_challenge (string): dns challenge name to use for domain validation

Server Mode

Manage certificate with API endpoints in a secured way.

HTTP Method Endpoint Auth Type Supported
GET /api/v1/certificate/metadata Bearer Token
GET, POST, PUT, DELETE /api/v1/certificate Bearer Token
GET, POST, PUT, DELETE /api/v1/token API key Header

See swagger page: http://localhost:8989/swagger/index.html

Generate an API key:

# Generate a random string
API_KEY=$(openssl rand -base64 32)

# Hash the random string with SHA1 and put it in the `api_key_hash` of acme manager config
$ echo -n $API_KEY | sha1sum
96a0585f6d3c3f90f74cdb963e7664f2ee8a10bb  -

# Your API KEY to use for curl command and others.
$ echo $API_KEY
GMZgFB3nYxTgISIqr8YAezgNpxePJqgOeU9o3/JRwS8=

Token endpoint

Required parameters:

  • username (string): token username
  • scope (list of string): token scope

Optional parameters:

  • expires (string): token duration (if not set, expires never)
Obtain a new token:
curl -XPOST \
  'http://localhost:8989/api/v1/token' \
  -H "X-API-Key: GMZgFB3nYxTgISIqr8YAezgNpxePJqgOeU9o3/JRwS8=" \
  -d '{
  "username":"testfgx",
  "scope":["read","create","update","delete"]
}'

{
  "expires": "Never",
  "id": "94e0c649-de98-476a-a5cc-ff1201512605",
  "scope": [
    "read",
    "create",
    "update",
    "delete"
  ],
  "token": "OTRlMGM2NDktZGU5OC00NzZhLWE1Y2MtZmYxMjAxNTEyNjA1OkczdTFUSkUtc1FCM05veEhtQXNVcW0xYXd4OXp4Z19V",
  "tokenHash": "2fba65b7e4c953148427407cd556c9b49043e1a4",
  "username": "testfgx"
}
Update the token scope and add expiration time to 30days
curl -XPUT \
  'http://localhost:8989/api/v1/token' \
  -H "X-API-Key: GMZgFB3nYxTgISIqr8YAezgNpxePJqgOeU9o3/JRwS8=" \
  -d '{
  "id": "94e0c649-de98-476a-a5cc-ff1201512605",
  "username":"testfgx",
  "scope":["read"], "expires":"30d"}
}'

{
  "expires": "2025-02-09 11:09:01 +0000 UTC",
  "id": "94e0c649-de98-476a-a5cc-ff1201512605",
  "scope": [
    "read"
  ],
  "token": "OTRlMGM2NDktZGU5OC00NzZhLWE1Y2MtZmYxMjAxNTEyNjA1OmpTeGtoUUwzd0MwQWl4Vzk1aU9mVjM4RzdIbWwzQ0F6",
  "tokenHash": "e7bf79d0b679fe56014cb8e87358ac459880f6dd",
  "username": "testfgx"
}
Read the token (no token value, contain only the hash)
curl -XGET \
  'http://localhost:8989/api/v1/token/94e0c649-de98-476a-a5cc-ff1201512605' \
  -H "X-API-Key: GMZgFB3nYxTgISIqr8YAezgNpxePJqgOeU9o3/JRwS8="

{
  "hash": "e7bf79d0b679fe56014cb8e87358ac459880f6dd",
  "scope": [
    "read"
  ],
  "username": "testfgx",
  "expires": "2025-02-09 11:09:01 +0000 UTC"
}
Revoke the token
curl -XDELETE \
  'http://localhost:8989/api/v1/token/94e0c649-de98-476a-a5cc-ff1201512605' \
  -H "X-API-Key: GMZgFB3nYxTgISIqr8YAezgNpxePJqgOeU9o3/JRwS8=" \

Certificate endpoint

Required parameters:

  • domain (string): domain certificate
  • issuer (string): issuer certificate
  • csr (string): certificate signing request in PEM format and base64 encoded

Optional parameters:

  • bundle (bool): if true, add the issuers certificate to the new certificate
  • renewal_days (int): number of days before automatic certificate renewal
  • days (int): number of days before certificate expiration
  • san (string, comma separated): DNS domain names to add to certificate
  • http_challenge (string): http challenge name to use for domain validation
  • dns_challenge (string): dns challenge name to use for domain validation
Get the certificate
curl -X 'GET' \
  'http://localhost:8989/api/v1/certificate/letsencrypt/testfgx01.example.com' \
  -H 'accept: application/json' \
  -H 'Authorization: Bearer MDIxYjUwNzUtMmQ....'

{
  "cert": "-----BEGIN CERTIFICATE-----\nMIIFUT...\n-----END CERTIFICATE-----\n",
  "csr": "LS0...",
  "ca_issuer": "\n-----BEGIN CERTIFICATE-----\nMIIFTT...\n-----END CERTIFICATE-----\n",
  "issuer": "letsencrypt",
  "url": "https://acme-staging-v02.api.letsencrypt.org/acme/cert/2b8cfad6a7516ac17349...",
  "domain": "testfgx01.example.com",
  "owner": "testfgx"
}
Obtain a new certificate
curl -X 'POST' \
  'http://localhost:8989/api/v1/certificate' \
  -H 'accept: application/json' \
  -H 'Authorization: Bearer MDIxYjUwNzUtMmQ....' \
  -H 'Content-Type: application/json' \
  -d '{
  "dns_challenge": "ns1",
  "domain": "testfgx01.example.com",
  "issuer": "letsencrypt",
  "renewal_days": 30,
  "csr": "LS0..."
}'

{
  "cert": "-----BEGIN CERTIFICATE-----\nMIIFUT...\n-----END CERTIFICATE-----\n",
  "csr": "LS0...",
  "ca_issuer": "\n-----BEGIN CERTIFICATE-----\nMIIFTT...\n-----END CERTIFICATE-----\n",
  "issuer": "letsencrypt",
  "url": "https://acme-staging-v02.api.letsencrypt.org/acme/cert/2b8cfad6a7516ac17349...",
  "domain": "testfgx01.example.com",
  "owner": "testfgx"
}
Update a certificate (will revoke the old one and create a new one)
curl -X 'PUT' \
  'http://localhost:8989/api/v1/certificate' \
  -H 'accept: application/json' \
  -H 'Authorization: Bearer MDIxYjUwNzUtMmQ....' \
  -H 'Content-Type: application/json' \
  -d '{
  "dns_challenge": "ns1",
  "domain": "testfgx01.example.com",
  "issuer": "letsencrypt",
  "renewal_days": 30,
  "san": "testfgx02.example.com",
  "csr": "LS0..."
}'

{
  "cert": "-----BEGIN CERTIFICATE-----\nMIIFUT...\n-----END CERTIFICATE-----\n",
  "csr": "LS0...",
  "ca_issuer": "\n-----BEGIN CERTIFICATE-----\nMIIFTT...\n-----END CERTIFICATE-----\n",
  "issuer": "letsencrypt",
  "url": "https://acme-staging-v02.api.letsencrypt.org/acme/cert/2b8cfad6a7516ac17349...",
  "domain": "testfgx01.example.com",
  "owner": "testfgx"
}
Revoke a certificate
curl -X 'DELETE' \
  'http://localhost:8989/api/v1/certificate/letsencrypt/testfgx01.example.com' \
  -H 'accept: application/json' \
  -H 'Authorization: Bearer MDIxYjUwNzUtMmQ....'

Token and certificate are retrieved from vault for each get api call.

Client Mode

Acme manager could run in client mode to obtain certificate from acme manager server.

It need the acme manager server url and a token.

The client start with reading the config file, check certificates from acme manager server and deploy them. It regulary check if certificate have been renewed/changed and redeploy them.

Local certificate deployment are controlled by certificate_deploy in common block.

It is also possible to execute a custom command once certificate have been generated/revoked wih cmd_enabled.

The client start a webserver to expose some metrics.

$ acme_manager --client.config-path config.yml --client

ts=2025-01-10T10:18:49.077Z caller=client.go:40 level=info msg="Checking certificates from config file"
ts=2025-01-10T10:18:49.165Z caller=client.go:224 level=info msg="Deployed certificate /etc/myapp/ssl/letsencrypt/testfgx01.example.com.crt"
ts=2025-01-10T10:18:49.165Z caller=client.go:233 level=info msg="Deployed private key /etc/myapp/ssl/letsencrypt/testfgx01.example.com.key"
ts=2025-01-10T10:18:49.173Z caller=cmd.go:29 level=info msg="Command '/usr/bin/systemctl reload myapp' successfully executed"
ts=2025-01-10T10:18:49.174Z caller=main.go:269 level=info msg="Listening on" address=:8989
ts=2025-01-10T10:18:49.174Z caller=main.go:271 level=info msg="TLS is disabled." address=:8989

Client Certificate config file

Optional Common parameters:

  • certificate_deploy (bool): If set to true, deploy certificate and private key in given certificate_dir
  • certificate_dir (string): Directory in which to deploy issuers certificates and private keys
  • certificate_dir_perm (uint32): Unix permission for certificate directory in octal format (default: 0700)
  • certificate_file_perm (uint32): Unix permission for certificate file in octal format (default: 0600)
  • certificate_keyfile_perm (uint32): Unix permission for certificate key file in octal format (default: 0600)
  • cmd_enabled (bool): If set to true, run a custom command after deploying certificates.
  • cmd_run (string): Command to run.
  • cmd_timeout (int): Command timeout.

Optional Certificate parameters:

  • bundle (bool): if true, add the issuers certificate to the new certificate
  • renewal_days (int): number of days before automatic certificate renewal
  • days (int): number of days before certificate expiration
  • san (string, comma separated): DNS domain names to add to certificate
  • http_challenge (string): http challenge name to use for domain validation
  • dns_challenge (string): dns challenge name to use for domain validation
common:
  certificate_deploy: true
  certificate_dir: /etc/myapp/ssl/

  cmd_enabled: true
  cmd_run: /usr/bin/systemcl reload myapp
  cmd_timeout: 30


certificate:
  - domain: testfgx01.example.com
    issuer: letsencrypt

  - domain: testfgx02.example.com
    issuer: sectigo

DNS and HTTP Challenge

acme-manager support DNS and HTTP challenge (thanks to lego lib).

There is a custom HTTP Challenge based on kvring, that allow http domain validation with the embedded http endpoint in acme manager.

Setting the http_challenge: kvring, will store the challenge token in kvring and it could be retrieved with a call like:

curl http://testfgx01.example.com/.well-known/acme-challenge/NClsmGOVJqV9jx8xBLO6kabcxBufpLGcu5oUjjhhu1o

Once the domain is validated, the challenge token value is removed from kvring.

Managed certificate web UI

The endpoint http://localhost:8989/certificates return the page for all managed certificate.

Managed token web UI

The endpoint http://localhost:8989/tokens return the page for all managed tokens.

Metrics Exposed

App metrics

This endpoint return metrics about app itself.

# HELP acme_manager_build_info A metric with a constant '1' value labeled by version, revision, branch, goversion from which acme_manager was built, and the goos and goarch for the build.
# TYPE acme_manager_build_info gauge
acme_manager_build_info{branch="HEAD",goarch="amd64",goos="linux",goversion="go1.22.10",revision="f9b7946ad9150bfd1e9b19ff5d1f8b47ceffdbc3",tags="unknown",version="0.1.5"} 1
# HELP acme_manager_certificate_created_total Number of created certificates by issuer and owner
# TYPE acme_manager_certificate_created_total counter
acme_manager_certificate_created_total{issuer="letsencrypt",owner="testfgx"} 1
acme_manager_certificate_created_total{issuer="sectigo",owner="testfgx"} 1
# HELP acme_manager_certificate_revoked_total Number of revoked certificates by issuer and owner
# TYPE acme_manager_certificate_revoked_total counter
acme_manager_certificate_revoked_total{issuer="sectigo",owner="testfgx"} 1
# HELP acme_manager_certificate_total Number of managed certificates by issuer and owner
# TYPE acme_manager_certificate_total gauge
acme_manager_certificate_total{issuer="letsencrypt",owner="testfgx"} 1
acme_manager_certificate_total{issuer="sectigo",owner="testfgx"} 1
# HELP acme_manager_issuer_config_error 1 if there was an error with issuer config, 0 otherwise
# TYPE acme_manager_issuer_config_error gauge
acme_manager_issuer_config_error{issuer="letsencrypt"} 0
acme_manager_issuer_config_error{issuer="sectigo"} 0
# HELP acme_manager_vault_delete_secret_success_total Number of created vault secrets
# TYPE acme_manager_vault_delete_secret_success_total counter
acme_manager_vault_delete_secret_success_total 1
# HELP acme_manager_vault_get_secret_success_total Number of retrieved vault secrets
# TYPE acme_manager_vault_get_secret_success_total counter
acme_manager_vault_get_secret_success_total 5
# HELP acme_manager_vault_put_secret_success_total Number of created/updated vault secrets
# TYPE acme_manager_vault_put_secret_success_total counter
acme_manager_vault_put_secret_success_total 2

Limitations

Currently only vault storage with app role login is supported.

TLS and basic authentication

Acme Manager supports TLS and basic authentication. This enables better control of the various HTTP endpoints.

To use TLS and/or basic authentication, you need to pass a configuration file using the --web.config.file parameter. The format of the file is described in the exporter-toolkit repository.

Sources

About

Manage acme certificates and deploy them on servers

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages