core/flow/slot.go (75 lines of code) (raw):
// Copyright 1999-2020 Alibaba Group Holding Ltd.
//
// 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 flow
import (
"github.com/alibaba/sentinel-golang/core/base"
"github.com/alibaba/sentinel-golang/core/stat"
metric_exporter "github.com/alibaba/sentinel-golang/exporter/metric"
"github.com/alibaba/sentinel-golang/logging"
"github.com/alibaba/sentinel-golang/util"
"github.com/pkg/errors"
)
const (
RuleCheckSlotOrder = 2000
)
var (
DefaultSlot = &Slot{}
flowWaitCount = metric_exporter.NewCounter(
"flow_wait_total",
"Flow wait count",
[]string{"resource"})
)
func init() {
metric_exporter.Register(flowWaitCount)
}
type Slot struct {
}
func (s *Slot) Order() uint32 {
return RuleCheckSlotOrder
}
func (s *Slot) Check(ctx *base.EntryContext) *base.TokenResult {
res := ctx.Resource.Name()
tcs := getTrafficControllerListFor(res)
result := ctx.RuleCheckResult
// Check rules in order
for _, tc := range tcs {
if tc == nil {
logging.Warn("[FlowSlot Check]Nil traffic controller found", "resourceName", res)
continue
}
r := canPassCheck(tc, ctx.StatNode, ctx.Input.BatchCount)
if r == nil {
// nil means pass
continue
}
if r.Status() == base.ResultStatusBlocked {
return r
}
if r.Status() == base.ResultStatusShouldWait {
if nanosToWait := r.NanosToWait(); nanosToWait > 0 {
flowWaitCount.Add(float64(ctx.Input.BatchCount), ctx.Resource.Name())
// Handle waiting action.
util.Sleep(nanosToWait)
}
continue
}
}
return result
}
func canPassCheck(tc *TrafficShapingController, node base.StatNode, batchCount uint32) *base.TokenResult {
return canPassCheckWithFlag(tc, node, batchCount, 0)
}
func canPassCheckWithFlag(tc *TrafficShapingController, node base.StatNode, batchCount uint32, flag int32) *base.TokenResult {
return checkInLocal(tc, node, batchCount, flag)
}
func selectNodeByRelStrategy(rule *Rule, node base.StatNode) base.StatNode {
if rule.RelationStrategy == AssociatedResource {
return stat.GetResourceNode(rule.RefResource)
}
return node
}
func checkInLocal(tc *TrafficShapingController, resStat base.StatNode, batchCount uint32, flag int32) *base.TokenResult {
actual := selectNodeByRelStrategy(tc.rule, resStat)
if actual == nil {
logging.FrequentErrorOnce.Do(func() {
logging.Error(errors.Errorf("nil resource node"), "No resource node for flow rule in FlowSlot.checkInLocal()", "rule", tc.rule)
})
return base.NewTokenResultPass()
}
return tc.PerformChecking(actual, batchCount, flag)
}