connection.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. package websocket
  2. import (
  3. "io"
  4. "net"
  5. "time"
  6. "github.com/gorilla/websocket"
  7. "github.com/xtls/xray-core/common/buf"
  8. "github.com/xtls/xray-core/common/errors"
  9. "github.com/xtls/xray-core/common/serial"
  10. )
  11. var _ buf.Writer = (*connection)(nil)
  12. // connection is a wrapper for net.Conn over WebSocket connection.
  13. type connection struct {
  14. conn *websocket.Conn
  15. reader io.Reader
  16. remoteAddr net.Addr
  17. }
  18. func newConnection(conn *websocket.Conn, remoteAddr net.Addr, extraReader io.Reader) *connection {
  19. return &connection{
  20. conn: conn,
  21. remoteAddr: remoteAddr,
  22. reader: extraReader,
  23. }
  24. }
  25. // Read implements net.Conn.Read()
  26. func (c *connection) Read(b []byte) (int, error) {
  27. for {
  28. reader, err := c.getReader()
  29. if err != nil {
  30. return 0, err
  31. }
  32. nBytes, err := reader.Read(b)
  33. if errors.Cause(err) == io.EOF {
  34. c.reader = nil
  35. continue
  36. }
  37. return nBytes, err
  38. }
  39. }
  40. func (c *connection) getReader() (io.Reader, error) {
  41. if c.reader != nil {
  42. return c.reader, nil
  43. }
  44. _, reader, err := c.conn.NextReader()
  45. if err != nil {
  46. return nil, err
  47. }
  48. c.reader = reader
  49. return reader, nil
  50. }
  51. // Write implements io.Writer.
  52. func (c *connection) Write(b []byte) (int, error) {
  53. if err := c.conn.WriteMessage(websocket.BinaryMessage, b); err != nil {
  54. return 0, err
  55. }
  56. return len(b), nil
  57. }
  58. func (c *connection) WriteMultiBuffer(mb buf.MultiBuffer) error {
  59. mb = buf.Compact(mb)
  60. mb, err := buf.WriteMultiBuffer(c, mb)
  61. buf.ReleaseMulti(mb)
  62. return err
  63. }
  64. func (c *connection) Close() error {
  65. var errors []interface{}
  66. if err := c.conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseNormalClosure, ""), time.Now().Add(time.Second*5)); err != nil {
  67. errors = append(errors, err)
  68. }
  69. if err := c.conn.Close(); err != nil {
  70. errors = append(errors, err)
  71. }
  72. if len(errors) > 0 {
  73. return newError("failed to close connection").Base(newError(serial.Concat(errors...)))
  74. }
  75. return nil
  76. }
  77. func (c *connection) LocalAddr() net.Addr {
  78. return c.conn.LocalAddr()
  79. }
  80. func (c *connection) RemoteAddr() net.Addr {
  81. return c.remoteAddr
  82. }
  83. func (c *connection) SetDeadline(t time.Time) error {
  84. if err := c.SetReadDeadline(t); err != nil {
  85. return err
  86. }
  87. return c.SetWriteDeadline(t)
  88. }
  89. func (c *connection) SetReadDeadline(t time.Time) error {
  90. return c.conn.SetReadDeadline(t)
  91. }
  92. func (c *connection) SetWriteDeadline(t time.Time) error {
  93. return c.conn.SetWriteDeadline(t)
  94. }