123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- package v1
- import (
- "bytes"
- "encoding/json"
- "errors"
- "fmt"
- "reflect"
- "github.com/samber/lo"
- "github.com/fatedier/frp/pkg/util/util"
- )
- type VisitorTransport struct {
- UseEncryption bool `json:"useEncryption,omitempty"`
- UseCompression bool `json:"useCompression,omitempty"`
- }
- type VisitorBaseConfig struct {
- Name string `json:"name"`
- Type string `json:"type"`
- Transport VisitorTransport `json:"transport,omitempty"`
- SecretKey string `json:"secretKey,omitempty"`
-
- ServerUser string `json:"serverUser,omitempty"`
- ServerName string `json:"serverName,omitempty"`
- BindAddr string `json:"bindAddr,omitempty"`
-
-
-
- BindPort int `json:"bindPort,omitempty"`
- }
- func (c *VisitorBaseConfig) GetBaseConfig() *VisitorBaseConfig {
- return c
- }
- func (c *VisitorBaseConfig) Complete(g *ClientCommonConfig) {
- if c.BindAddr == "" {
- c.BindAddr = "127.0.0.1"
- }
- namePrefix := ""
- if g.User != "" {
- namePrefix = g.User + "."
- }
- c.Name = namePrefix + c.Name
- if c.ServerUser != "" {
- c.ServerName = c.ServerUser + "." + c.ServerName
- } else {
- c.ServerName = namePrefix + c.ServerName
- }
- }
- type VisitorConfigurer interface {
- Complete(*ClientCommonConfig)
- GetBaseConfig() *VisitorBaseConfig
- }
- type VisitorType string
- const (
- VisitorTypeSTCP VisitorType = "stcp"
- VisitorTypeXTCP VisitorType = "xtcp"
- VisitorTypeSUDP VisitorType = "sudp"
- )
- var visitorConfigTypeMap = map[VisitorType]reflect.Type{
- VisitorTypeSTCP: reflect.TypeOf(STCPVisitorConfig{}),
- VisitorTypeXTCP: reflect.TypeOf(XTCPVisitorConfig{}),
- VisitorTypeSUDP: reflect.TypeOf(SUDPVisitorConfig{}),
- }
- type TypedVisitorConfig struct {
- Type string `json:"type"`
- VisitorConfigurer
- }
- func (c *TypedVisitorConfig) UnmarshalJSON(b []byte) error {
- if len(b) == 4 && string(b) == "null" {
- return errors.New("type is required")
- }
- typeStruct := struct {
- Type string `json:"type"`
- }{}
- if err := json.Unmarshal(b, &typeStruct); err != nil {
- return err
- }
- c.Type = typeStruct.Type
- configurer := NewVisitorConfigurerByType(VisitorType(typeStruct.Type))
- if configurer == nil {
- return fmt.Errorf("unknown visitor type: %s", typeStruct.Type)
- }
- decoder := json.NewDecoder(bytes.NewBuffer(b))
- if DisallowUnknownFields {
- decoder.DisallowUnknownFields()
- }
- if err := decoder.Decode(configurer); err != nil {
- return fmt.Errorf("unmarshal VisitorConfig error: %v", err)
- }
- c.VisitorConfigurer = configurer
- return nil
- }
- func (c *TypedVisitorConfig) MarshalJSON() ([]byte, error) {
- return json.Marshal(c.VisitorConfigurer)
- }
- func NewVisitorConfigurerByType(t VisitorType) VisitorConfigurer {
- v, ok := visitorConfigTypeMap[t]
- if !ok {
- return nil
- }
- vc := reflect.New(v).Interface().(VisitorConfigurer)
- vc.GetBaseConfig().Type = string(t)
- return vc
- }
- var _ VisitorConfigurer = &STCPVisitorConfig{}
- type STCPVisitorConfig struct {
- VisitorBaseConfig
- }
- var _ VisitorConfigurer = &SUDPVisitorConfig{}
- type SUDPVisitorConfig struct {
- VisitorBaseConfig
- }
- var _ VisitorConfigurer = &XTCPVisitorConfig{}
- type XTCPVisitorConfig struct {
- VisitorBaseConfig
- Protocol string `json:"protocol,omitempty"`
- KeepTunnelOpen bool `json:"keepTunnelOpen,omitempty"`
- MaxRetriesAnHour int `json:"maxRetriesAnHour,omitempty"`
- MinRetryInterval int `json:"minRetryInterval,omitempty"`
- FallbackTo string `json:"fallbackTo,omitempty"`
- FallbackTimeoutMs int `json:"fallbackTimeoutMs,omitempty"`
- }
- func (c *XTCPVisitorConfig) Complete(g *ClientCommonConfig) {
- c.VisitorBaseConfig.Complete(g)
- c.Protocol = util.EmptyOr(c.Protocol, "quic")
- c.MaxRetriesAnHour = util.EmptyOr(c.MaxRetriesAnHour, 8)
- c.MinRetryInterval = util.EmptyOr(c.MinRetryInterval, 90)
- c.FallbackTimeoutMs = util.EmptyOr(c.FallbackTimeoutMs, 1000)
- if c.FallbackTo != "" {
- c.FallbackTo = lo.Ternary(g.User == "", "", g.User+".") + c.FallbackTo
- }
- }
|