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 }