| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325 |
- 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)
- }
|