From edb94ef54d67ef9486239633ebf84ea28fb070bc Mon Sep 17 00:00:00 2001 From: Remi Gacogne Date: Thu, 19 Jun 2025 12:39:11 +0200 Subject: [PATCH] dnsdist: Tentative h2o fix for MadeYouReset Signed-off-by: Remi Gacogne (cherry picked from commit 2406bc461dbac384b07ed7dbb73a87def6354bd6) --- pdns/dnsdistdist/doh.cc | 38 +++++++++++++++++++++++++++++++++++++- 1 file changed, 37 insertions(+), 1 deletion(-) diff --git a/pdns/dnsdistdist/doh.cc b/pdns/dnsdistdist/doh.cc index 5cdf7a3b50..6dfff33223 100644 --- a/pdns/dnsdistdist/doh.cc +++ b/pdns/dnsdistdist/doh.cc @@ -316,6 +316,7 @@ struct DOHConnection struct timeval d_connectionStartTime{0, 0}; size_t d_nbQueries{0}; int d_desc{-1}; + uint8_t d_concurrentStreams{0}; }; static thread_local std::unordered_map t_conns; @@ -389,6 +390,17 @@ static const std::string& getReasonFromStatusCode(uint16_t statusCode) return reasonIt->second; } +static DOHConnection* getConnectionFromQuery(const h2o_req_t* req) +{ + h2o_socket_t* sock = req->conn->callbacks->get_socket(req->conn); + const int descriptor = h2o_socket_get_fd(sock); + if (descriptor == -1) { + /* this should not happen, but let's not crash on it */ + return nullptr; + } + return &t_conns.at(descriptor); +} + /* Always called from the main DoH thread */ static void handleResponse(DOHFrontend& dohFrontend, st_h2o_req_t* req, uint16_t statusCode, const PacketBuffer& response, const std::unordered_map& customResponseHeaders, const std::string& contentType, bool addContentType) { @@ -464,6 +476,10 @@ static void handleResponse(DOHFrontend& dohFrontend, st_h2o_req_t* req, uint16_t ++dohFrontend.d_errorresponses; } + + if (auto* conn = getConnectionFromQuery(req)) { + --conn->d_concurrentStreams; + } } static std::unique_ptr getDUFromIDS(InternalQueryState& ids) @@ -920,6 +936,8 @@ static void on_generator_dispose(void *_self) via a pipe */ static void doh_dispatch_query(DOHServerConfig* dsc, h2o_handler_t* self, h2o_req_t* req, PacketBuffer&& query, const ComboAddress& local, const ComboAddress& remote, std::string&& path) { + auto* conn = getConnectionFromQuery(req); + try { /* we only parse it there as a sanity check, we will parse it again later */ // NOLINTNEXTLINE(cppcoreguidelines-pro-type-reinterpret-cast) @@ -951,6 +969,9 @@ static void doh_dispatch_query(DOHServerConfig* dsc, h2o_handler_t* self, h2o_re } } + if (conn) { + ++conn->d_concurrentStreams; + } #ifdef HAVE_H2O_SOCKET_GET_SSL_SERVER_NAME h2o_socket_t* sock = req->conn->callbacks->get_socket(req->conn); const char * sni = h2o_socket_get_ssl_server_name(sock); @@ -969,17 +990,26 @@ static void doh_dispatch_query(DOHServerConfig* dsc, h2o_handler_t* self, h2o_re ++dnsdist::metrics::g_stats.dohQueryPipeFull; vinfolog("Unable to pass a DoH query to the DoH worker thread because the pipe is full"); h2o_send_error_500(req, "Internal Server Error", "Internal Server Error", 0); + if (conn) { + --conn->d_concurrentStreams; + } } } catch (...) { vinfolog("Unable to pass a DoH query to the DoH worker thread because we couldn't write to the pipe: %s", stringerror()); h2o_send_error_500(req, "Internal Server Error", "Internal Server Error", 0); + if (conn) { + --conn->d_concurrentStreams; + } } #endif /* USE_SINGLE_ACCEPTOR_THREAD */ } catch (const std::exception& e) { vinfolog("Had error parsing DoH DNS packet from %s: %s", remote.toStringWithPort(), e.what()); h2o_send_error_400(req, "Bad Request", "The DNS query could not be parsed", 0); + if (conn) { + --conn->d_concurrentStreams; + } } } @@ -1049,14 +1079,19 @@ static int doh_handler(h2o_handler_t *self, h2o_req_t *req) // NOLINTNEXTLINE(cppcoreguidelines-pro-bounds-pointer-arithmetic): h2o API auto* dsc = static_cast(req->conn->ctx->storage.entries[0].data); h2o_socket_t* sock = req->conn->callbacks->get_socket(req->conn); - const int descriptor = h2o_socket_get_fd(sock); if (descriptor == -1) { return 0; } auto& conn = t_conns.at(descriptor); + if (conn.d_concurrentStreams >= 100U) { + vinfolog("Too many concurrent streams on connection from %d", conn.d_remote.toStringWithPort()); + return 0; + } + ++conn.d_nbQueries; + if (conn.d_nbQueries == 1) { if (h2o_socket_get_ssl_session_reused(sock) == 0) { ++dsc->clientState->tlsNewSessions; @@ -1122,6 +1157,7 @@ static int doh_handler(h2o_handler_t *self, h2o_req_t *req) for (const auto& entry : *responsesMap) { if (entry->matches(path)) { const auto& customHeaders = entry->getHeaders(); + ++conn.d_concurrentStreams; handleResponse(*dsc->dohFrontend, req, entry->getStatusCode(), entry->getContent(), customHeaders ? *customHeaders : dsc->dohFrontend->d_customResponseHeaders, std::string(), false); return 0; } -- 2.47.3