-
Notifications
You must be signed in to change notification settings - Fork 837
/
Copy pathctx.go
81 lines (72 loc) · 2.24 KB
/
ctx.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
/*
* Copyright 2021 CloudWeGo Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package rpcinfo
import (
"context"
"github.com/cloudwego/kitex/internal"
)
type ctxRPCInfoKeyType struct{}
var ctxRPCInfoKey ctxRPCInfoKeyType
// NewCtxWithRPCInfo creates a new context with the RPCInfo given.
func NewCtxWithRPCInfo(ctx context.Context, ri RPCInfo) context.Context {
if ri != nil {
return context.WithValue(ctx, ctxRPCInfoKey, ri)
}
return ctx
}
// GetRPCInfo gets RPCInfo from ctx.
// Returns nil if not found.
func GetRPCInfo(ctx context.Context) RPCInfo {
if ri, ok := ctx.Value(ctxRPCInfoKey).(RPCInfo); ok {
return ri
}
return nil
}
// PutRPCInfo recycles the RPCInfo. This function is for internal use only.
func PutRPCInfo(ri RPCInfo) {
if v, ok := ri.(internal.Reusable); ok {
v.Recycle()
}
}
// FreezeRPCInfo returns a new context containing an RPCInfo that is safe
// to be used asynchronically.
// Note that the RPCStats of the freezed RPCInfo will be nil and
// the FreezeRPCInfo itself should not be used asynchronically.
//
// Example:
//
// func (p *MyServiceImpl) MyMethod(ctx context.Context, req *MyRequest) (resp *MyResponse, err error) {
// ri := rpcinfo.GetRPCInfo(ctx)
// go func(ctx context.Context) {
// ...
// ri := rpcinfo.GetRPCInfo(ctx) // not concurrent-safe
// ...
// }(ctx)
//
// ctx2 := rpcinfo.FreezeRPCInfo(ctx) // this creates a read-only copy of `ri` and attaches it to the new context
// go func(ctx context.Context) {
// ...
// ri := rpcinfo.GetRPCInfo(ctx) // OK
// ...
// }(ctx2)
// }
func FreezeRPCInfo(ctx context.Context) context.Context {
ri := GetRPCInfo(ctx)
if ri == nil {
return ctx
}
return NewCtxWithRPCInfo(ctx, freeze(ri))
}