| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- package httpupgrade
- import (
- "context"
- "github.com/v2fly/v2ray-core/v5/common/buf"
- "github.com/v2fly/v2ray-core/v5/common/net"
- "github.com/v2fly/v2ray-core/v5/common/serial"
- "io"
- "time"
- )
- type connection struct {
- conn net.Conn
- reader io.Reader
- remoteAddr net.Addr
- shouldWait bool
- delayedDialFinish context.Context
- finishedDial context.CancelFunc
- dialer delayedDialer
- }
- type delayedDialer func(earlyData []byte) (conn net.Conn, earlyReply io.Reader, err error)
- func newConnectionWithPendingRead(conn net.Conn, remoteAddr net.Addr, earlyReplyReader io.Reader) *connection {
- return &connection{
- conn: conn,
- remoteAddr: remoteAddr,
- reader: earlyReplyReader,
- }
- }
- func newConnectionWithDelayedDial(dialer delayedDialer) *connection {
- ctx, cancel := context.WithCancel(context.Background())
- return &connection{
- shouldWait: true,
- delayedDialFinish: ctx,
- finishedDial: cancel,
- dialer: dialer,
- }
- }
- // Read implements net.Conn.Read()
- func (c *connection) Read(b []byte) (int, error) {
- if c.shouldWait {
- <-c.delayedDialFinish.Done()
- if c.conn == nil {
- return 0, newError("unable to read delayed dial websocket connection as it do not exist")
- }
- }
- if c.reader != nil {
- n, err := c.reader.Read(b)
- if err == io.EOF {
- c.reader = nil
- return c.conn.Read(b)
- }
- return n, err
- }
- return c.conn.Read(b)
- }
- // Write implements io.Writer.
- func (c *connection) Write(b []byte) (int, error) {
- if c.shouldWait {
- var err error
- var earlyReply io.Reader
- c.conn, earlyReply, err = c.dialer(b)
- if earlyReply != nil {
- c.reader = earlyReply
- }
- 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
- }
- return c.conn.Write(b)
- }
- 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 closeErrors []interface{}
- if err := c.conn.Close(); err != nil {
- closeErrors = append(closeErrors, err)
- }
- if len(closeErrors) > 0 {
- return newError("failed to close connection").Base(newError(serial.Concat(closeErrors...)))
- }
- 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("httpupgrade 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("httpupgrade transport is not materialized when SetWriteDeadline() is called").AtWarning().WriteToLog()
- return nil
- }
- }
- return c.conn.SetWriteDeadline(t)
- }
|