2023-02-24 13:53:44 +08:00
|
|
|
package dialer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"time"
|
2024-08-21 10:51:49 +08:00
|
|
|
|
|
|
|
"github.com/metacubex/tfo-go"
|
2023-02-24 13:53:44 +08:00
|
|
|
)
|
|
|
|
|
2024-08-21 10:51:49 +08:00
|
|
|
var DisableTFO = false
|
|
|
|
|
2023-02-24 13:53:44 +08:00
|
|
|
type tfoConn struct {
|
|
|
|
net.Conn
|
|
|
|
closed bool
|
|
|
|
dialed chan bool
|
|
|
|
cancel context.CancelFunc
|
|
|
|
ctx context.Context
|
|
|
|
dialFn func(ctx context.Context, earlyData []byte) (net.Conn, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) Dial(earlyData []byte) (err error) {
|
2023-05-16 14:55:50 +08:00
|
|
|
conn, err := c.dialFn(c.ctx, earlyData)
|
2023-02-24 13:53:44 +08:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
2023-05-16 14:55:50 +08:00
|
|
|
c.Conn = conn
|
2023-02-24 13:53:44 +08:00
|
|
|
c.dialed <- true
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) Read(b []byte) (n int, err error) {
|
|
|
|
if c.closed {
|
|
|
|
return 0, io.ErrClosedPipe
|
|
|
|
}
|
|
|
|
if c.Conn == nil {
|
|
|
|
select {
|
|
|
|
case <-c.ctx.Done():
|
|
|
|
return 0, io.ErrUnexpectedEOF
|
|
|
|
case <-c.dialed:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return c.Conn.Read(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) Write(b []byte) (n int, err error) {
|
|
|
|
if c.closed {
|
|
|
|
return 0, io.ErrClosedPipe
|
|
|
|
}
|
|
|
|
if c.Conn == nil {
|
|
|
|
if err := c.Dial(b); err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return len(b), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Conn.Write(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) Close() error {
|
|
|
|
c.closed = true
|
|
|
|
c.cancel()
|
|
|
|
if c.Conn == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c.Conn.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) LocalAddr() net.Addr {
|
|
|
|
if c.Conn == nil {
|
2024-01-24 17:52:45 +08:00
|
|
|
return &net.TCPAddr{}
|
2023-02-24 13:53:44 +08:00
|
|
|
}
|
|
|
|
return c.Conn.LocalAddr()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) RemoteAddr() net.Addr {
|
|
|
|
if c.Conn == nil {
|
2024-01-24 17:52:45 +08:00
|
|
|
return &net.TCPAddr{}
|
2023-02-24 13:53:44 +08:00
|
|
|
}
|
|
|
|
return c.Conn.RemoteAddr()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) SetDeadline(t time.Time) error {
|
|
|
|
if err := c.SetReadDeadline(t); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return c.SetWriteDeadline(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) SetReadDeadline(t time.Time) error {
|
|
|
|
if c.Conn == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c.Conn.SetReadDeadline(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) SetWriteDeadline(t time.Time) error {
|
|
|
|
if c.Conn == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c.Conn.SetWriteDeadline(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) Upstream() any {
|
|
|
|
if c.Conn == nil { // ensure return a nil interface not an interface with nil value
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return c.Conn
|
|
|
|
}
|
|
|
|
|
2023-04-17 19:29:07 +08:00
|
|
|
func (c *tfoConn) NeedAdditionalReadDeadline() bool {
|
|
|
|
return c.Conn == nil
|
|
|
|
}
|
|
|
|
|
2023-02-27 00:26:49 +08:00
|
|
|
func (c *tfoConn) NeedHandshake() bool {
|
|
|
|
return c.Conn == nil
|
|
|
|
}
|
|
|
|
|
2023-04-25 23:01:05 +08:00
|
|
|
func (c *tfoConn) ReaderReplaceable() bool {
|
|
|
|
return c.Conn != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *tfoConn) WriterReplaceable() bool {
|
|
|
|
return c.Conn != nil
|
|
|
|
}
|
2024-08-21 10:51:49 +08:00
|
|
|
|
|
|
|
func dialTFO(ctx context.Context, netDialer net.Dialer, network, address string) (net.Conn, error) {
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), DefaultTCPTimeout)
|
|
|
|
dialer := tfo.Dialer{Dialer: netDialer, DisableTFO: false}
|
|
|
|
return &tfoConn{
|
|
|
|
dialed: make(chan bool, 1),
|
|
|
|
cancel: cancel,
|
|
|
|
ctx: ctx,
|
|
|
|
dialFn: func(ctx context.Context, earlyData []byte) (net.Conn, error) {
|
|
|
|
return dialer.DialContext(ctx, network, address, earlyData)
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|