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:
parent
4b747c2890
commit
e6c0a8f19e
102
lib/resolv.rb
102
lib/resolv.rb
@ -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
|
||||||
|
@ -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
|
||||||
|
Loading…
x
Reference in New Issue
Block a user