mirror of
https://github.com/caddyserver/caddy.git
synced 2024-11-26 10:13:39 +08:00
15fa5cf2da
OnStartup and OnShutdown callbacks now run as part of restarts, too. The startup and shutdown directives only run their commands NOT as part of restarts, as before. Some middleware that use OnStartup may need to switch to OnFirstStartup and implement OnFinalShutdown to do any cleanup as needed.
160 lines
3.3 KiB
Go
160 lines
3.3 KiB
Go
package errors
|
|
|
|
import (
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
|
|
"github.com/hashicorp/go-syslog"
|
|
"github.com/mholt/caddy"
|
|
"github.com/mholt/caddy/caddyhttp/httpserver"
|
|
)
|
|
|
|
// setup configures a new errors middleware instance.
|
|
func setup(c *caddy.Controller) error {
|
|
handler, err := errorsParse(c)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Open the log file for writing when the server starts
|
|
c.OnStartup(func() error {
|
|
var err error
|
|
var writer io.Writer
|
|
|
|
switch handler.LogFile {
|
|
case "visible":
|
|
handler.Debug = true
|
|
case "stdout":
|
|
writer = os.Stdout
|
|
case "stderr":
|
|
writer = os.Stderr
|
|
case "syslog":
|
|
writer, err = gsyslog.NewLogger(gsyslog.LOG_ERR, "LOCAL0", "caddy")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
if handler.LogFile == "" {
|
|
writer = os.Stderr // default
|
|
break
|
|
}
|
|
|
|
var file *os.File
|
|
file, err = os.OpenFile(handler.LogFile, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if handler.LogRoller != nil {
|
|
file.Close()
|
|
handler.LogRoller.Filename = handler.LogFile
|
|
writer = handler.LogRoller.GetLogWriter()
|
|
} else {
|
|
handler.file = file
|
|
writer = file
|
|
}
|
|
}
|
|
|
|
handler.Log = log.New(writer, "", 0)
|
|
return nil
|
|
})
|
|
|
|
// When server stops, close any open log file
|
|
c.OnShutdown(func() error {
|
|
if handler.file != nil {
|
|
handler.file.Close()
|
|
}
|
|
return nil
|
|
})
|
|
|
|
httpserver.GetConfig(c).AddMiddleware(func(next httpserver.Handler) httpserver.Handler {
|
|
handler.Next = next
|
|
return handler
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
func errorsParse(c *caddy.Controller) (*ErrorHandler, error) {
|
|
// Very important that we make a pointer because the startup
|
|
// function that opens the log file must have access to the
|
|
// same instance of the handler, not a copy.
|
|
handler := &ErrorHandler{ErrorPages: make(map[int]string)}
|
|
|
|
cfg := httpserver.GetConfig(c)
|
|
|
|
optionalBlock := func() (bool, error) {
|
|
var hadBlock bool
|
|
|
|
for c.NextBlock() {
|
|
hadBlock = true
|
|
|
|
what := c.Val()
|
|
if !c.NextArg() {
|
|
return hadBlock, c.ArgErr()
|
|
}
|
|
where := c.Val()
|
|
|
|
if what == "log" {
|
|
if where == "visible" {
|
|
handler.Debug = true
|
|
} else {
|
|
handler.LogFile = where
|
|
if c.NextArg() {
|
|
if c.Val() == "{" {
|
|
c.IncrNest()
|
|
logRoller, err := httpserver.ParseRoller(c)
|
|
if err != nil {
|
|
return hadBlock, err
|
|
}
|
|
handler.LogRoller = logRoller
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
// Error page; ensure it exists
|
|
where = filepath.Join(cfg.Root, where)
|
|
f, err := os.Open(where)
|
|
if err != nil {
|
|
log.Printf("[WARNING] Unable to open error page '%s': %v", where, err)
|
|
}
|
|
f.Close()
|
|
|
|
whatInt, err := strconv.Atoi(what)
|
|
if err != nil {
|
|
return hadBlock, c.Err("Expecting a numeric status code, got '" + what + "'")
|
|
}
|
|
handler.ErrorPages[whatInt] = where
|
|
}
|
|
}
|
|
return hadBlock, nil
|
|
}
|
|
|
|
for c.Next() {
|
|
// weird hack to avoid having the handler values overwritten.
|
|
if c.Val() == "}" {
|
|
continue
|
|
}
|
|
// Configuration may be in a block
|
|
hadBlock, err := optionalBlock()
|
|
if err != nil {
|
|
return handler, err
|
|
}
|
|
|
|
// Otherwise, the only argument would be an error log file name or 'visible'
|
|
if !hadBlock {
|
|
if c.NextArg() {
|
|
if c.Val() == "visible" {
|
|
handler.Debug = true
|
|
} else {
|
|
handler.LogFile = c.Val()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return handler, nil
|
|
}
|