1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 11:18:50 +08:00

update context of the original http request (#2717)

This commit is contained in:
John Guo 2023-06-28 21:04:49 +08:00 committed by GitHub
parent 8c4a0b61b8
commit 6eb0de42f8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 41 additions and 30 deletions

View File

@ -21,6 +21,7 @@ import (
"github.com/gogf/gf/v2/net/goai"
"github.com/gogf/gf/v2/net/gsvc"
"github.com/gogf/gf/v2/os/gcache"
"github.com/gogf/gf/v2/os/gctx"
"github.com/gogf/gf/v2/os/gsession"
"github.com/gogf/gf/v2/util/gtag"
)
@ -129,19 +130,19 @@ const (
)
const (
supportedHttpMethods = "GET,PUT,POST,DELETE,PATCH,HEAD,CONNECT,OPTIONS,TRACE"
defaultMethod = "ALL"
routeCacheDuration = time.Hour
ctxKeyForRequest = "gHttpRequestObject"
contentTypeXml = "text/xml"
contentTypeHtml = "text/html"
contentTypeJson = "application/json"
swaggerUIPackedPath = "/goframe/swaggerui"
responseHeaderTraceID = "Trace-ID"
responseHeaderContentLength = "Content-Length"
specialMethodNameInit = "Init"
specialMethodNameShut = "Shut"
specialMethodNameIndex = "Index"
supportedHttpMethods = "GET,PUT,POST,DELETE,PATCH,HEAD,CONNECT,OPTIONS,TRACE"
defaultMethod = "ALL"
routeCacheDuration = time.Hour
ctxKeyForRequest gctx.StrKey = "gHttpRequestObject"
contentTypeXml = "text/xml"
contentTypeHtml = "text/html"
contentTypeJson = "application/json"
swaggerUIPackedPath = "/goframe/swaggerui"
responseHeaderTraceID = "Trace-ID"
responseHeaderContentLength = "Content-Length"
specialMethodNameInit = "Init"
specialMethodNameShut = "Shut"
specialMethodNameIndex = "Index"
)
const (

View File

@ -7,7 +7,6 @@
package ghttp
import (
"context"
"fmt"
"net/http"
"strings"
@ -40,7 +39,6 @@ type Request struct {
// Private attributes for internal usage purpose.
// =================================================================================================================
context context.Context // Custom context for internal usage purpose.
handlers []*HandlerItemParsed // All matched handlers containing handler, hook and middleware for this request.
serveHandler *HandlerItemParsed // Real handler serving for this request, not hook or middleware.
handlerResponse interface{} // Handler response object for Request/Response handler.

View File

@ -45,7 +45,8 @@ func (m *middleware) Next() {
// Router values switching.
m.request.routerMap = item.Values
ctx := m.request.context
var ctx = m.request.Context()
gutil.TryCatch(ctx, func(ctx context.Context) {
// Execute bound middleware array of the item if it's not empty.
if m.handlerMDIndex < len(item.Handler.Middleware) {

View File

@ -46,29 +46,38 @@ func RequestFromCtx(ctx context.Context) *Request {
// This function overwrites the http.Request.Context function.
// See GetCtx.
func (r *Request) Context() context.Context {
if r.context == nil {
// It forbids the context manually done,
// to make the context can be propagated to asynchronous goroutines.
r.context = &neverDoneCtx{
r.Request.Context(),
}
r.context = gctx.WithCtx(r.context)
var ctx = r.Request.Context()
// Check and inject Request object into context.
if RequestFromCtx(ctx) == nil {
// Inject Request object into context.
ctx = context.WithValue(ctx, ctxKeyForRequest, r)
// Add default tracing info if using default tracing provider.
ctx = gctx.WithCtx(ctx)
// Update the values of the original HTTP request.
*r.Request = *r.Request.WithContext(ctx)
}
// Inject Request object into context.
if RequestFromCtx(r.context) == nil {
r.context = context.WithValue(r.context, ctxKeyForRequest, r)
}
return r.context
return ctx
}
// GetCtx retrieves and returns the request's context.
// Its alias of function Context,to be relevant with function SetCtx.
func (r *Request) GetCtx() context.Context {
return r.Context()
}
// GetNeverDoneCtx creates and returns a never done context object,
// which forbids the context manually done, to make the context can be propagated to asynchronous goroutines,
// which will not be affected by the HTTP request ends.
//
// This change is considered for common usage habits of developers for context propagation
// in multiple goroutines creation in one HTTP request.
func (r *Request) GetNeverDoneCtx() context.Context {
return &neverDoneCtx{r.Context()}
}
// SetCtx custom context for current request.
func (r *Request) SetCtx(ctx context.Context) {
r.context = ctx
*r.Request = *r.WithContext(ctx)
}
// GetCtxVar retrieves and returns a Var with a given key name.
@ -84,5 +93,7 @@ func (r *Request) GetCtxVar(key interface{}, def ...interface{}) *gvar.Var {
// SetCtxVar sets custom parameter to context with key-value pairs.
func (r *Request) SetCtxVar(key interface{}, value interface{}) {
r.context = context.WithValue(r.Context(), key, value)
var ctx = r.Context()
ctx = context.WithValue(ctx, key, value)
*r.Request = *r.Request.WithContext(ctx)
}