mirror of
https://github.com/shankar0123/certctl.git
synced 2026-06-08 18:48:55 +00:00
Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 6c8d4eca40 | |||
| 836534f2a7 | |||
| 648e2f7ab1 | |||
| 6375909591 | |||
| 3e5ff4b9c3 | |||
| 76d0ce2a0f | |||
| 207f2c6879 | |||
| 46a58d518a | |||
| c5be6d059f | |||
| ec209c9736 | |||
| d4f02c5f4b | |||
| 2409f2e464 | |||
| 225c7141b8 | |||
| 8807a7303d |
@@ -62,6 +62,7 @@ certctl-agent
|
|||||||
certctl-cli
|
certctl-cli
|
||||||
/server
|
/server
|
||||||
/agent
|
/agent
|
||||||
|
/cli
|
||||||
|
|
||||||
# Private strategy docs
|
# Private strategy docs
|
||||||
roadmap.md
|
roadmap.md
|
||||||
|
|||||||
@@ -7,6 +7,15 @@
|
|||||||
|
|
||||||
# certctl — Self-Hosted Certificate Lifecycle Platform
|
# certctl — Self-Hosted Certificate Lifecycle Platform
|
||||||
|
|
||||||
|
[](LICENSE)
|
||||||
|
[](https://goreportcard.com/report/github.com/shankar0123/certctl)
|
||||||
|
[](https://github.com/shankar0123/certctl/releases)
|
||||||
|
[](https://github.com/shankar0123/certctl/stargazers)
|
||||||
|
|
||||||
|
TLS certificate lifespans are shrinking fast. The CA/Browser Forum passed [Ballot SC-081v3](https://cabforum.org/2025/04/11/ballot-sc081v3-introduce-schedule-of-reducing-validity-and-data-reuse-periods/) unanimously in April 2025, setting a phased reduction: **200 days** by March 2026, **100 days** by March 2027, and **47 days** by March 2029. Organizations managing dozens or hundreds of certificates can no longer rely on spreadsheets, calendar reminders, or manual renewal workflows. The math doesn't work — at 47-day lifespans, a team managing 100 certificates is processing 7+ renewals per week, every week, forever.
|
||||||
|
|
||||||
|
certctl is a self-hosted platform that automates the entire certificate lifecycle — from issuance through renewal to deployment — with zero human intervention. It works with any certificate authority, deploys to any server, and keeps private keys on your infrastructure where they belong.
|
||||||
|
|
||||||
```mermaid
|
```mermaid
|
||||||
timeline
|
timeline
|
||||||
title TLS Certificate Maximum Lifespan (CA/Browser Forum Ballot SC-081v3)
|
title TLS Certificate Maximum Lifespan (CA/Browser Forum Ballot SC-081v3)
|
||||||
@@ -18,14 +27,6 @@ timeline
|
|||||||
March 2029 : 47 days
|
March 2029 : 47 days
|
||||||
```
|
```
|
||||||
|
|
||||||
TLS certificate lifespans are shrinking fast. The CA/Browser Forum passed [Ballot SC-081v3](https://cabforum.org/2025/04/11/ballot-sc081v3-introduce-schedule-of-reducing-validity-and-data-reuse-periods/) unanimously in April 2025, setting a phased reduction: **200 days** by March 2026, **100 days** by March 2027, and **47 days** by March 2029. Organizations managing dozens or hundreds of certificates can no longer rely on spreadsheets, calendar reminders, or manual renewal workflows. The math doesn't work — at 47-day lifespans, a team managing 100 certificates is processing 7+ renewals per week, every week, forever.
|
|
||||||
|
|
||||||
certctl is a self-hosted platform that automates the entire certificate lifecycle — from issuance through renewal to deployment — with zero human intervention. It works with any certificate authority, deploys to any server, and keeps private keys on your infrastructure where they belong.
|
|
||||||
|
|
||||||
[](LICENSE)
|
|
||||||
[](https://goreportcard.com/report/github.com/shankar0123/certctl)
|
|
||||||
[](https://github.com/shankar0123/certctl/releases)
|
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
|
|
||||||
| Guide | Description |
|
| Guide | Description |
|
||||||
@@ -42,7 +43,7 @@ certctl is a self-hosted platform that automates the entire certificate lifecycl
|
|||||||
| [Migrate from acme.sh](docs/migrate-from-acmesh.md) | Migration guide for acme.sh users with DNS-01 scripts |
|
| [Migrate from acme.sh](docs/migrate-from-acmesh.md) | Migration guide for acme.sh users with DNS-01 scripts |
|
||||||
| [certctl for cert-manager Users](docs/certctl-for-cert-manager-users.md) | Using certctl alongside cert-manager for non-Kubernetes infrastructure |
|
| [certctl for cert-manager Users](docs/certctl-for-cert-manager-users.md) | Using certctl alongside cert-manager for non-Kubernetes infrastructure |
|
||||||
|
|
||||||
> **Next release:** v2.1.0 will be tagged after the full V2 feature suite passes manual QA across all 34 sections of the [testing guide](docs/testing-guide.md). Automated CI (1,471 Go tests + 193 frontend tests) gates every commit; the manual playbook covers integration, deployment, and UX verification that unit tests can't reach.
|
> **Next release:** v2.1.0 will be tagged after the full V2 feature suite passes manual QA across all 34 sections of the [testing guide](docs/testing-guide.md). Automated CI (1,300+ Go tests + 211 frontend tests) gates every commit; the manual playbook covers integration, deployment, and UX verification that unit tests can't reach.
|
||||||
|
|
||||||
## Why certctl Exists
|
## Why certctl Exists
|
||||||
|
|
||||||
@@ -58,8 +59,8 @@ For a detailed comparison with CertKit, KeyTalk, and enterprise platforms (Venaf
|
|||||||
|
|
||||||
certctl gives you a single pane of glass for every TLS certificate in your organization:
|
certctl gives you a single pane of glass for every TLS certificate in your organization:
|
||||||
|
|
||||||
- **Web dashboard** — 22 operational pages: certificate inventory, deployment timeline with TLS verification, bulk operations (renew/revoke/reassign), discovery triage, network scan management, approval workflows, audit trail with CSV/JSON export, agent fleet overview with OS/arch grouping, short-lived credential monitoring, digest email preview
|
- **Web dashboard** — 24 operational pages: certificate inventory, deployment timeline with TLS verification, bulk operations (renew/revoke/reassign), discovery triage, network scan management, approval workflows, audit trail with CSV/JSON export, agent fleet overview with OS/arch grouping, short-lived credential monitoring, digest email preview
|
||||||
- **REST API** — 99 endpoints under `/api/v1/` + `/.well-known/est/` for complete automation, with sparse fields, sort, cursor pagination, and time-range filters
|
- **REST API** — 97 endpoints under `/api/v1/` + `/.well-known/est/` for complete automation, with sparse fields, sort, cursor pagination, and time-range filters
|
||||||
- **Agents** — generate private keys locally (ECDSA P-256), discover existing certs on disk (PEM/DER), submit CSRs only (private keys never leave your servers)
|
- **Agents** — generate private keys locally (ECDSA P-256), discover existing certs on disk (PEM/DER), submit CSRs only (private keys never leave your servers)
|
||||||
- **Network scanner** — discovers certificates on TLS endpoints across CIDR ranges without requiring agents, concurrent scanning with configurable timeouts
|
- **Network scanner** — discovers certificates on TLS endpoints across CIDR ranges without requiring agents, concurrent scanning with configurable timeouts
|
||||||
- **Certificate export** — PEM (JSON or file download) and PKCS#12 formats, with audit trail; private keys never included
|
- **Certificate export** — PEM (JSON or file download) and PKCS#12 formats, with audit trail; private keys never included
|
||||||
@@ -84,8 +85,10 @@ For the full capability breakdown — revocation infrastructure, policy engine,
|
|||||||
| ACME EAB (ZeroSSL, Google Trust) | Implemented (auto-fetch EAB from ZeroSSL) | `ACME` |
|
| ACME EAB (ZeroSSL, Google Trust) | Implemented (auto-fetch EAB from ZeroSSL) | `ACME` |
|
||||||
| step-ca | Implemented | `StepCA` |
|
| step-ca | Implemented | `StepCA` |
|
||||||
| OpenSSL / Custom CA | Implemented | `OpenSSL` |
|
| OpenSSL / Custom CA | Implemented | `OpenSSL` |
|
||||||
| Vault PKI | Future | — |
|
| Vault PKI | Beta | `VaultPKI` |
|
||||||
| DigiCert | Future | — |
|
| DigiCert CertCentral | Beta | `DigiCert` |
|
||||||
|
|
||||||
|
**Vault PKI and DigiCert connectors are in beta.** If you hit any bugs or unexpected behavior, please [open a GitHub issue](https://github.com/shankar0123/certctl/issues) -- we're actively testing these and want to hear from real users.
|
||||||
|
|
||||||
**Note:** ADCS integration is handled via the Local CA's sub-CA mode — certctl operates as a subordinate CA with its signing certificate issued by ADCS. Any CA with a shell-accessible signing interface can be integrated today via the OpenSSL/Custom CA connector.
|
**Note:** ADCS integration is handled via the Local CA's sub-CA mode — certctl operates as a subordinate CA with its signing certificate issued by ADCS. Any CA with a shell-accessible signing interface can be integrated today via the OpenSSL/Custom CA connector.
|
||||||
|
|
||||||
@@ -128,7 +131,7 @@ All connectors are pluggable — build your own by implementing the [connector i
|
|||||||
<tr>
|
<tr>
|
||||||
<td><a href="docs/screenshots/v2-policies.png"><img src="docs/screenshots/v2-policies.png" width="270" alt="Policies"></a><br><b>Policies</b><br><sub>Ownership, lifetime, renewal rules</sub></td>
|
<td><a href="docs/screenshots/v2-policies.png"><img src="docs/screenshots/v2-policies.png" width="270" alt="Policies"></a><br><b>Policies</b><br><sub>Ownership, lifetime, renewal rules</sub></td>
|
||||||
<td><a href="docs/screenshots/v2-profiles.png"><img src="docs/screenshots/v2-profiles.png" width="270" alt="Profiles"></a><br><b>Profiles</b><br><sub>Key types, max TTL, crypto constraints</sub></td>
|
<td><a href="docs/screenshots/v2-profiles.png"><img src="docs/screenshots/v2-profiles.png" width="270" alt="Profiles"></a><br><b>Profiles</b><br><sub>Key types, max TTL, crypto constraints</sub></td>
|
||||||
<td><a href="docs/screenshots/v2-issuers.png"><img src="docs/screenshots/v2-issuers.png" width="270" alt="Issuers"></a><br><b>Issuers</b><br><sub>Local CA, ACME, step-ca connectors</sub></td>
|
<td><a href="docs/screenshots/v2-issuers.png"><img src="docs/screenshots/v2-issuers.png" width="270" alt="Issuers"></a><br><b>Issuers</b><br><sub>Local CA, ACME, step-ca, Vault PKI, DigiCert</sub></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td><a href="docs/screenshots/v2-targets.png"><img src="docs/screenshots/v2-targets.png" width="270" alt="Targets"></a><br><b>Targets</b><br><sub>NGINX, Apache, HAProxy, Traefik, Caddy deployment</sub></td>
|
<td><a href="docs/screenshots/v2-targets.png"><img src="docs/screenshots/v2-targets.png" width="270" alt="Targets"></a><br><b>Targets</b><br><sub>NGINX, Apache, HAProxy, Traefik, Caddy deployment</sub></td>
|
||||||
@@ -142,7 +145,7 @@ All connectors are pluggable — build your own by implementing the [connector i
|
|||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
> **22 operational GUI pages** covering the full certificate lifecycle: dashboard, certificates (list + detail with EKU badges, deployment timeline, TLS verification status), agents, fleet overview, jobs (with approval workflow), notifications, policies, profiles, issuers, targets (wizard with NGINX/Apache/HAProxy/Traefik/Caddy/F5/IIS), owners, teams, agent groups, audit trail, short-lived credentials, discovery triage, and network scan management.
|
> **24 operational GUI pages** covering the full certificate lifecycle: dashboard, certificates (list + detail with EKU badges, deployment timeline, TLS verification status), agents, fleet overview, jobs (list + detail with approval workflow), notifications, policies, profiles, issuers (catalog + detail), targets (list + detail + wizard), owners, teams, agent groups, audit trail, short-lived credentials, discovery triage, network scan management, digest email preview, and observability metrics.
|
||||||
|
|
||||||
## Quick Start
|
## Quick Start
|
||||||
|
|
||||||
@@ -163,7 +166,7 @@ docker compose -f deploy/docker-compose.yml up -d --build
|
|||||||
|
|
||||||
Wait ~30 seconds, then open **http://localhost:8443** in your browser.
|
Wait ~30 seconds, then open **http://localhost:8443** in your browser.
|
||||||
|
|
||||||
The dashboard comes pre-loaded with 35 demo certificates across 5 issuers, 8 agents, 90 days of job history, discovery scan data, and network scan targets — a realistic snapshot of a certificate inventory that looks like it's been running for months.
|
The dashboard comes pre-loaded with 32 demo certificates across 7 issuers, 8 agents, 180 days of job history, discovery scan data, and network scan targets — a realistic snapshot of a certificate inventory that looks like it's been running for months.
|
||||||
|
|
||||||
Verify the API:
|
Verify the API:
|
||||||
```bash
|
```bash
|
||||||
@@ -171,7 +174,7 @@ curl http://localhost:8443/health
|
|||||||
# {"status":"healthy"}
|
# {"status":"healthy"}
|
||||||
|
|
||||||
curl -s http://localhost:8443/api/v1/certificates | jq '.total'
|
curl -s http://localhost:8443/api/v1/certificates | jq '.total'
|
||||||
# 35
|
# 32
|
||||||
```
|
```
|
||||||
|
|
||||||
### Agent Install (One-Liner)
|
### Agent Install (One-Liner)
|
||||||
@@ -371,7 +374,7 @@ make docker-clean # Stop + remove volumes
|
|||||||
|
|
||||||
## API Overview
|
## API Overview
|
||||||
|
|
||||||
99 endpoints under `/api/v1/` + `/.well-known/est/`, all returning JSON. List endpoints support pagination, sparse field selection (`?fields=`), sort (`?sort=-notAfter`), time-range filters, and cursor-based pagination. Full request/response schemas in the [OpenAPI 3.1 spec](api/openapi.yaml).
|
97 endpoints under `/api/v1/` + `/.well-known/est/`, all returning JSON. List endpoints support pagination, sparse field selection (`?fields=`), sort (`?sort=-notAfter`), time-range filters, and cursor-based pagination. Full request/response schemas in the [OpenAPI 3.1 spec](api/openapi.yaml).
|
||||||
|
|
||||||
### Key Endpoints
|
### Key Endpoints
|
||||||
```
|
```
|
||||||
@@ -448,7 +451,7 @@ certctl-cli certs list --format json # JSON output (default: table)
|
|||||||
|
|
||||||
## MCP Server (AI Integration)
|
## MCP Server (AI Integration)
|
||||||
|
|
||||||
certctl ships a standalone MCP (Model Context Protocol) server that exposes all 78 API endpoints as tools for AI assistants — Claude, Cursor, Windsurf, OpenClaw, VS Code Copilot, and any MCP-compatible client.
|
certctl ships a standalone MCP (Model Context Protocol) server that exposes all 80 API endpoints as tools for AI assistants — Claude, Cursor, Windsurf, OpenClaw, VS Code Copilot, and any MCP-compatible client.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Install
|
# Install
|
||||||
@@ -484,7 +487,7 @@ Core lifecycle management — Local CA + ACME v2 issuers, NGINX target connector
|
|||||||
|
|
||||||
### V2: Operational Maturity
|
### V2: Operational Maturity
|
||||||
|
|
||||||
30 milestones complete, 1500+ tests. See the [Feature Inventory](docs/features.md) for details on every capability.
|
30+ milestones complete, 1,500+ tests. See the [Feature Inventory](docs/features.md) for details on every capability.
|
||||||
|
|
||||||
**What shipped (all ✅):**
|
**What shipped (all ✅):**
|
||||||
|
|
||||||
@@ -496,7 +499,7 @@ Core lifecycle management — Local CA + ACME v2 issuers, NGINX target connector
|
|||||||
- **Observability** — Prometheus + JSON metrics, 5 stats API endpoints, dashboard charts (heatmap, trends, distribution), agent fleet overview, structured logging
|
- **Observability** — Prometheus + JSON metrics, 5 stats API endpoints, dashboard charts (heatmap, trends, distribution), agent fleet overview, structured logging
|
||||||
- **EST Server** (RFC 7030) — device/WiFi certificate enrollment, PKCS#7 wire format, configurable issuer + profile binding
|
- **EST Server** (RFC 7030) — device/WiFi certificate enrollment, PKCS#7 wire format, configurable issuer + profile binding
|
||||||
- **MCP Server** — 78 API operations as AI tools for Claude, Cursor, and any MCP-compatible client
|
- **MCP Server** — 78 API operations as AI tools for Claude, Cursor, and any MCP-compatible client
|
||||||
- **CLI** — 12 subcommands (list/get/renew/revoke certs, agents, jobs, import, status), JSON/table output
|
- **CLI** — 10 subcommands (list/get/renew/revoke certs, list agents/jobs, import, status, health, metrics), JSON/table output
|
||||||
- **Notifications** — Email (SMTP), Webhooks, Slack, Microsoft Teams, PagerDuty, OpsGenie connectors
|
- **Notifications** — Email (SMTP), Webhooks, Slack, Microsoft Teams, PagerDuty, OpsGenie connectors
|
||||||
- **API Enhancements** — sparse fields, sort, time-range filters, cursor pagination, immutable API audit logging
|
- **API Enhancements** — sparse fields, sort, time-range filters, cursor pagination, immutable API audit logging
|
||||||
- **Compliance Mapping** — SOC 2 Type II, PCI-DSS 4.0, NIST SP 800-57 alignment guides
|
- **Compliance Mapping** — SOC 2 Type II, PCI-DSS 4.0, NIST SP 800-57 alignment guides
|
||||||
@@ -509,16 +512,17 @@ Core lifecycle management — Local CA + ACME v2 issuers, NGINX target connector
|
|||||||
- **Scheduled Certificate Digest** — HTML email digests with certificate stats, expiration timeline, job trends, and agent health; configurable daily/hourly/weekly briefings via SMTP
|
- **Scheduled Certificate Digest** — HTML email digests with certificate stats, expiration timeline, job trends, and agent health; configurable daily/hourly/weekly briefings via SMTP
|
||||||
- **Helm Chart** — Production-ready Kubernetes with server Deployment, PostgreSQL StatefulSet with PVC, Agent DaemonSet, security contexts, resource limits, optional Ingress
|
- **Helm Chart** — Production-ready Kubernetes with server Deployment, PostgreSQL StatefulSet with PVC, Agent DaemonSet, security contexts, resource limits, optional Ingress
|
||||||
|
|
||||||
**Coming in v2.1.0:**
|
**Also shipped:**
|
||||||
- Vault PKI issuer connector (HashiCorp Vault /sign API)
|
|
||||||
- DigiCert CertCentral issuer connector (enterprise CA)
|
|
||||||
- Dynamic issuer and target configuration via GUI (no env var restarts)
|
|
||||||
- Issuer catalog page (see all supported CAs, configure from dashboard)
|
- Issuer catalog page (see all supported CAs, configure from dashboard)
|
||||||
- First-run onboarding wizard
|
- Vault PKI and DigiCert CertCentral issuer connectors (Beta)
|
||||||
- Turnkey deployment examples (ACME+NGINX, wildcard+DNS-01, private CA+Traefik, step-ca+HAProxy, multi-issuer)
|
- Turnkey deployment examples (ACME+NGINX, wildcard+DNS-01, private CA+Traefik, step-ca+HAProxy, multi-issuer)
|
||||||
- Migration guides (Certbot, acme.sh, cert-manager complement)
|
- Migration guides (Certbot, acme.sh, cert-manager complement)
|
||||||
- One-line agent install script with cross-compiled binaries
|
- One-line agent install script with cross-compiled binaries
|
||||||
|
|
||||||
|
**Coming in v2.1.0:**
|
||||||
|
- Dynamic issuer and target configuration via GUI (no env var restarts)
|
||||||
|
- First-run onboarding wizard
|
||||||
|
|
||||||
### V3: certctl Pro
|
### V3: certctl Pro
|
||||||
|
|
||||||
Team access controls, identity provider integration, enterprise deployment targets, compliance and risk scoring, advanced fleet operations, event-driven architecture, advanced search, real-time operational views.
|
Team access controls, identity provider integration, enterprise deployment targets, compliance and risk scoring, advanced fleet operations, event-driven architecture, advanced search, real-time operational views.
|
||||||
|
|||||||
+1
-1
@@ -2643,7 +2643,7 @@ components:
|
|||||||
# ─── Issuers ─────────────────────────────────────────────────────
|
# ─── Issuers ─────────────────────────────────────────────────────
|
||||||
IssuerType:
|
IssuerType:
|
||||||
type: string
|
type: string
|
||||||
enum: [ACME, GenericCA, StepCA]
|
enum: [ACME, GenericCA, StepCA, VaultPKI, DigiCert]
|
||||||
|
|
||||||
Issuer:
|
Issuer:
|
||||||
type: object
|
type: object
|
||||||
|
|||||||
@@ -19,8 +19,10 @@ import (
|
|||||||
"github.com/shankar0123/certctl/internal/domain"
|
"github.com/shankar0123/certctl/internal/domain"
|
||||||
acmeissuer "github.com/shankar0123/certctl/internal/connector/issuer/acme"
|
acmeissuer "github.com/shankar0123/certctl/internal/connector/issuer/acme"
|
||||||
"github.com/shankar0123/certctl/internal/connector/issuer/local"
|
"github.com/shankar0123/certctl/internal/connector/issuer/local"
|
||||||
|
digicertissuer "github.com/shankar0123/certctl/internal/connector/issuer/digicert"
|
||||||
opensslissuer "github.com/shankar0123/certctl/internal/connector/issuer/openssl"
|
opensslissuer "github.com/shankar0123/certctl/internal/connector/issuer/openssl"
|
||||||
stepcaissuer "github.com/shankar0123/certctl/internal/connector/issuer/stepca"
|
stepcaissuer "github.com/shankar0123/certctl/internal/connector/issuer/stepca"
|
||||||
|
vaultissuer "github.com/shankar0123/certctl/internal/connector/issuer/vault"
|
||||||
notifyemail "github.com/shankar0123/certctl/internal/connector/notifier/email"
|
notifyemail "github.com/shankar0123/certctl/internal/connector/notifier/email"
|
||||||
notifyopsgenie "github.com/shankar0123/certctl/internal/connector/notifier/opsgenie"
|
notifyopsgenie "github.com/shankar0123/certctl/internal/connector/notifier/opsgenie"
|
||||||
notifypagerduty "github.com/shankar0123/certctl/internal/connector/notifier/pagerduty"
|
notifypagerduty "github.com/shankar0123/certctl/internal/connector/notifier/pagerduty"
|
||||||
@@ -133,6 +135,27 @@ func main() {
|
|||||||
}, logger)
|
}, logger)
|
||||||
logger.Info("initialized OpenSSL/Custom CA issuer connector")
|
logger.Info("initialized OpenSSL/Custom CA issuer connector")
|
||||||
|
|
||||||
|
// Initialize Vault PKI issuer connector (for HashiCorp Vault internal PKI).
|
||||||
|
// Uses the Vault HTTP API with token authentication.
|
||||||
|
vaultConnector := vaultissuer.New(&vaultissuer.Config{
|
||||||
|
Addr: os.Getenv("CERTCTL_VAULT_ADDR"),
|
||||||
|
Token: os.Getenv("CERTCTL_VAULT_TOKEN"),
|
||||||
|
Mount: getEnvDefault("CERTCTL_VAULT_MOUNT", "pki"),
|
||||||
|
Role: os.Getenv("CERTCTL_VAULT_ROLE"),
|
||||||
|
TTL: getEnvDefault("CERTCTL_VAULT_TTL", "8760h"),
|
||||||
|
}, logger)
|
||||||
|
logger.Info("initialized Vault PKI issuer connector")
|
||||||
|
|
||||||
|
// Initialize DigiCert CertCentral issuer connector (for enterprise public CA).
|
||||||
|
// Uses the DigiCert REST API with async order model.
|
||||||
|
digicertConnector := digicertissuer.New(&digicertissuer.Config{
|
||||||
|
APIKey: os.Getenv("CERTCTL_DIGICERT_API_KEY"),
|
||||||
|
OrgID: os.Getenv("CERTCTL_DIGICERT_ORG_ID"),
|
||||||
|
ProductType: getEnvDefault("CERTCTL_DIGICERT_PRODUCT_TYPE", "ssl_basic"),
|
||||||
|
BaseURL: getEnvDefault("CERTCTL_DIGICERT_BASE_URL", "https://www.digicert.com/services/v2"),
|
||||||
|
}, logger)
|
||||||
|
logger.Info("initialized DigiCert CertCentral issuer connector")
|
||||||
|
|
||||||
// Build issuer registry: maps issuer IDs (from database) to connector implementations.
|
// Build issuer registry: maps issuer IDs (from database) to connector implementations.
|
||||||
// "iss-local" matches the seed data issuer ID for the Local CA.
|
// "iss-local" matches the seed data issuer ID for the Local CA.
|
||||||
// "iss-acme-staging" and "iss-acme-prod" are conventional IDs for ACME issuers.
|
// "iss-acme-staging" and "iss-acme-prod" are conventional IDs for ACME issuers.
|
||||||
@@ -145,6 +168,19 @@ func main() {
|
|||||||
"iss-stepca": service.NewIssuerConnectorAdapter(stepcaConnector),
|
"iss-stepca": service.NewIssuerConnectorAdapter(stepcaConnector),
|
||||||
"iss-openssl": service.NewIssuerConnectorAdapter(opensslConnector),
|
"iss-openssl": service.NewIssuerConnectorAdapter(opensslConnector),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Conditionally register Vault PKI (only if CERTCTL_VAULT_ADDR is set)
|
||||||
|
if os.Getenv("CERTCTL_VAULT_ADDR") != "" {
|
||||||
|
issuerRegistry["iss-vault"] = service.NewIssuerConnectorAdapter(vaultConnector)
|
||||||
|
logger.Info("Vault PKI issuer registered", "id", "iss-vault")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Conditionally register DigiCert (only if CERTCTL_DIGICERT_API_KEY is set)
|
||||||
|
if os.Getenv("CERTCTL_DIGICERT_API_KEY") != "" {
|
||||||
|
issuerRegistry["iss-digicert"] = service.NewIssuerConnectorAdapter(digicertConnector)
|
||||||
|
logger.Info("DigiCert CertCentral issuer registered", "id", "iss-digicert")
|
||||||
|
}
|
||||||
|
|
||||||
logger.Info("issuer registry configured", "issuers", len(issuerRegistry))
|
logger.Info("issuer registry configured", "issuers", len(issuerRegistry))
|
||||||
|
|
||||||
// Initialize revocation repository
|
// Initialize revocation repository
|
||||||
@@ -544,6 +580,14 @@ func main() {
|
|||||||
logger.Info("certctl server stopped")
|
logger.Info("certctl server stopped")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// getEnvDefault reads an environment variable with a default fallback.
|
||||||
|
func getEnvDefault(key, defaultVal string) string {
|
||||||
|
if val := os.Getenv(key); val != "" {
|
||||||
|
return val
|
||||||
|
}
|
||||||
|
return defaultVal
|
||||||
|
}
|
||||||
|
|
||||||
// getEnvIntDefault parses an integer from a string with a default fallback.
|
// getEnvIntDefault parses an integer from a string with a default fallback.
|
||||||
func getEnvIntDefault(s string, defaultVal int) int {
|
func getEnvIntDefault(s string, defaultVal int) int {
|
||||||
if s == "" {
|
if s == "" {
|
||||||
|
|||||||
@@ -80,13 +80,16 @@ flowchart TB
|
|||||||
CA2["ACME\n(HTTP-01 + DNS-01 + DNS-PERSIST-01)\n(EAB, ZeroSSL auto-EAB)"]
|
CA2["ACME\n(HTTP-01 + DNS-01 + DNS-PERSIST-01)\n(EAB, ZeroSSL auto-EAB)"]
|
||||||
CA3["step-ca\n(/sign API)"]
|
CA3["step-ca\n(/sign API)"]
|
||||||
CA4["OpenSSL / Custom CA\n(script-based)"]
|
CA4["OpenSSL / Custom CA\n(script-based)"]
|
||||||
CA6["Vault PKI\n(planned)"]
|
CA6["Vault PKI\n(token auth, /sign API)"]
|
||||||
|
CA7["DigiCert CertCentral\n(async order model)"]
|
||||||
end
|
end
|
||||||
|
|
||||||
subgraph "Target Systems"
|
subgraph "Target Systems"
|
||||||
T1["NGINX\n(file write + reload)"]
|
T1["NGINX\n(file write + reload)"]
|
||||||
T4["Apache httpd\n(file write + reload)"]
|
T4["Apache httpd\n(file write + reload)"]
|
||||||
T5["HAProxy\n(combined PEM + reload)"]
|
T5["HAProxy\n(combined PEM + reload)"]
|
||||||
|
T6["Traefik\n(file provider)"]
|
||||||
|
T7["Caddy\n(admin API / file)"]
|
||||||
T2["F5 BIG-IP\n(proxy agent + iControl REST, planned)"]
|
T2["F5 BIG-IP\n(proxy agent + iControl REST, planned)"]
|
||||||
T3["IIS\n(agent-local PowerShell, planned)"]
|
T3["IIS\n(agent-local PowerShell, planned)"]
|
||||||
end
|
end
|
||||||
@@ -96,7 +99,7 @@ flowchart TB
|
|||||||
SVC --> REPO
|
SVC --> REPO
|
||||||
REPO --> PG
|
REPO --> PG
|
||||||
SCHED --> SVC
|
SCHED --> SVC
|
||||||
SVC -->|"Issue/Renew"| CA1 & CA2 & CA3
|
SVC -->|"Issue/Renew"| CA1 & CA2 & CA3 & CA4 & CA6 & CA7
|
||||||
|
|
||||||
A1 & A2 & A3 -->|"CSR + Heartbeat"| API
|
A1 & A2 & A3 -->|"CSR + Heartbeat"| API
|
||||||
API -->|"Cert + Chain\n(NO private key)"| A1 & A2 & A3
|
API -->|"Cert + Chain\n(NO private key)"| A1 & A2 & A3
|
||||||
@@ -506,7 +509,8 @@ flowchart TB
|
|||||||
II --> ACME["ACME v2"]
|
II --> ACME["ACME v2"]
|
||||||
II --> SC["step-ca"]
|
II --> SC["step-ca"]
|
||||||
II --> OC["OpenSSL / Custom CA"]
|
II --> OC["OpenSSL / Custom CA"]
|
||||||
II --> VP["Vault PKI (planned)"]
|
II --> VP["Vault PKI"]
|
||||||
|
II --> DC["DigiCert CertCentral"]
|
||||||
end
|
end
|
||||||
|
|
||||||
subgraph "Target Connectors"
|
subgraph "Target Connectors"
|
||||||
@@ -570,7 +574,7 @@ type Connector interface {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Built-in issuers: **Local CA** (self-signed or sub-CA mode using `crypto/x509`), **ACME v2** (HTTP-01, DNS-01, and DNS-PERSIST-01 challenges, compatible with Let's Encrypt, ZeroSSL, Sectigo, Google Trust Services, and any ACME-compliant CA), **step-ca** (Smallstep private CA via native /sign API with JWK provisioner auth), and **OpenSSL/Custom CA** (script-based signing delegating to user-provided shell scripts). The ACME connector uses `golang.org/x/crypto/acme`, generates an ECDSA P-256 account key, handles account registration with ToS acceptance and optional External Account Binding (EAB) for CAs that require it (ZeroSSL, Google Trust Services, SSL.com), order creation, challenge solving (HTTP-01 via built-in server, DNS-01 via script-based hooks, DNS-PERSIST-01 via standing TXT records with auto-fallback to DNS-01), order finalization, and DER-to-PEM chain conversion. For ZeroSSL, EAB credentials are auto-fetched from ZeroSSL's public API when the directory URL is detected as ZeroSSL and no EAB credentials are provided — zero-friction onboarding with no dashboard visit required.
|
Built-in issuers: **Local CA** (self-signed or sub-CA mode using `crypto/x509`), **ACME v2** (HTTP-01, DNS-01, and DNS-PERSIST-01 challenges, compatible with Let's Encrypt, ZeroSSL, Sectigo, Google Trust Services, and any ACME-compliant CA), **step-ca** (Smallstep private CA via native /sign API with JWK provisioner auth), **OpenSSL/Custom CA** (script-based signing delegating to user-provided shell scripts), **Vault PKI** (HashiCorp Vault's PKI secrets engine via /sign API with token auth), and **DigiCert** (commercial CA via CertCentral REST API with async order processing). The ACME connector uses `golang.org/x/crypto/acme`, generates an ECDSA P-256 account key, handles account registration with ToS acceptance and optional External Account Binding (EAB) for CAs that require it (ZeroSSL, Google Trust Services, SSL.com), order creation, challenge solving (HTTP-01 via built-in server, DNS-01 via script-based hooks, DNS-PERSIST-01 via standing TXT records with auto-fallback to DNS-01), order finalization, and DER-to-PEM chain conversion. For ZeroSSL, EAB credentials are auto-fetched from ZeroSSL's public API when the directory URL is detected as ZeroSSL and no EAB credentials are provided — zero-friction onboarding with no dashboard visit required.
|
||||||
|
|
||||||
**ACME Renewal Information (ARI, RFC 9702):** The ACME connector supports CA-directed renewal timing via the `GetRenewalInfo()` method. Instead of using fixed thresholds (e.g., renew 30 days before expiry), the CA tells certctl when to renew by providing a `suggestedWindow` with start and end times. This is useful for distributing renewal load during maintenance windows and coordinating mass-revocation scenarios. Enable with `CERTCTL_ACME_ARI_ENABLED=true`. Cert ID is computed as `base64url(SHA-256(DER cert))` per RFC 9702. If the CA doesn't support ARI (404 from the ARI endpoint), certctl automatically falls back to threshold-based renewal — no operator intervention required. Errors from the CA are logged as warnings.
|
**ACME Renewal Information (ARI, RFC 9702):** The ACME connector supports CA-directed renewal timing via the `GetRenewalInfo()` method. Instead of using fixed thresholds (e.g., renew 30 days before expiry), the CA tells certctl when to renew by providing a `suggestedWindow` with start and end times. This is useful for distributing renewal load during maintenance windows and coordinating mass-revocation scenarios. Enable with `CERTCTL_ACME_ARI_ENABLED=true`. Cert ID is computed as `base64url(SHA-256(DER cert))` per RFC 9702. If the CA doesn't support ARI (404 from the ARI endpoint), certctl automatically falls back to threshold-based renewal — no operator intervention required. Errors from the CA are logged as warnings.
|
||||||
|
|
||||||
@@ -647,7 +651,7 @@ type ESTService interface {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
**Issuer connector extension:** EST required adding `GetCACertPEM(ctx) (string, error)` to the issuer connector interface so the `/cacerts` endpoint can serve the CA chain. The Local CA connector returns its CA certificate PEM; ACME, step-ca, and OpenSSL connectors return errors (they don't expose a static CA chain — their chains are per-issuance).
|
**Issuer connector extension:** EST required adding `GetCACertPEM(ctx) (string, error)` to the issuer connector interface so the `/cacerts` endpoint can serve the CA chain. The Local CA connector returns its CA certificate PEM; ACME, step-ca, OpenSSL, Vault, and DigiCert connectors return errors (they don't expose a static CA chain — their chains are per-issuance).
|
||||||
|
|
||||||
**Audit:** Every EST enrollment is recorded in the audit trail with `protocol: "EST"`, the CN, SANs, issuer ID, serial number, and optional profile ID.
|
**Audit:** Every EST enrollment is recorded in the audit trail with `protocol: "EST"`, the CN, SANs, issuer ID, serial number, and optional profile ID.
|
||||||
|
|
||||||
|
|||||||
+47
-4
@@ -312,12 +312,55 @@ The `GetCACertPEM()` method returns the PEM-encoded CA certificate chain, used b
|
|||||||
|
|
||||||
Note: EST (Enrollment over Secure Transport) is not a connector — it's a protocol handler (`internal/api/handler/est.go`) that delegates certificate issuance to whichever issuer connector is configured via `CERTCTL_EST_ISSUER_ID`. See the [Architecture Guide](architecture.md#est-server-rfc-7030) for details.
|
Note: EST (Enrollment over Secure Transport) is not a connector — it's a protocol handler (`internal/api/handler/est.go`) that delegates certificate issuance to whichever issuer connector is configured via `CERTCTL_EST_ISSUER_ID`. See the [Architecture Guide](architecture.md#est-server-rfc-7030) for details.
|
||||||
|
|
||||||
### Coming in V2.1
|
### Built-in: Vault PKI
|
||||||
|
|
||||||
The following issuer connectors are planned for the v2.1.0 release:
|
The Vault PKI connector integrates with HashiCorp Vault's PKI secrets engine using its native `/sign` API with token-based authentication. This is ideal for organizations using Vault as their internal certificate authority — synchronous issuance without the complexity of ACME or challenge solving.
|
||||||
|
|
||||||
- **Vault PKI** — HashiCorp Vault's PKI secrets engine (`/v1/{mount}/sign/{role}` API) for organizations using Vault as their internal CA. Token auth, configurable mount and role.
|
**Configuration:**
|
||||||
- **DigiCert** — Commercial CA integration via DigiCert CertCentral REST API. Async order model (submit → poll for completion). OV/EV certificate support.
|
|
||||||
|
| Variable | Default | Description |
|
||||||
|
|----------|---------|-------------|
|
||||||
|
| `CERTCTL_VAULT_ADDR` | — | Vault server address (e.g., `https://vault.internal:8200`) |
|
||||||
|
| `CERTCTL_VAULT_TOKEN` | — | Vault auth token with permissions on the PKI mount |
|
||||||
|
| `CERTCTL_VAULT_MOUNT` | `pki` | PKI secrets engine mount path |
|
||||||
|
| `CERTCTL_VAULT_ROLE` | — | PKI role name for certificate signing |
|
||||||
|
| `CERTCTL_VAULT_TTL` | `8760h` | Certificate validity period (TTL) |
|
||||||
|
|
||||||
|
The connector is registered in the issuer registry under `iss-vault`. Vault issues certificates synchronously via the `/v1/{mount}/sign/{role}` API with `X-Vault-Token` header authentication. The issued certificate is parsed to extract serial number, validity dates, and chain information.
|
||||||
|
|
||||||
|
**Note:** CRL and OCSP are managed by Vault itself. Clients should validate certificate status against Vault's own CRL/OCSP endpoints (`GET /v1/{mount}/crl` and Vault's OCSP responder). certctl does not generate local CRL/OCSP for Vault-issued certificates. Revocation is recorded locally but Vault is the authoritative source.
|
||||||
|
|
||||||
|
Location: `internal/connector/issuer/vault/vault.go`
|
||||||
|
|
||||||
|
### Built-in: DigiCert CertCentral
|
||||||
|
|
||||||
|
The DigiCert connector integrates with DigiCert's CertCentral REST API for ordering and managing certificates from DigiCert's commercial CA. It supports both Domain Validated (DV) and Organization/Extended Validated (OV/EV) certificates, with async order processing.
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
|
||||||
|
| Variable | Default | Description |
|
||||||
|
|----------|---------|-------------|
|
||||||
|
| `CERTCTL_DIGICERT_API_KEY` | — | DigiCert API key (X-DC-DEVKEY header) |
|
||||||
|
| `CERTCTL_DIGICERT_ORG_ID` | — | DigiCert organization ID |
|
||||||
|
| `CERTCTL_DIGICERT_PRODUCT_TYPE` | `ssl_basic` | Certificate product (e.g., `ssl_basic`, `ssl_plus`, `ssl_ev`) |
|
||||||
|
| `CERTCTL_DIGICERT_BASE_URL` | `https://www.digicert.com/services/v2` | DigiCert API base URL |
|
||||||
|
|
||||||
|
The connector submits certificate orders to DigiCert's `/order/certificate/create` API. DV certificates may issue immediately; OV/EV certificates require validation (handled by DigiCert) and poll-based completion. The connector periodically checks order status via `/order/certificate/{order_id}` until the certificate is available.
|
||||||
|
|
||||||
|
**Authentication:** API key passed via `X-DC-DEVKEY` header, with organization ID in request body.
|
||||||
|
|
||||||
|
**Note:** CRL and OCSP are managed by DigiCert. Clients should validate certificate status against DigiCert's infrastructure. certctl records the revocation locally but does not notify DigiCert for revocation — use DigiCert's dashboard for revocation management.
|
||||||
|
|
||||||
|
Location: `internal/connector/issuer/digicert/digicert.go`
|
||||||
|
|
||||||
|
### Coming in V2.2+
|
||||||
|
|
||||||
|
The following issuer connectors are planned for future releases:
|
||||||
|
|
||||||
|
- **Entrust** — Enterprise CA via Entrust API
|
||||||
|
- **Sectigo** — Commercial CA integration via Sectigo REST API
|
||||||
|
- **Google CAS** — Google Cloud Certificate Authority Service
|
||||||
|
- **AWS ACM Private CA** — AWS-managed private CA
|
||||||
|
|
||||||
Note: ADCS (Active Directory Certificate Services) integration is handled via the **sub-CA mode** of the Local CA issuer, not as a separate connector. certctl operates as a subordinate CA with its signing certificate issued by ADCS, so all certctl-issued certs chain to the enterprise ADCS root. See the Local CA section above.
|
Note: ADCS (Active Directory Certificate Services) integration is handled via the **sub-CA mode** of the Local CA issuer, not as a separate connector. certctl operates as a subordinate CA with its signing certificate issued by ADCS, so all certctl-issued certs chain to the enterprise ADCS root. See the Local CA section above.
|
||||||
|
|
||||||
|
|||||||
+2
-2
@@ -1469,8 +1469,8 @@ Each guide includes an evidence summary table mapping specific criteria to certc
|
|||||||
| **Bulk revocation** | ✗ | ✓ | Planned V3 (paid) |
|
| **Bulk revocation** | ✗ | ✓ | Planned V3 (paid) |
|
||||||
| **Certificate health scores** | ✗ | ✓ | Planned V3 |
|
| **Certificate health scores** | ✗ | ✓ | Planned V3 |
|
||||||
| **Compliance scoring** | ✗ | ✓ | Planned V3 |
|
| **Compliance scoring** | ✗ | ✓ | Planned V3 |
|
||||||
| **DigiCert issuer** | ✗ | ✓ | Planned V2.1 (free) |
|
| **DigiCert issuer** | ✗ | ✓ | Implemented (Beta) |
|
||||||
| **Vault PKI issuer** | ✗ | ✓ | Planned V2.1 (free) |
|
| **Vault PKI issuer** | ✗ | ✓ | Implemented (Beta) |
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
+471
-3
@@ -42,6 +42,10 @@ Comprehensive manual testing playbook. Every test has a concrete command, an exp
|
|||||||
- [Part 35: ARI (RFC 9702) Scheduler Integration](#part-35-ari-rfc-9702-scheduler-integration)
|
- [Part 35: ARI (RFC 9702) Scheduler Integration](#part-35-ari-rfc-9702-scheduler-integration)
|
||||||
- [Part 36: Agent Work Routing (M31)](#part-36-agent-work-routing-m31)
|
- [Part 36: Agent Work Routing (M31)](#part-36-agent-work-routing-m31)
|
||||||
- [Part 37: GUI Completeness (Pre-2.1.0-E)](#part-37-gui-completeness-pre-210-e)
|
- [Part 37: GUI Completeness (Pre-2.1.0-E)](#part-37-gui-completeness-pre-210-e)
|
||||||
|
- [Part 38: Vault PKI Connector (M32)](#part-38-vault-pki-connector-m32)
|
||||||
|
- [Part 39: DigiCert Connector (M37)](#part-39-digicert-connector-m37)
|
||||||
|
- [Part 40: Issuer Catalog Page (M33)](#part-40-issuer-catalog-page-m33)
|
||||||
|
- [Part 41: Frontend Audit Fixes](#part-41-frontend-audit-fixes)
|
||||||
- [Release Sign-Off](#release-sign-off)
|
- [Release Sign-Off](#release-sign-off)
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -5166,6 +5170,393 @@ curl -s -H "Authorization: Bearer $API_KEY" \
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## Part 38: Vault PKI Connector (M32)
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
|
||||||
|
- Vault server running with PKI secrets engine enabled at `pki` mount
|
||||||
|
- PKI role created with appropriate certificate generation policy
|
||||||
|
- Vault token with read/sign permissions on the PKI path
|
||||||
|
- Environment variables configured:
|
||||||
|
```bash
|
||||||
|
export CERTCTL_VAULT_ADDR="https://vault.internal:8200"
|
||||||
|
export CERTCTL_VAULT_TOKEN="s.xxxxxxxxxxxxxxxx"
|
||||||
|
export CERTCTL_VAULT_MOUNT="pki"
|
||||||
|
export CERTCTL_VAULT_ROLE="certctl-role"
|
||||||
|
export CERTCTL_VAULT_TTL="8760h"
|
||||||
|
```
|
||||||
|
|
||||||
|
### 38.1 Register Vault PKI Issuer
|
||||||
|
|
||||||
|
**Test:** Register a Vault PKI issuer via the API.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -X POST -H "$AUTH" -H "$CT" \
|
||||||
|
"$SERVER/api/v1/issuers" \
|
||||||
|
-d '{
|
||||||
|
"id": "iss-vault-prod",
|
||||||
|
"name": "Vault PKI Production",
|
||||||
|
"type": "VaultPKI",
|
||||||
|
"config": {
|
||||||
|
"vault_addr": "'"$CERTCTL_VAULT_ADDR"'",
|
||||||
|
"vault_token": "'"$CERTCTL_VAULT_TOKEN"'",
|
||||||
|
"vault_mount": "'"$CERTCTL_VAULT_MOUNT"'",
|
||||||
|
"vault_role": "'"$CERTCTL_VAULT_ROLE"'",
|
||||||
|
"vault_ttl": "'"$CERTCTL_VAULT_TTL"'"
|
||||||
|
}
|
||||||
|
}' | jq '.id'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Returns issuer ID `iss-vault-prod`.
|
||||||
|
**PASS if** issuer is registered and appears in `GET /api/v1/issuers`.
|
||||||
|
|
||||||
|
### 38.2 Issue Certificate via Vault PKI
|
||||||
|
|
||||||
|
**Test:** Create a certificate and issue it through Vault PKI.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
CERT_ID=$(curl -s -X POST -H "$AUTH" -H "$CT" \
|
||||||
|
"$SERVER/api/v1/certificates" \
|
||||||
|
-d '{
|
||||||
|
"common_name": "vault-test.example.com",
|
||||||
|
"issuer_id": "iss-vault-prod",
|
||||||
|
"key_algorithm": "RSA-2048"
|
||||||
|
}' | jq -r '.id')
|
||||||
|
|
||||||
|
curl -s -X POST -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID/renew" | jq '.job_id'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Renewal job created and eventually moves to Completed status.
|
||||||
|
**PASS if** certificate is issued by Vault with valid serial number and chain.
|
||||||
|
|
||||||
|
### 38.3 Verify Certificate Serial and Subject
|
||||||
|
|
||||||
|
**Test:** Check that the issued certificate has correct Vault metadata.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID" | jq '.versions[0] | {serial, subject_dn, not_before, not_after}'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Serial, DN, and validity dates from Vault PKI.
|
||||||
|
**PASS if** certificate metadata is populated from Vault's response.
|
||||||
|
|
||||||
|
### 38.4 Revocation Records Locally
|
||||||
|
|
||||||
|
**Test:** Revoke the certificate and verify local recording.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s -X POST -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID/revoke" \
|
||||||
|
-d '{"reason": "superseded"}' | jq '.revoked_at'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Returns `revoked_at` timestamp.
|
||||||
|
**PASS if** revocation is recorded locally in the audit trail but not propagated to Vault (Vault is authoritative for its own revocation).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Part 39: DigiCert Connector (M37)
|
||||||
|
|
||||||
|
### Prerequisites
|
||||||
|
|
||||||
|
- DigiCert CertCentral account with API access
|
||||||
|
- API key and organization ID from DigiCert
|
||||||
|
- Environment variables configured:
|
||||||
|
```bash
|
||||||
|
export CERTCTL_DIGICERT_API_KEY="xxxxxxxxxxxxxxxxxxxxxxxx"
|
||||||
|
export CERTCTL_DIGICERT_ORG_ID="123456"
|
||||||
|
export CERTCTL_DIGICERT_PRODUCT_TYPE="ssl_basic"
|
||||||
|
export CERTCTL_DIGICERT_BASE_URL="https://www.digicert.com/services/v2"
|
||||||
|
```
|
||||||
|
|
||||||
|
### 39.1 Register DigiCert Issuer
|
||||||
|
|
||||||
|
**Test:** Register a DigiCert CertCentral issuer via the API.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -X POST -H "$AUTH" -H "$CT" \
|
||||||
|
"$SERVER/api/v1/issuers" \
|
||||||
|
-d '{
|
||||||
|
"id": "iss-digicert-prod",
|
||||||
|
"name": "DigiCert CertCentral",
|
||||||
|
"type": "DigiCert",
|
||||||
|
"config": {
|
||||||
|
"api_key": "'"$CERTCTL_DIGICERT_API_KEY"'",
|
||||||
|
"org_id": "'"$CERTCTL_DIGICERT_ORG_ID"'",
|
||||||
|
"product_type": "'"$CERTCTL_DIGICERT_PRODUCT_TYPE"'",
|
||||||
|
"base_url": "'"$CERTCTL_DIGICERT_BASE_URL"'"
|
||||||
|
}
|
||||||
|
}' | jq '.id'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Returns issuer ID `iss-digicert-prod`.
|
||||||
|
**PASS if** issuer is registered and appears in `GET /api/v1/issuers`.
|
||||||
|
|
||||||
|
### 39.2 Issue DV Certificate via DigiCert
|
||||||
|
|
||||||
|
**Test:** Create a DV certificate order and track it to completion.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
CERT_ID=$(curl -s -X POST -H "$AUTH" -H "$CT" \
|
||||||
|
"$SERVER/api/v1/certificates" \
|
||||||
|
-d '{
|
||||||
|
"common_name": "dv-test.example.com",
|
||||||
|
"issuer_id": "iss-digicert-prod",
|
||||||
|
"key_algorithm": "RSA-2048"
|
||||||
|
}' | jq -r '.id')
|
||||||
|
|
||||||
|
JOB_ID=$(curl -s -X POST -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID/renew" | jq -r '.job_id')
|
||||||
|
|
||||||
|
# Poll for job completion (DV certs may issue immediately)
|
||||||
|
for i in {1..30}; do
|
||||||
|
STATUS=$(curl -s -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/jobs/$JOB_ID" | jq -r '.status')
|
||||||
|
echo "Job status: $STATUS"
|
||||||
|
[ "$STATUS" = "Completed" ] && break
|
||||||
|
sleep 2
|
||||||
|
done
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Job eventually reaches Completed status with certificate issued.
|
||||||
|
**PASS if** certificate has DigiCert serial number and chain.
|
||||||
|
|
||||||
|
### 39.3 Verify Order ID Tracking
|
||||||
|
|
||||||
|
**Test:** Check that the job record includes the DigiCert order ID for auditing.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/jobs/$JOB_ID" | jq '.metadata'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Metadata includes `order_id` from DigiCert for order tracking.
|
||||||
|
**PASS if** audit trail shows the DigiCert order lifecycle.
|
||||||
|
|
||||||
|
### 39.4 Async Poll Behavior
|
||||||
|
|
||||||
|
**Test:** Verify the connector polls for certificate completion (OV certs take longer).
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Submit OV certificate order (requires validation)
|
||||||
|
CERT_ID=$(curl -s -X POST -H "$AUTH" -H "$CT" \
|
||||||
|
"$SERVER/api/v1/certificates" \
|
||||||
|
-d '{
|
||||||
|
"common_name": "ov-test.example.com",
|
||||||
|
"issuer_id": "iss-digicert-prod",
|
||||||
|
"key_algorithm": "RSA-2048"
|
||||||
|
}' | jq -r '.id')
|
||||||
|
|
||||||
|
JOB_ID=$(curl -s -X POST -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID/renew" | jq -r '.job_id')
|
||||||
|
|
||||||
|
# Check job status transitions
|
||||||
|
curl -s -H "$AUTH" "$SERVER/api/v1/jobs/$JOB_ID" | jq '.status'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Job status transitions through pending states as DigiCert validates.
|
||||||
|
**PASS if** polling mechanism works and job reaches completion once DigiCert issues the certificate.
|
||||||
|
|
||||||
|
### 39.5 Revocation Records Locally
|
||||||
|
|
||||||
|
**Test:** Revoke a DigiCert-issued certificate.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -s -X POST -H "$AUTH" \
|
||||||
|
"$SERVER/api/v1/certificates/$CERT_ID/revoke" \
|
||||||
|
-d '{"reason": "cessationOfOperation"}' | jq '.revoked_at'
|
||||||
|
```
|
||||||
|
|
||||||
|
**Expected:** Returns `revoked_at` timestamp.
|
||||||
|
**PASS if** revocation is recorded locally; operator manages revocation in DigiCert CertCentral dashboard.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Part 40: Issuer Catalog Page (M33)
|
||||||
|
|
||||||
|
Frontend-only milestone. No backend changes. All tests are automated via `qa-smoke-test.sh` and `vitest`.
|
||||||
|
|
||||||
|
### 40.1 Shared Issuer Type Config
|
||||||
|
|
||||||
|
**Test:** Verify shared config file exists with all 6 supported types + 2 coming soon stubs.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
test -f web/src/config/issuerTypes.ts
|
||||||
|
grep -c 'VaultPKI' web/src/config/issuerTypes.ts # >= 1
|
||||||
|
grep -c 'DigiCert' web/src/config/issuerTypes.ts # >= 1
|
||||||
|
grep -cE 'eab_kid|eab_hmac' web/src/config/issuerTypes.ts # >= 1
|
||||||
|
grep -c 'sensitive' web/src/config/issuerTypes.ts # >= 1
|
||||||
|
```
|
||||||
|
|
||||||
|
**PASS if** file exists, all types present, EAB fields and sensitive flags included.
|
||||||
|
|
||||||
|
### 40.2 Composable Wizard Components
|
||||||
|
|
||||||
|
**Test:** Verify reusable components exist.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
test -f web/src/components/issuer/TypeSelector.tsx
|
||||||
|
test -f web/src/components/issuer/ConfigForm.tsx
|
||||||
|
test -f web/src/components/issuer/ConfigDetailModal.tsx
|
||||||
|
```
|
||||||
|
|
||||||
|
**PASS if** all 3 component files exist.
|
||||||
|
|
||||||
|
### 40.3 Frontend Build
|
||||||
|
|
||||||
|
**Test:** Verify frontend builds with zero errors.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd web && npm run build 2>&1 | tail -1 | grep -q 'built in'
|
||||||
|
```
|
||||||
|
|
||||||
|
**PASS if** build succeeds.
|
||||||
|
|
||||||
|
### 40.4 Frontend Tests
|
||||||
|
|
||||||
|
**Test:** Verify all Vitest tests pass including new VaultPKI/DigiCert create tests.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd web && npx vitest run 2>&1 | grep -qE 'Tests.*passed'
|
||||||
|
```
|
||||||
|
|
||||||
|
**PASS if** all tests pass.
|
||||||
|
|
||||||
|
### 40.5 (Manual) Create VaultPKI Issuer via Wizard
|
||||||
|
|
||||||
|
**Test:** Open Issuers page, click "Configure" on Vault PKI card, fill in form (addr, token, mount, role, ttl), submit.
|
||||||
|
**PASS if** issuer appears in configured issuers table.
|
||||||
|
|
||||||
|
### 40.6 (Manual) Create DigiCert Issuer via Wizard
|
||||||
|
|
||||||
|
**Test:** Open Issuers page, click "Configure" on DigiCert card, fill in form (api_key, org_id, product_type), submit.
|
||||||
|
**PASS if** issuer appears in configured issuers table.
|
||||||
|
|
||||||
|
### 40.7 (Manual) Create ACME Issuer with EAB Fields
|
||||||
|
|
||||||
|
**Test:** Open create wizard, select ACME, verify EAB Key ID and EAB HMAC Key fields are visible.
|
||||||
|
**PASS if** EAB fields render and accept input.
|
||||||
|
|
||||||
|
### 40.8 (Manual) Catalog Cards Show Correct Status
|
||||||
|
|
||||||
|
**Test:** Verify catalog cards show "Connected" (green, count) for types with configured issuers, "Available" (blue) for unconfigured types, and "Coming Soon" (grey) for Sectigo/Entrust.
|
||||||
|
**PASS if** all 8 cards render with correct status.
|
||||||
|
|
||||||
|
### 40.9 (Manual) Config Detail Modal Shows Full Redacted Config
|
||||||
|
|
||||||
|
**Test:** Click "View Config" on a configured issuer row. Verify modal shows full config JSON with sensitive fields (token, key, hmac, password, private, secret) redacted as `********`.
|
||||||
|
**PASS if** modal opens, full config visible, sensitive fields redacted.
|
||||||
|
|
||||||
|
### 40.10 (Manual) Issuer Type Filter Works
|
||||||
|
|
||||||
|
**Test:** Use the type filter dropdown above the configured issuers table. Select a specific type.
|
||||||
|
**PASS if** table filters to show only issuers of the selected type.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Part 41: Frontend Audit Fixes
|
||||||
|
|
||||||
|
Comprehensive frontend coverage audit closed 60 gaps between backend capabilities and GUI surfaces. This part validates the critical fixes.
|
||||||
|
|
||||||
|
### Automated Tests (qa-smoke-test.sh Part 41)
|
||||||
|
|
||||||
|
| # | Test | Assertion |
|
||||||
|
|---|------|-----------|
|
||||||
|
| 41.1 | Certificate TS type has lifecycle fields | `types.ts` contains `last_renewal_at`, `last_deployment_at`, `target_ids` |
|
||||||
|
| 41.2 | API client has new endpoint functions | `client.ts` exports `updateIssuer`, `updateTarget`, `getCertificateDeployments`, `getCRL`, `getOCSPStatus`, `getPolicy` |
|
||||||
|
| 41.3 | CertificatesPage has filter dropdowns | Contains `issuerFilter`, `ownerFilter`, `profileFilter` state vars |
|
||||||
|
| 41.4 | CertificatesPage shows last_renewal_at | Column renders `last_renewal_at` field |
|
||||||
|
| 41.5 | JobsPage shows error_message | Error column displays first 80 chars for failed jobs |
|
||||||
|
| 41.6 | ProfilesPage has key algorithm fields | Create form includes `allowed_key_algorithms` with add/remove rows |
|
||||||
|
| 41.7 | ProfilesPage has EKU checkboxes | Create form includes `allowed_ekus` checkbox group |
|
||||||
|
| 41.8 | DiscoveryPage shows is_ca badge | CA badge renders for discovered CA certificates |
|
||||||
|
| 41.9 | TargetDetailPage has Edit functionality | Edit button wired to `updateTarget` API call |
|
||||||
|
| 41.10 | CertificatesPage has tags field | Create form includes tags input (key=value pairs) |
|
||||||
|
| 41.11 | AgentFleetPage maps darwin to macOS | OS display mapping applied to pie chart and platform headers |
|
||||||
|
| 41.12 | Frontend builds after audit fixes | `npm run build` succeeds |
|
||||||
|
|
||||||
|
### Manual Tests
|
||||||
|
|
||||||
|
**41.M1: Profile Create Form — Key Algorithm Configuration**
|
||||||
|
|
||||||
|
1. Navigate to Profiles page, click "+ New Profile"
|
||||||
|
2. Verify default algorithms shown: ECDSA 256+, RSA 2048+
|
||||||
|
3. Click "Remove" on RSA row — verify it disappears
|
||||||
|
4. Click "+ Add" — verify Ed25519 appears (with "fixed" instead of size dropdown)
|
||||||
|
5. Submit form, verify profile created with correct `allowed_key_algorithms` array
|
||||||
|
|
||||||
|
**PASS if** algorithms are configurable and persisted correctly.
|
||||||
|
|
||||||
|
**41.M2: Profile Create Form — EKU Selection**
|
||||||
|
|
||||||
|
1. In Create Profile modal, verify EKU checkboxes visible (serverAuth checked by default)
|
||||||
|
2. Check "Email Protection (S/MIME)" and "Client Authentication"
|
||||||
|
3. Submit, verify profile has `allowed_ekus: ["serverAuth", "emailProtection", "clientAuth"]`
|
||||||
|
|
||||||
|
**PASS if** EKUs are selectable and sent to backend.
|
||||||
|
|
||||||
|
**41.M3: Certificate Create Form — Tags**
|
||||||
|
|
||||||
|
1. Navigate to Certificates page, click "+ New Certificate"
|
||||||
|
2. Enter tags: `env=prod, team=platform, app=api`
|
||||||
|
3. Submit, verify certificate created with `tags: {"env": "prod", "team": "platform", "app": "api"}`
|
||||||
|
|
||||||
|
**PASS if** tags are parsed and persisted as key-value pairs.
|
||||||
|
|
||||||
|
**41.M4: Jobs Table — Error Message Column**
|
||||||
|
|
||||||
|
1. Navigate to Jobs page, filter to "Failed" status
|
||||||
|
2. Verify "Error" column shows truncated error message (max 80 chars with "...")
|
||||||
|
3. Hover over truncated message, verify full text in tooltip
|
||||||
|
|
||||||
|
**PASS if** error messages visible for failed jobs.
|
||||||
|
|
||||||
|
**41.M5: Certificates Table — Lifecycle Columns**
|
||||||
|
|
||||||
|
1. Navigate to Certificates page
|
||||||
|
2. Verify "Last Renewal" and "Last Deploy" columns visible
|
||||||
|
3. Verify dates shown for certs with data, "—" for certs without
|
||||||
|
|
||||||
|
**PASS if** lifecycle timestamps displayed.
|
||||||
|
|
||||||
|
**41.M6: Certificate Filters — Issuer/Owner/Profile Dropdowns**
|
||||||
|
|
||||||
|
1. Navigate to Certificates page
|
||||||
|
2. Verify Issuer, Owner, Profile dropdown filters visible
|
||||||
|
3. Select an issuer — verify table filters to matching certificates
|
||||||
|
4. Clear filter, select a profile — verify filtering works
|
||||||
|
|
||||||
|
**PASS if** all three filter dropdowns functional.
|
||||||
|
|
||||||
|
**41.M7: Target Detail — Edit Button**
|
||||||
|
|
||||||
|
1. Navigate to a target detail page
|
||||||
|
2. Click "Edit" button
|
||||||
|
3. Modify name, click "Save"
|
||||||
|
4. Verify name updated on the page
|
||||||
|
|
||||||
|
**PASS if** target edit persists via API.
|
||||||
|
|
||||||
|
**41.M8: Discovery Table — CA Badge**
|
||||||
|
|
||||||
|
1. Navigate to Discovery page
|
||||||
|
2. Verify "Key" column shows algorithm + key size
|
||||||
|
3. For CA certificates, verify purple "CA" badge displayed
|
||||||
|
|
||||||
|
**PASS if** CA certificates visually distinguished.
|
||||||
|
|
||||||
|
**41.M9: Fleet Overview — macOS Display**
|
||||||
|
|
||||||
|
1. Navigate to Fleet Overview page
|
||||||
|
2. Verify OS pie chart shows "macOS" instead of "darwin"
|
||||||
|
3. Verify platform section headers show "macOS / amd64" (not "darwin / amd64")
|
||||||
|
|
||||||
|
**PASS if** darwin correctly mapped to macOS in all locations.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
## Release Sign-Off
|
## Release Sign-Off
|
||||||
|
|
||||||
All tests below must pass before tagging v2.1.0. Each row is one individual test from the guide above. The **Method** column indicates whether `qa-smoke-test.sh` covers the test automatically (**Auto**) or requires hands-on verification (**Manual**).
|
All tests below must pass before tagging v2.1.0. Each row is one individual test from the guide above. The **Method** column indicates whether `qa-smoke-test.sh` covers the test automatically (**Auto**) or requires hands-on verification (**Manual**).
|
||||||
@@ -5715,14 +6106,91 @@ These must be green before starting manual QA:
|
|||||||
| 37.16 | TargetsPage — target names clickable to /targets/:id | Manual | ☐ | | |
|
| 37.16 | TargetsPage — target names clickable to /targets/:id | Manual | ☐ | | |
|
||||||
| 37.17 | Sidebar — Digest and Observability nav items | Manual | ☐ | | |
|
| 37.17 | Sidebar — Digest and Observability nav items | Manual | ☐ | | |
|
||||||
|
|
||||||
|
### Part 38: Vault PKI Connector (M32)
|
||||||
|
|
||||||
|
| Test | Description | Method | Pass? | Date | Notes |
|
||||||
|
|------|-------------|--------|-------|------|-------|
|
||||||
|
| 38.s1 | Vault PKI issuer exists in seed data | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.1 |
|
||||||
|
| 38.s2 | Vault issuer type is VaultPKI | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.2 |
|
||||||
|
| 38.s3 | Vault issuer is enabled | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.3 |
|
||||||
|
| 38.s4 | Vault connector passes go vet | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.4 |
|
||||||
|
| 38.s5 | Vault connector tests pass | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.5 |
|
||||||
|
| 38.s6 | OpenAPI spec includes VaultPKI type | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 38.6 |
|
||||||
|
| 38.1 | Register Vault PKI issuer | Manual | ☐ | | Requires live Vault server |
|
||||||
|
| 38.2 | Issue certificate via Vault PKI | Manual | ☐ | | Requires live Vault server |
|
||||||
|
| 38.3 | Verify certificate serial and subject | Manual | ☐ | | Requires live Vault server |
|
||||||
|
| 38.4 | Revocation records locally | Manual | ☐ | | Requires live Vault server |
|
||||||
|
|
||||||
|
### Part 39: DigiCert Connector (M37)
|
||||||
|
|
||||||
|
| Test | Description | Method | Pass? | Date | Notes |
|
||||||
|
|------|-------------|--------|-------|------|-------|
|
||||||
|
| 39.s1 | DigiCert issuer exists in seed data | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.1 |
|
||||||
|
| 39.s2 | DigiCert issuer type is DigiCert | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.2 |
|
||||||
|
| 39.s3 | DigiCert issuer is enabled | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.3 |
|
||||||
|
| 39.s4 | DigiCert connector passes go vet | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.4 |
|
||||||
|
| 39.s5 | DigiCert connector tests pass | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.5 |
|
||||||
|
| 39.s6 | OpenAPI spec includes DigiCert type | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 39.6 |
|
||||||
|
| 39.1 | Register DigiCert issuer | Manual | ☐ | | Requires DigiCert sandbox |
|
||||||
|
| 39.2 | Issue DV certificate via DigiCert | Manual | ☐ | | Requires DigiCert sandbox |
|
||||||
|
| 39.3 | Verify order ID tracking | Manual | ☐ | | Requires DigiCert sandbox |
|
||||||
|
| 39.4 | Async poll behavior | Manual | ☐ | | Requires DigiCert sandbox |
|
||||||
|
| 39.5 | Revocation records locally | Manual | ☐ | | Requires DigiCert sandbox |
|
||||||
|
|
||||||
|
### Part 40: Issuer Catalog Page (M33)
|
||||||
|
|
||||||
|
| Test | Description | Method | Pass? | Date | Notes |
|
||||||
|
|------|-------------|--------|-------|------|-------|
|
||||||
|
| 40.s1 | Shared issuerTypes config exists | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.1 |
|
||||||
|
| 40.s2 | VaultPKI in issuerTypes config | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.2 |
|
||||||
|
| 40.s3 | DigiCert in issuerTypes config | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.3 |
|
||||||
|
| 40.s4 | ACME EAB fields in config | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.4 |
|
||||||
|
| 40.s5 | Sensitive field flag in config | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.5 |
|
||||||
|
| 40.s6 | ConfigDetailModal component exists | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.6 |
|
||||||
|
| 40.s7 | Frontend build succeeds | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.7 |
|
||||||
|
| 40.s8 | Frontend tests pass | Auto | ☑ | 2026-03-30 | qa-smoke-test.sh 40.8 |
|
||||||
|
| 40.m1 | Create VaultPKI issuer via wizard | Manual | ☐ | | |
|
||||||
|
| 40.m2 | Create DigiCert issuer via wizard | Manual | ☐ | | |
|
||||||
|
| 40.m3 | Create ACME issuer with EAB fields | Manual | ☐ | | |
|
||||||
|
| 40.m4 | Catalog cards show correct status | Manual | ☐ | | |
|
||||||
|
| 40.m5 | Config detail modal shows full redacted config | Manual | ☐ | | |
|
||||||
|
| 40.m6 | Issuer type filter works | Manual | ☐ | | |
|
||||||
|
|
||||||
|
### Part 41: Frontend Audit Fixes
|
||||||
|
|
||||||
|
| Test | Description | Method | Pass? | Date | Notes |
|
||||||
|
|------|-------------|--------|-------|------|-------|
|
||||||
|
| 41.s1 | Certificate TS type has lifecycle fields | Auto | ☐ | | qa-smoke-test.sh 41.1 |
|
||||||
|
| 41.s2 | API client has new endpoint functions | Auto | ☐ | | qa-smoke-test.sh 41.2 |
|
||||||
|
| 41.s3 | CertificatesPage has filter dropdowns | Auto | ☐ | | qa-smoke-test.sh 41.3 |
|
||||||
|
| 41.s4 | CertificatesPage shows last_renewal_at | Auto | ☐ | | qa-smoke-test.sh 41.4 |
|
||||||
|
| 41.s5 | JobsPage shows error_message | Auto | ☐ | | qa-smoke-test.sh 41.5 |
|
||||||
|
| 41.s6 | ProfilesPage has key algorithm fields | Auto | ☐ | | qa-smoke-test.sh 41.6 |
|
||||||
|
| 41.s7 | ProfilesPage has EKU checkboxes | Auto | ☐ | | qa-smoke-test.sh 41.7 |
|
||||||
|
| 41.s8 | DiscoveryPage shows is_ca badge | Auto | ☐ | | qa-smoke-test.sh 41.8 |
|
||||||
|
| 41.s9 | TargetDetailPage has Edit functionality | Auto | ☐ | | qa-smoke-test.sh 41.9 |
|
||||||
|
| 41.s10 | CertificatesPage has tags field | Auto | ☐ | | qa-smoke-test.sh 41.10 |
|
||||||
|
| 41.s11 | AgentFleetPage maps darwin to macOS | Auto | ☐ | | qa-smoke-test.sh 41.11 |
|
||||||
|
| 41.s12 | Frontend builds after audit fixes | Auto | ☐ | | qa-smoke-test.sh 41.12 |
|
||||||
|
| 41.m1 | Profile create form — key algorithm config | Manual | ☐ | | |
|
||||||
|
| 41.m2 | Profile create form — EKU selection | Manual | ☐ | | |
|
||||||
|
| 41.m3 | Certificate create form — tags | Manual | ☐ | | |
|
||||||
|
| 41.m4 | Jobs table — error message column | Manual | ☐ | | |
|
||||||
|
| 41.m5 | Certificates table — lifecycle columns | Manual | ☐ | | |
|
||||||
|
| 41.m6 | Certificate filters — issuer/owner/profile | Manual | ☐ | | |
|
||||||
|
| 41.m7 | Target detail — edit button | Manual | ☐ | | |
|
||||||
|
| 41.m8 | Discovery table — CA badge | Manual | ☐ | | |
|
||||||
|
| 41.m9 | Fleet overview — macOS display | Manual | ☐ | | |
|
||||||
|
|
||||||
### Summary
|
### Summary
|
||||||
|
|
||||||
| Category | Count |
|
| Category | Count |
|
||||||
|----------|-------|
|
|----------|-------|
|
||||||
| ☑ Auto (passed in `qa-smoke-test.sh`) | 127 |
|
| ☑ Auto (passed in `qa-smoke-test.sh`) | 144 |
|
||||||
|
| ☐ Auto (not yet run) | 12 |
|
||||||
| — Skipped (preconditions not met in demo) | 5 |
|
| — Skipped (preconditions not met in demo) | 5 |
|
||||||
| ☐ Manual (requires hands-on verification) | 217 |
|
| ☐ Manual (requires hands-on verification) | 241 |
|
||||||
| **Total** | **349** |
|
| **Total** | **402** |
|
||||||
|
|
||||||
**Automated tests must also be green.** CI passing is necessary but not sufficient — this manual QA catches integration issues that isolated unit tests miss.
|
**Automated tests must also be green.** CI passing is necessary but not sufficient — this manual QA catches integration issues that isolated unit tests miss.
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,187 @@
|
|||||||
|
-- =============================================================================
|
||||||
|
-- Comprehensive Referential Integrity Check for seed_demo.sql
|
||||||
|
-- Run AFTER migrations and seed data are loaded
|
||||||
|
-- =============================================================================
|
||||||
|
|
||||||
|
-- 1. Verify certificate_versions.certificate_id references valid managed_certificates.id
|
||||||
|
SELECT 'FK VIOLATION: certificate_versions.certificate_id' AS issue, cv.id, cv.certificate_id
|
||||||
|
FROM certificate_versions cv
|
||||||
|
WHERE cv.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY cv.id;
|
||||||
|
|
||||||
|
-- 2. Verify certificate_target_mappings references valid IDs
|
||||||
|
SELECT 'FK VIOLATION: certificate_target_mappings.certificate_id' AS issue, ctm.certificate_id
|
||||||
|
FROM certificate_target_mappings ctm
|
||||||
|
WHERE ctm.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY ctm.certificate_id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: certificate_target_mappings.target_id' AS issue, ctm.target_id
|
||||||
|
FROM certificate_target_mappings ctm
|
||||||
|
WHERE ctm.target_id NOT IN (SELECT id FROM deployment_targets)
|
||||||
|
ORDER BY ctm.target_id;
|
||||||
|
|
||||||
|
-- 3. Verify jobs references valid IDs
|
||||||
|
SELECT 'FK VIOLATION: jobs.certificate_id' AS issue, j.id, j.certificate_id
|
||||||
|
FROM jobs j
|
||||||
|
WHERE j.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY j.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: jobs.target_id' AS issue, j.id, j.target_id
|
||||||
|
FROM jobs j
|
||||||
|
WHERE j.target_id IS NOT NULL AND j.target_id NOT IN (SELECT id FROM deployment_targets)
|
||||||
|
ORDER BY j.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: jobs.agent_id' AS issue, j.id, j.agent_id
|
||||||
|
FROM jobs j
|
||||||
|
WHERE j.agent_id NOT IN (SELECT id FROM agents)
|
||||||
|
ORDER BY j.id;
|
||||||
|
|
||||||
|
-- 4. Verify discovered_certificates references valid IDs
|
||||||
|
SELECT 'FK VIOLATION: discovered_certificates.agent_id' AS issue, dc.id, dc.agent_id
|
||||||
|
FROM discovered_certificates dc
|
||||||
|
WHERE dc.agent_id NOT IN (SELECT id FROM agents)
|
||||||
|
ORDER BY dc.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: discovered_certificates.discovery_scan_id' AS issue, dc.id, dc.discovery_scan_id
|
||||||
|
FROM discovered_certificates dc
|
||||||
|
WHERE dc.discovery_scan_id IS NOT NULL AND dc.discovery_scan_id NOT IN (SELECT id FROM discovery_scans)
|
||||||
|
ORDER BY dc.id;
|
||||||
|
|
||||||
|
-- 5. Verify notification_events references valid certificate_id
|
||||||
|
SELECT 'FK VIOLATION: notification_events.certificate_id' AS issue, ne.id, ne.certificate_id
|
||||||
|
FROM notification_events ne
|
||||||
|
WHERE ne.certificate_id IS NOT NULL AND ne.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY ne.id;
|
||||||
|
|
||||||
|
-- 6. Verify policy_violations references valid certificate_id
|
||||||
|
SELECT 'FK VIOLATION: policy_violations.certificate_id' AS issue, pv.id, pv.certificate_id
|
||||||
|
FROM policy_violations pv
|
||||||
|
WHERE pv.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY pv.id;
|
||||||
|
|
||||||
|
-- 7. Verify certificate_revocations references valid IDs
|
||||||
|
SELECT 'FK VIOLATION: certificate_revocations.certificate_id' AS issue, cr.id, cr.certificate_id
|
||||||
|
FROM certificate_revocations cr
|
||||||
|
WHERE cr.certificate_id NOT IN (SELECT id FROM managed_certificates)
|
||||||
|
ORDER BY cr.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: certificate_revocations.issuer_id' AS issue, cr.id, cr.issuer_id
|
||||||
|
FROM certificate_revocations cr
|
||||||
|
WHERE cr.issuer_id NOT IN (SELECT id FROM issuers)
|
||||||
|
ORDER BY cr.id;
|
||||||
|
|
||||||
|
-- 8. Verify agent_group_members references valid IDs
|
||||||
|
SELECT 'FK VIOLATION: agent_group_members.agent_group_id' AS issue, agm.agent_group_id
|
||||||
|
FROM agent_group_members agm
|
||||||
|
WHERE agm.agent_group_id NOT IN (SELECT id FROM agent_groups)
|
||||||
|
ORDER BY agm.agent_group_id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: agent_group_members.agent_id' AS issue, agm.agent_id
|
||||||
|
FROM agent_group_members agm
|
||||||
|
WHERE agm.agent_id NOT IN (SELECT id FROM agents)
|
||||||
|
ORDER BY agm.agent_id;
|
||||||
|
|
||||||
|
-- 9. Verify owners.team_id references valid teams.id
|
||||||
|
SELECT 'FK VIOLATION: owners.team_id' AS issue, o.id, o.team_id
|
||||||
|
FROM owners o
|
||||||
|
WHERE o.team_id IS NOT NULL AND o.team_id NOT IN (SELECT id FROM teams)
|
||||||
|
ORDER BY o.id;
|
||||||
|
|
||||||
|
-- 10. Verify deployment_targets.agent_id references valid agents.id
|
||||||
|
SELECT 'FK VIOLATION: deployment_targets.agent_id' AS issue, dt.id, dt.agent_id
|
||||||
|
FROM deployment_targets dt
|
||||||
|
WHERE dt.agent_id NOT IN (SELECT id FROM agents)
|
||||||
|
ORDER BY dt.id;
|
||||||
|
|
||||||
|
-- 11. Verify managed_certificates FK columns
|
||||||
|
SELECT 'FK VIOLATION: managed_certificates.owner_id' AS issue, mc.id, mc.owner_id
|
||||||
|
FROM managed_certificates mc
|
||||||
|
WHERE mc.owner_id IS NOT NULL AND mc.owner_id NOT IN (SELECT id FROM owners)
|
||||||
|
ORDER BY mc.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: managed_certificates.team_id' AS issue, mc.id, mc.team_id
|
||||||
|
FROM managed_certificates mc
|
||||||
|
WHERE mc.team_id IS NOT NULL AND mc.team_id NOT IN (SELECT id FROM teams)
|
||||||
|
ORDER BY mc.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: managed_certificates.issuer_id' AS issue, mc.id, mc.issuer_id
|
||||||
|
FROM managed_certificates mc
|
||||||
|
WHERE mc.issuer_id NOT IN (SELECT id FROM issuers)
|
||||||
|
ORDER BY mc.id;
|
||||||
|
|
||||||
|
SELECT 'FK VIOLATION: managed_certificates.renewal_policy_id' AS issue, mc.id, mc.renewal_policy_id
|
||||||
|
FROM managed_certificates mc
|
||||||
|
WHERE mc.renewal_policy_id IS NOT NULL AND mc.renewal_policy_id NOT IN (SELECT id FROM renewal_policies)
|
||||||
|
ORDER BY mc.id;
|
||||||
|
|
||||||
|
-- 12. Check for duplicate primary keys
|
||||||
|
SELECT 'DUPLICATE PK: teams' AS issue, id, COUNT(*) as count
|
||||||
|
FROM teams GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: owners' AS issue, id, COUNT(*) as count
|
||||||
|
FROM owners GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: agents' AS issue, id, COUNT(*) as count
|
||||||
|
FROM agents GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: deployment_targets' AS issue, id, COUNT(*) as count
|
||||||
|
FROM deployment_targets GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: managed_certificates' AS issue, id, COUNT(*) as count
|
||||||
|
FROM managed_certificates GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: certificate_versions' AS issue, id, COUNT(*) as count
|
||||||
|
FROM certificate_versions GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: issuers' AS issue, id, COUNT(*) as count
|
||||||
|
FROM issuers GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: renewal_policies' AS issue, id, COUNT(*) as count
|
||||||
|
FROM renewal_policies GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: jobs' AS issue, id, COUNT(*) as count
|
||||||
|
FROM jobs GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: certificate_profiles' AS issue, id, COUNT(*) as count
|
||||||
|
FROM certificate_profiles GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
SELECT 'DUPLICATE PK: certificate_revocations' AS issue, id, COUNT(*) as count
|
||||||
|
FROM certificate_revocations GROUP BY id HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
-- 13. Check fingerprint_sha256 uniqueness in certificate_versions
|
||||||
|
SELECT 'DUPLICATE FINGERPRINT: certificate_versions' AS issue, fingerprint_sha256, COUNT(*) as count
|
||||||
|
FROM certificate_versions
|
||||||
|
WHERE fingerprint_sha256 IS NOT NULL
|
||||||
|
GROUP BY fingerprint_sha256
|
||||||
|
HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
-- 14. Check serial number uniqueness in certificate_versions
|
||||||
|
SELECT 'DUPLICATE SERIAL: certificate_versions' AS issue, serial_number, COUNT(*) as count
|
||||||
|
FROM certificate_versions
|
||||||
|
WHERE serial_number IS NOT NULL
|
||||||
|
GROUP BY serial_number
|
||||||
|
HAVING COUNT(*) > 1;
|
||||||
|
|
||||||
|
-- 15. Verify discovery_scan_id references are valid
|
||||||
|
SELECT 'FK VIOLATION: discovered_certificates.discovery_scan_id references' AS issue,
|
||||||
|
dc.id, dc.discovery_scan_id, ds.id
|
||||||
|
FROM discovered_certificates dc
|
||||||
|
LEFT JOIN discovery_scans ds ON dc.discovery_scan_id = ds.id
|
||||||
|
WHERE dc.discovery_scan_id IS NOT NULL AND ds.id IS NULL;
|
||||||
|
|
||||||
|
-- Summary: Count total records
|
||||||
|
SELECT 'SUMMARY: teams' AS table_name, COUNT(*) as count FROM teams UNION ALL
|
||||||
|
SELECT 'SUMMARY: owners', COUNT(*) FROM owners UNION ALL
|
||||||
|
SELECT 'SUMMARY: agents', COUNT(*) FROM agents UNION ALL
|
||||||
|
SELECT 'SUMMARY: deployment_targets', COUNT(*) FROM deployment_targets UNION ALL
|
||||||
|
SELECT 'SUMMARY: managed_certificates', COUNT(*) FROM managed_certificates UNION ALL
|
||||||
|
SELECT 'SUMMARY: certificate_versions', COUNT(*) FROM certificate_versions UNION ALL
|
||||||
|
SELECT 'SUMMARY: certificate_target_mappings', COUNT(*) FROM certificate_target_mappings UNION ALL
|
||||||
|
SELECT 'SUMMARY: issuers', COUNT(*) FROM issuers UNION ALL
|
||||||
|
SELECT 'SUMMARY: renewal_policies', COUNT(*) FROM renewal_policies UNION ALL
|
||||||
|
SELECT 'SUMMARY: jobs', COUNT(*) FROM jobs UNION ALL
|
||||||
|
SELECT 'SUMMARY: certificate_profiles', COUNT(*) FROM certificate_profiles UNION ALL
|
||||||
|
SELECT 'SUMMARY: certificate_revocations', COUNT(*) FROM certificate_revocations UNION ALL
|
||||||
|
SELECT 'SUMMARY: audit_events', COUNT(*) FROM audit_events UNION ALL
|
||||||
|
SELECT 'SUMMARY: discovery_scans', COUNT(*) FROM discovery_scans UNION ALL
|
||||||
|
SELECT 'SUMMARY: discovered_certificates', COUNT(*) FROM discovered_certificates;
|
||||||
@@ -243,6 +243,7 @@ func (h CertificateHandler) CreateCertificate(w http.ResponseWriter, r *http.Req
|
|||||||
|
|
||||||
created, err := h.svc.CreateCertificate(cert)
|
created, err := h.svc.CreateCertificate(cert)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
slog.Error("failed to create certificate", "error", err, "request_id", requestID, "common_name", cert.CommonName, "name", cert.Name)
|
||||||
ErrorWithRequestID(w, http.StatusInternalServerError, "Failed to create certificate", requestID)
|
ErrorWithRequestID(w, http.StatusInternalServerError, "Failed to create certificate", requestID)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,6 +25,8 @@ type Config struct {
|
|||||||
EST ESTConfig
|
EST ESTConfig
|
||||||
Verification VerificationConfig
|
Verification VerificationConfig
|
||||||
ACME ACMEConfig
|
ACME ACMEConfig
|
||||||
|
Vault VaultConfig
|
||||||
|
DigiCert DigiCertConfig
|
||||||
Digest DigestConfig
|
Digest DigestConfig
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -141,6 +143,57 @@ type StepCAConfig struct {
|
|||||||
ProvisionerPassword string
|
ProvisionerPassword string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// VaultConfig contains HashiCorp Vault PKI issuer connector configuration.
|
||||||
|
type VaultConfig struct {
|
||||||
|
// Addr is the Vault server address (e.g., "https://vault.example.com:8200").
|
||||||
|
// Required for Vault PKI integration.
|
||||||
|
// Setting: CERTCTL_VAULT_ADDR environment variable.
|
||||||
|
Addr string
|
||||||
|
|
||||||
|
// Token is the Vault token for authentication.
|
||||||
|
// Required for Vault PKI integration.
|
||||||
|
// Setting: CERTCTL_VAULT_TOKEN environment variable.
|
||||||
|
Token string
|
||||||
|
|
||||||
|
// Mount is the PKI secrets engine mount path.
|
||||||
|
// Default: "pki".
|
||||||
|
// Setting: CERTCTL_VAULT_MOUNT environment variable.
|
||||||
|
Mount string
|
||||||
|
|
||||||
|
// Role is the PKI role name used for signing certificates.
|
||||||
|
// Required for Vault PKI integration.
|
||||||
|
// Setting: CERTCTL_VAULT_ROLE environment variable.
|
||||||
|
Role string
|
||||||
|
|
||||||
|
// TTL is the requested certificate time-to-live.
|
||||||
|
// Default: "8760h" (1 year).
|
||||||
|
// Setting: CERTCTL_VAULT_TTL environment variable.
|
||||||
|
TTL string
|
||||||
|
}
|
||||||
|
|
||||||
|
// DigiCertConfig contains DigiCert CertCentral issuer connector configuration.
|
||||||
|
type DigiCertConfig struct {
|
||||||
|
// APIKey is the CertCentral API key for authentication.
|
||||||
|
// Required for DigiCert integration.
|
||||||
|
// Setting: CERTCTL_DIGICERT_API_KEY environment variable.
|
||||||
|
APIKey string
|
||||||
|
|
||||||
|
// OrgID is the DigiCert organization ID for certificate orders.
|
||||||
|
// Required for DigiCert integration.
|
||||||
|
// Setting: CERTCTL_DIGICERT_ORG_ID environment variable.
|
||||||
|
OrgID string
|
||||||
|
|
||||||
|
// ProductType is the DigiCert product type for certificate orders.
|
||||||
|
// Default: "ssl_basic". Common values: "ssl_basic", "ssl_wildcard", "ssl_ev_basic".
|
||||||
|
// Setting: CERTCTL_DIGICERT_PRODUCT_TYPE environment variable.
|
||||||
|
ProductType string
|
||||||
|
|
||||||
|
// BaseURL is the DigiCert CertCentral API base URL.
|
||||||
|
// Default: "https://www.digicert.com/services/v2".
|
||||||
|
// Setting: CERTCTL_DIGICERT_BASE_URL environment variable.
|
||||||
|
BaseURL string
|
||||||
|
}
|
||||||
|
|
||||||
// DigestConfig controls the scheduled certificate digest email feature.
|
// DigestConfig controls the scheduled certificate digest email feature.
|
||||||
type DigestConfig struct {
|
type DigestConfig struct {
|
||||||
// Enabled controls whether periodic digest emails are generated and sent.
|
// Enabled controls whether periodic digest emails are generated and sent.
|
||||||
@@ -429,6 +482,19 @@ func Load() (*Config, error) {
|
|||||||
Timeout: getEnvDuration("CERTCTL_VERIFY_TIMEOUT", 10*time.Second),
|
Timeout: getEnvDuration("CERTCTL_VERIFY_TIMEOUT", 10*time.Second),
|
||||||
Delay: getEnvDuration("CERTCTL_VERIFY_DELAY", 2*time.Second),
|
Delay: getEnvDuration("CERTCTL_VERIFY_DELAY", 2*time.Second),
|
||||||
},
|
},
|
||||||
|
Vault: VaultConfig{
|
||||||
|
Addr: getEnv("CERTCTL_VAULT_ADDR", ""),
|
||||||
|
Token: getEnv("CERTCTL_VAULT_TOKEN", ""),
|
||||||
|
Mount: getEnv("CERTCTL_VAULT_MOUNT", "pki"),
|
||||||
|
Role: getEnv("CERTCTL_VAULT_ROLE", ""),
|
||||||
|
TTL: getEnv("CERTCTL_VAULT_TTL", "8760h"),
|
||||||
|
},
|
||||||
|
DigiCert: DigiCertConfig{
|
||||||
|
APIKey: getEnv("CERTCTL_DIGICERT_API_KEY", ""),
|
||||||
|
OrgID: getEnv("CERTCTL_DIGICERT_ORG_ID", ""),
|
||||||
|
ProductType: getEnv("CERTCTL_DIGICERT_PRODUCT_TYPE", "ssl_basic"),
|
||||||
|
BaseURL: getEnv("CERTCTL_DIGICERT_BASE_URL", "https://www.digicert.com/services/v2"),
|
||||||
|
},
|
||||||
ACME: ACMEConfig{
|
ACME: ACMEConfig{
|
||||||
DirectoryURL: getEnv("CERTCTL_ACME_DIRECTORY_URL", ""),
|
DirectoryURL: getEnv("CERTCTL_ACME_DIRECTORY_URL", ""),
|
||||||
Email: getEnv("CERTCTL_ACME_EMAIL", ""),
|
Email: getEnv("CERTCTL_ACME_EMAIL", ""),
|
||||||
|
|||||||
@@ -0,0 +1,524 @@
|
|||||||
|
// Package digicert implements the issuer.Connector interface for DigiCert CertCentral.
|
||||||
|
//
|
||||||
|
// DigiCert CertCentral is an enterprise certificate authority offering DV, OV, and EV
|
||||||
|
// certificates. Unlike synchronous issuers (Vault, step-ca), DigiCert uses an
|
||||||
|
// asynchronous order model: submit an order, receive an order ID, then poll for
|
||||||
|
// completion. OV/EV certificates require organization validation which may take hours
|
||||||
|
// or days; DV certificates may be issued immediately.
|
||||||
|
//
|
||||||
|
// This connector maps to certctl's existing job state machine:
|
||||||
|
// - IssueCertificate submits the order; if status is "issued", returns cert immediately.
|
||||||
|
// If status is "pending", returns OrderID with empty CertPEM — the job system polls
|
||||||
|
// via GetOrderStatus.
|
||||||
|
// - GetOrderStatus polls the order; when status becomes "issued", downloads and
|
||||||
|
// parses the PEM bundle.
|
||||||
|
//
|
||||||
|
// Authentication: API key via X-DC-DEVKEY header.
|
||||||
|
//
|
||||||
|
// DigiCert CertCentral API used:
|
||||||
|
//
|
||||||
|
// POST /order/certificate/{product_type} - Submit certificate order
|
||||||
|
// GET /order/certificate/{order_id} - Check order status
|
||||||
|
// GET /certificate/{certificate_id}/download/format/pem_all - Download cert bundle
|
||||||
|
// PUT /certificate/{certificate_id}/revoke - Revoke certificate
|
||||||
|
// GET /user/me - Validate API credentials
|
||||||
|
package digicert
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"crypto/x509"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/pem"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log/slog"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Config represents the DigiCert CertCentral issuer connector configuration.
|
||||||
|
type Config struct {
|
||||||
|
// APIKey is the CertCentral API key for authentication.
|
||||||
|
// Required. Set via CERTCTL_DIGICERT_API_KEY environment variable.
|
||||||
|
APIKey string `json:"api_key"`
|
||||||
|
|
||||||
|
// OrgID is the DigiCert organization ID for certificate orders.
|
||||||
|
// Required. Set via CERTCTL_DIGICERT_ORG_ID environment variable.
|
||||||
|
OrgID string `json:"org_id"`
|
||||||
|
|
||||||
|
// ProductType is the DigiCert product type for certificate orders.
|
||||||
|
// Default: "ssl_basic". Set via CERTCTL_DIGICERT_PRODUCT_TYPE environment variable.
|
||||||
|
// Common values: "ssl_basic", "ssl_wildcard", "ssl_ev_basic", "ssl_plus", "ssl_multi_domain".
|
||||||
|
ProductType string `json:"product_type"`
|
||||||
|
|
||||||
|
// BaseURL is the DigiCert CertCentral API base URL.
|
||||||
|
// Default: "https://www.digicert.com/services/v2".
|
||||||
|
// Set via CERTCTL_DIGICERT_BASE_URL environment variable.
|
||||||
|
BaseURL string `json:"base_url"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Connector implements the issuer.Connector interface for DigiCert CertCentral.
|
||||||
|
type Connector struct {
|
||||||
|
config *Config
|
||||||
|
logger *slog.Logger
|
||||||
|
httpClient *http.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a new DigiCert CertCentral connector with the given configuration and logger.
|
||||||
|
func New(config *Config, logger *slog.Logger) *Connector {
|
||||||
|
if config != nil {
|
||||||
|
if config.ProductType == "" {
|
||||||
|
config.ProductType = "ssl_basic"
|
||||||
|
}
|
||||||
|
if config.BaseURL == "" {
|
||||||
|
config.BaseURL = "https://www.digicert.com/services/v2"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &Connector{
|
||||||
|
config: config,
|
||||||
|
logger: logger,
|
||||||
|
httpClient: &http.Client{
|
||||||
|
Timeout: 30 * time.Second,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// orderRequest is the JSON body for DigiCert certificate order submission.
|
||||||
|
type orderRequest struct {
|
||||||
|
Certificate orderCert `json:"certificate"`
|
||||||
|
Organization orderOrg `json:"organization"`
|
||||||
|
ValidityYears int `json:"validity_years"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type orderCert struct {
|
||||||
|
CommonName string `json:"common_name"`
|
||||||
|
CSR string `json:"csr"`
|
||||||
|
DNSNames []string `json:"dns_names,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type orderOrg struct {
|
||||||
|
ID json.Number `json:"id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// orderResponse is the JSON response from a certificate order submission.
|
||||||
|
type orderResponse struct {
|
||||||
|
ID int `json:"id"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
CertificateID int `json:"certificate_id,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// orderStatusResponse is the JSON response from an order status check.
|
||||||
|
type orderStatusResponse struct {
|
||||||
|
ID int `json:"id"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
Certificate struct {
|
||||||
|
ID int `json:"id"`
|
||||||
|
CommonName string `json:"common_name"`
|
||||||
|
} `json:"certificate"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateConfig checks that the DigiCert configuration is valid and API access works.
|
||||||
|
func (c *Connector) ValidateConfig(ctx context.Context, rawConfig json.RawMessage) error {
|
||||||
|
var cfg Config
|
||||||
|
if err := json.Unmarshal(rawConfig, &cfg); err != nil {
|
||||||
|
return fmt.Errorf("invalid DigiCert config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.APIKey == "" {
|
||||||
|
return fmt.Errorf("DigiCert api_key is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.OrgID == "" {
|
||||||
|
return fmt.Errorf("DigiCert org_id is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.ProductType == "" {
|
||||||
|
cfg.ProductType = "ssl_basic"
|
||||||
|
}
|
||||||
|
if cfg.BaseURL == "" {
|
||||||
|
cfg.BaseURL = "https://www.digicert.com/services/v2"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test API access via /user/me
|
||||||
|
meURL := cfg.BaseURL + "/user/me"
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, meURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create API test request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("X-DC-DEVKEY", cfg.APIKey)
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("DigiCert API not reachable at %s: %w", cfg.BaseURL, err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode == http.StatusForbidden || resp.StatusCode == http.StatusUnauthorized {
|
||||||
|
return fmt.Errorf("DigiCert API key is invalid (status %d)", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return fmt.Errorf("DigiCert API returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.config = &cfg
|
||||||
|
c.logger.Info("DigiCert CertCentral configuration validated",
|
||||||
|
"base_url", cfg.BaseURL,
|
||||||
|
"product_type", cfg.ProductType)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// IssueCertificate submits a certificate order to DigiCert CertCentral.
|
||||||
|
// If the certificate is issued immediately (DV certs), returns the cert.
|
||||||
|
// If pending (OV/EV certs), returns OrderID with empty CertPEM for polling.
|
||||||
|
func (c *Connector) IssueCertificate(ctx context.Context, request issuer.IssuanceRequest) (*issuer.IssuanceResult, error) {
|
||||||
|
c.logger.Info("processing DigiCert issuance request",
|
||||||
|
"common_name", request.CommonName,
|
||||||
|
"san_count", len(request.SANs),
|
||||||
|
"product_type", c.config.ProductType)
|
||||||
|
|
||||||
|
orderReq := orderRequest{
|
||||||
|
Certificate: orderCert{
|
||||||
|
CommonName: request.CommonName,
|
||||||
|
CSR: request.CSRPEM,
|
||||||
|
DNSNames: request.SANs,
|
||||||
|
},
|
||||||
|
Organization: orderOrg{
|
||||||
|
ID: json.Number(c.config.OrgID),
|
||||||
|
},
|
||||||
|
ValidityYears: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := json.Marshal(orderReq)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to marshal order request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
orderURL := fmt.Sprintf("%s/order/certificate/%s", c.config.BaseURL, c.config.ProductType)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, orderURL, bytes.NewReader(body))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create order request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("X-DC-DEVKEY", c.config.APIKey)
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("DigiCert order request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read order response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusCreated {
|
||||||
|
return nil, fmt.Errorf("DigiCert order returned status %d: %s", resp.StatusCode, string(respBody))
|
||||||
|
}
|
||||||
|
|
||||||
|
var orderResp orderResponse
|
||||||
|
if err := json.Unmarshal(respBody, &orderResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse order response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
orderID := fmt.Sprintf("%d", orderResp.ID)
|
||||||
|
|
||||||
|
c.logger.Info("DigiCert order submitted",
|
||||||
|
"order_id", orderID,
|
||||||
|
"status", orderResp.Status)
|
||||||
|
|
||||||
|
// If issued immediately (DV certs), download the certificate
|
||||||
|
if orderResp.Status == "issued" && orderResp.CertificateID > 0 {
|
||||||
|
certPEM, chainPEM, serial, notBefore, notAfter, err := c.downloadCertificate(ctx, orderResp.CertificateID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to download certificate: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.logger.Info("DigiCert certificate issued immediately",
|
||||||
|
"order_id", orderID,
|
||||||
|
"serial", serial)
|
||||||
|
|
||||||
|
return &issuer.IssuanceResult{
|
||||||
|
CertPEM: certPEM,
|
||||||
|
ChainPEM: chainPEM,
|
||||||
|
Serial: serial,
|
||||||
|
NotBefore: notBefore,
|
||||||
|
NotAfter: notAfter,
|
||||||
|
OrderID: orderID,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pending — return OrderID for polling via GetOrderStatus
|
||||||
|
c.logger.Info("DigiCert order pending validation",
|
||||||
|
"order_id", orderID,
|
||||||
|
"status", orderResp.Status)
|
||||||
|
|
||||||
|
return &issuer.IssuanceResult{
|
||||||
|
OrderID: orderID,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RenewCertificate renews a certificate by submitting a new order.
|
||||||
|
// DigiCert uses reissue for renewal, but for simplicity we submit a new order
|
||||||
|
// (reissue requires the original order ID which may not be available).
|
||||||
|
func (c *Connector) RenewCertificate(ctx context.Context, request issuer.RenewalRequest) (*issuer.IssuanceResult, error) {
|
||||||
|
c.logger.Info("processing DigiCert renewal request",
|
||||||
|
"common_name", request.CommonName,
|
||||||
|
"san_count", len(request.SANs))
|
||||||
|
|
||||||
|
return c.IssueCertificate(ctx, issuer.IssuanceRequest{
|
||||||
|
CommonName: request.CommonName,
|
||||||
|
SANs: request.SANs,
|
||||||
|
CSRPEM: request.CSRPEM,
|
||||||
|
EKUs: request.EKUs,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// RevokeCertificate revokes a certificate at DigiCert CertCentral.
|
||||||
|
// DigiCert revocation uses certificate_id, so we extract it from the serial
|
||||||
|
// by looking up the order. For simplicity, we use the serial as the cert ID
|
||||||
|
// (the caller should provide the DigiCert certificate ID).
|
||||||
|
func (c *Connector) RevokeCertificate(ctx context.Context, request issuer.RevocationRequest) error {
|
||||||
|
c.logger.Info("processing DigiCert revocation request", "serial", request.Serial)
|
||||||
|
|
||||||
|
reason := "unspecified"
|
||||||
|
if request.Reason != nil {
|
||||||
|
reason = *request.Reason
|
||||||
|
}
|
||||||
|
|
||||||
|
revokeBody := map[string]interface{}{
|
||||||
|
"reason": reason,
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := json.Marshal(revokeBody)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to marshal revoke request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DigiCert uses certificate_id in the URL path for revocation
|
||||||
|
revokeURL := fmt.Sprintf("%s/certificate/%s/revoke", c.config.BaseURL, request.Serial)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodPut, revokeURL, bytes.NewReader(body))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create revoke request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("X-DC-DEVKEY", c.config.APIKey)
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("DigiCert revoke request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
// DigiCert returns 204 No Content on successful revocation
|
||||||
|
if resp.StatusCode != http.StatusNoContent && resp.StatusCode != http.StatusOK {
|
||||||
|
respBody, _ := io.ReadAll(resp.Body)
|
||||||
|
return fmt.Errorf("DigiCert revoke returned status %d: %s", resp.StatusCode, string(respBody))
|
||||||
|
}
|
||||||
|
|
||||||
|
c.logger.Info("DigiCert certificate revoked", "serial", request.Serial, "reason", reason)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetOrderStatus checks the status of a DigiCert certificate order.
|
||||||
|
// If the order is "issued", downloads the certificate and returns it.
|
||||||
|
// If still "pending", returns pending status for continued polling.
|
||||||
|
func (c *Connector) GetOrderStatus(ctx context.Context, orderID string) (*issuer.OrderStatus, error) {
|
||||||
|
c.logger.Debug("checking DigiCert order status", "order_id", orderID)
|
||||||
|
|
||||||
|
statusURL := fmt.Sprintf("%s/order/certificate/%s", c.config.BaseURL, orderID)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, statusURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create status request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("X-DC-DEVKEY", c.config.APIKey)
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("DigiCert status request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read status response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return nil, fmt.Errorf("DigiCert order status returned %d: %s", resp.StatusCode, string(respBody))
|
||||||
|
}
|
||||||
|
|
||||||
|
var statusResp orderStatusResponse
|
||||||
|
if err := json.Unmarshal(respBody, &statusResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse status response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
|
||||||
|
switch statusResp.Status {
|
||||||
|
case "issued":
|
||||||
|
if statusResp.Certificate.ID == 0 {
|
||||||
|
return nil, fmt.Errorf("order is issued but certificate_id is missing")
|
||||||
|
}
|
||||||
|
|
||||||
|
certPEM, chainPEM, serial, notBefore, notAfter, err := c.downloadCertificate(ctx, statusResp.Certificate.ID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to download certificate: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.logger.Info("DigiCert order completed",
|
||||||
|
"order_id", orderID,
|
||||||
|
"serial", serial)
|
||||||
|
|
||||||
|
return &issuer.OrderStatus{
|
||||||
|
OrderID: orderID,
|
||||||
|
Status: "completed",
|
||||||
|
CertPEM: &certPEM,
|
||||||
|
ChainPEM: &chainPEM,
|
||||||
|
Serial: &serial,
|
||||||
|
NotBefore: ¬Before,
|
||||||
|
NotAfter: ¬After,
|
||||||
|
UpdatedAt: now,
|
||||||
|
}, nil
|
||||||
|
|
||||||
|
case "pending", "processing":
|
||||||
|
msg := fmt.Sprintf("order %s is %s", orderID, statusResp.Status)
|
||||||
|
return &issuer.OrderStatus{
|
||||||
|
OrderID: orderID,
|
||||||
|
Status: "pending",
|
||||||
|
Message: &msg,
|
||||||
|
UpdatedAt: now,
|
||||||
|
}, nil
|
||||||
|
|
||||||
|
case "rejected", "denied":
|
||||||
|
msg := fmt.Sprintf("order %s was %s", orderID, statusResp.Status)
|
||||||
|
return &issuer.OrderStatus{
|
||||||
|
OrderID: orderID,
|
||||||
|
Status: "failed",
|
||||||
|
Message: &msg,
|
||||||
|
UpdatedAt: now,
|
||||||
|
}, nil
|
||||||
|
|
||||||
|
default:
|
||||||
|
msg := fmt.Sprintf("unknown order status: %s", statusResp.Status)
|
||||||
|
return &issuer.OrderStatus{
|
||||||
|
OrderID: orderID,
|
||||||
|
Status: "pending",
|
||||||
|
Message: &msg,
|
||||||
|
UpdatedAt: now,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// downloadCertificate downloads the PEM bundle for a DigiCert certificate.
|
||||||
|
func (c *Connector) downloadCertificate(ctx context.Context, certificateID int) (certPEM string, chainPEM string, serial string, notBefore time.Time, notAfter time.Time, err error) {
|
||||||
|
downloadURL := fmt.Sprintf("%s/certificate/%d/download/format/pem_all", c.config.BaseURL, certificateID)
|
||||||
|
req, reqErr := http.NewRequestWithContext(ctx, http.MethodGet, downloadURL, nil)
|
||||||
|
if reqErr != nil {
|
||||||
|
err = fmt.Errorf("failed to create download request: %w", reqErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
req.Header.Set("X-DC-DEVKEY", c.config.APIKey)
|
||||||
|
|
||||||
|
resp, doErr := c.httpClient.Do(req)
|
||||||
|
if doErr != nil {
|
||||||
|
err = fmt.Errorf("DigiCert download request failed: %w", doErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
body, _ := io.ReadAll(resp.Body)
|
||||||
|
err = fmt.Errorf("DigiCert download returned status %d: %s", resp.StatusCode, string(body))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
body, readErr := io.ReadAll(resp.Body)
|
||||||
|
if readErr != nil {
|
||||||
|
err = fmt.Errorf("failed to read download response: %w", readErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the PEM bundle: first cert is the leaf, rest are intermediates
|
||||||
|
certPEM, chainPEM, serial, notBefore, notAfter, err = parsePEMBundle(string(body))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// parsePEMBundle splits a PEM bundle into leaf cert and chain, extracting metadata.
|
||||||
|
func parsePEMBundle(bundle string) (certPEM string, chainPEM string, serial string, notBefore time.Time, notAfter time.Time, err error) {
|
||||||
|
var certs []string
|
||||||
|
remaining := bundle
|
||||||
|
|
||||||
|
for {
|
||||||
|
var block *pem.Block
|
||||||
|
block, rest := pem.Decode([]byte(remaining))
|
||||||
|
if block == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if block.Type == "CERTIFICATE" {
|
||||||
|
certs = append(certs, string(pem.EncodeToMemory(block)))
|
||||||
|
}
|
||||||
|
remaining = string(rest)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(certs) == 0 {
|
||||||
|
err = fmt.Errorf("no certificates found in PEM bundle")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
certPEM = certs[0]
|
||||||
|
if len(certs) > 1 {
|
||||||
|
chainPEM = strings.Join(certs[1:], "")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse leaf cert for metadata
|
||||||
|
block, _ := pem.Decode([]byte(certPEM))
|
||||||
|
if block == nil {
|
||||||
|
err = fmt.Errorf("failed to decode leaf certificate PEM")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
cert, parseErr := x509.ParseCertificate(block.Bytes)
|
||||||
|
if parseErr != nil {
|
||||||
|
err = fmt.Errorf("failed to parse leaf certificate: %w", parseErr)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
serial = cert.SerialNumber.String()
|
||||||
|
notBefore = cert.NotBefore
|
||||||
|
notAfter = cert.NotAfter
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// GenerateCRL is not supported because DigiCert manages CRL distribution.
|
||||||
|
func (c *Connector) GenerateCRL(ctx context.Context, revokedCerts []issuer.RevokedCertEntry) ([]byte, error) {
|
||||||
|
return nil, fmt.Errorf("DigiCert manages CRL distribution; use DigiCert's CRL endpoints")
|
||||||
|
}
|
||||||
|
|
||||||
|
// SignOCSPResponse is not supported because DigiCert manages OCSP.
|
||||||
|
func (c *Connector) SignOCSPResponse(ctx context.Context, req issuer.OCSPSignRequest) ([]byte, error) {
|
||||||
|
return nil, fmt.Errorf("DigiCert manages OCSP; use DigiCert's OCSP responder")
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCACertPEM is not directly supported. DigiCert intermediate certificates
|
||||||
|
// come with each certificate issuance as part of the PEM bundle.
|
||||||
|
func (c *Connector) GetCACertPEM(ctx context.Context) (string, error) {
|
||||||
|
return "", fmt.Errorf("DigiCert intermediate certificates are included with each issued certificate")
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetRenewalInfo returns nil, nil as DigiCert does not support ACME Renewal Information (ARI).
|
||||||
|
func (c *Connector) GetRenewalInfo(ctx context.Context, certPEM string) (*issuer.RenewalInfoResult, error) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure Connector implements the issuer.Connector interface.
|
||||||
|
var _ issuer.Connector = (*Connector)(nil)
|
||||||
@@ -0,0 +1,591 @@
|
|||||||
|
package digicert_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/rand"
|
||||||
|
"crypto/rsa"
|
||||||
|
"crypto/x509"
|
||||||
|
"crypto/x509/pkix"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/pem"
|
||||||
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
"math/big"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer"
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer/digicert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestDigiCertConnector(t *testing.T) {
|
||||||
|
logger := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelDebug}))
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_Success", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/user/me" {
|
||||||
|
if r.Header.Get("X-DC-DEVKEY") == "dc-test-api-key" {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"id":12345,"first_name":"Test","last_name":"User"}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
w.Write([]byte(`{"errors":[{"code":"invalid_api_key"}]}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := digicert.Config{
|
||||||
|
APIKey: "dc-test-api-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
ProductType: "ssl_basic",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := digicert.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ValidateConfig failed: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_MissingAPIKey", func(t *testing.T) {
|
||||||
|
config := digicert.Config{
|
||||||
|
OrgID: "12345",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := digicert.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for missing api_key")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "api_key is required") {
|
||||||
|
t.Errorf("Expected api_key required error, got: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_MissingOrgID", func(t *testing.T) {
|
||||||
|
config := digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := digicert.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for missing org_id")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "org_id is required") {
|
||||||
|
t.Errorf("Expected org_id required error, got: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_InvalidKey", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/user/me" {
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
w.Write([]byte(`{"errors":[{"code":"invalid_api_key"}]}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := digicert.Config{
|
||||||
|
APIKey: "dc-bad-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := digicert.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for invalid API key")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "invalid") {
|
||||||
|
t.Logf("Got error: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_ImmediateSuccess", func(t *testing.T) {
|
||||||
|
testCertPEM, _ := generateTestCert(t)
|
||||||
|
testChainPEM, _ := generateTestCert(t)
|
||||||
|
pemBundle := testCertPEM + testChainPEM
|
||||||
|
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/order/certificate/ssl_basic"):
|
||||||
|
w.WriteHeader(http.StatusCreated)
|
||||||
|
w.Write([]byte(`{"id":99001,"status":"issued","certificate_id":88001}`))
|
||||||
|
case r.URL.Path == "/certificate/88001/download/format/pem_all":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(pemBundle))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
ProductType: "ssl_basic",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "app.example.com")
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "app.example.com",
|
||||||
|
SANs: []string{"app.example.com"},
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("IssueCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if result.CertPEM == "" {
|
||||||
|
t.Error("CertPEM should not be empty for immediate issuance")
|
||||||
|
}
|
||||||
|
if result.Serial == "" {
|
||||||
|
t.Error("Serial should not be empty for immediate issuance")
|
||||||
|
}
|
||||||
|
if result.OrderID != "99001" {
|
||||||
|
t.Errorf("Expected OrderID '99001', got '%s'", result.OrderID)
|
||||||
|
}
|
||||||
|
t.Logf("DigiCert issued cert: serial=%s, orderID=%s", result.Serial, result.OrderID)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_Pending", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/order/certificate/ssl_ev_basic"):
|
||||||
|
w.WriteHeader(http.StatusCreated)
|
||||||
|
w.Write([]byte(`{"id":99002,"status":"pending"}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
ProductType: "ssl_ev_basic",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "secure.example.com")
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "secure.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("IssueCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if result.OrderID != "99002" {
|
||||||
|
t.Errorf("Expected OrderID '99002', got '%s'", result.OrderID)
|
||||||
|
}
|
||||||
|
if result.CertPEM != "" {
|
||||||
|
t.Error("CertPEM should be empty for pending order")
|
||||||
|
}
|
||||||
|
if result.Serial != "" {
|
||||||
|
t.Error("Serial should be empty for pending order")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_ServerError", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte(`{"errors":[{"code":"invalid_csr","message":"CSR is malformed"}]}`))
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
ProductType: "ssl_basic",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "test.example.com",
|
||||||
|
CSRPEM: "invalid-csr",
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for server error response")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetOrderStatus_Issued", func(t *testing.T) {
|
||||||
|
testCertPEM, _ := generateTestCert(t)
|
||||||
|
testChainPEM, _ := generateTestCert(t)
|
||||||
|
pemBundle := testCertPEM + testChainPEM
|
||||||
|
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.URL.Path {
|
||||||
|
case "/order/certificate/99001":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"id":99001,"status":"issued","certificate":{"id":88001,"common_name":"app.example.com"}}`))
|
||||||
|
case "/certificate/88001/download/format/pem_all":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(pemBundle))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
status, err := connector.GetOrderStatus(ctx, "99001")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetOrderStatus failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if status.Status != "completed" {
|
||||||
|
t.Errorf("Expected status 'completed', got '%s'", status.Status)
|
||||||
|
}
|
||||||
|
if status.CertPEM == nil || *status.CertPEM == "" {
|
||||||
|
t.Error("CertPEM should not be empty for issued order")
|
||||||
|
}
|
||||||
|
if status.Serial == nil || *status.Serial == "" {
|
||||||
|
t.Error("Serial should not be empty for issued order")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetOrderStatus_Pending", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/order/certificate/99002" {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"id":99002,"status":"pending","certificate":{"id":0}}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
status, err := connector.GetOrderStatus(ctx, "99002")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetOrderStatus failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if status.Status != "pending" {
|
||||||
|
t.Errorf("Expected status 'pending', got '%s'", status.Status)
|
||||||
|
}
|
||||||
|
if status.CertPEM != nil {
|
||||||
|
t.Error("CertPEM should be nil for pending order")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetOrderStatus_Rejected", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/order/certificate/99003" {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"id":99003,"status":"rejected","certificate":{"id":0}}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
status, err := connector.GetOrderStatus(ctx, "99003")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetOrderStatus failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if status.Status != "failed" {
|
||||||
|
t.Errorf("Expected status 'failed', got '%s'", status.Status)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RenewCertificate_NewOrder", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/order/certificate/"):
|
||||||
|
w.WriteHeader(http.StatusCreated)
|
||||||
|
w.Write([]byte(`{"id":99010,"status":"pending"}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
ProductType: "ssl_basic",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "renew.example.com")
|
||||||
|
renewReq := issuer.RenewalRequest{
|
||||||
|
CommonName: "renew.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.RenewCertificate(ctx, renewReq)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("RenewCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if result.OrderID == "" {
|
||||||
|
t.Error("OrderID should not be empty")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RevokeCertificate_Success", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if strings.HasSuffix(r.URL.Path, "/revoke") && r.Method == http.MethodPut {
|
||||||
|
if r.Header.Get("X-DC-DEVKEY") == "" {
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.WriteHeader(http.StatusNoContent)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
reason := "keyCompromise"
|
||||||
|
revokeReq := issuer.RevocationRequest{
|
||||||
|
Serial: "88001",
|
||||||
|
Reason: &reason,
|
||||||
|
}
|
||||||
|
|
||||||
|
err := connector.RevokeCertificate(ctx, revokeReq)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("RevokeCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RevokeCertificate_Error", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte(`{"errors":[{"code":"certificate_not_found"}]}`))
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
revokeReq := issuer.RevocationRequest{
|
||||||
|
Serial: "00000",
|
||||||
|
}
|
||||||
|
|
||||||
|
err := connector.RevokeCertificate(ctx, revokeReq)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for revocation of nonexistent cert")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetOrderStatus_DownloadError", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.URL.Path {
|
||||||
|
case "/order/certificate/99004":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"id":99004,"status":"issued","certificate":{"id":88004}}`))
|
||||||
|
case "/certificate/88004/download/format/pem_all":
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
w.Write([]byte(`{"errors":["internal server error"]}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: srv.URL,
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
_, err := connector.GetOrderStatus(ctx, "99004")
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error when download fails")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "download") {
|
||||||
|
t.Logf("Got error: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetRenewalInfo_ReturnsNil", func(t *testing.T) {
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
BaseURL: "https://api.digicert.com",
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
result, err := connector.GetRenewalInfo(ctx, "-----BEGIN CERTIFICATE-----\ntest\n-----END CERTIFICATE-----")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetRenewalInfo should not return error, got: %v", err)
|
||||||
|
}
|
||||||
|
if result != nil {
|
||||||
|
t.Fatal("GetRenewalInfo should return nil for DigiCert")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("DefaultProductType", func(t *testing.T) {
|
||||||
|
config := &digicert.Config{
|
||||||
|
APIKey: "dc-test-key",
|
||||||
|
OrgID: "12345",
|
||||||
|
// ProductType intentionally left empty
|
||||||
|
}
|
||||||
|
connector := digicert.New(config, logger)
|
||||||
|
|
||||||
|
// Verify the connector was created (the default is set in New())
|
||||||
|
if connector == nil {
|
||||||
|
t.Fatal("Connector should not be nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify via a request that uses the product type
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// Verify the path includes the default product type
|
||||||
|
if strings.Contains(r.URL.Path, "ssl_basic") {
|
||||||
|
w.WriteHeader(http.StatusCreated)
|
||||||
|
w.Write([]byte(`{"id":99099,"status":"pending"}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
t.Errorf("Expected path to contain 'ssl_basic', got: %s", r.URL.Path)
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
// Reconfigure with test server URL
|
||||||
|
config.BaseURL = srv.URL
|
||||||
|
connector = digicert.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "test.example.com")
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "test.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("IssueCertificate with default product type failed: %v", err)
|
||||||
|
}
|
||||||
|
if result.OrderID == "" {
|
||||||
|
t.Error("OrderID should not be empty")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// generateTestCert creates a self-signed test certificate and returns the PEM strings.
|
||||||
|
func generateTestCert(t *testing.T) (certPEM string, keyPEM string) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
key, err := rsa.GenerateKey(rand.Reader, 2048)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to generate key: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serial, _ := rand.Int(rand.Reader, new(big.Int).Lsh(big.NewInt(1), 128))
|
||||||
|
template := &x509.Certificate{
|
||||||
|
SerialNumber: serial,
|
||||||
|
Subject: pkix.Name{
|
||||||
|
CommonName: fmt.Sprintf("Test Certificate %s", serial.String()[:8]),
|
||||||
|
},
|
||||||
|
DNSNames: []string{"test.example.com"},
|
||||||
|
KeyUsage: x509.KeyUsageDigitalSignature,
|
||||||
|
ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth},
|
||||||
|
BasicConstraintsValid: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
certBytes, err := x509.CreateCertificate(rand.Reader, template, template, &key.PublicKey, key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create certificate: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
certPEM = string(pem.EncodeToMemory(&pem.Block{Type: "CERTIFICATE", Bytes: certBytes}))
|
||||||
|
keyPEM = string(pem.EncodeToMemory(&pem.Block{Type: "RSA PRIVATE KEY", Bytes: x509.MarshalPKCS1PrivateKey(key)}))
|
||||||
|
|
||||||
|
return certPEM, keyPEM
|
||||||
|
}
|
||||||
|
|
||||||
|
// generateTestCSR creates a test CSR for the given common name.
|
||||||
|
func generateTestCSR(t *testing.T, commonName string) (*x509.CertificateRequest, string) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
key, err := rsa.GenerateKey(rand.Reader, 2048)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to generate key: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
csrTemplate := x509.CertificateRequest{
|
||||||
|
Subject: pkix.Name{
|
||||||
|
CommonName: commonName,
|
||||||
|
},
|
||||||
|
DNSNames: []string{commonName},
|
||||||
|
SignatureAlgorithm: x509.SHA256WithRSA,
|
||||||
|
}
|
||||||
|
|
||||||
|
csrBytes, err := x509.CreateCertificateRequest(rand.Reader, &csrTemplate, key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create CSR: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
csrPEM := string(pem.EncodeToMemory(&pem.Block{
|
||||||
|
Type: "CERTIFICATE REQUEST",
|
||||||
|
Bytes: csrBytes,
|
||||||
|
}))
|
||||||
|
|
||||||
|
csr, err := x509.ParseCertificateRequest(csrBytes)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to parse CSR: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return csr, csrPEM
|
||||||
|
}
|
||||||
@@ -0,0 +1,372 @@
|
|||||||
|
// Package vault implements the issuer.Connector interface for HashiCorp Vault PKI
|
||||||
|
// secrets engine.
|
||||||
|
//
|
||||||
|
// Vault PKI provides a full-featured private CA with certificate signing, revocation,
|
||||||
|
// CRL, and OCSP capabilities. This connector uses the Vault HTTP API to sign CSRs
|
||||||
|
// via the /v1/{mount}/sign/{role} endpoint, authenticated with a Vault token.
|
||||||
|
//
|
||||||
|
// Vault issues certificates synchronously (like step-ca), so GetOrderStatus always
|
||||||
|
// returns "completed". CRL and OCSP are delegated to Vault's own endpoints.
|
||||||
|
//
|
||||||
|
// Authentication: Vault token via X-Vault-Token header.
|
||||||
|
//
|
||||||
|
// Vault API used:
|
||||||
|
//
|
||||||
|
// GET /v1/sys/health - Health check
|
||||||
|
// POST /v1/{mount}/sign/{role} - Sign CSR
|
||||||
|
// POST /v1/{mount}/revoke - Revoke certificate
|
||||||
|
// GET /v1/{mount}/ca/pem - Get CA certificate
|
||||||
|
package vault
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"crypto/x509"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/pem"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log/slog"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Config represents the Vault PKI issuer connector configuration.
|
||||||
|
type Config struct {
|
||||||
|
// Addr is the Vault server address (e.g., "https://vault.example.com:8200").
|
||||||
|
// Required. Set via CERTCTL_VAULT_ADDR environment variable.
|
||||||
|
Addr string `json:"addr"`
|
||||||
|
|
||||||
|
// Token is the Vault token for authentication.
|
||||||
|
// Required. Set via CERTCTL_VAULT_TOKEN environment variable.
|
||||||
|
Token string `json:"token"`
|
||||||
|
|
||||||
|
// Mount is the PKI secrets engine mount path.
|
||||||
|
// Default: "pki". Set via CERTCTL_VAULT_MOUNT environment variable.
|
||||||
|
Mount string `json:"mount"`
|
||||||
|
|
||||||
|
// Role is the PKI role name used for signing certificates.
|
||||||
|
// Required. Set via CERTCTL_VAULT_ROLE environment variable.
|
||||||
|
Role string `json:"role"`
|
||||||
|
|
||||||
|
// TTL is the requested certificate TTL (e.g., "8760h" for 1 year).
|
||||||
|
// Default: "8760h". Set via CERTCTL_VAULT_TTL environment variable.
|
||||||
|
TTL string `json:"ttl"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Connector implements the issuer.Connector interface for Vault PKI.
|
||||||
|
type Connector struct {
|
||||||
|
config *Config
|
||||||
|
logger *slog.Logger
|
||||||
|
httpClient *http.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
// New creates a new Vault PKI connector with the given configuration and logger.
|
||||||
|
func New(config *Config, logger *slog.Logger) *Connector {
|
||||||
|
if config != nil {
|
||||||
|
if config.Mount == "" {
|
||||||
|
config.Mount = "pki"
|
||||||
|
}
|
||||||
|
if config.TTL == "" {
|
||||||
|
config.TTL = "8760h"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &Connector{
|
||||||
|
config: config,
|
||||||
|
logger: logger,
|
||||||
|
httpClient: &http.Client{
|
||||||
|
Timeout: 30 * time.Second,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// vaultResponse is the standard Vault API response wrapper.
|
||||||
|
type vaultResponse struct {
|
||||||
|
Data json.RawMessage `json:"data"`
|
||||||
|
Errors []string `json:"errors,omitempty"`
|
||||||
|
Warnings []string `json:"warnings,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// signData holds the data returned from the /sign endpoint.
|
||||||
|
type signData struct {
|
||||||
|
Certificate string `json:"certificate"`
|
||||||
|
IssuingCA string `json:"issuing_ca"`
|
||||||
|
CAChain []string `json:"ca_chain"`
|
||||||
|
SerialNumber string `json:"serial_number"`
|
||||||
|
Expiration int64 `json:"expiration"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateConfig checks that the Vault configuration is valid and the server is reachable.
|
||||||
|
func (c *Connector) ValidateConfig(ctx context.Context, rawConfig json.RawMessage) error {
|
||||||
|
var cfg Config
|
||||||
|
if err := json.Unmarshal(rawConfig, &cfg); err != nil {
|
||||||
|
return fmt.Errorf("invalid Vault config: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.Addr == "" {
|
||||||
|
return fmt.Errorf("Vault addr is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.Token == "" {
|
||||||
|
return fmt.Errorf("Vault token is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.Role == "" {
|
||||||
|
return fmt.Errorf("Vault role is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
if cfg.Mount == "" {
|
||||||
|
cfg.Mount = "pki"
|
||||||
|
}
|
||||||
|
if cfg.TTL == "" {
|
||||||
|
cfg.TTL = "8760h"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Health check
|
||||||
|
healthURL := cfg.Addr + "/v1/sys/health"
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, healthURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create health check request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Vault not reachable at %s: %w", cfg.Addr, err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
// Vault health returns 200 for initialized+unsealed, 429 for standby, 472 for DR secondary,
|
||||||
|
// 473 for perf standby, 501 for uninitialized, 503 for sealed
|
||||||
|
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusTooManyRequests {
|
||||||
|
return fmt.Errorf("Vault health check returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.config = &cfg
|
||||||
|
c.logger.Info("Vault PKI configuration validated",
|
||||||
|
"addr", cfg.Addr,
|
||||||
|
"mount", cfg.Mount,
|
||||||
|
"role", cfg.Role)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// IssueCertificate submits a CSR to Vault PKI for signing.
|
||||||
|
func (c *Connector) IssueCertificate(ctx context.Context, request issuer.IssuanceRequest) (*issuer.IssuanceResult, error) {
|
||||||
|
c.logger.Info("processing Vault PKI issuance request",
|
||||||
|
"common_name", request.CommonName,
|
||||||
|
"san_count", len(request.SANs))
|
||||||
|
|
||||||
|
// Build the sign request body
|
||||||
|
signBody := map[string]interface{}{
|
||||||
|
"csr": request.CSRPEM,
|
||||||
|
"common_name": request.CommonName,
|
||||||
|
"ttl": c.config.TTL,
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(request.SANs) > 0 {
|
||||||
|
signBody["alt_names"] = strings.Join(request.SANs, ",")
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := json.Marshal(signBody)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to marshal sign request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// POST /v1/{mount}/sign/{role}
|
||||||
|
signURL := fmt.Sprintf("%s/v1/%s/sign/%s", c.config.Addr, c.config.Mount, c.config.Role)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, signURL, bytes.NewReader(body))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create sign request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
req.Header.Set("X-Vault-Token", c.config.Token)
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("Vault sign request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read sign response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
var vaultResp vaultResponse
|
||||||
|
if jsonErr := json.Unmarshal(respBody, &vaultResp); jsonErr == nil && len(vaultResp.Errors) > 0 {
|
||||||
|
return nil, fmt.Errorf("Vault sign returned status %d: %s", resp.StatusCode, strings.Join(vaultResp.Errors, "; "))
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("Vault sign returned status %d: %s", resp.StatusCode, string(respBody))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the Vault response
|
||||||
|
var vaultResp vaultResponse
|
||||||
|
if err := json.Unmarshal(respBody, &vaultResp); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse Vault response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var data signData
|
||||||
|
if err := json.Unmarshal(vaultResp.Data, &data); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse Vault sign data: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if data.Certificate == "" {
|
||||||
|
return nil, fmt.Errorf("no certificate in Vault sign response")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the leaf certificate to extract metadata
|
||||||
|
certPEM := data.Certificate
|
||||||
|
block, _ := pem.Decode([]byte(certPEM))
|
||||||
|
if block == nil {
|
||||||
|
return nil, fmt.Errorf("failed to decode certificate PEM from Vault")
|
||||||
|
}
|
||||||
|
|
||||||
|
cert, err := x509.ParseCertificate(block.Bytes)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to parse certificate: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build chain PEM from ca_chain or issuing_ca
|
||||||
|
var chainPEM string
|
||||||
|
if len(data.CAChain) > 0 {
|
||||||
|
chainPEM = strings.Join(data.CAChain, "\n")
|
||||||
|
} else if data.IssuingCA != "" {
|
||||||
|
chainPEM = data.IssuingCA
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normalize serial: Vault uses colon-separated hex (e.g., "aa:bb:cc"), convert to plain string
|
||||||
|
serial := normalizeSerial(data.SerialNumber)
|
||||||
|
|
||||||
|
orderID := fmt.Sprintf("vault-%s", serial)
|
||||||
|
|
||||||
|
c.logger.Info("Vault PKI certificate issued",
|
||||||
|
"common_name", request.CommonName,
|
||||||
|
"serial", serial,
|
||||||
|
"not_after", cert.NotAfter)
|
||||||
|
|
||||||
|
return &issuer.IssuanceResult{
|
||||||
|
CertPEM: certPEM,
|
||||||
|
ChainPEM: chainPEM,
|
||||||
|
Serial: serial,
|
||||||
|
NotBefore: cert.NotBefore,
|
||||||
|
NotAfter: cert.NotAfter,
|
||||||
|
OrderID: orderID,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RenewCertificate renews a certificate by creating a new signing request.
|
||||||
|
// For Vault PKI, renewal is functionally identical to issuance (new cert signed from CSR).
|
||||||
|
func (c *Connector) RenewCertificate(ctx context.Context, request issuer.RenewalRequest) (*issuer.IssuanceResult, error) {
|
||||||
|
c.logger.Info("processing Vault PKI renewal request",
|
||||||
|
"common_name", request.CommonName,
|
||||||
|
"san_count", len(request.SANs))
|
||||||
|
|
||||||
|
return c.IssueCertificate(ctx, issuer.IssuanceRequest{
|
||||||
|
CommonName: request.CommonName,
|
||||||
|
SANs: request.SANs,
|
||||||
|
CSRPEM: request.CSRPEM,
|
||||||
|
EKUs: request.EKUs,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// RevokeCertificate revokes a certificate at Vault PKI.
|
||||||
|
func (c *Connector) RevokeCertificate(ctx context.Context, request issuer.RevocationRequest) error {
|
||||||
|
c.logger.Info("processing Vault PKI revocation request", "serial", request.Serial)
|
||||||
|
|
||||||
|
revokeBody := map[string]interface{}{
|
||||||
|
"serial_number": request.Serial,
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := json.Marshal(revokeBody)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to marshal revoke request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
revokeURL := fmt.Sprintf("%s/v1/%s/revoke", c.config.Addr, c.config.Mount)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodPost, revokeURL, bytes.NewReader(body))
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to create revoke request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
req.Header.Set("X-Vault-Token", c.config.Token)
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("Vault revoke request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
respBody, _ := io.ReadAll(resp.Body)
|
||||||
|
return fmt.Errorf("Vault revoke returned status %d: %s", resp.StatusCode, string(respBody))
|
||||||
|
}
|
||||||
|
|
||||||
|
c.logger.Info("Vault PKI certificate revoked", "serial", request.Serial)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetOrderStatus returns the status of a Vault PKI order.
|
||||||
|
// Vault signs synchronously, so orders are always "completed" immediately.
|
||||||
|
func (c *Connector) GetOrderStatus(ctx context.Context, orderID string) (*issuer.OrderStatus, error) {
|
||||||
|
return &issuer.OrderStatus{
|
||||||
|
OrderID: orderID,
|
||||||
|
Status: "completed",
|
||||||
|
UpdatedAt: time.Now(),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GenerateCRL is not supported because Vault serves CRL directly at /v1/{mount}/crl.
|
||||||
|
func (c *Connector) GenerateCRL(ctx context.Context, revokedCerts []issuer.RevokedCertEntry) ([]byte, error) {
|
||||||
|
return nil, fmt.Errorf("Vault serves CRL directly at /v1/%s/crl; use Vault's endpoint", c.config.Mount)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SignOCSPResponse is not supported because Vault serves OCSP directly at /v1/{mount}/ocsp.
|
||||||
|
func (c *Connector) SignOCSPResponse(ctx context.Context, req issuer.OCSPSignRequest) ([]byte, error) {
|
||||||
|
return nil, fmt.Errorf("Vault serves OCSP directly at /v1/%s/ocsp; use Vault's endpoint", c.config.Mount)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCACertPEM retrieves the CA certificate from Vault PKI.
|
||||||
|
func (c *Connector) GetCACertPEM(ctx context.Context) (string, error) {
|
||||||
|
caURL := fmt.Sprintf("%s/v1/%s/ca/pem", c.config.Addr, c.config.Mount)
|
||||||
|
req, err := http.NewRequestWithContext(ctx, http.MethodGet, caURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("failed to create CA cert request: %w", err)
|
||||||
|
}
|
||||||
|
req.Header.Set("X-Vault-Token", c.config.Token)
|
||||||
|
|
||||||
|
resp, err := c.httpClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("Vault CA cert request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return "", fmt.Errorf("Vault CA cert returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("failed to read CA cert response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return string(body), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetRenewalInfo returns nil, nil as Vault does not support ACME Renewal Information (ARI).
|
||||||
|
func (c *Connector) GetRenewalInfo(ctx context.Context, certPEM string) (*issuer.RenewalInfoResult, error) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// normalizeSerial converts Vault's colon-separated hex serial (e.g., "aa:bb:cc:dd")
|
||||||
|
// to a plain string representation suitable for storage.
|
||||||
|
func normalizeSerial(serial string) string {
|
||||||
|
return strings.ReplaceAll(serial, ":", "-")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure Connector implements the issuer.Connector interface.
|
||||||
|
var _ issuer.Connector = (*Connector)(nil)
|
||||||
@@ -0,0 +1,527 @@
|
|||||||
|
package vault_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/rand"
|
||||||
|
"crypto/rsa"
|
||||||
|
"crypto/x509"
|
||||||
|
"crypto/x509/pkix"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/pem"
|
||||||
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
"math/big"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer"
|
||||||
|
"github.com/shankar0123/certctl/internal/connector/issuer/vault"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestVaultConnector(t *testing.T) {
|
||||||
|
logger := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelDebug}))
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_Success", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.URL.Path == "/v1/sys/health" {
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"initialized":true,"sealed":false,"standby":false}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token-12345",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
TTL: "8760h",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := vault.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ValidateConfig failed: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_MissingAddr", func(t *testing.T) {
|
||||||
|
config := vault.Config{
|
||||||
|
Token: "s.test-token",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := vault.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for missing addr")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "addr is required") {
|
||||||
|
t.Errorf("Expected addr required error, got: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_MissingToken", func(t *testing.T) {
|
||||||
|
config := vault.Config{
|
||||||
|
Addr: "https://vault.example.com:8200",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := vault.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for missing token")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "token is required") {
|
||||||
|
t.Errorf("Expected token required error, got: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_MissingRole", func(t *testing.T) {
|
||||||
|
config := vault.Config{
|
||||||
|
Addr: "https://vault.example.com:8200",
|
||||||
|
Token: "s.test-token",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := vault.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for missing role")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "role is required") {
|
||||||
|
t.Errorf("Expected role required error, got: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ValidateConfig_UnreachableVault", func(t *testing.T) {
|
||||||
|
config := vault.Config{
|
||||||
|
Addr: "http://localhost:19999",
|
||||||
|
Token: "s.test-token",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
|
||||||
|
connector := vault.New(nil, logger)
|
||||||
|
rawConfig, _ := json.Marshal(config)
|
||||||
|
err := connector.ValidateConfig(ctx, rawConfig)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for unreachable Vault")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_Success", func(t *testing.T) {
|
||||||
|
testCertPEM, _ := generateTestCert(t)
|
||||||
|
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case r.URL.Path == "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"initialized":true,"sealed":false}`))
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/v1/pki/sign/"):
|
||||||
|
// Verify auth header
|
||||||
|
if r.Header.Get("X-Vault-Token") != "s.test-token" {
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
w.Write([]byte(`{"errors":["permission denied"]}`))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
resp := fmt.Sprintf(`{
|
||||||
|
"data": {
|
||||||
|
"certificate": %q,
|
||||||
|
"issuing_ca": %q,
|
||||||
|
"ca_chain": [%q],
|
||||||
|
"serial_number": "aa:bb:cc:dd:ee:ff",
|
||||||
|
"expiration": 1893456000
|
||||||
|
}
|
||||||
|
}`, testCertPEM, testCertPEM, testCertPEM)
|
||||||
|
w.Write([]byte(resp))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
TTL: "8760h",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "app.example.com")
|
||||||
|
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "app.example.com",
|
||||||
|
SANs: []string{"app.example.com", "www.example.com"},
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("IssueCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if result.CertPEM == "" {
|
||||||
|
t.Error("CertPEM is empty")
|
||||||
|
}
|
||||||
|
if result.Serial == "" {
|
||||||
|
t.Error("Serial is empty")
|
||||||
|
}
|
||||||
|
if result.OrderID == "" {
|
||||||
|
t.Error("OrderID is empty")
|
||||||
|
}
|
||||||
|
if !strings.HasPrefix(result.OrderID, "vault-") {
|
||||||
|
t.Errorf("Expected OrderID to start with 'vault-', got '%s'", result.OrderID)
|
||||||
|
}
|
||||||
|
// Verify serial normalization (colons replaced with dashes)
|
||||||
|
if strings.Contains(result.Serial, ":") {
|
||||||
|
t.Errorf("Serial should not contain colons, got '%s'", result.Serial)
|
||||||
|
}
|
||||||
|
t.Logf("Vault issued cert: serial=%s, orderID=%s", result.Serial, result.OrderID)
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_ServerError", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case r.URL.Path == "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/v1/pki/sign/"):
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte(`{"errors":["invalid CSR"]}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "test.example.com")
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "test.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for server error response")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "invalid CSR") {
|
||||||
|
t.Logf("Got error: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("IssueCertificate_Forbidden", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case r.URL.Path == "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/v1/pki/sign/"):
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
w.Write([]byte(`{"errors":["permission denied"]}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.bad-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "test.example.com")
|
||||||
|
req := issuer.IssuanceRequest{
|
||||||
|
CommonName: "test.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := connector.IssueCertificate(ctx, req)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for forbidden response")
|
||||||
|
}
|
||||||
|
if !strings.Contains(err.Error(), "permission denied") {
|
||||||
|
t.Logf("Got error: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RenewCertificate_Success", func(t *testing.T) {
|
||||||
|
testCertPEM, _ := generateTestCert(t)
|
||||||
|
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch {
|
||||||
|
case r.URL.Path == "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
case strings.HasPrefix(r.URL.Path, "/v1/pki/sign/"):
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
resp := fmt.Sprintf(`{
|
||||||
|
"data": {
|
||||||
|
"certificate": %q,
|
||||||
|
"issuing_ca": %q,
|
||||||
|
"serial_number": "11:22:33:44:55:66",
|
||||||
|
"expiration": 1893456000
|
||||||
|
}
|
||||||
|
}`, testCertPEM, testCertPEM)
|
||||||
|
w.Write([]byte(resp))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
_, csrPEM := generateTestCSR(t, "renew.example.com")
|
||||||
|
renewReq := issuer.RenewalRequest{
|
||||||
|
CommonName: "renew.example.com",
|
||||||
|
CSRPEM: csrPEM,
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := connector.RenewCertificate(ctx, renewReq)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("RenewCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if result.Serial == "" {
|
||||||
|
t.Error("Serial is empty")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RevokeCertificate_Success", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.URL.Path {
|
||||||
|
case "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
case "/v1/pki/revoke":
|
||||||
|
// Verify token
|
||||||
|
if r.Header.Get("X-Vault-Token") == "" {
|
||||||
|
w.WriteHeader(http.StatusForbidden)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(`{"data":{"revocation_time":1234567890}}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
reason := "keyCompromise"
|
||||||
|
revokeReq := issuer.RevocationRequest{
|
||||||
|
Serial: "aa-bb-cc-dd-ee-ff",
|
||||||
|
Reason: &reason,
|
||||||
|
}
|
||||||
|
|
||||||
|
err := connector.RevokeCertificate(ctx, revokeReq)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("RevokeCertificate failed: %v", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("RevokeCertificate_ServerError", func(t *testing.T) {
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.URL.Path {
|
||||||
|
case "/v1/sys/health":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
case "/v1/pki/revoke":
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Write([]byte(`{"errors":["serial not found"]}`))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
revokeReq := issuer.RevocationRequest{
|
||||||
|
Serial: "00-00-00-00",
|
||||||
|
}
|
||||||
|
|
||||||
|
err := connector.RevokeCertificate(ctx, revokeReq)
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected error for server error response")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetCACertPEM_Success", func(t *testing.T) {
|
||||||
|
expectedPEM := "-----BEGIN CERTIFICATE-----\nTESTCA\n-----END CERTIFICATE-----\n"
|
||||||
|
|
||||||
|
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
switch r.URL.Path {
|
||||||
|
case "/v1/pki/ca/pem":
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
w.Write([]byte(expectedPEM))
|
||||||
|
default:
|
||||||
|
http.NotFound(w, r)
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
defer srv.Close()
|
||||||
|
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: srv.URL,
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
caPEM, err := connector.GetCACertPEM(ctx)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetCACertPEM failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if caPEM != expectedPEM {
|
||||||
|
t.Errorf("Expected CA PEM %q, got %q", expectedPEM, caPEM)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetOrderStatus_Synchronous", func(t *testing.T) {
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: "https://vault.example.com:8200",
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
status, err := connector.GetOrderStatus(ctx, "vault-aa-bb-cc")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetOrderStatus failed: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if status.Status != "completed" {
|
||||||
|
t.Errorf("Expected status 'completed', got '%s'", status.Status)
|
||||||
|
}
|
||||||
|
if status.OrderID != "vault-aa-bb-cc" {
|
||||||
|
t.Errorf("Expected OrderID 'vault-aa-bb-cc', got '%s'", status.OrderID)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetRenewalInfo_ReturnsNil", func(t *testing.T) {
|
||||||
|
config := &vault.Config{
|
||||||
|
Addr: "https://vault.example.com:8200",
|
||||||
|
Token: "s.test-token",
|
||||||
|
Mount: "pki",
|
||||||
|
Role: "web-certs",
|
||||||
|
}
|
||||||
|
connector := vault.New(config, logger)
|
||||||
|
|
||||||
|
result, err := connector.GetRenewalInfo(ctx, "-----BEGIN CERTIFICATE-----\ntest\n-----END CERTIFICATE-----")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("GetRenewalInfo should not return error, got: %v", err)
|
||||||
|
}
|
||||||
|
if result != nil {
|
||||||
|
t.Fatal("GetRenewalInfo should return nil for Vault")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// generateTestCert creates a self-signed test certificate and returns the PEM strings.
|
||||||
|
func generateTestCert(t *testing.T) (certPEM string, keyPEM string) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
key, err := rsa.GenerateKey(rand.Reader, 2048)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to generate key: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
serial, _ := rand.Int(rand.Reader, new(big.Int).Lsh(big.NewInt(1), 128))
|
||||||
|
template := &x509.Certificate{
|
||||||
|
SerialNumber: serial,
|
||||||
|
Subject: pkix.Name{
|
||||||
|
CommonName: "Test Certificate",
|
||||||
|
},
|
||||||
|
DNSNames: []string{"test.example.com"},
|
||||||
|
KeyUsage: x509.KeyUsageDigitalSignature,
|
||||||
|
ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth},
|
||||||
|
BasicConstraintsValid: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
certBytes, err := x509.CreateCertificate(rand.Reader, template, template, &key.PublicKey, key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create certificate: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
certPEM = string(pem.EncodeToMemory(&pem.Block{Type: "CERTIFICATE", Bytes: certBytes}))
|
||||||
|
keyPEM = string(pem.EncodeToMemory(&pem.Block{Type: "RSA PRIVATE KEY", Bytes: x509.MarshalPKCS1PrivateKey(key)}))
|
||||||
|
|
||||||
|
return certPEM, keyPEM
|
||||||
|
}
|
||||||
|
|
||||||
|
// generateTestCSR creates a test CSR for the given common name.
|
||||||
|
func generateTestCSR(t *testing.T, commonName string) (*x509.CertificateRequest, string) {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
key, err := rsa.GenerateKey(rand.Reader, 2048)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to generate key: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
csrTemplate := x509.CertificateRequest{
|
||||||
|
Subject: pkix.Name{
|
||||||
|
CommonName: commonName,
|
||||||
|
},
|
||||||
|
DNSNames: []string{commonName},
|
||||||
|
SignatureAlgorithm: x509.SHA256WithRSA,
|
||||||
|
}
|
||||||
|
|
||||||
|
csrBytes, err := x509.CreateCertificateRequest(rand.Reader, &csrTemplate, key)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create CSR: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
csrPEM := string(pem.EncodeToMemory(&pem.Block{
|
||||||
|
Type: "CERTIFICATE REQUEST",
|
||||||
|
Bytes: csrBytes,
|
||||||
|
}))
|
||||||
|
|
||||||
|
csr, err := x509.ParseCertificateRequest(csrBytes)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to parse CSR: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return csr, csrPEM
|
||||||
|
}
|
||||||
@@ -69,6 +69,8 @@ const (
|
|||||||
IssuerTypeGenericCA IssuerType = "GenericCA"
|
IssuerTypeGenericCA IssuerType = "GenericCA"
|
||||||
IssuerTypeStepCA IssuerType = "StepCA"
|
IssuerTypeStepCA IssuerType = "StepCA"
|
||||||
IssuerTypeOpenSSL IssuerType = "OpenSSL"
|
IssuerTypeOpenSSL IssuerType = "OpenSSL"
|
||||||
|
IssuerTypeVault IssuerType = "VaultPKI"
|
||||||
|
IssuerTypeDigiCert IssuerType = "DigiCert"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TargetType represents the type of deployment target.
|
// TargetType represents the type of deployment target.
|
||||||
|
|||||||
@@ -304,6 +304,14 @@ func (s *CertificateService) CreateCertificate(cert domain.ManagedCertificate) (
|
|||||||
if cert.UpdatedAt.IsZero() {
|
if cert.UpdatedAt.IsZero() {
|
||||||
cert.UpdatedAt = now
|
cert.UpdatedAt = now
|
||||||
}
|
}
|
||||||
|
// Default status to Pending if not set (DB column DEFAULT only applies when column is omitted from INSERT)
|
||||||
|
if cert.Status == "" {
|
||||||
|
cert.Status = domain.CertificateStatusPending
|
||||||
|
}
|
||||||
|
// Default tags to empty map if nil (avoids JSON null in JSONB column)
|
||||||
|
if cert.Tags == nil {
|
||||||
|
cert.Tags = make(map[string]string)
|
||||||
|
}
|
||||||
if err := s.certRepo.Create(context.Background(), &cert); err != nil {
|
if err := s.certRepo.Create(context.Background(), &cert); err != nil {
|
||||||
return nil, fmt.Errorf("failed to create certificate: %w", err)
|
return nil, fmt.Errorf("failed to create certificate: %w", err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,7 +43,9 @@ INSERT INTO issuers (id, name, type, config, enabled, created_at, updated_at) VA
|
|||||||
('iss-acme-le', 'Let''s Encrypt Staging', 'acme', '{"directory_url": "https://acme-staging-v02.api.letsencrypt.org/directory", "email": "admin@example.com", "challenge_type": "http-01"}', true, NOW() - INTERVAL '150 days', NOW() - INTERVAL '150 days'),
|
('iss-acme-le', 'Let''s Encrypt Staging', 'acme', '{"directory_url": "https://acme-staging-v02.api.letsencrypt.org/directory", "email": "admin@example.com", "challenge_type": "http-01"}', true, NOW() - INTERVAL '150 days', NOW() - INTERVAL '150 days'),
|
||||||
('iss-stepca', 'step-ca Internal', 'stepca', '{"ca_url": "https://ca.internal:9000", "provisioner_name": "certctl", "validity_days": 90}', true, NOW() - INTERVAL '120 days', NOW() - INTERVAL '120 days'),
|
('iss-stepca', 'step-ca Internal', 'stepca', '{"ca_url": "https://ca.internal:9000", "provisioner_name": "certctl", "validity_days": 90}', true, NOW() - INTERVAL '120 days', NOW() - INTERVAL '120 days'),
|
||||||
('iss-acme-zs', 'ZeroSSL (EAB)', 'acme', '{"directory_url": "https://acme.zerossl.com/v2/DV90", "email": "admin@example.com", "challenge_type": "http-01"}', true, NOW() - INTERVAL '60 days', NOW() - INTERVAL '60 days'),
|
('iss-acme-zs', 'ZeroSSL (EAB)', 'acme', '{"directory_url": "https://acme.zerossl.com/v2/DV90", "email": "admin@example.com", "challenge_type": "http-01"}', true, NOW() - INTERVAL '60 days', NOW() - INTERVAL '60 days'),
|
||||||
('iss-openssl', 'Custom OpenSSL CA', 'openssl', '{"sign_script": "/opt/ca/sign.sh", "timeout_seconds": 30}', false, NOW() - INTERVAL '30 days', NOW() - INTERVAL '30 days')
|
('iss-openssl', 'Custom OpenSSL CA', 'openssl', '{"sign_script": "/opt/ca/sign.sh", "timeout_seconds": 30}', false, NOW() - INTERVAL '30 days', NOW() - INTERVAL '30 days'),
|
||||||
|
('iss-vault', 'HashiCorp Vault PKI', 'VaultPKI', '{"addr": "https://vault.internal:8200", "mount": "pki", "role": "web-certs", "ttl": "8760h"}', true, NOW() - INTERVAL '20 days', NOW() - INTERVAL '20 days'),
|
||||||
|
('iss-digicert', 'DigiCert CertCentral', 'DigiCert', '{"base_url": "https://www.digicert.com/services/v2", "product_type": "ssl_basic"}', true, NOW() - INTERVAL '15 days', NOW() - INTERVAL '15 days')
|
||||||
ON CONFLICT (id) DO NOTHING;
|
ON CONFLICT (id) DO NOTHING;
|
||||||
|
|
||||||
-- ============================================================
|
-- ============================================================
|
||||||
|
|||||||
@@ -83,6 +83,12 @@ import {
|
|||||||
getIssuer,
|
getIssuer,
|
||||||
getTarget,
|
getTarget,
|
||||||
getPrometheusMetrics,
|
getPrometheusMetrics,
|
||||||
|
getCertificateDeployments,
|
||||||
|
getCRL,
|
||||||
|
getOCSPStatus,
|
||||||
|
updateIssuer,
|
||||||
|
updateTarget,
|
||||||
|
getPolicy,
|
||||||
} from './client';
|
} from './client';
|
||||||
|
|
||||||
// Mock global fetch
|
// Mock global fetch
|
||||||
@@ -632,6 +638,50 @@ describe('API Client', () => {
|
|||||||
expect(url).toBe('/api/v1/issuers');
|
expect(url).toBe('/api/v1/issuers');
|
||||||
expect(init.method).toBe('POST');
|
expect(init.method).toBe('POST');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('createIssuer sends correct payload for VaultPKI type', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ id: 'iss-vault', name: 'Vault PKI' }));
|
||||||
|
const vaultPayload = {
|
||||||
|
name: 'Vault PKI',
|
||||||
|
type: 'VaultPKI',
|
||||||
|
config: {
|
||||||
|
addr: 'https://vault.internal:8200',
|
||||||
|
token: 'hvs.test-token',
|
||||||
|
mount: 'pki',
|
||||||
|
role: 'web-certs',
|
||||||
|
ttl: '8760h',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
await createIssuer(vaultPayload);
|
||||||
|
const [url, init] = mockFetch.mock.calls[0];
|
||||||
|
expect(url).toBe('/api/v1/issuers');
|
||||||
|
expect(init.method).toBe('POST');
|
||||||
|
const body = JSON.parse(init.body);
|
||||||
|
expect(body.type).toBe('VaultPKI');
|
||||||
|
expect(body.config.addr).toBe('https://vault.internal:8200');
|
||||||
|
expect(body.config.role).toBe('web-certs');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('createIssuer sends correct payload for DigiCert type', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ id: 'iss-digicert', name: 'DigiCert' }));
|
||||||
|
const digicertPayload = {
|
||||||
|
name: 'DigiCert CertCentral',
|
||||||
|
type: 'DigiCert',
|
||||||
|
config: {
|
||||||
|
api_key: 'test-api-key',
|
||||||
|
org_id: '12345',
|
||||||
|
product_type: 'ssl_basic',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
await createIssuer(digicertPayload);
|
||||||
|
const [url, init] = mockFetch.mock.calls[0];
|
||||||
|
expect(url).toBe('/api/v1/issuers');
|
||||||
|
expect(init.method).toBe('POST');
|
||||||
|
const body = JSON.parse(init.body);
|
||||||
|
expect(body.type).toBe('DigiCert');
|
||||||
|
expect(body.config.org_id).toBe('12345');
|
||||||
|
expect(body.config.product_type).toBe('ssl_basic');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// ─── Audit ──────────────────────────────────────────
|
// ─── Audit ──────────────────────────────────────────
|
||||||
@@ -1106,4 +1156,53 @@ describe('API Client', () => {
|
|||||||
expect(init.headers['Authorization']).toBe('Bearer prom-key');
|
expect(init.headers['Authorization']).toBe('Bearer prom-key');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('Frontend Audit: New API Functions', () => {
|
||||||
|
it('getCertificateDeployments sends GET with cert ID', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ data: [], total: 0 }));
|
||||||
|
await getCertificateDeployments('mc-1');
|
||||||
|
expect(mockFetch.mock.calls[0][0]).toContain('/api/v1/certificates/mc-1/deployments');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('getCRL sends GET to /crl', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ entries: [], total: 0 }));
|
||||||
|
await getCRL();
|
||||||
|
expect(mockFetch.mock.calls[0][0]).toBe('/api/v1/crl');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('getOCSPStatus sends GET with issuer and serial', async () => {
|
||||||
|
const buf = new ArrayBuffer(8);
|
||||||
|
mockFetch.mockReturnValueOnce(
|
||||||
|
Promise.resolve({
|
||||||
|
ok: true,
|
||||||
|
status: 200,
|
||||||
|
arrayBuffer: () => Promise.resolve(buf),
|
||||||
|
} as Response)
|
||||||
|
);
|
||||||
|
await getOCSPStatus('iss-local', 'ABC123');
|
||||||
|
expect(mockFetch.mock.calls[0][0]).toBe('/api/v1/ocsp/iss-local/ABC123');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('updateIssuer sends PUT with data', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ id: 'iss-1', name: 'Updated' }));
|
||||||
|
await updateIssuer('iss-1', { name: 'Updated' });
|
||||||
|
const [url, init] = mockFetch.mock.calls[0];
|
||||||
|
expect(url).toBe('/api/v1/issuers/iss-1');
|
||||||
|
expect(init.method).toBe('PUT');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('updateTarget sends PUT with data', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ id: 't-1', name: 'Updated' }));
|
||||||
|
await updateTarget('t-1', { name: 'Updated' });
|
||||||
|
const [url, init] = mockFetch.mock.calls[0];
|
||||||
|
expect(url).toBe('/api/v1/targets/t-1');
|
||||||
|
expect(init.method).toBe('PUT');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('getPolicy sends GET with policy ID', async () => {
|
||||||
|
mockFetch.mockReturnValueOnce(mockJsonResponse({ id: 'pol-1', name: 'Test' }));
|
||||||
|
await getPolicy('pol-1');
|
||||||
|
expect(mockFetch.mock.calls[0][0]).toBe('/api/v1/policies/pol-1');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -122,6 +122,26 @@ export const exportCertificatePKCS12 = (id: string, password: string = '') => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Certificate Deployments
|
||||||
|
export const getCertificateDeployments = (id: string, params: Record<string, string> = {}) => {
|
||||||
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
||||||
|
return fetchJSON<PaginatedResponse<Job>>(`${BASE}/certificates/${id}/deployments?${qs}`);
|
||||||
|
};
|
||||||
|
|
||||||
|
// CRL / OCSP
|
||||||
|
export const getCRL = () =>
|
||||||
|
fetchJSON<{ version: number; entries: unknown[]; total: number; generated_at: string }>(`${BASE}/crl`);
|
||||||
|
|
||||||
|
export const getOCSPStatus = (issuerId: string, serial: string) => {
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
if (apiKey) headers['Authorization'] = `Bearer ${apiKey}`;
|
||||||
|
return fetch(`${BASE}/ocsp/${issuerId}/${serial}`, { headers })
|
||||||
|
.then(r => {
|
||||||
|
if (!r.ok) throw new Error(`OCSP request failed: ${r.status}`);
|
||||||
|
return r.arrayBuffer();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
// Agents
|
// Agents
|
||||||
export const getAgents = (params: Record<string, string> = {}) => {
|
export const getAgents = (params: Record<string, string> = {}) => {
|
||||||
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
const qs = new URLSearchParams({ page: '1', per_page: '50', ...params }).toString();
|
||||||
@@ -170,6 +190,9 @@ export const createPolicy = (data: Partial<PolicyRule>) =>
|
|||||||
export const updatePolicy = (id: string, data: Partial<PolicyRule>) =>
|
export const updatePolicy = (id: string, data: Partial<PolicyRule>) =>
|
||||||
fetchJSON<PolicyRule>(`${BASE}/policies/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
fetchJSON<PolicyRule>(`${BASE}/policies/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
||||||
|
|
||||||
|
export const getPolicy = (id: string) =>
|
||||||
|
fetchJSON<PolicyRule>(`${BASE}/policies/${id}`);
|
||||||
|
|
||||||
export const deletePolicy = (id: string) =>
|
export const deletePolicy = (id: string) =>
|
||||||
fetchJSON<{ message: string }>(`${BASE}/policies/${id}`, { method: 'DELETE' });
|
fetchJSON<{ message: string }>(`${BASE}/policies/${id}`, { method: 'DELETE' });
|
||||||
|
|
||||||
@@ -188,6 +211,9 @@ export const createIssuer = (data: Partial<Issuer>) =>
|
|||||||
export const testIssuerConnection = (id: string) =>
|
export const testIssuerConnection = (id: string) =>
|
||||||
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}/test`, { method: 'POST' });
|
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}/test`, { method: 'POST' });
|
||||||
|
|
||||||
|
export const updateIssuer = (id: string, data: Partial<Issuer>) =>
|
||||||
|
fetchJSON<Issuer>(`${BASE}/issuers/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
||||||
|
|
||||||
export const deleteIssuer = (id: string) =>
|
export const deleteIssuer = (id: string) =>
|
||||||
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}`, { method: 'DELETE' });
|
fetchJSON<{ message: string }>(`${BASE}/issuers/${id}`, { method: 'DELETE' });
|
||||||
|
|
||||||
@@ -200,6 +226,9 @@ export const getTargets = (params: Record<string, string> = {}) => {
|
|||||||
export const createTarget = (data: Partial<Target>) =>
|
export const createTarget = (data: Partial<Target>) =>
|
||||||
fetchJSON<Target>(`${BASE}/targets`, { method: 'POST', body: JSON.stringify(data) });
|
fetchJSON<Target>(`${BASE}/targets`, { method: 'POST', body: JSON.stringify(data) });
|
||||||
|
|
||||||
|
export const updateTarget = (id: string, data: Partial<Target>) =>
|
||||||
|
fetchJSON<Target>(`${BASE}/targets/${id}`, { method: 'PUT', body: JSON.stringify(data) });
|
||||||
|
|
||||||
export const deleteTarget = (id: string) =>
|
export const deleteTarget = (id: string) =>
|
||||||
fetchJSON<{ message: string }>(`${BASE}/targets/${id}`, { method: 'DELETE' });
|
fetchJSON<{ message: string }>(`${BASE}/targets/${id}`, { method: 'DELETE' });
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,10 @@ export interface Certificate {
|
|||||||
expires_at: string;
|
expires_at: string;
|
||||||
revoked_at?: string;
|
revoked_at?: string;
|
||||||
revocation_reason?: string;
|
revocation_reason?: string;
|
||||||
|
target_ids?: string[];
|
||||||
tags: Record<string, string>;
|
tags: Record<string, string>;
|
||||||
|
last_renewal_at?: string;
|
||||||
|
last_deployment_at?: string;
|
||||||
created_at: string;
|
created_at: string;
|
||||||
updated_at: string;
|
updated_at: string;
|
||||||
}
|
}
|
||||||
@@ -45,6 +48,8 @@ export interface CertificateVersion {
|
|||||||
csr_pem: string;
|
csr_pem: string;
|
||||||
not_before: string;
|
not_before: string;
|
||||||
not_after: string;
|
not_after: string;
|
||||||
|
key_algorithm?: string;
|
||||||
|
key_size?: number;
|
||||||
created_at: string;
|
created_at: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -135,7 +140,10 @@ export interface Issuer {
|
|||||||
type: string;
|
type: string;
|
||||||
config: Record<string, unknown>;
|
config: Record<string, unknown>;
|
||||||
status: string;
|
status: string;
|
||||||
|
/** Backend returns enabled boolean; status is derived from this */
|
||||||
|
enabled: boolean;
|
||||||
created_at: string;
|
created_at: string;
|
||||||
|
updated_at?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Target {
|
export interface Target {
|
||||||
@@ -147,6 +155,7 @@ export interface Target {
|
|||||||
config: Record<string, unknown>;
|
config: Record<string, unknown>;
|
||||||
status: string;
|
status: string;
|
||||||
created_at: string;
|
created_at: string;
|
||||||
|
updated_at?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface KeyAlgorithmRule {
|
export interface KeyAlgorithmRule {
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ export default function Layout() {
|
|||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
<div className="px-5 py-3 border-t border-white/10 flex items-center justify-between">
|
<div className="px-5 py-3 border-t border-white/10 flex items-center justify-between">
|
||||||
<span className="text-[10px] text-brand-300/60 font-mono">v2.0.14</span>
|
<span className="text-[10px] text-brand-300/60 font-mono">v2.0.20</span>
|
||||||
{authRequired && (
|
{authRequired && (
|
||||||
<button
|
<button
|
||||||
onClick={logout}
|
onClick={logout}
|
||||||
|
|||||||
@@ -23,6 +23,9 @@ const statusStyles: Record<string, string> = {
|
|||||||
Unmanaged: 'badge-warning',
|
Unmanaged: 'badge-warning',
|
||||||
Managed: 'badge-success',
|
Managed: 'badge-success',
|
||||||
Dismissed: 'badge-neutral',
|
Dismissed: 'badge-neutral',
|
||||||
|
// Issuer statuses
|
||||||
|
Enabled: 'badge-success',
|
||||||
|
Disabled: 'badge-neutral',
|
||||||
// Notification statuses
|
// Notification statuses
|
||||||
sent: 'badge-success',
|
sent: 'badge-success',
|
||||||
pending: 'badge-warning',
|
pending: 'badge-warning',
|
||||||
|
|||||||
@@ -0,0 +1,56 @@
|
|||||||
|
/**
|
||||||
|
* Full config viewer modal with sensitive field redaction.
|
||||||
|
* Replaces the 60-char truncation in the issuers table.
|
||||||
|
* Reusable for targets in M35 — no IssuersPage-specific imports.
|
||||||
|
*/
|
||||||
|
import { isSensitiveKey } from '../../config/issuerTypes';
|
||||||
|
|
||||||
|
interface ConfigDetailModalProps {
|
||||||
|
title: string;
|
||||||
|
config: Record<string, unknown>;
|
||||||
|
onClose: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ConfigDetailModal({ title, config, onClose }: ConfigDetailModalProps) {
|
||||||
|
const entries = Object.entries(config);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center">
|
||||||
|
<div className="bg-surface border border-surface-border rounded-lg shadow-lg max-w-lg w-full mx-4">
|
||||||
|
<div className="border-b border-surface-border px-6 py-4 flex justify-between items-center">
|
||||||
|
<h2 className="text-lg font-semibold text-ink">{title}</h2>
|
||||||
|
<button onClick={onClose} className="text-ink-muted hover:text-ink transition-colors">
|
||||||
|
✕
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div className="px-6 py-4 max-h-96 overflow-y-auto">
|
||||||
|
{entries.length === 0 ? (
|
||||||
|
<div className="text-sm text-ink-faint py-4 text-center">No configuration data</div>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-0">
|
||||||
|
{entries.map(([key, val]) => {
|
||||||
|
const redacted = isSensitiveKey(key);
|
||||||
|
return (
|
||||||
|
<div key={key} className="flex justify-between py-2 border-b border-surface-border/50">
|
||||||
|
<span className="text-sm text-ink-muted">{key}</span>
|
||||||
|
<span className="text-sm text-ink font-mono text-right max-w-xs break-all">
|
||||||
|
{redacted ? '********' : String(val ?? '')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="border-t border-surface-border px-6 py-4 flex justify-end">
|
||||||
|
<button
|
||||||
|
onClick={onClose}
|
||||||
|
className="px-4 py-2 border border-surface-border rounded text-ink hover:bg-surface-hover transition-colors text-sm font-medium"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,139 @@
|
|||||||
|
/**
|
||||||
|
* Renders config fields from an IssuerTypeConfig.configFields definition.
|
||||||
|
* Handles sensitive field masking. M34 will reuse this directly for its
|
||||||
|
* dynamic config wizard. M35 can reuse it for target config forms.
|
||||||
|
*/
|
||||||
|
import type { ConfigField } from '../../config/issuerTypes';
|
||||||
|
|
||||||
|
interface ConfigFormProps {
|
||||||
|
fields: ConfigField[];
|
||||||
|
values: Record<string, unknown>;
|
||||||
|
onChange: (key: string, value: unknown) => void;
|
||||||
|
/** When true, sensitive fields show as ******** with a "Change" button.
|
||||||
|
* Used in edit mode — empty value means "keep existing". */
|
||||||
|
editMode?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ConfigForm({ fields, values, onChange, editMode }: ConfigFormProps) {
|
||||||
|
return (
|
||||||
|
<div className="space-y-5">
|
||||||
|
{fields.map((field) => (
|
||||||
|
<ConfigFieldInput
|
||||||
|
key={field.key}
|
||||||
|
field={field}
|
||||||
|
value={values[field.key]}
|
||||||
|
onChange={(v) => onChange(field.key, v)}
|
||||||
|
editMode={editMode}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ConfigFieldInput({
|
||||||
|
field,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
editMode,
|
||||||
|
}: {
|
||||||
|
field: ConfigField;
|
||||||
|
value: unknown;
|
||||||
|
onChange: (v: unknown) => void;
|
||||||
|
editMode?: boolean;
|
||||||
|
}) {
|
||||||
|
const inputCls =
|
||||||
|
'w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors';
|
||||||
|
|
||||||
|
// In edit mode, sensitive fields that haven't been touched show as masked
|
||||||
|
if (editMode && field.sensitive && value === undefined) {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<FieldLabel field={field} />
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-sm text-ink-muted font-mono">********</span>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => onChange('')}
|
||||||
|
className="text-xs text-brand-400 hover:text-brand-500"
|
||||||
|
>
|
||||||
|
Change
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (field.type === 'select') {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<FieldLabel field={field} />
|
||||||
|
<select
|
||||||
|
value={(value as string) || ''}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
className={inputCls}
|
||||||
|
>
|
||||||
|
<option value="">Select {field.label}</option>
|
||||||
|
{field.options?.map((opt) => (
|
||||||
|
<option key={opt} value={opt}>{opt}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (field.type === 'textarea') {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<FieldLabel field={field} />
|
||||||
|
<textarea
|
||||||
|
value={(value as string) || ''}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
placeholder={field.placeholder}
|
||||||
|
rows={4}
|
||||||
|
className={`${inputCls} font-mono text-xs`}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (field.type === 'number') {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<FieldLabel field={field} />
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
value={(value as number | string) ?? ''}
|
||||||
|
onChange={(e) => onChange(e.target.value ? parseInt(e.target.value, 10) : '')}
|
||||||
|
placeholder={field.placeholder}
|
||||||
|
className={inputCls}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// text or password
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<FieldLabel field={field} />
|
||||||
|
<input
|
||||||
|
type={field.type === 'password' ? 'password' : 'text'}
|
||||||
|
value={(value as string) || ''}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
placeholder={field.placeholder}
|
||||||
|
className={inputCls}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function FieldLabel({ field }: { field: ConfigField }) {
|
||||||
|
return (
|
||||||
|
<label className="block text-sm font-medium text-ink mb-2">
|
||||||
|
{field.label}
|
||||||
|
{field.required && <span className="text-red-600 ml-1">*</span>}
|
||||||
|
{field.sensitive && (
|
||||||
|
<span className="ml-2 text-xs text-yellow-500 font-normal">sensitive</span>
|
||||||
|
)}
|
||||||
|
</label>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
/**
|
||||||
|
* Issuer type selector grid. Used in both the catalog view and create wizard.
|
||||||
|
* M34 will reuse this for its 3-step wizard (Select Type step).
|
||||||
|
*/
|
||||||
|
import { issuerTypes, type IssuerTypeConfig } from '../../config/issuerTypes';
|
||||||
|
|
||||||
|
interface TypeSelectorProps {
|
||||||
|
onSelect: (typeId: string) => void;
|
||||||
|
/** Filter to only show these type IDs. If not provided, shows all non-comingSoon types. */
|
||||||
|
filterIds?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function TypeSelector({ onSelect, filterIds }: TypeSelectorProps) {
|
||||||
|
const types = filterIds
|
||||||
|
? issuerTypes.filter(t => filterIds.includes(t.id))
|
||||||
|
: issuerTypes.filter(t => !t.comingSoon);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="grid grid-cols-2 gap-4">
|
||||||
|
{types.map((type: IssuerTypeConfig) => (
|
||||||
|
<button
|
||||||
|
key={type.id}
|
||||||
|
onClick={() => onSelect(type.id)}
|
||||||
|
className="p-4 border border-surface-border rounded-lg hover:border-brand-500 hover:bg-opacity-5 transition-all text-left"
|
||||||
|
>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-lg">{type.icon}</span>
|
||||||
|
<span className="font-medium text-ink">{type.name}</span>
|
||||||
|
</div>
|
||||||
|
<div className="text-sm text-ink-muted mt-1">{type.description}</div>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,179 @@
|
|||||||
|
/**
|
||||||
|
* Shared issuer type configuration.
|
||||||
|
* Imported by IssuersPage.tsx (M33), and will be reused by M34 (Dynamic Issuer Config)
|
||||||
|
* for its 3-step wizard config forms.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export interface ConfigField {
|
||||||
|
key: string;
|
||||||
|
label: string;
|
||||||
|
type?: 'text' | 'password' | 'number' | 'select' | 'textarea';
|
||||||
|
placeholder?: string;
|
||||||
|
required: boolean;
|
||||||
|
options?: string[];
|
||||||
|
defaultValue?: string;
|
||||||
|
/** Mark fields that contain secrets (tokens, keys, passwords).
|
||||||
|
* Display as ******** when viewing existing config. M34 will use this
|
||||||
|
* for AES-GCM encryption decisions. */
|
||||||
|
sensitive?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IssuerTypeConfig {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
description: string;
|
||||||
|
icon: string;
|
||||||
|
configFields: ConfigField[];
|
||||||
|
/** If true, this type is not yet implemented — show as "Coming Soon" */
|
||||||
|
comingSoon?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Canonical type label map. Keys match what the backend API returns.
|
||||||
|
* DB stores: local, acme, stepca, openssl, VaultPKI, DigiCert
|
||||||
|
*/
|
||||||
|
export const typeLabels: Record<string, string> = {
|
||||||
|
local: 'Local CA',
|
||||||
|
local_ca: 'Local CA', // backward compat (some frontend references)
|
||||||
|
acme: 'ACME',
|
||||||
|
stepca: 'step-ca',
|
||||||
|
openssl: 'OpenSSL/Custom',
|
||||||
|
VaultPKI: 'Vault PKI',
|
||||||
|
DigiCert: 'DigiCert',
|
||||||
|
manual: 'Manual',
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* All supported issuer types + 2 "Coming Soon" stubs.
|
||||||
|
* Order: most common first, coming-soon last.
|
||||||
|
*/
|
||||||
|
export const issuerTypes: IssuerTypeConfig[] = [
|
||||||
|
{
|
||||||
|
id: 'acme',
|
||||||
|
name: 'ACME',
|
||||||
|
description: "Let's Encrypt, ZeroSSL, or any ACME-compatible CA",
|
||||||
|
icon: '\uD83D\uDD12',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'directory_url', label: 'Directory URL', placeholder: 'https://acme-v02.api.letsencrypt.org/directory', required: true },
|
||||||
|
{ key: 'email', label: 'Email', placeholder: 'admin@example.com', required: true },
|
||||||
|
{ key: 'challenge_type', label: 'Challenge Type', type: 'select', options: ['http-01', 'dns-01', 'dns-persist-01'], required: false, defaultValue: 'http-01' },
|
||||||
|
{ key: 'eab_kid', label: 'EAB Key ID', placeholder: 'External Account Binding Key ID (optional)', required: false },
|
||||||
|
{ key: 'eab_hmac', label: 'EAB HMAC Key', placeholder: 'External Account Binding HMAC key', required: false, type: 'password', sensitive: true },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'local',
|
||||||
|
name: 'Local CA',
|
||||||
|
description: 'Self-signed or subordinate CA for internal certificates',
|
||||||
|
icon: '\uD83C\uDFE0',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'ca_cert_path', label: 'CA Cert Path (optional)', placeholder: '/path/to/ca.crt', required: false },
|
||||||
|
{ key: 'ca_key_path', label: 'CA Key Path (optional)', placeholder: '/path/to/ca.key', required: false, sensitive: true },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'stepca',
|
||||||
|
name: 'step-ca',
|
||||||
|
description: 'Smallstep private CA with JWK provisioner auth',
|
||||||
|
icon: '\uD83D\uDC63',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'ca_url', label: 'CA URL', placeholder: 'https://ca.example.com', required: true },
|
||||||
|
{ key: 'provisioner_name', label: 'Provisioner Name', placeholder: 'my-provisioner', required: true },
|
||||||
|
{ key: 'provisioner_key', label: 'Provisioner Key (JWK)', placeholder: '{...}', type: 'textarea', required: true, sensitive: true },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'VaultPKI',
|
||||||
|
name: 'Vault PKI',
|
||||||
|
description: 'HashiCorp Vault PKI secrets engine',
|
||||||
|
icon: '\uD83D\uDD10',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'addr', label: 'Vault Address', placeholder: 'https://vault.internal:8200', required: true },
|
||||||
|
{ key: 'token', label: 'Vault Token', placeholder: 'hvs.CAES...', required: true, type: 'password', sensitive: true },
|
||||||
|
{ key: 'mount', label: 'PKI Mount Path', placeholder: 'pki', required: false, defaultValue: 'pki' },
|
||||||
|
{ key: 'role', label: 'PKI Role Name', placeholder: 'web-certs', required: true },
|
||||||
|
{ key: 'ttl', label: 'Certificate TTL', placeholder: '8760h', required: false, defaultValue: '8760h' },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'DigiCert',
|
||||||
|
name: 'DigiCert CertCentral',
|
||||||
|
description: 'DigiCert CertCentral for OV/EV certificates',
|
||||||
|
icon: '\uD83C\uDF10',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'api_key', label: 'DigiCert API Key', placeholder: 'Your DigiCert API key', required: true, type: 'password', sensitive: true },
|
||||||
|
{ key: 'org_id', label: 'Organization ID', placeholder: '12345', required: true },
|
||||||
|
{ key: 'product_type', label: 'Product Type', type: 'select', options: ['ssl_basic', 'ssl_plus', 'ssl_wildcard', 'ssl_ev_basic', 'ssl_ev_plus'], required: false, defaultValue: 'ssl_basic' },
|
||||||
|
{ key: 'base_url', label: 'API Base URL Override', placeholder: 'https://www.digicert.com/services/v2', required: false },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'openssl',
|
||||||
|
name: 'OpenSSL/Custom',
|
||||||
|
description: 'Script-based signing with your own CA',
|
||||||
|
icon: '\uD83D\uDD27',
|
||||||
|
configFields: [
|
||||||
|
{ key: 'sign_script', label: 'Sign Script Path', placeholder: '/path/to/sign.sh', required: true },
|
||||||
|
{ key: 'revoke_script', label: 'Revoke Script Path (optional)', placeholder: '/path/to/revoke.sh', required: false },
|
||||||
|
{ key: 'crl_script', label: 'CRL Script Path (optional)', placeholder: '/path/to/crl.sh', required: false },
|
||||||
|
{ key: 'timeout_seconds', label: 'Timeout (seconds)', placeholder: '30', type: 'number', required: false },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'sectigo',
|
||||||
|
name: 'Sectigo',
|
||||||
|
description: 'Sectigo Certificate Manager \u2014 coming soon',
|
||||||
|
icon: '\uD83D\uDCE6',
|
||||||
|
configFields: [],
|
||||||
|
comingSoon: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'entrust',
|
||||||
|
name: 'Entrust',
|
||||||
|
description: 'Entrust Certificate Services \u2014 coming soon',
|
||||||
|
icon: '\uD83D\uDCE6',
|
||||||
|
configFields: [],
|
||||||
|
comingSoon: true,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/** Sensitive config key patterns for redaction in display */
|
||||||
|
const SENSITIVE_PATTERNS = ['password', 'secret', 'token', 'key', 'hmac', 'private'];
|
||||||
|
|
||||||
|
/** Check if a config key should be redacted */
|
||||||
|
export function isSensitiveKey(key: string): boolean {
|
||||||
|
const lower = key.toLowerCase();
|
||||||
|
return SENSITIVE_PATTERNS.some(p => lower.includes(p));
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Redact sensitive values in a config object */
|
||||||
|
export function redactConfig(config: Record<string, unknown>): Record<string, unknown> {
|
||||||
|
return Object.fromEntries(
|
||||||
|
Object.entries(config).map(([k, v]) => [k, isSensitiveKey(k) ? '********' : v])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns catalog status info per issuer type.
|
||||||
|
* M36 (Onboarding) will use this to detect first-run state.
|
||||||
|
*/
|
||||||
|
export function getIssuerCatalogStatus(
|
||||||
|
configuredIssuers: { type: string }[]
|
||||||
|
): { type: IssuerTypeConfig; status: 'connected' | 'available' | 'coming_soon'; count: number }[] {
|
||||||
|
return issuerTypes.map(t => {
|
||||||
|
if (t.comingSoon) {
|
||||||
|
return { type: t, status: 'coming_soon' as const, count: 0 };
|
||||||
|
}
|
||||||
|
// Match both the canonical id and common aliases
|
||||||
|
const aliases: Record<string, string[]> = {
|
||||||
|
local: ['local', 'local_ca'],
|
||||||
|
};
|
||||||
|
const matchIds = aliases[t.id] || [t.id];
|
||||||
|
const matching = configuredIssuers.filter(i => matchIds.includes(i.type));
|
||||||
|
return {
|
||||||
|
type: t,
|
||||||
|
status: matching.length > 0 ? 'connected' as const : 'available' as const,
|
||||||
|
count: matching.length,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -13,6 +13,14 @@ const OS_COLORS: Record<string, string> = {
|
|||||||
unknown: '#64748b',
|
unknown: '#64748b',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const OS_DISPLAY_NAMES: Record<string, string> = {
|
||||||
|
darwin: 'macOS',
|
||||||
|
};
|
||||||
|
|
||||||
|
function displayOS(os: string): string {
|
||||||
|
return OS_DISPLAY_NAMES[os.toLowerCase()] || os;
|
||||||
|
}
|
||||||
|
|
||||||
const STATUS_COLORS: Record<string, string> = {
|
const STATUS_COLORS: Record<string, string> = {
|
||||||
Online: '#10b981',
|
Online: '#10b981',
|
||||||
Offline: '#ef4444',
|
Offline: '#ef4444',
|
||||||
@@ -86,7 +94,7 @@ export default function AgentFleetPage() {
|
|||||||
return acc;
|
return acc;
|
||||||
}, {});
|
}, {});
|
||||||
const osPieData = Object.entries(osDistribution).map(([name, value]) => ({
|
const osPieData = Object.entries(osDistribution).map(([name, value]) => ({
|
||||||
name,
|
name: displayOS(name),
|
||||||
value,
|
value,
|
||||||
fill: OS_COLORS[name.toLowerCase()] || '#64748b',
|
fill: OS_COLORS[name.toLowerCase()] || '#64748b',
|
||||||
}));
|
}));
|
||||||
@@ -216,7 +224,7 @@ export default function AgentFleetPage() {
|
|||||||
style={{ backgroundColor: OS_COLORS[group.os.toLowerCase()] || '#64748b' }}
|
style={{ backgroundColor: OS_COLORS[group.os.toLowerCase()] || '#64748b' }}
|
||||||
/>
|
/>
|
||||||
<h4 className="text-sm font-medium text-ink">
|
<h4 className="text-sm font-medium text-ink">
|
||||||
{group.os} / {group.arch}
|
{displayOS(group.os)} / {group.arch}
|
||||||
</h4>
|
</h4>
|
||||||
<span className="text-xs text-ink-faint">
|
<span className="text-xs text-ink-faint">
|
||||||
{group.agents.length} agent{group.agents.length !== 1 ? 's' : ''}
|
{group.agents.length} agent{group.agents.length !== 1 ? 's' : ''}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
import { getCertificates, createCertificate, triggerRenewal, revokeCertificate, updateCertificate, getOwners } from '../api/client';
|
import { getCertificates, createCertificate, triggerRenewal, revokeCertificate, updateCertificate, getOwners, getProfiles, getIssuers } from '../api/client';
|
||||||
import { REVOCATION_REASONS } from '../api/types';
|
import { REVOCATION_REASONS } from '../api/types';
|
||||||
import PageHeader from '../components/PageHeader';
|
import PageHeader from '../components/PageHeader';
|
||||||
import DataTable from '../components/DataTable';
|
import DataTable from '../components/DataTable';
|
||||||
@@ -16,20 +16,66 @@ function CreateCertificateModal({ onClose, onSuccess }: { onClose: () => void; o
|
|||||||
name: '',
|
name: '',
|
||||||
id: '',
|
id: '',
|
||||||
common_name: '',
|
common_name: '',
|
||||||
|
sans: '',
|
||||||
environment: 'production',
|
environment: 'production',
|
||||||
issuer_id: '',
|
issuer_id: '',
|
||||||
|
certificate_profile_id: '',
|
||||||
owner_id: '',
|
owner_id: '',
|
||||||
team_id: '',
|
team_id: '',
|
||||||
renewal_policy_id: '',
|
renewal_policy_id: '',
|
||||||
|
tags: '',
|
||||||
});
|
});
|
||||||
const [error, setError] = useState('');
|
const [error, setError] = useState('');
|
||||||
|
|
||||||
|
const { data: profilesResp } = useQuery({
|
||||||
|
queryKey: ['profiles'],
|
||||||
|
queryFn: () => getProfiles(),
|
||||||
|
});
|
||||||
|
const { data: issuersResp } = useQuery({
|
||||||
|
queryKey: ['issuers'],
|
||||||
|
queryFn: () => getIssuers(),
|
||||||
|
});
|
||||||
|
const profiles = profilesResp?.data || [];
|
||||||
|
const issuers = issuersResp?.data || [];
|
||||||
|
|
||||||
|
const selectedProfile = profiles.find(p => p.id === form.certificate_profile_id);
|
||||||
|
const ttlLabel = selectedProfile
|
||||||
|
? selectedProfile.max_ttl_seconds < 3600
|
||||||
|
? `${Math.round(selectedProfile.max_ttl_seconds / 60)}m`
|
||||||
|
: selectedProfile.max_ttl_seconds < 86400
|
||||||
|
? `${Math.round(selectedProfile.max_ttl_seconds / 3600)}h`
|
||||||
|
: `${Math.round(selectedProfile.max_ttl_seconds / 86400)}d`
|
||||||
|
: null;
|
||||||
|
|
||||||
const mutation = useMutation({
|
const mutation = useMutation({
|
||||||
mutationFn: () => createCertificate(form),
|
mutationFn: () => {
|
||||||
|
const payload: Record<string, unknown> = { ...form };
|
||||||
|
// Convert comma-separated SANs to array
|
||||||
|
if (form.sans.trim()) {
|
||||||
|
payload.sans = form.sans.split(',').map(s => s.trim()).filter(Boolean);
|
||||||
|
} else {
|
||||||
|
delete payload.sans;
|
||||||
|
}
|
||||||
|
// Convert comma-separated key=value tags to object
|
||||||
|
if (form.tags.trim()) {
|
||||||
|
const tags: Record<string, string> = {};
|
||||||
|
form.tags.split(',').forEach(pair => {
|
||||||
|
const [k, ...v] = pair.split('=');
|
||||||
|
if (k?.trim()) tags[k.trim()] = v.join('=').trim();
|
||||||
|
});
|
||||||
|
payload.tags = tags;
|
||||||
|
} else {
|
||||||
|
delete payload.tags;
|
||||||
|
}
|
||||||
|
return createCertificate(payload);
|
||||||
|
},
|
||||||
onSuccess: () => onSuccess(),
|
onSuccess: () => onSuccess(),
|
||||||
onError: (err: Error) => setError(err.message),
|
onError: (err: Error) => setError(err.message),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const inputClass = "w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20";
|
||||||
|
const selectClass = "w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="fixed inset-0 bg-black/40 flex items-center justify-center z-50" onClick={onClose}>
|
<div className="fixed inset-0 bg-black/40 flex items-center justify-center z-50" onClick={onClose}>
|
||||||
<div className="bg-surface border border-surface-border rounded p-6 w-full max-w-lg shadow-xl" onClick={e => e.stopPropagation()}>
|
<div className="bg-surface border border-surface-border rounded p-6 w-full max-w-lg shadow-xl" onClick={e => e.stopPropagation()}>
|
||||||
@@ -39,57 +85,90 @@ function CreateCertificateModal({ onClose, onSuccess }: { onClose: () => void; o
|
|||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Name *</label>
|
<label className="text-xs text-ink-muted block mb-1">Name *</label>
|
||||||
<input value={form.name} onChange={e => setForm(f => ({ ...f, name: e.target.value }))}
|
<input value={form.name} onChange={e => setForm(f => ({ ...f, name: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="API Production Cert" />
|
placeholder="API Production Cert" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">ID (optional)</label>
|
<label className="text-xs text-ink-muted block mb-1">ID (optional)</label>
|
||||||
<input value={form.id} onChange={e => setForm(f => ({ ...f, id: e.target.value }))}
|
<input value={form.id} onChange={e => setForm(f => ({ ...f, id: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="mc-api-prod (auto-generated if empty)" />
|
placeholder="mc-api-prod (auto-generated if empty)" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Common Name *</label>
|
<label className="text-xs text-ink-muted block mb-1">Common Name *</label>
|
||||||
<input value={form.common_name} onChange={e => setForm(f => ({ ...f, common_name: e.target.value }))}
|
<input value={form.common_name} onChange={e => setForm(f => ({ ...f, common_name: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="api.example.com" />
|
placeholder="api.example.com" />
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="text-xs text-ink-muted block mb-1">SANs (comma-separated)</label>
|
||||||
|
<input value={form.sans} onChange={e => setForm(f => ({ ...f, sans: e.target.value }))}
|
||||||
|
className={inputClass}
|
||||||
|
placeholder="api.example.com, api-v2.example.com" />
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-2 gap-3">
|
||||||
|
<div>
|
||||||
|
<label className="text-xs text-ink-muted block mb-1">Issuer *</label>
|
||||||
|
<select value={form.issuer_id} onChange={e => setForm(f => ({ ...f, issuer_id: e.target.value }))}
|
||||||
|
className={selectClass}>
|
||||||
|
<option value="">Select issuer...</option>
|
||||||
|
{issuers.map(i => (
|
||||||
|
<option key={i.id} value={i.id}>{i.name}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="text-xs text-ink-muted block mb-1">
|
||||||
|
Profile {ttlLabel && <span className="text-brand-400 font-medium">(TTL: {ttlLabel})</span>}
|
||||||
|
</label>
|
||||||
|
<select value={form.certificate_profile_id} onChange={e => setForm(f => ({ ...f, certificate_profile_id: e.target.value }))}
|
||||||
|
className={selectClass}>
|
||||||
|
<option value="">Select profile...</option>
|
||||||
|
{profiles.map(p => (
|
||||||
|
<option key={p.id} value={p.id}>
|
||||||
|
{p.name}{p.max_ttl_seconds ? ` (${p.max_ttl_seconds < 3600 ? `${Math.round(p.max_ttl_seconds / 60)}m` : p.max_ttl_seconds < 86400 ? `${Math.round(p.max_ttl_seconds / 3600)}h` : `${Math.round(p.max_ttl_seconds / 86400)}d`})` : ''}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<div className="grid grid-cols-2 gap-3">
|
<div className="grid grid-cols-2 gap-3">
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Environment</label>
|
<label className="text-xs text-ink-muted block mb-1">Environment</label>
|
||||||
<select value={form.environment} onChange={e => setForm(f => ({ ...f, environment: e.target.value }))}
|
<select value={form.environment} onChange={e => setForm(f => ({ ...f, environment: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink">
|
className={selectClass}>
|
||||||
<option value="production">Production</option>
|
<option value="production">Production</option>
|
||||||
<option value="staging">Staging</option>
|
<option value="staging">Staging</option>
|
||||||
<option value="development">Development</option>
|
<option value="development">Development</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Issuer ID *</label>
|
<label className="text-xs text-ink-muted block mb-1">Policy</label>
|
||||||
<input value={form.issuer_id} onChange={e => setForm(f => ({ ...f, issuer_id: e.target.value }))}
|
<input value={form.renewal_policy_id} onChange={e => setForm(f => ({ ...f, renewal_policy_id: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="iss-local" />
|
placeholder="rp-standard" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="grid grid-cols-3 gap-3">
|
<div className="grid grid-cols-2 gap-3">
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Owner ID</label>
|
<label className="text-xs text-ink-muted block mb-1">Owner</label>
|
||||||
<input value={form.owner_id} onChange={e => setForm(f => ({ ...f, owner_id: e.target.value }))}
|
<input value={form.owner_id} onChange={e => setForm(f => ({ ...f, owner_id: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="o-alice" />
|
placeholder="o-alice" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Team ID</label>
|
<label className="text-xs text-ink-muted block mb-1">Team</label>
|
||||||
<input value={form.team_id} onChange={e => setForm(f => ({ ...f, team_id: e.target.value }))}
|
<input value={form.team_id} onChange={e => setForm(f => ({ ...f, team_id: e.target.value }))}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
className={inputClass}
|
||||||
placeholder="t-platform" />
|
placeholder="t-platform" />
|
||||||
</div>
|
</div>
|
||||||
<div>
|
</div>
|
||||||
<label className="text-xs text-ink-muted block mb-1">Policy ID</label>
|
<div>
|
||||||
<input value={form.renewal_policy_id} onChange={e => setForm(f => ({ ...f, renewal_policy_id: e.target.value }))}
|
<label className="text-xs text-ink-muted block mb-1">Tags</label>
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400 focus:ring-1 focus:ring-brand-400/20"
|
<input value={form.tags} onChange={e => setForm(f => ({ ...f, tags: e.target.value }))}
|
||||||
placeholder="rp-standard" />
|
className={inputClass}
|
||||||
</div>
|
placeholder="env=prod, team=platform, app=api" />
|
||||||
|
<p className="text-xs text-ink-faint mt-0.5">Comma-separated key=value pairs</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex justify-end gap-3 mt-6">
|
<div className="flex justify-end gap-3 mt-6">
|
||||||
@@ -245,15 +324,25 @@ export default function CertificatesPage() {
|
|||||||
const queryClient = useQueryClient();
|
const queryClient = useQueryClient();
|
||||||
const [statusFilter, setStatusFilter] = useState('');
|
const [statusFilter, setStatusFilter] = useState('');
|
||||||
const [envFilter, setEnvFilter] = useState('');
|
const [envFilter, setEnvFilter] = useState('');
|
||||||
|
const [issuerFilter, setIssuerFilter] = useState('');
|
||||||
|
const [ownerFilter, setOwnerFilter] = useState('');
|
||||||
|
const [profileFilter, setProfileFilter] = useState('');
|
||||||
const [showCreate, setShowCreate] = useState(false);
|
const [showCreate, setShowCreate] = useState(false);
|
||||||
const [selectedIds, setSelectedIds] = useState<Set<string>>(new Set());
|
const [selectedIds, setSelectedIds] = useState<Set<string>>(new Set());
|
||||||
const [showBulkRevoke, setShowBulkRevoke] = useState(false);
|
const [showBulkRevoke, setShowBulkRevoke] = useState(false);
|
||||||
const [showBulkReassign, setShowBulkReassign] = useState(false);
|
const [showBulkReassign, setShowBulkReassign] = useState(false);
|
||||||
const [bulkRenewProgress, setBulkRenewProgress] = useState<{ done: number; total: number; running: boolean } | null>(null);
|
const [bulkRenewProgress, setBulkRenewProgress] = useState<{ done: number; total: number; running: boolean } | null>(null);
|
||||||
|
|
||||||
|
const { data: issuersData } = useQuery({ queryKey: ['issuers-filter'], queryFn: () => getIssuers({ per_page: '100' }) });
|
||||||
|
const { data: ownersData } = useQuery({ queryKey: ['owners-filter'], queryFn: () => getOwners({ per_page: '100' }) });
|
||||||
|
const { data: profilesData } = useQuery({ queryKey: ['profiles-filter'], queryFn: () => getProfiles({ per_page: '100' }) });
|
||||||
|
|
||||||
const params: Record<string, string> = {};
|
const params: Record<string, string> = {};
|
||||||
if (statusFilter) params.status = statusFilter;
|
if (statusFilter) params.status = statusFilter;
|
||||||
if (envFilter) params.environment = envFilter;
|
if (envFilter) params.environment = envFilter;
|
||||||
|
if (issuerFilter) params.issuer_id = issuerFilter;
|
||||||
|
if (ownerFilter) params.owner_id = ownerFilter;
|
||||||
|
if (profileFilter) params.profile_id = profileFilter;
|
||||||
|
|
||||||
const { data, isLoading, error, refetch } = useQuery({
|
const { data, isLoading, error, refetch } = useQuery({
|
||||||
queryKey: ['certificates', params],
|
queryKey: ['certificates', params],
|
||||||
@@ -302,7 +391,8 @@ export default function CertificatesPage() {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{ key: 'env', label: 'Environment', render: (c) => <span className="text-ink-muted">{c.environment || '—'}</span> },
|
{ key: 'last_renewal', label: 'Last Renewal', render: (c) => <span className="text-xs text-ink-muted">{c.last_renewal_at ? formatDate(c.last_renewal_at) : '—'}</span> },
|
||||||
|
{ key: 'last_deploy', label: 'Last Deploy', render: (c) => <span className="text-xs text-ink-muted">{c.last_deployment_at ? formatDate(c.last_deployment_at) : '—'}</span> },
|
||||||
{ key: 'issuer', label: 'Issuer', render: (c) => <span className="text-ink-muted text-xs">{c.issuer_id}</span> },
|
{ key: 'issuer', label: 'Issuer', render: (c) => <span className="text-ink-muted text-xs">{c.issuer_id}</span> },
|
||||||
{ key: 'owner', label: 'Owner', render: (c) => <span className="text-ink-muted text-xs">{c.owner_id}</span> },
|
{ key: 'owner', label: 'Owner', render: (c) => <span className="text-ink-muted text-xs">{c.owner_id}</span> },
|
||||||
];
|
];
|
||||||
@@ -382,6 +472,36 @@ export default function CertificatesPage() {
|
|||||||
<option value="staging">Staging</option>
|
<option value="staging">Staging</option>
|
||||||
<option value="development">Development</option>
|
<option value="development">Development</option>
|
||||||
</select>
|
</select>
|
||||||
|
<select
|
||||||
|
value={issuerFilter}
|
||||||
|
onChange={e => setIssuerFilter(e.target.value)}
|
||||||
|
className="bg-white border border-surface-border rounded px-3 py-1.5 text-sm text-ink"
|
||||||
|
>
|
||||||
|
<option value="">All issuers</option>
|
||||||
|
{issuersData?.data?.map(i => (
|
||||||
|
<option key={i.id} value={i.id}>{i.name}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
<select
|
||||||
|
value={ownerFilter}
|
||||||
|
onChange={e => setOwnerFilter(e.target.value)}
|
||||||
|
className="bg-white border border-surface-border rounded px-3 py-1.5 text-sm text-ink"
|
||||||
|
>
|
||||||
|
<option value="">All owners</option>
|
||||||
|
{ownersData?.data?.map(o => (
|
||||||
|
<option key={o.id} value={o.id}>{o.name}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
<select
|
||||||
|
value={profileFilter}
|
||||||
|
onChange={e => setProfileFilter(e.target.value)}
|
||||||
|
className="bg-white border border-surface-border rounded px-3 py-1.5 text-sm text-ink"
|
||||||
|
>
|
||||||
|
<option value="">All profiles</option>
|
||||||
|
{profilesData?.data?.map(p => (
|
||||||
|
<option key={p.id} value={p.id}>{p.name}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex-1 overflow-y-auto">
|
<div className="flex-1 overflow-y-auto">
|
||||||
{error ? (
|
{error ? (
|
||||||
|
|||||||
@@ -197,6 +197,18 @@ export default function DiscoveryPage() {
|
|||||||
label: 'Expiry',
|
label: 'Expiry',
|
||||||
render: (c) => <span className="text-xs">{formatExpiry(c.not_after)}</span>,
|
render: (c) => <span className="text-xs">{formatExpiry(c.not_after)}</span>,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: 'key_info',
|
||||||
|
label: 'Key',
|
||||||
|
render: (c) => (
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
<span className="text-xs text-ink-muted">{c.key_algorithm}{c.key_size ? ` ${c.key_size}` : ''}</span>
|
||||||
|
{c.is_ca && (
|
||||||
|
<span className="text-[10px] px-1.5 py-0.5 rounded bg-purple-100 text-purple-700 font-medium">CA</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: 'fingerprint',
|
key: 'fingerprint',
|
||||||
label: 'Fingerprint',
|
label: 'Fingerprint',
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useParams } from 'react-router-dom';
|
import { useParams, useNavigate } from 'react-router-dom';
|
||||||
import { useQuery, useMutation } from '@tanstack/react-query';
|
import { useQuery, useMutation } from '@tanstack/react-query';
|
||||||
import { getIssuer, testIssuerConnection, getCertificates } from '../api/client';
|
import { getIssuer, testIssuerConnection, getCertificates } from '../api/client';
|
||||||
import PageHeader from '../components/PageHeader';
|
import PageHeader from '../components/PageHeader';
|
||||||
@@ -7,15 +7,8 @@ import DataTable from '../components/DataTable';
|
|||||||
import type { Column } from '../components/DataTable';
|
import type { Column } from '../components/DataTable';
|
||||||
import ErrorState from '../components/ErrorState';
|
import ErrorState from '../components/ErrorState';
|
||||||
import { formatDateTime } from '../api/utils';
|
import { formatDateTime } from '../api/utils';
|
||||||
import type { Certificate } from '../api/types';
|
import type { Certificate, Issuer } from '../api/types';
|
||||||
|
import { typeLabels, redactConfig } from '../config/issuerTypes';
|
||||||
const typeLabels: Record<string, string> = {
|
|
||||||
local_ca: 'Local CA',
|
|
||||||
acme: 'ACME (Let\'s Encrypt)',
|
|
||||||
step_ca: 'step-ca',
|
|
||||||
openssl: 'OpenSSL / Custom',
|
|
||||||
vault: 'Vault PKI',
|
|
||||||
};
|
|
||||||
|
|
||||||
function InfoRow({ label, value }: { label: string; value: React.ReactNode }) {
|
function InfoRow({ label, value }: { label: string; value: React.ReactNode }) {
|
||||||
return (
|
return (
|
||||||
@@ -26,8 +19,17 @@ function InfoRow({ label, value }: { label: string; value: React.ReactNode }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Derive display status from backend enabled boolean */
|
||||||
|
function issuerStatus(issuer: Issuer): string {
|
||||||
|
if (issuer.enabled !== undefined) {
|
||||||
|
return issuer.enabled ? 'Enabled' : 'Disabled';
|
||||||
|
}
|
||||||
|
return issuer.status || 'Unknown';
|
||||||
|
}
|
||||||
|
|
||||||
export default function IssuerDetailPage() {
|
export default function IssuerDetailPage() {
|
||||||
const { id } = useParams<{ id: string }>();
|
const { id } = useParams<{ id: string }>();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
|
||||||
const { data: issuer, isLoading, error, refetch } = useQuery({
|
const { data: issuer, isLoading, error, refetch } = useQuery({
|
||||||
queryKey: ['issuer', id],
|
queryKey: ['issuer', id],
|
||||||
@@ -65,13 +67,7 @@ export default function IssuerDetailPage() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Redact sensitive config fields
|
const safeConfig = issuer.config ? redactConfig(issuer.config) : {};
|
||||||
const safeConfig = issuer.config ? Object.fromEntries(
|
|
||||||
Object.entries(issuer.config).map(([k, v]) => {
|
|
||||||
const sensitive = ['password', 'secret', 'token', 'key', 'hmac', 'private'].some(s => k.toLowerCase().includes(s));
|
|
||||||
return [k, sensitive ? '********' : v];
|
|
||||||
})
|
|
||||||
) : {};
|
|
||||||
|
|
||||||
const certColumns: Column<Certificate>[] = [
|
const certColumns: Column<Certificate>[] = [
|
||||||
{
|
{
|
||||||
@@ -94,13 +90,21 @@ export default function IssuerDetailPage() {
|
|||||||
title={issuer.name}
|
title={issuer.name}
|
||||||
subtitle={typeLabels[issuer.type] || issuer.type}
|
subtitle={typeLabels[issuer.type] || issuer.type}
|
||||||
action={
|
action={
|
||||||
<button
|
<div className="flex gap-2">
|
||||||
onClick={() => testMutation.mutate()}
|
<button
|
||||||
disabled={testMutation.isPending}
|
onClick={() => navigate(`/issuers?edit=${issuer.id}`)}
|
||||||
className="btn btn-primary text-xs disabled:opacity-50"
|
className="px-3 py-1.5 border border-surface-border rounded text-ink text-xs hover:bg-surface-hover transition-colors font-medium"
|
||||||
>
|
>
|
||||||
{testMutation.isPending ? 'Testing...' : 'Test Connection'}
|
Edit
|
||||||
</button>
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => testMutation.mutate()}
|
||||||
|
disabled={testMutation.isPending}
|
||||||
|
className="btn btn-primary text-xs disabled:opacity-50"
|
||||||
|
>
|
||||||
|
{testMutation.isPending ? 'Testing...' : 'Test Connection'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -123,7 +127,7 @@ export default function IssuerDetailPage() {
|
|||||||
<InfoRow label="ID" value={<span className="font-mono text-xs">{issuer.id}</span>} />
|
<InfoRow label="ID" value={<span className="font-mono text-xs">{issuer.id}</span>} />
|
||||||
<InfoRow label="Name" value={issuer.name} />
|
<InfoRow label="Name" value={issuer.name} />
|
||||||
<InfoRow label="Type" value={typeLabels[issuer.type] || issuer.type} />
|
<InfoRow label="Type" value={typeLabels[issuer.type] || issuer.type} />
|
||||||
<InfoRow label="Status" value={<StatusBadge status={issuer.status} />} />
|
<InfoRow label="Status" value={<StatusBadge status={issuerStatus(issuer)} />} />
|
||||||
<InfoRow label="Created" value={formatDateTime(issuer.created_at)} />
|
<InfoRow label="Created" value={formatDateTime(issuer.created_at)} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
+202
-209
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useState, useMemo } from 'react';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
||||||
import { getIssuers, testIssuerConnection, deleteIssuer, createIssuer } from '../api/client';
|
import { getIssuers, testIssuerConnection, deleteIssuer, createIssuer } from '../api/client';
|
||||||
@@ -9,83 +9,27 @@ import StatusBadge from '../components/StatusBadge';
|
|||||||
import ErrorState from '../components/ErrorState';
|
import ErrorState from '../components/ErrorState';
|
||||||
import { formatDateTime } from '../api/utils';
|
import { formatDateTime } from '../api/utils';
|
||||||
import type { Issuer } from '../api/types';
|
import type { Issuer } from '../api/types';
|
||||||
|
import { issuerTypes, typeLabels, getIssuerCatalogStatus, type IssuerTypeConfig } from '../config/issuerTypes';
|
||||||
|
import TypeSelector from '../components/issuer/TypeSelector';
|
||||||
|
import ConfigForm from '../components/issuer/ConfigForm';
|
||||||
|
import ConfigDetailModal from '../components/issuer/ConfigDetailModal';
|
||||||
|
|
||||||
const typeLabels: Record<string, string> = {
|
/** Derive display status from backend enabled boolean */
|
||||||
local_ca: 'Local CA',
|
function issuerStatus(issuer: Issuer): string {
|
||||||
acme: 'ACME',
|
if (issuer.enabled !== undefined) {
|
||||||
stepca: 'step-ca',
|
return issuer.enabled ? 'Enabled' : 'Disabled';
|
||||||
openssl: 'OpenSSL/Custom',
|
}
|
||||||
vault: 'Vault PKI',
|
// Fallback for legacy data that may have status string
|
||||||
manual: 'Manual',
|
return issuer.status || 'Unknown';
|
||||||
};
|
|
||||||
|
|
||||||
interface IssuerConfigField {
|
|
||||||
key: string;
|
|
||||||
label: string;
|
|
||||||
placeholder?: string;
|
|
||||||
required: boolean;
|
|
||||||
type?: string;
|
|
||||||
options?: string[];
|
|
||||||
defaultValue?: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IssuerTypeConfig {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
description: string;
|
|
||||||
configFields: IssuerConfigField[];
|
|
||||||
}
|
|
||||||
|
|
||||||
const issuerTypes: IssuerTypeConfig[] = [
|
|
||||||
{
|
|
||||||
id: 'local_ca',
|
|
||||||
name: 'Local CA',
|
|
||||||
description: 'Self-signed or subordinate CA for certificate issuance',
|
|
||||||
configFields: [
|
|
||||||
{ key: 'ca_cert_path', label: 'CA Cert Path (optional)', placeholder: '/path/to/ca.crt', required: false },
|
|
||||||
{ key: 'ca_key_path', label: 'CA Key Path (optional)', placeholder: '/path/to/ca.key', required: false },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'acme',
|
|
||||||
name: 'ACME',
|
|
||||||
description: "Let's Encrypt or other ACME-compatible CA",
|
|
||||||
configFields: [
|
|
||||||
{ key: 'directory_url', label: 'Directory URL', placeholder: 'https://acme-v02.api.letsencrypt.org/directory', required: true },
|
|
||||||
{ key: 'email', label: 'Email', placeholder: 'admin@example.com', required: true },
|
|
||||||
{ key: 'challenge_type', label: 'Challenge Type', type: 'select', options: ['http-01', 'dns-01', 'dns-persist-01'], required: false, defaultValue: 'http-01' },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'stepca',
|
|
||||||
name: 'step-ca',
|
|
||||||
description: 'Smallstep private CA',
|
|
||||||
configFields: [
|
|
||||||
{ key: 'ca_url', label: 'CA URL', placeholder: 'https://ca.example.com', required: true },
|
|
||||||
{ key: 'provisioner_name', label: 'Provisioner Name', placeholder: 'my-provisioner', required: true },
|
|
||||||
{ key: 'provisioner_key', label: 'Provisioner Key (JWK)', placeholder: '{...}', type: 'textarea', required: true },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'openssl',
|
|
||||||
name: 'OpenSSL/Custom',
|
|
||||||
description: 'Script-based signing with your own CA',
|
|
||||||
configFields: [
|
|
||||||
{ key: 'sign_script', label: 'Sign Script Path', placeholder: '/path/to/sign.sh', required: true },
|
|
||||||
{ key: 'revoke_script', label: 'Revoke Script Path (optional)', placeholder: '/path/to/revoke.sh', required: false },
|
|
||||||
{ key: 'crl_script', label: 'CRL Script Path (optional)', placeholder: '/path/to/crl.sh', required: false },
|
|
||||||
{ key: 'timeout_seconds', label: 'Timeout (seconds)', placeholder: '30', type: 'number', required: false },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
export default function IssuersPage() {
|
export default function IssuersPage() {
|
||||||
const queryClient = useQueryClient();
|
const queryClient = useQueryClient();
|
||||||
const [testResult, setTestResult] = useState<{ id: string; ok: boolean; msg: string } | null>(null);
|
const [testResult, setTestResult] = useState<{ id: string; ok: boolean; msg: string } | null>(null);
|
||||||
const [showCreateModal, setShowCreateModal] = useState(false);
|
const [showCreateModal, setShowCreateModal] = useState(false);
|
||||||
const [createStep, setCreateStep] = useState<'type' | 'config'>('type');
|
const [preselectedType, setPreselectedType] = useState<string | null>(null);
|
||||||
const [selectedType, setSelectedType] = useState<string | null>(null);
|
const [typeFilter, setTypeFilter] = useState<string>('');
|
||||||
const [createForm, setCreateForm] = useState<Record<string, unknown>>({});
|
const [configModal, setConfigModal] = useState<{ title: string; config: Record<string, unknown> } | null>(null);
|
||||||
|
|
||||||
const { data, isLoading, error, refetch } = useQuery({
|
const { data, isLoading, error, refetch } = useQuery({
|
||||||
queryKey: ['issuers'],
|
queryKey: ['issuers'],
|
||||||
@@ -109,12 +53,22 @@ export default function IssuersPage() {
|
|||||||
onSuccess: () => {
|
onSuccess: () => {
|
||||||
queryClient.invalidateQueries({ queryKey: ['issuers'] });
|
queryClient.invalidateQueries({ queryKey: ['issuers'] });
|
||||||
setShowCreateModal(false);
|
setShowCreateModal(false);
|
||||||
setCreateStep('type');
|
setPreselectedType(null);
|
||||||
setSelectedType(null);
|
|
||||||
setCreateForm({});
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const catalogStatus = useMemo(
|
||||||
|
() => getIssuerCatalogStatus(data?.data || []),
|
||||||
|
[data?.data]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Filter issuers by type
|
||||||
|
const filteredIssuers = useMemo(() => {
|
||||||
|
if (!data?.data) return [];
|
||||||
|
if (!typeFilter) return data.data;
|
||||||
|
return data.data.filter(i => i.type === typeFilter);
|
||||||
|
}, [data?.data, typeFilter]);
|
||||||
|
|
||||||
const columns: Column<Issuer>[] = [
|
const columns: Column<Issuer>[] = [
|
||||||
{
|
{
|
||||||
key: 'name',
|
key: 'name',
|
||||||
@@ -138,7 +92,7 @@ export default function IssuersPage() {
|
|||||||
{
|
{
|
||||||
key: 'status',
|
key: 'status',
|
||||||
label: 'Status',
|
label: 'Status',
|
||||||
render: (i) => <StatusBadge status={i.status} />,
|
render: (i) => <StatusBadge status={issuerStatus(i)} />,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
key: 'config',
|
key: 'config',
|
||||||
@@ -146,9 +100,15 @@ export default function IssuersPage() {
|
|||||||
render: (i) => {
|
render: (i) => {
|
||||||
if (!i.config || Object.keys(i.config).length === 0) return <span className="text-ink-faint">—</span>;
|
if (!i.config || Object.keys(i.config).length === 0) return <span className="text-ink-faint">—</span>;
|
||||||
return (
|
return (
|
||||||
<span className="text-xs text-ink-muted font-mono truncate max-w-xs block">
|
<button
|
||||||
{JSON.stringify(i.config).slice(0, 60)}
|
onClick={(e) => {
|
||||||
</span>
|
e.stopPropagation();
|
||||||
|
setConfigModal({ title: `${i.name} Configuration`, config: i.config });
|
||||||
|
}}
|
||||||
|
className="text-xs text-brand-400 hover:text-brand-500 transition-colors"
|
||||||
|
>
|
||||||
|
View Config
|
||||||
|
</button>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -184,14 +144,12 @@ export default function IssuersPage() {
|
|||||||
<>
|
<>
|
||||||
<PageHeader
|
<PageHeader
|
||||||
title="Issuers"
|
title="Issuers"
|
||||||
subtitle={data ? `${data.total} issuers` : undefined}
|
subtitle={data ? `${data.total} configured` : undefined}
|
||||||
action={
|
action={
|
||||||
<button
|
<button
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
|
setPreselectedType(null);
|
||||||
setShowCreateModal(true);
|
setShowCreateModal(true);
|
||||||
setCreateStep('type');
|
|
||||||
setSelectedType(null);
|
|
||||||
setCreateForm({});
|
|
||||||
}}
|
}}
|
||||||
className="px-4 py-2 bg-brand-600 text-white rounded font-medium hover:bg-brand-700 transition-colors text-sm"
|
className="px-4 py-2 bg-brand-600 text-white rounded font-medium hover:bg-brand-700 transition-colors text-sm"
|
||||||
>
|
>
|
||||||
@@ -205,49 +163,83 @@ export default function IssuersPage() {
|
|||||||
<button onClick={() => setTestResult(null)} className="ml-3 text-xs opacity-60 hover:opacity-100">dismiss</button>
|
<button onClick={() => setTestResult(null)} className="ml-3 text-xs opacity-60 hover:opacity-100">dismiss</button>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div className="flex-1 overflow-y-auto">
|
<div className="flex-1 overflow-y-auto">
|
||||||
{error ? (
|
{error ? (
|
||||||
<ErrorState error={error as Error} onRetry={() => refetch()} />
|
<ErrorState error={error as Error} onRetry={() => refetch()} />
|
||||||
) : (
|
) : (
|
||||||
<DataTable columns={columns} data={data?.data || []} isLoading={isLoading} emptyMessage="No issuers configured" />
|
<>
|
||||||
|
{/* Issuer Type Catalog Cards */}
|
||||||
|
<div className="px-6 py-4">
|
||||||
|
<h3 className="text-sm font-semibold text-ink-muted mb-3">Issuer Types</h3>
|
||||||
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-3">
|
||||||
|
{catalogStatus.map(({ type, status, count }) => (
|
||||||
|
<CatalogCard
|
||||||
|
key={type.id}
|
||||||
|
type={type}
|
||||||
|
status={status}
|
||||||
|
count={count}
|
||||||
|
onConfigure={() => {
|
||||||
|
setPreselectedType(type.id);
|
||||||
|
setShowCreateModal(true);
|
||||||
|
}}
|
||||||
|
onFilter={() => {
|
||||||
|
// Match both the canonical id and aliases
|
||||||
|
const filterValue = type.id === 'local' ? 'local' : type.id;
|
||||||
|
setTypeFilter(prev => prev === filterValue ? '' : filterValue);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Configured Issuers Table */}
|
||||||
|
<div className="px-6 pb-4">
|
||||||
|
<div className="flex items-center justify-between mb-3">
|
||||||
|
<h3 className="text-sm font-semibold text-ink-muted">Configured Issuers</h3>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<select
|
||||||
|
value={typeFilter}
|
||||||
|
onChange={(e) => setTypeFilter(e.target.value)}
|
||||||
|
className="text-xs px-2 py-1.5 bg-surface border border-surface-border rounded text-ink focus:outline-none focus:border-brand-500"
|
||||||
|
>
|
||||||
|
<option value="">All Types</option>
|
||||||
|
{issuerTypes.filter(t => !t.comingSoon).map(t => (
|
||||||
|
<option key={t.id} value={t.id}>{t.name}</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<DataTable
|
||||||
|
columns={columns}
|
||||||
|
data={filteredIssuers}
|
||||||
|
isLoading={isLoading}
|
||||||
|
emptyMessage={typeFilter ? `No ${typeLabels[typeFilter] || typeFilter} issuers configured` : 'No issuers configured'}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Config Detail Modal */}
|
||||||
|
{configModal && (
|
||||||
|
<ConfigDetailModal
|
||||||
|
title={configModal.title}
|
||||||
|
config={configModal.config}
|
||||||
|
onClose={() => setConfigModal(null)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Create Issuer Modal */}
|
||||||
{showCreateModal && (
|
{showCreateModal && (
|
||||||
<CreateIssuerModal
|
<CreateIssuerModal
|
||||||
step={createStep}
|
preselectedType={preselectedType}
|
||||||
selectedType={selectedType}
|
onSubmit={(name, type, config) => {
|
||||||
form={createForm}
|
createMutation.mutate({ name, type, config });
|
||||||
onTypeSelect={(type) => {
|
|
||||||
setSelectedType(type);
|
|
||||||
const typeConfig = issuerTypes.find((t) => t.id === type);
|
|
||||||
const defaultConfig: Record<string, unknown> = {};
|
|
||||||
if (typeConfig) {
|
|
||||||
typeConfig.configFields.forEach((field) => {
|
|
||||||
if (field.defaultValue) {
|
|
||||||
defaultConfig[field.key] = field.defaultValue;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
setCreateForm({ ...defaultConfig });
|
|
||||||
setCreateStep('config');
|
|
||||||
}}
|
|
||||||
onFormChange={(field, value) => {
|
|
||||||
setCreateForm({ ...createForm, [field]: value });
|
|
||||||
}}
|
|
||||||
onBack={() => setCreateStep('type')}
|
|
||||||
onSubmit={() => {
|
|
||||||
if (!selectedType || !createForm.name) return;
|
|
||||||
const config: Record<string, unknown> = { ...createForm };
|
|
||||||
const name = config.name as string;
|
|
||||||
delete config.name;
|
|
||||||
createMutation.mutate({ name, type: selectedType, config });
|
|
||||||
}}
|
}}
|
||||||
onCancel={() => {
|
onCancel={() => {
|
||||||
setShowCreateModal(false);
|
setShowCreateModal(false);
|
||||||
setCreateStep('type');
|
setPreselectedType(null);
|
||||||
setSelectedType(null);
|
|
||||||
setCreateForm({});
|
|
||||||
}}
|
}}
|
||||||
isSubmitting={createMutation.isPending}
|
isSubmitting={createMutation.isPending}
|
||||||
/>
|
/>
|
||||||
@@ -256,30 +248,94 @@ export default function IssuersPage() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ─── Catalog Card ───────────────────────────────────────────────
|
||||||
|
|
||||||
|
interface CatalogCardProps {
|
||||||
|
type: IssuerTypeConfig;
|
||||||
|
status: 'connected' | 'available' | 'coming_soon';
|
||||||
|
count: number;
|
||||||
|
onConfigure: () => void;
|
||||||
|
onFilter: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
function CatalogCard({ type, status, count, onConfigure, onFilter }: CatalogCardProps) {
|
||||||
|
const statusConfig = {
|
||||||
|
connected: { label: `${count} configured`, cls: 'bg-emerald-500/10 text-emerald-400 border-emerald-500/30' },
|
||||||
|
available: { label: 'Available', cls: 'bg-brand-500/10 text-brand-400 border-brand-500/30' },
|
||||||
|
coming_soon: { label: 'Coming Soon', cls: 'bg-gray-500/10 text-gray-400 border-gray-500/30' },
|
||||||
|
};
|
||||||
|
const { label, cls } = statusConfig[status];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={`p-4 border rounded-lg ${status === 'coming_soon' ? 'border-surface-border/50 opacity-60' : 'border-surface-border'}`}>
|
||||||
|
<div className="flex items-start justify-between mb-2">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="text-lg">{type.icon}</span>
|
||||||
|
<span className="font-medium text-ink text-sm">{type.name}</span>
|
||||||
|
</div>
|
||||||
|
<span className={`text-xs px-2 py-0.5 rounded-full border ${cls}`}>{label}</span>
|
||||||
|
</div>
|
||||||
|
<p className="text-xs text-ink-muted mb-3">{type.description}</p>
|
||||||
|
{status === 'connected' && (
|
||||||
|
<button
|
||||||
|
onClick={onFilter}
|
||||||
|
className="text-xs text-brand-400 hover:text-brand-500 transition-colors"
|
||||||
|
>
|
||||||
|
View issuers
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
{status === 'available' && (
|
||||||
|
<button
|
||||||
|
onClick={onConfigure}
|
||||||
|
className="text-xs px-3 py-1 bg-brand-600 text-white rounded hover:bg-brand-700 transition-colors"
|
||||||
|
>
|
||||||
|
Configure
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ─── Create Issuer Modal ────────────────────────────────────────
|
||||||
|
|
||||||
interface CreateIssuerModalProps {
|
interface CreateIssuerModalProps {
|
||||||
step: 'type' | 'config';
|
preselectedType: string | null;
|
||||||
selectedType: string | null;
|
onSubmit: (name: string, type: string, config: Record<string, unknown>) => void;
|
||||||
form: Record<string, unknown>;
|
|
||||||
onTypeSelect: (type: string) => void;
|
|
||||||
onFormChange: (field: string, value: unknown) => void;
|
|
||||||
onBack: () => void;
|
|
||||||
onSubmit: () => void;
|
|
||||||
onCancel: () => void;
|
onCancel: () => void;
|
||||||
isSubmitting: boolean;
|
isSubmitting: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
function CreateIssuerModal({
|
function CreateIssuerModal({ preselectedType, onSubmit, onCancel, isSubmitting }: CreateIssuerModalProps) {
|
||||||
step,
|
const [step, setStep] = useState<'type' | 'config'>(preselectedType ? 'config' : 'type');
|
||||||
selectedType,
|
const [selectedType, setSelectedType] = useState<string | null>(preselectedType);
|
||||||
form,
|
const [form, setForm] = useState<Record<string, unknown>>(() => {
|
||||||
onTypeSelect,
|
if (preselectedType) {
|
||||||
onFormChange,
|
const tc = issuerTypes.find(t => t.id === preselectedType);
|
||||||
onBack,
|
const defaults: Record<string, unknown> = {};
|
||||||
onSubmit,
|
tc?.configFields.forEach(f => { if (f.defaultValue) defaults[f.key] = f.defaultValue; });
|
||||||
onCancel,
|
return defaults;
|
||||||
isSubmitting,
|
}
|
||||||
}: CreateIssuerModalProps) {
|
return {};
|
||||||
const selectedTypeConfig = issuerTypes.find((t) => t.id === selectedType);
|
});
|
||||||
|
|
||||||
|
const selectedTypeConfig = issuerTypes.find(t => t.id === selectedType);
|
||||||
|
|
||||||
|
function handleTypeSelect(typeId: string) {
|
||||||
|
setSelectedType(typeId);
|
||||||
|
const tc = issuerTypes.find(t => t.id === typeId);
|
||||||
|
const defaults: Record<string, unknown> = {};
|
||||||
|
tc?.configFields.forEach(f => { if (f.defaultValue) defaults[f.key] = f.defaultValue; });
|
||||||
|
setForm(defaults);
|
||||||
|
setStep('config');
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSubmit() {
|
||||||
|
if (!selectedType || !form.name) return;
|
||||||
|
const config = { ...form };
|
||||||
|
const name = config.name as string;
|
||||||
|
delete config.name;
|
||||||
|
onSubmit(name, selectedType, config);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center">
|
<div className="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center">
|
||||||
@@ -289,10 +345,7 @@ function CreateIssuerModal({
|
|||||||
<h2 className="text-lg font-semibold text-ink">
|
<h2 className="text-lg font-semibold text-ink">
|
||||||
{step === 'type' ? 'Create Issuer' : `Configure ${selectedTypeConfig?.name || 'Issuer'}`}
|
{step === 'type' ? 'Create Issuer' : `Configure ${selectedTypeConfig?.name || 'Issuer'}`}
|
||||||
</h2>
|
</h2>
|
||||||
<button
|
<button onClick={onCancel} className="text-ink-muted hover:text-ink transition-colors">
|
||||||
onClick={onCancel}
|
|
||||||
className="text-ink-muted hover:text-ink transition-colors"
|
|
||||||
>
|
|
||||||
✕
|
✕
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -300,79 +353,28 @@ function CreateIssuerModal({
|
|||||||
{/* Content */}
|
{/* Content */}
|
||||||
<div className="px-6 py-6">
|
<div className="px-6 py-6">
|
||||||
{step === 'type' ? (
|
{step === 'type' ? (
|
||||||
<div className="grid grid-cols-2 gap-4">
|
<TypeSelector onSelect={handleTypeSelect} />
|
||||||
{issuerTypes.map((type) => (
|
|
||||||
<button
|
|
||||||
key={type.id}
|
|
||||||
onClick={() => onTypeSelect(type.id)}
|
|
||||||
className="p-4 border border-surface-border rounded-lg hover:border-brand-500 hover:bg-opacity-5 transition-all text-left"
|
|
||||||
>
|
|
||||||
<div className="font-medium text-ink">{type.name}</div>
|
|
||||||
<div className="text-sm text-ink-muted mt-1">{type.description}</div>
|
|
||||||
</button>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
) : (
|
) : (
|
||||||
<div className="space-y-5">
|
<div className="space-y-5">
|
||||||
{/* Name field always shown */}
|
{/* Name field */}
|
||||||
<div>
|
<div>
|
||||||
<label className="block text-sm font-medium text-ink mb-2">Issuer Name *</label>
|
<label className="block text-sm font-medium text-ink mb-2">Issuer Name *</label>
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
value={(form.name as string) || ''}
|
value={(form.name as string) || ''}
|
||||||
onChange={(e) => onFormChange('name', e.target.value)}
|
onChange={(e) => setForm({ ...form, name: e.target.value })}
|
||||||
placeholder="e.g., Production CA"
|
placeholder="e.g., Production CA"
|
||||||
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors"
|
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
{/* Type-specific fields via ConfigForm */}
|
||||||
{/* Type-specific fields */}
|
{selectedTypeConfig && (
|
||||||
{selectedTypeConfig?.configFields.map((field) => (
|
<ConfigForm
|
||||||
<div key={field.key}>
|
fields={selectedTypeConfig.configFields}
|
||||||
<label className="block text-sm font-medium text-ink mb-2">
|
values={form}
|
||||||
{field.label}
|
onChange={(key, value) => setForm({ ...form, [key]: value })}
|
||||||
{field.required && <span className="text-red-600 ml-1">*</span>}
|
/>
|
||||||
</label>
|
)}
|
||||||
{field.type === 'select' ? (
|
|
||||||
<select
|
|
||||||
value={(form[field.key] as string) || ''}
|
|
||||||
onChange={(e) => onFormChange(field.key, e.target.value)}
|
|
||||||
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink focus:outline-none focus:border-brand-500 transition-colors"
|
|
||||||
>
|
|
||||||
<option value="">Select {field.label}</option>
|
|
||||||
{field.options?.map((opt) => (
|
|
||||||
<option key={opt} value={opt}>
|
|
||||||
{opt}
|
|
||||||
</option>
|
|
||||||
))}
|
|
||||||
</select>
|
|
||||||
) : field.type === 'textarea' ? (
|
|
||||||
<textarea
|
|
||||||
value={(form[field.key] as string) || ''}
|
|
||||||
onChange={(e) => onFormChange(field.key, e.target.value)}
|
|
||||||
placeholder={field.placeholder}
|
|
||||||
rows={4}
|
|
||||||
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors font-mono text-xs"
|
|
||||||
/>
|
|
||||||
) : field.type === 'number' ? (
|
|
||||||
<input
|
|
||||||
type="number"
|
|
||||||
value={(form[field.key] as number | string) || ''}
|
|
||||||
onChange={(e) => onFormChange(field.key, e.target.value ? parseInt(e.target.value, 10) : '')}
|
|
||||||
placeholder={field.placeholder}
|
|
||||||
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors"
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={(form[field.key] as string) || ''}
|
|
||||||
onChange={(e) => onFormChange(field.key, e.target.value)}
|
|
||||||
placeholder={field.placeholder}
|
|
||||||
className="w-full px-3 py-2 bg-surface border border-surface-border rounded text-ink placeholder-ink-faint focus:outline-none focus:border-brand-500 transition-colors"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
@@ -381,7 +383,7 @@ function CreateIssuerModal({
|
|||||||
<div className="border-t border-surface-border px-6 py-4 flex justify-end gap-3">
|
<div className="border-t border-surface-border px-6 py-4 flex justify-end gap-3">
|
||||||
{step === 'config' && (
|
{step === 'config' && (
|
||||||
<button
|
<button
|
||||||
onClick={onBack}
|
onClick={() => setStep('type')}
|
||||||
className="px-4 py-2 border border-surface-border rounded text-ink hover:bg-surface-hover transition-colors text-sm font-medium"
|
className="px-4 py-2 border border-surface-border rounded text-ink hover:bg-surface-hover transition-colors text-sm font-medium"
|
||||||
>
|
>
|
||||||
Back
|
Back
|
||||||
@@ -395,22 +397,13 @@ function CreateIssuerModal({
|
|||||||
</button>
|
</button>
|
||||||
{step === 'config' && (
|
{step === 'config' && (
|
||||||
<button
|
<button
|
||||||
onClick={onSubmit}
|
onClick={handleSubmit}
|
||||||
disabled={isSubmitting || !form.name}
|
disabled={isSubmitting || !form.name}
|
||||||
className="px-4 py-2 bg-brand-600 text-white rounded text-sm font-medium hover:bg-brand-700 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
className="px-4 py-2 bg-brand-600 text-white rounded text-sm font-medium hover:bg-brand-700 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
>
|
>
|
||||||
{isSubmitting ? 'Creating...' : 'Create Issuer'}
|
{isSubmitting ? 'Creating...' : 'Create Issuer'}
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
{step === 'type' && (
|
|
||||||
<button
|
|
||||||
onClick={() => selectedType && onTypeSelect(selectedType)}
|
|
||||||
disabled={!selectedType}
|
|
||||||
className="px-4 py-2 bg-brand-600 text-white rounded text-sm font-medium hover:bg-brand-700 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
|
||||||
>
|
|
||||||
Next
|
|
||||||
</button>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -136,6 +136,15 @@ export default function JobsPage() {
|
|||||||
label: 'Attempts',
|
label: 'Attempts',
|
||||||
render: (j) => <span className="text-ink-muted">{j.attempts}/{j.max_attempts}</span>,
|
render: (j) => <span className="text-ink-muted">{j.attempts}/{j.max_attempts}</span>,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: 'error',
|
||||||
|
label: 'Error',
|
||||||
|
render: (j) => j.status === 'Failed' && j.error_message ? (
|
||||||
|
<span className="text-xs text-red-600 truncate max-w-[200px] inline-block" title={j.error_message}>
|
||||||
|
{j.error_message.length > 80 ? j.error_message.substring(0, 80) + '...' : j.error_message}
|
||||||
|
</span>
|
||||||
|
) : <span className="text-xs text-ink-faint">—</span>,
|
||||||
|
},
|
||||||
{ key: 'scheduled', label: 'Scheduled', render: (j) => <span className="text-xs text-ink-muted">{formatDateTime(j.scheduled_at)}</span> },
|
{ key: 'scheduled', label: 'Scheduled', render: (j) => <span className="text-xs text-ink-muted">{formatDateTime(j.scheduled_at)}</span> },
|
||||||
{ key: 'completed', label: 'Completed', render: (j) => <span className="text-xs text-ink-muted">{formatDateTime(j.completed_at)}</span> },
|
{ key: 'completed', label: 'Completed', render: (j) => <span className="text-xs text-ink-muted">{formatDateTime(j.completed_at)}</span> },
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -25,11 +25,63 @@ interface CreateProfileModalProps {
|
|||||||
error: string | null;
|
error: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const AVAILABLE_ALGORITHMS = ['RSA', 'ECDSA', 'Ed25519'];
|
||||||
|
const ALGORITHM_MIN_SIZES: Record<string, number[]> = {
|
||||||
|
RSA: [2048, 3072, 4096],
|
||||||
|
ECDSA: [256, 384],
|
||||||
|
Ed25519: [0],
|
||||||
|
};
|
||||||
|
|
||||||
|
const AVAILABLE_EKUS = [
|
||||||
|
{ value: 'serverAuth', label: 'Server Authentication (TLS)' },
|
||||||
|
{ value: 'clientAuth', label: 'Client Authentication' },
|
||||||
|
{ value: 'codeSigning', label: 'Code Signing' },
|
||||||
|
{ value: 'emailProtection', label: 'Email Protection (S/MIME)' },
|
||||||
|
{ value: 'timeStamping', label: 'Time Stamping' },
|
||||||
|
];
|
||||||
|
|
||||||
|
interface KeyAlgorithmEntry {
|
||||||
|
algorithm: string;
|
||||||
|
min_size: number;
|
||||||
|
}
|
||||||
|
|
||||||
function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: CreateProfileModalProps) {
|
function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: CreateProfileModalProps) {
|
||||||
const [name, setName] = useState('');
|
const [name, setName] = useState('');
|
||||||
const [description, setDescription] = useState('');
|
const [description, setDescription] = useState('');
|
||||||
const [ttl, setTtl] = useState('86400');
|
const [ttl, setTtl] = useState('86400');
|
||||||
const [shortLived, setShortLived] = useState(false);
|
const [shortLived, setShortLived] = useState(false);
|
||||||
|
const [keyAlgorithms, setKeyAlgorithms] = useState<KeyAlgorithmEntry[]>([
|
||||||
|
{ algorithm: 'ECDSA', min_size: 256 },
|
||||||
|
{ algorithm: 'RSA', min_size: 2048 },
|
||||||
|
]);
|
||||||
|
const [selectedEkus, setSelectedEkus] = useState<string[]>(['serverAuth']);
|
||||||
|
const [sanPatterns, setSanPatterns] = useState('');
|
||||||
|
const [spiffePattern, setSpiffePattern] = useState('');
|
||||||
|
|
||||||
|
const addAlgorithm = () => {
|
||||||
|
const unused = AVAILABLE_ALGORITHMS.find(a => !keyAlgorithms.some(ka => ka.algorithm === a));
|
||||||
|
if (unused) {
|
||||||
|
setKeyAlgorithms([...keyAlgorithms, { algorithm: unused, min_size: ALGORITHM_MIN_SIZES[unused][0] }]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeAlgorithm = (idx: number) => {
|
||||||
|
setKeyAlgorithms(keyAlgorithms.filter((_, i) => i !== idx));
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateAlgorithm = (idx: number, field: 'algorithm' | 'min_size', value: string | number) => {
|
||||||
|
const updated = [...keyAlgorithms];
|
||||||
|
if (field === 'algorithm') {
|
||||||
|
updated[idx] = { algorithm: value as string, min_size: ALGORITHM_MIN_SIZES[value as string]?.[0] || 0 };
|
||||||
|
} else {
|
||||||
|
updated[idx] = { ...updated[idx], min_size: value as number };
|
||||||
|
}
|
||||||
|
setKeyAlgorithms(updated);
|
||||||
|
};
|
||||||
|
|
||||||
|
const toggleEku = (eku: string) => {
|
||||||
|
setSelectedEkus(prev => prev.includes(eku) ? prev.filter(e => e !== eku) : [...prev, eku]);
|
||||||
|
};
|
||||||
|
|
||||||
const handleSubmit = async (e: React.FormEvent) => {
|
const handleSubmit = async (e: React.FormEvent) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@@ -39,20 +91,31 @@ function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: Cr
|
|||||||
description: description.trim(),
|
description: description.trim(),
|
||||||
max_ttl_seconds: parseInt(ttl) || 86400,
|
max_ttl_seconds: parseInt(ttl) || 86400,
|
||||||
allow_short_lived: shortLived,
|
allow_short_lived: shortLived,
|
||||||
|
allowed_key_algorithms: keyAlgorithms,
|
||||||
|
allowed_ekus: selectedEkus,
|
||||||
|
required_san_patterns: sanPatterns.trim() ? sanPatterns.split(',').map(s => s.trim()).filter(Boolean) : [],
|
||||||
|
spiffe_uri_pattern: spiffePattern.trim() || '',
|
||||||
enabled: true,
|
enabled: true,
|
||||||
});
|
});
|
||||||
setName('');
|
setName('');
|
||||||
setDescription('');
|
setDescription('');
|
||||||
setTtl('86400');
|
setTtl('86400');
|
||||||
setShortLived(false);
|
setShortLived(false);
|
||||||
|
setKeyAlgorithms([{ algorithm: 'ECDSA', min_size: 256 }, { algorithm: 'RSA', min_size: 2048 }]);
|
||||||
|
setSelectedEkus(['serverAuth']);
|
||||||
|
setSanPatterns('');
|
||||||
|
setSpiffePattern('');
|
||||||
onSuccess();
|
onSuccess();
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!isOpen) return null;
|
if (!isOpen) return null;
|
||||||
|
|
||||||
|
const inputClass = 'w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400';
|
||||||
|
const selectClass = 'bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="fixed inset-0 bg-black/40 flex items-center justify-center z-50" onClick={onClose}>
|
<div className="fixed inset-0 bg-black/40 flex items-center justify-center z-50" onClick={onClose}>
|
||||||
<div className="bg-surface border border-surface-border rounded p-5 w-full max-w-md shadow-xl" onClick={e => e.stopPropagation()}>
|
<div className="bg-surface border border-surface-border rounded p-5 w-full max-w-lg shadow-xl max-h-[90vh] overflow-y-auto" onClick={e => e.stopPropagation()}>
|
||||||
<h2 className="text-lg font-semibold text-ink mb-4">Create Profile</h2>
|
<h2 className="text-lg font-semibold text-ink mb-4">Create Profile</h2>
|
||||||
{error && <div className="mb-4 p-3 bg-red-50 border border-red-200 rounded text-sm text-red-700">{error}</div>}
|
{error && <div className="mb-4 p-3 bg-red-50 border border-red-200 rounded text-sm text-red-700">{error}</div>}
|
||||||
<form onSubmit={handleSubmit} className="space-y-4">
|
<form onSubmit={handleSubmit} className="space-y-4">
|
||||||
@@ -61,7 +124,7 @@ function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: Cr
|
|||||||
<input
|
<input
|
||||||
value={name}
|
value={name}
|
||||||
onChange={e => setName(e.target.value)}
|
onChange={e => setName(e.target.value)}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400"
|
className={inputClass}
|
||||||
placeholder="e.g., Web Server Certs"
|
placeholder="e.g., Web Server Certs"
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
@@ -71,7 +134,7 @@ function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: Cr
|
|||||||
<textarea
|
<textarea
|
||||||
value={description}
|
value={description}
|
||||||
onChange={e => setDescription(e.target.value)}
|
onChange={e => setDescription(e.target.value)}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400"
|
className={inputClass}
|
||||||
placeholder="Optional description"
|
placeholder="Optional description"
|
||||||
rows={2}
|
rows={2}
|
||||||
/>
|
/>
|
||||||
@@ -82,7 +145,7 @@ function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: Cr
|
|||||||
type="number"
|
type="number"
|
||||||
value={ttl}
|
value={ttl}
|
||||||
onChange={e => setTtl(e.target.value)}
|
onChange={e => setTtl(e.target.value)}
|
||||||
className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400"
|
className={inputClass}
|
||||||
placeholder="86400"
|
placeholder="86400"
|
||||||
/>
|
/>
|
||||||
<p className="text-xs text-ink-muted mt-1">
|
<p className="text-xs text-ink-muted mt-1">
|
||||||
@@ -109,6 +172,97 @@ function CreateProfileModal({ isOpen, onClose, onSuccess, isLoading, error }: Cr
|
|||||||
/>
|
/>
|
||||||
<label htmlFor="shortLived" className="text-sm text-ink">Allow short-lived certs</label>
|
<label htmlFor="shortLived" className="text-sm text-ink">Allow short-lived certs</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Allowed Key Algorithms */}
|
||||||
|
<div>
|
||||||
|
<div className="flex items-center justify-between mb-1">
|
||||||
|
<label className="block text-sm font-medium text-ink">Allowed Key Algorithms</label>
|
||||||
|
{keyAlgorithms.length < AVAILABLE_ALGORITHMS.length && (
|
||||||
|
<button type="button" onClick={addAlgorithm} className="text-xs text-brand-600 hover:text-brand-700 font-medium">
|
||||||
|
+ Add
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
{keyAlgorithms.map((ka, idx) => (
|
||||||
|
<div key={idx} className="flex items-center gap-2">
|
||||||
|
<select
|
||||||
|
value={ka.algorithm}
|
||||||
|
onChange={e => updateAlgorithm(idx, 'algorithm', e.target.value)}
|
||||||
|
className={selectClass + ' flex-1'}
|
||||||
|
>
|
||||||
|
{AVAILABLE_ALGORITHMS.map(a => (
|
||||||
|
<option key={a} value={a} disabled={a !== ka.algorithm && keyAlgorithms.some(k => k.algorithm === a)}>
|
||||||
|
{a}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
{ka.algorithm !== 'Ed25519' ? (
|
||||||
|
<select
|
||||||
|
value={ka.min_size}
|
||||||
|
onChange={e => updateAlgorithm(idx, 'min_size', parseInt(e.target.value))}
|
||||||
|
className={selectClass + ' w-24'}
|
||||||
|
>
|
||||||
|
{(ALGORITHM_MIN_SIZES[ka.algorithm] || []).map(s => (
|
||||||
|
<option key={s} value={s}>{s}+</option>
|
||||||
|
))}
|
||||||
|
</select>
|
||||||
|
) : (
|
||||||
|
<span className="text-xs text-ink-muted w-24 text-center">fixed</span>
|
||||||
|
)}
|
||||||
|
<button type="button" onClick={() => removeAlgorithm(idx)} className="text-xs text-red-500 hover:text-red-600">
|
||||||
|
Remove
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{keyAlgorithms.length === 0 && (
|
||||||
|
<p className="text-xs text-ink-faint">No algorithms configured. Click + Add to allow key types.</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Allowed EKUs */}
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-ink mb-1">Allowed Extended Key Usages</label>
|
||||||
|
<div className="space-y-1.5">
|
||||||
|
{AVAILABLE_EKUS.map(eku => (
|
||||||
|
<label key={eku.value} className="flex items-center gap-2 cursor-pointer">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={selectedEkus.includes(eku.value)}
|
||||||
|
onChange={() => toggleEku(eku.value)}
|
||||||
|
className="w-4 h-4"
|
||||||
|
/>
|
||||||
|
<span className="text-sm text-ink">{eku.label}</span>
|
||||||
|
</label>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Required SAN Patterns */}
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-ink mb-1">Required SAN Patterns</label>
|
||||||
|
<input
|
||||||
|
value={sanPatterns}
|
||||||
|
onChange={e => setSanPatterns(e.target.value)}
|
||||||
|
className={inputClass}
|
||||||
|
placeholder="e.g., *.example.com, api.internal"
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-ink-muted mt-1">Comma-separated patterns. Leave empty for no constraints.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* SPIFFE URI Pattern */}
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-ink mb-1">SPIFFE URI Pattern</label>
|
||||||
|
<input
|
||||||
|
value={spiffePattern}
|
||||||
|
onChange={e => setSpiffePattern(e.target.value)}
|
||||||
|
className={inputClass}
|
||||||
|
placeholder="e.g., spiffe://example.org/service/*"
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-ink-muted mt-1">Optional workload identity URI SAN pattern.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="flex gap-2 pt-4">
|
<div className="flex gap-2 pt-4">
|
||||||
<button
|
<button
|
||||||
type="submit"
|
type="submit"
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
|
import { useState } from 'react';
|
||||||
import { useParams, Link } from 'react-router-dom';
|
import { useParams, Link } from 'react-router-dom';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
||||||
import { getTarget, getJobs } from '../api/client';
|
import { getTarget, getJobs, updateTarget } from '../api/client';
|
||||||
import PageHeader from '../components/PageHeader';
|
import PageHeader from '../components/PageHeader';
|
||||||
import StatusBadge from '../components/StatusBadge';
|
import StatusBadge from '../components/StatusBadge';
|
||||||
import DataTable from '../components/DataTable';
|
import DataTable from '../components/DataTable';
|
||||||
@@ -30,6 +31,18 @@ function InfoRow({ label, value }: { label: string; value: React.ReactNode }) {
|
|||||||
|
|
||||||
export default function TargetDetailPage() {
|
export default function TargetDetailPage() {
|
||||||
const { id } = useParams<{ id: string }>();
|
const { id } = useParams<{ id: string }>();
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const [isEditing, setIsEditing] = useState(false);
|
||||||
|
const [editName, setEditName] = useState('');
|
||||||
|
const [editHostname, setEditHostname] = useState('');
|
||||||
|
|
||||||
|
const updateMutation = useMutation({
|
||||||
|
mutationFn: (data: Partial<{ name: string; hostname: string }>) => updateTarget(id!, data),
|
||||||
|
onSuccess: () => {
|
||||||
|
queryClient.invalidateQueries({ queryKey: ['target', id] });
|
||||||
|
setIsEditing(false);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const { data: target, isLoading, error, refetch } = useQuery({
|
const { data: target, isLoading, error, refetch } = useQuery({
|
||||||
queryKey: ['target', id],
|
queryKey: ['target', id],
|
||||||
@@ -112,6 +125,18 @@ export default function TargetDetailPage() {
|
|||||||
<PageHeader
|
<PageHeader
|
||||||
title={target.name}
|
title={target.name}
|
||||||
subtitle={typeLabels[target.type] || target.type}
|
subtitle={typeLabels[target.type] || target.type}
|
||||||
|
action={
|
||||||
|
<button
|
||||||
|
onClick={() => {
|
||||||
|
setEditName(target.name);
|
||||||
|
setEditHostname(target.hostname || '');
|
||||||
|
setIsEditing(true);
|
||||||
|
}}
|
||||||
|
className="px-3 py-1.5 border border-surface-border rounded text-ink text-xs hover:bg-surface-hover transition-colors font-medium"
|
||||||
|
>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex-1 overflow-y-auto px-6 py-4 space-y-6">
|
<div className="flex-1 overflow-y-auto px-6 py-4 space-y-6">
|
||||||
@@ -164,6 +189,36 @@ export default function TargetDetailPage() {
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{/* Edit Modal */}
|
||||||
|
{isEditing && (
|
||||||
|
<div className="fixed inset-0 bg-black/40 flex items-center justify-center z-50" onClick={() => setIsEditing(false)}>
|
||||||
|
<div className="bg-surface border border-surface-border rounded p-5 w-full max-w-md shadow-xl" onClick={e => e.stopPropagation()}>
|
||||||
|
<h2 className="text-lg font-semibold text-ink mb-4">Edit Target</h2>
|
||||||
|
{updateMutation.isError && (
|
||||||
|
<div className="mb-4 p-3 bg-red-50 border border-red-200 rounded text-sm text-red-700">
|
||||||
|
{(updateMutation.error as Error).message}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<form onSubmit={e => { e.preventDefault(); updateMutation.mutate({ name: editName, hostname: editHostname }); }} className="space-y-4">
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-ink mb-1">Name</label>
|
||||||
|
<input value={editName} onChange={e => setEditName(e.target.value)} className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label className="block text-sm font-medium text-ink mb-1">Hostname</label>
|
||||||
|
<input value={editHostname} onChange={e => setEditHostname(e.target.value)} className="w-full bg-white border border-surface-border rounded px-3 py-2 text-sm text-ink focus:outline-none focus:border-brand-400" />
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2 pt-2">
|
||||||
|
<button type="submit" disabled={updateMutation.isPending} className="flex-1 btn btn-primary disabled:opacity-50">
|
||||||
|
{updateMutation.isPending ? 'Saving...' : 'Save'}
|
||||||
|
</button>
|
||||||
|
<button type="button" onClick={() => setIsEditing(false)} className="flex-1 btn btn-ghost">Cancel</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user