| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 |
- package websocket
- import (
- "context"
- "io"
- "net"
- "time"
- "github.com/gorilla/websocket"
- "github.com/v2fly/v2ray-core/v4/common/buf"
- "github.com/v2fly/v2ray-core/v4/common/errors"
- "github.com/v2fly/v2ray-core/v4/common/serial"
- )
- var _ buf.Writer = (*connection)(nil)
- // connection is a wrapper for net.Conn over WebSocket connection.
- type connection struct {
- conn *websocket.Conn
- reader io.Reader
- remoteAddr net.Addr
- shouldWait bool
- delayedDialFinish context.Context
- finishedDial context.CancelFunc
- dialer DelayedDialer
- }
- type DelayedDialer interface {
- Dial(earlyData []byte) (*websocket.Conn, error)
- }
- func newConnection(conn *websocket.Conn, remoteAddr net.Addr) *connection {
- return &connection{
- conn: conn,
- remoteAddr: remoteAddr,
- }
- }
- func newConnectionWithEarlyData(conn *websocket.Conn, remoteAddr net.Addr, earlyData io.Reader) *connection {
- return &connection{
- conn: conn,
- remoteAddr: remoteAddr,
- reader: earlyData,
- }
- }
- func newConnectionWithDelayedDial(dialer DelayedDialer) *connection {
- delayedDialContext, cancelFunc := context.WithCancel(context.Background())
- return &connection{
- shouldWait: true,
- delayedDialFinish: delayedDialContext,
- finishedDial: cancelFunc,
- dialer: dialer,
- }
- }
- func newRelayedConnectionWithDelayedDial(dialer DelayedDialerForwarded) *connectionForwarder {
- delayedDialContext, cancelFunc := context.WithCancel(context.Background())
- return &connectionForwarder{
- shouldWait: true,
- delayedDialFinish: delayedDialContext,
- finishedDial: cancelFunc,
- dialer: dialer,
- }
- }
- func newRelayedConnection(conn io.ReadWriteCloser) *connectionForwarder {
- return &connectionForwarder{
- ReadWriteCloser: conn,
- shouldWait: false,
- }
- }
- // Read implements net.Conn.Read()
- func (c *connection) Read(b []byte) (int, error) {
- for {
- reader, err := c.getReader()
- if err != nil {
- return 0, err
- }
- nBytes, err := reader.Read(b)
- if errors.Cause(err) == io.EOF {
- c.reader = nil
- continue
- }
- return nBytes, err
- }
- }
- func (c *connection) getReader() (io.Reader, error) {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- return nil, newError("unable to read delayed dial websocket connection as it do not exist")
- }
- }
- if c.reader != nil {
- return c.reader, nil
- }
- _, reader, err := c.conn.NextReader()
- if err != nil {
- return nil, err
- }
- c.reader = reader
- return reader, nil
- }
- // Write implements io.Writer.
- func (c *connection) Write(b []byte) (int, error) {
- if c.shouldWait {
- var err error
- c.conn, err = c.dialer.Dial(b)
- c.finishedDial()
- if err != nil {
- return 0, newError("Unable to proceed with delayed write").Base(err)
- }
- c.remoteAddr = c.conn.RemoteAddr()
- c.shouldWait = false
- return len(b), nil
- }
- if err := c.conn.WriteMessage(websocket.BinaryMessage, b); err != nil {
- return 0, err
- }
- return len(b), nil
- }
- func (c *connection) WriteMultiBuffer(mb buf.MultiBuffer) error {
- mb = buf.Compact(mb)
- mb, err := buf.WriteMultiBuffer(c, mb)
- buf.ReleaseMulti(mb)
- return err
- }
- func (c *connection) Close() error {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- return newError("unable to close delayed dial websocket connection as it do not exist")
- }
- }
- var errors []interface{}
- if err := c.conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseNormalClosure, ""), time.Now().Add(time.Second*5)); err != nil {
- errors = append(errors, err)
- }
- if err := c.conn.Close(); err != nil {
- errors = append(errors, err)
- }
- if len(errors) > 0 {
- return newError("failed to close connection").Base(newError(serial.Concat(errors...)))
- }
- return nil
- }
- func (c *connection) LocalAddr() net.Addr {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- newError("websocket transport is not materialized when LocalAddr() is called").AtWarning().WriteToLog()
- return &net.UnixAddr{
- Name: "@placeholder",
- Net: "unix",
- }
- }
- }
- return c.conn.LocalAddr()
- }
- func (c *connection) RemoteAddr() net.Addr {
- return c.remoteAddr
- }
- func (c *connection) SetDeadline(t time.Time) error {
- if err := c.SetReadDeadline(t); err != nil {
- return err
- }
- return c.SetWriteDeadline(t)
- }
- func (c *connection) SetReadDeadline(t time.Time) error {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- newError("websocket transport is not materialized when SetReadDeadline() is called").AtWarning().WriteToLog()
- return nil
- }
- }
- return c.conn.SetReadDeadline(t)
- }
- func (c *connection) SetWriteDeadline(t time.Time) error {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- newError("websocket transport is not materialized when SetWriteDeadline() is called").AtWarning().WriteToLog()
- return nil
- }
- }
- return c.conn.SetWriteDeadline(t)
- }
|