|
@@ -1,119 +1,65 @@
|
|
|
#include <boost/bind.hpp>
|
|
|
-#include <iostream>
|
|
|
-
|
|
|
#include <boost/format.hpp>
|
|
|
+#include <functional>
|
|
|
+#include <iostream>
|
|
|
// #include <boost/log/core.hpp>
|
|
|
// #include <boost/log/trivial.hpp>
|
|
|
|
|
|
#include <regex>
|
|
|
+#include <string>
|
|
|
|
|
|
#include "config.h"
|
|
|
+#include "galaxy.h"
|
|
|
#include "logging.h"
|
|
|
#include "session.h"
|
|
|
-
|
|
|
-#include "galaxy.h"
|
|
|
-
|
|
|
-#include <string>
|
|
|
+#include "utils.h"
|
|
|
|
|
|
// #include <boost/log/attributes/named_scope.hpp>
|
|
|
|
|
|
-bool replace(std::string &str, const std::string &from, const std::string &to) {
|
|
|
- size_t start_pos = str.find(from);
|
|
|
- if (start_pos == std::string::npos)
|
|
|
- return false;
|
|
|
- do {
|
|
|
- str.replace(start_pos, from.length(), to);
|
|
|
- } while ((start_pos = str.find(from)) != std::string::npos);
|
|
|
- return true;
|
|
|
-}
|
|
|
-
|
|
|
-bool replace(std::string &str, const char *from, const char *to) {
|
|
|
- size_t start_pos = str.find(from);
|
|
|
- if (start_pos == std::string::npos)
|
|
|
- return false;
|
|
|
- do {
|
|
|
- str.replace(start_pos, strlen(from), to);
|
|
|
- } while ((start_pos = str.find(from)) != std::string::npos);
|
|
|
- return true;
|
|
|
-}
|
|
|
-
|
|
|
-void ansi_clean(std::string &str) {
|
|
|
- static std::regex ansi_cleaner("\x1b\[[0-9;]*[A-Zmh]",
|
|
|
- std::regex_constants::ECMAScript);
|
|
|
- str = std::regex_replace(str, ansi_cleaner, "");
|
|
|
-}
|
|
|
-
|
|
|
-void high_ascii(std::string &str) {
|
|
|
- // the + replaces all of them into one. I want each high ascii replaced with
|
|
|
- // #.
|
|
|
- static std::regex high_cleaner("[\x80-\xff]",
|
|
|
- std::regex_constants::ECMAScript);
|
|
|
- str = std::regex_replace(str, high_cleaner, "#");
|
|
|
-}
|
|
|
-
|
|
|
-std::smatch ansi_newline(const std::string &str) {
|
|
|
- static std::regex ansi_nl("\x1b\[[0-9;]*[JK]",
|
|
|
- std::regex_constants::ECMAScript);
|
|
|
- std::smatch m;
|
|
|
- std::regex_search(str, m, ansi_nl);
|
|
|
- return m;
|
|
|
-}
|
|
|
-
|
|
|
-std::string clean_string(const std::string &source) {
|
|
|
- std::string clean = source;
|
|
|
-
|
|
|
- replace(clean, "\n", "\\n");
|
|
|
- replace(clean, "\r", "\\r");
|
|
|
- replace(clean, "\b", "\\b");
|
|
|
-
|
|
|
- // ANSI too
|
|
|
- ansi_clean(clean);
|
|
|
- // BUGZ_LOG(error) << "cleaned: " << clean;
|
|
|
- high_ascii(clean);
|
|
|
-
|
|
|
- replace(clean, "\x1b", "^");
|
|
|
-
|
|
|
- return clean;
|
|
|
-}
|
|
|
-
|
|
|
-std::vector<std::string> split(const std::string &line) {
|
|
|
- static std::regex rx_split("[^\\s]+");
|
|
|
- std::vector<std::string> results;
|
|
|
-
|
|
|
- for (auto it = std::sregex_iterator(line.begin(), line.end(), rx_split);
|
|
|
- it != std::sregex_iterator(); ++it) {
|
|
|
- results.push_back(it->str());
|
|
|
- }
|
|
|
- return results;
|
|
|
-}
|
|
|
-
|
|
|
Session::Session(boost::asio::ip::tcp::socket socket,
|
|
|
boost::asio::io_service &io_service, std::string hostname,
|
|
|
std::string port)
|
|
|
- : main(this), socket_(std::move(socket)), io_service_{io_service},
|
|
|
- resolver_{io_service}, server_{io_service}, prompt_timer_{io_service},
|
|
|
- keep_alive_{io_service}, host{hostname}, port{port} {
|
|
|
+ : socket_(std::move(socket)),
|
|
|
+ io_service_{io_service},
|
|
|
+ resolver_{io_service},
|
|
|
+ server_{io_service},
|
|
|
+ prompt_timer_{io_service},
|
|
|
+ keep_alive_{io_service},
|
|
|
+ host{hostname},
|
|
|
+ port{port} {
|
|
|
BUGZ_LOG(info) << "Session::Session()";
|
|
|
// server_sent = 0;
|
|
|
time_ms = 50;
|
|
|
- if (CONFIG["prompt_timeout"])
|
|
|
- time_ms = CONFIG["prompt_timeout"].as<int>();
|
|
|
+ if (CONFIG["prompt_timeout"]) time_ms = CONFIG["prompt_timeout"].as<int>();
|
|
|
|
|
|
keepalive_secs = 45;
|
|
|
- if (CONFIG["keepalive"])
|
|
|
- keepalive_secs = CONFIG["keepalive"].as<int>();
|
|
|
-}
|
|
|
+ if (CONFIG["keepalive"]) keepalive_secs = CONFIG["keepalive"].as<int>();
|
|
|
|
|
|
-void Session::start(void) {
|
|
|
- // BOOST_LOG_NAMED_SCOPE();
|
|
|
+ // Initialize the director
|
|
|
+ director.to_server = boost::bind(&Session::to_server, this, _1);
|
|
|
+ director.to_client = boost::bind(&Session::to_client, this, _1);
|
|
|
|
|
|
- // If I want the file and line number information, here's how to do it:
|
|
|
- // BUGZ_LOG(info) << boost::format("(%1%:%2%) ") % __FILE__ % __LINE__
|
|
|
+ // replace emit_ with below: if (director.server_line)
|
|
|
+ // director.server_line(s);
|
|
|
+ /*
|
|
|
+ emit_server_line = [this](const std::string &s) {
|
|
|
+ if (director.server_line) {
|
|
|
+ director.server_line(s);
|
|
|
+ }
|
|
|
+ };
|
|
|
+ emit_server_prompt = [this](const std::string &s) {
|
|
|
+ if (director.server_prompt) {
|
|
|
+ director.server_prompt(s);
|
|
|
+ }
|
|
|
+ };
|
|
|
+ emit_client_input ... => director.client_input
|
|
|
+ */
|
|
|
+}
|
|
|
|
|
|
+void Session::start(void) {
|
|
|
BUGZ_LOG(info) << "Session::start()";
|
|
|
- auto self(shared_from_this());
|
|
|
- // read_buffer.reserve(1024);
|
|
|
- // do_write("Welcome!\n");
|
|
|
+ // auto self(shared_from_this());
|
|
|
+
|
|
|
client_read();
|
|
|
}
|
|
|
|
|
@@ -156,8 +102,6 @@ void Session::on_connect(const boost::system::error_code error) {
|
|
|
// We've connected to the server! WOOT WOOT!
|
|
|
// BOOST_LOG_NAMED_SCOPE("Session");
|
|
|
|
|
|
- SL_parser = nullptr;
|
|
|
-
|
|
|
if (!error) {
|
|
|
BUGZ_LOG(info) << "Connected to " << host;
|
|
|
to_client("Connected...\n\r");
|
|
@@ -195,7 +139,9 @@ void Session::on_connect(const boost::system::error_code error) {
|
|
|
*/
|
|
|
void Session::on_server_line(const std::string &line) {
|
|
|
BUGZ_LOG(info) << "SL: [" << line << "]";
|
|
|
+ director.server_line(line);
|
|
|
|
|
|
+#ifdef DECOUPLE
|
|
|
if (line.find("TradeWars Game Server ") != std::string::npos) {
|
|
|
to_client("\rTradeWars Proxy v2++ READY (~ or ESC to activate)\n\r");
|
|
|
game = 0;
|
|
@@ -228,8 +174,7 @@ void Session::on_server_line(const std::string &line) {
|
|
|
BUGZ_LOG(warning) << "GAME " << game << " activated!";
|
|
|
}
|
|
|
// not needed (handled by above Game Server check).
|
|
|
- if (ch == 'Q')
|
|
|
- game = 0;
|
|
|
+ if (ch == 'Q') game = 0;
|
|
|
}
|
|
|
|
|
|
// Do I need to run through the tests (below) before calling the parser here?
|
|
@@ -267,93 +212,8 @@ void Session::on_server_line(const std::string &line) {
|
|
|
// should I have an internal emit_server_line for parsing sections?
|
|
|
// rather then having a weird state machine to track where we are?
|
|
|
|
|
|
- if (emit_server_line)
|
|
|
- emit_server_line(line);
|
|
|
-}
|
|
|
-
|
|
|
-void Session::SL_cimline(const std::string &line) {
|
|
|
- if (line == ": ENDINTERROG") {
|
|
|
- SL_parser = nullptr;
|
|
|
- return;
|
|
|
- }
|
|
|
- if (line == ": ") {
|
|
|
- // do I need to do anything special here for this?
|
|
|
- return;
|
|
|
- }
|
|
|
- if (line.empty()) {
|
|
|
- SL_parser = nullptr;
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- // parse cimline
|
|
|
- size_t pos = line.find('%');
|
|
|
- std::string work = line;
|
|
|
-
|
|
|
- if (pos == line.npos) {
|
|
|
- // warpcim
|
|
|
- BUGZ_LOG(fatal) << "warpcim: [" << line << "]";
|
|
|
- auto warps = split(line);
|
|
|
- sector_warps sw;
|
|
|
- for( auto const & w : warps) {
|
|
|
- if (sw.sector == 0) {
|
|
|
- sw.sector = stoi(w);
|
|
|
- } else {
|
|
|
- sw.add(stoi(w));
|
|
|
- }
|
|
|
- }
|
|
|
- BUGZ_LOG(fatal) << "warpcim: " << sw;
|
|
|
-
|
|
|
- } else {
|
|
|
- // portcim
|
|
|
- struct port p = parse_portcim(line);
|
|
|
- if (p.sector == 0)
|
|
|
- BUGZ_LOG(fatal) << "portcim: [" << line << "]";
|
|
|
- else
|
|
|
- BUGZ_LOG(fatal) << "portcim: " << p;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void Session::SL_thiefline(const std::string &line) {
|
|
|
- size_t pos = line.find("Suddenly you're Busted!");
|
|
|
- bool busted = pos != line.npos;
|
|
|
- if (busted) {
|
|
|
- BUGZ_LOG(fatal) << "set bust";
|
|
|
- SL_parser = nullptr;
|
|
|
- } else {
|
|
|
- pos = line.find("(You realize the guards saw you last time!)");
|
|
|
- if (pos != line.npos)
|
|
|
- SL_parser = nullptr;
|
|
|
- }
|
|
|
-
|
|
|
- // Are those the two ways to exit from this state?
|
|
|
-}
|
|
|
-void Session::SL_sectorline(const std::string &line) {
|
|
|
- BUGZ_LOG(fatal) << "sectorline: [" << line << "]";
|
|
|
-}
|
|
|
-
|
|
|
-void Session::SL_portline(const std::string &line) {
|
|
|
- if (line.empty()) {
|
|
|
- SL_parser = nullptr;
|
|
|
- return;
|
|
|
- }
|
|
|
- BUGZ_LOG(info) << "portline : " << line;
|
|
|
- size_t pos = line.find('%');
|
|
|
- if (pos != line.npos) {
|
|
|
- // Ok, this is a valid portline
|
|
|
- std::string work = line;
|
|
|
- replace(work, "Fuel Ore", "Fuel");
|
|
|
- BUGZ_LOG(fatal) << "re.split? : [" << work << "]";
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void Session::SL_warpline(const std::string &line) {
|
|
|
- if (line.empty()) {
|
|
|
- SL_parser = nullptr;
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- // process warp line
|
|
|
- BUGZ_LOG(fatal) << "warpline: [" << line << "]";
|
|
|
+ if (emit_server_line) emit_server_line(line);
|
|
|
+#endif
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -401,7 +261,6 @@ void Session::process_lines(std::string &received) {
|
|
|
// I also need to break on r"\x1b[\[0-9;]*JK", treat these like \n
|
|
|
|
|
|
while ((pos = server_prompt.find('\n', 0)) != std::string::npos) {
|
|
|
-
|
|
|
std::string line;
|
|
|
|
|
|
std::smatch m = ansi_newline(server_prompt);
|
|
@@ -522,11 +381,11 @@ void Session::set_prompt_timer(void) {
|
|
|
|
|
|
void Session::reset_prompt_timer(void) { prompt_timer_.cancel(); }
|
|
|
|
|
|
-void Session::on_server_prompt(const std::string &prompt) {
|
|
|
+// probably no longer needed --
|
|
|
+void Session::on_server_prompt(const std::string &prompt,
|
|
|
+ const std::string &raw_prompt) {
|
|
|
BUGZ_LOG(warning) << "SP: [" << prompt << "]";
|
|
|
- if (emit_server_prompt) {
|
|
|
- emit_server_prompt(prompt);
|
|
|
- }
|
|
|
+ director.server_prompt(prompt, raw_prompt);
|
|
|
}
|
|
|
|
|
|
void Session::on_prompt_timeout(const boost::system::error_code error) {
|
|
@@ -539,7 +398,7 @@ void Session::on_prompt_timeout(const boost::system::error_code error) {
|
|
|
|
|
|
std::string clean = clean_string(server_prompt);
|
|
|
if (!clean.empty()) {
|
|
|
- on_server_prompt(clean);
|
|
|
+ on_server_prompt(clean, server_prompt);
|
|
|
}
|
|
|
// BUGZ_LOG(trace) << "SP: [" << server_prompt << "]";
|
|
|
}
|
|
@@ -609,9 +468,10 @@ void Session::on_resolve(
|
|
|
}
|
|
|
|
|
|
void Session::client_input(const std::string &input) {
|
|
|
-
|
|
|
BUGZ_LOG(info) << "CI: " << input;
|
|
|
+ director.client_input(input);
|
|
|
|
|
|
+#ifdef DECOUPLE
|
|
|
// Is "proxy" active
|
|
|
if (active) {
|
|
|
// do something amazing with the user's input.
|
|
@@ -621,8 +481,9 @@ void Session::client_input(const std::string &input) {
|
|
|
BUGZ_LOG(trace) << "CI: ACTIVATE prompt shows: [" << prompt << "]";
|
|
|
|
|
|
if (prompt == "Selection (? for menu): ") {
|
|
|
- to_client("\n\rThere's not much we can do here. Activate in-game at a "
|
|
|
- "Command prompt.\n\r");
|
|
|
+ to_client(
|
|
|
+ "\n\rThere's not much we can do here. Activate in-game at a "
|
|
|
+ "Command prompt.\n\r");
|
|
|
to_client(get_prompt());
|
|
|
return;
|
|
|
}
|
|
@@ -630,8 +491,9 @@ void Session::client_input(const std::string &input) {
|
|
|
// easter-eggs:
|
|
|
|
|
|
if (prompt == "Enter your choice: ") {
|
|
|
- to_client("\n\r\x1b[1;36mI'd choose \x1b[1;37m`T`\x1b[1;36m, but "
|
|
|
- "that's how I was coded.\n\r");
|
|
|
+ to_client(
|
|
|
+ "\n\r\x1b[1;36mI'd choose \x1b[1;37m`T`\x1b[1;36m, but "
|
|
|
+ "that's how I was coded.\n\r");
|
|
|
to_client(get_prompt());
|
|
|
return;
|
|
|
}
|
|
@@ -682,8 +544,10 @@ void Session::client_input(const std::string &input) {
|
|
|
if (emit_client_input) {
|
|
|
emit_client_input(input);
|
|
|
}
|
|
|
+#endif
|
|
|
}
|
|
|
|
|
|
+/*
|
|
|
DispatchSettings Session::save_settings(void) {
|
|
|
DispatchSettings ss{emit_server_line, emit_server_prompt, emit_client_input,
|
|
|
show_client, talk_direct};
|
|
@@ -700,7 +564,7 @@ void Session::restore_settings(const DispatchSettings &ss) {
|
|
|
|
|
|
void Session::proxy_activate(void) {
|
|
|
active = true;
|
|
|
- start_keepin_alive(); // kickstart the keepalive timer
|
|
|
+ start_keepin_alive(); // kickstart the keepalive timer
|
|
|
main.setNotify([this](void) { this->proxy_deactivate(); });
|
|
|
main.activate();
|
|
|
}
|
|
@@ -711,11 +575,12 @@ void Session::proxy_deactivate(void) {
|
|
|
to_client(get_prompt());
|
|
|
// to_client(" \b");
|
|
|
}
|
|
|
+*/
|
|
|
|
|
|
void Session::client_read(void) {
|
|
|
auto self(shared_from_this());
|
|
|
|
|
|
- boost::asio::async_read( // why can't I async_read_some here?
|
|
|
+ boost::asio::async_read( // why can't I async_read_some here?
|
|
|
socket_, boost::asio::buffer(read_buffer, sizeof(read_buffer) - 1),
|
|
|
boost::asio::transfer_at_least(1),
|
|
|
[this, self](boost::system::error_code ec, std::size_t length) {
|
|
@@ -779,7 +644,6 @@ void Session::to_client(const std::string &message) {
|
|
|
socket_, boost::asio::buffer(message),
|
|
|
[this, self](boost::system::error_code ec, std::size_t /*length*/) {
|
|
|
if (!ec) {
|
|
|
-
|
|
|
} else {
|
|
|
BUGZ_LOG(warning) << "2C: write failed? closed? Server.shutdown()";
|
|
|
if (connected) {
|
|
@@ -792,11 +656,11 @@ void Session::to_client(const std::string &message) {
|
|
|
|
|
|
void Session::to_server(const std::string &message) {
|
|
|
auto self(shared_from_this());
|
|
|
+ BUGZ_LOG(trace) << "2S: " << message;
|
|
|
boost::asio::async_write(
|
|
|
server_, boost::asio::buffer(message),
|
|
|
[this, self](boost::system::error_code ec, std::size_t /*length*/) {
|
|
|
if (!ec) {
|
|
|
-
|
|
|
} else {
|
|
|
BUGZ_LOG(warning) << "S: write failed? closed? socket.shutdown()";
|
|
|
// we're no longer connected.
|
|
@@ -805,7 +669,7 @@ void Session::to_server(const std::string &message) {
|
|
|
}
|
|
|
});
|
|
|
|
|
|
- if (active) {
|
|
|
+ if (director.active) {
|
|
|
start_keepin_alive();
|
|
|
}
|
|
|
}
|
|
@@ -820,7 +684,7 @@ void Session::start_keepin_alive(void) {
|
|
|
void Session::stayin_alive(const boost::system::error_code error) {
|
|
|
if (error != boost::asio::error::operation_aborted) {
|
|
|
// stayin' alive, stayin' alive...
|
|
|
- if (active) {
|
|
|
+ if (director.active) {
|
|
|
to_server(" ");
|
|
|
BUGZ_LOG(warning) << "Session::stayin_alive()";
|
|
|
}
|
|
@@ -830,8 +694,11 @@ void Session::stayin_alive(const boost::system::error_code error) {
|
|
|
Server::Server(boost::asio::io_service &io_service,
|
|
|
const boost::asio::ip::tcp::endpoint &endpoint,
|
|
|
const std::string &host, const std::string &port)
|
|
|
- : io_service_{io_service}, acceptor_{io_service_, endpoint},
|
|
|
- signal_{io_service, SIGUSR1, SIGTERM}, host_{host}, port_{port} {
|
|
|
+ : io_service_{io_service},
|
|
|
+ acceptor_{io_service_, endpoint},
|
|
|
+ signal_{io_service, SIGUSR1, SIGTERM},
|
|
|
+ host_{host},
|
|
|
+ port_{port} {
|
|
|
keep_accepting = true;
|
|
|
|
|
|
BUGZ_LOG(info) << "Server::Server()";
|