diff options
author | wxiaoguang <wxiaoguang@gmail.com> | 2021-12-15 14:59:57 +0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-12-15 14:59:57 +0800 |
commit | 4da1d9781025aa4a85899b1ddeb25aabbaa82703 (patch) | |
tree | 9714f71f2fb042e44d418fbcfc56cd8440a1bbb1 /modules/context/context.go | |
parent | 9d943bf374e56e4d403303a6a2caafc1c79cdb6f (diff) | |
download | gitea-4da1d9781025aa4a85899b1ddeb25aabbaa82703.tar.gz gitea-4da1d9781025aa4a85899b1ddeb25aabbaa82703.zip |
Refactor HTTP request context (#17979)
Diffstat (limited to 'modules/context/context.go')
-rw-r--r-- | modules/context/context.go | 157 |
1 files changed, 60 insertions, 97 deletions
diff --git a/modules/context/context.go b/modules/context/context.go index 5a37f8b0bf..88cf498f82 100644 --- a/modules/context/context.go +++ b/modules/context/context.go @@ -9,7 +9,6 @@ import ( "context" "crypto/sha256" "encoding/hex" - "fmt" "html" "html/template" "io" @@ -156,6 +155,7 @@ func (ctx *Context) GetErrMsg() string { } // HasError returns true if error occurs in form validation. +// Attention: this function changes ctx.Data and ctx.Flash func (ctx *Context) HasError() bool { hasErr, ok := ctx.Data["HasError"] if !ok { @@ -191,29 +191,25 @@ func (ctx *Context) RedirectToFirst(location ...string) { ctx.Redirect(setting.AppSubURL + "/") } -// HTML calls Context.HTML and converts template name to string. +// HTML calls Context.HTML and renders the template to HTTP response func (ctx *Context) HTML(status int, name base.TplName) { log.Debug("Template: %s", name) - var startTime = time.Now() + tmplStartTime := time.Now() ctx.Data["TmplLoadTimes"] = func() string { - return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms" + return strconv.FormatInt(time.Since(tmplStartTime).Nanoseconds()/1e6, 10) + "ms" } if err := ctx.Render.HTML(ctx.Resp, status, string(name), ctx.Data); err != nil { if status == http.StatusInternalServerError && name == base.TplName("status/500") { - ctx.PlainText(http.StatusInternalServerError, []byte("Unable to find status/500 template")) + ctx.PlainText(http.StatusInternalServerError, "Unable to find status/500 template") return } ctx.ServerError("Render failed", err) } } -// HTMLString render content to a string but not http.ResponseWriter -func (ctx *Context) HTMLString(name string, data interface{}) (string, error) { +// RenderToString renders the template content to a string +func (ctx *Context) RenderToString(name base.TplName, data map[string]interface{}) (string, error) { var buf strings.Builder - var startTime = time.Now() - ctx.Data["TmplLoadTimes"] = func() string { - return fmt.Sprint(time.Since(startTime).Nanoseconds()/1e6) + "ms" - } err := ctx.Render.HTML(&buf, 200, string(name), data) return buf.String(), err } @@ -229,33 +225,30 @@ func (ctx *Context) RenderWithErr(msg string, tpl base.TplName, form interface{} } // NotFound displays a 404 (Not Found) page and prints the given error, if any. -func (ctx *Context) NotFound(title string, err error) { - ctx.notFoundInternal(title, err) +func (ctx *Context) NotFound(logMsg string, logErr error) { + ctx.notFoundInternal(logMsg, logErr) } -func (ctx *Context) notFoundInternal(title string, err error) { - if err != nil { - log.ErrorWithSkip(2, "%s: %v", title, err) +func (ctx *Context) notFoundInternal(logMsg string, logErr error) { + if logErr != nil { + log.ErrorWithSkip(2, "%s: %v", logMsg, logErr) if !setting.IsProd { - ctx.Data["ErrorMsg"] = err + ctx.Data["ErrorMsg"] = logErr } } - // response simple meesage if Accept isn't text/html - reqTypes, has := ctx.Req.Header["Accept"] - if has && len(reqTypes) > 0 { - notHTML := true - for _, part := range reqTypes { - if strings.Contains(part, "text/html") { - notHTML = false - break - } + // response simple message if Accept isn't text/html + showHTML := false + for _, part := range ctx.Req.Header["Accept"] { + if strings.Contains(part, "text/html") { + showHTML = true + break } + } - if notHTML { - ctx.PlainText(404, []byte("Not found.\n")) - return - } + if !showHTML { + ctx.PlainText(http.StatusNotFound, "Not found.\n") + return } ctx.Data["IsRepo"] = ctx.Repo.Repository != nil @@ -263,17 +256,16 @@ func (ctx *Context) notFoundInternal(title string, err error) { ctx.HTML(http.StatusNotFound, base.TplName("status/404")) } -// ServerError displays a 500 (Internal Server Error) page and prints the given -// error, if any. -func (ctx *Context) ServerError(title string, err error) { - ctx.serverErrorInternal(title, err) +// ServerError displays a 500 (Internal Server Error) page and prints the given error, if any. +func (ctx *Context) ServerError(logMsg string, logErr error) { + ctx.serverErrorInternal(logMsg, logErr) } -func (ctx *Context) serverErrorInternal(title string, err error) { - if err != nil { - log.ErrorWithSkip(2, "%s: %v", title, err) +func (ctx *Context) serverErrorInternal(logMsg string, logErr error) { + if logErr != nil { + log.ErrorWithSkip(2, "%s: %v", logMsg, logErr) if !setting.IsProd { - ctx.Data["ErrorMsg"] = err + ctx.Data["ErrorMsg"] = logErr } } @@ -282,37 +274,45 @@ func (ctx *Context) serverErrorInternal(title string, err error) { } // NotFoundOrServerError use error check function to determine if the error -// is about not found. It responses with 404 status code for not found 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(title string, errck func(error) bool, err error) { - if errck(err) { - ctx.notFoundInternal(title, err) +func (ctx *Context) NotFoundOrServerError(logMsg string, errCheck func(error) bool, err error) { + if errCheck(err) { + ctx.notFoundInternal(logMsg, err) return } + ctx.serverErrorInternal(logMsg, err) +} - ctx.serverErrorInternal(title, err) +// PlainTextBytes renders bytes as plain text +func (ctx *Context) PlainTextBytes(status int, bs []byte) { + if (status/100 == 4) || (status/100 == 5) { + log.Error("PlainTextBytes: %s", string(bs)) + } + ctx.Resp.WriteHeader(status) + ctx.Resp.Header().Set("Content-Type", "text/plain;charset=utf-8") + if _, err := ctx.Resp.Write(bs); err != nil { + log.Error("Write bytes failed: %v", err) + } } -// Header returns a header -func (ctx *Context) Header() http.Header { - return ctx.Resp.Header() +// PlainText renders content as plain text +func (ctx *Context) PlainText(status int, text string) { + ctx.PlainTextBytes(status, []byte(text)) } -// HandleText handles HTTP status code -func (ctx *Context) HandleText(status int, title string) { - if (status/100 == 4) || (status/100 == 5) { - log.Error("%s", title) - } - ctx.PlainText(status, []byte(title)) +// RespHeader returns the response header +func (ctx *Context) RespHeader() http.Header { + return ctx.Resp.Header() } // ServeContent serves content to http request func (ctx *Context) ServeContent(name string, r io.ReadSeeker, params ...interface{}) { - modtime := time.Now() + modTime := time.Now() for _, p := range params { switch v := p.(type) { case time.Time: - modtime = v + modTime = v } } ctx.Resp.Header().Set("Content-Description", "File Transfer") @@ -323,16 +323,7 @@ func (ctx *Context) ServeContent(name string, r io.ReadSeeker, params ...interfa ctx.Resp.Header().Set("Cache-Control", "must-revalidate") ctx.Resp.Header().Set("Pragma", "public") ctx.Resp.Header().Set("Access-Control-Expose-Headers", "Content-Disposition") - http.ServeContent(ctx.Resp, ctx.Req, name, modtime, r) -} - -// PlainText render content as plain text -func (ctx *Context) PlainText(status int, bs []byte) { - ctx.Resp.WriteHeader(status) - ctx.Resp.Header().Set("Content-Type", "text/plain;charset=utf-8") - if _, err := ctx.Resp.Write(bs); err != nil { - ctx.ServerError("Write bytes failed", err) - } + http.ServeContent(ctx.Resp, ctx.Req, name, modTime, r) } // ServeFile serves given file to response. @@ -386,7 +377,7 @@ func (ctx *Context) JSON(status int, content interface{}) { } } -// Redirect redirect the request +// Redirect redirects the request func (ctx *Context) Redirect(location string, status ...int) { code := http.StatusFound if len(status) == 1 { @@ -506,7 +497,7 @@ func (ctx *Context) SetParams(k, v string) { chiCtx.URLParams.Add(strings.TrimPrefix(k, ":"), url.PathEscape(v)) } -// Write writes data to webbrowser +// Write writes data to web browser func (ctx *Context) Write(bs []byte) (int, error) { return ctx.Resp.Write(bs) } @@ -544,10 +535,9 @@ func (ctx *Context) Value(key interface{}) interface{} { // Handler represents a custom handler type Handler func(*Context) -// enumerate all content -var ( - contextKey interface{} = "default_context" -) +type contextKeyType struct{} + +var contextKey interface{} = contextKeyType{} // WithContext set up install context in request func WithContext(req *http.Request, ctx *Context) *http.Request { @@ -570,31 +560,6 @@ func GetContextUser(req *http.Request) *user_model.User { return nil } -// SignedUserName returns signed user's name via context -func SignedUserName(req *http.Request) string { - if middleware.IsInternalPath(req) { - return "" - } - if middleware.IsAPIPath(req) { - ctx, ok := req.Context().Value(apiContextKey).(*APIContext) - if ok { - v := ctx.Data["SignedUserName"] - if res, ok := v.(string); ok { - return res - } - } - } else { - ctx, ok := req.Context().Value(contextKey).(*Context) - if ok { - v := ctx.Data["SignedUserName"] - if res, ok := v.(string); ok { - return res - } - } - } - return "" -} - func getCsrfOpts() CsrfOptions { return CsrfOptions{ Secret: setting.SecretKey, @@ -727,8 +692,6 @@ func Contexter() func(next http.Handler) http.Handler { ctx.Data["CsrfToken"] = html.EscapeString(ctx.csrf.GetToken()) ctx.Data["CsrfTokenHtml"] = template.HTML(`<input type="hidden" name="_csrf" value="` + ctx.Data["CsrfToken"].(string) + `">`) - log.Debug("Session ID: %s", ctx.Session.ID()) - log.Debug("CSRF Token: %v", ctx.Data["CsrfToken"]) // FIXME: do we really always need these setting? There should be someway to have to avoid having to always set these ctx.Data["IsLandingPageHome"] = setting.LandingPageURL == setting.LandingPageHome |