Crow/include/http_server.h

128 lines
4.0 KiB
C
Raw Normal View History

#pragma once
#include <boost/date_time/posix_time/posix_time.hpp>
#include <boost/asio.hpp>
#include <cstdint>
#include <atomic>
2014-08-07 16:14:27 +00:00
#include <future>
2014-10-23 17:45:34 +00:00
#include <vector>
#include <memory>
#include "http_connection.h"
#include "datetime.h"
2014-05-20 16:17:56 +00:00
#include "logging.h"
#include "dumb_timer_queue.h"
2014-04-26 17:19:59 +00:00
namespace crow
{
using namespace boost;
using tcp = asio::ip::tcp;
2014-09-06 19:30:53 +00:00
template <typename Handler, typename ... Middlewares>
class Server
{
public:
2014-04-17 06:50:28 +00:00
Server(Handler* handler, uint16_t port, uint16_t concurrency = 1)
: acceptor_(io_service_, tcp::endpoint(asio::ip::address(), port)),
signals_(io_service_, SIGINT, SIGTERM),
handler_(handler),
2014-05-20 16:17:56 +00:00
concurrency_(concurrency),
port_(port)
{
}
void run()
{
if (concurrency_ < 0)
concurrency_ = 1;
for(int i = 0; i < concurrency_; i++)
io_service_pool_.emplace_back(new boost::asio::io_service());
2014-04-17 06:50:28 +00:00
std::vector<std::future<void>> v;
for(uint16_t i = 0; i < concurrency_; i ++)
v.push_back(
std::async(std::launch::async, [this, i]{
2014-09-06 19:30:53 +00:00
// initializing timer queue
auto& timer_queue = detail::dumb_timer_queue::get_current_dumb_timer_queue();
2014-09-06 19:30:53 +00:00
timer_queue.set_io_service(*io_service_pool_[i]);
boost::asio::deadline_timer timer(*io_service_pool_[i]);
timer.expires_from_now(boost::posix_time::seconds(1));
2014-09-06 19:30:53 +00:00
std::function<void(const boost::system::error_code& ec)> handler;
handler = [&](const boost::system::error_code& ec){
if (ec)
return;
timer_queue.process();
timer.expires_from_now(boost::posix_time::seconds(1));
timer.async_wait(handler);
};
timer.async_wait(handler);
2014-09-06 19:30:53 +00:00
io_service_pool_[i]->run();
}));
2014-05-20 22:30:51 +00:00
CROW_LOG_INFO << server_name_ << " server is running, local port " << port_;
2014-05-20 16:17:56 +00:00
signals_.async_wait(
[&](const boost::system::error_code& error, int signal_number){
stop();
});
do_accept();
v.push_back(std::async(std::launch::async, [this]{
io_service_.run();
CROW_LOG_INFO << "Exiting.";
}));
}
2014-04-15 13:08:23 +00:00
void stop()
{
io_service_.stop();
for(auto& io_service:io_service_pool_)
io_service->stop();
2014-04-15 13:08:23 +00:00
}
private:
asio::io_service& pick_io_service()
{
// TODO load balancing
roundrobin_index_++;
if (roundrobin_index_ >= io_service_pool_.size())
roundrobin_index_ = 0;
return *io_service_pool_[roundrobin_index_];
}
void do_accept()
{
2014-09-06 19:30:53 +00:00
auto p = new Connection<Handler, Middlewares...>(pick_io_service(), handler_, server_name_, middlewares_);
acceptor_.async_accept(p->socket(),
[this, p](boost::system::error_code ec)
{
if (!ec)
{
p->start();
}
do_accept();
});
}
private:
asio::io_service io_service_;
std::vector<std::unique_ptr<asio::io_service>> io_service_pool_;
tcp::acceptor acceptor_;
boost::asio::signal_set signals_;
Handler* handler_;
2014-05-02 09:22:02 +00:00
uint16_t concurrency_{1};
std::string server_name_ = "Crow/0.1";
2014-05-20 16:17:56 +00:00
uint16_t port_;
unsigned int roundrobin_index_{};
2014-09-06 19:30:53 +00:00
std::tuple<Middlewares...> middlewares_;
};
}