2018-01-03 21:34:26 +08:00
|
|
|
#include <fstream>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include <fmt/format.h>
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include <glog/logging.h>
|
|
|
|
|
2018-01-24 19:16:14 +08:00
|
|
|
#include "communication/rpc/server.hpp"
|
2018-01-03 21:34:26 +08:00
|
|
|
#include "messages.hpp"
|
|
|
|
#include "utils/signals/handler.hpp"
|
|
|
|
#include "utils/terminate_handler.hpp"
|
|
|
|
|
|
|
|
using namespace communication::rpc;
|
|
|
|
using namespace std::literals::chrono_literals;
|
|
|
|
|
|
|
|
DEFINE_string(interface, "127.0.0.1",
|
|
|
|
"Communication interface on which to listen.");
|
|
|
|
DEFINE_string(port, "10000", "Communication port on which to listen.");
|
|
|
|
DEFINE_string(log, "log.txt", "Entries log file");
|
|
|
|
|
|
|
|
volatile sig_atomic_t is_shutting_down = 0;
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
google::SetUsageMessage("Raft RPC Server");
|
|
|
|
|
|
|
|
gflags::ParseCommandLineFlags(&argc, &argv, true);
|
|
|
|
google::InitGoogleLogging(argv[0]);
|
|
|
|
|
|
|
|
// Unhandled exception handler init.
|
|
|
|
std::set_terminate(&terminate_handler);
|
|
|
|
|
2018-02-23 17:56:56 +08:00
|
|
|
Server server(io::network::Endpoint(FLAGS_interface, stoul(FLAGS_port)));
|
2018-01-03 21:34:26 +08:00
|
|
|
std::ofstream log(FLAGS_log, std::ios_base::app);
|
|
|
|
|
|
|
|
// Handler for regular termination signals.
|
2018-01-10 20:56:12 +08:00
|
|
|
auto shutdown = [&log]() {
|
2018-01-03 21:34:26 +08:00
|
|
|
if (is_shutting_down) return;
|
|
|
|
is_shutting_down = 1;
|
|
|
|
log.close();
|
|
|
|
exit(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Prevent handling shutdown inside a shutdown. For example, SIGINT handler
|
|
|
|
// being interrupted by SIGTERM before is_shutting_down is set, thus causing
|
|
|
|
// double shutdown.
|
|
|
|
sigset_t block_shutdown_signals;
|
|
|
|
sigemptyset(&block_shutdown_signals);
|
|
|
|
sigaddset(&block_shutdown_signals, SIGTERM);
|
|
|
|
sigaddset(&block_shutdown_signals, SIGINT);
|
|
|
|
|
|
|
|
CHECK(SignalHandler::RegisterHandler(Signal::Terminate, shutdown,
|
|
|
|
block_shutdown_signals))
|
|
|
|
<< "Unable to register SIGTERM handler!";
|
|
|
|
CHECK(SignalHandler::RegisterHandler(Signal::Interupt, shutdown,
|
|
|
|
block_shutdown_signals))
|
|
|
|
<< "Unable to register SIGINT handler!";
|
|
|
|
|
|
|
|
// Example callback.
|
|
|
|
server.Register<AppendEntry>([&log](const AppendEntryReq &request) {
|
|
|
|
log << request.val << std::endl;
|
|
|
|
log.flush();
|
|
|
|
LOG(INFO) << fmt::format("AppendEntry: {}", request.val);
|
|
|
|
return std::make_unique<AppendEntryRes>(200, FLAGS_interface,
|
|
|
|
stol(FLAGS_port));
|
|
|
|
});
|
|
|
|
|
|
|
|
LOG(INFO) << "Raft RPC server started";
|
|
|
|
// Sleep until shutdown detected.
|
|
|
|
std::this_thread::sleep_until(
|
|
|
|
std::chrono::time_point<std::chrono::system_clock>::max());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|