123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138 |
- 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"
- "github.com/sagernet/sing-box/option"
- "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())
- }
- func (d *defaultDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
- return d.ListenConfig.ListenPacket(ctx, "udp", "")
- }
- func newDialer(options option.DialerOptions) N.Dialer {
- 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
- options option.DialerOptions
- dialer N.Dialer
- initOnce sync.Once
- initErr error
- }
- func NewDialer(router adapter.Router, options option.DialerOptions) N.Dialer {
- 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)
- }
- func (d *lazyDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
- dialer, err := d.Dialer()
- if err != nil {
- return nil, err
- }
- return dialer.ListenPacket(ctx, destination)
- }
- 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)
- }
|