From bf4f45a7c2742431a4d40e12f44e4dd03bf1b0f7 Mon Sep 17 00:00:00 2001 From: Roman Gershman Date: Mon, 9 Dec 2024 18:12:07 +0200 Subject: [PATCH] chore: let resp parser provide more useful logs (#4273) * chore: let resp parser provide more useful logs 1. More warning logs around bad BAD_ARRAYLEN messages 2. Lift the restriction around big bulk strings and log a warning instead. 3. Pull helio Probably fixes #4213 --------- Signed-off-by: Roman Gershman --- helio | 2 +- src/facade/redis_parser.cc | 26 ++++++++++++++++++-------- src/facade/redis_parser.h | 2 -- src/facade/redis_parser_test.cc | 14 +++++++++++++- src/server/server_family.cc | 1 + 5 files changed, 33 insertions(+), 12 deletions(-) diff --git a/helio b/helio index ff9b6cd35bf0..32e8c4ec830e 160000 --- a/helio +++ b/helio @@ -1 +1 @@ -Subproject commit ff9b6cd35bf082a9d48cf0904b0e8557cf31b6d2 +Subproject commit 32e8c4ec830e8d4224d8a13a11c1607f357da80f diff --git a/src/facade/redis_parser.cc b/src/facade/redis_parser.cc index f7ebd2db4beb..2e2967761764 100644 --- a/src/facade/redis_parser.cc +++ b/src/facade/redis_parser.cc @@ -11,6 +11,7 @@ namespace facade { using namespace std; +constexpr static long kMaxBulkLen = 256 * (1ul << 20); // 256MB. auto RedisParser::Parse(Buffer str, uint32_t* consumed, RespExpr::Vec* res) -> Result { DCHECK(!str.empty()); @@ -218,7 +219,11 @@ auto RedisParser::ParseLen(Buffer str, int64_t* res) -> ResultConsumed { const char* s = reinterpret_cast(str.data()); const char* pos = reinterpret_cast(memchr(s, '\n', str.size())); if (!pos) { - Result r = str.size() < 32 ? INPUT_PENDING : BAD_ARRAYLEN; + Result r = INPUT_PENDING; + if (str.size() >= 32) { + LOG(WARNING) << "Unexpected format " << string_view{s, str.size()}; + r = BAD_ARRAYLEN; + } return {r, 0}; } @@ -227,10 +232,16 @@ auto RedisParser::ParseLen(Buffer str, int64_t* res) -> ResultConsumed { } // Skip the first character and 2 last ones (\r\n). - bool success = absl::SimpleAtoi(std::string_view{s + 1, size_t(pos - 1 - s)}, res); + string_view len_token{s + 1, size_t(pos - 1 - s)}; + bool success = absl::SimpleAtoi(len_token, res); + unsigned consumed = pos - s + 1; + if (success && *res >= -1) { + return ResultConsumed{OK, consumed}; + } - return ResultConsumed{success ? OK : BAD_ARRAYLEN, consumed}; + LOG(WARNING) << "Failed to parse len " << len_token; + return ResultConsumed{BAD_ARRAYLEN, consumed}; } auto RedisParser::ConsumeArrayLen(Buffer str) -> ResultConsumed { @@ -247,8 +258,8 @@ auto RedisParser::ConsumeArrayLen(Buffer str) -> ResultConsumed { len *= 2; } - if (len < -1 || len > max_arr_len_) { - LOG_IF(WARNING, len > max_arr_len_) << "Multibulk len is too large " << len; + if (len > max_arr_len_) { + LOG(WARNING) << "Multibulk len is too large " << len; return {BAD_ARRAYLEN, res.second}; } @@ -310,15 +321,14 @@ auto RedisParser::ParseArg(Buffer str) -> ResultConsumed { return res; } - if (len < -1 || len > kMaxBulkLen) - return {BAD_ARRAYLEN, res.second}; - if (len == -1) { // Resp2 NIL cached_expr_->emplace_back(RespExpr::NIL); cached_expr_->back().u = Buffer{}; HandleFinishArg(); } else { DVLOG(1) << "String(" << len << ")"; + LOG_IF(WARNING, len > kMaxBulkLen) << "Large bulk len: " << len; + cached_expr_->emplace_back(RespExpr::STRING); cached_expr_->back().u = Buffer{}; bulk_len_ = len; diff --git a/src/facade/redis_parser.h b/src/facade/redis_parser.h index f9986dd4d37e..e092be06ae98 100644 --- a/src/facade/redis_parser.h +++ b/src/facade/redis_parser.h @@ -22,8 +22,6 @@ namespace facade { */ class RedisParser { public: - constexpr static long kMaxBulkLen = 256 * (1ul << 20); // 256MB. - enum Result : uint8_t { OK, INPUT_PENDING, diff --git a/src/facade/redis_parser_test.cc b/src/facade/redis_parser_test.cc index 29237ca06dae..1bb7864e4310 100644 --- a/src/facade/redis_parser_test.cc +++ b/src/facade/redis_parser_test.cc @@ -170,7 +170,7 @@ TEST_F(RedisParserTest, Empty) { } TEST_F(RedisParserTest, LargeBulk) { - std::string_view prefix("*1\r\n$1024\r\n"); + string_view prefix("*1\r\n$1024\r\n"); ASSERT_EQ(RedisParser::INPUT_PENDING, Parse(prefix)); ASSERT_EQ(prefix.size(), consumed_); @@ -191,6 +191,18 @@ TEST_F(RedisParserTest, LargeBulk) { ASSERT_EQ(RedisParser::INPUT_PENDING, Parse(part1)); ASSERT_EQ(RedisParser::INPUT_PENDING, Parse(half)); ASSERT_EQ(RedisParser::OK, Parse("\r\n")); + + prefix = "*1\r\n$270000000\r\n"; + ASSERT_EQ(RedisParser::INPUT_PENDING, Parse(prefix)); + ASSERT_EQ(prefix.size(), consumed_); + string chunk(1000000, 'a'); + for (unsigned i = 0; i < 270; ++i) { + ASSERT_EQ(RedisParser::INPUT_PENDING, Parse(chunk)); + ASSERT_EQ(chunk.size(), consumed_); + } + ASSERT_EQ(RedisParser::OK, Parse("\r\n")); + ASSERT_THAT(args_, ElementsAre(ArgType(RespExpr::STRING))); + EXPECT_EQ(270000000, args_[0].GetBuf().size()); } TEST_F(RedisParserTest, NILs) { diff --git a/src/server/server_family.cc b/src/server/server_family.cc index 7e9638c10bd1..632409deb461 100644 --- a/src/server/server_family.cc +++ b/src/server/server_family.cc @@ -877,6 +877,7 @@ void ServerFamily::Init(util::AcceptServer* acceptor, std::vector();