hub.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  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. return nil, errors.Base(err).Message("WebSocket: Failed to create apply TLS config.")
  68. }
  69. tlsConfig, ok := securitySettings.(*v2tls.Config)
  70. if ok {
  71. l.tlsConfig = tlsConfig.GetTLSConfig()
  72. }
  73. }
  74. err = l.listenws(address, port)
  75. return l, err
  76. }
  77. func (ln *Listener) listenws(address v2net.Address, port v2net.Port) error {
  78. netAddr := address.String() + ":" + strconv.Itoa(int(port.Value()))
  79. var listener net.Listener
  80. if ln.tlsConfig == nil {
  81. l, err := net.Listen("tcp", netAddr)
  82. if err != nil {
  83. return errors.Base(err).Message("WebSocket|Listener: Failed to listen TCP ", netAddr)
  84. }
  85. listener = l
  86. } else {
  87. l, err := tls.Listen("tcp", netAddr, ln.tlsConfig)
  88. if err != nil {
  89. return errors.Base(err).Message("WebSocket|Listener: Failed to listen TLS ", netAddr)
  90. }
  91. listener = l
  92. }
  93. ln.listener = listener
  94. go func() {
  95. http.Serve(listener, &requestHandler{
  96. path: ln.config.GetNormailzedPath(),
  97. conns: ln.awaitingConns,
  98. })
  99. }()
  100. return nil
  101. }
  102. func converttovws(w http.ResponseWriter, r *http.Request) (*wsconn, error) {
  103. var upgrader = websocket.Upgrader{
  104. ReadBufferSize: 32 * 1024,
  105. WriteBufferSize: 32 * 1024,
  106. }
  107. conn, err := upgrader.Upgrade(w, r, nil)
  108. if err != nil {
  109. return nil, err
  110. }
  111. return &wsconn{wsc: conn}, nil
  112. }
  113. func (ln *Listener) Accept() (internet.Connection, error) {
  114. for ln.acccepting {
  115. select {
  116. case connErr, open := <-ln.awaitingConns:
  117. if !open {
  118. return nil, ErrClosedListener
  119. }
  120. if connErr.err != nil {
  121. return nil, connErr.err
  122. }
  123. return internal.NewConnection(internal.ConnectionID{}, connErr.conn, ln, internal.ReuseConnection(ln.config.IsConnectionReuse())), nil
  124. case <-time.After(time.Second * 2):
  125. }
  126. }
  127. return nil, ErrClosedListener
  128. }
  129. func (ln *Listener) Put(id internal.ConnectionID, conn net.Conn) {
  130. ln.Lock()
  131. defer ln.Unlock()
  132. if !ln.acccepting {
  133. return
  134. }
  135. select {
  136. case ln.awaitingConns <- &ConnectionWithError{conn: conn}:
  137. default:
  138. conn.Close()
  139. }
  140. }
  141. func (ln *Listener) Addr() net.Addr {
  142. return ln.listener.Addr()
  143. }
  144. func (ln *Listener) Close() error {
  145. ln.Lock()
  146. defer ln.Unlock()
  147. ln.acccepting = false
  148. ln.listener.Close()
  149. close(ln.awaitingConns)
  150. for connErr := range ln.awaitingConns {
  151. if connErr.conn != nil {
  152. connErr.conn.Close()
  153. }
  154. }
  155. return nil
  156. }
  157. func init() {
  158. common.Must(internet.RegisterTransportListener(internet.TransportProtocol_WebSocket, ListenWS))
  159. }