0
Fork 0
mirror of https://github.com/caddyserver/caddy.git synced 2025-01-06 22:40:31 -05:00

caddytls: Configurable storage clean interval

Can drastically reduce costs on storage backends where scans are expensive.

Also reduced default interval to 24h.

See https://github.com/silinternational/certmagic-storage-dynamodb/issues/18
This commit is contained in:
Matthew Holt 2021-04-12 15:41:22 -06:00
parent 3903642aa7
commit 42b7134ffa
No known key found for this signature in database
GPG key ID: 2A349DD577D586A5
2 changed files with 52 additions and 9 deletions

View file

@ -56,6 +56,19 @@ type AutomationConfig struct {
// performed. Default: 10m // performed. Default: 10m
RenewCheckInterval caddy.Duration `json:"renew_interval,omitempty"` RenewCheckInterval caddy.Duration `json:"renew_interval,omitempty"`
// How often to scan storage units for old or expired
// assets and remove them. These scans exert lots of
// reads (and list operations) on the storage module, so
// choose a longer interval for large deployments.
// Default: 24h
//
// Storage will always be cleaned when the process first
// starts. Then, a new cleaning will be started this
// duration after the previous cleaning started if the
// previous cleaning finished in less than half the time
// of this interval (otherwise next start will be skipped).
StorageCleanInterval caddy.Duration `json:"storage_clean_interval,omitempty"`
defaultPublicAutomationPolicy *AutomationPolicy defaultPublicAutomationPolicy *AutomationPolicy
defaultInternalAutomationPolicy *AutomationPolicy // only initialized if necessary defaultInternalAutomationPolicy *AutomationPolicy // only initialized if necessary
} }

View file

@ -414,7 +414,7 @@ func (t *TLS) AllMatchingCertificates(san string) []certmagic.Certificate {
// known storage units if it was not recently done, and then runs the // known storage units if it was not recently done, and then runs the
// operation at every tick from t.storageCleanTicker. // operation at every tick from t.storageCleanTicker.
func (t *TLS) keepStorageClean() { func (t *TLS) keepStorageClean() {
t.storageCleanTicker = time.NewTicker(storageCleanInterval) t.storageCleanTicker = time.NewTicker(t.storageCleanInterval())
t.storageCleanStop = make(chan struct{}) t.storageCleanStop = make(chan struct{})
go func() { go func() {
defer func() { defer func() {
@ -438,31 +438,61 @@ func (t *TLS) cleanStorageUnits() {
storageCleanMu.Lock() storageCleanMu.Lock()
defer storageCleanMu.Unlock() defer storageCleanMu.Unlock()
if !storageClean.IsZero() && time.Since(storageClean) < storageCleanInterval { // If storage was cleaned recently, don't do it again for now. Although the ticker
// drops missed ticks for us, config reloads discard the old ticker and replace it
// with a new one, possibly invoking a cleaning to happen again too soon.
// (We divide the interval by 2 because the actual cleaning takes non-zero time,
// and we don't want to skip cleanings if we don't have to; whereas if a cleaning
// took the entire interval, we'd probably want to skip the next one so we aren't
// constantly cleaning. This allows cleanings to take up to half the interval's
// duration before we decide to skip the next one.)
if !storageClean.IsZero() && time.Since(storageClean) < t.storageCleanInterval()/2 {
return return
} }
// mark when storage cleaning was last initiated
storageClean = time.Now()
options := certmagic.CleanStorageOptions{ options := certmagic.CleanStorageOptions{
OCSPStaples: true, OCSPStaples: true,
ExpiredCerts: true, ExpiredCerts: true,
ExpiredCertGracePeriod: 24 * time.Hour * 14, ExpiredCertGracePeriod: 24 * time.Hour * 14,
} }
// start with the default storage // avoid cleaning same storage more than once per cleaning cycle
certmagic.CleanStorage(t.ctx, t.ctx.Storage(), options) storagesCleaned := make(map[string]struct{})
// start with the default/global storage
storage := t.ctx.Storage()
storageStr := fmt.Sprintf("%v", storage)
t.logger.Info("cleaning storage unit", zap.String("description", storageStr))
certmagic.CleanStorage(t.ctx, storage, options)
storagesCleaned[storageStr] = struct{}{}
// then clean each storage defined in ACME automation policies // then clean each storage defined in ACME automation policies
if t.Automation != nil { if t.Automation != nil {
for _, ap := range t.Automation.Policies { for _, ap := range t.Automation.Policies {
if ap.storage != nil { if ap.storage == nil {
continue
}
storageStr := fmt.Sprintf("%v", ap.storage)
if _, ok := storagesCleaned[storageStr]; ok {
continue
}
t.logger.Info("cleaning storage unit", zap.String("description", storageStr))
certmagic.CleanStorage(t.ctx, ap.storage, options) certmagic.CleanStorage(t.ctx, ap.storage, options)
} storagesCleaned[storageStr] = struct{}{}
} }
} }
storageClean = time.Now() t.logger.Info("finished cleaning storage units")
}
t.logger.Info("cleaned up storage units") func (t *TLS) storageCleanInterval() time.Duration {
if t.Automation != nil && t.Automation.StorageCleanInterval > 0 {
return time.Duration(t.Automation.StorageCleanInterval)
}
return defaultStorageCleanInterval
} }
// CertificateLoader is a type that can load certificates. // CertificateLoader is a type that can load certificates.
@ -507,7 +537,7 @@ type CertCacheOptions struct {
// Variables related to storage cleaning. // Variables related to storage cleaning.
var ( var (
storageCleanInterval = 12 * time.Hour defaultStorageCleanInterval = 24 * time.Hour
storageClean time.Time storageClean time.Time
storageCleanMu sync.Mutex storageCleanMu sync.Mutex