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:
parent
8c4a0b61b8
commit
6eb0de42f8
@ -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 (
|
||||
|
@ -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.
|
||||
|
@ -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) {
|
||||
|
@ -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)
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user