sing-box/outbound/socks.go

104 lines
3.1 KiB
Go
Raw Permalink Normal View History

2022-07-03 05:14:49 +00:00
package outbound
import (
"context"
"net"
"github.com/sagernet/sing-box/adapter"
2022-07-07 13:47:21 +00:00
"github.com/sagernet/sing-box/common/dialer"
2022-07-03 05:14:49 +00:00
C "github.com/sagernet/sing-box/constant"
"github.com/sagernet/sing-box/log"
"github.com/sagernet/sing-box/option"
2022-07-29 16:29:22 +00:00
E "github.com/sagernet/sing/common/exceptions"
2022-07-08 15:03:57 +00:00
M "github.com/sagernet/sing/common/metadata"
N "github.com/sagernet/sing/common/network"
2022-08-12 09:55:52 +00:00
"github.com/sagernet/sing/common/uot"
2022-07-08 15:03:57 +00:00
"github.com/sagernet/sing/protocol/socks"
2022-07-03 05:14:49 +00:00
)
var _ adapter.Outbound = (*Socks)(nil)
type Socks struct {
myOutboundAdapter
client *socks.Client
2022-08-12 09:55:52 +00:00
uot bool
2022-07-03 05:14:49 +00:00
}
2022-07-12 07:17:29 +00:00
func NewSocks(router adapter.Router, logger log.ContextLogger, tag string, options option.SocksOutboundOptions) (*Socks, error) {
2022-07-07 15:36:32 +00:00
detour := dialer.NewOutbound(router, options.OutboundDialerOptions)
2022-07-03 05:14:49 +00:00
var version socks.Version
var err error
if options.Version != "" {
version, err = socks.ParseVersion(options.Version)
} else {
version = socks.Version5
}
if err != nil {
return nil, err
}
return &Socks{
myOutboundAdapter{
protocol: C.TypeSocks,
2022-07-23 01:15:47 +00:00
network: options.Network.Build(),
router: router,
2022-07-03 05:14:49 +00:00
logger: logger,
tag: tag,
},
2022-07-07 13:47:21 +00:00
socks.NewClient(detour, options.ServerOptions.Build(), version, options.Username, options.Password),
2022-08-12 09:55:52 +00:00
options.UoT,
2022-07-03 05:14:49 +00:00
}, nil
}
func (h *Socks) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
2022-07-07 13:47:21 +00:00
ctx, metadata := adapter.AppendContext(ctx)
metadata.Outbound = h.tag
2022-07-07 15:36:32 +00:00
metadata.Destination = destination
2022-07-29 16:29:22 +00:00
switch N.NetworkName(network) {
case N.NetworkTCP:
2022-07-12 07:17:29 +00:00
h.logger.InfoContext(ctx, "outbound connection to ", destination)
2022-07-29 16:29:22 +00:00
case N.NetworkUDP:
2022-08-12 09:55:52 +00:00
if h.uot {
h.logger.InfoContext(ctx, "outbound UoT packet connection to ", destination)
tcpConn, err := h.client.DialContext(ctx, N.NetworkTCP, M.Socksaddr{
Fqdn: uot.UOTMagicAddress,
Port: destination.Port,
})
if err != nil {
return nil, err
}
return uot.NewClientConn(tcpConn), nil
}
2022-07-12 07:17:29 +00:00
h.logger.InfoContext(ctx, "outbound packet connection to ", destination)
2022-07-03 05:14:49 +00:00
default:
2022-07-29 16:29:22 +00:00
return nil, E.Extend(N.ErrUnknownNetwork, network)
2022-07-03 05:14:49 +00:00
}
return h.client.DialContext(ctx, network, destination)
}
func (h *Socks) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
2022-07-07 13:47:21 +00:00
ctx, metadata := adapter.AppendContext(ctx)
metadata.Outbound = h.tag
2022-07-07 15:36:32 +00:00
metadata.Destination = destination
2022-08-12 09:55:52 +00:00
if h.uot {
h.logger.InfoContext(ctx, "outbound UoT packet connection to ", destination)
tcpConn, err := h.client.DialContext(ctx, N.NetworkTCP, M.Socksaddr{
Fqdn: uot.UOTMagicAddress,
Port: destination.Port,
})
if err != nil {
return nil, err
}
return uot.NewClientConn(tcpConn), nil
}
2022-07-12 07:17:29 +00:00
h.logger.InfoContext(ctx, "outbound packet connection to ", destination)
2022-07-03 05:14:49 +00:00
return h.client.ListenPacket(ctx, destination)
}
2022-07-07 15:36:32 +00:00
func (h *Socks) NewConnection(ctx context.Context, conn net.Conn, metadata adapter.InboundContext) error {
return NewConnection(ctx, h, conn, metadata)
2022-07-03 05:14:49 +00:00
}
2022-07-07 15:36:32 +00:00
func (h *Socks) NewPacketConnection(ctx context.Context, conn N.PacketConn, metadata adapter.InboundContext) error {
return NewPacketConnection(ctx, h, conn, metadata)
2022-07-03 05:14:49 +00:00
}