2022-07-01 11:34:02 +00:00
|
|
|
package outbound
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net"
|
|
|
|
"runtime"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/database64128/tfo-go"
|
|
|
|
"github.com/sagernet/sing-box/adapter"
|
|
|
|
"github.com/sagernet/sing-box/log"
|
2022-07-02 06:07:50 +00:00
|
|
|
"github.com/sagernet/sing-box/option"
|
2022-07-01 11:34:02 +00:00
|
|
|
"github.com/sagernet/sing/common"
|
|
|
|
"github.com/sagernet/sing/common/buf"
|
|
|
|
"github.com/sagernet/sing/common/bufio"
|
|
|
|
"github.com/sagernet/sing/common/control"
|
|
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
|
|
M "github.com/sagernet/sing/common/metadata"
|
|
|
|
N "github.com/sagernet/sing/common/network"
|
|
|
|
)
|
|
|
|
|
|
|
|
type myOutboundAdapter struct {
|
|
|
|
protocol string
|
|
|
|
logger log.Logger
|
|
|
|
tag string
|
|
|
|
dialer N.Dialer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *myOutboundAdapter) Type() string {
|
|
|
|
return a.protocol
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *myOutboundAdapter) Tag() string {
|
|
|
|
return a.tag
|
|
|
|
}
|
|
|
|
|
|
|
|
type defaultDialer struct {
|
|
|
|
tfo.Dialer
|
|
|
|
net.ListenConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *defaultDialer) DialContext(ctx context.Context, network string, address M.Socksaddr) (net.Conn, error) {
|
|
|
|
return d.Dialer.DialContext(ctx, network, address.String())
|
|
|
|
}
|
|
|
|
|
2022-07-03 05:14:49 +00:00
|
|
|
func (d *defaultDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
|
2022-07-01 11:34:02 +00:00
|
|
|
return d.ListenConfig.ListenPacket(ctx, "udp", "")
|
|
|
|
}
|
|
|
|
|
2022-07-02 06:07:50 +00:00
|
|
|
func newDialer(options option.DialerOptions) N.Dialer {
|
2022-07-01 11:34:02 +00:00
|
|
|
var dialer net.Dialer
|
|
|
|
var listener net.ListenConfig
|
|
|
|
if options.BindInterface != "" {
|
|
|
|
dialer.Control = control.Append(dialer.Control, control.BindToInterface(options.BindInterface))
|
|
|
|
listener.Control = control.Append(listener.Control, control.BindToInterface(options.BindInterface))
|
|
|
|
}
|
|
|
|
if options.RoutingMark != 0 {
|
|
|
|
dialer.Control = control.Append(dialer.Control, control.RoutingMark(options.RoutingMark))
|
|
|
|
listener.Control = control.Append(listener.Control, control.RoutingMark(options.RoutingMark))
|
|
|
|
}
|
|
|
|
if options.ReuseAddr {
|
|
|
|
listener.Control = control.Append(listener.Control, control.ReuseAddr())
|
|
|
|
}
|
|
|
|
if options.ConnectTimeout != 0 {
|
|
|
|
dialer.Timeout = time.Duration(options.ConnectTimeout) * time.Second
|
|
|
|
}
|
|
|
|
return &defaultDialer{tfo.Dialer{Dialer: dialer, DisableTFO: !options.TCPFastOpen}, listener}
|
|
|
|
}
|
|
|
|
|
|
|
|
type lazyDialer struct {
|
|
|
|
router adapter.Router
|
2022-07-02 06:07:50 +00:00
|
|
|
options option.DialerOptions
|
2022-07-01 11:34:02 +00:00
|
|
|
dialer N.Dialer
|
|
|
|
initOnce sync.Once
|
|
|
|
initErr error
|
|
|
|
}
|
|
|
|
|
2022-07-02 06:07:50 +00:00
|
|
|
func NewDialer(router adapter.Router, options option.DialerOptions) N.Dialer {
|
2022-07-01 11:34:02 +00:00
|
|
|
if options.Detour == "" {
|
|
|
|
return newDialer(options)
|
|
|
|
}
|
|
|
|
return &lazyDialer{
|
|
|
|
router: router,
|
|
|
|
options: options,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *lazyDialer) Dialer() (N.Dialer, error) {
|
|
|
|
d.initOnce.Do(func() {
|
|
|
|
var loaded bool
|
|
|
|
d.dialer, loaded = d.router.Outbound(d.options.Detour)
|
|
|
|
if !loaded {
|
|
|
|
d.initErr = E.New("outbound detour not found: ", d.options.Detour)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return d.dialer, d.initErr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *lazyDialer) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
|
|
|
|
dialer, err := d.Dialer()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return dialer.DialContext(ctx, network, destination)
|
|
|
|
}
|
|
|
|
|
2022-07-03 05:14:49 +00:00
|
|
|
func (d *lazyDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
|
2022-07-01 11:34:02 +00:00
|
|
|
dialer, err := d.Dialer()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-03 05:14:49 +00:00
|
|
|
return dialer.ListenPacket(ctx, destination)
|
2022-07-01 11:34:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func CopyEarlyConn(ctx context.Context, conn net.Conn, serverConn net.Conn) error {
|
|
|
|
_payload := buf.StackNew()
|
|
|
|
payload := common.Dup(_payload)
|
|
|
|
err := conn.SetReadDeadline(time.Now().Add(100 * time.Millisecond))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = payload.ReadFrom(conn)
|
|
|
|
if err != nil && !E.IsTimeout(err) {
|
|
|
|
return E.Cause(err, "read payload")
|
|
|
|
}
|
|
|
|
err = conn.SetReadDeadline(time.Time{})
|
|
|
|
if err != nil {
|
|
|
|
payload.Release()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = serverConn.Write(payload.Bytes())
|
|
|
|
if err != nil {
|
|
|
|
return E.Cause(err, "client handshake")
|
|
|
|
}
|
|
|
|
runtime.KeepAlive(_payload)
|
|
|
|
return bufio.CopyConn(ctx, conn, serverConn)
|
|
|
|
}
|