mirror of
https://github.com/caddyserver/caddy.git
synced 2026-05-26 08:42:31 -04:00
Add clustering plugin types; use latest certmagic.Storage interface
This commit is contained in:
+19
-40
@@ -12,6 +12,25 @@
|
||||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
// Package certmagic automates the obtaining and renewal of TLS certificates,
|
||||
// including TLS & HTTPS best practices such as robust OCSP stapling, caching,
|
||||
// HTTP->HTTPS redirects, and more.
|
||||
//
|
||||
// Its high-level API serves your HTTP handlers over HTTPS by simply giving
|
||||
// the domain name(s) and the http.Handler; CertMagic will create and run
|
||||
// the HTTPS server for you, fully managing certificates during the lifetime
|
||||
// of the server. Similarly, it can be used to start TLS listeners or return
|
||||
// a ready-to-use tls.Config -- whatever layer you need TLS for, CertMagic
|
||||
// makes it easy.
|
||||
//
|
||||
// If you need more control, create a Config using New() and then call
|
||||
// Manage() on the config; but you'll have to be sure to solve the HTTP
|
||||
// and TLS-ALPN challenges yourself (unless you disabled them or use the
|
||||
// DNS challenge) by using the provided Config.GetCertificate function
|
||||
// in your tls.Config and/or Config.HTTPChallangeHandler in your HTTP
|
||||
// handler.
|
||||
//
|
||||
// See the package's README for more instruction.
|
||||
package certmagic
|
||||
|
||||
import (
|
||||
@@ -166,46 +185,6 @@ func manageWithDefaultConfig(domainNames []string, disableHTTPChallenge bool) (*
|
||||
return cfg, cfg.Manage(domainNames)
|
||||
}
|
||||
|
||||
// Locker facilitates synchronization of certificate tasks across
|
||||
// machines and networks.
|
||||
type Locker interface {
|
||||
// TryLock will attempt to acquire the lock for key. If a
|
||||
// lock could be obtained, nil values are returned as no
|
||||
// waiting is required. If not (meaning another process is
|
||||
// already working on key), a Waiter value will be returned,
|
||||
// upon which you should Wait() until it is finished.
|
||||
//
|
||||
// The key should be a carefully-chosen value that uniquely
|
||||
// and precisely identifies the operation being locked. For
|
||||
// example, if it is for a certificate obtain or renew with
|
||||
// the ACME protocol to the same CA endpoint (remembering
|
||||
// that an obtain and renew are the same according to ACME,
|
||||
// thus both obtain and renew should share a lock key), a
|
||||
// good key would identify that operation by some name,
|
||||
// concatenated with the domain name and the CA endpoint.
|
||||
//
|
||||
// TryLock never blocks; it always returns without waiting.
|
||||
//
|
||||
// To prevent deadlocks, all implementations (where this concern
|
||||
// is relevant) should put a reasonable expiration on the lock in
|
||||
// case Unlock is unable to be called due to some sort of storage
|
||||
// system failure or crash.
|
||||
TryLock(key string) (Waiter, error)
|
||||
|
||||
// Unlock releases the lock for key. This method must ONLY be
|
||||
// called after a successful call to TryLock where no Waiter was
|
||||
// returned, and only after the operation requiring the lock is
|
||||
// finished, even if it returned an error or timed out. Unlock
|
||||
// should also clean up any unused resources allocated during
|
||||
// TryLock.
|
||||
Unlock(key string) error
|
||||
}
|
||||
|
||||
// Waiter is a type that can block until a lock is released.
|
||||
type Waiter interface {
|
||||
Wait()
|
||||
}
|
||||
|
||||
// OnDemandConfig contains some state relevant for providing
|
||||
// on-demand TLS.
|
||||
type OnDemandConfig struct {
|
||||
|
||||
+28
-32
@@ -217,23 +217,21 @@ func (cfg *Config) lockKey(op, domainName string) string {
|
||||
// Callers who have access to a Config value should use the ObtainCert
|
||||
// method on that instead of this lower-level method.
|
||||
func (c *acmeClient) Obtain(name string) error {
|
||||
if c.config.Sync != nil {
|
||||
lockKey := c.config.lockKey("cert_acme", name)
|
||||
waiter, err := c.config.Sync.TryLock(lockKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if waiter != nil {
|
||||
log.Printf("[INFO] Certificate for %s is already being obtained elsewhere and stored; waiting", name)
|
||||
waiter.Wait()
|
||||
return nil // we assume the process with the lock succeeded, rather than hammering this execution path again
|
||||
}
|
||||
defer func() {
|
||||
if err := c.config.Sync.Unlock(lockKey); err != nil {
|
||||
log.Printf("[ERROR] Unable to unlock obtain call for %s: %v", name, err)
|
||||
}
|
||||
}()
|
||||
lockKey := c.config.lockKey("cert_acme", name)
|
||||
waiter, err := c.config.certCache.storage.TryLock(lockKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if waiter != nil {
|
||||
log.Printf("[INFO] Certificate for %s is already being obtained elsewhere and stored; waiting", name)
|
||||
waiter.Wait()
|
||||
return nil // we assume the process with the lock succeeded, rather than hammering this execution path again
|
||||
}
|
||||
defer func() {
|
||||
if err := c.config.certCache.storage.Unlock(lockKey); err != nil {
|
||||
log.Printf("[ERROR] Unable to unlock obtain call for %s: %v", name, err)
|
||||
}
|
||||
}()
|
||||
|
||||
for attempts := 0; attempts < 2; attempts++ {
|
||||
request := certificate.ObtainRequest{
|
||||
@@ -276,23 +274,21 @@ func (c *acmeClient) Obtain(name string) error {
|
||||
// Callers who have access to a Config value should use the RenewCert
|
||||
// method on that instead of this lower-level method.
|
||||
func (c *acmeClient) Renew(name string) error {
|
||||
if c.config.Sync != nil {
|
||||
lockKey := c.config.lockKey("cert_acme", name)
|
||||
waiter, err := c.config.Sync.TryLock(lockKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if waiter != nil {
|
||||
log.Printf("[INFO] Certificate for %s is already being renewed elsewhere and stored; waiting", name)
|
||||
waiter.Wait()
|
||||
return nil // assume that the worker that renewed the cert succeeded; avoid hammering this path over and over
|
||||
}
|
||||
defer func() {
|
||||
if err := c.config.Sync.Unlock(lockKey); err != nil {
|
||||
log.Printf("[ERROR] Unable to unlock renew call for %s: %v", name, err)
|
||||
}
|
||||
}()
|
||||
lockKey := c.config.lockKey("cert_acme", name)
|
||||
waiter, err := c.config.certCache.storage.TryLock(lockKey)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if waiter != nil {
|
||||
log.Printf("[INFO] Certificate for %s is already being renewed elsewhere and stored; waiting", name)
|
||||
waiter.Wait()
|
||||
return nil // assume that the worker that renewed the cert succeeded to avoid hammering this path over and over
|
||||
}
|
||||
defer func() {
|
||||
if err := c.config.certCache.storage.Unlock(lockKey); err != nil {
|
||||
log.Printf("[ERROR] Unable to unlock renew call for %s: %v", name, err)
|
||||
}
|
||||
}()
|
||||
|
||||
// Prepare for renewal (load PEM cert, key, and meta)
|
||||
certRes, err := c.config.loadCertResource(name)
|
||||
|
||||
+6
-24
@@ -38,15 +38,6 @@ type Config struct {
|
||||
// selecting an existing ACME server account
|
||||
Email string
|
||||
|
||||
// The synchronization implementation - although
|
||||
// it is not strictly required to have a Sync
|
||||
// value in general, all instances running in
|
||||
// in a cluster for the same domain names must
|
||||
// specify a Sync and use the same one, otherwise
|
||||
// some cert operations will not be properly
|
||||
// coordinated
|
||||
Sync Locker
|
||||
|
||||
// Set to true if agreed to the CA's
|
||||
// subscriber agreement
|
||||
Agreed bool
|
||||
@@ -198,20 +189,6 @@ func NewWithCache(certCache *Cache, cfg Config) *Config {
|
||||
cfg.MustStaple = MustStaple
|
||||
}
|
||||
|
||||
// if no sync facility is provided, we'll default to
|
||||
// a file system synchronizer backed by the storage
|
||||
// given to certCache (if it is one), or just a simple
|
||||
// in-memory sync facility otherwise (strictly speaking,
|
||||
// a sync is not required; only if running multiple
|
||||
// instances for the same domain names concurrently)
|
||||
if cfg.Sync == nil {
|
||||
if ccfs, ok := certCache.storage.(FileStorage); ok {
|
||||
cfg.Sync = NewFileStorageLocker(ccfs)
|
||||
} else {
|
||||
cfg.Sync = NewMemoryLocker()
|
||||
}
|
||||
}
|
||||
|
||||
// ensure the unexported fields are valid
|
||||
cfg.certificates = make(map[string]string)
|
||||
cfg.certCache = certCache
|
||||
@@ -222,7 +199,12 @@ func NewWithCache(certCache *Cache, cfg Config) *Config {
|
||||
}
|
||||
|
||||
// Manage causes the certificates for domainNames to be managed
|
||||
// according to cfg.
|
||||
// according to cfg. If cfg is enabled for OnDemand, then this
|
||||
// simply whitelists the domain names. Otherwise, the certificate(s)
|
||||
// for each name are loaded from storage or obtained from the CA;
|
||||
// and if loaded from storage, renewed if they are expiring or
|
||||
// expired. It then caches the certificate in memory and is
|
||||
// prepared to serve them up during TLS handshakes.
|
||||
func (cfg *Config) Manage(domainNames []string) error {
|
||||
for _, domainName := range domainNames {
|
||||
// if on-demand is configured, simply whitelist this name
|
||||
|
||||
+113
@@ -15,10 +15,13 @@
|
||||
package certmagic
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"runtime"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
// FileStorage facilitates forming file paths derived from a root
|
||||
@@ -123,4 +126,114 @@ func dataDir() string {
|
||||
return filepath.Join(baseDir, "certmagic")
|
||||
}
|
||||
|
||||
// TryLock attempts to get a lock for name, otherwise it returns
|
||||
// a Waiter value to wait until the other process is finished.
|
||||
func (fs FileStorage) TryLock(key string) (Waiter, error) {
|
||||
fileStorageNameLocksMu.Lock()
|
||||
defer fileStorageNameLocksMu.Unlock()
|
||||
|
||||
// see if lock already exists within this process - allows
|
||||
// for faster unlocking since we don't have to poll the disk
|
||||
fw, ok := fileStorageNameLocks[key]
|
||||
if ok {
|
||||
// lock already created within process, let caller wait on it
|
||||
return fw, nil
|
||||
}
|
||||
|
||||
// attempt to persist lock to disk by creating lock file
|
||||
|
||||
// parent dir must exist
|
||||
lockDir := fs.lockDir()
|
||||
if err := os.MkdirAll(lockDir, 0700); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
fw = &fileStorageWaiter{
|
||||
filename: filepath.Join(lockDir, safeKey(key)+".lock"),
|
||||
wg: new(sync.WaitGroup),
|
||||
}
|
||||
|
||||
// create the file in a special mode such that an
|
||||
// error is returned if it already exists
|
||||
lf, err := os.OpenFile(fw.filename, os.O_CREATE|os.O_EXCL, 0644)
|
||||
if err != nil {
|
||||
if os.IsExist(err) {
|
||||
// another process has the lock; use it to wait
|
||||
return fw, nil
|
||||
}
|
||||
// otherwise, this was some unexpected error
|
||||
return nil, err
|
||||
}
|
||||
lf.Close()
|
||||
|
||||
// looks like we get the lock
|
||||
fw.wg.Add(1)
|
||||
fileStorageNameLocks[key] = fw
|
||||
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// Unlock releases the lock for name.
|
||||
func (fs FileStorage) Unlock(key string) error {
|
||||
fileStorageNameLocksMu.Lock()
|
||||
defer fileStorageNameLocksMu.Unlock()
|
||||
|
||||
fw, ok := fileStorageNameLocks[key]
|
||||
if !ok {
|
||||
return fmt.Errorf("FileStorage: no lock to release for %s", key)
|
||||
}
|
||||
|
||||
// remove lock file
|
||||
os.Remove(fw.filename)
|
||||
|
||||
// if parent folder is now empty, remove it too to keep it tidy
|
||||
dir, err := os.Open(fs.lockDir()) // OK to ignore error here
|
||||
if err == nil {
|
||||
items, _ := dir.Readdirnames(3) // OK to ignore error here
|
||||
if len(items) == 0 {
|
||||
os.Remove(dir.Name())
|
||||
}
|
||||
dir.Close()
|
||||
}
|
||||
|
||||
// clean up in memory
|
||||
fw.wg.Done()
|
||||
delete(fileStorageNameLocks, key)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (fs FileStorage) lockDir() string {
|
||||
return filepath.Join(fs.Path, "locks")
|
||||
}
|
||||
|
||||
// fileStorageWaiter waits for a file to disappear; it
|
||||
// polls the file system to check for the existence of
|
||||
// a file. It also uses a WaitGroup to optimize the
|
||||
// polling in the case when this process is the only
|
||||
// one waiting. (Other processes that are waiting for
|
||||
// the lock will still block, but must wait for the
|
||||
// polling to get their answer.)
|
||||
type fileStorageWaiter struct {
|
||||
filename string
|
||||
wg *sync.WaitGroup
|
||||
}
|
||||
|
||||
// Wait waits until the lock is released.
|
||||
func (fw *fileStorageWaiter) Wait() {
|
||||
start := time.Now()
|
||||
fw.wg.Wait()
|
||||
for time.Since(start) < 1*time.Hour {
|
||||
_, err := os.Stat(fw.filename)
|
||||
if os.IsNotExist(err) {
|
||||
return
|
||||
}
|
||||
time.Sleep(1 * time.Second)
|
||||
}
|
||||
}
|
||||
|
||||
var fileStorageNameLocks = make(map[string]*fileStorageWaiter)
|
||||
var fileStorageNameLocksMu sync.Mutex
|
||||
|
||||
var _ Storage = FileStorage{}
|
||||
var _ Waiter = &fileStorageWaiter{}
|
||||
|
||||
-146
@@ -1,146 +0,0 @@
|
||||
// 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 (
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
// FileStorageLocker implements the Locker interface
|
||||
// using the file system. An empty value is NOT VALID,
|
||||
// so you must use NewFileStorageLocker() to get one.
|
||||
type FileStorageLocker struct {
|
||||
fs FileStorage
|
||||
}
|
||||
|
||||
// NewFileStorageLocker returns a valid Locker backed by fs.
|
||||
func NewFileStorageLocker(fs FileStorage) *FileStorageLocker {
|
||||
return &FileStorageLocker{fs: fs}
|
||||
}
|
||||
|
||||
// TryLock attempts to get a lock for name, otherwise it returns
|
||||
// a Waiter value to wait until the other process is finished.
|
||||
func (l *FileStorageLocker) TryLock(name string) (Waiter, error) {
|
||||
fileStorageNameLocksMu.Lock()
|
||||
defer fileStorageNameLocksMu.Unlock()
|
||||
|
||||
// see if lock already exists within this process
|
||||
fw, ok := fileStorageNameLocks[name]
|
||||
if ok {
|
||||
// lock already created within process, let caller wait on it
|
||||
return fw, nil
|
||||
}
|
||||
|
||||
// attempt to persist lock to disk by creating lock file
|
||||
|
||||
// parent dir must exist
|
||||
lockDir := l.lockDir()
|
||||
if err := os.MkdirAll(lockDir, 0700); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
fw = &FileStorageWaiter{
|
||||
filename: filepath.Join(lockDir, safeKey(name)+".lock"),
|
||||
wg: new(sync.WaitGroup),
|
||||
}
|
||||
|
||||
// create the file in a special mode such that an
|
||||
// error is returned if it already exists
|
||||
lf, err := os.OpenFile(fw.filename, os.O_CREATE|os.O_EXCL, 0644)
|
||||
if err != nil {
|
||||
if os.IsExist(err) {
|
||||
// another process has the lock; use it to wait
|
||||
return fw, nil
|
||||
}
|
||||
// otherwise, this was some unexpected error
|
||||
return nil, err
|
||||
}
|
||||
lf.Close()
|
||||
|
||||
// looks like we get the lock
|
||||
fw.wg.Add(1)
|
||||
fileStorageNameLocks[name] = fw
|
||||
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// Unlock releases the lock for name.
|
||||
func (l *FileStorageLocker) Unlock(name string) error {
|
||||
fileStorageNameLocksMu.Lock()
|
||||
defer fileStorageNameLocksMu.Unlock()
|
||||
|
||||
fw, ok := fileStorageNameLocks[name]
|
||||
if !ok {
|
||||
return fmt.Errorf("FileStorageLocker: no lock to release for %s", name)
|
||||
}
|
||||
|
||||
// remove lock file
|
||||
os.Remove(fw.filename)
|
||||
|
||||
// if parent folder is now empty, remove it too to keep it tidy
|
||||
dir, err := os.Open(l.lockDir()) // OK to ignore error here
|
||||
if err == nil {
|
||||
items, _ := dir.Readdirnames(3) // OK to ignore error here
|
||||
if len(items) == 0 {
|
||||
os.Remove(dir.Name())
|
||||
}
|
||||
dir.Close()
|
||||
}
|
||||
|
||||
// clean up in memory
|
||||
fw.wg.Done()
|
||||
delete(fileStorageNameLocks, name)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (l *FileStorageLocker) lockDir() string {
|
||||
return filepath.Join(l.fs.Path, "locks")
|
||||
}
|
||||
|
||||
// FileStorageWaiter waits for a file to disappear; it
|
||||
// polls the file system to check for the existence of
|
||||
// a file. It also uses a WaitGroup to optimize the
|
||||
// polling in the case when this process is the only
|
||||
// one waiting. (Other processes that are waiting
|
||||
// for the lock will still block, but must wait
|
||||
// for the poll intervals to get their answer.)
|
||||
type FileStorageWaiter struct {
|
||||
filename string
|
||||
wg *sync.WaitGroup
|
||||
}
|
||||
|
||||
// Wait waits until the lock is released.
|
||||
func (fw *FileStorageWaiter) Wait() {
|
||||
start := time.Now()
|
||||
fw.wg.Wait()
|
||||
for time.Since(start) < 1*time.Hour {
|
||||
_, err := os.Stat(fw.filename)
|
||||
if os.IsNotExist(err) {
|
||||
return
|
||||
}
|
||||
time.Sleep(1 * time.Second)
|
||||
}
|
||||
}
|
||||
|
||||
var fileStorageNameLocks = make(map[string]*FileStorageWaiter)
|
||||
var fileStorageNameLocksMu sync.Mutex
|
||||
|
||||
var _ Locker = &FileStorageLocker{}
|
||||
var _ Waiter = &FileStorageWaiter{}
|
||||
+42
-6
@@ -31,9 +31,9 @@ import (
|
||||
// in order to share certificates and other TLS resources
|
||||
// with the cluster.
|
||||
type Storage interface {
|
||||
// Exists returns true if the key exists
|
||||
// and there was no error checking.
|
||||
Exists(key string) bool
|
||||
// Locker provides atomic synchronization
|
||||
// operations, making Storage safe to share.
|
||||
Locker
|
||||
|
||||
// Store puts value at key.
|
||||
Store(key string, value []byte) error
|
||||
@@ -44,6 +44,10 @@ type Storage interface {
|
||||
// Delete deletes key.
|
||||
Delete(key string) error
|
||||
|
||||
// Exists returns true if the key exists
|
||||
// and there was no error checking.
|
||||
Exists(key string) bool
|
||||
|
||||
// List returns all keys that match prefix.
|
||||
List(prefix string) ([]string, error)
|
||||
|
||||
@@ -51,6 +55,41 @@ type Storage interface {
|
||||
Stat(key string) (KeyInfo, error)
|
||||
}
|
||||
|
||||
// Locker facilitates synchronization of certificate tasks across
|
||||
// machines and networks.
|
||||
type Locker interface {
|
||||
// TryLock will attempt to acquire the lock for key. If a
|
||||
// lock could be obtained, nil values are returned as no
|
||||
// waiting is required. If not (meaning another process is
|
||||
// already working on key), a Waiter value will be returned,
|
||||
// upon which you should Wait() until it is finished.
|
||||
//
|
||||
// The actual implementation of obtaining of a lock must be
|
||||
// an atomic operation so that multiple TryLock calls at the
|
||||
// same time always results in only one caller receiving the
|
||||
// lock. TryLock always returns without waiting.
|
||||
//
|
||||
// To prevent deadlocks, all implementations (where this concern
|
||||
// is relevant) should put a reasonable expiration on the lock in
|
||||
// case Unlock is unable to be called due to some sort of network
|
||||
// or system failure or crash.
|
||||
TryLock(key string) (Waiter, error)
|
||||
|
||||
// Unlock releases the lock for key. This method must ONLY be
|
||||
// called after a successful call to TryLock where no Waiter was
|
||||
// returned, and only after the operation requiring the lock is
|
||||
// finished, even if it errored or timed out. It is INCORRECT to
|
||||
// call Unlock if any non-nil value was returned from a call to
|
||||
// TryLock or if Unlock was not called at all. Unlock should also
|
||||
// clean up any unused resources allocated during TryLock.
|
||||
Unlock(key string) error
|
||||
}
|
||||
|
||||
// Waiter is a type that can block until a lock is released.
|
||||
type Waiter interface {
|
||||
Wait()
|
||||
}
|
||||
|
||||
// KeyInfo holds information about a key in storage.
|
||||
type KeyInfo struct {
|
||||
Key string
|
||||
@@ -207,6 +246,3 @@ var defaultFileStorage = FileStorage{Path: dataDir()}
|
||||
|
||||
// DefaultStorage is the default Storage implementation.
|
||||
var DefaultStorage Storage = defaultFileStorage
|
||||
|
||||
// DefaultSync is a default sync to use.
|
||||
var DefaultSync Locker
|
||||
|
||||
Reference in New Issue
Block a user