irc.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577
  1. #include "irc.h"
  2. #include <boost/algorithm/string.hpp>
  3. #include <iostream>
  4. void string_toupper(std::string &str) {
  5. std::transform(str.begin(), str.end(), str.begin(), ::toupper);
  6. }
  7. /**
  8. * @brief remove channel modes (op,voice,hop,...)
  9. *
  10. * @param nick
  11. */
  12. void remove_channel_modes(std::string &nick) {
  13. // ~&@%+
  14. std::string remove("~&@%+");
  15. std::string::size_type pos;
  16. do {
  17. pos = remove.find(nick[0]);
  18. if (pos != std::string::npos)
  19. nick.erase(0, 1);
  20. } while (pos != std::string::npos);
  21. }
  22. /**
  23. * @brief split on spaces, with limit
  24. *
  25. * max is the maximum number of splits we will do.
  26. * default -1 is split all.
  27. *
  28. * "this is a test", 3 => [this][is][a test]
  29. *
  30. * @param text
  31. * @param max
  32. * @return std::vector<std::string>
  33. */
  34. std::vector<std::string> split_limit(std::string &text, int max) {
  35. std::vector<std::string> ret;
  36. int t = 0;
  37. boost::split(ret, text, [&t, max](char c) {
  38. if (c == ' ') {
  39. ++t;
  40. return ((max == -1) or (t < max));
  41. };
  42. return false;
  43. });
  44. return ret;
  45. }
  46. /**
  47. * @brief irc split
  48. *
  49. * If it doesn't start with a ':', split into two parts.
  50. * Otherwise 4 parts
  51. * [from] [command] [to] [message]
  52. *
  53. * @param text
  54. * @return std::vector<std::string>
  55. */
  56. std::vector<std::string> irc_split(std::string &text) {
  57. if (text[0] != ':')
  58. return split_limit(text, 2);
  59. return split_limit(text, 4);
  60. }
  61. /**
  62. * @brief parse_nick
  63. *
  64. * Parse out the nick from nick!username@host
  65. *
  66. * @param name
  67. * @return std::string
  68. */
  69. std::string parse_nick(std::string &name) {
  70. std::string to = name;
  71. if (to[0] == ':')
  72. to.erase(0, 1);
  73. size_t pos = to.find('!');
  74. if (pos != std::string::npos) {
  75. to.erase(pos);
  76. }
  77. return to;
  78. }
  79. // namespace io = boost::asio;
  80. // namespace ip = io::ip;
  81. // using tcp = boost::asio::ip; // ip::tcp;
  82. using error_code = boost::system::error_code;
  83. using namespace std::placeholders;
  84. // #define DEBUG_OUTPUT
  85. typedef std::function<void(std::string &)> receiveFunction;
  86. ircClient::ircClient(boost::asio::io_context &io_context)
  87. : resolver{io_context}, ssl_context{boost::asio::ssl::context::tls},
  88. socket{io_context, ssl_context}, context{io_context} {
  89. registered = false;
  90. nick_retry = 1;
  91. shutdown = false;
  92. logging = false;
  93. }
  94. std::ofstream &ircClient::log(void) {
  95. std::time_t t = std::time(nullptr);
  96. std::tm tm = *std::localtime(&t);
  97. debug_file << std::put_time(&tm, "%c ");
  98. return debug_file;
  99. }
  100. void ircClient::begin(void) {
  101. original_nick = nick;
  102. resolver.async_resolve(hostname, port,
  103. std::bind(&ircClient::on_resolve, this, _1, _2));
  104. if (!debug_output.empty()) {
  105. debug_file.open(debug_output.c_str(),
  106. std::ofstream::out | std::ofstream::app);
  107. logging = true;
  108. }
  109. }
  110. void ircClient::write(std::string output) {
  111. if (logging) {
  112. log() << "<< " << output << std::endl;
  113. }
  114. error_code error;
  115. socket.write_some(boost::asio::buffer(output + "\r\n"), error);
  116. if (error) {
  117. if (logging) {
  118. log() << "Write: " << error.message() << std::endl;
  119. }
  120. }
  121. }
  122. void ircClient::message_append(message_stamp &msg) {
  123. lock.lock();
  124. messages.push_back(msg);
  125. lock.unlock();
  126. }
  127. boost::optional<message_stamp> ircClient::message_pop(void) {
  128. lock.lock();
  129. message_stamp msg;
  130. if (messages.empty()) {
  131. lock.unlock();
  132. return boost::optional<message_stamp>{};
  133. }
  134. msg = messages.front();
  135. messages.erase(messages.begin());
  136. lock.unlock();
  137. return msg;
  138. }
  139. /*
  140. void ircClient::buffer_append(std::vector<std::string> &data) {
  141. lock.lock();
  142. buffer.push_back(data);
  143. lock.unlock();
  144. }
  145. int ircClient::buffer_size(void) {
  146. lock.lock();
  147. int size = buffer.size();
  148. lock.unlock();
  149. return size;
  150. }
  151. std::vector<std::string> ircClient::buffer_pop(void) {
  152. lock.lock();
  153. std::vector<std::string> ret = buffer.front();
  154. buffer.erase(buffer.begin());
  155. lock.unlock();
  156. return ret;
  157. }
  158. boost::optional<std::vector<std::string>> ircClient::buffer_maybe_pop(void) {
  159. lock.lock();
  160. if (buffer.empty()) {
  161. lock.unlock();
  162. return boost::optional<std::vector<std::string>>{};
  163. }
  164. std::vector<std::string> ret = buffer.front();
  165. buffer.erase(buffer.begin());
  166. lock.unlock();
  167. return ret;
  168. }
  169. */
  170. void ircClient::on_resolve(
  171. error_code error, boost::asio::ip::tcp::resolver::results_type results) {
  172. if (logging) {
  173. log() << "Resolve: " << error.message() << std::endl;
  174. }
  175. if (error) {
  176. std::string output = "Unable to resolve (DNS Issue?): " + error.message();
  177. message(output);
  178. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  179. }
  180. boost::asio::async_connect(socket.next_layer(), results,
  181. std::bind(&ircClient::on_connect, this, _1, _2));
  182. }
  183. void ircClient::on_connect(error_code error,
  184. boost::asio::ip::tcp::endpoint const &endpoint) {
  185. if (logging) {
  186. log() << "Connect: " << error.message() << ", endpoint: " << endpoint
  187. << std::endl;
  188. }
  189. if (error) {
  190. std::string output = "Unable to connect: " + error.message();
  191. message(output);
  192. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  193. }
  194. socket.async_handshake(boost::asio::ssl::stream_base::client,
  195. std::bind(&ircClient::on_handshake, this, _1));
  196. }
  197. void ircClient::on_handshake(error_code error) {
  198. if (logging) {
  199. log() << "Handshake: " << error.message() << std::endl;
  200. }
  201. if (error) {
  202. std::string output = "Handshake: " + error.message();
  203. message(output);
  204. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  205. }
  206. std::string request = registration();
  207. boost::asio::async_write(socket, boost::asio::buffer(request),
  208. std::bind(&ircClient::on_write, this, _1, _2));
  209. // socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  210. }
  211. void ircClient::on_write(error_code error, std::size_t bytes_transferred) {
  212. if ((error) and (logging)) {
  213. log() << "Write: " << error.message() << std::endl;
  214. }
  215. // << ", bytes transferred: " << bytes_transferred << "\n";
  216. boost::asio::async_read_until(
  217. socket, response, '\n', std::bind(&ircClient::read_until, this, _1, _2));
  218. }
  219. void ircClient::on_shutdown(error_code error) {
  220. if (logging) {
  221. log() << "SHUTDOWN: " << error.message() << std::endl;
  222. }
  223. shutdown = true;
  224. context.stop();
  225. }
  226. void ircClient::read_until(error_code error, std::size_t bytes) {
  227. // std::cout << "Read: " << bytes << ", " << error << "\n";
  228. // auto data = response.data();
  229. if (bytes == 0) {
  230. if (logging) {
  231. log() << "Read 0 bytes, shutdown..." << std::endl;
  232. }
  233. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  234. return;
  235. };
  236. // Only try to get the data -- if we're read some bytes.
  237. auto data = response.data();
  238. response.consume(bytes);
  239. std::string text{(const char *)data.data(), bytes};
  240. while ((text[text.size() - 1] == '\r') or (text[text.size() - 1] == '\n'))
  241. text.erase(text.size() - 1);
  242. receive(text);
  243. // repeat until closed
  244. boost::asio::async_read_until(
  245. socket, response, '\n', std::bind(&ircClient::read_until, this, _1, _2));
  246. }
  247. void ircClient::message(std::string msg) {
  248. message_stamp ms;
  249. ms.buffer.push_back(msg);
  250. message_append(ms);
  251. }
  252. void ircClient::receive(std::string &text) {
  253. message_stamp ms;
  254. ms.buffer = irc_split(text);
  255. std::vector<std::string> &parts = ms.buffer; // irc_split(text);
  256. if (logging) {
  257. // this also shows our parser working
  258. std::ofstream &l = log();
  259. l << ">> ";
  260. for (auto &s : parts) {
  261. l << "[" << s << "] ";
  262. }
  263. l << std::endl;
  264. }
  265. // INTERNAL IRC PARSING/TRACKING
  266. if (parts.size() == 2) {
  267. // hide PING / PONG messages
  268. if (parts[0] == "PING") {
  269. std::string output = "PONG " + parts[1];
  270. write(output);
  271. return;
  272. }
  273. }
  274. if (parts.size() >= 3) {
  275. std::string source = parse_nick(parts[0]);
  276. std::string cmd = parts[1];
  277. std::string msg_to = parts[2];
  278. std::string msg;
  279. if (parts.size() == 4) {
  280. msg = parts[3];
  281. }
  282. if (cmd == "JOIN") {
  283. msg_to.erase(0, 1); // channel
  284. channels_lock.lock();
  285. if (nick == source) {
  286. // yes, we are joining
  287. std::string output =
  288. "You have joined " + msg_to + " [talkto = " + msg_to + "]";
  289. message(output);
  290. talkto(msg_to);
  291. // insert empty set here.
  292. std::set<std::string> empty;
  293. channels[msg_to] = empty;
  294. } else {
  295. // Someone else is joining
  296. std::string output = source + " has joined " += msg_to;
  297. message(output);
  298. channels[msg_to].insert(source);
  299. }
  300. channels_lock.unlock();
  301. }
  302. if (cmd == "PART") {
  303. channels_lock.lock();
  304. if (nick == source) {
  305. std::string output = "You left " + msg_to;
  306. if (logging) {
  307. for (auto c : channels) {
  308. log() << c.first << " ";
  309. for (auto s : c.second) {
  310. log() << s << " ";
  311. }
  312. log() << std::endl;
  313. }
  314. }
  315. {
  316. auto ch = channels.find(msg_to);
  317. if (ch != channels.end()) {
  318. channels.erase(ch);
  319. log() << "erase ! " << msg_to << std::endl;
  320. } else {
  321. log() << "failed to find " << msg_to << std::endl;
  322. }
  323. }
  324. if (logging) {
  325. for (auto c : channels) {
  326. log() << c.first << " ";
  327. for (auto s : c.second) {
  328. log() << s << " ";
  329. }
  330. log() << std::endl;
  331. }
  332. }
  333. if (!channels.empty()) {
  334. talkto(channels.begin()->first);
  335. output += " [talkto = " + talkto() + "]";
  336. } else {
  337. talkto("");
  338. }
  339. message(output);
  340. } else {
  341. std::string output = source + " has left " + msg_to;
  342. if (!msg.empty()) {
  343. output += " " + msg;
  344. }
  345. message(output);
  346. channels[msg_to].erase(source);
  347. }
  348. channels_lock.unlock();
  349. }
  350. if (cmd == "KICK") {
  351. std::string wholeft = split_limit(parts[3], 2)[0];
  352. std::string output =
  353. source + " has kicked " + wholeft + " from " + msg_to;
  354. channels_lock.lock();
  355. if (wholeft == nick) {
  356. channels.erase(msg_to);
  357. if (!channels.empty()) {
  358. talkto(channels.begin()->first);
  359. output += " [talkto = " + talkto() + "]";
  360. } else {
  361. talkto("");
  362. }
  363. } else {
  364. channels[msg_to].erase(wholeft);
  365. }
  366. channels_lock.unlock();
  367. message(output);
  368. }
  369. if (cmd == "QUIT") {
  370. std::string output = "* " + source + " has quit ";
  371. message(output);
  372. channels_lock.lock();
  373. if (source == nick) {
  374. // We've quit?
  375. channels.erase(channels.begin(), channels.end());
  376. } else {
  377. for (auto c : channels) {
  378. c.second.erase(source);
  379. // would it be possible that channel is empty now?
  380. }
  381. }
  382. channels_lock.unlock();
  383. }
  384. if (cmd == "353") {
  385. // NAMES list for channel
  386. std::vector<std::string> names_list = split_limit(msg);
  387. names_list.erase(names_list.begin());
  388. std::string channel = names_list.front();
  389. names_list.erase(names_list.begin());
  390. if ((names_list.size() > 0) and (names_list[0][0] == ':')) {
  391. names_list[0].erase(0, 1);
  392. }
  393. channels_lock.lock();
  394. if (channels.find(channel) == channels.end()) {
  395. // does not exist
  396. channels.insert({channel, std::set<std::string>{}});
  397. }
  398. for (auto name : names_list) {
  399. remove_channel_modes(name);
  400. channels[channel].insert(name);
  401. }
  402. channels_lock.unlock();
  403. }
  404. if (cmd == "PRIVMSG") {
  405. // Possibly a CTCP request. Let's see
  406. std::string message = msg;
  407. if ((message[0] == ':') and (message[1] == '\x01') and
  408. (message[message.size() - 1] == '\x01')) {
  409. // CTCP MESSAGE FOUND strip \x01's
  410. message.erase(0, 2);
  411. message.erase(message.size() - 1);
  412. std::vector<std::string> ctcp_cmd = split_limit(message, 2);
  413. if (ctcp_cmd[0] != "ACTION") {
  414. std::string msg =
  415. "Received CTCP " + ctcp_cmd[0] + " from " + parse_nick(parts[0]);
  416. this->message(msg);
  417. if (logging) {
  418. log() << "CTCP : [" << message << "] from " + parse_nick(parts[0])
  419. << std::endl;
  420. }
  421. }
  422. if (message == "VERSION") {
  423. std::string reply_to = parse_nick(parts[0]);
  424. boost::format fmt =
  425. boost::format("NOTICE %1% :\x01VERSION Bugz IRC thing V0.1\x01") %
  426. reply_to;
  427. std::string response = fmt.str();
  428. write(response);
  429. return;
  430. }
  431. if (message.substr(0, 5) == "PING ") {
  432. message.erase(0, 5);
  433. boost::format fmt = boost::format("NOTICE %1% :\x01PING %2%\x01") %
  434. parse_nick(parts[0]) % message;
  435. std::string response = fmt.str();
  436. write(response);
  437. return;
  438. }
  439. if (message == "TIME") {
  440. auto now = std::chrono::system_clock::now();
  441. auto in_time_t = std::chrono::system_clock::to_time_t(now);
  442. std::string datetime = boost::lexical_cast<std::string>(
  443. std::put_time(std::localtime(&in_time_t), "%c"));
  444. boost::format fmt = boost::format("NOTICE %1% :\x01TIME %2%\x01") %
  445. parse_nick(parts[0]) % datetime;
  446. std::string response = fmt.str();
  447. write(response);
  448. return;
  449. }
  450. if (message.substr(0, 7) == "ACTION ") {
  451. message.erase(0, 7);
  452. parts[1] = "ACTION"; // change PRIVMSG to ACTION
  453. parts[3] = message;
  454. }
  455. // I have this parsed this far, now what can I do with it?!
  456. }
  457. }
  458. }
  459. // CTCP handler
  460. // NOTE: When sent to a channel, the response is sent to the sender.
  461. if (!registered) {
  462. // We're not registered yet
  463. if (parts[1] == "433") {
  464. // nick collision! Nick already in use
  465. if (nick == original_nick) {
  466. // try something basic
  467. nick += "_";
  468. std::string output = "NICK " + nick;
  469. write(output);
  470. return;
  471. } else {
  472. // Ok, go advanced
  473. nick = original_nick + "_" + std::to_string(nick_retry);
  474. ++nick_retry;
  475. std::string output = "NICK " + nick;
  476. write(output);
  477. return;
  478. }
  479. }
  480. if ((parts[1] == "376") or (parts[1] == "422")) {
  481. // END MOTD, or MOTD MISSING
  482. registered = true;
  483. if (!autojoin.empty()) {
  484. std::string msg = "JOIN " + autojoin;
  485. write(msg);
  486. }
  487. }
  488. }
  489. if (parts[0] == "ERROR") {
  490. // we're outta here. :O
  491. // std::cout << "BANG!" << std::endl;
  492. }
  493. message_append(ms);
  494. // :FROM command TO :rest and ':' is optional
  495. // std::cout << text << "\n";
  496. }
  497. std::string ircClient::registration(void) {
  498. std::string text;
  499. text = "NICK " + nick + "\r\n" + "USER " + username + " 0 * :" + realname +
  500. "\r\n";
  501. return text;
  502. }