mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-19 01:05:37 +03:00
129 lines
3.4 KiB
Go
129 lines
3.4 KiB
Go
// Package server implements a configurable, general-purpose web server.
|
|
// It relies on configurations obtained from the adjacent config package
|
|
// and can execute middleware as defined by the adjacent middleware package.
|
|
package server
|
|
|
|
import (
|
|
"errors"
|
|
"log"
|
|
"net/http"
|
|
"runtime"
|
|
|
|
"github.com/bradfitz/http2"
|
|
"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
|
|
// static content at a particular address (host and port).
|
|
type Server struct {
|
|
config config.Config
|
|
reqlog *log.Logger
|
|
errlog *log.Logger
|
|
fileServer http.Handler
|
|
stack http.HandlerFunc
|
|
}
|
|
|
|
// New creates a new Server and registers it with the list
|
|
// of servers created. Each server must have a unique host:port
|
|
// combination. This function does not start serving.
|
|
func New(conf config.Config) (*Server, error) {
|
|
addr := conf.Address()
|
|
|
|
// Unique address check
|
|
if _, exists := servers[addr]; exists {
|
|
return nil, errors.New("Address " + addr + " is already in use")
|
|
}
|
|
|
|
// Use all CPUs (if needed) by default
|
|
if conf.MaxCPU == 0 {
|
|
conf.MaxCPU = runtime.NumCPU()
|
|
}
|
|
|
|
// Initialize
|
|
s := new(Server)
|
|
s.config = conf
|
|
|
|
// Register the server
|
|
servers[addr] = s
|
|
|
|
return s, nil
|
|
}
|
|
|
|
// Serve starts the server. It blocks until the server quits.
|
|
func (s *Server) Serve() error {
|
|
err := s.buildStack()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// use highest value across all configurations
|
|
if s.config.MaxCPU > 0 && s.config.MaxCPU > runtime.GOMAXPROCS(0) {
|
|
runtime.GOMAXPROCS(s.config.MaxCPU)
|
|
}
|
|
|
|
server := &http.Server{
|
|
Addr: s.config.Address(),
|
|
Handler: s,
|
|
}
|
|
|
|
http2.ConfigureServer(server, nil) // TODO: This may not be necessary after HTTP/2 merged into std lib
|
|
|
|
if s.config.TLS.Enabled {
|
|
return server.ListenAndServeTLS(s.config.TLS.Certificate, s.config.TLS.Key)
|
|
} else {
|
|
return server.ListenAndServe()
|
|
}
|
|
}
|
|
|
|
// ServeHTTP is the entry point for each request to s.
|
|
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
s.stack(w, r)
|
|
}
|
|
|
|
// Log writes a message to the server's configured error log,
|
|
// if there is one, or if there isn't, to the default stderr log.
|
|
func (s *Server) Log(v ...interface{}) {
|
|
if s.errlog != nil {
|
|
s.errlog.Println(v)
|
|
} else {
|
|
log.Println(v)
|
|
}
|
|
}
|
|
|
|
// buildStack builds the server's middleware stack based
|
|
// on its config. This method should be called last before
|
|
// ListenAndServe begins.
|
|
func (s *Server) buildStack() error {
|
|
s.fileServer = http.FileServer(http.Dir(s.config.Root))
|
|
|
|
for _, start := range s.config.Startup {
|
|
err := start()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// TODO: We only compile middleware for the "/" scope.
|
|
// Partial support for multiple location contexts already
|
|
// exists in 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 generator
|
|
// function calls like handler1(handler2(handler3(finalHandler))).
|
|
func (s *Server) compile(layers []middleware.Middleware) {
|
|
s.stack = s.fileServer.ServeHTTP // core app layer
|
|
for _, layer := range layers {
|
|
s.stack = layer(s.stack)
|
|
}
|
|
}
|