2020-07-26 17:09:05 +08:00
|
|
|
package serverinterceptors
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
|
2020-08-08 16:40:10 +08:00
|
|
|
"github.com/tal-tech/go-zero/core/load"
|
|
|
|
"github.com/tal-tech/go-zero/core/stat"
|
2020-07-26 17:09:05 +08:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
)
|
|
|
|
|
|
|
|
const serviceType = "rpc"
|
|
|
|
|
|
|
|
var (
|
|
|
|
sheddingStat *load.SheddingStat
|
|
|
|
lock sync.Mutex
|
|
|
|
)
|
|
|
|
|
2021-03-01 23:52:44 +08:00
|
|
|
// UnarySheddingInterceptor returns a func that does load shedding on processing unary requests.
|
2020-07-26 17:09:05 +08:00
|
|
|
func UnarySheddingInterceptor(shedder load.Shedder, metrics *stat.Metrics) grpc.UnaryServerInterceptor {
|
|
|
|
ensureSheddingStat()
|
|
|
|
|
|
|
|
return func(ctx context.Context, req interface{}, info *grpc.UnaryServerInfo,
|
|
|
|
handler grpc.UnaryHandler) (val interface{}, err error) {
|
|
|
|
sheddingStat.IncrementTotal()
|
|
|
|
var promise load.Promise
|
|
|
|
promise, err = shedder.Allow()
|
|
|
|
if err != nil {
|
|
|
|
metrics.AddDrop()
|
|
|
|
sheddingStat.IncrementDrop()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err == context.DeadlineExceeded {
|
|
|
|
promise.Fail()
|
|
|
|
} else {
|
|
|
|
sheddingStat.IncrementPass()
|
|
|
|
promise.Pass()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return handler(ctx, req)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ensureSheddingStat() {
|
|
|
|
lock.Lock()
|
|
|
|
if sheddingStat == nil {
|
|
|
|
sheddingStat = load.NewSheddingStat(serviceType)
|
|
|
|
}
|
|
|
|
lock.Unlock()
|
|
|
|
}
|