session.h 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. #ifndef SESSION_H
  2. #define SESSION_H
  3. #include <boost/asio.hpp>
  4. #include <boost/asio/ip/basic_resolver.hpp>
  5. #include <string>
  6. #include <map>
  7. #define MAX_BUFFER 256
  8. /*
  9. My ideas on how to do this.
  10. */
  11. struct Request {
  12. int x, y;
  13. std::string str;
  14. };
  15. struct Result {
  16. int x;
  17. std::string str;
  18. };
  19. typedef std::function<void (const struct Result&)> ResultCallbackFunc;
  20. typedef std::function<void (const struct Request&, ResultCallbackFunc)> CommandFunc;
  21. // Yes, this needs to be in session.cpp (not the header) Whoops!
  22. // std::map<std::string, CommandFunc> commands;
  23. // call_command("goto", Request{1, 0, ""}, my_callback );
  24. void register_command(const std::string name, CommandFunc command);
  25. bool call_command(const std::string name, const struct Request &param, ResultCallbackFunc callback);
  26. // look up name, if found, call it with params, callback. return true (success)
  27. // otherwise fail (false).
  28. typedef std::function<void (const std::string &)> StringFunc;
  29. class OldState {
  30. StringFunc server_line;
  31. StringFunc server_prompt;
  32. StringFunc client_input;
  33. bool server_to_client;
  34. bool client_to_server;
  35. };
  36. class Dispatcher {
  37. std::function<void (std::string)> server_line;
  38. std::function<void (std::string)> server_prompt;
  39. std::function<void (std::string)> client_input;
  40. bool server_to_client;
  41. bool client_to_server;
  42. OldState save_state(void);
  43. void restore_state(OldState);
  44. };
  45. /*
  46. Dispatch Manager:
  47. Handles:
  48. [ ] Does whatever we received from the server get sent to the client?
  49. [SHOW-CLIENT]
  50. [ ] Where does client input go? [CLIENT]
  51. [ ] Client input go directly to the server? [TALK-DIRECT]
  52. [ ] Where does server input go? [SERVER-LINE] [SERVER-PROMPT]
  53. I'm thinking a dispatcher would know how to:
  54. [ ] Save the current state (and current dispatcher/setup).
  55. [ ] Set itself up with the director.
  56. [ ] Restore the previous state (when destructing -- but only if it is the
  57. current active receiver).
  58. [ ] Take a command [goto SECTOR][trade FOE] (fuel organics equipment),
  59. and return a result. [1 SUCCESS][0 FAIL][-1 USER ABORTED]
  60. [ ] Tell director where results should go (callback).
  61. [ ] Tell director where callback messages/events should go. (dynamic)
  62. Director:
  63. [ ] Handles server lines, server prompt, client input.
  64. [ ] Has to_client, to_server.
  65. [ ] Has low-level flags for show_client, talk-direct
  66. [ ] Has a std::stack of dispatchers. Uses .top() for calls.
  67. [ ] Reset -- would pop all but the last/master dispatchers.
  68. It seems like the director wants to be the session, or at least parts of it.
  69. */
  70. /*
  71. The Session:
  72. */
  73. class Session : public std::enable_shared_from_this<Session> {
  74. public:
  75. Session(boost::asio::ip::tcp::socket socket,
  76. boost::asio::io_service &io_service, std::string hostname,
  77. std::string port);
  78. void start(void);
  79. ~Session();
  80. void parse_auth(void);
  81. void on_connect(const boost::system::error_code error);
  82. void server_read(void);
  83. void on_resolve(const boost::system::error_code error,
  84. const boost::asio::ip::tcp::resolver::results_type results);
  85. void client_read(void);
  86. void to_client(std::string message);
  87. void to_server(std::string message);
  88. void on_shutdown(boost::system::error_code ec);
  89. void dispatch_line(std::string line);
  90. void process_lines(std::string &received);
  91. private:
  92. void set_timer(void);
  93. void reset_timer(void);
  94. void on_timer(const boost::system::error_code error);
  95. int time_ms;
  96. // FOR NOW: These will go into the director
  97. bool show_client = true;
  98. bool talk_direct = true;
  99. /**
  100. * The client's socket
  101. */
  102. boost::asio::ip::tcp::socket socket_;
  103. boost::asio::io_service &io_service_;
  104. boost::asio::ip::tcp::resolver resolver_;
  105. /**
  106. * The server's socket
  107. */
  108. boost::asio::ip::tcp::socket server_;
  109. /**
  110. * The time that we'll use to fire off the "prompt" event.
  111. *
  112. * The idea being that we'd receive chars, processing lines.
  113. * And if we have something in server_prompt start the timer.
  114. * If we receive more, process lines, if !server_prompt.empty()
  115. * reset the timer.
  116. *
  117. * If the timer fires (and isn't aborted), fire off the prompt.
  118. *
  119. * I'm not so sure about this -- because this adds delay to the
  120. * proxy. [It might be better to just fire off "some" text, and
  121. * have it ignored, rather then adding a delay.] Or, this might
  122. * not matter at all, I'm thinking milliseconds here!
  123. */
  124. boost::asio::high_resolution_timer timer_;
  125. /**
  126. * What characters have been received from the server,
  127. * that weren't \n terminated?
  128. *
  129. * This needs to be reset/cleared if there's a \r (carriage return).
  130. */
  131. std::string server_prompt;
  132. /**
  133. * FAIL-WHALE: This was supposed to hold the number of characters
  134. * already sent to the user at this point in time, but I'm having
  135. * a hard time tracking those.
  136. *
  137. */
  138. int server_sent;
  139. /**
  140. * The client read buffer.
  141. *
  142. * This is too big, we don't get that many characters from the client.
  143. *
  144. */
  145. char read_buffer[MAX_BUFFER + 1];
  146. /**
  147. * The server read buffer.
  148. * This is MAX_BUFFER + 1 (to store the \0 to terminate the string)
  149. */
  150. char server_buffer[MAX_BUFFER + 1];
  151. /**
  152. * The rlogin information received from the client.
  153. *
  154. * We check this, and if it isn't valid, we spoof some rlogin
  155. * connection.
  156. */
  157. std::string rlogin_auth;
  158. /**
  159. * The username passed in via rlogin. We need this so we know what
  160. * name we need to store the data under.
  161. */
  162. std::string rlogin_name;
  163. std::string host;
  164. std::string port;
  165. /**
  166. * Are we connected to the server?
  167. *
  168. * Don't shutdown the server socket if we aren't connected.
  169. */
  170. bool connected = false;
  171. };
  172. /*
  173. maybe move the resolver part to the server, so I don't need io_service?
  174. I'm not sure what the socket connection part is going to need just yet,
  175. so I probably won't move that just yet. [NNY!]
  176. */
  177. class Server {
  178. public:
  179. Server(boost::asio::io_service &io_service,
  180. const boost::asio::ip::tcp::endpoint &endpoint, std::string host,
  181. std::string port);
  182. private:
  183. void do_accept(void);
  184. boost::asio::io_service &io_service_;
  185. boost::asio::ip::tcp::acceptor acceptor_;
  186. /**
  187. * The host to connect to (from config)
  188. */
  189. std::string host_;
  190. /**
  191. * The port to connect to (from config)
  192. *
  193. */
  194. std::string port_;
  195. };
  196. // The simple way to get the boost logging to log file and line number
  197. // information.
  198. #include <iomanip>
  199. #include <string.h>
  200. // I don't gain anything from this being constexpr
  201. const char *trim_path(const char *filepath);
  202. #define BUGZ_LOG(severity) \
  203. BOOST_LOG_TRIVIAL(severity) \
  204. << "(" << std::setw(15) << trim_path(__FILE__) << ":" << std::setw(4) \
  205. << std::left << __LINE__ << ") "
  206. #endif