Files
Archive/echo/internal/relay/conf/cfg.go
2024-04-07 20:25:37 +02:00

139 lines
3.3 KiB
Go

package conf
import (
"errors"
"fmt"
"github.com/Ehco1996/ehco/internal/constant"
"github.com/Ehco1996/ehco/pkg/lb"
"go.uber.org/zap"
)
type Config struct {
Listen string `json:"listen"`
ListenType string `json:"listen_type"`
TransportType string `json:"transport_type"`
TCPRemotes []string `json:"tcp_remotes"`
UDPRemotes []string `json:"udp_remotes"`
Label string `json:"label,omitempty"`
}
func (r *Config) Validate() error {
if r.Adjust() != nil {
return errors.New("adjust config failed")
}
if r.ListenType != constant.RelayTypeRaw &&
r.ListenType != constant.RelayTypeWS &&
r.ListenType != constant.RelayTypeWSS &&
r.ListenType != constant.RelayTypeMTCP &&
r.ListenType != constant.RelayTypeMWSS {
return fmt.Errorf("invalid listen type:%s", r.ListenType)
}
if r.TransportType != constant.RelayTypeRaw &&
r.TransportType != constant.RelayTypeWS &&
r.TransportType != constant.RelayTypeWSS &&
r.TransportType != constant.RelayTypeMTCP &&
r.TransportType != constant.RelayTypeMWSS {
return fmt.Errorf("invalid transport type:%s", r.ListenType)
}
if r.Listen == "" {
return fmt.Errorf("invalid listen:%s", r.Listen)
}
for _, addr := range r.TCPRemotes {
if addr == "" {
return fmt.Errorf("invalid tcp remote addr:%s", addr)
}
}
for _, addr := range r.UDPRemotes {
if addr == "" {
return fmt.Errorf("invalid udp remote addr:%s", addr)
}
}
if len(r.TCPRemotes) == 0 && len(r.UDPRemotes) == 0 {
return errors.New("both tcp and udp remotes are empty")
}
if len(r.UDPRemotes) > 0 {
zap.S().Warn("UDP RELAY WAS DISABLED FOR NOW, THIS FEATURE WILL BE AVAILABLE IN THE FUTURE")
}
return nil
}
func (r *Config) Clone() *Config {
new := &Config{
Listen: r.Listen,
ListenType: r.ListenType,
TransportType: r.TransportType,
Label: r.Label,
}
new.TCPRemotes = make([]string, len(r.TCPRemotes))
copy(new.TCPRemotes, r.TCPRemotes)
new.UDPRemotes = make([]string, len(r.UDPRemotes))
copy(new.UDPRemotes, r.UDPRemotes)
return new
}
func (r *Config) Different(new *Config) bool {
if r.Listen != new.Listen ||
r.ListenType != new.ListenType ||
r.TransportType != new.TransportType ||
r.Label != new.Label {
return true
}
if len(r.TCPRemotes) != len(new.TCPRemotes) {
return true
}
for i, addr := range r.TCPRemotes {
if addr != new.TCPRemotes[i] {
return true
}
}
if len(r.UDPRemotes) != len(new.UDPRemotes) {
return true
}
for i, addr := range r.UDPRemotes {
if addr != new.UDPRemotes[i] {
return true
}
}
return false
}
// todo make this shorter and more readable
func (r *Config) DefaultLabel() string {
defaultLabel := fmt.Sprintf("<At=%s TCP-To=%s TP=%s>",
r.Listen, r.TCPRemotes, r.TransportType)
return defaultLabel
}
func (r *Config) Adjust() error {
if r.Label == "" {
r.Label = r.DefaultLabel()
zap.S().Debugf("label is empty, set default label:%s", r.Label)
}
return nil
}
func (r *Config) ToTCPRemotes() lb.RoundRobin {
tcpNodeList := make([]*lb.Node, len(r.TCPRemotes))
for idx, addr := range r.TCPRemotes {
tcpNodeList[idx] = &lb.Node{
Address: addr,
Label: fmt.Sprintf("%s-%s", r.Label, addr),
}
}
return lb.NewRoundRobin(tcpNodeList)
}
func (r *Config) GetLoggerName() string {
return fmt.Sprintf("%s(%s<->%s)", r.Label, r.ListenType, r.TransportType)
}