improved error page handling by tapping into Echo and its middleware
This commit is contained in:
@@ -10,7 +10,12 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
"git.erbosoft.com/amy/amsterdam/ui"
|
"git.erbosoft.com/amy/amsterdam/ui"
|
||||||
|
"github.com/labstack/echo/v4"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
/* NotImplPage is used for all TODO links, to show that something hasn't yet been implemented.
|
/* NotImplPage is used for all TODO links, to show that something hasn't yet been implemented.
|
||||||
@@ -27,3 +32,39 @@ func NotImplPage(ctxt ui.AmContext) (string, any, error) {
|
|||||||
ctxt.VarMap().Set("path", ctxt.URLPath())
|
ctxt.VarMap().Set("path", ctxt.URLPath())
|
||||||
return "framed_template", "notimpl.jet", nil
|
return "framed_template", "notimpl.jet", nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* AmErrorHandler handles all the mundane HTTP errors generated by the Echo engine.
|
||||||
|
* Parameters:
|
||||||
|
* err - The error to be handled.
|
||||||
|
* c - The Echo context error is being handled on.
|
||||||
|
*/
|
||||||
|
func AmErrorHandler(err error, c echo.Context) {
|
||||||
|
if c.Response().Committed {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
errCode := http.StatusInternalServerError
|
||||||
|
if he, ok := err.(*echo.HTTPError); ok {
|
||||||
|
errCode = he.Code
|
||||||
|
}
|
||||||
|
|
||||||
|
amctxt := ui.AmContextFromEchoContext(c)
|
||||||
|
if amctxt == nil {
|
||||||
|
var qerr error
|
||||||
|
amctxt, qerr = ui.AmCreateContext(c)
|
||||||
|
if qerr != nil {
|
||||||
|
log.Errorf("failed to create AmContext in error handler: %v", qerr)
|
||||||
|
c.String(errCode, fmt.Sprintf("error %d: %v", errCode, err))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
amctxt.SetLeftMenu("top")
|
||||||
|
amctxt.SetRC(errCode)
|
||||||
|
amctxt.VarMap().Set("error", err.Error())
|
||||||
|
// TODO: come up with a way to shift templates and titles for different error codes
|
||||||
|
amctxt.VarMap().Set("amsterdam_pageTitle", "Amsterdam Internal Server Error")
|
||||||
|
cerr := ui.AmSendPageData(c, amctxt, "framed_template", "error.jet")
|
||||||
|
if cerr != nil {
|
||||||
|
log.Errorf("Error rendering error %d (%v): %v", errCode, err, cerr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
+14
-1
@@ -11,7 +11,10 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
|
"bytes"
|
||||||
"io"
|
"io"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/labstack/echo/v4"
|
"github.com/labstack/echo/v4"
|
||||||
@@ -119,8 +122,18 @@ func LogrusMiddleware(next echo.HandlerFunc) echo.HandlerFunc {
|
|||||||
"uri": req.RequestURI,
|
"uri": req.RequestURI,
|
||||||
"status": res.Status,
|
"status": res.Status,
|
||||||
"latency_ms": stop.Sub(start).Milliseconds(),
|
"latency_ms": stop.Sub(start).Milliseconds(),
|
||||||
"user_agent": req.UserAgent(),
|
|
||||||
}).Info("handled request")
|
}).Info("handled request")
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// LogrusPanicLogging is a log function hooked into the recovery middleware.
|
||||||
|
func LogrusPanicLogging(c echo.Context, err error, stack []byte) error {
|
||||||
|
log.Errorf("[PANIC RECOVERY] %v", err)
|
||||||
|
scanner := bufio.NewScanner(bytes.NewReader(stack))
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := strings.ReplaceAll(scanner.Text(), "\t", " ")
|
||||||
|
log.Error(line)
|
||||||
|
}
|
||||||
|
return scanner.Err()
|
||||||
|
}
|
||||||
|
|||||||
@@ -33,8 +33,11 @@ func setupEcho() *echo.Echo {
|
|||||||
e := echo.New()
|
e := echo.New()
|
||||||
e.Logger = &EchoLogrusAdapter{}
|
e.Logger = &EchoLogrusAdapter{}
|
||||||
e.Renderer = &ui.TemplateRenderer{}
|
e.Renderer = &ui.TemplateRenderer{}
|
||||||
|
e.HTTPErrorHandler = AmErrorHandler
|
||||||
if !config.CommandLine.DebugPanic {
|
if !config.CommandLine.DebugPanic {
|
||||||
e.Use(middleware.Recover())
|
e.Use(middleware.RecoverWithConfig(middleware.RecoverConfig{
|
||||||
|
LogErrorFunc: LogrusPanicLogging,
|
||||||
|
}))
|
||||||
} else {
|
} else {
|
||||||
log.Warn("WARNING: --debug-panic in effect - DO NOT use this in production!")
|
log.Warn("WARNING: --debug-panic in effect - DO NOT use this in production!")
|
||||||
}
|
}
|
||||||
|
|||||||
+5
-5
@@ -21,7 +21,7 @@ import (
|
|||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
func sendPageData(ctxt echo.Context, amctxt AmContext, command string, data any) error {
|
func AmSendPageData(ctxt echo.Context, amctxt AmContext, command string, data any) error {
|
||||||
var err error
|
var err error
|
||||||
switch command {
|
switch command {
|
||||||
case "bytes":
|
case "bytes":
|
||||||
@@ -103,7 +103,7 @@ func AmWrap(myfunc func(AmContext) (string, any, error)) echo.HandlerFunc {
|
|||||||
amctxt.VarMap().Set("amsterdam_pageTitle", "IP Address Banned")
|
amctxt.VarMap().Set("amsterdam_pageTitle", "IP Address Banned")
|
||||||
amctxt.VarMap().Set("message", banmsg)
|
amctxt.VarMap().Set("message", banmsg)
|
||||||
amctxt.SetRC(http.StatusForbidden)
|
amctxt.SetRC(http.StatusForbidden)
|
||||||
return sendPageData(ctxt, amctxt, "framed_template", "ipban.jet")
|
return AmSendPageData(ctxt, amctxt, "framed_template", "ipban.jet")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for cookie login.
|
// Check for cookie login.
|
||||||
@@ -123,7 +123,7 @@ func AmWrap(myfunc func(AmContext) (string, any, error)) echo.HandlerFunc {
|
|||||||
}
|
}
|
||||||
if !user.VerifyEMail {
|
if !user.VerifyEMail {
|
||||||
// bounce to E-mail verification before we go anywhere
|
// bounce to E-mail verification before we go anywhere
|
||||||
return sendPageData(ctxt, amctxt, "redirect",
|
return AmSendPageData(ctxt, amctxt, "redirect",
|
||||||
"/verify?tgt="+url.QueryEscape(ctxt.Request().URL.Path))
|
"/verify?tgt="+url.QueryEscape(ctxt.Request().URL.Path))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -140,7 +140,7 @@ func AmWrap(myfunc func(AmContext) (string, any, error)) echo.HandlerFunc {
|
|||||||
ctxt.Logger().Errorf("Session save error: %v", err)
|
ctxt.Logger().Errorf("Session save error: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = sendPageData(ctxt, amctxt, what, rc)
|
err = AmSendPageData(ctxt, amctxt, what, rc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctxt.Logger().Errorf("Rendering error: %v", err)
|
ctxt.Logger().Errorf("Rendering error: %v", err)
|
||||||
}
|
}
|
||||||
@@ -148,7 +148,7 @@ func AmWrap(myfunc func(AmContext) (string, any, error)) echo.HandlerFunc {
|
|||||||
ctxt.Logger().Errorf("Page function error: %v", err)
|
ctxt.Logger().Errorf("Page function error: %v", err)
|
||||||
_, rc, _ = ErrorPage(amctxt, err)
|
_, rc, _ = ErrorPage(amctxt, err)
|
||||||
amctxt.SetRC(http.StatusInternalServerError)
|
amctxt.SetRC(http.StatusInternalServerError)
|
||||||
newerr := sendPageData(ctxt, amctxt, "framed_template", rc)
|
newerr := AmSendPageData(ctxt, amctxt, "framed_template", rc)
|
||||||
err = newerr
|
err = newerr
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
|
|||||||
+1
-1
@@ -15,7 +15,7 @@
|
|||||||
The Amsterdam server encountered an error: {{ CapitalizeString(error) }}.
|
The Amsterdam server encountered an error: {{ CapitalizeString(error) }}.
|
||||||
</p>
|
</p>
|
||||||
<p class="text-black text-sm mb-4">
|
<p class="text-black text-sm mb-4">
|
||||||
<a href="/">Click here</a> to return to the home page.
|
<a href="/">Click here to return to the home page.</a>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user