Fix edge case related to reloaded configs and ACME challenge

If Caddy is running but not listening on port 80, reloading Caddy with a new Caddyfile that needs to obtain a TLS cert from the CA would fail, because it was just assumed that, if reloading, port 80 as already in use. That is not always the case, so we scan the servers to see if one of them is listening on port 80, and we configure the ACME client accordingly. Kind of a hack... but it works.
This commit is contained in:
Matthew Holt 2016-02-11 15:37:51 -07:00
parent 1fe39e4633
commit 7bd2adf0dc
2 changed files with 35 additions and 7 deletions

View File

@ -49,7 +49,7 @@ func Activate(configs []server.Config) ([]server.Config, error) {
MarkQualified(configs) MarkQualified(configs)
// place certificates and keys on disk // place certificates and keys on disk
err := ObtainCerts(configs, true) err := ObtainCerts(configs, true, false)
if err != nil { if err != nil {
return configs, err return configs, err
} }
@ -109,10 +109,12 @@ func MarkQualified(configs []server.Config) {
} }
} }
// ObtainCerts obtains certificates for all these configs as long as a certificate does not // ObtainCerts obtains certificates for all these configs as long as a
// already exist on disk. It does not modify the configs at all; it only obtains and stores // certificate does not already exist on disk. It does not modify the
// certificates and keys to the disk. // configs at all; it only obtains and stores certificates and keys to
func ObtainCerts(configs []server.Config, allowPrompts bool) error { // the disk. If allowPrompts is true, the user may be shown a prompt.
// If proxyACME is true, the ACME challenges will be proxied to our alt port.
func ObtainCerts(configs []server.Config, allowPrompts, proxyACME bool) error {
// We group configs by email so we don't make the same clients over and // We group configs by email so we don't make the same clients over and
// over. This has the potential to prompt the user for an email, but we // over. This has the potential to prompt the user for an email, but we
// prevent that by assuming that if we already have a listener that can // prevent that by assuming that if we already have a listener that can
@ -131,7 +133,19 @@ func ObtainCerts(configs []server.Config, allowPrompts bool) error {
continue continue
} }
client.Configure(cfg.BindHost) // c.Configure assumes that allowPrompts == !proxyACME,
// but that's not always true. For example, a restart where
// the user isn't present and we're not listening on port 80.
// TODO: This could probably be refactored better.
if proxyACME {
client.SetHTTPAddress(net.JoinHostPort(cfg.BindHost, AlternatePort))
client.SetTLSAddress(net.JoinHostPort(cfg.BindHost, AlternatePort))
client.ExcludeChallenges([]acme.Challenge{acme.TLSSNI01, acme.DNS01})
} else {
client.SetHTTPAddress(net.JoinHostPort(cfg.BindHost, ""))
client.SetTLSAddress(net.JoinHostPort(cfg.BindHost, ""))
client.ExcludeChallenges([]acme.Challenge{acme.DNS01})
}
err := client.Obtain([]string{cfg.Host}) err := client.Obtain([]string{cfg.Host})
if err != nil { if err != nil {

View File

@ -8,6 +8,7 @@ import (
"errors" "errors"
"io/ioutil" "io/ioutil"
"log" "log"
"net"
"os" "os"
"os/exec" "os/exec"
"path" "path"
@ -142,8 +143,21 @@ func getCertsForNewCaddyfile(newCaddyfile Input) error {
// (can ignore error since we aren't actually using the certs) // (can ignore error since we aren't actually using the certs)
https.EnableTLS(configs, false) https.EnableTLS(configs, false)
// find out if we can let the acme package start its own challenge listener
// on port 80
var proxyACME bool
serversMu.Lock()
for _, s := range servers {
_, port, _ := net.SplitHostPort(s.Addr)
if port == "80" {
proxyACME = true
break
}
}
serversMu.Unlock()
// place certs on the disk // place certs on the disk
err = https.ObtainCerts(configs, false) err = https.ObtainCerts(configs, false, proxyACME)
if err != nil { if err != nil {
return errors.New("obtaining certs: " + err.Error()) return errors.New("obtaining certs: " + err.Error())
} }