mirror of
https://github.com/shankar0123/certctl.git
synced 2026-06-07 18:01:37 +00:00
f92c997a50
Three related ACME ecosystem changes shipped as a single milestone: 1. ACME Certificate Profile Selection: Custom JWS-signed newOrder POST with `profile` field (e.g., `tlsserver`, `shortlived` for 6-day certs) bypassing acme.Client.AuthorizeOrder() since golang.org/x/crypto lacks profile support. ES256 JWS signing with kid mode, nonce management, directory discovery. Empty profile delegates to standard library path (zero behavior change). Configurable via CERTCTL_ACME_PROFILE env var. GUI: profile dropdown on ACME issuer config. 2. ARI RFC 9702 → 9773 Renumber: All 25+ references updated across Go source, docs, README, and examples. Zero remaining occurrences of RFC 9702. 3. 45-Day / Short-Lived Certificate Positioning: 5 domain tests validating renewal thresholds against SC-081v3 validity reduction timeline (200→100→47 days) and Let's Encrypt 45-day/6-day profiles. ARI (RFC 9773) is the expected renewal path for 6-day shortlived certs. New tests: 13 profile + 5 domain threshold + 1 frontend = 19 new tests. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
36 lines
1.5 KiB
Go
36 lines
1.5 KiB
Go
package domain
|
|
|
|
import "time"
|
|
|
|
// RenewalInfo represents ACME Renewal Information (ARI) per RFC 9773.
|
|
// It provides CA-directed renewal timing via a suggested renewal window.
|
|
type RenewalInfo struct {
|
|
// SuggestedWindowStart is the beginning of the time window during which the CA suggests renewal.
|
|
SuggestedWindowStart time.Time `json:"suggested_window_start"`
|
|
|
|
// SuggestedWindowEnd is the end of the time window during which the CA suggests renewal.
|
|
SuggestedWindowEnd time.Time `json:"suggested_window_end"`
|
|
|
|
// RetryAfter is the earliest time the client should re-poll for updated ARI.
|
|
// Zero value means no retry constraint.
|
|
RetryAfter time.Time `json:"retry_after,omitempty"`
|
|
|
|
// ExplanationURL is an optional URL with human-readable explanation for the renewal timing.
|
|
ExplanationURL string `json:"explanation_url,omitempty"`
|
|
}
|
|
|
|
// ShouldRenewNow returns true if the current time is within or past the suggested renewal window.
|
|
// This is the primary decision point: if true, renewal should proceed immediately.
|
|
func (r *RenewalInfo) ShouldRenewNow() bool {
|
|
now := time.Now()
|
|
return !now.Before(r.SuggestedWindowStart)
|
|
}
|
|
|
|
// OptimalRenewalTime returns the midpoint of the suggested renewal window,
|
|
// which is the recommended time to initiate renewal per RFC 9773.
|
|
// This can be used for scheduling if the current time is before the window.
|
|
func (r *RenewalInfo) OptimalRenewalTime() time.Time {
|
|
duration := r.SuggestedWindowEnd.Sub(r.SuggestedWindowStart)
|
|
return r.SuggestedWindowStart.Add(duration / 2)
|
|
}
|