123456789_123456789_123456789_123456789_123456789_

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:

Constant Summary

Class Method Summary

Instance Attribute Summary

Instance Method Summary

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, :search and :ndots keys.

:nameserver_port can be used to specify port number of nameserver address. :raise_timeout_errors can be used to raise timeout errors as exceptions instead of treating the same as an NXDOMAIN response.

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)
[ GitHub ]

  
# File 'lib/resolv.rb', line 335

def initialize(config_info=nil)
  @mutex = Thread::Mutex.new
  @config = Config.new(config_info)
  @initialized = nil
end

Class Method Details

.allocate_request_id(host, port)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 638

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")

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 662

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)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 650

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.

[ GitHub ]

  
# File 'lib/resolv.rb', line 298

def self.open(*args)
  dns = new(*args)
  return dns unless block_given?
  begin
    yield dns
  ensure
    dns.close
  end
end

.random(arg)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 627

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
[ GitHub ]

  
# File 'lib/resolv.rb', line 352

def timeouts=(values)
  @config.timeouts = values
end

#use_ipv6?Boolean (readonly, private)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 414

def use_ipv6? # :nodoc:
  @config.lazy_initialize unless @config.instance_variable_get(:@initialized)

  use_ipv6 = @config.use_ipv6?
  unless use_ipv6.nil?
    return use_ipv6
  end

  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.

[ GitHub ]

  
# File 'lib/resolv.rb', line 369

def close
  @mutex.synchronize {
    if @initialized
      @initialized = false
    end
  }
end

#each_address(name)

Iterates over all IP addresses for name retrieved from the DNS resolver.

name can be a DNS::Name or a String. Retrieved addresses will be a IPv4 or IPv6

[ GitHub ]

  
# File 'lib/resolv.rb', line 407

def each_address(name)
  if use_ipv6?
    each_resource(name, Resource::IN::AAAA) {|resource| yield resource.address}
  end
  each_resource(name, Resource::IN::A) {|resource| yield resource.address}
end

#each_name(address)

Iterates over all hostnames for address retrieved from the DNS resolver.

address must be a IPv4, IPv6 or a String. Retrieved names will be DNS::Name instances.

[ GitHub ]

  
# File 'lib/resolv.rb', line 461

def each_name(address)
  case address
  when Name
    ptr = address
  when IPv4, IPv6
    ptr = address.to_name
  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.

[ GitHub ]

  
# File 'lib/resolv.rb', line 519

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)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 596

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 525

def fetch_resource(name, typeclass)
  lazy_initialize
  truncated = {}
  requesters = {}
  udp_requester = begin
    make_udp_requester
  rescue Errno::EACCES
    # fall back to TCP
  end
  senders = {}

  begin
    @config.resolv(name) do |candidate, tout, nameserver, port|
      msg = Message.new
      msg.rd = 1
      msg.add_question(candidate, typeclass)

      requester = requesters.fetch([nameserver, port]) do
        if !truncated[candidate] && udp_requester
          udp_requester
        else
          requesters[[nameserver, port]] = make_tcp_requester(nameserver, port)
        end
      end

      unless sender = senders[[candidate, requester, nameserver, port]]
        sender = requester.sender(msg, candidate, nameserver, port)
        next if !sender
        senders[[candidate, requester, 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
          # Retry via TCP:
          truncated[candidate] = true
          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
    end
  ensure
    udp_requester&.close
    requesters.each_value { |requester| requester&.close }
  end
end

#getaddress(name)

Gets the IP address of name from the DNS resolver.

name can be a DNS::Name or a String. Retrieved address will be a IPv4 or IPv6

Raises:

[ GitHub ]

  
# File 'lib/resolv.rb', line 383

def getaddress(name)
  each_address(name) {|address| return address}
  raise ResolvError.new("DNS result has no information for #{name}")
end

#getaddresses(name)

Gets all IP addresses for name from the DNS resolver.

name can be a DNS::Name or a String. Retrieved addresses will be a IPv4 or IPv6

[ GitHub ]

  
# File 'lib/resolv.rb', line 394

def getaddresses(name)
  ret = []
  each_address(name) {|address| ret << address}
  return ret
end

#getname(address)

Gets the hostname for address from the DNS resolver.

address must be a IPv4, IPv6 or a String. Retrieved name will be a DNS::Name.

Raises:

[ GitHub ]

  
# File 'lib/resolv.rb', line 437

def getname(address)
  each_name(address) {|name| return name}
  raise ResolvError.new("DNS result has no information for #{address}")
end

#getnames(address)

Gets all hostnames for address from the DNS resolver.

address must be a IPv4, IPv6 or a String. Retrieved names will be DNS::Name instances.

[ GitHub ]

  
# File 'lib/resolv.rb', line 448

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:

  • DNS::Resource::IN::A

  • DNS::Resource::IN::AAAA

  • 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

  • DNS::Resource::IN::WKS

Returned resource is represented as a DNS::Resource instance, i.e. DNS::Resource::IN::A.

Raises:

[ GitHub ]

  
# File 'lib/resolv.rb', line 500

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.

[ GitHub ]

  
# File 'lib/resolv.rb', line 509

def getresources(name, typeclass)
  ret = []
  each_resource(name, typeclass) {|resource| ret << resource}
  return ret
end

#lazy_initialize

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 356

def lazy_initialize # :nodoc:
  @mutex.synchronize {
    unless @initialized
      @config.lazy_initialize
      @initialized = true
    end
  }
  self
end

#make_tcp_requester(host, port)

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 587

def make_tcp_requester(host, port) # :nodoc:
  return Requester::TCP.new(host, port)
rescue Errno::ECONNREFUSED
  # Treat a refused TCP connection attempt to a nameserver like a timeout,
  # as Resolv::DNS::Config#resolv considers ResolvTimeout exceptions as a
  # hint to try the next nameserver:
  raise ResolvTimeout
end

#make_udp_requester

This method is for internal use only.
[ GitHub ]

  
# File 'lib/resolv.rb', line 578

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