resolv.rb: close socket

* lib/resolv.rb (UnconnectedUDP#lazy_initialize): store new
  sockets before binding, so the sockets get closed when the
  requester is closing.

* lib/resolv.rb (ConnectedUDP#lazy_initialize): ditto.

* lib/resolv.rb (UnconnectedUDP#close): synchronize to get rid of
  race condition.

* lib/resolv.rb (ConnectedUDP#close): ditto.

[ruby-core:85901] [Bug #14571]

From: quixoten (Devin Christensen) <quixoten@gmail.com>

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@62671 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
nobu 2018-03-06 02:48:17 +00:00
parent 4b747c2890
commit e6c0a8f19e
2 changed files with 88 additions and 33 deletions

View File

@ -732,35 +732,47 @@ class Resolv
def initialize(*nameserver_port) def initialize(*nameserver_port)
super() super()
@nameserver_port = nameserver_port @nameserver_port = nameserver_port
@socks_hash = {} @initialized = false
@socks = [] @mutex = Thread::Mutex.new
nameserver_port.each {|host, port| end
if host.index(':')
bind_host = "::" def lazy_initialize
af = Socket::AF_INET6 @mutex.synchronize {
else next if @initialized
bind_host = "0.0.0.0" @initialized = true
af = Socket::AF_INET @socks_hash = {}
end @socks = []
next if @socks_hash[bind_host] @nameserver_port.each {|host, port|
begin if host.index(':')
sock = UDPSocket.new(af) bind_host = "::"
rescue Errno::EAFNOSUPPORT af = Socket::AF_INET6
next # The kernel doesn't support the address family. else
end bind_host = "0.0.0.0"
sock.do_not_reverse_lookup = true af = Socket::AF_INET
DNS.bind_random_port(sock, bind_host) end
@socks << sock next if @socks_hash[bind_host]
@socks_hash[bind_host] = sock begin
sock = UDPSocket.new(af)
rescue Errno::EAFNOSUPPORT
next # The kernel doesn't support the address family.
end
@socks << sock
@socks_hash[bind_host] = sock
sock.do_not_reverse_lookup = true
DNS.bind_random_port(sock, bind_host)
}
} }
self
end end
def recv_reply(readable_socks) def recv_reply(readable_socks)
lazy_initialize
reply, from = readable_socks[0].recvfrom(UDPSize) reply, from = readable_socks[0].recvfrom(UDPSize)
return reply, [from[3],from[1]] return reply, [from[3],from[1]]
end end
def sender(msg, data, host, port=Port) def sender(msg, data, host, port=Port)
lazy_initialize
sock = @socks_hash[host.index(':') ? "::" : "0.0.0.0"] sock = @socks_hash[host.index(':') ? "::" : "0.0.0.0"]
return nil if !sock return nil if !sock
service = [host, port] service = [host, port]
@ -772,9 +784,14 @@ class Resolv
end end
def close def close
super @mutex.synchronize {
@senders.each_key {|service, id| if @initialized
DNS.free_request_id(service[0], service[1], id) super
@senders.each_key {|service, id|
DNS.free_request_id(service[0], service[1], id)
}
@initialized = false
end
} }
end end
@ -798,20 +815,32 @@ class Resolv
super() super()
@host = host @host = host
@port = port @port = port
is_ipv6 = host.index(':') @mutex = Thread::Mutex.new
sock = UDPSocket.new(is_ipv6 ? Socket::AF_INET6 : Socket::AF_INET) @initialized = false
@socks = [sock] end
sock.do_not_reverse_lookup = true
DNS.bind_random_port(sock, is_ipv6 ? "::" : "0.0.0.0") def lazy_initialize
sock.connect(host, port) @mutex.synchronize {
next if @initialized
@initialized = true
is_ipv6 = @host.index(':')
sock = UDPSocket.new(is_ipv6 ? Socket::AF_INET6 : Socket::AF_INET)
@socks = [sock]
sock.do_not_reverse_lookup = true
DNS.bind_random_port(sock, is_ipv6 ? "::" : "0.0.0.0")
sock.connect(@host, @port)
}
self
end end
def recv_reply(readable_socks) def recv_reply(readable_socks)
lazy_initialize
reply = readable_socks[0].recv(UDPSize) reply = readable_socks[0].recv(UDPSize)
return reply, nil return reply, nil
end end
def sender(msg, data, host=@host, port=@port) def sender(msg, data, host=@host, port=@port)
lazy_initialize
unless host == @host && port == @port unless host == @host && port == @port
raise RequestError.new("host/port don't match: #{host}:#{port}") raise RequestError.new("host/port don't match: #{host}:#{port}")
end end
@ -822,10 +851,15 @@ class Resolv
end end
def close def close
super @mutex.synchronize do
@senders.each_key {|from, id| if @initialized
DNS.free_request_id(@host, @port, id) super
} @senders.each_key {|from, id|
DNS.free_request_id(@host, @port, id)
}
@initialized = false
end
end
end end
class Sender < Requester::Sender # :nodoc: class Sender < Requester::Sender # :nodoc:
@ -839,6 +873,7 @@ class Resolv
class MDNSOneShot < UnconnectedUDP # :nodoc: class MDNSOneShot < UnconnectedUDP # :nodoc:
def sender(msg, data, host, port=Port) def sender(msg, data, host, port=Port)
lazy_initialize
id = DNS.allocate_request_id(host, port) id = DNS.allocate_request_id(host, port)
request = msg.encode request = msg.encode
request[0,2] = [id].pack('n') request[0,2] = [id].pack('n')
@ -848,6 +883,7 @@ class Resolv
end end
def sender_for(addr, msg) def sender_for(addr, msg)
lazy_initialize
@senders[msg.id] @senders[msg.id]
end end
end end

View File

@ -3,6 +3,7 @@ require 'test/unit'
require 'resolv' require 'resolv'
require 'socket' require 'socket'
require 'tempfile' require 'tempfile'
require 'minitest/mock'
class TestResolvDNS < Test::Unit::TestCase class TestResolvDNS < Test::Unit::TestCase
def setup def setup
@ -246,4 +247,22 @@ class TestResolvDNS < Test::Unit::TestCase
} }
assert_operator(2**14, :<, m.to_s.length) assert_operator(2**14, :<, m.to_s.length)
end end
def assert_no_fd_leak
socket = assert_throw(self) do |tag|
Resolv::DNS.stub(:bind_random_port, ->(s, *) {throw(tag, s)}) do
yield.getname("8.8.8.8")
end
end
assert_predicate(socket, :closed?, "file descriptor leaked")
end
def test_no_fd_leak_connected
assert_no_fd_leak {Resolv::DNS.new(nameserver_port: [['127.0.0.1', 53]])}
end
def test_no_fd_leak_unconnected
assert_no_fd_leak {Resolv::DNS.new}
end
end end