freedom.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. package conf
  2. import (
  3. "net"
  4. "strconv"
  5. "strings"
  6. v2net "github.com/xtls/xray-core/common/net"
  7. "github.com/xtls/xray-core/common/protocol"
  8. "github.com/xtls/xray-core/proxy/freedom"
  9. "google.golang.org/protobuf/proto"
  10. )
  11. type FreedomConfig struct {
  12. DomainStrategy string `json:"domainStrategy"`
  13. Timeout *uint32 `json:"timeout"`
  14. Redirect string `json:"redirect"`
  15. UserLevel uint32 `json:"userLevel"`
  16. Fragment *Fragment `json:"fragment"`
  17. }
  18. type Fragment struct {
  19. Packets string `json:"packets"`
  20. Length string `json:"length"`
  21. Interval string `json:"interval"`
  22. }
  23. // Build implements Buildable
  24. func (c *FreedomConfig) Build() (proto.Message, error) {
  25. config := new(freedom.Config)
  26. switch strings.ToLower(c.DomainStrategy) {
  27. case "asis", "":
  28. config.DomainStrategy = freedom.Config_AS_IS
  29. case "useip":
  30. config.DomainStrategy = freedom.Config_USE_IP
  31. case "useipv4":
  32. config.DomainStrategy = freedom.Config_USE_IP4
  33. case "useipv6":
  34. config.DomainStrategy = freedom.Config_USE_IP6
  35. case "useipv4v6":
  36. config.DomainStrategy = freedom.Config_USE_IP46
  37. case "useipv6v4":
  38. config.DomainStrategy = freedom.Config_USE_IP64
  39. case "forceip":
  40. config.DomainStrategy = freedom.Config_FORCE_IP
  41. case "forceipv4":
  42. config.DomainStrategy = freedom.Config_FORCE_IP4
  43. case "forceipv6":
  44. config.DomainStrategy = freedom.Config_FORCE_IP6
  45. case "forceipv4v6":
  46. config.DomainStrategy = freedom.Config_FORCE_IP46
  47. case "forceipv6v4":
  48. config.DomainStrategy = freedom.Config_FORCE_IP64
  49. default:
  50. return nil, newError("unsupported domain strategy: ", c.DomainStrategy)
  51. }
  52. if c.Fragment != nil {
  53. config.Fragment = new(freedom.Fragment)
  54. var err, err2 error
  55. switch strings.ToLower(c.Fragment.Packets) {
  56. case "tlshello":
  57. // TLS Hello Fragmentation (into multiple handshake messages)
  58. config.Fragment.PacketsFrom = 0
  59. config.Fragment.PacketsTo = 1
  60. case "":
  61. // TCP Segmentation (all packets)
  62. config.Fragment.PacketsFrom = 0
  63. config.Fragment.PacketsTo = 0
  64. default:
  65. // TCP Segmentation (range)
  66. packetsFromTo := strings.Split(c.Fragment.Packets, "-")
  67. if len(packetsFromTo) == 2 {
  68. config.Fragment.PacketsFrom, err = strconv.ParseUint(packetsFromTo[0], 10, 64)
  69. config.Fragment.PacketsTo, err2 = strconv.ParseUint(packetsFromTo[1], 10, 64)
  70. } else {
  71. config.Fragment.PacketsFrom, err = strconv.ParseUint(packetsFromTo[0], 10, 64)
  72. config.Fragment.PacketsTo = config.Fragment.PacketsFrom
  73. }
  74. if err != nil {
  75. return nil, newError("Invalid PacketsFrom").Base(err)
  76. }
  77. if err2 != nil {
  78. return nil, newError("Invalid PacketsTo").Base(err2)
  79. }
  80. if config.Fragment.PacketsFrom > config.Fragment.PacketsTo {
  81. config.Fragment.PacketsFrom, config.Fragment.PacketsTo = config.Fragment.PacketsTo, config.Fragment.PacketsFrom
  82. }
  83. if config.Fragment.PacketsFrom == 0 {
  84. return nil, newError("PacketsFrom can't be 0")
  85. }
  86. }
  87. {
  88. if c.Fragment.Length == "" {
  89. return nil, newError("Length can't be empty")
  90. }
  91. lengthMinMax := strings.Split(c.Fragment.Length, "-")
  92. if len(lengthMinMax) == 2 {
  93. config.Fragment.LengthMin, err = strconv.ParseUint(lengthMinMax[0], 10, 64)
  94. config.Fragment.LengthMax, err2 = strconv.ParseUint(lengthMinMax[1], 10, 64)
  95. } else {
  96. config.Fragment.LengthMin, err = strconv.ParseUint(lengthMinMax[0], 10, 64)
  97. config.Fragment.LengthMax = config.Fragment.LengthMin
  98. }
  99. if err != nil {
  100. return nil, newError("Invalid LengthMin").Base(err)
  101. }
  102. if err2 != nil {
  103. return nil, newError("Invalid LengthMax").Base(err2)
  104. }
  105. if config.Fragment.LengthMin > config.Fragment.LengthMax {
  106. config.Fragment.LengthMin, config.Fragment.LengthMax = config.Fragment.LengthMax, config.Fragment.LengthMin
  107. }
  108. if config.Fragment.LengthMin == 0 {
  109. return nil, newError("LengthMin can't be 0")
  110. }
  111. }
  112. {
  113. if c.Fragment.Interval == "" {
  114. return nil, newError("Interval can't be empty")
  115. }
  116. intervalMinMax := strings.Split(c.Fragment.Interval, "-")
  117. if len(intervalMinMax) == 2 {
  118. config.Fragment.IntervalMin, err = strconv.ParseUint(intervalMinMax[0], 10, 64)
  119. config.Fragment.IntervalMax, err2 = strconv.ParseUint(intervalMinMax[1], 10, 64)
  120. } else {
  121. config.Fragment.IntervalMin, err = strconv.ParseUint(intervalMinMax[0], 10, 64)
  122. config.Fragment.IntervalMax = config.Fragment.IntervalMin
  123. }
  124. if err != nil {
  125. return nil, newError("Invalid IntervalMin").Base(err)
  126. }
  127. if err2 != nil {
  128. return nil, newError("Invalid IntervalMax").Base(err2)
  129. }
  130. if config.Fragment.IntervalMin > config.Fragment.IntervalMax {
  131. config.Fragment.IntervalMin, config.Fragment.IntervalMax = config.Fragment.IntervalMax, config.Fragment.IntervalMin
  132. }
  133. }
  134. }
  135. if c.Timeout != nil {
  136. config.Timeout = *c.Timeout
  137. }
  138. config.UserLevel = c.UserLevel
  139. if len(c.Redirect) > 0 {
  140. host, portStr, err := net.SplitHostPort(c.Redirect)
  141. if err != nil {
  142. return nil, newError("invalid redirect address: ", c.Redirect, ": ", err).Base(err)
  143. }
  144. port, err := v2net.PortFromString(portStr)
  145. if err != nil {
  146. return nil, newError("invalid redirect port: ", c.Redirect, ": ", err).Base(err)
  147. }
  148. config.DestinationOverride = &freedom.DestinationOverride{
  149. Server: &protocol.ServerEndpoint{
  150. Port: uint32(port),
  151. },
  152. }
  153. if len(host) > 0 {
  154. config.DestinationOverride.Server.Address = v2net.NewIPOrDomain(v2net.ParseAddress(host))
  155. }
  156. }
  157. return config, nil
  158. }