pkg/filter/traffic/traffic.go (156 lines of code) (raw):

/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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 traffic import ( "errors" "fmt" "math/rand" "strings" "time" ) import ( "github.com/apache/dubbo-go-pixiu/pkg/common/constant" "github.com/apache/dubbo-go-pixiu/pkg/common/extension/filter" "github.com/apache/dubbo-go-pixiu/pkg/context/http" "github.com/apache/dubbo-go-pixiu/pkg/logger" ) const ( canaryByHeader CanaryHeaders = "canary-by-header" unInitialize int = -1 ) func init() { filter.RegisterHttpFilter(&Plugin{}) } type ( CanaryHeaders string Plugin struct { } FilterFactory struct { cfg *Config record map[string]struct{} rulesMap map[string][]*ClusterWrapper } Filter struct { weight int Rules []*ClusterWrapper } Config struct { Traffics []*Cluster `yaml:"traffics" json:"traffics" mapstructure:"traffics"` } ClusterWrapper struct { weightCeil int weightFloor int header string Cluster *Cluster } Cluster struct { Name string `yaml:"name" json:"name" mapstructure:"name"` Router string `yaml:"router" json:"router" mapstructure:"router"` CanaryByHeader string `yaml:"canary-by-header" json:"canary-by-header" mapstructure:"canary-by-header"` CanaryWeight int `yaml:"canary-weight" json:"canary-weight" mapstructure:"canary-weight"` } ) func (p *Plugin) Kind() string { return constant.HTTPTrafficFilter } func (p *Plugin) CreateFilterFactory() (filter.HttpFilterFactory, error) { return &FilterFactory{ cfg: &Config{}, record: map[string]struct{}{}, rulesMap: map[string][]*ClusterWrapper{}, }, nil } func (factory *FilterFactory) Config() any { return factory.cfg } func (factory *FilterFactory) Apply() error { var ( router string up int ) for _, cluster := range factory.cfg.Traffics { up = 0 router = cluster.Router if len(factory.rulesMap[router]) != 0 { lastCeil := factory.rulesMap[router][len(factory.rulesMap[router])-1].weightCeil if lastCeil != -1 { up = lastCeil } } wp := &ClusterWrapper{ Cluster: cluster, weightCeil: -1, weightFloor: -1, } if cluster.CanaryByHeader != "" { if _, ok := factory.record[cluster.CanaryByHeader]; ok { return errors.New("duplicate canary-by-header values") } else { factory.record[cluster.CanaryByHeader] = struct{}{} wp.header = cluster.CanaryByHeader } } if cluster.CanaryWeight > 0 && cluster.CanaryWeight <= 100 { if up+cluster.CanaryWeight > 100 { return fmt.Errorf("[dubbo-go-pixiu] clusters' weight sum more than 100 in %v service!", cluster.Router) } else { wp.weightFloor = up up += cluster.CanaryWeight wp.weightCeil = up } } factory.rulesMap[router] = append(factory.rulesMap[router], wp) } return nil } func (factory *FilterFactory) PrepareFilterChain(ctx *http.HttpContext, chain filter.FilterChain) error { f := &Filter{ weight: unInitialize, } f.Rules = factory.rulesMatch(ctx.Request.RequestURI) chain.AppendDecodeFilters(f) return nil } func (f *Filter) Decode(ctx *http.HttpContext) filter.FilterStatus { if f.Rules != nil { var cluster string for _, wp := range f.Rules { if f.trafficHeader(wp, ctx) { cluster = wp.Cluster.Name logger.Debugf("[dubbo-go-pixiu] execute traffic split to cluster %s", wp.Cluster.Name) break } } if cluster != "" { ctx.Route.Cluster = cluster } else { for _, wp := range f.Rules { if f.trafficWeight(wp) { ctx.Route.Cluster = wp.Cluster.Name cluster = wp.Cluster.Name logger.Debugf("[dubbo-go-pixiu] execute traffic split to cluster %s", cluster) break } } } } else { logger.Warnf("[dubbo-go-pixiu] execute traffic split fail because of empty rules.") } return filter.Continue } func (f *Filter) trafficHeader(c *ClusterWrapper, ctx *http.HttpContext) bool { return spiltHeader(ctx.Request, c.header) } func (f *Filter) trafficWeight(c *ClusterWrapper) bool { if f.weight == unInitialize { rng := rand.New(rand.NewSource(time.Now().UnixNano())) f.weight = rng.Intn(100) + 1 } return spiltWeight(f.weight, c.weightFloor, c.weightCeil) } func (factory *FilterFactory) rulesMatch(path string) []*ClusterWrapper { for key := range factory.rulesMap { if strings.HasPrefix(path, key) { return factory.rulesMap[key] } } return nil }