transport_dialer.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  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. options.UDPFragmentDefault = true
  20. return dialer.NewWithOptions(dialer.Options{
  21. Context: ctx,
  22. Options: options.DialerOptions,
  23. DirectResolver: true,
  24. LegacyDNSDialer: options.Legacy,
  25. })
  26. }
  27. }
  28. func NewRemoteDialer(ctx context.Context, options option.RemoteDNSServerOptions) (N.Dialer, error) {
  29. if options.LegacyDefaultDialer {
  30. transportDialer := dialer.NewDefaultOutbound(ctx)
  31. if options.LegacyAddressResolver != "" {
  32. transport := service.FromContext[adapter.DNSTransportManager](ctx)
  33. resolverTransport, loaded := transport.Transport(options.LegacyAddressResolver)
  34. if !loaded {
  35. return nil, E.New("address resolver not found: ", options.LegacyAddressResolver)
  36. }
  37. transportDialer = newTransportDialer(transportDialer, service.FromContext[adapter.DNSRouter](ctx), resolverTransport, C.DomainStrategy(options.LegacyAddressStrategy), time.Duration(options.LegacyAddressFallbackDelay))
  38. } else if options.ServerIsDomain() {
  39. return nil, E.New("missing address resolver for server: ", options.Server)
  40. }
  41. return transportDialer, nil
  42. } else {
  43. return dialer.NewWithOptions(dialer.Options{
  44. Context: ctx,
  45. Options: options.DialerOptions,
  46. RemoteIsDomain: options.ServerIsDomain(),
  47. DirectResolver: true,
  48. LegacyDNSDialer: options.Legacy,
  49. })
  50. }
  51. }
  52. type legacyTransportDialer struct {
  53. dialer N.Dialer
  54. dnsRouter adapter.DNSRouter
  55. transport adapter.DNSTransport
  56. strategy C.DomainStrategy
  57. fallbackDelay time.Duration
  58. }
  59. func newTransportDialer(dialer N.Dialer, dnsRouter adapter.DNSRouter, transport adapter.DNSTransport, strategy C.DomainStrategy, fallbackDelay time.Duration) *legacyTransportDialer {
  60. return &legacyTransportDialer{
  61. dialer,
  62. dnsRouter,
  63. transport,
  64. strategy,
  65. fallbackDelay,
  66. }
  67. }
  68. func (d *legacyTransportDialer) DialContext(ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
  69. if destination.IsIP() {
  70. return d.dialer.DialContext(ctx, network, 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. return N.DialParallel(ctx, d.dialer, network, destination, addresses, d.strategy == C.DomainStrategyPreferIPv6, d.fallbackDelay)
  80. }
  81. func (d *legacyTransportDialer) ListenPacket(ctx context.Context, destination M.Socksaddr) (net.PacketConn, error) {
  82. if destination.IsIP() {
  83. return d.dialer.ListenPacket(ctx, destination)
  84. }
  85. addresses, err := d.dnsRouter.Lookup(ctx, destination.Fqdn, adapter.DNSQueryOptions{
  86. Transport: d.transport,
  87. Strategy: d.strategy,
  88. })
  89. if err != nil {
  90. return nil, err
  91. }
  92. conn, _, err := N.ListenSerial(ctx, d.dialer, destination, addresses)
  93. return conn, err
  94. }
  95. func (d *legacyTransportDialer) Upstream() any {
  96. return d.dialer
  97. }