dialer.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. package websocket
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/base64"
  6. "github.com/v2fly/v2ray-core/v5/transport/internet/security"
  7. "io"
  8. gonet "net"
  9. "net/http"
  10. "time"
  11. "github.com/gorilla/websocket"
  12. core "github.com/v2fly/v2ray-core/v5"
  13. "github.com/v2fly/v2ray-core/v5/common"
  14. "github.com/v2fly/v2ray-core/v5/common/net"
  15. "github.com/v2fly/v2ray-core/v5/common/session"
  16. "github.com/v2fly/v2ray-core/v5/features/extension"
  17. "github.com/v2fly/v2ray-core/v5/transport/internet"
  18. )
  19. // Dial dials a WebSocket connection to the given destination.
  20. func Dial(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (internet.Connection, error) {
  21. newError("creating connection to ", dest).WriteToLog(session.ExportIDToError(ctx))
  22. conn, err := dialWebsocket(ctx, dest, streamSettings)
  23. if err != nil {
  24. return nil, newError("failed to dial WebSocket").Base(err)
  25. }
  26. return internet.Connection(conn), nil
  27. }
  28. func init() {
  29. common.Must(internet.RegisterTransportDialer(protocolName, Dial))
  30. }
  31. func dialWebsocket(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (net.Conn, error) {
  32. wsSettings := streamSettings.ProtocolSettings.(*Config)
  33. dialer := &websocket.Dialer{
  34. NetDial: func(network, addr string) (net.Conn, error) {
  35. return internet.DialSystem(ctx, dest, streamSettings.SocketSettings)
  36. },
  37. ReadBufferSize: 4 * 1024,
  38. WriteBufferSize: 4 * 1024,
  39. HandshakeTimeout: time.Second * 8,
  40. }
  41. protocol := "ws"
  42. securityEngine, err := security.CreateSecurityEngineFromSettings(ctx, streamSettings)
  43. if err != nil {
  44. return nil, newError("unable to create security engine").Base(err)
  45. }
  46. if securityEngine != nil {
  47. protocol = "wss"
  48. dialer.NetDialTLSContext = func(ctx context.Context, network, addr string) (gonet.Conn, error) {
  49. conn, err := dialer.NetDial(network, addr)
  50. if err != nil {
  51. return nil, newError("dial TLS connection failed").Base(err)
  52. }
  53. conn, err = securityEngine.Client(conn,
  54. security.OptionWithDestination{Dest: dest},
  55. security.OptionWithALPN{ALPNs: []string{"http/1.1"}})
  56. if err != nil {
  57. return nil, newError("unable to create security protocol client from security engine").Base(err)
  58. }
  59. return conn, nil
  60. }
  61. }
  62. host := dest.NetAddr()
  63. if (protocol == "ws" && dest.Port == 80) || (protocol == "wss" && dest.Port == 443) {
  64. host = dest.Address.String()
  65. }
  66. uri := protocol + "://" + host + wsSettings.GetNormalizedPath()
  67. if wsSettings.UseBrowserForwarding {
  68. var forwarder extension.BrowserForwarder
  69. err := core.RequireFeatures(ctx, func(Forwarder extension.BrowserForwarder) {
  70. forwarder = Forwarder
  71. })
  72. if err != nil {
  73. return nil, newError("cannot find browser forwarder service").Base(err)
  74. }
  75. if wsSettings.MaxEarlyData != 0 {
  76. return newRelayedConnectionWithDelayedDial(&dialerWithEarlyDataRelayed{
  77. forwarder: forwarder,
  78. uriBase: uri,
  79. config: wsSettings,
  80. }), nil
  81. }
  82. conn, err := forwarder.DialWebsocket(uri, nil)
  83. if err != nil {
  84. return nil, newError("cannot dial with browser forwarder service").Base(err)
  85. }
  86. return newRelayedConnection(conn), nil
  87. }
  88. if wsSettings.MaxEarlyData != 0 {
  89. return newConnectionWithDelayedDial(&dialerWithEarlyData{
  90. dialer: dialer,
  91. uriBase: uri,
  92. config: wsSettings,
  93. }), nil
  94. }
  95. conn, resp, err := dialer.Dial(uri, wsSettings.GetRequestHeader()) // nolint: bodyclose
  96. if err != nil {
  97. var reason string
  98. if resp != nil {
  99. reason = resp.Status
  100. }
  101. return nil, newError("failed to dial to (", uri, "): ", reason).Base(err)
  102. }
  103. return newConnection(conn, conn.RemoteAddr()), nil
  104. }
  105. type dialerWithEarlyData struct {
  106. dialer *websocket.Dialer
  107. uriBase string
  108. config *Config
  109. }
  110. func (d dialerWithEarlyData) Dial(earlyData []byte) (*websocket.Conn, error) {
  111. earlyDataBuf := bytes.NewBuffer(nil)
  112. base64EarlyDataEncoder := base64.NewEncoder(base64.RawURLEncoding, earlyDataBuf)
  113. earlydata := bytes.NewReader(earlyData)
  114. limitedEarlyDatareader := io.LimitReader(earlydata, int64(d.config.MaxEarlyData))
  115. n, encerr := io.Copy(base64EarlyDataEncoder, limitedEarlyDatareader)
  116. if encerr != nil {
  117. return nil, newError("websocket delayed dialer cannot encode early data").Base(encerr)
  118. }
  119. if errc := base64EarlyDataEncoder.Close(); errc != nil {
  120. return nil, newError("websocket delayed dialer cannot encode early data tail").Base(errc)
  121. }
  122. dialFunction := func() (*websocket.Conn, *http.Response, error) {
  123. return d.dialer.Dial(d.uriBase+earlyDataBuf.String(), d.config.GetRequestHeader())
  124. }
  125. if d.config.EarlyDataHeaderName != "" {
  126. dialFunction = func() (*websocket.Conn, *http.Response, error) {
  127. earlyDataStr := earlyDataBuf.String()
  128. currentHeader := d.config.GetRequestHeader()
  129. currentHeader.Set(d.config.EarlyDataHeaderName, earlyDataStr)
  130. return d.dialer.Dial(d.uriBase, currentHeader)
  131. }
  132. }
  133. conn, resp, err := dialFunction() // nolint: bodyclose
  134. if err != nil {
  135. var reason string
  136. if resp != nil {
  137. reason = resp.Status
  138. }
  139. return nil, newError("failed to dial to (", d.uriBase, ") with early data: ", reason).Base(err)
  140. }
  141. if n != int64(len(earlyData)) {
  142. if errWrite := conn.WriteMessage(websocket.BinaryMessage, earlyData[n:]); errWrite != nil {
  143. return nil, newError("failed to dial to (", d.uriBase, ") with early data as write of remainder early data failed: ").Base(err)
  144. }
  145. }
  146. return conn, nil
  147. }
  148. type dialerWithEarlyDataRelayed struct {
  149. forwarder extension.BrowserForwarder
  150. uriBase string
  151. config *Config
  152. }
  153. func (d dialerWithEarlyDataRelayed) Dial(earlyData []byte) (io.ReadWriteCloser, error) {
  154. earlyDataBuf := bytes.NewBuffer(nil)
  155. base64EarlyDataEncoder := base64.NewEncoder(base64.RawURLEncoding, earlyDataBuf)
  156. earlydata := bytes.NewReader(earlyData)
  157. limitedEarlyDatareader := io.LimitReader(earlydata, int64(d.config.MaxEarlyData))
  158. n, encerr := io.Copy(base64EarlyDataEncoder, limitedEarlyDatareader)
  159. if encerr != nil {
  160. return nil, newError("websocket delayed dialer cannot encode early data").Base(encerr)
  161. }
  162. if errc := base64EarlyDataEncoder.Close(); errc != nil {
  163. return nil, newError("websocket delayed dialer cannot encode early data tail").Base(errc)
  164. }
  165. dialFunction := func() (io.ReadWriteCloser, error) {
  166. return d.forwarder.DialWebsocket(d.uriBase+earlyDataBuf.String(), d.config.GetRequestHeader())
  167. }
  168. if d.config.EarlyDataHeaderName != "" {
  169. earlyDataStr := earlyDataBuf.String()
  170. currentHeader := d.config.GetRequestHeader()
  171. currentHeader.Set(d.config.EarlyDataHeaderName, earlyDataStr)
  172. return d.forwarder.DialWebsocket(d.uriBase, currentHeader)
  173. }
  174. conn, err := dialFunction()
  175. if err != nil {
  176. var reason string
  177. return nil, newError("failed to dial to (", d.uriBase, ") with early data: ", reason).Base(err)
  178. }
  179. if n != int64(len(earlyData)) {
  180. if _, errWrite := conn.Write(earlyData[n:]); errWrite != nil {
  181. return nil, newError("failed to dial to (", d.uriBase, ") with early data as write of remainder early data failed: ").Base(err)
  182. }
  183. }
  184. return conn, nil
  185. }