我有一个 golang API 端点及其关联的上下文。
端点需要在幕后执行一些繁重的工作,因此我在主端点内创建一个新的子例程,然后返回响应本身。
为了处理上下文取消,我创建了一个后台上下文并将其作为新上下文传递给子例程。
问题是通过这样做,是的,我可以执行后台子例程,但是主上下文中用于跟踪的值,如请求 id、span id 等(大多数键对我来说是未知的)将是丢失的。
即使在响应发送到客户端之后,如何将父上下文传递给子例程而不取消执行。
我没有将任何值传递到上下文中。但最初我们传递的是跟踪所需的 request-id、span-id 等。这些信息都在上下文中。这是一个内部库,上下文是我们保存它的地方。
我知道这是使用上下文传递值的反模式,除了请求 ID 和其他对库而不是业务逻辑重要的值之外,不会传递任何值
当您取消父上下文时,从其派生的所有上下文也将取消。因此,为从请求处理程序生成的 goroutine 创建新上下文是正确的。
当您创建新上下文时,您应该将您感兴趣的所有值从原始上下文复制到新上下文。然而,你说你不知道所有的钥匙。因此,您仍然可以保留对父上下文的引用,以便可以查询它的值。像这样的东西:
type nestedContext struct {
context.Context
parent context.Context
}
func (n nestedContext) Value(key any) any {
return n.parent.Value(key)
}
...
newContext := nestedContext{
Context:context.Background(),
parent: parentContext,
}
Run Code Online (Sandbox Code Playgroud)
这将从中创建一个新的上下文,context.Background()
该上下文将从取消的父上下文中查找值。
作为上下文传递newContext
给从处理程序创建的 goroutine。