2

Resolv :: DNSを使用するときにTCPを強制する方法を知っている人はいますか?
レコードを要求するANYと、出力が切り捨てられ、部分的な結果が得られるようです。多くのクエリ(レコードタイプごとに1つ)を実行すると、より多くの結果が得られます。また、一貫性のない結果が得られます(マシン間で異なり、2つの連続したクエリが異なる結果を返します...)

UDPがパケットサイズに制限されていることと関係があるのではないかと思いました。

TCPを使用するように強制する方法はありますか?使用できる他のDNSパッケージはありますか?

4

1 に答える 1

2

Resolv非常に大きな結果セットを期待していたので、TCPのみのクエリに使用したいという同じ問題がありました。最終的にResolvのソースコードを調べてみたところ、デフォルトでは、TCPクエリはUDPクエリが失敗した場合にのみ実行されることがわかりました。Resolv::DNSメソッドをサブクラス化してオーバーライドできることがわかりましたeach_resource。これが私の情報源です:

require 'resolv'

# A TCP-only resolver built from `Resolv::DNS`. See the docs for what it's about.
# http://ruby-doc.org/stdlib-1.9.3/libdoc/resolv/rdoc/Resolv/DNS.html
class TcpDNS < Resolv::DNS
  # Override fetch_resource to use a TCP requester instead of a UDP requester. This
  # is mostly borrowed from `lib/resolv.rb` with the UDP->TCP fallback logic removed.
  def each_resource(name, typeclass, &proc)
    lazy_initialize
    senders = {}
    requester = nil
    begin
      @config.resolv(name) { |candidate, tout, nameserver, port|
        requester = make_tcp_requester(nameserver, port)
        msg = Message.new
        msg.rd = 1
        msg.add_question(candidate, typeclass)
        unless sender = senders[[candidate, nameserver, port]]
          sender = senders[[candidate, nameserver, port]] =
            requester.sender(msg, candidate, nameserver, port)
        end

        begin # HACK
          reply, reply_name = requester.request(sender, tout)
        rescue
          return
        end

        case reply.rcode
        when RCode::NoError
          extract_resources(reply, reply_name, typeclass, &proc)
          return
        when RCode::NXDomain
          raise Config::NXDomain.new(reply_name.to_s)
        else
          raise Config::OtherResolvError.new(reply_name.to_s)
        end
      }
    ensure
      requester.close
    end
  end
end

次に、それを使用するのは次のように簡単です。

TcpDNS.open :nameserver => ns_addrs, :search => '', :ndots => 1 do |dns|
  resp = dns.getresources target, Resolv::DNS::Resource::IN::ANY
end
于 2013-08-16T06:17:55.820 に答える