v2fly/app/router/balancing.go

69 lines
1.7 KiB
Go
Raw Normal View History

//go:build !confonly
2019-02-01 19:08:21 +00:00
// +build !confonly
2018-11-07 20:08:20 +00:00
package router
import (
2021-04-08 19:55:25 +00:00
"context"
"github.com/v2fly/v2ray-core/v4/features/routing"
2021-04-08 19:55:25 +00:00
"github.com/v2fly/v2ray-core/v4/features/extension"
2021-02-16 20:31:50 +00:00
"github.com/v2fly/v2ray-core/v4/features/outbound"
2018-11-07 20:08:20 +00:00
)
type BalancingStrategy interface {
PickOutbound([]string) string
}
type Balancer struct {
selectors []string
strategy routing.BalancingStrategy
ohm outbound.Manager
fallbackTag string
override override
2018-11-07 20:08:20 +00:00
}
// PickOutbound picks the tag of a outbound
2018-11-07 20:08:20 +00:00
func (b *Balancer) PickOutbound() (string, error) {
candidates, err := b.SelectOutbounds()
if err != nil {
if b.fallbackTag != "" {
newError("fallback to [", b.fallbackTag, "], due to error: ", err).AtInfo().WriteToLog()
return b.fallbackTag, nil
}
return "", err
2018-11-07 20:08:20 +00:00
}
var tag string
if o := b.override.Get(); o != nil {
tag = b.strategy.Pick(o.selects)
} else {
tag = b.strategy.SelectAndPick(candidates)
2018-11-07 20:08:20 +00:00
}
if tag == "" {
if b.fallbackTag != "" {
newError("fallback to [", b.fallbackTag, "], due to empty tag returned").AtInfo().WriteToLog()
return b.fallbackTag, nil
}
// will use default handler
2018-11-07 20:08:20 +00:00
return "", newError("balancing strategy returns empty tag")
}
return tag, nil
}
2021-05-19 21:28:52 +00:00
2021-04-08 19:55:25 +00:00
func (b *Balancer) InjectContext(ctx context.Context) {
if contextReceiver, ok := b.strategy.(extension.ContextReceiver); ok {
contextReceiver.InjectContext(ctx)
}
}
// SelectOutbounds select outbounds with selectors of the Balancer
func (b *Balancer) SelectOutbounds() ([]string, error) {
hs, ok := b.ohm.(outbound.HandlerSelector)
if !ok {
return nil, newError("outbound.Manager is not a HandlerSelector")
}
tags := hs.Select(b.selectors)
return tags, nil
}