mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-13 22:51:08 -05:00
390 lines
11 KiB
Go
390 lines
11 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 (
|
|
"crypto/tls"
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/xenolf/lego/certcrypto"
|
|
"github.com/xenolf/lego/certificate"
|
|
"github.com/xenolf/lego/challenge"
|
|
"github.com/xenolf/lego/challenge/tlsalpn01"
|
|
"github.com/xenolf/lego/lego"
|
|
)
|
|
|
|
// Config configures a certificate manager instance.
|
|
// An empty Config is not valid: use New() to obtain
|
|
// a valid Config.
|
|
type Config struct {
|
|
// The endpoint of the directory for the ACME
|
|
// CA we are to use
|
|
CA string
|
|
|
|
// The email address to use when creating or
|
|
// selecting an existing ACME server account
|
|
Email string
|
|
|
|
// Set to true if agreed to the CA's
|
|
// subscriber agreement
|
|
Agreed bool
|
|
|
|
// Disable all HTTP challenges
|
|
DisableHTTPChallenge bool
|
|
|
|
// Disable all TLS-ALPN challenges
|
|
DisableTLSALPNChallenge bool
|
|
|
|
// How long before expiration to renew certificates
|
|
RenewDurationBefore time.Duration
|
|
|
|
// How long before expiration to require a renewed
|
|
// certificate when in interactive mode, like when
|
|
// the program is first starting up (see
|
|
// mholt/caddy#1680). A wider window between
|
|
// RenewDurationBefore and this value will suppress
|
|
// errors under duress (bad) but hopefully this duration
|
|
// will give it enough time for the blockage to be
|
|
// relieved.
|
|
RenewDurationBeforeAtStartup time.Duration
|
|
|
|
// An optional event callback clients can set
|
|
// to subscribe to certain things happening
|
|
// internally by this config; invocations are
|
|
// synchronous, so make them return quickly!
|
|
OnEvent func(event string, data interface{})
|
|
|
|
// The host (ONLY the host, not port) to listen
|
|
// on if necessary to start a listener to solve
|
|
// an ACME challenge
|
|
ListenHost string
|
|
|
|
// The alternate port to use for the ACME HTTP
|
|
// challenge; if non-empty, this port will be
|
|
// used instead of HTTPChallengePort to spin up
|
|
// a listener for the HTTP challenge
|
|
AltHTTPPort int
|
|
|
|
// The alternate port to use for the ACME
|
|
// TLS-ALPN challenge; the system must forward
|
|
// TLSALPNChallengePort to this port for
|
|
// challenge to succeed
|
|
AltTLSALPNPort int
|
|
|
|
// The DNS provider to use when solving the
|
|
// ACME DNS challenge
|
|
DNSProvider challenge.Provider
|
|
|
|
// The type of key to use when generating
|
|
// certificates
|
|
KeyType certcrypto.KeyType
|
|
|
|
// The state needed to operate on-demand TLS
|
|
OnDemand *OnDemandConfig
|
|
|
|
// Add the must staple TLS extension to the
|
|
// CSR generated by lego/acme
|
|
MustStaple bool
|
|
|
|
// Map of hostname to certificate hash; used
|
|
// to complete handshakes and serve the right
|
|
// certificate given SNI
|
|
certificates map[string]string
|
|
|
|
// Pointer to the certificate store to use
|
|
certCache *Cache
|
|
|
|
// Map of client config key to ACME clients
|
|
// so they can be reused
|
|
acmeClients map[string]*lego.Client
|
|
acmeClientsMu *sync.Mutex
|
|
}
|
|
|
|
// NewDefault returns a new, valid, default config.
|
|
//
|
|
// Calling this function signifies your acceptance to
|
|
// the CA's Subscriber Agreement and/or Terms of Service.
|
|
func NewDefault() *Config {
|
|
return New(Config{Agreed: true})
|
|
}
|
|
|
|
// New makes a valid config based on cfg and uses
|
|
// a default certificate cache. All calls to
|
|
// New() will use the same certificate cache.
|
|
func New(cfg Config) *Config {
|
|
return NewWithCache(nil, cfg)
|
|
}
|
|
|
|
// NewWithCache makes a valid new config based on cfg
|
|
// and uses the provided certificate cache. If certCache
|
|
// is nil, a new, default one will be created using
|
|
// DefaultStorage; or, if a default cache has already
|
|
// been created, it will be reused.
|
|
func NewWithCache(certCache *Cache, cfg Config) *Config {
|
|
// avoid nil pointers with sensible defaults,
|
|
// careful to initialize a default cache (which
|
|
// begins its maintenance goroutine) only if
|
|
// needed - and only once (we don't initialize
|
|
// it at package init to give importers a chance
|
|
// to set DefaultStorage if they so desire)
|
|
if certCache == nil {
|
|
defaultCacheMu.Lock()
|
|
if defaultCache == nil {
|
|
defaultCache = NewCache(DefaultStorage)
|
|
}
|
|
certCache = defaultCache
|
|
defaultCacheMu.Unlock()
|
|
}
|
|
if certCache.storage == nil {
|
|
certCache.storage = DefaultStorage
|
|
}
|
|
|
|
// fill in default values
|
|
if cfg.CA == "" {
|
|
cfg.CA = CA
|
|
}
|
|
if cfg.Email == "" {
|
|
cfg.Email = Email
|
|
}
|
|
if cfg.OnDemand == nil {
|
|
cfg.OnDemand = OnDemand
|
|
}
|
|
if !cfg.Agreed {
|
|
cfg.Agreed = Agreed
|
|
}
|
|
if !cfg.DisableHTTPChallenge {
|
|
cfg.DisableHTTPChallenge = DisableHTTPChallenge
|
|
}
|
|
if !cfg.DisableTLSALPNChallenge {
|
|
cfg.DisableTLSALPNChallenge = DisableTLSALPNChallenge
|
|
}
|
|
if cfg.RenewDurationBefore == 0 {
|
|
cfg.RenewDurationBefore = RenewDurationBefore
|
|
}
|
|
if cfg.RenewDurationBeforeAtStartup == 0 {
|
|
cfg.RenewDurationBeforeAtStartup = RenewDurationBeforeAtStartup
|
|
}
|
|
if cfg.OnEvent == nil {
|
|
cfg.OnEvent = OnEvent
|
|
}
|
|
if cfg.ListenHost == "" {
|
|
cfg.ListenHost = ListenHost
|
|
}
|
|
if cfg.AltHTTPPort == 0 {
|
|
cfg.AltHTTPPort = AltHTTPPort
|
|
}
|
|
if cfg.AltTLSALPNPort == 0 {
|
|
cfg.AltTLSALPNPort = AltTLSALPNPort
|
|
}
|
|
if cfg.DNSProvider == nil {
|
|
cfg.DNSProvider = DNSProvider
|
|
}
|
|
if cfg.KeyType == "" {
|
|
cfg.KeyType = KeyType
|
|
}
|
|
if cfg.OnDemand == nil {
|
|
cfg.OnDemand = OnDemand
|
|
}
|
|
if !cfg.MustStaple {
|
|
cfg.MustStaple = MustStaple
|
|
}
|
|
|
|
// ensure the unexported fields are valid
|
|
cfg.certificates = make(map[string]string)
|
|
cfg.certCache = certCache
|
|
cfg.acmeClients = make(map[string]*lego.Client)
|
|
cfg.acmeClientsMu = new(sync.Mutex)
|
|
|
|
return &cfg
|
|
}
|
|
|
|
// Manage causes the certificates for domainNames to be managed
|
|
// 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
|
|
if cfg.OnDemand != nil {
|
|
if !cfg.OnDemand.whitelistContains(domainName) {
|
|
cfg.OnDemand.HostWhitelist = append(cfg.OnDemand.HostWhitelist, domainName)
|
|
}
|
|
continue
|
|
}
|
|
|
|
// try loading an existing certificate; if it doesn't
|
|
// exist yet, obtain one and try loading it again
|
|
cert, err := cfg.CacheManagedCertificate(domainName)
|
|
if err != nil {
|
|
if _, ok := err.(ErrNotExist); ok {
|
|
// if it doesn't exist, get it, then try loading it again
|
|
err := cfg.ObtainCert(domainName, false)
|
|
if err != nil {
|
|
return fmt.Errorf("%s: obtaining certificate: %v", domainName, err)
|
|
}
|
|
cert, err = cfg.CacheManagedCertificate(domainName)
|
|
if err != nil {
|
|
return fmt.Errorf("%s: caching certificate after obtaining it: %v", domainName, err)
|
|
}
|
|
continue
|
|
}
|
|
return fmt.Errorf("%s: caching certificate: %v", domainName, err)
|
|
}
|
|
|
|
// for existing certificates, make sure it is renewed
|
|
if cert.NeedsRenewal() {
|
|
err := cfg.RenewCert(domainName, false)
|
|
if err != nil {
|
|
return fmt.Errorf("%s: renewing certificate: %v", domainName, err)
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// ObtainCert obtains a certificate for name using cfg, as long
|
|
// as a certificate does not already exist in storage for that
|
|
// name. The name must qualify and cfg must be flagged as Managed.
|
|
// This function is a no-op if storage already has a certificate
|
|
// for name.
|
|
//
|
|
// It only obtains and stores certificates (and their keys),
|
|
// it does not load them into memory. If interactive is true,
|
|
// the user may be shown a prompt.
|
|
func (cfg *Config) ObtainCert(name string, interactive bool) error {
|
|
skip, err := cfg.preObtainOrRenewChecks(name, interactive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if skip {
|
|
return nil
|
|
}
|
|
|
|
client, err := cfg.newACMEClient(interactive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return client.Obtain(name)
|
|
}
|
|
|
|
// RenewCert renews the certificate for name using cfg. It stows the
|
|
// renewed certificate and its assets in storage if successful.
|
|
func (cfg *Config) RenewCert(name string, interactive bool) error {
|
|
skip, err := cfg.preObtainOrRenewChecks(name, interactive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if skip {
|
|
return nil
|
|
}
|
|
client, err := cfg.newACMEClient(interactive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return client.Renew(name)
|
|
}
|
|
|
|
// RevokeCert revokes the certificate for domain via ACME protocol.
|
|
func (cfg *Config) RevokeCert(domain string, interactive bool) error {
|
|
client, err := cfg.newACMEClient(interactive)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return client.Revoke(domain)
|
|
}
|
|
|
|
// TLSConfig is an opinionated method that returns a
|
|
// recommended, modern TLS configuration that can be
|
|
// used to configure TLS listeners, which also supports
|
|
// the TLS-ALPN challenge and serves up certificates
|
|
// managed by cfg.
|
|
//
|
|
// Unlike the package TLS() function, this method does
|
|
// not, by itself, enable certificate management for
|
|
// any domain names.
|
|
//
|
|
// Feel free to further customize the returned tls.Config,
|
|
// but do not mess with the GetCertificate or NextProtos
|
|
// fields unless you know what you're doing, as they're
|
|
// necessary to solve the TLS-ALPN challenge.
|
|
func (cfg *Config) TLSConfig() *tls.Config {
|
|
return &tls.Config{
|
|
// these two fields necessary for TLS-ALPN challenge
|
|
GetCertificate: cfg.GetCertificate,
|
|
NextProtos: []string{"h2", "http/1.1", tlsalpn01.ACMETLS1Protocol},
|
|
|
|
// the rest recommended for modern TLS servers
|
|
MinVersion: tls.VersionTLS12,
|
|
CurvePreferences: []tls.CurveID{
|
|
tls.X25519,
|
|
tls.CurveP256,
|
|
},
|
|
CipherSuites: preferredDefaultCipherSuites(),
|
|
PreferServerCipherSuites: true,
|
|
}
|
|
}
|
|
|
|
// preObtainOrRenewChecks perform a few simple checks before
|
|
// obtaining or renewing a certificate with ACME, and returns
|
|
// whether this name should be skipped (like if it's not
|
|
// managed TLS) as well as any error. It ensures that the
|
|
// config is Managed, that the name qualifies for a certificate,
|
|
// and that an email address is available.
|
|
func (cfg *Config) preObtainOrRenewChecks(name string, allowPrompts bool) (bool, error) {
|
|
if !HostQualifies(name) {
|
|
return true, nil
|
|
}
|
|
|
|
if cfg.Email == "" {
|
|
var err error
|
|
cfg.Email, err = cfg.getEmail(allowPrompts)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
// storageHasCertResources returns true if the storage
|
|
// associated with cfg's certificate cache has all the
|
|
// resources related to the certificate for domain: the
|
|
// certificate, the private key, and the metadata.
|
|
func (cfg *Config) storageHasCertResources(domain string) bool {
|
|
certKey := StorageKeys.SiteCert(cfg.CA, domain)
|
|
keyKey := StorageKeys.SitePrivateKey(cfg.CA, domain)
|
|
metaKey := StorageKeys.SiteMeta(cfg.CA, domain)
|
|
return cfg.certCache.storage.Exists(certKey) &&
|
|
cfg.certCache.storage.Exists(keyKey) &&
|
|
cfg.certCache.storage.Exists(metaKey)
|
|
}
|
|
|
|
// managedCertNeedsRenewal returns true if certRes is
|
|
// expiring soon or already expired, or if the process
|
|
// of checking the expiration returned an error.
|
|
func (cfg *Config) managedCertNeedsRenewal(certRes certificate.Resource) bool {
|
|
cert, err := cfg.makeCertificate(certRes.Certificate, certRes.PrivateKey)
|
|
if err != nil {
|
|
return true
|
|
}
|
|
return cert.NeedsRenewal()
|
|
}
|