Virtual hosts and SNI support

This commit is contained in:
Matthew Holt 2015-04-15 14:11:32 -06:00
parent 07964a6112
commit feec7c5b40
4 changed files with 226 additions and 100 deletions

View file

@ -3,6 +3,7 @@
package config package config
import ( import (
"net"
"os" "os"
"github.com/mholt/caddy/middleware" "github.com/mholt/caddy/middleware"
@ -12,13 +13,16 @@ const (
defaultHost = "localhost" defaultHost = "localhost"
defaultPort = "8080" defaultPort = "8080"
defaultRoot = "." defaultRoot = "."
// The default configuration file to load if none is specified
DefaultConfigFile = "Caddyfile"
) )
// config represents a server configuration. It // config represents a server configuration. It
// is populated by parsing a config file (via the // is populated by parsing a config file (via the
// Load function). // Load function).
type Config struct { type Config struct {
// The hostname or IP to which to bind the server // The hostname or IP on which to serve
Host string Host string
// The port to listen on // The port to listen on
@ -51,7 +55,7 @@ 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 {
return c.Host + ":" + c.Port return net.JoinHostPort(c.Host, c.Port)
} }
// TLSConfig describes how TLS should be configured and used, // TLSConfig describes how TLS should be configured and used,

58
main.go
View file

@ -2,7 +2,9 @@ package main
import ( import (
"flag" "flag"
"fmt"
"log" "log"
"net"
"sync" "sync"
"github.com/mholt/caddy/config" "github.com/mholt/caddy/config"
@ -15,7 +17,7 @@ var (
) )
func init() { func init() {
flag.StringVar(&conf, "conf", server.DefaultConfigFile, "the configuration file to use") flag.StringVar(&conf, "conf", config.DefaultConfigFile, "the configuration file to use")
flag.BoolVar(&http2, "http2", true, "enable HTTP/2 support") // temporary flag until http2 merged into std lib flag.BoolVar(&http2, "http2", true, "enable HTTP/2 support") // temporary flag until http2 merged into std lib
} }
@ -24,17 +26,25 @@ func main() {
flag.Parse() flag.Parse()
vhosts, err := config.Load(conf) // Load config from file
allConfigs, err := config.Load(conf)
if err != nil { if err != nil {
if config.IsNotFound(err) { if config.IsNotFound(err) {
vhosts = config.Default() allConfigs = config.Default()
} else { } else {
log.Fatal(err) log.Fatal(err)
} }
} }
for _, conf := range vhosts { // Group by address (virtual hosting)
s, err := server.New(conf) addresses, err := arrangeBindings(allConfigs)
if err != nil {
log.Fatal(err)
}
// Start each server with its one or more configurations
for addr, configs := range addresses {
s, err := server.New(addr, configs, configs[0].TLS.Enabled)
if err != nil { if err != nil {
log.Fatal(err) log.Fatal(err)
} }
@ -51,3 +61,41 @@ func main() {
wg.Wait() wg.Wait()
} }
// arrangeBindings groups configurations by their bind address. For example,
// 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
// if the address lookup fails or if a TLS listener is configured on the
// same address as a plaintext HTTP listener.
func arrangeBindings(allConfigs []config.Config) (map[string][]config.Config, error) {
addresses := make(map[string][]config.Config)
// Group configs by bind address
for _, conf := range allConfigs {
addr, err := net.ResolveTCPAddr("tcp", conf.Address())
if err != nil {
return addresses, err
}
addresses[addr.String()] = append(addresses[addr.String()], conf)
}
// Don't allow HTTP and HTTPS to be served on the same address
for _, configs := range addresses {
isTLS := configs[0].TLS.Enabled
for _, config := range configs {
if config.TLS.Enabled != isTLS {
thisConfigProto, otherConfigProto := "HTTP", "HTTP"
if config.TLS.Enabled {
thisConfigProto = "HTTPS"
}
if configs[0].TLS.Enabled {
otherConfigProto = "HTTPS"
}
return addresses, fmt.Errorf("Configuration error: Cannot multiplex %s (%s) and %s (%s) on same address",
configs[0].Address(), otherConfigProto, config.Address(), thisConfigProto)
}
}
}
return addresses, nil
}

View file

@ -4,9 +4,10 @@
package server package server
import ( import (
"errors" "crypto/tls"
"fmt" "fmt"
"log" "log"
"net"
"net/http" "net/http"
"os" "os"
"os/signal" "os/signal"
@ -14,73 +15,55 @@ import (
"github.com/bradfitz/http2" "github.com/bradfitz/http2"
"github.com/mholt/caddy/config" "github.com/mholt/caddy/config"
"github.com/mholt/caddy/middleware"
) )
// The default configuration file to load if none is specified
const DefaultConfigFile = "Caddyfile"
// servers maintains a registry of running servers, keyed by address.
var servers = make(map[string]*Server)
// Server represents an instance of a server, which serves // Server represents an instance of a server, which serves
// static content at a particular address (host and port). // static content at a particular address (host and port).
type Server struct { type Server struct {
HTTP2 bool // temporary while http2 is not in std lib (TODO: remove flag when part of std lib) HTTP2 bool // temporary while http2 is not in std lib (TODO: remove flag when part of std lib)
config config.Config address string
fileServer middleware.Handler tls bool
stack middleware.Handler vhosts map[string]virtualHost
} }
// New creates a new Server and registers it with the list // New creates a new Server which will bind to addr and serve
// of servers created. Each server must have a unique host:port // the sites/hosts configured in configs. This function does
// combination. This function does not start serving. // not start serving.
func New(conf config.Config) (*Server, error) { func New(addr string, configs []config.Config, tls bool) (*Server, error) {
addr := conf.Address() s := &Server{
address: addr,
// Unique address check tls: tls,
if _, exists := servers[addr]; exists { vhosts: make(map[string]virtualHost),
return nil, errors.New("Address " + addr + " is already in use")
} }
// Use all CPUs (if needed) by default for _, conf := range configs {
if conf.MaxCPU == 0 { if _, exists := s.vhosts[conf.Host]; exists {
conf.MaxCPU = runtime.NumCPU() return nil, fmt.Errorf("Cannot serve %s - host already defined for address %s", conf.Address(), s.address)
}
// Use all CPUs (if needed) by default
if conf.MaxCPU == 0 {
conf.MaxCPU = runtime.NumCPU()
}
vh := virtualHost{config: conf}
// Build middleware stack
err := vh.buildStack()
if err != nil {
return nil, err
}
s.vhosts[conf.Host] = vh
} }
// Initialize
s := new(Server)
s.config = conf
// Register the server
servers[addr] = s
return s, nil return s, nil
} }
// Serve starts the server. It blocks until the server quits. // Serve starts the server. It blocks until the server quits.
func (s *Server) Serve() error { func (s *Server) Serve() error {
// Execute startup functions
for _, start := range s.config.Startup {
err := start()
if err != nil {
return err
}
}
// Build middleware stack
err := s.buildStack()
if err != nil {
return err
}
// Use highest procs value across all configurations
if s.config.MaxCPU > 0 && s.config.MaxCPU > runtime.GOMAXPROCS(0) {
runtime.GOMAXPROCS(s.config.MaxCPU)
}
server := &http.Server{ server := &http.Server{
Addr: s.config.Address(), Addr: s.address,
Handler: s, Handler: s,
} }
@ -89,28 +72,91 @@ func (s *Server) Serve() error {
http2.ConfigureServer(server, nil) http2.ConfigureServer(server, nil)
} }
// Execute shutdown commands on exit for _, vh := range s.vhosts {
go func() { // Execute startup functions
interrupt := make(chan os.Signal, 1) for _, start := range vh.config.Startup {
signal.Notify(interrupt, os.Interrupt, os.Kill) // TODO: syscall.SIGQUIT? (Ctrl+\, Unix-only) err := start()
<-interrupt
for _, shutdownFunc := range s.config.Shutdown {
err := shutdownFunc()
if err != nil { if err != nil {
log.Fatal(err) return err
} }
} }
os.Exit(0)
}()
if s.config.TLS.Enabled { // Use highest procs value across all configurations
return server.ListenAndServeTLS(s.config.TLS.Certificate, s.config.TLS.Key) if vh.config.MaxCPU > 0 && vh.config.MaxCPU > runtime.GOMAXPROCS(0) {
runtime.GOMAXPROCS(vh.config.MaxCPU)
}
if len(vh.config.Shutdown) > 0 {
// Execute shutdown commands on exit
go func() {
interrupt := make(chan os.Signal, 1)
signal.Notify(interrupt, os.Interrupt, os.Kill) // TODO: syscall.SIGQUIT? (Ctrl+\, Unix-only)
<-interrupt
for _, shutdownFunc := range vh.config.Shutdown {
err := shutdownFunc()
if err != nil {
log.Fatal(err)
}
}
os.Exit(0)
}()
}
}
if s.tls {
var tlsConfigs []config.TLSConfig
for _, vh := range s.vhosts {
tlsConfigs = append(tlsConfigs, vh.config.TLS)
}
return ListenAndServeTLSWithSNI(server, tlsConfigs)
} else { } else {
return server.ListenAndServe() return server.ListenAndServe()
} }
} }
// ServeHTTP is the entry point for every request to s. // ListenAndServeTLSWithSNI serves TLS with Server Name Indication (SNI) support, which allows
// multiple sites (different hostnames) to be served from the same address. This method is
// adapted directly from the std lib's net/http ListenAndServeTLS function, which was
// written by the Go Authors. It has been modified to support multiple certificate/key pairs.
func ListenAndServeTLSWithSNI(srv *http.Server, tlsConfigs []config.TLSConfig) error {
addr := srv.Addr
if addr == "" {
addr = ":https"
}
config := new(tls.Config)
if srv.TLSConfig != nil {
*config = *srv.TLSConfig
}
if config.NextProtos == nil {
config.NextProtos = []string{"http/1.1"}
}
// Here we diverge from the stdlib a bit by loading multiple certs/key pairs
// then we map the server names to their certs
var err error
config.Certificates = make([]tls.Certificate, len(tlsConfigs))
for i, tlsConfig := range tlsConfigs {
config.Certificates[i], err = tls.LoadX509KeyPair(tlsConfig.Certificate, tlsConfig.Key)
if err != nil {
return err
}
}
config.BuildNameToCertificate()
conn, err := net.Listen("tcp", addr)
if err != nil {
return err
}
tlsListener := tls.NewListener(conn, config)
return srv.Serve(tlsListener)
}
// ServeHTTP is the entry point for every request to the address that s
// is bound to. It acts as a multiplexer for the requests hostname as
// defined in the Host header so that the correct virtualhost
// (configuration and middleware stack) will handle the request.
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) { func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
defer func() { defer func() {
// In case the user doesn't enable error middleware, we still // In case the user doesn't enable error middleware, we still
@ -121,35 +167,21 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
} }
}() }()
status, _ := s.stack.ServeHTTP(w, r) host, _, err := net.SplitHostPort(r.Host)
if err != nil {
host = r.Host // oh well
}
// Fallback error response in case error handling wasn't chained in if vh, ok := s.vhosts[host]; ok {
if status >= 400 { status, _ := vh.stack.ServeHTTP(w, r)
w.WriteHeader(status)
fmt.Fprintf(w, "%d %s", status, http.StatusText(status)) // Fallback error response in case error handling wasn't chained in
} if status >= 400 {
} w.WriteHeader(status)
fmt.Fprintf(w, "%d %s", status, http.StatusText(status))
// buildStack builds the server's middleware stack based }
// on its config. This method should be called last before } else {
// ListenAndServe begins. w.WriteHeader(http.StatusNotFound)
func (s *Server) buildStack() error { fmt.Fprintf(w, "No such host at %s", s.address)
s.fileServer = FileServer(http.Dir(s.config.Root), []string{s.config.ConfigFile})
// TODO: We only compile middleware for the "/" scope.
// Partial support for multiple location contexts already
// exists at the parser and config levels, but until full
// support is implemented, this is all we do right here.
s.compile(s.config.Middleware["/"])
return nil
}
// compile is an elegant alternative to nesting middleware function
// calls like handler1(handler2(handler3(finalHandler))).
func (s *Server) compile(layers []middleware.Middleware) {
s.stack = s.fileServer // core app layer
for i := len(layers) - 1; i >= 0; i-- {
s.stack = layers[i](s.stack)
} }
} }

