vmess.go 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. package proxy
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "math/rand"
  7. "net"
  8. "net/url"
  9. "regexp"
  10. "strconv"
  11. "strings"
  12. "github.com/zu1k/proxypool/pkg/tool"
  13. )
  14. var (
  15. ErrorNotVmessLink = errors.New("not a correct vmess link")
  16. ErrorVmessPayloadParseFail = errors.New("vmess link payload parse failed")
  17. )
  18. type Vmess struct {
  19. Base
  20. UUID string `yaml:"uuid" json:"uuid"`
  21. AlterID int `yaml:"alterId" json:"alterId"`
  22. Cipher string `yaml:"cipher" json:"cipher"`
  23. TLS bool `yaml:"tls,omitempty" json:"tls,omitempty"`
  24. Network string `yaml:"network,omitempty" json:"network,omitempty"`
  25. HTTPOpts HTTPOptions `yaml:"http-opts,omitempty" json:"http-opts,omitempty"`
  26. WSPath string `yaml:"ws-path,omitempty" json:"ws-path,omitempty"`
  27. WSHeaders map[string]string `yaml:"ws-headers,omitempty" json:"ws-headers,omitempty"`
  28. SkipCertVerify bool `yaml:"skip-cert-verify,omitempty" json:"skip-cert-verify,omitempty"`
  29. ServerName string `yaml:"servername,omitempty" json:"servername,omitempty"`
  30. }
  31. type HTTPOptions struct {
  32. Method string `yaml:"method,omitempty" json:"method,omitempty"`
  33. Path []string `yaml:"path,omitempty" json:"path,omitempty"`
  34. Headers map[string][]string `yaml:"headers,omitempty" json:"headers,omitempty"`
  35. }
  36. func (v Vmess) Identifier() string {
  37. return net.JoinHostPort(v.Server, strconv.Itoa(v.Port)) + v.Cipher + v.UUID
  38. }
  39. func (v Vmess) String() string {
  40. data, err := json.Marshal(v)
  41. if err != nil {
  42. return ""
  43. }
  44. return string(data)
  45. }
  46. func (v Vmess) ToClash() string {
  47. data, err := json.Marshal(v)
  48. if err != nil {
  49. return ""
  50. }
  51. return "- " + string(data)
  52. }
  53. func (v Vmess) ToSurge() string {
  54. // node2 = vmess, server, port, username=, ws=true, ws-path=, ws-headers=
  55. if v.Network == "ws" {
  56. wsHeasers := ""
  57. for k, v := range v.WSHeaders {
  58. if wsHeasers == "" {
  59. wsHeasers = k + ":" + v
  60. } else {
  61. wsHeasers += "|" + k + ":" + v
  62. }
  63. }
  64. text := fmt.Sprintf("%s = vmess, %s, %d, username=%s, ws=true, tls=%t, ws-path=%s",
  65. v.Name, v.Server, v.Port, v.UUID, v.TLS, v.WSPath)
  66. if wsHeasers != "" {
  67. text += ", ws-headers=" + wsHeasers
  68. }
  69. return text
  70. } else {
  71. return fmt.Sprintf("%s = vmess, %s, %d, username=%s, tls=%t",
  72. v.Name, v.Server, v.Port, v.UUID, v.TLS)
  73. }
  74. }
  75. func (v Vmess) Clone() Proxy {
  76. return &v
  77. }
  78. type vmessLinkJson struct {
  79. Add string `json:"add"`
  80. V string `json:"v"`
  81. Ps string `json:"ps"`
  82. Port interface{} `json:"port"`
  83. Id string `json:"id"`
  84. Aid string `json:"aid"`
  85. Net string `json:"net"`
  86. Type string `json:"type"`
  87. Host string `json:"host"`
  88. Path string `json:"path"`
  89. Tls string `json:"tls"`
  90. }
  91. func ParseVmessLink(link string) (*Vmess, error) {
  92. if !strings.HasPrefix(link, "vmess") {
  93. return nil, ErrorNotVmessLink
  94. }
  95. vmessmix := strings.SplitN(link, "://", 2)
  96. if len(vmessmix) < 2 {
  97. return nil, ErrorNotVmessLink
  98. }
  99. linkPayload := vmessmix[1]
  100. if strings.Contains(linkPayload, "?") {
  101. // 使用第二种解析方法
  102. var infoPayloads []string
  103. if strings.Contains(linkPayload, "/?") {
  104. infoPayloads = strings.SplitN(linkPayload, "/?", 2)
  105. } else {
  106. infoPayloads = strings.SplitN(linkPayload, "?", 2)
  107. }
  108. if len(infoPayloads) < 2 {
  109. return nil, ErrorNotVmessLink
  110. }
  111. baseInfo, err := tool.Base64DecodeString(infoPayloads[0])
  112. if err != nil {
  113. return nil, ErrorVmessPayloadParseFail
  114. }
  115. baseInfoPath := strings.Split(baseInfo, ":")
  116. if len(baseInfoPath) < 3 {
  117. return nil, ErrorPathNotComplete
  118. }
  119. // base info
  120. cipher := baseInfoPath[0]
  121. mixInfo := strings.SplitN(baseInfoPath[1], "@", 2)
  122. if len(mixInfo) < 2 {
  123. return nil, ErrorVmessPayloadParseFail
  124. }
  125. uuid := mixInfo[0]
  126. server := mixInfo[1]
  127. portStr := baseInfoPath[2]
  128. port, err := strconv.Atoi(portStr)
  129. if err != nil {
  130. return nil, ErrorVmessPayloadParseFail
  131. }
  132. moreInfo, _ := url.ParseQuery(infoPayloads[1])
  133. remarks := moreInfo.Get("remarks")
  134. obfs := moreInfo.Get("obfs")
  135. network := "tcp"
  136. if obfs == "websocket" {
  137. network = "ws"
  138. }
  139. //obfsParam := moreInfo.Get("obfsParam")
  140. path := moreInfo.Get("path")
  141. if path == "" {
  142. path = "/"
  143. }
  144. tls := moreInfo.Get("tls") == "1"
  145. wsHeaders := make(map[string]string)
  146. return &Vmess{
  147. Base: Base{
  148. Name: remarks + "_" + strconv.Itoa(rand.Int()),
  149. Server: server,
  150. Port: port,
  151. Type: "vmess",
  152. UDP: false,
  153. },
  154. UUID: uuid,
  155. AlterID: 0,
  156. Cipher: cipher,
  157. TLS: tls,
  158. Network: network,
  159. HTTPOpts: HTTPOptions{},
  160. WSPath: path,
  161. WSHeaders: wsHeaders,
  162. SkipCertVerify: true,
  163. ServerName: server,
  164. }, nil
  165. } else {
  166. payload, err := tool.Base64DecodeString(linkPayload)
  167. if err != nil {
  168. return nil, ErrorVmessPayloadParseFail
  169. }
  170. vmessJson := vmessLinkJson{}
  171. err = json.Unmarshal([]byte(payload), &vmessJson)
  172. if err != nil {
  173. return nil, err
  174. }
  175. port := 443
  176. portInterface := vmessJson.Port
  177. if i, ok := portInterface.(int); ok {
  178. port = i
  179. } else if s, ok := portInterface.(string); ok {
  180. port, _ = strconv.Atoi(s)
  181. }
  182. alterId, err := strconv.Atoi(vmessJson.Aid)
  183. if err != nil {
  184. alterId = 0
  185. }
  186. tls := vmessJson.Tls == "tls"
  187. wsHeaders := make(map[string]string)
  188. if vmessJson.Host != "" {
  189. wsHeaders["HOST"] = vmessJson.Host
  190. }
  191. if vmessJson.Path == "" {
  192. vmessJson.Path = "/"
  193. }
  194. return &Vmess{
  195. Base: Base{
  196. Name: vmessJson.Ps + "_" + strconv.Itoa(rand.Int()),
  197. Server: vmessJson.Add,
  198. Port: port,
  199. Type: "vmess",
  200. UDP: false,
  201. },
  202. UUID: vmessJson.Id,
  203. AlterID: alterId,
  204. Cipher: "auto",
  205. TLS: tls,
  206. Network: vmessJson.Net,
  207. HTTPOpts: HTTPOptions{},
  208. WSPath: vmessJson.Path,
  209. WSHeaders: wsHeaders,
  210. SkipCertVerify: true,
  211. ServerName: vmessJson.Host,
  212. }, nil
  213. }
  214. }
  215. var (
  216. vmessPlainRe = regexp.MustCompile("vmess://([A-Za-z0-9+/_?&=-])+")
  217. )
  218. func GrepVmessLinkFromString(text string) []string {
  219. results := make([]string, 0)
  220. texts := strings.Split(text, "vmess://")
  221. for _, text := range texts {
  222. results = append(results, vmessPlainRe.FindAllString("vmess://"+text, -1)...)
  223. }
  224. return results
  225. }