Class: Bones::RPC::DNSResolver

Inherits:
Object
  • Object
show all
Defined in:
lib/bones/rpc/dns_resolver.rb

Constant Summary collapse

RESOLV_CONF =
'/etc/resolv.conf'
DNS_PORT =
53

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initializeDNSResolver

Returns a new instance of DNSResolver.



20
21
22
23
24
25
26
27
# File 'lib/bones/rpc/dns_resolver.rb', line 20

def initialize
  @nameservers = self.class.nameservers

  # TODO: fall back on other nameservers if the first one is unavailable
  @server = @nameservers.first

  @socket = UDPSocket.new
end

Class Method Details

.generate_idObject



12
13
14
# File 'lib/bones/rpc/dns_resolver.rb', line 12

def self.generate_id
  @mutex.synchronize { @identifier = (@identifier + 1) & 0xFFFF }
end

.nameservers(config = RESOLV_CONF) ⇒ Object



16
17
18
# File 'lib/bones/rpc/dns_resolver.rb', line 16

def self.nameservers(config = RESOLV_CONF)
  File.read(config).scan(/^\s*nameserver\s+([0-9.:]+)/).flatten
end

Instance Method Details

#resolve(hostname) ⇒ Object



29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
# File 'lib/bones/rpc/dns_resolver.rb', line 29

def resolve(hostname)
  if host = resolve_hostname(hostname)
    unless ip_address = resolve_host(host)
      raise Resolv::ResolvError, "invalid entry in hosts file: #{host}"
    end
    return ip_address
  end

  query = build_query(hostname)
  @socket.send query.encode, 0, @server, DNS_PORT
  data, _ = @socket.recvfrom(512)
  response = Resolv::DNS::Message.decode(data)

  addrs = []
  # The answer might include IN::CNAME entries so filters them out
  # to include IN::A & IN::AAAA entries only.
  response.each_answer { |name, ttl, value| addrs << value.address if value.respond_to?(:address) }

  return if addrs.empty?
  return addrs.first if addrs.size == 1
  addrs
end