Class: Resolv::DNS
| Relationships & Source Files | |
| Namespace Children | |
| Modules: | |
| Classes: | |
| Exceptions: | |
| Extension / Inclusion / Inheritance Descendants | |
| Subclasses: | |
| Inherits: | Object | 
| Defined in: | lib/resolv.rb | 
Overview
DNS is a DNS stub resolver.
Information taken from the following places:
- 
STD0013 
- 
RFC 1035 
- 
etc. 
Constant Summary
- 
    Port =
    # File 'lib/resolv.rb', line 283Default DNS Port53
- 
    RequestID =
    Internal use only
    
 # File 'lib/resolv.rb', line 612{}
- 
    RequestIDMutex =
    Internal use only
    
 # File 'lib/resolv.rb', line 613Thread::Mutex.new 
- 
    UDPSize =
    # File 'lib/resolv.rb', line 288Default DNS UDP packet size 512
Class Method Summary
- 
    
      .new(config_info = nil)  ⇒ DNS 
    
    constructor
    Creates a new DNSresolver.
- 
    
      .open(*args)  
    
    Creates a new DNSresolver.
- .allocate_request_id(host, port) Internal use only
- .bind_random_port(udpsock, bind_host = "0.0.0.0") Internal use only
- .free_request_id(host, port, id) Internal use only
- .random(arg) Internal use only
Instance Attribute Summary
- 
    
      #timeouts=(values)  
    
    writeonly
    Sets the resolver timeouts. 
- #use_ipv6? ⇒ Boolean readonly private Internal use only
Instance Method Summary
- 
    
      #close  
    
    Closes the DNSresolver.
- 
    
      #each_address(name)  
    
    Iterates over all IP addresses for nameretrieved from theDNSresolver.
- 
    
      #each_name(address)  
    
    Iterates over all hostnames for addressretrieved from theDNSresolver.
- 
    
      #each_resource(name, typeclass, &proc)  
    
    Iterates over all typeclassDNSresources forname.
- #fetch_resource(name, typeclass)
- 
    
      #getaddress(name)  
    
    Gets the IP address of namefrom theDNSresolver.
- 
    
      #getaddresses(name)  
    
    Gets all IP addresses for namefrom theDNSresolver.
- 
    
      #getname(address)  
    
    Gets the hostname for addressfrom theDNSresolver.
- 
    
      #getnames(address)  
    
    Gets all hostnames for addressfrom theDNSresolver.
- 
    
      #getresource(name, typeclass)  
    
    Look up the typeclassDNSresource ofname.
- 
    
      #getresources(name, typeclass)  
    
    Looks up all typeclassDNSresources forname.
- #extract_resources(msg, name, typeclass) Internal use only
- #lazy_initialize Internal use only
- #make_tcp_requester(host, port) Internal use only
- #make_udp_requester Internal use only
Constructor Details
    .new(config_info = nil)  ⇒ DNS 
  
Creates a new DNS resolver.
config_info can be:
- nil
- 
Uses /etc/resolv.conf. 
- String
- 
Path to a file using /etc/resolv.conf’s format. 
- Hash
- 
Must contain :nameserver,:searchand:ndotskeys.
:nameserver_port can be used to specify port number of nameserver address.
The value of :nameserver should be an address string or an array of address strings.
- 
:nameserver=> ‘8.8.8.8’
- 
:nameserver=> [‘8.8.8.8’, ‘8.8.4.4’]
The value of :nameserver_port should be an array of pair of nameserver address and port number.
- 
:nameserver_port=> [[‘8.8.8.8’, 53], [‘8.8.4.4’, 53]]
Example:
Resolv::DNS.new(:nameserver => ['210.251.121.21'],
                :search => ['ruby-lang.org'],
                :ndots => 1)Class Method Details
.allocate_request_id(host, port)
# File 'lib/resolv.rb', line 615
def self.allocate_request_id(host, port) # :nodoc: id = nil RequestIDMutex.synchronize { h = (RequestID[[host, port]] ||= {}) begin id = random(0x0000..0xffff) end while h[id] h[id] = true } id end
.bind_random_port(udpsock, bind_host = "0.0.0.0")
# File 'lib/resolv.rb', line 639
def self.bind_random_port(udpsock, bind_host="0.0.0.0") # :nodoc: begin port = random(1024..65535) udpsock.bind(bind_host, port) rescue Errno::EADDRINUSE, # POSIX Errno::EACCES, # SunOS: See PRIV_SYS_NFS in privileges(5) Errno::EPERM # FreeBSD: security.mac.portacl.port_high is configurable. See mac_portacl(4). retry end end
.free_request_id(host, port, id)
# File 'lib/resolv.rb', line 627
def self.free_request_id(host, port, id) # :nodoc: RequestIDMutex.synchronize { key = [host, port] if h = RequestID[key] h.delete id if h.empty? RequestID.delete key end end } end
.open(*args)
Creates a new DNS resolver.  See .new for argument details.
Yields the created DNS resolver to the block, if given, otherwise returns it.
.random(arg)
# File 'lib/resolv.rb', line 599
def self.random(arg) # :nodoc: begin SecureRandom.random_number(arg) rescue NotImplementedError rand(arg) end end
Instance Attribute Details
#timeouts=(values) (writeonly)
Sets the resolver timeouts.  This may be a single positive number or an array of positive numbers representing timeouts in seconds. If an array is specified, a DNS request will retry and wait for each successive interval in the array until a successful response is received.  Specifying nil reverts to the default timeouts:
- 5, second = 5 * 2 / nameserver_count, 2 * second, 4 * second
- 
Example: dns.timeouts = 3
# File 'lib/resolv.rb', line 348
def timeouts=(values) @config.timeouts = values end
    #use_ipv6?  ⇒ Boolean  (readonly, private)
  
  # File 'lib/resolv.rb', line 410
