irc.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600
  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. size_t msgpos = text.find(" :");
  58. std::string message;
  59. if (msgpos != std::string::npos) {
  60. message = text.substr(msgpos + 2);
  61. text.erase(msgpos);
  62. }
  63. std::vector<std::string> results;
  64. /*
  65. if (text[0] != ':')
  66. results = split_limit(text, 2);
  67. results = split_limit(text, 4);
  68. */
  69. results = split_limit(text, -1);
  70. if (!message.empty())
  71. results.push_back(message);
  72. return results;
  73. }
  74. /**
  75. * @brief parse_nick
  76. *
  77. * Parse out the nick from nick!username@host
  78. *
  79. * @param name
  80. * @return std::string
  81. */
  82. std::string parse_nick(std::string &name) {
  83. std::string to = name;
  84. if (to[0] == ':')
  85. to.erase(0, 1);
  86. size_t pos = to.find('!');
  87. if (pos != std::string::npos) {
  88. to.erase(pos);
  89. }
  90. return to;
  91. }
  92. // namespace io = boost::asio;
  93. // namespace ip = io::ip;
  94. // using tcp = boost::asio::ip; // ip::tcp;
  95. using error_code = boost::system::error_code;
  96. using namespace std::placeholders;
  97. // #define DEBUG_OUTPUT
  98. typedef std::function<void(std::string &)> receiveFunction;
  99. ircClient::ircClient(boost::asio::io_context &io_context)
  100. : resolver{io_context}, ssl_context{boost::asio::ssl::context::tls},
  101. socket{io_context, ssl_context}, context{io_context} {
  102. registered = false;
  103. nick_retry = 1;
  104. shutdown = false;
  105. logging = false;
  106. channels_updated = false;
  107. version = "Bugz IRC thing V0.1";
  108. }
  109. std::ofstream &ircClient::log(void) {
  110. std::time_t t = std::time(nullptr);
  111. std::tm tm = *std::localtime(&t);
  112. debug_file << std::put_time(&tm, "%c ");
  113. return debug_file;
  114. }
  115. void ircClient::begin(void) {
  116. original_nick = nick;
  117. resolver.async_resolve(hostname, port,
  118. std::bind(&ircClient::on_resolve, this, _1, _2));
  119. if (!debug_output.empty()) {
  120. debug_file.open(debug_output.c_str(),
  121. std::ofstream::out | std::ofstream::app);
  122. logging = true;
  123. }
  124. }
  125. void ircClient::write(std::string output) {
  126. if (logging) {
  127. log() << "<< " << output << std::endl;
  128. }
  129. error_code error;
  130. socket.write_some(boost::asio::buffer(output + "\r\n"), error);
  131. if (error) {
  132. if (logging) {
  133. log() << "Write: " << error.message() << std::endl;
  134. }
  135. }
  136. }
  137. /**
  138. * @brief thread safe messages.push_back
  139. *
  140. * @param msg
  141. */
  142. void ircClient::message_append(message_stamp &msg) {
  143. lock.lock();
  144. messages.push_back(msg);
  145. channels_updated = true;
  146. lock.unlock();
  147. }
  148. /**
  149. * @brief thread safe message_stamp pop
  150. *
  151. * @return boost::optional<message_stamp>
  152. */
  153. boost::optional<message_stamp> ircClient::message_pop(void) {
  154. lock.lock();
  155. message_stamp msg;
  156. if (messages.empty()) {
  157. channels_updated = false;
  158. lock.unlock();
  159. return boost::optional<message_stamp>{};
  160. }
  161. msg = messages.front();
  162. messages.erase(messages.begin());
  163. lock.unlock();
  164. return msg;
  165. }
  166. void ircClient::on_resolve(
  167. error_code error, boost::asio::ip::tcp::resolver::results_type results) {
  168. if (logging) {
  169. log() << "Resolve: " << error.message() << std::endl;
  170. }
  171. if (error) {
  172. std::string output = "Unable to resolve (DNS Issue?): " + error.message();
  173. errors.push_back(output);
  174. message(output);
  175. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  176. }
  177. boost::asio::async_connect(socket.next_layer(), results,
  178. std::bind(&ircClient::on_connect, this, _1, _2));
  179. }
  180. void ircClient::on_connect(error_code error,
  181. boost::asio::ip::tcp::endpoint const &endpoint) {
  182. if (logging) {
  183. log() << "Connect: " << error.message() << ", endpoint: " << endpoint
  184. << std::endl;
  185. }
  186. if (error) {
  187. std::string output = "Unable to connect: " + error.message();
  188. message(output);
  189. errors.push_back(output);
  190. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  191. }
  192. socket.async_handshake(boost::asio::ssl::stream_base::client,
  193. std::bind(&ircClient::on_handshake, this, _1));
  194. }
  195. void ircClient::on_handshake(error_code error) {
  196. if (logging) {
  197. log() << "Handshake: " << error.message() << std::endl;
  198. }
  199. if (error) {
  200. std::string output = "Handshake Failure: " + error.message();
  201. message(output);
  202. errors.push_back(output);
  203. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  204. }
  205. std::string request = registration();
  206. boost::asio::async_write(socket, boost::asio::buffer(request),
  207. std::bind(&ircClient::on_write, this, _1, _2));
  208. // socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  209. }
  210. void ircClient::on_write(error_code error, std::size_t bytes_transferred) {
  211. if ((error) and (logging)) {
  212. log() << "Write: " << error.message() << std::endl;
  213. }
  214. // << ", bytes transferred: " << bytes_transferred << "\n";
  215. boost::asio::async_read_until(
  216. socket, response, '\n', std::bind(&ircClient::read_until, this, _1, _2));
  217. }
  218. void ircClient::on_shutdown(error_code error) {
  219. if (logging) {
  220. log() << "SHUTDOWN: " << error.message() << std::endl;
  221. }
  222. shutdown = true;
  223. context.stop();
  224. }
  225. void ircClient::read_until(error_code error, std::size_t bytes) {
  226. // std::cout << "Read: " << bytes << ", " << error << "\n";
  227. // auto data = response.data();
  228. if (bytes == 0) {
  229. if (logging) {
  230. log() << "Read 0 bytes, shutdown..." << std::endl;
  231. }
  232. socket.async_shutdown(std::bind(&ircClient::on_shutdown, this, _1));
  233. return;
  234. };
  235. // Only try to get the data -- if we're read some bytes.
  236. auto data = response.data();
  237. response.consume(bytes);
  238. std::string text{(const char *)data.data(), bytes};
  239. while ((text[text.size() - 1] == '\r') or (text[text.size() - 1] == '\n'))
  240. text.erase(text.size() - 1);
  241. receive(text);
  242. // repeat until closed
  243. boost::asio::async_read_until(
  244. socket, response, '\n', std::bind(&ircClient::read_until, this, _1, _2));
  245. }
  246. /**
  247. * @brief Append a system message to the messages.
  248. *
  249. * @param msg
  250. */
  251. void ircClient::message(std::string msg) {
  252. message_stamp ms;
  253. ms.buffer.push_back(msg);
  254. message_append(ms);
  255. }
  256. void ircClient::receive(std::string &text) {
  257. message_stamp ms;
  258. ms.buffer = irc_split(text);
  259. std::vector<std::string> &parts = ms.buffer;
  260. if (logging) {
  261. // this also shows our parser working
  262. std::ofstream &l = log();
  263. l << ">> ";
  264. for (auto &s : parts) {
  265. l << "[" << s << "] ";
  266. }
  267. l << std::endl;
  268. }
  269. // INTERNAL IRC PARSING/TRACKING
  270. if (parts.size() == 2) {
  271. // hide PING / PONG messages
  272. if (parts[0] == "PING") {
  273. std::string output = "PONG " + parts[1];
  274. write(output);
  275. return;
  276. }
  277. }
  278. if (parts.size() >= 3) {
  279. std::string source = parse_nick(parts[0]);
  280. std::string &cmd = parts[1];
  281. std::string &msg_to = parts[2];
  282. std::string msg;
  283. if (parts.size() >= 4) {
  284. msg = parts[parts.size() - 1];
  285. }
  286. if (logging) {
  287. // this also shows our parser working
  288. std::ofstream &l = log();
  289. l << "IRC: [SRC:" << source << "] [CMD:" << cmd << "] [TO:" << msg_to
  290. << "] [MSG:" << msg << "]" << std::endl;
  291. }
  292. if (cmd == "JOIN") {
  293. channels_lock.lock();
  294. if (nick == source) {
  295. // yes, we are joining
  296. std::string output = "You have joined " + msg_to;
  297. message(output);
  298. talkto(msg_to);
  299. // insert empty set here.
  300. std::set<std::string> empty;
  301. channels[msg_to] = empty;
  302. } else {
  303. // Someone else is joining
  304. std::string output = source + " has joined " += msg_to;
  305. message(output);
  306. channels[msg_to].insert(source);
  307. if ((int)source.size() > max_nick_length)
  308. max_nick_length = (int)source.size();
  309. }
  310. channels_lock.unlock();
  311. }
  312. if (cmd == "PART") {
  313. channels_lock.lock();
  314. if (nick == source) {
  315. std::string output = "You left " + msg_to;
  316. auto ch = channels.find(msg_to);
  317. if (ch != channels.end())
  318. channels.erase(ch);
  319. if (!channels.empty()) {
  320. talkto(channels.begin()->first);
  321. // output += " [talkto = " + talkto() + "]";
  322. } else {
  323. talkto("");
  324. }
  325. // message(output);
  326. } else {
  327. std::string output = source + " has left " + msg_to;
  328. if (!msg.empty()) {
  329. output += " " + msg;
  330. }
  331. message(output);
  332. channels[msg_to].erase(source);
  333. }
  334. find_max_nick_length();
  335. channels_lock.unlock();
  336. }
  337. if (cmd == "KICK") {
  338. std::string output =
  339. source + " has kicked " + parts[3] + " from " + msg_to;
  340. channels_lock.lock();
  341. if (parts[3] == nick) {
  342. channels.erase(msg_to);
  343. if (!channels.empty()) {
  344. talkto(channels.begin()->first);
  345. output += " [talkto = " + talkto() + "]";
  346. } else {
  347. talkto("");
  348. }
  349. } else {
  350. channels[msg_to].erase(parts[3]);
  351. }
  352. find_max_nick_length();
  353. channels_lock.unlock();
  354. message(output);
  355. }
  356. if (cmd == "QUIT") {
  357. std::string output = "* " + source + " has quit ";
  358. message(output);
  359. channels_lock.lock();
  360. if (source == nick) {
  361. // We've quit?
  362. channels.erase(channels.begin(), channels.end());
  363. } else {
  364. for (auto &c : channels) {
  365. c.second.erase(source);
  366. // would it be possible that channel is empty now?
  367. // no, because we're still in it.
  368. }
  369. find_max_nick_length();
  370. }
  371. channels_lock.unlock();
  372. }
  373. if (cmd == "353") {
  374. // NAMES list for channel
  375. std::vector<std::string> names_list = split_limit(msg);
  376. std::string channel = parts[4];
  377. channels_lock.lock();
  378. if (channels.find(channel) == channels.end()) {
  379. // does not exist
  380. channels.insert({channel, std::set<std::string>{}});
  381. }
  382. for (auto name : names_list) {
  383. remove_channel_modes(name);
  384. channels[channel].insert(name);
  385. }
  386. find_max_nick_length();
  387. channels_lock.unlock();
  388. }
  389. if (cmd == "NICK") {
  390. // msg_to.erase(0, 1);
  391. channels_lock.lock();
  392. for (auto &ch : channels) {
  393. if (ch.second.erase(source) == 1) {
  394. ch.second.insert(msg_to);
  395. }
  396. }
  397. // Is this us? If so, change our nick.
  398. if (source == nick)
  399. nick = msg_to;
  400. find_max_nick_length();
  401. channels_lock.unlock();
  402. }
  403. if (cmd == "PRIVMSG") {
  404. // Possibly a CTCP request. Let's see
  405. std::string message = msg;
  406. if ((message[0] == '\x01') and (message[message.size() - 1] == '\x01')) {
  407. // CTCP MESSAGE FOUND strip \x01's
  408. message.erase(0, 1);
  409. message.erase(message.size() - 1);
  410. std::vector<std::string> ctcp_cmd = split_limit(message, 2);
  411. if (ctcp_cmd[0] != "ACTION") {
  412. std::string msg =
  413. "Received CTCP " + ctcp_cmd[0] + " from " + parse_nick(source);
  414. this->message(msg);
  415. if (logging) {
  416. log() << "CTCP : [" << message << "] from " + parse_nick(source)
  417. << std::endl;
  418. }
  419. }
  420. if (message == "VERSION") {
  421. std::string reply_to = parse_nick(source);
  422. boost::format fmt = boost::format("NOTICE %1% :\x01VERSION %2%\x01") %
  423. reply_to % version;
  424. std::string response = fmt.str();
  425. write(response);
  426. return;
  427. }
  428. if (message.substr(0, 5) == "PING ") {
  429. message.erase(0, 5);
  430. boost::format fmt = boost::format("NOTICE %1% :\x01PING %2%\x01") %
  431. parse_nick(source) % message;
  432. std::string response = fmt.str();
  433. write(response);
  434. return;
  435. }
  436. if (message == "TIME") {
  437. auto now = std::chrono::system_clock::now();
  438. auto in_time_t = std::chrono::system_clock::to_time_t(now);
  439. std::string datetime = boost::lexical_cast<std::string>(
  440. std::put_time(std::localtime(&in_time_t), "%c"));
  441. boost::format fmt = boost::format("NOTICE %1% :\x01TIME %2%\x01") %
  442. parse_nick(source) % datetime;
  443. std::string response = fmt.str();
  444. write(response);
  445. return;
  446. }
  447. if (message.substr(0, 7) == "ACTION ") {
  448. message.erase(0, 7);
  449. parts[1] = "ACTION"; // change PRIVMSG to ACTION
  450. parts[3] = message;
  451. } else {
  452. // What should I do with unknown CTCP commands?
  453. // Unknown CTCP command. Eat it.
  454. return;
  455. }
  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. find_max_nick_length(); // start with ourself.
  483. registered = true;
  484. if (!autojoin.empty()) {
  485. std::string msg = "JOIN " + autojoin;
  486. write(msg);
  487. }
  488. }
  489. }
  490. if (parts[0] == "ERROR") {
  491. // we're outta here. :O
  492. // std::cout << "BANG!" << std::endl;
  493. }
  494. message_append(ms);
  495. // :FROM command TO :rest and ':' is optional
  496. // std::cout << text << "\n";
  497. }
  498. /**
  499. * @brief find max nick length
  500. *
  501. * This is for formatting the messages.
  502. * We run through the channels checking all the users,
  503. * and also checking our own nick.
  504. *
  505. * This updates \ref max_nick_length
  506. */
  507. void ircClient::find_max_nick_length(void) {
  508. int max = 0;
  509. for (auto const &ch : channels) {
  510. for (auto const &nick : ch.second) {
  511. if ((int)nick.size() > max)
  512. max = (int)nick.size();
  513. }
  514. }
  515. // check our nick against this too.
  516. if ((int)nick.size() > max)
  517. max = (int)nick.size();
  518. max_nick_length = max;
  519. }
  520. std::string ircClient::registration(void) {
  521. std::string text;
  522. text = "NICK " + nick + "\r\n" + "USER " + username + " 0 * :" + realname +
  523. "\r\n";
  524. return text;
  525. }