2016-08-02 05:14:09 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <atomic>
|
2016-08-11 11:47:30 +08:00
|
|
|
#include <cstdio>
|
|
|
|
#include <iomanip>
|
2016-08-02 05:14:09 +08:00
|
|
|
#include <memory>
|
2016-08-11 11:47:30 +08:00
|
|
|
#include <sstream>
|
2016-08-02 05:14:09 +08:00
|
|
|
#include <thread>
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
#include "communication/bolt/v1/bolt.hpp"
|
|
|
|
#include "communication/bolt/v1/session.hpp"
|
|
|
|
#include "io/network/stream_reader.hpp"
|
2016-08-30 22:12:30 +08:00
|
|
|
#include "logging/default.hpp"
|
2016-08-02 05:14:09 +08:00
|
|
|
|
|
|
|
namespace bolt
|
|
|
|
{
|
|
|
|
|
|
|
|
template <class Worker>
|
|
|
|
class Server;
|
|
|
|
|
|
|
|
class Worker : public io::StreamReader<Worker, Session>
|
|
|
|
{
|
|
|
|
friend class bolt::Server<Worker>;
|
|
|
|
|
|
|
|
public:
|
|
|
|
using sptr = std::shared_ptr<Worker>;
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
Worker(Bolt &bolt) : bolt(bolt)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
logger = logging::log->logger("Network");
|
|
|
|
}
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
Session &on_connect(io::Socket &&socket)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
logger.trace("Accepting connection on socket {}", socket.id());
|
|
|
|
|
|
|
|
return *bolt.get().create_session(std::forward<io::Socket>(socket));
|
|
|
|
}
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
void on_error(Session &)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
logger.trace("[on_error] errno = {}", errno);
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
auto err = io::NetworkError("");
|
|
|
|
logger.debug("{}", err.what());
|
|
|
|
#endif
|
|
|
|
|
|
|
|
logger.error("Error occured in this session");
|
|
|
|
}
|
|
|
|
|
|
|
|
void on_wait_timeout() {}
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
Buffer on_alloc(Session &)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
/* logger.trace("[on_alloc] Allocating {}B", sizeof buf); */
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
return Buffer{buf, sizeof buf};
|
2016-08-02 05:14:09 +08:00
|
|
|
}
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
void on_read(Session &session, Buffer &buf)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
logger.trace("[on_read] Received {}B", buf.len);
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
std::stringstream stream;
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
for (size_t i = 0; i < buf.len; ++i)
|
2016-08-02 05:14:09 +08:00
|
|
|
stream << fmt::format("{:02X} ", static_cast<byte>(buf.ptr[i]));
|
|
|
|
|
|
|
|
logger.trace("[on_read] {}", stream.str());
|
|
|
|
#endif
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
try {
|
|
|
|
session.execute(reinterpret_cast<const byte *>(buf.ptr), buf.len);
|
|
|
|
} catch (const std::exception &e) {
|
2016-08-02 05:14:09 +08:00
|
|
|
logger.error("Error occured while executing statement.");
|
|
|
|
logger.error("{}", e.what());
|
2016-08-30 22:12:30 +08:00
|
|
|
// TODO: report to client
|
2016-08-02 05:14:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
void on_close(Session &session)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
logger.trace("[on_close] Client closed the connection");
|
|
|
|
session.close();
|
|
|
|
}
|
|
|
|
|
2016-08-30 22:12:30 +08:00
|
|
|
template <class... Args>
|
|
|
|
void on_exception(Session &session, Args &&... args)
|
|
|
|
{
|
|
|
|
logger.error("Error occured in this session");
|
|
|
|
logger.error(args...);
|
|
|
|
|
|
|
|
// TODO: Do something about it
|
|
|
|
}
|
|
|
|
|
2016-08-02 05:14:09 +08:00
|
|
|
char buf[65536];
|
|
|
|
|
|
|
|
protected:
|
|
|
|
std::reference_wrapper<Bolt> bolt;
|
|
|
|
|
|
|
|
Logger logger;
|
|
|
|
std::thread thread;
|
|
|
|
|
2016-08-11 11:47:30 +08:00
|
|
|
void start(std::atomic<bool> &alive)
|
2016-08-02 05:14:09 +08:00
|
|
|
{
|
|
|
|
thread = std::thread([&, this]() {
|
2016-08-11 11:47:30 +08:00
|
|
|
while (alive)
|
2016-08-02 05:14:09 +08:00
|
|
|
wait_and_process_events();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|