mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-19 09:05:41 +03:00
Add support for default (wildcard) error page
This commit is contained in:
parent
68be4a9161
commit
f7003bee3f
4 changed files with 145 additions and 21 deletions
|
@ -24,13 +24,14 @@ func init() {
|
||||||
|
|
||||||
// ErrorHandler handles HTTP errors (and errors from other middleware).
|
// ErrorHandler handles HTTP errors (and errors from other middleware).
|
||||||
type ErrorHandler struct {
|
type ErrorHandler struct {
|
||||||
Next httpserver.Handler
|
Next httpserver.Handler
|
||||||
ErrorPages map[int]string // map of status code to filename
|
GenericErrorPage string // default error page filename
|
||||||
LogFile string
|
ErrorPages map[int]string // map of status code to filename
|
||||||
Log *log.Logger
|
LogFile string
|
||||||
LogRoller *httpserver.LogRoller
|
Log *log.Logger
|
||||||
Debug bool // if true, errors are written out to client rather than to a log
|
LogRoller *httpserver.LogRoller
|
||||||
file *os.File // a log file to close when done
|
Debug bool // if true, errors are written out to client rather than to a log
|
||||||
|
file *os.File // a log file to close when done
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h ErrorHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
func (h ErrorHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||||
|
@ -63,7 +64,7 @@ func (h ErrorHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, er
|
||||||
// message is written instead, and the extra error is logged.
|
// message is written instead, and the extra error is logged.
|
||||||
func (h ErrorHandler) errorPage(w http.ResponseWriter, r *http.Request, code int) {
|
func (h ErrorHandler) errorPage(w http.ResponseWriter, r *http.Request, code int) {
|
||||||
// See if an error page for this status code was specified
|
// See if an error page for this status code was specified
|
||||||
if pagePath, ok := h.ErrorPages[code]; ok {
|
if pagePath, ok := h.findErrorPage(code); ok {
|
||||||
// Try to open it
|
// Try to open it
|
||||||
errorPage, err := os.Open(pagePath)
|
errorPage, err := os.Open(pagePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -94,6 +95,18 @@ func (h ErrorHandler) errorPage(w http.ResponseWriter, r *http.Request, code int
|
||||||
httpserver.DefaultErrorFunc(w, r, code)
|
httpserver.DefaultErrorFunc(w, r, code)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h ErrorHandler) findErrorPage(code int) (string, bool) {
|
||||||
|
if pagePath, ok := h.ErrorPages[code]; ok {
|
||||||
|
return pagePath, true
|
||||||
|
}
|
||||||
|
|
||||||
|
if h.GenericErrorPage != "" {
|
||||||
|
return h.GenericErrorPage, true
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
|
||||||
func (h ErrorHandler) recovery(w http.ResponseWriter, r *http.Request) {
|
func (h ErrorHandler) recovery(w http.ResponseWriter, r *http.Request) {
|
||||||
rec := recover()
|
rec := recover()
|
||||||
if rec == nil {
|
if rec == nil {
|
||||||
|
|
|
@ -18,20 +18,14 @@ import (
|
||||||
|
|
||||||
func TestErrors(t *testing.T) {
|
func TestErrors(t *testing.T) {
|
||||||
// create a temporary page
|
// create a temporary page
|
||||||
path := filepath.Join(os.TempDir(), "errors_test.html")
|
const content = "This is a error page"
|
||||||
f, err := os.Create(path)
|
|
||||||
|
path, err := createErrorPageFile("errors_test.html", content)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
defer os.Remove(path)
|
defer os.Remove(path)
|
||||||
|
|
||||||
const content = "This is a error page"
|
|
||||||
_, err = f.WriteString(content)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
f.Close()
|
|
||||||
|
|
||||||
buf := bytes.Buffer{}
|
buf := bytes.Buffer{}
|
||||||
em := ErrorHandler{
|
em := ErrorHandler{
|
||||||
ErrorPages: map[int]string{
|
ErrorPages: map[int]string{
|
||||||
|
@ -157,6 +151,87 @@ func TestVisibleErrorWithPanic(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestGenericErrorPage(t *testing.T) {
|
||||||
|
// create temporary generic error page
|
||||||
|
const genericErrorContent = "This is a generic error page"
|
||||||
|
|
||||||
|
genericErrorPagePath, err := createErrorPageFile("generic_error_test.html", genericErrorContent)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(genericErrorPagePath)
|
||||||
|
|
||||||
|
// create temporary error page
|
||||||
|
const notFoundErrorContent = "This is a error page"
|
||||||
|
|
||||||
|
notFoundErrorPagePath, err := createErrorPageFile("not_found.html", notFoundErrorContent)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
defer os.Remove(notFoundErrorPagePath)
|
||||||
|
|
||||||
|
buf := bytes.Buffer{}
|
||||||
|
em := ErrorHandler{
|
||||||
|
GenericErrorPage: genericErrorPagePath,
|
||||||
|
ErrorPages: map[int]string{
|
||||||
|
http.StatusNotFound: notFoundErrorPagePath,
|
||||||
|
},
|
||||||
|
Log: log.New(&buf, "", 0),
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
next httpserver.Handler
|
||||||
|
expectedCode int
|
||||||
|
expectedBody string
|
||||||
|
expectedLog string
|
||||||
|
expectedErr error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
next: genErrorHandler(http.StatusNotFound, nil, ""),
|
||||||
|
expectedCode: 0,
|
||||||
|
expectedBody: notFoundErrorContent,
|
||||||
|
expectedLog: "",
|
||||||
|
expectedErr: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
next: genErrorHandler(http.StatusInternalServerError, nil, ""),
|
||||||
|
expectedCode: 0,
|
||||||
|
expectedBody: genericErrorContent,
|
||||||
|
expectedLog: "",
|
||||||
|
expectedErr: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("GET", "/", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, test := range tests {
|
||||||
|
em.Next = test.next
|
||||||
|
buf.Reset()
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
code, err := em.ServeHTTP(rec, req)
|
||||||
|
|
||||||
|
if err != test.expectedErr {
|
||||||
|
t.Errorf("Test %d: Expected error %v, but got %v",
|
||||||
|
i, test.expectedErr, err)
|
||||||
|
}
|
||||||
|
if code != test.expectedCode {
|
||||||
|
t.Errorf("Test %d: Expected status code %d, but got %d",
|
||||||
|
i, test.expectedCode, code)
|
||||||
|
}
|
||||||
|
if body := rec.Body.String(); body != test.expectedBody {
|
||||||
|
t.Errorf("Test %d: Expected body %q, but got %q",
|
||||||
|
i, test.expectedBody, body)
|
||||||
|
}
|
||||||
|
if log := buf.String(); !strings.Contains(log, test.expectedLog) {
|
||||||
|
t.Errorf("Test %d: Expected log %q, but got %q",
|
||||||
|
i, test.expectedLog, log)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func genErrorHandler(status int, err error, body string) httpserver.Handler {
|
func genErrorHandler(status int, err error, body string) httpserver.Handler {
|
||||||
return httpserver.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
|
return httpserver.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||||
if len(body) > 0 {
|
if len(body) > 0 {
|
||||||
|
@ -166,3 +241,19 @@ func genErrorHandler(status int, err error, body string) httpserver.Handler {
|
||||||
return status, err
|
return status, err
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func createErrorPageFile(name string, content string) (string, error) {
|
||||||
|
errorPageFilePath := filepath.Join(os.TempDir(), name)
|
||||||
|
f, err := os.Create(errorPageFilePath)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = f.WriteString(content)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
f.Close()
|
||||||
|
|
||||||
|
return errorPageFilePath, nil
|
||||||
|
}
|
||||||
|
|
|
@ -122,11 +122,15 @@ func errorsParse(c *caddy.Controller) (*ErrorHandler, error) {
|
||||||
}
|
}
|
||||||
f.Close()
|
f.Close()
|
||||||
|
|
||||||
whatInt, err := strconv.Atoi(what)
|
if what == "*" {
|
||||||
if err != nil {
|
handler.GenericErrorPage = where
|
||||||
return hadBlock, c.Err("Expecting a numeric status code, got '" + what + "'")
|
} else {
|
||||||
|
whatInt, err := strconv.Atoi(what)
|
||||||
|
if err != nil {
|
||||||
|
return hadBlock, c.Err("Expecting a numeric status code or '*', got '" + what + "'")
|
||||||
|
}
|
||||||
|
handler.ErrorPages[whatInt] = where
|
||||||
}
|
}
|
||||||
handler.ErrorPages[whatInt] = where
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return hadBlock, nil
|
return hadBlock, nil
|
||||||
|
|
|
@ -103,6 +103,18 @@ func TestErrorsParse(t *testing.T) {
|
||||||
LocalTime: true,
|
LocalTime: true,
|
||||||
},
|
},
|
||||||
}},
|
}},
|
||||||
|
{`errors { log errors.txt
|
||||||
|
* generic_error.html
|
||||||
|
404 404.html
|
||||||
|
503 503.html
|
||||||
|
}`, false, ErrorHandler{
|
||||||
|
LogFile: "errors.txt",
|
||||||
|
GenericErrorPage: "generic_error.html",
|
||||||
|
ErrorPages: map[int]string{
|
||||||
|
404: "404.html",
|
||||||
|
503: "503.html",
|
||||||
|
},
|
||||||
|
}},
|
||||||
}
|
}
|
||||||
for i, test := range tests {
|
for i, test := range tests {
|
||||||
actualErrorsRule, err := errorsParse(caddy.NewTestController("http", test.inputErrorsRules))
|
actualErrorsRule, err := errorsParse(caddy.NewTestController("http", test.inputErrorsRules))
|
||||||
|
@ -150,6 +162,10 @@ func TestErrorsParse(t *testing.T) {
|
||||||
i, test.expectedErrorHandler.LogRoller.LocalTime, actualErrorsRule.LogRoller.LocalTime)
|
i, test.expectedErrorHandler.LogRoller.LocalTime, actualErrorsRule.LogRoller.LocalTime)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if actualErrorsRule.GenericErrorPage != test.expectedErrorHandler.GenericErrorPage {
|
||||||
|
t.Fatalf("Test %d expected GenericErrorPage to be %v, but got %v",
|
||||||
|
i, test.expectedErrorHandler.GenericErrorPage, actualErrorsRule.GenericErrorPage)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue