Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
caddy
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
caddy
Commits
feec7c5b
Commit
feec7c5b
authored
Apr 15, 2015
by
Matthew Holt
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Virtual hosts and SNI support
parent
07964a61
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
222 additions
and
96 deletions
+222
-96
config/config.go
config/config.go
+6
-2
main.go
main.go
+53
-5
server/server.go
server/server.go
+121
-89
server/virtualhost.go
server/virtualhost.go
+42
-0
No files found.
config/config.go
View file @
feec7c5b
...
@@ -3,6 +3,7 @@
...
@@ -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 (
...
@@ -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 {
...
@@ -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,
...
...
main.go
View file @
feec7c5b
...
@@ -2,7 +2,9 @@ package main
...
@@ -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 (
...
@@ -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() {
...
@@ -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
)
{
vhost
s
=
config
.
Default
()
allConfig
s
=
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() {
...
@@ -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
}
server/server.go
View file @
feec7c5b
...
@@ -4,9 +4,10 @@
...
@@ -4,9 +4,10 @@
package
server
package
server
import
(
import
(
"
error
s"
"
crypto/tl
s"
"fmt"
"fmt"
"log"
"log"
"net"
"net/http"
"net/http"
"os"
"os"
"os/signal"
"os/signal"
...
@@ -14,33 +15,30 @@ import (
...
@@ -14,33 +15,30 @@ 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
.
Confi
g
address
strin
g
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
,
tls
:
tls
,
vhosts
:
make
(
map
[
string
]
virtualHost
),
}
// Unique address check
for
_
,
conf
:=
range
configs
{
if
_
,
exists
:=
servers
[
addr
];
exists
{
if
_
,
exists
:=
s
.
vhosts
[
conf
.
Host
];
exists
{
return
nil
,
errors
.
New
(
"Address "
+
addr
+
" is already in use"
)
return
nil
,
fmt
.
Errorf
(
"Cannot serve %s - host already defined for address %s"
,
conf
.
Address
(),
s
.
address
)
}
}
// Use all CPUs (if needed) by default
// Use all CPUs (if needed) by default
...
@@ -48,53 +46,53 @@ func New(conf config.Config) (*Server, error) {
...
@@ -48,53 +46,53 @@ func New(conf config.Config) (*Server, error) {
conf
.
MaxCPU
=
runtime
.
NumCPU
()
conf
.
MaxCPU
=
runtime
.
NumCPU
()
}
}
// Initialize
vh
:=
virtualHost
{
config
:
conf
}
s
:=
new
(
Server
)
s
.
config
=
conf
// Build middleware stack
err
:=
vh
.
buildStack
()
if
err
!=
nil
{
return
nil
,
err
}
// Register the server
s
.
vhosts
[
conf
.
Host
]
=
vh
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
server
:=
&
http
.
Server
{
for
_
,
start
:=
range
s
.
config
.
Startup
{
Addr
:
s
.
address
,
err
:=
start
()
Handler
:
s
,
if
err
!=
nil
{
return
err
}
}
if
s
.
HTTP2
{
// TODO: This call may not be necessary after HTTP/2 is merged into std lib
http2
.
ConfigureServer
(
server
,
nil
)
}
}
// Build middleware stack
for
_
,
vh
:=
range
s
.
vhosts
{
err
:=
s
.
buildStack
()
// Execute startup functions
for
_
,
start
:=
range
vh
.
config
.
Startup
{
err
:=
start
()
if
err
!=
nil
{
if
err
!=
nil
{
return
err
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
{
// Use highest procs value across all configurations
Addr
:
s
.
config
.
Address
(),
if
vh
.
config
.
MaxCPU
>
0
&&
vh
.
config
.
MaxCPU
>
runtime
.
GOMAXPROCS
(
0
)
{
Handler
:
s
,
runtime
.
GOMAXPROCS
(
vh
.
config
.
MaxCPU
)
}
if
s
.
HTTP2
{
// TODO: This call may not be necessary after HTTP/2 is merged into std lib
http2
.
ConfigureServer
(
server
,
nil
)
}
}
if
len
(
vh
.
config
.
Shutdown
)
>
0
{
// Execute shutdown commands on exit
// Execute shutdown commands on exit
go
func
()
{
go
func
()
{
interrupt
:=
make
(
chan
os
.
Signal
,
1
)
interrupt
:=
make
(
chan
os
.
Signal
,
1
)
signal
.
Notify
(
interrupt
,
os
.
Interrupt
,
os
.
Kill
)
// TODO: syscall.SIGQUIT? (Ctrl+\, Unix-only)
signal
.
Notify
(
interrupt
,
os
.
Interrupt
,
os
.
Kill
)
// TODO: syscall.SIGQUIT? (Ctrl+\, Unix-only)
<-
interrupt
<-
interrupt
for
_
,
shutdownFunc
:=
range
s
.
config
.
Shutdown
{
for
_
,
shutdownFunc
:=
range
vh
.
config
.
Shutdown
{
err
:=
shutdownFunc
()
err
:=
shutdownFunc
()
if
err
!=
nil
{
if
err
!=
nil
{
log
.
Fatal
(
err
)
log
.
Fatal
(
err
)
...
@@ -102,15 +100,63 @@ func (s *Server) Serve() error {
...
@@ -102,15 +100,63 @@ func (s *Server) Serve() error {
}
}
os
.
Exit
(
0
)
os
.
Exit
(
0
)
}()
}()
}
}
if
s
.
config
.
TLS
.
Enabled
{
if
s
.
tls
{
return
server
.
ListenAndServeTLS
(
s
.
config
.
TLS
.
Certificate
,
s
.
config
.
TLS
.
Key
)
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) {
...
@@ -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
}
if
vh
,
ok
:=
s
.
vhosts
[
host
];
ok
{
status
,
_
:=
vh
.
stack
.
ServeHTTP
(
w
,
r
)
// Fallback error response in case error handling wasn't chained in
// Fallback error response in case error handling wasn't chained in
if
status
>=
400
{
if
status
>=
400
{
w
.
WriteHeader
(
status
)
w
.
WriteHeader
(
status
)
fmt
.
Fprintf
(
w
,
"%d %s"
,
status
,
http
.
StatusText
(
status
))
fmt
.
Fprintf
(
w
,
"%d %s"
,
status
,
http
.
StatusText
(
status
))
}
}
}
}
else
{
w
.
WriteHeader
(
http
.
StatusNotFound
)
// buildStack builds the server's middleware stack based
fmt
.
Fprintf
(
w
,
"No such host at %s"
,
s
.
address
)
// on its config. This method should be called last before
// ListenAndServe begins.
func
(
s
*
Server
)
buildStack
()
error
{
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
)
}
}
}
}
server/virtualhost.go
0 → 100644
View file @
feec7c5b
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
)
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment