mirror of https://github.com/XTLS/Xray-core
326 lines
7.7 KiB
Go
326 lines
7.7 KiB
Go
package conf
|
|
|
|
import (
|
|
"encoding/json"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/xtls/xray-core/common/errors"
|
|
"github.com/xtls/xray-core/common/net"
|
|
"github.com/xtls/xray-core/common/platform"
|
|
"github.com/xtls/xray-core/common/protocol"
|
|
)
|
|
|
|
type StringList []string
|
|
|
|
func NewStringList(raw []string) *StringList {
|
|
list := StringList(raw)
|
|
return &list
|
|
}
|
|
|
|
func (v StringList) Len() int {
|
|
return len(v)
|
|
}
|
|
|
|
func (v *StringList) UnmarshalJSON(data []byte) error {
|
|
var strarray []string
|
|
if err := json.Unmarshal(data, &strarray); err == nil {
|
|
*v = *NewStringList(strarray)
|
|
return nil
|
|
}
|
|
|
|
var rawstr string
|
|
if err := json.Unmarshal(data, &rawstr); err == nil {
|
|
strlist := strings.Split(rawstr, ",")
|
|
*v = *NewStringList(strlist)
|
|
return nil
|
|
}
|
|
return errors.New("unknown format of a string list: " + string(data))
|
|
}
|
|
|
|
type Address struct {
|
|
net.Address
|
|
}
|
|
|
|
func (v Address) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(v.Address.String())
|
|
}
|
|
|
|
func (v *Address) UnmarshalJSON(data []byte) error {
|
|
var rawStr string
|
|
if err := json.Unmarshal(data, &rawStr); err != nil {
|
|
return errors.New("invalid address: ", string(data)).Base(err)
|
|
}
|
|
if strings.HasPrefix(rawStr, "env:") {
|
|
rawStr = platform.NewEnvFlag(rawStr[4:]).GetValue(func() string { return "" })
|
|
}
|
|
v.Address = net.ParseAddress(rawStr)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (v *Address) Build() *net.IPOrDomain {
|
|
return net.NewIPOrDomain(v.Address)
|
|
}
|
|
|
|
type Network string
|
|
|
|
func (v Network) Build() net.Network {
|
|
switch strings.ToLower(string(v)) {
|
|
case "tcp":
|
|
return net.Network_TCP
|
|
case "udp":
|
|
return net.Network_UDP
|
|
case "unix":
|
|
return net.Network_UNIX
|
|
default:
|
|
return net.Network_Unknown
|
|
}
|
|
}
|
|
|
|
type NetworkList []Network
|
|
|
|
func (v *NetworkList) UnmarshalJSON(data []byte) error {
|
|
var strarray []Network
|
|
if err := json.Unmarshal(data, &strarray); err == nil {
|
|
nl := NetworkList(strarray)
|
|
*v = nl
|
|
return nil
|
|
}
|
|
|
|
var rawstr Network
|
|
if err := json.Unmarshal(data, &rawstr); err == nil {
|
|
strlist := strings.Split(string(rawstr), ",")
|
|
nl := make([]Network, len(strlist))
|
|
for idx, network := range strlist {
|
|
nl[idx] = Network(network)
|
|
}
|
|
*v = nl
|
|
return nil
|
|
}
|
|
return errors.New("unknown format of a string list: " + string(data))
|
|
}
|
|
|
|
func (v *NetworkList) Build() []net.Network {
|
|
if v == nil {
|
|
return []net.Network{net.Network_TCP}
|
|
}
|
|
|
|
list := make([]net.Network, 0, len(*v))
|
|
for _, network := range *v {
|
|
list = append(list, network.Build())
|
|
}
|
|
return list
|
|
}
|
|
|
|
func parseIntPort(data []byte) (net.Port, error) {
|
|
var intPort uint32
|
|
err := json.Unmarshal(data, &intPort)
|
|
if err != nil {
|
|
return net.Port(0), err
|
|
}
|
|
return net.PortFromInt(intPort)
|
|
}
|
|
|
|
func parseStringPort(s string) (net.Port, net.Port, error) {
|
|
if strings.HasPrefix(s, "env:") {
|
|
s = platform.NewEnvFlag(s[4:]).GetValue(func() string { return "" })
|
|
}
|
|
|
|
pair := strings.SplitN(s, "-", 2)
|
|
if len(pair) == 0 {
|
|
return net.Port(0), net.Port(0), errors.New("invalid port range: ", s)
|
|
}
|
|
if len(pair) == 1 {
|
|
port, err := net.PortFromString(pair[0])
|
|
return port, port, err
|
|
}
|
|
|
|
fromPort, err := net.PortFromString(pair[0])
|
|
if err != nil {
|
|
return net.Port(0), net.Port(0), err
|
|
}
|
|
toPort, err := net.PortFromString(pair[1])
|
|
if err != nil {
|
|
return net.Port(0), net.Port(0), err
|
|
}
|
|
return fromPort, toPort, nil
|
|
}
|
|
|
|
func parseJSONStringPort(data []byte) (net.Port, net.Port, error) {
|
|
var s string
|
|
err := json.Unmarshal(data, &s)
|
|
if err != nil {
|
|
return net.Port(0), net.Port(0), err
|
|
}
|
|
return parseStringPort(s)
|
|
}
|
|
|
|
type PortRange struct {
|
|
From uint32
|
|
To uint32
|
|
}
|
|
|
|
func (v *PortRange) Build() *net.PortRange {
|
|
return &net.PortRange{
|
|
From: v.From,
|
|
To: v.To,
|
|
}
|
|
}
|
|
|
|
// UnmarshalJSON implements encoding/json.Unmarshaler.UnmarshalJSON
|
|
func (v *PortRange) UnmarshalJSON(data []byte) error {
|
|
port, err := parseIntPort(data)
|
|
if err == nil {
|
|
v.From = uint32(port)
|
|
v.To = uint32(port)
|
|
return nil
|
|
}
|
|
|
|
from, to, err := parseJSONStringPort(data)
|
|
if err == nil {
|
|
v.From = uint32(from)
|
|
v.To = uint32(to)
|
|
if v.From > v.To {
|
|
return errors.New("invalid port range ", v.From, " -> ", v.To)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
return errors.New("invalid port range: ", string(data))
|
|
}
|
|
|
|
type PortList struct {
|
|
Range []PortRange
|
|
}
|
|
|
|
func (list *PortList) Build() *net.PortList {
|
|
portList := new(net.PortList)
|
|
for _, r := range list.Range {
|
|
portList.Range = append(portList.Range, r.Build())
|
|
}
|
|
return portList
|
|
}
|
|
|
|
// UnmarshalJSON implements encoding/json.Unmarshaler.UnmarshalJSON
|
|
func (list *PortList) UnmarshalJSON(data []byte) error {
|
|
var listStr string
|
|
var number uint32
|
|
if err := json.Unmarshal(data, &listStr); err != nil {
|
|
if err2 := json.Unmarshal(data, &number); err2 != nil {
|
|
return errors.New("invalid port: ", string(data)).Base(err2)
|
|
}
|
|
}
|
|
rangelist := strings.Split(listStr, ",")
|
|
for _, rangeStr := range rangelist {
|
|
trimmed := strings.TrimSpace(rangeStr)
|
|
if len(trimmed) > 0 {
|
|
if strings.Contains(trimmed, "-") || strings.Contains(trimmed, "env:") {
|
|
from, to, err := parseStringPort(trimmed)
|
|
if err != nil {
|
|
return errors.New("invalid port range: ", trimmed).Base(err)
|
|
}
|
|
list.Range = append(list.Range, PortRange{From: uint32(from), To: uint32(to)})
|
|
} else {
|
|
port, err := parseIntPort([]byte(trimmed))
|
|
if err != nil {
|
|
return errors.New("invalid port: ", trimmed).Base(err)
|
|
}
|
|
list.Range = append(list.Range, PortRange{From: uint32(port), To: uint32(port)})
|
|
}
|
|
}
|
|
}
|
|
if number != 0 {
|
|
list.Range = append(list.Range, PortRange{From: number, To: number})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type User struct {
|
|
EmailString string `json:"email"`
|
|
LevelByte byte `json:"level"`
|
|
}
|
|
|
|
func (v *User) Build() *protocol.User {
|
|
return &protocol.User{
|
|
Email: v.EmailString,
|
|
Level: uint32(v.LevelByte),
|
|
}
|
|
}
|
|
|
|
// Int32Range deserializes from "1-2" or 1, so can deserialize from both int and number.
|
|
// Negative integers can be passed as sentinel values, but do not parse as ranges.
|
|
// Value will be exchanged if From > To, use .Left and .Right to get original value if need.
|
|
type Int32Range struct {
|
|
Left int32
|
|
Right int32
|
|
From int32
|
|
To int32
|
|
}
|
|
|
|
func (v *Int32Range) UnmarshalJSON(data []byte) error {
|
|
defer v.ensureOrder()
|
|
var str string
|
|
var rawint int32
|
|
if err := json.Unmarshal(data, &str); err == nil {
|
|
left, right, err := ParseRangeString(str)
|
|
if err == nil {
|
|
v.Left, v.Right = int32(left), int32(right)
|
|
return nil
|
|
}
|
|
} else if err := json.Unmarshal(data, &rawint); err == nil {
|
|
v.Left = rawint
|
|
v.Right = rawint
|
|
return nil
|
|
}
|
|
|
|
return errors.New("Invalid integer range, expected either string of form \"1-2\" or plain integer.")
|
|
}
|
|
|
|
// ensureOrder() gives value to .From & .To and make sure .From < .To
|
|
func (r *Int32Range) ensureOrder() {
|
|
r.From, r.To = r.Left, r.Right
|
|
if r.From > r.To {
|
|
r.From, r.To = r.To, r.From
|
|
}
|
|
}
|
|
|
|
// "-114-514" → ["-114","514"]
|
|
// "-1919--810" → ["-1919","-810"]
|
|
func splitFromSecondDash(s string) []string {
|
|
parts := strings.SplitN(s, "-", 3)
|
|
if len(parts) < 3 {
|
|
return []string{s}
|
|
}
|
|
return []string{parts[0] + "-" + parts[1], parts[2]}
|
|
}
|
|
|
|
// Parse rang in string. Support negative number.
|
|
// eg: "114-514" "-114-514" "-1919--810" "114514" ""(return 0)
|
|
func ParseRangeString(str string) (int, int, error) {
|
|
// for number in string format like "114" or "-1"
|
|
if value, err := strconv.Atoi(str); err == nil {
|
|
return value, value, nil
|
|
}
|
|
// for empty "", we treat it as 0
|
|
if str == "" {
|
|
return 0, 0, nil
|
|
}
|
|
// for range value, like "114-514"
|
|
var pair []string
|
|
// Process sth like "-114-514" "-1919--810"
|
|
if strings.HasPrefix(str, "-") {
|
|
pair = splitFromSecondDash(str)
|
|
} else {
|
|
pair = strings.SplitN(str, "-", 2)
|
|
}
|
|
if len(pair) == 2 {
|
|
left, err := strconv.Atoi(pair[0])
|
|
right, err2 := strconv.Atoi(pair[1])
|
|
if err == nil && err2 == nil {
|
|
return left, right, nil
|
|
}
|
|
}
|
|
return 0, 0, errors.New("invalid range string: ", str)
|
|
}
|