transport_dialer.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. package dns
  2. import (
  3. "context"
  4. "net"
  5. "time"
  6. "github.com/sagernet/sing-box/adapter"
  7. "github.com/sagernet/sing-box/common/dialer"
  8. C "github.com/sagernet/sing-box/constant"
  9. "github.com/sagernet/sing-box/option"
  10. E "github.com/sagernet/sing/common/exceptions"
  11. M "github.com/sagernet/sing/common/metadata"
  12. N "github.com/sagernet/sing/common/network"
  13. "github.com/sagernet/sing/service"
  14. )
  15. func NewLocalDialer(ctx context.Context, options option.LocalDNSServerOptions) (N.Dialer, error) {
  16. if options.LegacyDefaultDialer {
  17. return dialer.NewDefaultOutbound(ctx), nil
  18. } else {
  19. return dialer.New(ctx, options.DialerOptions)
  20. }
  21. }
  22. func NewRemoteDialer(ctx context.Context, options option.RemoteDNSServerOptions) (N.Dialer, error) {
  23. transportDialer, err := NewLocalDialer(ctx, options.LocalDNSServerOptions)
  24. if err != nil {
  25. return nil, err
  26. }
  27. if options.AddressResolver != "" {
  28. transport := service.FromContext[adapter.DNSTransportManager](ctx)
  29. resolverTransport, loaded := transport.Transport(options.AddressResolver)
  30. if !loaded {
  31. return nil, E.New("address resolver not found: ", options.AddressResolver)
  32. }
  33. transportDialer = NewTransportDialer(transportDialer, service.FromContext[adapter.DNSRouter](ctx), resolverTransport, C.DomainStrategy(options.AddressStrategy), time.Duration(options.AddressFallbackDelay))
  34. } else if M.IsDomainName(options.Server) {
  35. return nil, E.New("missing address resolver for server: ", options.Server)
  36. }
  37. return transportDialer, nil
  38. }
  39. type TransportDialer struct {
  40. dialer N.Dialer
  41. dnsRouter adapter.DNSRouter
  42. transport adapter.DNSTransport
  43. strategy C.DomainStrategy
  44. fallbackDelay time.Duration
  45. }
  46. func NewTransportDialer(dialer N.Dialer, dnsRouter adapter.DNSRouter, transport adapter.DNSTransport, strategy C.DomainStrategy, fallbackDelay time.Duration) *TransportDialer {
  47. return &TransportDialer{
  48. dialer,
  49. dnsRouter,
  50. transport,
  51. strategy,
  52. fallbackDelay,
  53. }
  54. }
  55. func (d *TransportDialer) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
  56. if destination.IsIP() {
  57. return d.dialer.DialContext(ctx, network, destination)
  58. }
  59. addresses, err := d.dnsRouter.Lookup(ctx, destination.Fqdn, adapter.DNSQueryOptions{
  60. Transport: d.transport,
  61. Strategy: d.strategy,
  62. })
  63. if err != nil {
  64. return nil, err
  65. }
  66. return N.DialParallel(ctx, d.dialer, network, destination, addresses, d.strategy == C.DomainStrategyPreferIPv6, d.fallbackDelay)
  67. }
  68. func (d *TransportDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
  69. if destination.IsIP() {
  70. return d.dialer.ListenPacket(ctx, destination)
  71. }
  72. addresses, err := d.dnsRouter.Lookup(ctx, destination.Fqdn, adapter.DNSQueryOptions{
  73. Transport: d.transport,
  74. Strategy: d.strategy,
  75. })
  76. if err != nil {
  77. return nil, err
  78. }
  79. conn, _, err := N.ListenSerial(ctx, d.dialer, destination, addresses)
  80. return conn, err
  81. }
  82. func (d *TransportDialer) Upstream() any {
  83. return d.dialer
  84. }