-2

我正在使用 go 编写一个或多或少简单的 Web 应用程序,它提供了一个 rest api。当请求进来时,我想将用户的 id 临时存储在请求上下文中,它是 api 令牌的一部分。在阅读 了一些 文章文档之后,我仍然很困惑如何确保使用 附加到上下文的值context.WithValue()可以在没有类型断言的情况下使用,而是使用某种结构。

到目前为止,这是我想出的:

// RequestContext contains the application-specific information that are carried around in a request.
type RequestContext interface {
    context.Context
    // UserID returns the ID of the user for the current request
    UserID() uuid.UUID
    // SetUserID sets the ID of the currently authenticated user
    SetUserID(id uuid.UUID)
}

type requestContext struct {
    context.Context           // the golang context
    now             time.Time // the time when the request is being processed
    userID          uuid.UUID // an ID identifying the current user
}

func (ctx *requestContext) UserID() uuid.UUID {
    return ctx.userID
}

func (ctx *requestContext) SetUserID(id uuid.UUID) {
    ctx.userID = id
}

func (ctx *requestContext) Now() time.Time {
    return ctx.now
}

// NewRequestContext creates a new RequestContext with the current request information.
func NewRequestContext(now time.Time, r *http.Request) RequestContext {
    return &requestContext{
        Context: r.Context(),
        now:     now,
    }
}

// RequestContextHandler is a middleware that sets up a new RequestContext and attaches it to the current request.
func RequestContextHandler(next http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        now := time.Now()
        next.ServeHTTP(w, r.WithContext(NewRequestContext(now, r)))
    })
}

我想知道如何在处理程序中访问请求上下文的SetUserID()UserID()函数,或者是否有替代的类似方法。

4

2 回答 2

4

要使用您构建的内容,您必须使用类型断言:

rctx:=request.Context().(RequestContext)

如果您有一个以与您相同的方式包装上下文的中间件,这将中断。

另一种方法是使用基本上下文,并添加带有私钥的助手来访问值:

type reqKey int

const key reqKey=iota

type RequestData struct {
   UserID uuid.UUID
   Now time.Time
}

func ContextWithRequest(ctx context.Context,req requestData) context.Context {
  return context.WithValue(ctx,key,req)
}

func GetRequest(ctx context) RequestData {
  x:=ctx.Value(key)
  if x!=nil {
     return x.(RequestData)
  }
  return RequestData{}
}

当然,如果要更改RequestData,则需要添加指向上下文的指针。

于 2019-12-04T23:28:16.777 回答
0

这就是我现在在我的应用程序中使用的。我使用简单的 settters 和 getter 从上下文中检索值。

import (
    "context"
    "github.com/google/uuid"
)

const userIdContextKey = "user_id" // type is uuid

func addUserIdToContext(ctx context.Context, user uuid.UUID) context.Context {
    return context.WithValue(ctx, UserIdContextKey, user)
}

func userIDFromContext(ctx context.Context) uuid.UUID {
    return ctx.Value(UserIdContextKey).(uuid.UUID)
}

然后在一个中间件中,我已经检索了 userId,我通过将上下文和用户 ID 传递给addUserIdToContext()函数来将值添加到上下文中。

ctx := addUserIdToContext(r.Context(), userId)

当我想从上下文中检索用户时,我只需要调用userIDFromContext(ctx)并将请求上下文传递给它。

func CreateResource(w http.ResponseWriter, r *http.Request) {
    createdBy := userIDFromContext(r.Context())
    // do something with createdBy
}

我知道这不是最优雅甚至类型安全的解决方案,但是,嘿,因为我正在尝试构建它,它工作得很好。

于 2019-12-17T21:52:21.630 回答