Breaking change release. Plaintext HTTP listener removed. The certctl control plane now terminates TLS 1.3 on :8443 via http.Server.ListenAndServeTLS. No CERTCTL_TLS_ENABLED=false escape hatch. No dual-listener mode. One-step cutover per docs/upgrade-to-tls.md. Server - cmd/server/tls.go: certHolder with SIGHUP hot-reload + atomic cert swap, buildServerTLSConfig (TLS 1.3 min, GetCertificate callback), preflightServerTLS validation - cmd/server/main.go: ListenAndServeTLS in place of ListenAndServe, watchSIGHUP wiring, cert/key path config threading - tls_test.go: 418-line regression coverage of reload, preflight, callback behavior, SAN validation Config - CERTCTL_TLS_CERT_PATH / CERTCTL_TLS_KEY_PATH (required) - Plaintext rejection: agents/CLI/MCP pre-flight-fail on http:// URLs with a pointer to docs/upgrade-to-tls.md Agents, CLI, MCP - All three pre-flight-reject http:// URLs with fail-loud diagnostic - CERTCTL_SERVER_CA_BUNDLE_PATH for private-CA trust - CERTCTL_SERVER_TLS_INSECURE_SKIP_VERIFY for dev-only bypass (loud warning on startup) - install-agent.sh emits both vars as commented template lines docker-compose - certctl-tls-init sidecar generates SAN-valid self-signed cert into deploy/test/certs/ on first boot - All demo-stack curls pin against ca.crt with --cacert Helm chart - Three TLS provisioning modes, exactly one required: - server.tls.existingSecret (operator-supplied) - server.tls.certManager.enabled (cert-manager integration) - server.tls.selfSigned.enabled (eval only — not for production) - server-certificate.yaml template for cert-manager mode - helm install without a TLS source fails at template render with a pointer to docs/tls.md CI - .github/workflows/ci.yml Helm Chart Validation step renders the chart in both existingSecret and cert-manager modes, plus an inverse guard-regression test that asserts helm template MUST refuse to render when no TLS source is configured. Previously the single `helm template` invocation hit the certctl.tls.required fail-loud guard and exit-1'd CI. Four invocations now: lint (existingSecret), template (existingSecret), template (cert-manager), template (no args — must fail). Integration tests - deploy/test/integration_test.go stands up the Compose stack over HTTPS, extracts the CA bundle, and exercises every certctl API over https://localhost:8443 - All 34 integration subtests green (per Phase 8 local CI-parity) Documentation - New: docs/tls.md (provisioning patterns, rotation, SIGHUP reload) - New: docs/upgrade-to-tls.md (one-step cutover, no-downgrade warnings, fleet-roll sequencing) - CHANGELOG.md: v2.2.0 "HTTPS Everywhere — The Irony" entry (file heading unchanged; release tag is v2.0.47) - All curls in docs/, examples/, deploy/helm/ guides use https://localhost:8443 --cacert Verification - grep -rn "ListenAndServe[^T]" cmd/ internal/ → 0 hits - grep -rn "\"http://" cmd/ internal/ → 2 benign hits (Caddy admin API default, SSRF doc comment) — zero certctl endpoints - Tasks #197–#206 (Phases 0–8) all closed in the tracker Files: 65 changed, 3489 insertions, 372 deletions (pre-CI-fix).
9.2 KiB
Multi-Issuer Example: ACME + Local CA
This example demonstrates certctl managing both public and internal certificates from a single dashboard. Public-facing services use Let's Encrypt (ACME), while internal services use a private Local CA — all visible and managed in one place.
The Use Case
You have:
- Public-facing services (web app, API, etc.) that need TLS certs signed by a trusted public CA (Let's Encrypt)
- Internal services (databases, microservices, middleware) that need TLS certs but don't require public trust
- One team managing certs across both, needing unified visibility and automated renewal
With certctl, both issuer types are configured and available. You assign each certificate to the appropriate issuer via its profile or at enrollment time. The dashboard shows all certs together, with renewal status, expiration timelines, and audit trails — regardless of which CA issued them.
Architecture
flowchart TD
subgraph Server ["certctl Server (Control Plane)"]
A["Let's Encrypt ACME issuer<br/>(HTTP-01 challenges)"]
B["Local CA issuer<br/>(self-signed or sub-CA mode)"]
C["PostgreSQL database<br/>(cert inventory, audit, jobs)"]
end
subgraph Agent ["certctl Agent"]
D["Discovers existing certs<br/>(/etc/nginx/ssl, /etc/app/ssl)"]
E["Polls server for<br/>renewal/issuance/deployment jobs"]
F["Generates keys locally<br/>(agent-side crypto)"]
G["Deploys certs to NGINX<br/>and app service directories"]
end
subgraph Targets ["Target Services"]
H["NGINX (public TLS)<br/>(Let's Encrypt certs)"]
I["App Services (internal TLS)<br/>(Local CA certs)"]
end
Server -->|API polling| Agent
Agent -->|Deploy| H
Agent -->|Deploy| I
Prerequisites
- Docker & Docker Compose — containers run everything
- Port access — 80 (HTTP-01 challenges) and 443 (HTTPS) for Let's Encrypt
- Domain for ACME (optional) — if using real Let's Encrypt, not needed for demo
- Internet connectivity — to reach Let's Encrypt's API (demo can use staging directory)
TLS Security
certctl is HTTPS-only as of v2.2. The demo compose stack provisions a self-signed certificate. When accessing https://localhost:8443, you can either:
- Use
curl --cacert ./deploy/test/certs/ca.crt ...to pin the CA certificate - Use
curl -k ...for quick smoke tests (never in production) - Import the CA at
./deploy/test/certs/ca.crtinto your OS trust store for browser visits
Quick Start
1. Clone or navigate to this directory
cd examples/multi-issuer
2. Set environment variables (optional, defaults provided)
# Email for Let's Encrypt account
export ACME_EMAIL="your-email@example.com"
# Database password (for demo, default is fine)
export DB_PASSWORD="certctl-dev-password"
# Agent API key
export AGENT_API_KEY="agent-demo-key"
# Server port (default 8443)
export SERVER_PORT="8443"
3. Start the services
docker compose up -d
This spins up:
- PostgreSQL database (certctl data store)
- certctl server with ACME and Local CA issuers configured
- certctl agent discovering existing certs and polling for work
- NGINX web server (target for public TLS certs)
4. Access the dashboard
Open your browser to https://localhost:8443 (or your configured SERVER_PORT)
You should see:
- Empty cert inventory (fresh start)
- Two configured issuers: "ACME" and "Local CA"
- One registered agent ("multi-issuer-agent-01")
5. Create test certificates
In the dashboard:
For a public cert (Let's Encrypt):
- Go to Certificates > + New Certificate
- Common Name:
example.com(or a test domain you control) - Issuer: Select "ACME"
- Profile: Select default or create one (key type: RSA 2048, TTL: 90 days)
- Create → The server submits an ACME order
For an internal cert (Local CA):
- Go to Certificates > + New Certificate
- Common Name:
internal-api.internal(or any internal name) - Issuer: Select "Local CA"
- Profile: Select default
- Create → The server issues immediately from the private CA
6. Monitor in the dashboard
- Dashboard — see cert counts by status and issuer
- Certificates page — filter by issuer, see renewal status, expiration timeline
- Audit Trail — track all operations (issuance, renewals, deployments)
- Agents — view agent health and pending work
How Issuer Assignment Works
Via Profiles
Create a profile for each issuer type:
- Profile public-tls → Issuer: ACME, TTL: 90 days, allowed domains:
*.example.com - Profile internal-tls → Issuer: Local CA, TTL: 1 year, allowed SANs: internal DNS names
Then create certificates using the appropriate profile.
Via Direct Assignment
When creating a certificate, explicitly select the issuer. The certificate remembers which issuer it belongs to.
ACME Configuration
The server is configured with Let's Encrypt's production directory:
CERTCTL_ACME_DIRECTORY_URL: https://acme-v02.api.letsencrypt.org/directory
CERTCTL_ACME_EMAIL: admin@example.com
CERTCTL_ACME_CHALLENGE_TYPE: http-01
For testing without a real domain, use Let's Encrypt's staging directory:
# Edit docker-compose.yml and change:
CERTCTL_ACME_DIRECTORY_URL: https://acme-staging-v02.api.letsencrypt.org/directory
Staging certs are untrusted (for testing only) but unlimited rate limits.
Local CA Configuration
The Local CA issuer can operate in two modes:
Mode 1: Self-Signed (Default)
Leave CERTCTL_CA_CERT_PATH and CERTCTL_CA_KEY_PATH empty. The server generates a self-signed root CA on first run.
CERTCTL_CA_CERT_PATH: ""
CERTCTL_CA_KEY_PATH: ""
Use case: Development, testing, internal services that trust a self-signed root.
Mode 2: Sub-CA (Enterprise)
Provide an existing CA cert + key (e.g., from your organization's PKI). The Local CA issues certs signed by that intermediate.
# In docker-compose.yml, volume-mount your CA cert+key:
volumes:
- /path/to/ca.crt:/etc/certctl/ca.crt:ro
- /path/to/ca.key:/etc/certctl/ca.key:ro
# And set env vars:
CERTCTL_CA_CERT_PATH: /etc/certctl/ca.crt
CERTCTL_CA_KEY_PATH: /etc/certctl/ca.key
Use case: Enterprise internal PKI where certs need to chain to a trusted root (e.g., Windows ADCS, OpenSSL, Vault PKI).
Deployment Flow
When you create a certificate and assign it for deployment:
-
Issuance — Server calls the issuer connector (ACME or Local CA)
- ACME: submit challenge, poll until DNS/HTTP validated, retrieve cert
- Local CA: generate and sign immediately
-
Agent picks up work — Agent polls
/api/v1/agents/{id}/work -
Agent deployment — Agent places cert+key in the target directory
- NGINX:
/etc/nginx/ssl/(mounted volume) - App services:
/etc/app/ssl/(mounted volume)
- NGINX:
-
Service reload — Agent triggers reload (NGINX:
nginx -s reload, etc.) -
Dashboard reflects status — Job transitions from
Running→Completed, cert shows asActive
Scaling Beyond Docker Compose
In production:
- Deploy certctl server on a single node (or HA cluster with external PostgreSQL)
- Deploy certctl agents on each server needing cert management
- Point agents to server URL via
CERTCTL_SERVER_URLenv var - Configure issuers on server via env vars or (in V3+) the dashboard UI
- Use profiles to segment issuers — operators select a profile at cert creation time
Each agent independently manages its local cert inventory and deployments. The server coordinates all agent work and provides the unified dashboard.
Troubleshooting
Certs aren't being issued
- Check server logs:
docker compose logs certctl-server - Verify issuer configuration: Dashboard → Issuers, click "Test Connection"
- For ACME, ensure ports 80/443 are open and your domain resolves
Agent can't reach server
- Check network:
docker compose exec certctl-agent curl http://certctl-server:8443/health - Verify
CERTCTL_SERVER_URLenvironment variable
No issuers showing up
- Ensure env vars are set on the server container
- Restart server:
docker compose restart certctl-server - Check server logs for validation errors
Let's Encrypt rate limits
- Use the staging directory for testing (unlimited, untrusted certs)
- Production directory: 50 certs per domain per week
- Read more: https://letsencrypt.org/docs/rate-limits/
Next Steps
- Create a certificate profile — Dashboard → Profiles → + New Profile
- Configure team ownership — Dashboard → Owners/Teams (assign certs to teams)
- Set renewal policies — Dashboard → Policies (expiration thresholds, auto-renewal)
- Enable notifications — Configure Slack/Teams webhook to get alerts on renewals and expirations
- Explore discovery — Agent scans
/etc/nginx/ssland/etc/app/ssl, Dashboard → Discovery shows what's already deployed