hub.go 4.0 KB

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