core/isolation/slot.go (53 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 isolation
import (
"github.com/alibaba/sentinel-golang/core/base"
"github.com/alibaba/sentinel-golang/logging"
"github.com/pkg/errors"
)
const (
RuleCheckSlotOrder = 3000
)
var (
DefaultSlot = &Slot{}
)
type Slot struct {
}
func (s *Slot) Order() uint32 {
return RuleCheckSlotOrder
}
func (s *Slot) Check(ctx *base.EntryContext) *base.TokenResult {
resource := ctx.Resource.Name()
result := ctx.RuleCheckResult
if len(resource) == 0 {
return result
}
if passed, rule, snapshot := checkPass(ctx); !passed {
msg := "concurrency exceeds threshold"
if result == nil {
result = base.NewTokenResultBlockedWithCause(base.BlockTypeIsolation, msg, rule, snapshot)
} else {
result.ResetToBlockedWithCause(base.BlockTypeIsolation, msg, rule, snapshot)
}
}
return result
}
func checkPass(ctx *base.EntryContext) (bool, *Rule, uint32) {
statNode := ctx.StatNode
batchCount := ctx.Input.BatchCount
curCount := uint32(0)
for _, rule := range getRulesOfResource(ctx.Resource.Name()) {
threshold := rule.Threshold
if rule.MetricType == Concurrency {
if cur := statNode.CurrentConcurrency(); cur >= 0 {
curCount = uint32(cur)
} else {
curCount = 0
logging.Error(errors.New("negative concurrency"), "Negative concurrency in isolation.checkPass()", "rule", rule)
}
if curCount+batchCount > threshold {
return false, rule, curCount
}
}
}
return true, nil, curCount
}