go-zero/zrpc/internal/serverinterceptors/sheddinginterceptor.go

58 lines
1.2 KiB
Go
Raw Normal View History

2020-07-26 17:09:05 +08:00
package serverinterceptors
import (
"context"
"errors"
2020-07-26 17:09:05 +08:00
"sync"
"github.com/zeromicro/go-zero/core/load"
"github.com/zeromicro/go-zero/core/stat"
2020-07-26 17:09:05 +08:00
"google.golang.org/grpc"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
2020-07-26 17:09:05 +08:00
)
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 any, info *grpc.UnaryServerInfo,
handler grpc.UnaryHandler) (val any, err error) {
2020-07-26 17:09:05 +08:00
sheddingStat.IncrementTotal()
var promise load.Promise
promise, err = shedder.Allow()
if err != nil {
metrics.AddDrop()
sheddingStat.IncrementDrop()
err = status.Error(codes.ResourceExhausted, err.Error())
2020-07-26 17:09:05 +08:00
return
}
defer func() {
if errors.Is(err, context.DeadlineExceeded) {
2020-07-26 17:09:05 +08:00
promise.Fail()
} else {
sheddingStat.IncrementPass()
promise.Pass()
}
}()
return handler(ctx, req)
}
}
func ensureSheddingStat() {
lock.Lock()
if sheddingStat == nil {
sheddingStat = load.NewSheddingStat(serviceType)
}
lock.Unlock()
}