340 lines
12 KiB
Go
340 lines
12 KiB
Go
// Copyright 2015 Matthew Holt
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package certmagic
|
|
|
|
import (
|
|
"log"
|
|
"time"
|
|
|
|
"golang.org/x/crypto/ocsp"
|
|
)
|
|
|
|
// maintainAssets is a permanently-blocking function
|
|
// that loops indefinitely and, on a regular schedule, checks
|
|
// certificates for expiration and initiates a renewal of certs
|
|
// that are expiring soon. It also updates OCSP stapling. It
|
|
// should only be called once per cache.
|
|
func (certCache *Cache) maintainAssets() {
|
|
renewalTicker := time.NewTicker(certCache.options.RenewCheckInterval)
|
|
ocspTicker := time.NewTicker(certCache.options.OCSPCheckInterval)
|
|
|
|
log.Printf("[INFO][cache:%p] Started certificate maintenance routine", certCache)
|
|
|
|
for {
|
|
select {
|
|
case <-renewalTicker.C:
|
|
log.Printf("[INFO][cache:%p] Scanning for expiring certificates", certCache)
|
|
err := certCache.RenewManagedCertificates(false)
|
|
if err != nil {
|
|
log.Printf("[ERROR][cache:%p] Renewing managed certificates: %v", certCache, err)
|
|
}
|
|
log.Printf("[INFO][cache:%p] Done scanning certificates", certCache)
|
|
case <-ocspTicker.C:
|
|
log.Printf("[INFO][cache:%p] Scanning for stale OCSP staples", certCache)
|
|
certCache.updateOCSPStaples()
|
|
// certCache.deleteOldStapleFiles()
|
|
log.Printf("[INFO][cache:%p] Done checking OCSP staples", certCache)
|
|
case <-certCache.stopChan:
|
|
renewalTicker.Stop()
|
|
ocspTicker.Stop()
|
|
// TODO: stop any in-progress maintenance operations and clear locks we made
|
|
log.Printf("[INFO][cache:%p] Stopped certificate maintenance routine", certCache)
|
|
close(certCache.doneChan)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// RenewManagedCertificates renews managed certificates,
|
|
// including ones loaded on-demand. Note that this is done
|
|
// automatically on a regular basis; normally you will not
|
|
// need to call this.
|
|
func (certCache *Cache) RenewManagedCertificates(interactive bool) error {
|
|
// configs will hold a map of certificate name to the config
|
|
// to use when managing that certificate
|
|
configs := make(map[string]*Config)
|
|
|
|
// we use the queues for a very important reason: to do any and all
|
|
// operations that could require an exclusive write lock outside
|
|
// of the read lock! otherwise we get a deadlock, yikes. in other
|
|
// words, our first iteration through the certificate cache does NOT
|
|
// perform any operations--only queues them--so that more fine-grained
|
|
// write locks may be obtained during the actual operations.
|
|
var renewQueue, reloadQueue, deleteQueue []Certificate
|
|
|
|
certCache.mu.RLock()
|
|
for certKey, cert := range certCache.cache {
|
|
if !cert.managed {
|
|
continue
|
|
}
|
|
|
|
// the list of names on this cert should never be empty... programmer error?
|
|
if cert.Names == nil || len(cert.Names) == 0 {
|
|
log.Printf("[WARNING] Certificate keyed by '%s' has no names: %v - removing from cache", certKey, cert.Names)
|
|
deleteQueue = append(deleteQueue, cert)
|
|
continue
|
|
}
|
|
|
|
// get the config associated with this certificate
|
|
cfg, err := certCache.getConfig(cert)
|
|
if err != nil {
|
|
log.Printf("[ERROR] Getting configuration to manage certificate for names %v; unable to renew: %v", cert.Names, err)
|
|
continue
|
|
}
|
|
if cfg == nil {
|
|
// this is bad if this happens, probably a programmer error (oops)
|
|
log.Printf("[ERROR] No configuration associated with certificate for names %v; unable to manage", cert.Names)
|
|
continue
|
|
}
|
|
configs[cert.Names[0]] = cfg
|
|
|
|
// if time is up or expires soon, we need to try to renew it
|
|
if cert.NeedsRenewal(cfg) {
|
|
// see if the certificate in storage has already been renewed, possibly by another
|
|
// instance that didn't coordinate with this one; if so, just load it (this
|
|
// might happen if another instance already renewed it - kinda sloppy but checking disk
|
|
// first is a simple way to possibly drastically reduce rate limit problems)
|
|
storedCertExpiring, err := cfg.managedCertInStorageExpiresSoon(cert)
|
|
if err != nil {
|
|
// hmm, weird, but not a big deal, maybe it was deleted or something
|
|
log.Printf("[NOTICE] Error while checking if certificate for %v in storage is also expiring soon: %v",
|
|
cert.Names, err)
|
|
} else if !storedCertExpiring {
|
|
// if the certificate is NOT expiring soon and there was no error, then we
|
|
// are good to just reload the certificate from storage instead of repeating
|
|
// a likely-unnecessary renewal procedure
|
|
reloadQueue = append(reloadQueue, cert)
|
|
continue
|
|
}
|
|
|
|
// the certificate in storage has not been renewed yet, so we will do it
|
|
// NOTE: It is super-important to note that the TLS-ALPN challenge requires
|
|
// a write lock on the cache in order to complete its challenge, so it is extra
|
|
// vital that this renew operation does not happen inside our read lock!
|
|
renewQueue = append(renewQueue, cert)
|
|
}
|
|
}
|
|
certCache.mu.RUnlock()
|
|
|
|
// Reload certificates that merely need to be updated in memory
|
|
for _, oldCert := range reloadQueue {
|
|
timeLeft := oldCert.NotAfter.Sub(time.Now().UTC())
|
|
log.Printf("[INFO] Certificate for %v expires in %v, but is already renewed in storage; reloading stored certificate",
|
|
oldCert.Names, timeLeft)
|
|
|
|
cfg := configs[oldCert.Names[0]]
|
|
|
|
err := cfg.reloadManagedCertificate(oldCert)
|
|
if err != nil {
|
|
if interactive {
|
|
return err // operator is present, so report error immediately
|
|
}
|
|
log.Printf("[ERROR] Loading renewed certificate: %v", err)
|
|
}
|
|
}
|
|
|
|
// Renewal queue
|
|
for _, oldCert := range renewQueue {
|
|
timeLeft := oldCert.NotAfter.Sub(time.Now().UTC())
|
|
log.Printf("[INFO] Certificate for %v expires in %v; attempting renewal", oldCert.Names, timeLeft)
|
|
|
|
cfg := configs[oldCert.Names[0]]
|
|
|
|
// Get the name which we should use to renew this certificate;
|
|
// we only support managing certificates with one name per cert,
|
|
// so this should be easy.
|
|
renewName := oldCert.Names[0]
|
|
|
|
// perform renewal
|
|
err := cfg.RenewCert(renewName, interactive)
|
|
if err != nil {
|
|
if interactive {
|
|
// Certificate renewal failed and the operator is present. See a discussion about
|
|
// this in issue mholt/caddy#642. For a while, we only stopped if the certificate
|
|
// was expired, but in reality, there is no difference between reporting it now
|
|
// versus later, except that there's somebody present to deal with it right now.
|
|
// Follow-up: See issue mholt/caddy#1680. Only fail in this case if the certificate
|
|
// is dangerously close to expiration.
|
|
timeLeft := oldCert.NotAfter.Sub(time.Now().UTC())
|
|
if timeLeft < cfg.RenewDurationBeforeAtStartup {
|
|
return err
|
|
}
|
|
}
|
|
log.Printf("[ERROR] %v", err)
|
|
if cfg.OnDemand != nil {
|
|
// loaded dynamically, remove dynamically
|
|
deleteQueue = append(deleteQueue, oldCert)
|
|
}
|
|
continue
|
|
}
|
|
|
|
// successful renewal, so update in-memory cache by loading
|
|
// renewed certificate so it will be used with handshakes
|
|
err = cfg.reloadManagedCertificate(oldCert)
|
|
if err != nil {
|
|
if interactive {
|
|
return err // operator is present, so report error immediately
|
|
}
|
|
log.Printf("[ERROR] %v", err)
|
|
}
|
|
}
|
|
|
|
// Deletion queue
|
|
for _, cert := range deleteQueue {
|
|
certCache.removeCertificate(cert)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// updateOCSPStaples updates the OCSP stapling in all
|
|
// eligible, cached certificates.
|
|
//
|
|
// OCSP maintenance strives to abide the relevant points on
|
|
// Ryan Sleevi's recommendations for good OCSP support:
|
|
// https://gist.github.com/sleevi/5efe9ef98961ecfb4da8
|
|
func (certCache *Cache) updateOCSPStaples() {
|
|
// Create a temporary place to store updates
|
|
// until we release the potentially long-lived
|
|
// read lock and use a short-lived write lock
|
|
// on the certificate cache.
|
|
type ocspUpdate struct {
|
|
rawBytes []byte
|
|
parsed *ocsp.Response
|
|
}
|
|
updated := make(map[string]ocspUpdate)
|
|
|
|
certCache.mu.RLock()
|
|
for certHash, cert := range certCache.cache {
|
|
// no point in updating OCSP for expired certificates
|
|
if time.Now().After(cert.NotAfter) {
|
|
continue
|
|
}
|
|
|
|
var lastNextUpdate time.Time
|
|
if cert.OCSP != nil {
|
|
lastNextUpdate = cert.OCSP.NextUpdate
|
|
if freshOCSP(cert.OCSP) {
|
|
continue // no need to update staple if ours is still fresh
|
|
}
|
|
}
|
|
|
|
cfg, err := certCache.getConfig(cert)
|
|
if err != nil {
|
|
log.Printf("[ERROR] Getting configuration to manage OCSP for certificate with names %v; unable to refresh: %v", cert.Names, err)
|
|
continue
|
|
}
|
|
if cfg == nil {
|
|
// this is bad if this happens, probably a programmer error (oops)
|
|
log.Printf("[ERROR] No configuration associated with certificate for names %v; unable to manage OCSP", cert.Names)
|
|
continue
|
|
}
|
|
|
|
err = stapleOCSP(cfg.Storage, &cert, nil)
|
|
if err != nil {
|
|
if cert.OCSP != nil {
|
|
// if there was no staple before, that's fine; otherwise we should log the error
|
|
log.Printf("[ERROR] Checking OCSP: %v", err)
|
|
}
|
|
continue
|
|
}
|
|
|
|
// By this point, we've obtained the latest OCSP response.
|
|
// If there was no staple before, or if the response is updated, make
|
|
// sure we apply the update to all names on the certificate.
|
|
if cert.OCSP != nil && (lastNextUpdate.IsZero() || lastNextUpdate != cert.OCSP.NextUpdate) {
|
|
log.Printf("[INFO] Advancing OCSP staple for %v from %s to %s",
|
|
cert.Names, lastNextUpdate, cert.OCSP.NextUpdate)
|
|
updated[certHash] = ocspUpdate{rawBytes: cert.Certificate.OCSPStaple, parsed: cert.OCSP}
|
|
}
|
|
}
|
|
certCache.mu.RUnlock()
|
|
|
|
// These write locks should be brief since we have all the info we need now.
|
|
for certKey, update := range updated {
|
|
certCache.mu.Lock()
|
|
cert := certCache.cache[certKey]
|
|
cert.OCSP = update.parsed
|
|
cert.Certificate.OCSPStaple = update.rawBytes
|
|
certCache.cache[certKey] = cert
|
|
certCache.mu.Unlock()
|
|
}
|
|
}
|
|
|
|
// CleanStorageOptions specifies how to clean up a storage unit.
|
|
type CleanStorageOptions struct {
|
|
OCSPStaples bool
|
|
// TODO: long-expired certificates
|
|
}
|
|
|
|
// CleanStorage tidies up the given storage according to opts; this
|
|
// generally involves deleting assets which are no longer required.
|
|
// TODO: We should do this for long-expired certificates, too.
|
|
func CleanStorage(storage Storage, opts CleanStorageOptions) {
|
|
if opts.OCSPStaples {
|
|
err := deleteOldOCSPStaples(storage)
|
|
if err != nil {
|
|
log.Printf("[ERROR] Deleting old OCSP staples: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func deleteOldOCSPStaples(storage Storage) error {
|
|
ocspKeys, err := storage.List(prefixOCSP, false)
|
|
if err != nil {
|
|
// maybe just hasn't been created yet; no big deal
|
|
return nil
|
|
}
|
|
for _, key := range ocspKeys {
|
|
ocspBytes, err := storage.Load(key)
|
|
if err != nil {
|
|
log.Printf("[ERROR] While deleting old OCSP staples, unable to load staple file: %v", err)
|
|
continue
|
|
}
|
|
resp, err := ocsp.ParseResponse(ocspBytes, nil)
|
|
if err != nil {
|
|
// contents are invalid; delete it
|
|
err = storage.Delete(key)
|
|
if err != nil {
|
|
log.Printf("[ERROR] Purging corrupt staple file %s: %v", key, err)
|
|
}
|
|
continue
|
|
}
|
|
if time.Now().After(resp.NextUpdate) {
|
|
// response has expired; delete it
|
|
err = storage.Delete(key)
|
|
if err != nil {
|
|
log.Printf("[ERROR] Purging expired staple file %s: %v", key, err)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
const (
|
|
// DefaultRenewCheckInterval is how often to check certificates for renewal.
|
|
DefaultRenewCheckInterval = 12 * time.Hour
|
|
|
|
// DefaultRenewDurationBefore is how long before expiration to renew certificates.
|
|
DefaultRenewDurationBefore = (24 * time.Hour) * 30
|
|
|
|
// DefaultRenewDurationBeforeAtStartup is how long before expiration to require
|
|
// a renewed certificate when the process is first starting up (see mholt/caddy#1680).
|
|
DefaultRenewDurationBeforeAtStartup = (24 * time.Hour) * 7
|
|
|
|
// DefaultOCSPCheckInterval is how often to check if OCSP stapling needs updating.
|
|
DefaultOCSPCheckInterval = 1 * time.Hour
|
|
)
|