kratos/selector/p2c/p2c.go

110 lines
2.5 KiB

package p2c
import (
"context"
"math/rand"
"sync"
"sync/atomic"
"time"
"github.com/go-kratos/kratos/v2/selector"
"github.com/go-kratos/kratos/v2/selector/node/ewma"
)
const (
forcePick = time.Second * 3
// Name is balancer name
Name = "p2c"
)
var _ selector.Balancer = &Balancer{}
// WithFilter with select filters
func WithFilter(filters ...selector.Filter) Option {
return func(o *options) {
o.filters = filters
}
}
// Option is random builder option.
type Option func(o *options)
// options is random builder options
type options struct {
filters []selector.Filter
}
// New creates a p2c selector.
func New(opts ...Option) selector.Selector {
return NewBuilder(opts...).Build()
}
// Balancer is p2c selector.
type Balancer struct {
mu sync.Mutex
r *rand.Rand
picked int64
}
// choose two distinct nodes.
func (s *Balancer) prePick(nodes []selector.WeightedNode) (nodeA selector.WeightedNode, nodeB selector.WeightedNode) {
s.mu.Lock()
a := s.r.Intn(len(nodes))
b := s.r.Intn(len(nodes) - 1)
s.mu.Unlock()
if b >= a {
b = b + 1
}
nodeA, nodeB = nodes[a], nodes[b]
return
}
// Pick pick a node.
func (s *Balancer) Pick(ctx context.Context, nodes []selector.WeightedNode) (selector.WeightedNode, selector.DoneFunc, error) {
if len(nodes) == 0 {
return nil, nil, selector.ErrNoAvailable
}
if len(nodes) == 1 {
done := nodes[0].Pick()
return nodes[0], done, nil
}
var pc, upc selector.WeightedNode
nodeA, nodeB := s.prePick(nodes)
// meta.Weight is the weight set by the service publisher in discovery
if nodeB.Weight() > nodeA.Weight() {
pc, upc = nodeB, nodeA
} else {
pc, upc = nodeA, nodeB
}
// If the failed node has never been selected once during forceGap, it is forced to be selected once
// Take advantage of forced opportunities to trigger updates of success rate and delay
if upc.PickElapsed() > forcePick && atomic.CompareAndSwapInt64(&s.picked, 0, 1) {
pc = upc
atomic.StoreInt64(&s.picked, 0)
}
done := pc.Pick()
return pc, done, nil
}
// NewBuilder returns a selector builder with p2c balancer
func NewBuilder(opts ...Option) selector.Builder {
var option options
for _, opt := range opts {
opt(&option)
}
return &selector.DefaultBuilder{
Filters: option.filters,
Balancer: &Builder{},
Node: &ewma.Builder{},
}
}
// Builder is p2c builder
type Builder struct{}
// Build creates Balancer
func (b *Builder) Build() selector.Balancer {
return &Balancer{r: rand.New(rand.NewSource(time.Now().UnixNano()))}
}