42
server/virtualhost.go Normal file
View file

@ -0,0 +1,42 @@
package server
import (
"net/http"
"github.com/mholt/caddy/config"
"github.com/mholt/caddy/middleware"
)
// virtualHost represents a virtual host/server. While a Server
// is what actually binds to the address, a user may want to serve
// multiple sites on a single address, and what is what a
// virtualHost allows us to do.
type virtualHost struct {
config config.Config
fileServer middleware.Handler
stack middleware.Handler
}
// buildStack builds the server's middleware stack based
// on its config. This method should be called last before
// ListenAndServe begins.
func (vh *virtualHost) buildStack() error {
vh.fileServer = FileServer(http.Dir(vh.config.Root), []string{vh.config.ConfigFile})
// TODO: We only compile middleware for the "/" scope.
// Partial support for multiple location contexts already
// exists at the parser and config levels, but until full
// support is implemented, this is all we do right here.
vh.compile(vh.config.Middleware["/"])
return nil
}
// compile is an elegant alternative to nesting middleware function
// calls like handler1(handler2(handler3(finalHandler))).
func (vh *virtualHost) compile(layers []middleware.Middleware) {
vh.stack = vh.fileServer // core app layer
for i := len(layers) - 1; i >= 0; i-- {
vh.stack = layers[i](vh.stack)
}
}