Skip to content

Commit

Permalink
rescue IO::TimeoutError raised by Ruby since 3.2.0 on blocking reads/…
Browse files Browse the repository at this point in the history
…writes (#973)

* rescue IO::TimeoutError raised by Ruby since 3.2.0 on blocking reads/writes

One scenario in which this happens is when a firewall silently drops
connections between clients and memcached instances.

With this patch the client reopens the connection and the request succeeds.
Without it, IO:TimeoutError is raised for the application to handle which
deviates from Dalli behavior under Ruby 3.1 and earlier.

* trying to make Rubocop happy

* Shift require to address lint

* Add changelog entry

---------

Co-authored-by: Stefan Kaes <[email protected]>
  • Loading branch information
petergoldstein and skaes authored Sep 25, 2023
1 parent d61dbcc commit 868f7a7
Show file tree
Hide file tree
Showing 4 changed files with 16 additions and 4 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ Dalli Changelog
Unreleased
==========

- Rescue IO::TimeoutError raised by Ruby since 3.2.0 on blocking reads/writes (skaes)

3.2.5
==========
Expand Down
11 changes: 11 additions & 0 deletions lib/dalli/protocol.rb
Original file line number Diff line number Diff line change
@@ -1,8 +1,19 @@
# frozen_string_literal: true

require 'timeout'

module Dalli
module Protocol
# Preserved for backwards compatibility. Should be removed in 4.0
NOT_FOUND = ::Dalli::NOT_FOUND

# Ruby 3.2 raises IO::TimeoutError on blocking reads/writes, but
# it is not defined in earlier Ruby versions.
TIMEOUT_ERRORS =
if defined?(IO::TimeoutError)
[Timeout::Error, IO::TimeoutError]
else
[Timeout::Error]
end
end
end
2 changes: 1 addition & 1 deletion lib/dalli/protocol/base.rb
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,7 @@ def pipeline_next_responses
end

values
rescue SystemCallError, Timeout::Error, EOFError => e
rescue SystemCallError, *TIMEOUT_ERRORS, EOFError => e
@connection_manager.error_on_request!(e)
end

Expand Down
6 changes: 3 additions & 3 deletions lib/dalli/protocol/connection_manager.rb
Original file line number Diff line number Diff line change
Expand Up @@ -150,19 +150,19 @@ def read_line
data = @sock.gets("\r\n")
error_on_request!('EOF in read_line') if data.nil?
data
rescue SystemCallError, Timeout::Error, EOFError => e
rescue SystemCallError, *TIMEOUT_ERRORS, EOFError => e
error_on_request!(e)
end

def read(count)
@sock.readfull(count)
rescue SystemCallError, Timeout::Error, EOFError => e
rescue SystemCallError, *TIMEOUT_ERRORS, EOFError => e
error_on_request!(e)
end

def write(bytes)
@sock.write(bytes)
rescue SystemCallError, Timeout::Error => e
rescue SystemCallError, *TIMEOUT_ERRORS => e
error_on_request!(e)
end

Expand Down

0 comments on commit 868f7a7

Please sign in to comment.