|
@@ -7,6 +7,7 @@
|
|
|
|
|
|
#include <regex>
|
|
|
|
|
|
+#include "config.h"
|
|
|
#include "session.h"
|
|
|
|
|
|
#include <string>
|
|
@@ -61,33 +62,32 @@ std::string clean_string(const std::string &source) {
|
|
|
return clean;
|
|
|
}
|
|
|
|
|
|
-session::session(boost::asio::ip::tcp::socket socket,
|
|
|
+Session::Session(boost::asio::ip::tcp::socket socket,
|
|
|
boost::asio::io_service &io_service, std::string hostname,
|
|
|
std::string port)
|
|
|
: socket_(std::move(socket)),
|
|
|
io_service_{io_service}, resolver_{io_service}, server_{io_service},
|
|
|
timer_{io_service}, host{hostname}, port{port} {
|
|
|
server_sent = 0;
|
|
|
+ time_ms = stoi(from_config("prompt_timeout", "50"));
|
|
|
}
|
|
|
|
|
|
-void session::start(void) {
|
|
|
+void Session::start(void) {
|
|
|
// BOOST_LOG_NAMED_SCOPE();
|
|
|
|
|
|
// If I want the file and line number information, here's how to do it:
|
|
|
// BUGZ_LOG(info) << boost::format("(%1%:%2%) ") % __FILE__ % __LINE__
|
|
|
-
|
|
|
- BUGZ_LOG(info) << "session::start()";
|
|
|
+
|
|
|
+ BUGZ_LOG(info) << "Session::start()";
|
|
|
auto self(shared_from_this());
|
|
|
// read_buffer.reserve(1024);
|
|
|
// do_write("Welcome!\n");
|
|
|
client_read();
|
|
|
}
|
|
|
|
|
|
-session::~session() {
|
|
|
- BUGZ_LOG(info) << "~session";
|
|
|
-}
|
|
|
+Session::~Session() { BUGZ_LOG(info) << "~Session"; }
|
|
|
|
|
|
-void session::parse_auth(void) {
|
|
|
+void Session::parse_auth(void) {
|
|
|
// how many nulls should I be seeing?
|
|
|
// \0user\0pass\0terminal/SPEED\0
|
|
|
// If I don't have a proper rlogin value here, it isn't going
|
|
@@ -99,9 +99,9 @@ void session::parse_auth(void) {
|
|
|
rlogin_name = "?";
|
|
|
}
|
|
|
|
|
|
-void session::on_connect(const boost::system::error_code error) {
|
|
|
+void Session::on_connect(const boost::system::error_code error) {
|
|
|
// We've connected to the server! WOOT WOOT!
|
|
|
- // BOOST_LOG_NAMED_SCOPE("session");
|
|
|
+ // BOOST_LOG_NAMED_SCOPE("Session");
|
|
|
|
|
|
if (!error) {
|
|
|
BUGZ_LOG(info) << "Connected to " << host;
|
|
@@ -129,11 +129,11 @@ void session::on_connect(const boost::system::error_code error) {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-void session::dispatch_line(std::string line) {
|
|
|
+void Session::dispatch_line(std::string line) {
|
|
|
// Does this have \n\r still on it? I don't want them.
|
|
|
|
|
|
std::string temp = clean_string(line);
|
|
|
- BUGZ_LOG(info) << "SL: " << temp;
|
|
|
+ BUGZ_LOG(info) << "SL: " << temp;
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -146,7 +146,7 @@ what was just added to server_prompt.
|
|
|
What about \r, \b ? Should that "reset" the server_prompt?
|
|
|
|
|
|
*/
|
|
|
-void session::process_lines(std::string &received) {
|
|
|
+void Session::process_lines(std::string &received) {
|
|
|
// break server_prompt into lines and send/process one by one.
|
|
|
|
|
|
size_t pos, rpos;
|
|
@@ -198,9 +198,13 @@ void session::process_lines(std::string &received) {
|
|
|
|
|
|
// How should I handle \r in lines? For now, remove it
|
|
|
// but LOG that we did.
|
|
|
+
|
|
|
+ replace(part, "\r", "");
|
|
|
+ /*
|
|
|
if (replace(part, "\r", "")) {
|
|
|
BUGZ_LOG(warning) << "\\r removed from line";
|
|
|
}
|
|
|
+ */
|
|
|
|
|
|
dispatch_line(part);
|
|
|
}
|
|
@@ -213,9 +217,45 @@ void session::process_lines(std::string &received) {
|
|
|
// std::string clean = clean_string(received);
|
|
|
// BUGZ_LOG(error) << "show_client/leftovers:" << clean;
|
|
|
}
|
|
|
+
|
|
|
+ // check the server prompt here:
|
|
|
+ if ((pos = server_prompt.rfind('\r')) != std::string::npos) {
|
|
|
+ // server_prompt contains \r, remove it.
|
|
|
+ server_prompt = server_prompt.substr(pos + 1);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!server_prompt.empty()) {
|
|
|
+ // We have something remaining -- start the timer!
|
|
|
+ set_timer();
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+void Session::set_timer(void) {
|
|
|
+ timer_.expires_after(std::chrono::milliseconds(time_ms));
|
|
|
+ timer_.async_wait(
|
|
|
+ boost::bind(&Session::on_timer, this, boost::asio::placeholders::error));
|
|
|
+}
|
|
|
+
|
|
|
+void Session::reset_timer(void) { timer_.cancel(); }
|
|
|
+void Session::on_timer(const boost::system::error_code error) {
|
|
|
+ if (error != boost::asio::error::operation_aborted) {
|
|
|
+ // Ok, VALID timeout
|
|
|
+ if (!server_prompt.empty()) {
|
|
|
+ // Here's what is happening:
|
|
|
+ // SP: [ESC[2JESC[H]
|
|
|
+ // which after clean_string is empty.
|
|
|
+
|
|
|
+ std::string clean = clean_string(server_prompt);
|
|
|
+ if (!clean.empty()) {
|
|
|
+ BUGZ_LOG(warning) << "SP: [" << clean << "]";
|
|
|
+ // emit
|
|
|
+ }
|
|
|
+ // BUGZ_LOG(trace) << "SP: [" << server_prompt << "]";
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
-void session::server_read(void) {
|
|
|
+void Session::server_read(void) {
|
|
|
auto self(shared_from_this());
|
|
|
|
|
|
boost::asio::async_read(
|
|
@@ -250,7 +290,7 @@ void session::server_read(void) {
|
|
|
});
|
|
|
}
|
|
|
|
|
|
-void session::on_resolve(
|
|
|
+void Session::on_resolve(
|
|
|
const boost::system::error_code error,
|
|
|
const boost::asio::ip::tcp::resolver::results_type results) {
|
|
|
//
|
|
@@ -261,12 +301,12 @@ void session::on_resolve(
|
|
|
boost::asio::ip::tcp::endpoint const &endpoint = *results;
|
|
|
|
|
|
server_.async_connect(endpoint,
|
|
|
- boost::bind(&session::on_connect, this,
|
|
|
+ boost::bind(&Session::on_connect, this,
|
|
|
boost::asio::placeholders::error));
|
|
|
|
|
|
} else {
|
|
|
// TO DO:
|
|
|
- // BOOST_LOG_NAMED_SCOPE("session");
|
|
|
+ // BOOST_LOG_NAMED_SCOPE("Session");
|
|
|
BUGZ_LOG(error) << "Unable to resolve: " << host;
|
|
|
std::string output =
|
|
|
str(boost::format("Unable to resolve: %1%\n\r") % host);
|
|
@@ -276,7 +316,7 @@ void session::on_resolve(
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-void session::client_read(void) {
|
|
|
+void Session::client_read(void) {
|
|
|
auto self(shared_from_this());
|
|
|
|
|
|
boost::asio::async_read( // why can't I async_read_some here?
|
|
@@ -301,19 +341,19 @@ void session::client_read(void) {
|
|
|
|
|
|
/* // this fails, and I'm not sure why. I've used code like this
|
|
|
before. resolver_.async_resolve( host, port, std::bind(
|
|
|
- &session::on_resolve, this, _1, _2)); */
|
|
|
+ &Session::on_resolve, this, _1, _2)); */
|
|
|
|
|
|
// This example shows using boost::bind, which WORKS.
|
|
|
// https://stackoverflow.com/questions/6025471/bind-resolve-handler-to-resolver-async-resolve-using-boostasio
|
|
|
resolver_.async_resolve(
|
|
|
host, port,
|
|
|
- boost::bind(&session::on_resolve, this,
|
|
|
+ boost::bind(&Session::on_resolve, this,
|
|
|
boost::asio::placeholders::error,
|
|
|
boost::asio::placeholders::iterator));
|
|
|
|
|
|
} else if (length) {
|
|
|
// Proxy Active?
|
|
|
- // BOOST_LOG_NAMED_SCOPE("session");
|
|
|
+ // BOOST_LOG_NAMED_SCOPE("Session");
|
|
|
|
|
|
if (talk_direct)
|
|
|
to_server(read_buffer);
|
|
@@ -325,14 +365,14 @@ void session::client_read(void) {
|
|
|
} else {
|
|
|
BUGZ_LOG(warning) << "C: read_failed";
|
|
|
if (connected) {
|
|
|
- BUGZ_LOG(warning) << "server.shutdown()";
|
|
|
+ BUGZ_LOG(warning) << "Server.shutdown()";
|
|
|
server_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
-void session::to_client(std::string message) {
|
|
|
+void Session::to_client(std::string message) {
|
|
|
auto self(shared_from_this());
|
|
|
// output the cleaned string (so I can see what we're sending in the
|
|
|
// logs)
|
|
@@ -346,17 +386,16 @@ void session::to_client(std::string message) {
|
|
|
if (!ec) {
|
|
|
|
|
|
} else {
|
|
|
- BUGZ_LOG(warning)
|
|
|
- << "C: write failed? closed? server.shutdown()";
|
|
|
+ BUGZ_LOG(warning) << "C: write failed? closed? Server.shutdown()";
|
|
|
if (connected) {
|
|
|
- BUGZ_LOG(warning) << "server.shutdown()";
|
|
|
+ BUGZ_LOG(warning) << "Server.shutdown()";
|
|
|
server_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
-void session::to_server(std::string message) {
|
|
|
+void Session::to_server(std::string message) {
|
|
|
auto self(shared_from_this());
|
|
|
boost::asio::async_write(
|
|
|
server_, boost::asio::buffer(message),
|
|
@@ -364,14 +403,13 @@ void session::to_server(std::string message) {
|
|
|
if (!ec) {
|
|
|
|
|
|
} else {
|
|
|
- BUGZ_LOG(warning)
|
|
|
- << "S: write failed? closed? socket.shutdown()";
|
|
|
+ BUGZ_LOG(warning) << "S: write failed? closed? socket.shutdown()";
|
|
|
socket_.shutdown(boost::asio::ip::tcp::socket::shutdown_both);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
-server::server(boost::asio::io_service &io_service,
|
|
|
+Server::Server(boost::asio::io_service &io_service,
|
|
|
const boost::asio::ip::tcp::endpoint &endpoint, std::string host,
|
|
|
std::string port)
|
|
|
: io_service_{io_service}, acceptor_{io_service_, endpoint}, host_{host},
|
|
@@ -386,12 +424,12 @@ server::server(boost::asio::io_service &io_service,
|
|
|
* session to automatically clean up when it is no longer active/has anything
|
|
|
* running in the reactor.
|
|
|
*/
|
|
|
-void server::do_accept(void) {
|
|
|
+void Server::do_accept(void) {
|
|
|
acceptor_.async_accept([this](boost::system::error_code ec,
|
|
|
boost::asio::ip::tcp::socket socket) {
|
|
|
if (!ec) {
|
|
|
- BUGZ_LOG(info) << "server::do_accept()";
|
|
|
- std::make_shared<session>(std::move(socket), io_service_, host_, port_)
|
|
|
+ BUGZ_LOG(info) << "Server::do_accept()";
|
|
|
+ std::make_shared<Session>(std::move(socket), io_service_, host_, port_)
|
|
|
->start();
|
|
|
}
|
|
|
|
|
@@ -401,13 +439,13 @@ void server::do_accept(void) {
|
|
|
|
|
|
/**
|
|
|
* Clean up the trailing ../ in __FILE__
|
|
|
- *
|
|
|
+ *
|
|
|
* This is used by the logging macro.
|
|
|
- *
|
|
|
- * @param filepath
|
|
|
- * @return const char*
|
|
|
+ *
|
|
|
+ * @param filepath
|
|
|
+ * @return const char*
|
|
|
*/
|
|
|
-const char * trim_path( const char * filepath ) {
|
|
|
+const char *trim_path(const char *filepath) {
|
|
|
if (strncmp(filepath, "../", 3) == 0) {
|
|
|
filepath += 3;
|
|
|
}
|