hub.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package websocket
  2. import (
  3. "crypto/tls"
  4. "net"
  5. "net/http"
  6. "strconv"
  7. "sync"
  8. "time"
  9. "github.com/gorilla/websocket"
  10. "v2ray.com/core/app/log"
  11. "v2ray.com/core/common"
  12. "v2ray.com/core/common/errors"
  13. v2net "v2ray.com/core/common/net"
  14. "v2ray.com/core/transport/internet"
  15. "v2ray.com/core/transport/internet/internal"
  16. v2tls "v2ray.com/core/transport/internet/tls"
  17. )
  18. var (
  19. ErrClosedListener = errors.New("Listener is closed.")
  20. )
  21. type ConnectionWithError struct {
  22. conn net.Conn
  23. err error
  24. }
  25. type requestHandler struct {
  26. path string
  27. conns chan *ConnectionWithError
  28. }
  29. func (h *requestHandler) ServeHTTP(writer http.ResponseWriter, request *http.Request) {
  30. if request.URL.Path != h.path {
  31. writer.WriteHeader(http.StatusNotFound)
  32. return
  33. }
  34. conn, err := converttovws(writer, request)
  35. if err != nil {
  36. log.Info("WebSocket|Listener: Failed to convert to WebSocket connection: ", err)
  37. return
  38. }
  39. select {
  40. case h.conns <- &ConnectionWithError{conn: conn}:
  41. default:
  42. conn.Close()
  43. }
  44. }
  45. type Listener struct {
  46. sync.Mutex
  47. acccepting bool
  48. awaitingConns chan *ConnectionWithError
  49. listener net.Listener
  50. tlsConfig *tls.Config
  51. config *Config
  52. }
  53. func ListenWS(address v2net.Address, port v2net.Port, options internet.ListenOptions) (internet.Listener, error) {
  54. networkSettings, err := options.Stream.GetEffectiveTransportSettings()
  55. if err != nil {
  56. return nil, err
  57. }
  58. wsSettings := networkSettings.(*Config)
  59. l := &Listener{
  60. acccepting: true,
  61. awaitingConns: make(chan *ConnectionWithError, 32),
  62. config: wsSettings,
  63. }
  64. if options.Stream != nil && options.Stream.HasSecuritySettings() {
  65. securitySettings, err := options.Stream.GetEffectiveSecuritySettings()
  66. if err != nil {
  67. log.Error("WebSocket: Failed to create apply TLS config: ", err)
  68. return nil, err
  69. }
  70. tlsConfig, ok := securitySettings.(*v2tls.Config)
  71. if ok {
  72. l.tlsConfig = tlsConfig.GetTLSConfig()
  73. }
  74. }
  75. err = l.listenws(address, port)
  76. return l, err
  77. }
  78. func (ln *Listener) listenws(address v2net.Address, port v2net.Port) error {
  79. netAddr := address.String() + ":" + strconv.Itoa(int(port.Value()))
  80. var listener net.Listener
  81. if ln.tlsConfig == nil {
  82. l, err := net.Listen("tcp", netAddr)
  83. if err != nil {
  84. return errors.Base(err).Message("WebSocket|Listener: Failed to listen TCP ", netAddr)
  85. }
  86. listener = l
  87. } else {
  88. l, err := tls.Listen("tcp", netAddr, ln.tlsConfig)
  89. if err != nil {
  90. return errors.Base(err).Message("WebSocket|Listener: Failed to listen TLS ", netAddr)
  91. }
  92. listener = l
  93. }
  94. ln.listener = listener
  95. go func() {
  96. http.Serve(listener, &requestHandler{
  97. path: "/" + ln.config.GetNormailzedPath(),
  98. conns: ln.awaitingConns,
  99. })
  100. }()
  101. return nil
  102. }
  103. func converttovws(w http.ResponseWriter, r *http.Request) (*wsconn, error) {
  104. var upgrader = websocket.Upgrader{
  105. ReadBufferSize: 32 * 1024,
  106. WriteBufferSize: 32 * 1024,
  107. }
  108. conn, err := upgrader.Upgrade(w, r, nil)
  109. if err != nil {
  110. return nil, err
  111. }
  112. return &wsconn{wsc: conn}, nil
  113. }
  114. func (ln *Listener) Accept() (internet.Connection, error) {
  115. for ln.acccepting {
  116. select {
  117. case connErr, open := <-ln.awaitingConns:
  118. if !open {
  119. return nil, ErrClosedListener
  120. }
  121. if connErr.err != nil {
  122. return nil, connErr.err
  123. }
  124. return internal.NewConnection(internal.ConnectionID{}, connErr.conn, ln, internal.ReuseConnection(ln.config.IsConnectionReuse())), nil
  125. case <-time.After(time.Second * 2):
  126. }
  127. }
  128. return nil, ErrClosedListener
  129. }
  130. func (ln *Listener) Put(id internal.ConnectionID, conn net.Conn) {
  131. ln.Lock()
  132. defer ln.Unlock()
  133. if !ln.acccepting {
  134. return
  135. }
  136. select {
  137. case ln.awaitingConns <- &ConnectionWithError{conn: conn}:
  138. default:
  139. conn.Close()
  140. }
  141. }
  142. func (ln *Listener) Addr() net.Addr {
  143. return ln.listener.Addr()
  144. }
  145. func (ln *Listener) Close() error {
  146. ln.Lock()
  147. defer ln.Unlock()
  148. ln.acccepting = false
  149. ln.listener.Close()
  150. close(ln.awaitingConns)
  151. for connErr := range ln.awaitingConns {
  152. if connErr.conn != nil {
  153. connErr.conn.Close()
  154. }
  155. }
  156. return nil
  157. }
  158. func init() {
  159. common.Must(internet.RegisterTransportListener(internet.TransportProtocol_WebSocket, ListenWS))
  160. }