Protect against slow read attack

Right now as long as an attacker continutes to
do a slow read, the connection will stay open forever.
Set a timeout so this can't happen.

Tested: Used slowhttptest to verify this wouldn't
happen

Change-Id: I4dbe2a18f9ccce0ba36875572ec3df6bf3be6a1e
Signed-off-by: James Feist <james.feist@linux.intel.com>
diff --git a/http/http_connection.h b/http/http_connection.h
index 4cd1e89..9f6c7b6 100644
--- a/http/http_connection.h
+++ b/http/http_connection.h
@@ -831,34 +831,39 @@
         }
     }
 
-    void startDeadline()
+    void startDeadline(size_t timerIterations = 0)
     {
+        // drop all connections after 1 minute, this time limit was chosen
+        // arbitrarily and can be adjusted later if needed
+        constexpr const size_t maxReadAttempts =
+            (60 / detail::timerQueueTimeoutSeconds);
+
         cancelDeadlineTimer();
 
-        timerCancelKey =
-            timerQueue.add([this, self(shared_from_this()),
-                            readCount{parser->get().body().size()}] {
-                // Mark timer as not active to avoid canceling it during
-                // Connection destructor which leads to double free issue
-                timerCancelKey.reset();
-                if (!isAlive())
-                {
-                    return;
-                }
+        timerCancelKey = timerQueue.add([this, self(shared_from_this()),
+                                         readCount{parser->get().body().size()},
+                                         timerIterations{timerIterations + 1}] {
+            // Mark timer as not active to avoid canceling it during
+            // Connection destructor which leads to double free issue
+            timerCancelKey.reset();
+            if (!isAlive())
+            {
+                return;
+            }
 
-                // Restart timer if read is in progress.
-                // With threshold can be used to drop slow connections
-                // to protect against slow-rate DoS attack
-                if (parser->get().body().size() > readCount)
-                {
-                    BMCWEB_LOG_DEBUG << this
-                                     << " restart timer - read in progress";
-                    startDeadline();
-                    return;
-                }
+            // Restart timer if read is in progress.
+            // With threshold can be used to drop slow connections
+            // to protect against slow-rate DoS attack
+            if ((parser->get().body().size() > readCount) &&
+                (timerIterations < maxReadAttempts))
+            {
+                BMCWEB_LOG_DEBUG << this << " restart timer - read in progress";
+                startDeadline(timerIterations);
+                return;
+            }
 
-                close();
-            });
+            close();
+        });
         BMCWEB_LOG_DEBUG << this << " timer added: " << &timerQueue << ' '
                          << *timerCancelKey;
     }
diff --git a/http/timer_queue.h b/http/timer_queue.h
index 7339d66..d7a427e 100644
--- a/http/timer_queue.h
+++ b/http/timer_queue.h
@@ -11,6 +11,9 @@
 {
 namespace detail
 {
+
+constexpr const size_t timerQueueTimeoutSeconds = 5;
+
 // fast timer queue for fixed tick value.
 class TimerQueue
 {
@@ -49,7 +52,8 @@
             // remove canceled ones immediately
             if (x.second)
             {
-                if (now - x.first < std::chrono::seconds(5))
+                if (now - x.first <
+                    std::chrono::seconds(timerQueueTimeoutSeconds))
                 {
                     break;
                 }