فهرست منبع

Working irc write_queue. See input for example.

Steve Thielemann 3 سال پیش
والد
کامیت
60cfb3c804
3فایلهای تغییر یافته به همراه112 افزوده شده و 0 حذف شده
  1. 17 0
      input.cpp
  2. 79 0
      irc.cpp
  3. 16 0
      irc.h

+ 17 - 0
input.cpp

@@ -166,6 +166,23 @@ void parse_input(door::Door &door, ircClient &irc) {
       door << "[ESC] aborts input" << door::nl;
     }
 
+    if (cmd[0] == "/flood") {
+      std::string target = irc.talkto();
+      std::string bugz = "bugz";
+      for (int x = 0; x < 20; ++x) {
+        std::string message = "PRIVMSG " + target +
+                              " : CHANNEL FLOOD TESTING THIS IS MESSAGE " +
+                              std::to_string(x + 1) + " TEST TEST TEST";
+        irc.write_queue(target, message);
+        message = "PRIVMSG " + bugz + " : USER FLOOD TESTING THIS IS MESSAGE " +
+                  std::to_string(x + 1) + " TEST TEST TEST";
+        irc.write_queue(bugz, message);
+        message = "PRIVMSG apollo : USER FLOOD TESTING THIS IS MESSAGE " +
+                  std::to_string(x + 1) + " TEST TEST TEST";
+        irc.write_queue("apollo", message);
+      }
+    }
+
     if (cmd[0] == "/info") {
       irc.channels_lock.lock();
       for (auto c : irc.channels) {

+ 79 - 0
irc.cpp

@@ -108,15 +108,28 @@ using namespace std::placeholders;
 
 typedef std::function<void(std::string &)> receiveFunction;
 
+#ifdef SENDQ
+ircClient::ircClient(boost::asio::io_context &io_context)
+    : resolver{io_context}, ssl_context{boost::asio::ssl::context::tls},
+      socket{io_context, ssl_context},
+      sendq_timer{io_context}, context{io_context} {
+#else
 ircClient::ircClient(boost::asio::io_context &io_context)
     : resolver{io_context}, ssl_context{boost::asio::ssl::context::tls},
       socket{io_context, ssl_context}, context{io_context} {
+#endif
   registered = false;
   nick_retry = 1;
   shutdown = false;
   logging = false;
   channels_updated = false;
   version = "Bugz IRC thing V0.1";
+#ifdef SENDQ
+  sendq_current = 0;
+  sendq_active = false;
+  sendq_ms = 500;
+  sendq_current = 0;
+#endif
 }
 
 std::ofstream &ircClient::log(void) {
@@ -150,6 +163,72 @@ void ircClient::write(std::string output) {
   }
 }
 
+#ifdef SENDQ
+void ircClient::write_queue(std::string target, std::string output) {
+  // is target in sendq_targets
+  bool found = false;
+  for (auto &t : sendq_targets) {
+    if (t == target) {
+      found = true;
+      break;
+    }
+  }
+
+  if (!found) {
+    sendq_targets.push_back(target);
+  };
+
+  sendq[target].push_back(output);
+
+  if (!sendq_active) {
+    sendq_active = true;
+    sendq_current = 0;
+    sendq_timer.expires_after(std::chrono::milliseconds(sendq_ms));
+    sendq_timer.async_wait(std::bind(&ircClient::on_sendq, this, _1));
+  }
+}
+
+void ircClient::on_sendq(error_code error) {
+  std::string &target = sendq_targets[sendq_current];
+
+  // calculate the next target
+  ++sendq_current;
+  if (sendq_current >= (int)sendq_targets.size())
+    sendq_current = 0;
+
+  std::string output = sendq[target].front();
+  sendq[target].erase(sendq[target].begin());
+  write(output);
+
+  if (sendq[target].size() == 0) {
+    // target queue is empty
+    sendq.erase(target);
+
+    // remove target from sendq_targets
+    for (auto pos = sendq_targets.begin(); pos != sendq_targets.end(); ++pos) {
+      // for (int x = 0; x < (int)sendq_targets.size(); ++x) {
+      if (*pos == target) {
+        sendq_targets.erase(pos);
+        break;
+      }
+    }
+
+    // verify the sendq_current is still valid
+    if (sendq_current >= (int)sendq_targets.size())
+      sendq_current = 0;
+  }
+
+  if (!sendq_targets.empty()) {
+    // more to do, let's do it again!
+    sendq_timer.expires_after(std::chrono::milliseconds(sendq_ms));
+    sendq_timer.async_wait(std::bind(&ircClient::on_sendq, this, _1));
+  } else {
+    // let write_queue know we aren't running anymore.
+    sendq_active = false;
+  }
+}
+#endif
+
 /**
  * @brief thread safe messages.push_back
  *

+ 16 - 0
irc.h

@@ -18,6 +18,8 @@
 
 #include <boost/asio/io_context.hpp>
 
+#define SENDQ
+
 void string_toupper(std::string &str);
 
 std::vector<std::string> split_limit(std::string &text, int max = -1);
@@ -45,6 +47,11 @@ public:
 
   // thread-safe write to IRC
   void write(std::string output);
+#ifdef SENDQ
+  // queued writer
+  void write_queue(std::string target, std::string output);
+  void on_sendq(error_code error);
+#endif
 
   // configuration
   std::string hostname;
@@ -130,6 +137,15 @@ private:
   boost::asio::streambuf response;
   boost::asio::ssl::stream<boost::asio::ip::tcp::socket> socket;
 
+#ifdef SENDQ
+  boost::asio::high_resolution_timer sendq_timer;
+  std::map<std::string, std::vector<std::string>> sendq;
+  std::vector<std::string> sendq_targets;
+  int sendq_current;
+  std::atomic<bool> sendq_active;
+  int sendq_ms;
+#endif
+
   boost::asio::io_context &context;
 };