aboutsummaryrefslogtreecommitdiffstats
path: root/modules/context
diff options
context:
space:
mode:
authorwxiaoguang <wxiaoguang@gmail.com>2023-04-14 13:19:11 +0800
committerGitHub <noreply@github.com>2023-04-14 13:19:11 +0800
commit1c8bc4081a4f4d0d921ac218cb724ce97924d410 (patch)
tree038587701606c7abb11b29f5b63a14e12cb2239e /modules/context
parent5768bafeb28e4e4212ae0e2abc7f22c9c8b7c653 (diff)
downloadgitea-1c8bc4081a4f4d0d921ac218cb724ce97924d410.tar.gz
gitea-1c8bc4081a4f4d0d921ac218cb724ce97924d410.zip
Show friendly 500 error page to users and developers (#24110)
Close #24104 This also introduces many tests to cover many complex error handling functions. ### Before The details are never shown in production. <details> ![image](https://user-images.githubusercontent.com/2114189/231805004-13214579-4fbe-465a-821c-be75c2749097.png) </details> ### After The details could be shown to site admin users. It is safe. ![image](https://user-images.githubusercontent.com/2114189/231803912-d5660994-416f-4b27-a4f1-a4cc962091d4.png)
Diffstat (limited to 'modules/context')
-rw-r--r--modules/context/context.go44
1 files changed, 10 insertions, 34 deletions
diff --git a/modules/context/context.go b/modules/context/context.go
index cee533e42a..2507cc10c0 100644
--- a/modules/context/context.go
+++ b/modules/context/context.go
@@ -16,10 +16,8 @@ import (
"net/http"
"net/url"
"path"
- "regexp"
"strconv"
"strings"
- texttemplate "text/template"
"time"
"code.gitea.io/gitea/models/db"
@@ -216,7 +214,7 @@ func (ctx *Context) RedirectToFirst(location ...string) {
ctx.Redirect(setting.AppSubURL + "/")
}
-var templateExecutingErr = regexp.MustCompile(`^template: (.*):([1-9][0-9]*):([1-9][0-9]*): executing (?:"(.*)" at <(.*)>: )?`)
+const tplStatus500 base.TplName = "status/500"
// HTML calls Context.HTML and renders the template to HTTP response
func (ctx *Context) HTML(status int, name base.TplName) {
@@ -229,34 +227,11 @@ func (ctx *Context) HTML(status int, name base.TplName) {
return strconv.FormatInt(time.Since(tmplStartTime).Nanoseconds()/1e6, 10) + "ms"
}
if err := ctx.Render.HTML(ctx.Resp, status, string(name), templates.BaseVars().Merge(ctx.Data)); err != nil {
- if status == http.StatusInternalServerError && name == base.TplName("status/500") {
+ if status == http.StatusInternalServerError && name == tplStatus500 {
ctx.PlainText(http.StatusInternalServerError, "Unable to find HTML templates, the template system is not initialized, or Gitea can't find your template files.")
return
}
- if execErr, ok := err.(texttemplate.ExecError); ok {
- if groups := templateExecutingErr.FindStringSubmatch(err.Error()); len(groups) > 0 {
- errorTemplateName, lineStr, posStr := groups[1], groups[2], groups[3]
- target := ""
- if len(groups) == 6 {
- target = groups[5]
- }
- line, _ := strconv.Atoi(lineStr) // Cannot error out as groups[2] is [1-9][0-9]*
- pos, _ := strconv.Atoi(posStr) // Cannot error out as groups[3] is [1-9][0-9]*
- assetLayerName := templates.AssetFS().GetFileLayerName(errorTemplateName + ".tmpl")
- filename := fmt.Sprintf("(%s) %s", assetLayerName, errorTemplateName)
- if errorTemplateName != string(name) {
- filename += " (subtemplate of " + string(name) + ")"
- }
- err = fmt.Errorf("failed to render %s, error: %w:\n%s", filename, err, templates.GetLineFromTemplate(errorTemplateName, line, target, pos))
- } else {
- assetLayerName := templates.AssetFS().GetFileLayerName(execErr.Name + ".tmpl")
- filename := fmt.Sprintf("(%s) %s", assetLayerName, execErr.Name)
- if execErr.Name != string(name) {
- filename += " (subtemplate of " + string(name) + ")"
- }
- err = fmt.Errorf("failed to render %s, error: %w", filename, err)
- }
- }
+ err = fmt.Errorf("failed to render template: %s, error: %s", name, templates.HandleTemplateRenderingError(err))
ctx.ServerError("Render failed", err)
}
}
@@ -324,24 +299,25 @@ func (ctx *Context) serverErrorInternal(logMsg string, logErr error) {
return
}
- if !setting.IsProd {
+ // it's safe to show internal error to admin users, and it helps
+ if !setting.IsProd || (ctx.Doer != nil && ctx.Doer.IsAdmin) {
ctx.Data["ErrorMsg"] = logErr
}
}
ctx.Data["Title"] = "Internal Server Error"
- ctx.HTML(http.StatusInternalServerError, base.TplName("status/500"))
+ ctx.HTML(http.StatusInternalServerError, tplStatus500)
}
// NotFoundOrServerError use error check function to determine if the error
// is about not found. It responds with 404 status code for not found error,
// or error context description for logging purpose of 500 server error.
-func (ctx *Context) NotFoundOrServerError(logMsg string, errCheck func(error) bool, err error) {
- if errCheck(err) {
- ctx.notFoundInternal(logMsg, err)
+func (ctx *Context) NotFoundOrServerError(logMsg string, errCheck func(error) bool, logErr error) {
+ if errCheck(logErr) {
+ ctx.notFoundInternal(logMsg, logErr)
return
}
- ctx.serverErrorInternal(logMsg, err)
+ ctx.serverErrorInternal(logMsg, logErr)
}
// PlainTextBytes renders bytes as plain text