|
| 1 | +From a21a3b7d23704a01d34bd79d09dc37897e00922a Mon Sep 17 00:00:00 2001 |
| 2 | +From: Yusuke Endoh < [email protected]> |
| 3 | +Date: Wed, 7 Jul 2021 12:06:44 +0900 |
| 4 | +Subject: [PATCH] Fix StartTLS stripping vulnerability |
| 5 | + |
| 6 | +Reported by Alexandr Savca in https://hackerone.com/reports/1178562 |
| 7 | + |
| 8 | +Co-authored-by: Shugo Maeda < [email protected]> |
| 9 | +--- |
| 10 | + lib/net/imap.rb | 8 +++++++- |
| 11 | + test/net/imap/test_imap.rb | 31 +++++++++++++++++++++++++++++++ |
| 12 | + version.h | 2 +- |
| 13 | + 3 files changed, 39 insertions(+), 2 deletions(-) |
| 14 | + |
| 15 | +diff --git a/lib/net/imap.rb b/lib/net/imap.rb |
| 16 | +index 1c7e89b..91df89b 100644 |
| 17 | +--- a/lib/net/imap.rb |
| 18 | ++++ b/lib/net/imap.rb |
| 19 | +@@ -1215,12 +1215,14 @@ module Net |
| 20 | + end |
| 21 | + resp = @tagged_responses.delete(tag) |
| 22 | + case resp.name |
| 23 | ++ when /\A(?:OK)\z/ni |
| 24 | ++ return resp |
| 25 | + when /\A(?:NO)\z/ni |
| 26 | + raise NoResponseError, resp |
| 27 | + when /\A(?:BAD)\z/ni |
| 28 | + raise BadResponseError, resp |
| 29 | + else |
| 30 | +- return resp |
| 31 | ++ raise UnknownResponseError, resp |
| 32 | + end |
| 33 | + end |
| 34 | + |
| 35 | +@@ -3716,6 +3718,10 @@ module Net |
| 36 | + class ByeResponseError < ResponseError |
| 37 | + end |
| 38 | + |
| 39 | ++ # Error raised upon an unknown response from the server. |
| 40 | ++ class UnknownResponseError < ResponseError |
| 41 | ++ end |
| 42 | ++ |
| 43 | + RESPONSE_ERRORS = Hash.new(ResponseError) |
| 44 | + RESPONSE_ERRORS["NO"] = NoResponseError |
| 45 | + RESPONSE_ERRORS["BAD"] = BadResponseError |
| 46 | +diff --git a/test/net/imap/test_imap.rb b/test/net/imap/test_imap.rb |
| 47 | +index 936f4e0..217b611 100644 |
| 48 | +--- a/test/net/imap/test_imap.rb |
| 49 | ++++ b/test/net/imap/test_imap.rb |
| 50 | +@@ -127,6 +127,16 @@ class IMAPTest < Test::Unit::TestCase |
| 51 | + imap.disconnect |
| 52 | + end |
| 53 | + end |
| 54 | ++ |
| 55 | ++ def test_starttls_stripping |
| 56 | ++ starttls_stripping_test do |port| |
| 57 | ++ imap = Net::IMAP.new("localhost", :port => port) |
| 58 | ++ assert_raise(Net::IMAP::UnknownResponseError) do |
| 59 | ++ imap.starttls(:ca_file => CA_FILE) |
| 60 | ++ end |
| 61 | ++ imap |
| 62 | ++ end |
| 63 | ++ end |
| 64 | + end |
| 65 | + |
| 66 | + def test_unexpected_eof |
| 67 | +@@ -762,6 +772,27 @@ EOF |
| 68 | + end |
| 69 | + end |
| 70 | + |
| 71 | ++ def starttls_stripping_test |
| 72 | ++ server = create_tcp_server |
| 73 | ++ port = server.addr[1] |
| 74 | ++ start_server do |
| 75 | ++ sock = server.accept |
| 76 | ++ begin |
| 77 | ++ sock.print("* OK test server\r\n") |
| 78 | ++ sock.gets |
| 79 | ++ sock.print("RUBY0001 BUG unhandled command\r\n") |
| 80 | ++ ensure |
| 81 | ++ sock.close |
| 82 | ++ server.close |
| 83 | ++ end |
| 84 | ++ end |
| 85 | ++ begin |
| 86 | ++ imap = yield(port) |
| 87 | ++ ensure |
| 88 | ++ imap.disconnect if imap && !imap.disconnected? |
| 89 | ++ end |
| 90 | ++ end |
| 91 | ++ |
| 92 | + def create_tcp_server |
| 93 | + return TCPServer.new(server_addr, 0) |
| 94 | + end |
| 95 | +diff --git a/version.h b/version.h |
| 96 | +index 1c491eb..2f4fcdf 100644 |
| 97 | +--- a/version.h |
| 98 | ++++ b/version.h |
| 99 | +@@ -1,6 +1,6 @@ |
| 100 | + #define RUBY_VERSION "2.6.7" |
| 101 | + #define RUBY_RELEASE_DATE "2021-04-05" |
| 102 | +-#define RUBY_PATCHLEVEL 197 |
| 103 | ++#define RUBY_PATCHLEVEL 198 |
| 104 | + |
| 105 | + #define RUBY_RELEASE_YEAR 2021 |
| 106 | + #define RUBY_RELEASE_MONTH 4 |
| 107 | +-- |
| 108 | +2.17.1 |
0 commit comments