def use_ipv6? # :nodoc: begin list = Socket.ip_address_list rescue NotImplementedError return true end list.any? {|a| a.ipv6? && !a.ipv6_loopback? && !a.ipv6_linklocal? } end
Instance Method Details
#close
Closes the DNS resolver.
# File 'lib/resolv.rb', line 365
def close @mutex.synchronize { if @initialized @initialized = false end } end
#each_address(name)
# File 'lib/resolv.rb', line 403
def each_address(name) each_resource(name, Resource::IN::A) {|resource| yield resource.address} if use_ipv6? each_resource(name, Resource::IN::AAAA) {|resource| yield resource.address} end end
#each_name(address)
# File 'lib/resolv.rb', line 450
def each_name(address) case address when Name ptr = address when IPv4::Regex ptr = IPv4.create(address).to_name when IPv6::Regex ptr = IPv6.create(address).to_name else raise ResolvError.new("cannot interpret as address: #{address}") end each_resource(ptr, Resource::IN::PTR) {|resource| yield resource.name} end
#each_resource(name, typeclass, &proc)
Iterates over all typeclass DNS resources for name.  See #getresource for argument details.
# File 'lib/resolv.rb', line 506
def each_resource(name, typeclass, &proc) fetch_resource(name, typeclass) {|reply, reply_name| extract_resources(reply, reply_name, typeclass, &proc) } end
#extract_resources(msg, name, typeclass)
# File 'lib/resolv.rb', line 567
def extract_resources(msg, name, typeclass) # :nodoc: if typeclass < Resource::ANY n0 = Name.create(name) msg.each_resource {|n, ttl, data| yield data if n0 == n } end yielded = false n0 = Name.create(name) msg.each_resource {|n, ttl, data| if n0 == n case data when typeclass yield data yielded = true when Resource::CNAME n0 = data.name end end } return if yielded msg.each_resource {|n, ttl, data| if n0 == n case data when typeclass yield data end end } end
#fetch_resource(name, typeclass)
[ GitHub ]# File 'lib/resolv.rb', line 512
def fetch_resource(name, typeclass) lazy_initialize requester = make_udp_requester senders = {} begin @config.resolv(name) {|candidate, tout, nameserver, port| msg = Message.new msg.rd = 1 msg.add_question(candidate, typeclass) unless sender = senders[[candidate, nameserver, port]] sender = requester.sender(msg, candidate, nameserver, port) next if !sender senders[[candidate, nameserver, port]] = sender end reply, reply_name = requester.request(sender, tout) case reply.rcode when RCode::NoError if reply.tc == 1 and not Requester::TCP === requester requester.close # Retry via TCP: requester = make_tcp_requester(nameserver, port) senders = {} # This will use TCP for all remaining candidates (assuming the # current candidate does not already respond successfully via # TCP). This makes sense because we already know the full # response will not fit in an untruncated UDP packet. redo else yield(reply, reply_name) end 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
#getaddress(name)
# File 'lib/resolv.rb', line 379
def getaddress(name) each_address(name) {|address| return address} raise ResolvError.new("DNS result has no information for #{name}") end
#getaddresses(name)
# File 'lib/resolv.rb', line 390
def getaddresses(name) ret = [] each_address(name) {|address| ret << address} return ret end
#getname(address)
# File 'lib/resolv.rb', line 426
def getname(address) each_name(address) {|name| return name} raise ResolvError.new("DNS result has no information for #{address}") end
#getnames(address)
# File 'lib/resolv.rb', line 437
def getnames(address) ret = [] each_name(address) {|name| ret << name} return ret end
#getresource(name, typeclass)
Look up the typeclass DNS resource of name.
name must be a DNS::Name or a String.
typeclass should be one of the following:
- 
Resolv::DNS::Resource::IN::ANY
- 
Resolv::DNS::Resource::IN::CNAME
- 
Resolv::DNS::Resource::IN::HINFO
- 
Resolv::DNS::Resource::IN::MINFO
- 
Resolv::DNS::Resource::IN::MX
- 
Resolv::DNS::Resource::IN::NS
- 
Resolv::DNS::Resource::IN::PTR
- 
Resolv::DNS::Resource::IN::SOA
- 
Resolv::DNS::Resource::IN::TXT
Returned resource is represented as a DNS::Resource instance, i.e. DNS::Resource::IN::A.
# File 'lib/resolv.rb', line 487
def getresource(name, typeclass) each_resource(name, typeclass) {|resource| return resource} raise ResolvError.new("DNS result has no information for #{name}") end
#getresources(name, typeclass)
Looks up all typeclass DNS resources for name.  See #getresource for argument details.
# File 'lib/resolv.rb', line 496
def getresources(name, typeclass) ret = [] each_resource(name, typeclass) {|resource| ret << resource} return ret end
#lazy_initialize
# File 'lib/resolv.rb', line 352
def lazy_initialize # :nodoc: @mutex.synchronize { unless @initialized @config.lazy_initialize @initialized = true end } self end
#make_tcp_requester(host, port)
#make_udp_requester
# File 'lib/resolv.rb', line 554
def make_udp_requester # :nodoc: nameserver_port = @config.nameserver_port if nameserver_port.length == 1 Requester::ConnectedUDP.new(*nameserver_port[0]) else Requester::UnconnectedUDP.new(*nameserver_port) end end