sing-box/protocol/socks/outbound.go

117 lines
3.7 KiB
Go
Raw Normal View History

2024-11-02 00:39:02 +08:00
package socks
2022-07-03 13:14:49 +08:00
import (
"context"
"net"
"github.com/sagernet/sing-box/adapter"
2024-11-02 00:39:02 +08:00
"github.com/sagernet/sing-box/adapter/outbound"
2022-07-07 21:47:21 +08:00
"github.com/sagernet/sing-box/common/dialer"
2022-07-03 13:14:49 +08:00
C "github.com/sagernet/sing-box/constant"
"github.com/sagernet/sing-box/log"
"github.com/sagernet/sing-box/option"
2023-03-17 12:24:29 +08:00
"github.com/sagernet/sing/common"
2022-07-30 00:29:22 +08:00
E "github.com/sagernet/sing/common/exceptions"
2024-11-02 00:39:02 +08:00
"github.com/sagernet/sing/common/logger"
2022-07-08 23:03:57 +08:00
M "github.com/sagernet/sing/common/metadata"
N "github.com/sagernet/sing/common/network"
2022-08-12 17:55:52 +08:00
"github.com/sagernet/sing/common/uot"
2022-07-08 23:03:57 +08:00
"github.com/sagernet/sing/protocol/socks"
2022-07-03 13:14:49 +08:00
)
2024-11-02 00:39:02 +08:00
func RegisterOutbound(registry *outbound.Registry) {
outbound.Register[option.SOCKSOutboundOptions](registry, C.TypeSOCKS, NewOutbound)
}
var _ adapter.Outbound = (*Outbound)(nil)
2022-07-03 13:14:49 +08:00
2024-11-02 00:39:02 +08:00
type Outbound struct {
outbound.Adapter
router adapter.Router
logger logger.ContextLogger
2023-03-17 12:24:29 +08:00
client *socks.Client
resolve bool
uotClient *uot.Client
2022-07-03 13:14:49 +08:00
}
2024-11-02 00:39:02 +08:00
func NewOutbound(ctx context.Context, router adapter.Router, logger log.ContextLogger, tag string, options option.SOCKSOutboundOptions) (adapter.Outbound, error) {
2022-07-03 13:14:49 +08: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
}
outboundDialer, err := dialer.New(ctx, options.DialerOptions)
2023-08-08 16:14:03 +08:00
if err != nil {
return nil, err
}
2024-11-02 00:39:02 +08:00
outbound := &Outbound{
2024-11-21 18:10:41 +08:00
Adapter: outbound.NewAdapterWithDialerOptions(C.TypeSOCKS, tag, options.Network.Build(), options.DialerOptions),
2024-11-02 00:39:02 +08:00
router: router,
logger: logger,
2023-08-08 16:14:03 +08:00
client: socks.NewClient(outboundDialer, options.ServerOptions.Build(), version, options.Username, options.Password),
2023-03-15 13:34:02 +08:00
resolve: version == socks.Version4,
}
uotOptions := common.PtrValueOrDefault(options.UDPOverTCP)
2023-03-17 12:24:29 +08:00
if uotOptions.Enabled {
outbound.uotClient = &uot.Client{
Dialer: outbound.client,
Version: uotOptions.Version,
}
2023-03-15 13:34:02 +08:00
}
return outbound, nil
2022-07-03 13:14:49 +08:00
}
2024-11-02 00:39:02 +08:00
func (h *Outbound) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
2024-10-27 07:45:15 +08:00
ctx, metadata := adapter.ExtendContext(ctx)
2024-11-02 00:39:02 +08:00
metadata.Outbound = h.Tag()
2022-07-07 23:36:32 +08:00
metadata.Destination = destination
2022-07-30 00:29:22 +08:00
switch N.NetworkName(network) {
case N.NetworkTCP:
2022-07-12 15:17:29 +08:00
h.logger.InfoContext(ctx, "outbound connection to ", destination)
2022-07-30 00:29:22 +08:00
case N.NetworkUDP:
2023-03-17 12:24:29 +08:00
if h.uotClient != nil {
h.logger.InfoContext(ctx, "outbound UoT connect packet connection to ", destination)
return h.uotClient.DialContext(ctx, network, destination)
2022-08-12 17:55:52 +08:00
}
2022-07-12 15:17:29 +08:00
h.logger.InfoContext(ctx, "outbound packet connection to ", destination)
2022-07-03 13:14:49 +08:00
default:
2022-07-30 00:29:22 +08:00
return nil, E.Extend(N.ErrUnknownNetwork, network)
2022-07-03 13:14:49 +08:00
}
2022-09-25 14:42:39 +08:00
if h.resolve && destination.IsFqdn() {
2023-09-06 19:13:39 +08:00
destinationAddresses, err := h.router.LookupDefault(ctx, destination.Fqdn)
2022-09-12 11:33:38 +08:00
if err != nil {
return nil, err
}
2023-09-06 19:13:39 +08:00
return N.DialSerial(ctx, h.client, network, destination, destinationAddresses)
2022-09-12 11:33:38 +08:00
}
2022-07-03 13:14:49 +08:00
return h.client.DialContext(ctx, network, destination)
}
2024-11-02 00:39:02 +08:00
func (h *Outbound) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
2024-10-27 07:45:15 +08:00
ctx, metadata := adapter.ExtendContext(ctx)
2024-11-02 00:39:02 +08:00
metadata.Outbound = h.Tag()
2022-07-07 23:36:32 +08:00
metadata.Destination = destination
2023-03-17 12:24:29 +08:00
if h.uotClient != nil {
2022-08-12 17:55:52 +08:00
h.logger.InfoContext(ctx, "outbound UoT packet connection to ", destination)
2023-03-17 12:24:29 +08:00
return h.uotClient.ListenPacket(ctx, destination)
2022-08-12 17:55:52 +08:00
}
2023-09-06 19:13:39 +08:00
if h.resolve && destination.IsFqdn() {
destinationAddresses, err := h.router.LookupDefault(ctx, destination.Fqdn)
if err != nil {
return nil, err
}
packetConn, _, err := N.ListenSerial(ctx, h.client, destination, destinationAddresses)
if err != nil {
return nil, err
}
return packetConn, nil
}
2022-07-12 15:17:29 +08:00
h.logger.InfoContext(ctx, "outbound packet connection to ", destination)
2022-07-03 13:14:49 +08:00
return h.client.ListenPacket(ctx, destination)
}