mirror of
https://github.com/caddyserver/caddy.git
synced 2024-12-23 22:27:38 -05:00
Addresses which fail to resolve are handled more gracefully in the two most common cases: the hostname doesn't resolve or the port is unknown (like "http" on a system that doesn't support that port name). If the hostname doesn't resolve, the host is served on the listener at host 0.0.0.0. If the port is unknown, we attempt to rewrite it as a number manually and try again.
This commit is contained in:
parent
640cd059ce
commit
d8391d6fbd
3 changed files with 123 additions and 9 deletions
|
@ -1,7 +1,6 @@
|
||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
"log"
|
||||||
|
@ -89,18 +88,22 @@ func Load(filename string, input io.Reader) ([]server.Config, error) {
|
||||||
// ArrangeBindings groups configurations by their bind address. For example,
|
// ArrangeBindings groups configurations by their bind address. For example,
|
||||||
// a server that should listen on localhost and another on 127.0.0.1 will
|
// a server that should listen on localhost and another on 127.0.0.1 will
|
||||||
// be grouped into the same address: 127.0.0.1. It will return an error
|
// be grouped into the same address: 127.0.0.1. It will return an error
|
||||||
// if the address lookup fails or if a TLS listener is configured on the
|
// if an address is malformed or a TLS listener is configured on the
|
||||||
// same address as a plaintext HTTP listener. The return value is a map of
|
// same address as a plaintext HTTP listener. The return value is a map of
|
||||||
// bind address to list of configs that would become VirtualHosts on that
|
// bind address to list of configs that would become VirtualHosts on that
|
||||||
// server.
|
// server. Use the keys of the returned map to create listeners, and use
|
||||||
|
// the associated values to set up the virtualhosts.
|
||||||
func ArrangeBindings(allConfigs []server.Config) (map[*net.TCPAddr][]server.Config, error) {
|
func ArrangeBindings(allConfigs []server.Config) (map[*net.TCPAddr][]server.Config, error) {
|
||||||
addresses := make(map[*net.TCPAddr][]server.Config)
|
addresses := make(map[*net.TCPAddr][]server.Config)
|
||||||
|
|
||||||
// Group configs by bind address
|
// Group configs by bind address
|
||||||
for _, conf := range allConfigs {
|
for _, conf := range allConfigs {
|
||||||
newAddr, err := net.ResolveTCPAddr("tcp", conf.Address())
|
newAddr, warnErr, fatalErr := resolveAddr(conf)
|
||||||
if err != nil {
|
if fatalErr != nil {
|
||||||
return addresses, errors.New("could not serve " + conf.Address() + " - " + err.Error())
|
return addresses, fatalErr
|
||||||
|
}
|
||||||
|
if warnErr != nil {
|
||||||
|
log.Println("[Warning]", warnErr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make sure to compare the string representation of the address,
|
// Make sure to compare the string representation of the address,
|
||||||
|
@ -139,6 +142,59 @@ func ArrangeBindings(allConfigs []server.Config) (map[*net.TCPAddr][]server.Conf
|
||||||
return addresses, nil
|
return addresses, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// resolveAddr determines the address (host and port) that a config will
|
||||||
|
// bind to. The returned address, resolvAddr, should be used to bind the
|
||||||
|
// listener or group the config with other configs using the same address.
|
||||||
|
// The first error, if not nil, is just a warning and should be reported
|
||||||
|
// but execution may continue. The second error, if not nil, is a real
|
||||||
|
// problem and the server should not be started.
|
||||||
|
//
|
||||||
|
// This function handles edge cases gracefully. If a port name like
|
||||||
|
// "http" or "https" is unknown to the system, this function will
|
||||||
|
// change them to 80 or 443 respectively. If a hostname fails to
|
||||||
|
// resolve, that host can still be served but will be listening on
|
||||||
|
// the wildcard host instead. This function takes care of this for you.
|
||||||
|
func resolveAddr(conf server.Config) (resolvAddr *net.TCPAddr, warnErr error, fatalErr error) {
|
||||||
|
// The host to bind to may be different from the (virtual)host to serve
|
||||||
|
bindHost := conf.BindHost
|
||||||
|
if bindHost == "" {
|
||||||
|
bindHost = conf.Host
|
||||||
|
}
|
||||||
|
|
||||||
|
resolvAddr, warnErr = net.ResolveTCPAddr("tcp", net.JoinHostPort(bindHost, conf.Port))
|
||||||
|
if warnErr != nil {
|
||||||
|
// Most likely the host lookup failed or the port is unknown
|
||||||
|
tryPort := conf.Port
|
||||||
|
|
||||||
|
switch errVal := warnErr.(type) {
|
||||||
|
case *net.AddrError:
|
||||||
|
if errVal.Err == "unknown port" {
|
||||||
|
// some odd Linux machines don't support these port names; see issue #136
|
||||||
|
switch conf.Port {
|
||||||
|
case "http":
|
||||||
|
tryPort = "80"
|
||||||
|
case "https":
|
||||||
|
tryPort = "443"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
resolvAddr, fatalErr = net.ResolveTCPAddr("tcp", net.JoinHostPort(bindHost, tryPort))
|
||||||
|
if fatalErr != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
// the hostname probably couldn't be resolved, just bind to wildcard then
|
||||||
|
resolvAddr, fatalErr = net.ResolveTCPAddr("tcp", net.JoinHostPort("0.0.0.0", tryPort))
|
||||||
|
if fatalErr != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// validDirective returns true if d is a valid
|
// validDirective returns true if d is a valid
|
||||||
// directive; false otherwise.
|
// directive; false otherwise.
|
||||||
func validDirective(d string) bool {
|
func validDirective(d string) bool {
|
||||||
|
|
61
config/config_test.go
Normal file
61
config/config_test.go
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/mholt/caddy/server"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestReolveAddr(t *testing.T) {
|
||||||
|
// NOTE: If tests fail due to comparing to string "127.0.0.1",
|
||||||
|
// it's possible that system env resolves with IPv6, or ::1.
|
||||||
|
// If that happens, maybe we should use actualAddr.IP.IsLoopback()
|
||||||
|
// for the assertion, rather than a direct string comparison.
|
||||||
|
|
||||||
|
for i, test := range []struct {
|
||||||
|
config server.Config
|
||||||
|
shouldWarnErr bool
|
||||||
|
shouldFatalErr bool
|
||||||
|
expectedIP string
|
||||||
|
expectedPort int
|
||||||
|
}{
|
||||||
|
{server.Config{Host: "localhost", Port: "1234"}, false, false, "127.0.0.1", 1234},
|
||||||
|
{server.Config{Host: "127.0.0.1", Port: "1234"}, false, false, "127.0.0.1", 1234},
|
||||||
|
{server.Config{Host: "should-not-resolve", Port: "1234"}, true, false, "0.0.0.0", 1234},
|
||||||
|
{server.Config{Host: "localhost", Port: "http"}, false, false, "127.0.0.1", 80},
|
||||||
|
{server.Config{Host: "localhost", Port: "https"}, false, false, "127.0.0.1", 443},
|
||||||
|
{server.Config{Host: "", Port: ""}, false, true, "", 0},
|
||||||
|
{server.Config{Host: "localhost", Port: ""}, false, true, "127.0.0.1", 0},
|
||||||
|
{server.Config{Host: "", Port: "1234"}, false, false, "<nil>", 1234},
|
||||||
|
{server.Config{Host: "localhost", Port: "abcd"}, false, true, "", 0},
|
||||||
|
{server.Config{BindHost: "127.0.0.1", Host: "should-not-be-used", Port: "1234"}, false, false, "127.0.0.1", 1234},
|
||||||
|
{server.Config{BindHost: "localhost", Host: "should-not-be-used", Port: "1234"}, false, false, "127.0.0.1", 1234},
|
||||||
|
{server.Config{BindHost: "should-not-resolve", Host: "localhost", Port: "1234"}, true, false, "0.0.0.0", 1234},
|
||||||
|
} {
|
||||||
|
actualAddr, warnErr, fatalErr := resolveAddr(test.config)
|
||||||
|
|
||||||
|
if test.shouldFatalErr && fatalErr == nil {
|
||||||
|
t.Errorf("Test %d: Expected error, but there wasn't any", i)
|
||||||
|
}
|
||||||
|
if !test.shouldFatalErr && fatalErr != nil {
|
||||||
|
t.Errorf("Test %d: Expected no error, but there was one: %v", i, fatalErr)
|
||||||
|
}
|
||||||
|
if fatalErr != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if test.shouldWarnErr && warnErr == nil {
|
||||||
|
t.Errorf("Test %d: Expected warning, but there wasn't any", i)
|
||||||
|
}
|
||||||
|
if !test.shouldWarnErr && warnErr != nil {
|
||||||
|
t.Errorf("Test %d: Expected no warning, but there was one: %v", i, warnErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
if actual, expected := actualAddr.IP.String(), test.expectedIP; actual != expected {
|
||||||
|
t.Errorf("Test %d: IP was %s but expected %s", i, actual, expected)
|
||||||
|
}
|
||||||
|
if actual, expected := actualAddr.Port, test.expectedPort; actual != expected {
|
||||||
|
t.Errorf("Test %d: Port was %d but expected %d", i, actual, expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -47,9 +47,6 @@ type Config struct {
|
||||||
|
|
||||||
// Address returns the host:port of c as a string.
|
// Address returns the host:port of c as a string.
|
||||||
func (c Config) Address() string {
|
func (c Config) Address() string {
|
||||||
if c.BindHost != "" {
|
|
||||||
return net.JoinHostPort(c.BindHost, c.Port)
|
|
||||||
}
|
|
||||||
return net.JoinHostPort(c.Host, c.Port)
|
return net.JoinHostPort(c.Host, c.Port)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue