session.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. #include <boost/bind.hpp>
  2. #include <iostream>
  3. #include <boost/format.hpp>
  4. #include <boost/log/core.hpp>
  5. #include <boost/log/trivial.hpp>
  6. #include <regex>
  7. #include "session.h"
  8. #include <string>
  9. // #include <boost/log/attributes/named_scope.hpp>
  10. bool replace(std::string &str, const std::string &from, const std::string &to) {
  11. size_t start_pos = str.find(from);
  12. if (start_pos == std::string::npos)
  13. return false;
  14. do {
  15. str.replace(start_pos, from.length(), to);
  16. } while ((start_pos = str.find(from)) != std::string::npos);
  17. return true;
  18. }
  19. bool replace(std::string &str, const char *from, const char *to) {
  20. size_t start_pos = str.find(from);
  21. if (start_pos == std::string::npos)
  22. return false;
  23. do {
  24. str.replace(start_pos, strlen(from), to);
  25. } while ((start_pos = str.find(from)) != std::string::npos);
  26. return true;
  27. }
  28. void ansi_clean(std::string &str) {
  29. static std::regex ansi_cleaner("\x1b\[[0-9;]*[A-Zmh]",
  30. std::regex_constants::ECMAScript);
  31. str = std::regex_replace(str, ansi_cleaner, "");
  32. }
  33. void high_ascii(std::string &str) {
  34. static std::regex high_cleaner("[\x80-\xff]+",
  35. std::regex_constants::ECMAScript);
  36. str = std::regex_replace(str, high_cleaner, "#");
  37. }
  38. std::string clean_string(const std::string &source) {
  39. // BOOST_LOG_NAMED_SCOPE("clean_string");
  40. std::string clean = source;
  41. replace(clean, "\n", "\\n");
  42. replace(clean, "\r", "\\r");
  43. // ANSI too
  44. ansi_clean(clean);
  45. // BOOST_LOG_TRIVIAL(error) << "cleaned: " << clean;
  46. high_ascii(clean);
  47. replace(clean, "\x1b", "^");
  48. return clean;
  49. }
  50. session::session(boost::asio::ip::tcp::socket socket,
  51. boost::asio::io_service &io_service, std::string hostname,
  52. std::string port)
  53. : socket_(std::move(socket)),
  54. io_service_{io_service}, resolver_{io_service}, server_{io_service},
  55. timer_{io_service}, host{hostname}, port{port} {
  56. server_sent = 0;
  57. // BOOST_LOG_NAMED_SCOPE("session");
  58. }
  59. void session::start(void) {
  60. // BOOST_LOG_NAMED_SCOPE();
  61. // If I want the file and line number information, here's how to do it:
  62. BOOST_LOG_TRIVIAL(info) << boost::format("(%1%:%2%) ") % __FILE__ % __LINE__
  63. << "session";
  64. auto self(shared_from_this());
  65. // read_buffer.reserve(1024);
  66. // do_write("Welcome!\n");
  67. do_read();
  68. }
  69. session::~session() {
  70. // BOOST_LOG_NAMED_SCOPE("session");
  71. BOOST_LOG_TRIVIAL(info) << "~session destructed";
  72. }
  73. void session::parse_auth(void) {
  74. // how many nulls should I be seeing?
  75. // \0user\0pass\0terminal/SPEED\0
  76. // If I don't have a proper rlogin value here, it isn't going
  77. // to work when I try to connect to the rlogin server.
  78. if (rlogin_auth.size() > 10)
  79. rlogin_name = rlogin_auth.c_str() + 1;
  80. else
  81. rlogin_name = "?";
  82. }
  83. void session::on_connect(const boost::system::error_code error) {
  84. // We've connected to the server! WOOT WOOT!
  85. // BOOST_LOG_NAMED_SCOPE("session");
  86. if (!error) {
  87. BOOST_LOG_TRIVIAL(info) << "Connected to " << host;
  88. to_client("Connected...\n\r");
  89. connected = true;
  90. if (rlogin_auth[0] != 0) {
  91. // Ok, the rlogin information was junk --
  92. to_client("Let me make up some fake rlogin data for you...\n\r");
  93. char temp[] = "\0test\0test\0terminal/9600\0";
  94. std::string tmp(temp, sizeof(temp));
  95. to_server(tmp);
  96. } else {
  97. to_server(rlogin_auth);
  98. }
  99. server_read();
  100. } else {
  101. // TODO:
  102. std::string output =
  103. str(boost::format("Failed to connect: %1%:%2%\n\r") % host % port);
  104. to_client(output);
  105. BOOST_LOG_TRIVIAL(error) << "Failed to connect to " << host << ":" << port;
  106. BOOST_LOG_TRIVIAL(warning) << "socket.shutdown()";
  107. socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  108. }
  109. }
  110. void session::dispatch_line(std::string line) {
  111. // Does this have \n\r still on it? I don't want them.
  112. std::string temp = clean_string(line);
  113. BOOST_LOG_TRIVIAL(info) << "SL: " << temp;
  114. }
  115. /*
  116. Call this with whatever we just received.
  117. That will allow me to send "just whatever I got"
  118. this time around, rather then trying to figure out
  119. what was just added to server_prompt.
  120. What about \r, \b ? Should that "reset" the server_prompt?
  121. */
  122. void session::process_lines(std::string &received) {
  123. // break server_prompt into lines and send/process one by one.
  124. size_t pos, rpos;
  125. server_prompt.append(received);
  126. while ((pos = server_prompt.find('\n', 0)) != std::string::npos) {
  127. std::string line;
  128. // process "line" in received
  129. rpos = received.find('\n', 0);
  130. // get line to send to the client
  131. if (show_client) {
  132. // that is, if we're sending to the client!
  133. line = received.substr(0, rpos + 1);
  134. /*
  135. std::string clean = clean_string(line);
  136. BOOST_LOG_TRIVIAL(error) << "rpos/show_client:" << clean;
  137. */
  138. to_client(line);
  139. }
  140. received = received.substr(rpos + 1);
  141. // process "line" in server_prompt
  142. line = server_prompt.substr(0, pos + 1);
  143. server_prompt = server_prompt.substr(pos + 1);
  144. // Remove \n for dispatching
  145. std::string part = line.substr(0, pos);
  146. if (server_sent != 0) {
  147. line = line.substr(server_sent);
  148. server_sent = 0;
  149. };
  150. // display on?
  151. // to_client(line);
  152. // NOTE: We get "TradeWars Game Server\n" (Missing \r)
  153. // We add the \r with our injection line.
  154. // TODO(stevet): MOVE TO DEFAULT dispatcher
  155. // our first injection
  156. if (line.find("TradeWars Game Server") != std::string::npos) {
  157. to_client("\rTradeWars Proxy v2++ READY (~ to activate)\n\r");
  158. }
  159. // How should I handle \r in lines? For now, remove it
  160. // but LOG that we did.
  161. if (replace(part, "\r", "")) {
  162. BOOST_LOG_TRIVIAL(warning) << "\\r removed from line";
  163. }
  164. dispatch_line(part);
  165. }
  166. // Ok, we have sent all of the \n lines.
  167. if (!received.empty())
  168. if (show_client) {
  169. to_client(received);
  170. std::string clean = clean_string(received);
  171. BOOST_LOG_TRIVIAL(error) << "show_client/leftovers:" << clean;
  172. }
  173. }
  174. void session::server_read(void) {
  175. auto self(shared_from_this());
  176. boost::asio::async_read(
  177. server_, boost::asio::buffer(server_buffer, sizeof(server_buffer) - 1),
  178. boost::asio::transfer_at_least(1),
  179. [this, self](boost::system::error_code ec, std::size_t length) {
  180. if (!ec) {
  181. server_buffer[length] = 0;
  182. // server_prompt.append(server_buffer, length);
  183. std::string received(server_buffer, length);
  184. process_lines(received);
  185. /*
  186. I don't believe I need to consume this,
  187. I'm not async_reading from a stream.
  188. */
  189. /*
  190. if (length) {
  191. // std::cout << length << std::endl;
  192. std::cout << "S: " << server_buffer << std::endl;
  193. do_write(server_buffer);
  194. }
  195. */
  196. server_read();
  197. } else {
  198. BOOST_LOG_TRIVIAL(warning) << "S: read_failed: socket.shutdown()";
  199. socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  200. }
  201. });
  202. }
  203. void session::on_resolve(
  204. const boost::system::error_code error,
  205. const boost::asio::ip::tcp::resolver::results_type results) {
  206. //
  207. auto self(shared_from_this());
  208. if (!error) {
  209. // Take the first endpoint.
  210. boost::asio::ip::tcp::endpoint const &endpoint = *results;
  211. server_.async_connect(endpoint,
  212. boost::bind(&session::on_connect, this,
  213. boost::asio::placeholders::error));
  214. } else {
  215. // TO DO:
  216. // BOOST_LOG_NAMED_SCOPE("session");
  217. BOOST_LOG_TRIVIAL(error) << "Unable to resolve: " << host;
  218. std::string output =
  219. str(boost::format("Unable to resolve: %1%\n\r") % host);
  220. to_client(output);
  221. BOOST_LOG_TRIVIAL(warning) << "socket.shutdown()";
  222. socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  223. }
  224. }
  225. void session::do_read(void) {
  226. auto self(shared_from_this());
  227. boost::asio::async_read( // why can't I async_read_some here?
  228. socket_, boost::asio::buffer(read_buffer, sizeof(read_buffer) - 1),
  229. boost::asio::transfer_at_least(1),
  230. [this, self](boost::system::error_code ec, std::size_t length) {
  231. if (!ec) {
  232. read_buffer[length] = 0;
  233. if (rlogin_auth.empty()) {
  234. // first read should be rlogin information
  235. rlogin_auth.assign(read_buffer, length);
  236. // parse authentication information
  237. parse_auth();
  238. to_client(std::string(1, 0));
  239. to_client("Welcome, ");
  240. to_client(rlogin_name);
  241. to_client("\n\r");
  242. // Activate the connection to the server
  243. /* // this fails, and I'm not sure why. I've used code like this
  244. before. resolver_.async_resolve( host, port, std::bind(
  245. &session::on_resolve, this, _1, _2)); */
  246. // This example shows using boost::bind, which WORKS.
  247. // https://stackoverflow.com/questions/6025471/bind-resolve-handler-to-resolver-async-resolve-using-boostasio
  248. resolver_.async_resolve(
  249. host, port,
  250. boost::bind(&session::on_resolve, this,
  251. boost::asio::placeholders::error,
  252. boost::asio::placeholders::iterator));
  253. } else if (length) {
  254. // Proxy Active?
  255. // BOOST_LOG_NAMED_SCOPE("session");
  256. if (talk_direct)
  257. to_server(read_buffer);
  258. BOOST_LOG_TRIVIAL(info) << "C: " << read_buffer;
  259. // do_write(output);
  260. }
  261. do_read();
  262. } else {
  263. BOOST_LOG_TRIVIAL(warning) << "C: read_failed";
  264. if (connected) {
  265. BOOST_LOG_TRIVIAL(warning) << "server.shutdown()";
  266. server_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  267. }
  268. }
  269. });
  270. }
  271. void session::to_client(std::string message) {
  272. auto self(shared_from_this());
  273. // output the cleaned string (so I can see what we're sending in the
  274. // logs)
  275. std::string clean = clean_string(message);
  276. BOOST_LOG_TRIVIAL(trace) << "C: >>" << clean;
  277. boost::asio::async_write(
  278. socket_, boost::asio::buffer(message),
  279. [this, self](boost::system::error_code ec, std::size_t /*length*/) {
  280. if (!ec) {
  281. } else {
  282. BOOST_LOG_TRIVIAL(warning)
  283. << "C: write failed? closed? server.shutdown()";
  284. if (connected) {
  285. BOOST_LOG_TRIVIAL(warning) << "server.shutdown()";
  286. server_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  287. }
  288. }
  289. });
  290. }
  291. void session::to_server(std::string message) {
  292. auto self(shared_from_this());
  293. boost::asio::async_write(
  294. server_, boost::asio::buffer(message),
  295. [this, self](boost::system::error_code ec, std::size_t /*length*/) {
  296. if (!ec) {
  297. } else {
  298. BOOST_LOG_TRIVIAL(warning)
  299. << "S: write failed? closed? socket.shutdown()";
  300. socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
  301. }
  302. });
  303. }
  304. /*
  305. void session::on_shutdown(boost::system::error_code ec) {
  306. std::cout << "shutdown." << std::endl;
  307. }
  308. */
  309. server::server(boost::asio::io_service &io_service,
  310. const boost::asio::ip::tcp::endpoint &endpoint, std::string host,
  311. std::string port)
  312. : io_service_{io_service}, acceptor_{io_service_, endpoint}, host_{host},
  313. port_{port} {
  314. do_accept();
  315. }
  316. /**
  317. * setup async connect accept
  318. *
  319. * This creates a session for each connection. Using make_shared allows the
  320. * session to automatically clean up when it is no longer active/has anything
  321. * running in the reactor.
  322. */
  323. void server::do_accept(void) {
  324. acceptor_.async_accept([this](boost::system::error_code ec,
  325. boost::asio::ip::tcp::socket socket) {
  326. if (!ec) {
  327. BOOST_LOG_TRIVIAL(info) << "server::do_accept()";
  328. std::make_shared<session>(std::move(socket), io_service_, host_, port_)
  329. ->start();
  330. }
  331. do_accept();
  332. });
  333. }