Current net/http and net/pop use Timeout.timeout to tigger open_timeout event.
Timeout.timeout is slow. It will create and destroy a thread every time.
Timeout.timeout is also dangerous. see [[http://www.mikeperham.com/2015/05/08/timeout-rubys-most-dangerous-api/]]
It is more effective and safe to use socket timeout to accomplish this.
Follow is the changes need to do.
Replace TCPSocket.open with Socket.new
Use socket.connect_nonblock and IO.select to connect and trigger timeout event.
The pull request is here:
[[https://github.com/ruby/ruby/pull/1480]]
net/http, net/pop, net/smtp and net/ftp use Timeout.timeout to calculate connect_timeout.
Timeout.timeout is slow. It creates and destroys a thread every time.
Timeout.timeout is also dangerous. see Timeout: Ruby's Most Dangerous API
I agree with eliminating Timeout, but I don't think your
solution is enough because it does not cover timeouts for
DNS resolution (getaddrinfo(3) calls).
For timeouts, we would need to use resolv.rb instead of getaddrinfo(3)
provided by libc to do timeouts without a separate thread. I started
adding timeouts to resolv.rb last year but can't remember how far I got...
I'm not sure if resolv.rb supports all the features of a modern
getaddrinfo(3), either, AFAIK, not many people use resolv.rb.
It is more effective and safe to use socket timeout to accomplish that.
Follow is the changes need to do.
Replace TCPSocket.open with Socket.tcp
Create TCPSocket with TCPSocket.for_fd
I don't think this step should be necessary; Socket and
TCPSocket should be usable interchangeably for stream sockets
(maybe some API calls need to be changed, but I'd rather avoid
the extra object entirely).
The only complication with trying to use Socket.tcp now would be that it has two separate timeouts that are specified at the same time, but the net stdlib uses a combine open timeout that include DNS resolving and TCP connecting. Preserving that behaviour would mean using the open timeout for DNS resolution, then using the remaining time for the connect timeout. Doing this outside Socket.tcp would mean duplicating a lot of the code in that method.
Should we add a combined timeout option to Socket.tcp` so we can easily use that in the net stdlib's open timeouts?