2020-11-25 14:01:53 +03:00
|
|
|
package conf
|
|
|
|
|
|
|
|
import (
|
2024-09-16 15:46:30 +03:00
|
|
|
"encoding/base64"
|
2025-01-02 12:17:03 +03:00
|
|
|
"encoding/hex"
|
2020-11-25 14:01:53 +03:00
|
|
|
"net"
|
|
|
|
"strings"
|
|
|
|
|
2024-06-29 21:32:57 +03:00
|
|
|
"github.com/xtls/xray-core/common/errors"
|
2020-12-04 04:36:16 +03:00
|
|
|
v2net "github.com/xtls/xray-core/common/net"
|
|
|
|
"github.com/xtls/xray-core/common/protocol"
|
|
|
|
"github.com/xtls/xray-core/proxy/freedom"
|
2023-08-10 07:43:34 +03:00
|
|
|
"google.golang.org/protobuf/proto"
|
2020-11-25 14:01:53 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type FreedomConfig struct {
|
2023-05-22 05:59:58 +03:00
|
|
|
DomainStrategy string `json:"domainStrategy"`
|
|
|
|
Redirect string `json:"redirect"`
|
|
|
|
UserLevel uint32 `json:"userLevel"`
|
|
|
|
Fragment *Fragment `json:"fragment"`
|
2024-08-29 00:10:11 +03:00
|
|
|
Noise *Noise `json:"noise"`
|
2024-09-16 15:46:30 +03:00
|
|
|
Noises []*Noise `json:"noises"`
|
2024-02-03 04:10:23 +03:00
|
|
|
ProxyProtocol uint32 `json:"proxyProtocol"`
|
2023-05-22 05:59:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type Fragment struct {
|
2024-10-15 07:05:22 +03:00
|
|
|
Packets string `json:"packets"`
|
|
|
|
Length *Int32Range `json:"length"`
|
|
|
|
Interval *Int32Range `json:"interval"`
|
2020-11-25 14:01:53 +03:00
|
|
|
}
|
|
|
|
|
2024-08-29 00:10:11 +03:00
|
|
|
type Noise struct {
|
2024-09-16 15:46:30 +03:00
|
|
|
Type string `json:"type"`
|
|
|
|
Packet string `json:"packet"`
|
|
|
|
Delay *Int32Range `json:"delay"`
|
2024-08-29 00:10:11 +03:00
|
|
|
}
|
|
|
|
|
2020-11-25 14:01:53 +03:00
|
|
|
// Build implements Buildable
|
|
|
|
func (c *FreedomConfig) Build() (proto.Message, error) {
|
|
|
|
config := new(freedom.Config)
|
|
|
|
switch strings.ToLower(c.DomainStrategy) {
|
2023-11-13 00:27:39 +03:00
|
|
|
case "asis", "":
|
|
|
|
config.DomainStrategy = freedom.Config_AS_IS
|
|
|
|
case "useip":
|
2020-11-25 14:01:53 +03:00
|
|
|
config.DomainStrategy = freedom.Config_USE_IP
|
2023-11-13 00:27:39 +03:00
|
|
|
case "useipv4":
|
2020-11-25 14:01:53 +03:00
|
|
|
config.DomainStrategy = freedom.Config_USE_IP4
|
2023-11-13 00:27:39 +03:00
|
|
|
case "useipv6":
|
2020-11-25 14:01:53 +03:00
|
|
|
config.DomainStrategy = freedom.Config_USE_IP6
|
2023-11-13 00:27:39 +03:00
|
|
|
case "useipv4v6":
|
|
|
|
config.DomainStrategy = freedom.Config_USE_IP46
|
|
|
|
case "useipv6v4":
|
|
|
|
config.DomainStrategy = freedom.Config_USE_IP64
|
|
|
|
case "forceip":
|
|
|
|
config.DomainStrategy = freedom.Config_FORCE_IP
|
|
|
|
case "forceipv4":
|
|
|
|
config.DomainStrategy = freedom.Config_FORCE_IP4
|
|
|
|
case "forceipv6":
|
|
|
|
config.DomainStrategy = freedom.Config_FORCE_IP6
|
|
|
|
case "forceipv4v6":
|
|
|
|
config.DomainStrategy = freedom.Config_FORCE_IP46
|
|
|
|
case "forceipv6v4":
|
|
|
|
config.DomainStrategy = freedom.Config_FORCE_IP64
|
|
|
|
default:
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("unsupported domain strategy: ", c.DomainStrategy)
|
2020-11-25 14:01:53 +03:00
|
|
|
}
|
|
|
|
|
2023-05-22 05:59:58 +03:00
|
|
|
if c.Fragment != nil {
|
2023-07-06 19:30:39 +03:00
|
|
|
config.Fragment = new(freedom.Fragment)
|
2023-05-22 05:59:58 +03:00
|
|
|
|
2023-06-18 17:12:42 +03:00
|
|
|
switch strings.ToLower(c.Fragment.Packets) {
|
|
|
|
case "tlshello":
|
|
|
|
// TLS Hello Fragmentation (into multiple handshake messages)
|
2023-07-06 19:30:39 +03:00
|
|
|
config.Fragment.PacketsFrom = 0
|
|
|
|
config.Fragment.PacketsTo = 1
|
2023-06-18 17:12:42 +03:00
|
|
|
case "":
|
|
|
|
// TCP Segmentation (all packets)
|
2023-07-06 19:30:39 +03:00
|
|
|
config.Fragment.PacketsFrom = 0
|
|
|
|
config.Fragment.PacketsTo = 0
|
2023-06-18 17:12:42 +03:00
|
|
|
default:
|
|
|
|
// TCP Segmentation (range)
|
2024-10-15 07:05:22 +03:00
|
|
|
from, to, err := ParseRangeString(c.Fragment.Packets)
|
2023-05-22 05:59:58 +03:00
|
|
|
if err != nil {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("Invalid PacketsFrom").Base(err)
|
2023-05-22 05:59:58 +03:00
|
|
|
}
|
2024-10-15 07:05:22 +03:00
|
|
|
config.Fragment.PacketsFrom = uint64(from)
|
|
|
|
config.Fragment.PacketsTo = uint64(to)
|
2023-07-06 19:30:39 +03:00
|
|
|
if config.Fragment.PacketsFrom == 0 {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("PacketsFrom can't be 0")
|
2023-07-06 19:30:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2024-10-15 07:05:22 +03:00
|
|
|
if c.Fragment.Length == nil {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("Length can't be empty")
|
2023-07-06 19:30:39 +03:00
|
|
|
}
|
2024-10-15 07:05:22 +03:00
|
|
|
config.Fragment.LengthMin = uint64(c.Fragment.Length.From)
|
|
|
|
config.Fragment.LengthMax = uint64(c.Fragment.Length.To)
|
2023-07-06 19:30:39 +03:00
|
|
|
if config.Fragment.LengthMin == 0 {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("LengthMin can't be 0")
|
2023-07-06 19:30:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2024-10-15 07:05:22 +03:00
|
|
|
if c.Fragment.Interval == nil {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("Interval can't be empty")
|
2023-07-06 19:30:39 +03:00
|
|
|
}
|
2024-10-15 07:05:22 +03:00
|
|
|
config.Fragment.IntervalMin = uint64(c.Fragment.Interval.From)
|
|
|
|
config.Fragment.IntervalMax = uint64(c.Fragment.Interval.To)
|
2023-05-22 05:59:58 +03:00
|
|
|
}
|
|
|
|
}
|
2024-08-29 00:10:11 +03:00
|
|
|
|
2024-09-16 15:46:30 +03:00
|
|
|
if c.Noise != nil {
|
|
|
|
return nil, errors.PrintRemovedFeatureError("noise = { ... }", "noises = [ { ... } ]")
|
|
|
|
}
|
2024-08-29 00:10:11 +03:00
|
|
|
|
2024-09-16 15:46:30 +03:00
|
|
|
if c.Noises != nil {
|
|
|
|
for _, n := range c.Noises {
|
|
|
|
NConfig, err := ParseNoise(n)
|
2024-08-29 00:10:11 +03:00
|
|
|
if err != nil {
|
2024-09-16 15:46:30 +03:00
|
|
|
return nil, err
|
2024-08-29 00:10:11 +03:00
|
|
|
}
|
2024-09-16 15:46:30 +03:00
|
|
|
config.Noises = append(config.Noises, NConfig)
|
2024-08-29 00:10:11 +03:00
|
|
|
}
|
|
|
|
}
|
2023-05-22 05:59:58 +03:00
|
|
|
|
2020-11-25 14:01:53 +03:00
|
|
|
config.UserLevel = c.UserLevel
|
|
|
|
if len(c.Redirect) > 0 {
|
|
|
|
host, portStr, err := net.SplitHostPort(c.Redirect)
|
|
|
|
if err != nil {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("invalid redirect address: ", c.Redirect, ": ", err).Base(err)
|
2020-11-25 14:01:53 +03:00
|
|
|
}
|
|
|
|
port, err := v2net.PortFromString(portStr)
|
|
|
|
if err != nil {
|
2024-06-29 21:32:57 +03:00
|
|
|
return nil, errors.New("invalid redirect port: ", c.Redirect, ": ", err).Base(err)
|
2020-11-25 14:01:53 +03:00
|
|
|
}
|
|
|
|
config.DestinationOverride = &freedom.DestinationOverride{
|
|
|
|
Server: &protocol.ServerEndpoint{
|
|
|
|
Port: uint32(port),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(host) > 0 {
|
|
|
|
config.DestinationOverride.Server.Address = v2net.NewIPOrDomain(v2net.ParseAddress(host))
|
|
|
|
}
|
|
|
|
}
|
2024-02-03 04:10:23 +03:00
|
|
|
if c.ProxyProtocol > 0 && c.ProxyProtocol <= 2 {
|
|
|
|
config.ProxyProtocol = c.ProxyProtocol
|
|
|
|
}
|
2020-11-25 14:01:53 +03:00
|
|
|
return config, nil
|
|
|
|
}
|
2024-09-16 15:46:30 +03:00
|
|
|
|
|
|
|
func ParseNoise(noise *Noise) (*freedom.Noise, error) {
|
2024-10-15 07:05:22 +03:00
|
|
|
var err error
|
2024-09-16 15:46:30 +03:00
|
|
|
NConfig := new(freedom.Noise)
|
2025-01-02 12:45:46 +03:00
|
|
|
noise.Packet = strings.TrimSpace(noise.Packet)
|
2024-09-16 15:46:30 +03:00
|
|
|
|
2025-01-02 12:45:46 +03:00
|
|
|
switch noise.Type {
|
2024-09-16 15:46:30 +03:00
|
|
|
case "rand":
|
2024-10-15 07:05:22 +03:00
|
|
|
min, max, err := ParseRangeString(noise.Packet)
|
2024-09-16 15:46:30 +03:00
|
|
|
if err != nil {
|
2024-10-15 07:05:22 +03:00
|
|
|
return nil, errors.New("invalid value for rand Length").Base(err)
|
2024-09-16 15:46:30 +03:00
|
|
|
}
|
2024-10-15 07:05:22 +03:00
|
|
|
NConfig.LengthMin = uint64(min)
|
|
|
|
NConfig.LengthMax = uint64(max)
|
2024-09-16 15:46:30 +03:00
|
|
|
if NConfig.LengthMin == 0 {
|
|
|
|
return nil, errors.New("rand lengthMin or lengthMax cannot be 0")
|
|
|
|
}
|
|
|
|
|
|
|
|
case "str":
|
2025-01-02 12:17:03 +03:00
|
|
|
// user input string
|
2025-01-02 12:45:46 +03:00
|
|
|
NConfig.Packet = []byte(noise.Packet)
|
2024-09-16 15:46:30 +03:00
|
|
|
|
2025-01-02 12:17:03 +03:00
|
|
|
case "hex":
|
|
|
|
// user input hex
|
|
|
|
NConfig.Packet, err = hex.DecodeString(noise.Packet)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Invalid hex string").Base(err)
|
|
|
|
}
|
|
|
|
|
2024-09-16 15:46:30 +03:00
|
|
|
case "base64":
|
2025-01-02 12:17:03 +03:00
|
|
|
// user input base64
|
2025-01-02 12:45:46 +03:00
|
|
|
NConfig.Packet, err = base64.RawURLEncoding.DecodeString(strings.NewReplacer("+", "-", "/", "_", "=", "").Replace(noise.Packet))
|
2024-09-16 15:46:30 +03:00
|
|
|
if err != nil {
|
2025-01-02 12:17:03 +03:00
|
|
|
return nil, errors.New("Invalid base64 string").Base(err)
|
2024-09-16 15:46:30 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
2025-01-02 12:17:03 +03:00
|
|
|
return nil, errors.New("Invalid packet, only rand/str/hex/base64 are supported")
|
2024-09-16 15:46:30 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if noise.Delay != nil {
|
2025-01-01 13:22:57 +03:00
|
|
|
NConfig.DelayMin = uint64(noise.Delay.From)
|
|
|
|
NConfig.DelayMax = uint64(noise.Delay.To)
|
2024-09-16 15:46:30 +03:00
|
|
|
}
|
|
|
|
return NConfig, nil
|
|
|
|
}
|