3 changed files with 997 additions and 3 deletions
@ -0,0 +1,220 @@ |
|||||||
|
#ifndef CRYPTO_HPP |
||||||
|
#define CRYPTO_HPP |
||||||
|
|
||||||
|
#include <string> |
||||||
|
#include <cmath> |
||||||
|
#include <sstream> |
||||||
|
#include <iomanip> |
||||||
|
#include <istream> |
||||||
|
#include <vector> |
||||||
|
|
||||||
|
//Moving these to a seperate namespace for minimal global namespace cluttering does not work with clang++
|
||||||
|
#include <openssl/evp.h> |
||||||
|
#include <openssl/buffer.h> |
||||||
|
#include <openssl/sha.h> |
||||||
|
#include <openssl/md5.h> |
||||||
|
|
||||||
|
namespace SimpleWeb { |
||||||
|
//TODO 2017: remove workaround for MSVS 2012
|
||||||
|
#if _MSC_VER == 1700 //MSVS 2012 has no definition for round()
|
||||||
|
inline double round(double x) { //custom definition of round() for positive numbers
|
||||||
|
return floor(x + 0.5); |
||||||
|
} |
||||||
|
#endif |
||||||
|
|
||||||
|
class Crypto { |
||||||
|
const static size_t buffer_size=131072; |
||||||
|
public: |
||||||
|
class Base64 { |
||||||
|
public: |
||||||
|
static std::string encode(const std::string &ascii) { |
||||||
|
std::string base64; |
||||||
|
|
||||||
|
BIO *bio, *b64; |
||||||
|
BUF_MEM *bptr=BUF_MEM_new(); |
||||||
|
|
||||||
|
b64 = BIO_new(BIO_f_base64()); |
||||||
|
BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); |
||||||
|
bio = BIO_new(BIO_s_mem()); |
||||||
|
BIO_push(b64, bio); |
||||||
|
BIO_set_mem_buf(b64, bptr, BIO_CLOSE); |
||||||
|
|
||||||
|
//Write directly to base64-buffer to avoid copy
|
||||||
|
int base64_length=static_cast<int>(round(4*ceil((double)ascii.size()/3.0))); |
||||||
|
base64.resize(base64_length); |
||||||
|
bptr->length=0; |
||||||
|
bptr->max=base64_length+1; |
||||||
|
bptr->data=(char*)&base64[0]; |
||||||
|
|
||||||
|
BIO_write(b64, &ascii[0], static_cast<int>(ascii.size())); |
||||||
|
BIO_flush(b64); |
||||||
|
|
||||||
|
//To keep &base64[0] through BIO_free_all(b64)
|
||||||
|
bptr->length=0; |
||||||
|
bptr->max=0; |
||||||
|
bptr->data=nullptr; |
||||||
|
|
||||||
|
BIO_free_all(b64); |
||||||
|
|
||||||
|
return base64; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string decode(const std::string &base64) { |
||||||
|
std::string ascii; |
||||||
|
|
||||||
|
//Resize ascii, however, the size is a up to two bytes too large.
|
||||||
|
ascii.resize((6*base64.size())/8); |
||||||
|
BIO *b64, *bio; |
||||||
|
|
||||||
|
b64 = BIO_new(BIO_f_base64()); |
||||||
|
BIO_set_flags(b64, BIO_FLAGS_BASE64_NO_NL); |
||||||
|
bio = BIO_new_mem_buf((char*)&base64[0], static_cast<int>(base64.size())); |
||||||
|
bio = BIO_push(b64, bio); |
||||||
|
|
||||||
|
int decoded_length = BIO_read(bio, &ascii[0], static_cast<int>(ascii.size())); |
||||||
|
ascii.resize(decoded_length); |
||||||
|
|
||||||
|
BIO_free_all(b64); |
||||||
|
|
||||||
|
return ascii; |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
/// Return hex string from bytes in input string.
|
||||||
|
static std::string to_hex_string(const std::string &input) { |
||||||
|
std::stringstream hex_stream; |
||||||
|
hex_stream << std::hex << std::internal << std::setfill('0'); |
||||||
|
for (auto &byte : input) |
||||||
|
hex_stream << std::setw(2) << static_cast<int>(static_cast<unsigned char>(byte)); |
||||||
|
return hex_stream.str(); |
||||||
|
} |
||||||
|
|
||||||
|
static std::string md5(const std::string &input, size_t iterations=1) { |
||||||
|
std::string hash; |
||||||
|
|
||||||
|
hash.resize(128 / 8); |
||||||
|
MD5(reinterpret_cast<const unsigned char*>(&input[0]), input.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
MD5(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string md5(std::istream &stream, size_t iterations=1) { |
||||||
|
MD5_CTX context; |
||||||
|
MD5_Init(&context); |
||||||
|
std::streamsize read_length; |
||||||
|
std::vector<char> buffer(buffer_size); |
||||||
|
while((read_length=stream.read(&buffer[0], buffer_size).gcount())>0) |
||||||
|
MD5_Update(&context, buffer.data(), read_length); |
||||||
|
std::string hash; |
||||||
|
hash.resize(128 / 8); |
||||||
|
MD5_Final(reinterpret_cast<unsigned char*>(&hash[0]), &context); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
MD5(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha1(const std::string &input, size_t iterations=1) { |
||||||
|
std::string hash; |
||||||
|
|
||||||
|
hash.resize(160 / 8); |
||||||
|
SHA1(reinterpret_cast<const unsigned char*>(&input[0]), input.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA1(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha1(std::istream &stream, size_t iterations=1) { |
||||||
|
SHA_CTX context; |
||||||
|
SHA1_Init(&context); |
||||||
|
std::streamsize read_length; |
||||||
|
std::vector<char> buffer(buffer_size); |
||||||
|
while((read_length=stream.read(&buffer[0], buffer_size).gcount())>0) |
||||||
|
SHA1_Update(&context, buffer.data(), read_length); |
||||||
|
std::string hash; |
||||||
|
hash.resize(160 / 8); |
||||||
|
SHA1_Final(reinterpret_cast<unsigned char*>(&hash[0]), &context); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA1(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha256(const std::string &input, size_t iterations=1) { |
||||||
|
std::string hash; |
||||||
|
|
||||||
|
hash.resize(256 / 8); |
||||||
|
SHA256(reinterpret_cast<const unsigned char*>(&input[0]), input.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA256(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha256(std::istream &stream, size_t iterations=1) { |
||||||
|
SHA256_CTX context; |
||||||
|
SHA256_Init(&context); |
||||||
|
std::streamsize read_length; |
||||||
|
std::vector<char> buffer(buffer_size); |
||||||
|
while((read_length=stream.read(&buffer[0], buffer_size).gcount())>0) |
||||||
|
SHA256_Update(&context, buffer.data(), read_length); |
||||||
|
std::string hash; |
||||||
|
hash.resize(256 / 8); |
||||||
|
SHA256_Final(reinterpret_cast<unsigned char*>(&hash[0]), &context); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA256(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha512(const std::string &input, size_t iterations=1) { |
||||||
|
std::string hash; |
||||||
|
|
||||||
|
hash.resize(512 / 8); |
||||||
|
SHA512(reinterpret_cast<const unsigned char*>(&input[0]), input.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA512(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
static std::string sha512(std::istream &stream, size_t iterations=1) { |
||||||
|
SHA512_CTX context; |
||||||
|
SHA512_Init(&context); |
||||||
|
std::streamsize read_length; |
||||||
|
std::vector<char> buffer(buffer_size); |
||||||
|
while((read_length=stream.read(&buffer[0], buffer_size).gcount())>0) |
||||||
|
SHA512_Update(&context, buffer.data(), read_length); |
||||||
|
std::string hash; |
||||||
|
hash.resize(512 / 8); |
||||||
|
SHA512_Final(reinterpret_cast<unsigned char*>(&hash[0]), &context); |
||||||
|
|
||||||
|
for (size_t c = 1; c < iterations; ++c) |
||||||
|
SHA512(reinterpret_cast<const unsigned char*>(&hash[0]), hash.size(), reinterpret_cast<unsigned char*>(&hash[0])); |
||||||
|
|
||||||
|
return hash; |
||||||
|
} |
||||||
|
|
||||||
|
/// key_size is number of bytes of the returned key.
|
||||||
|
static std::string pbkdf2(const std::string &password, const std::string &salt, int iterations, int key_size) { |
||||||
|
std::string key; |
||||||
|
key.resize(key_size); |
||||||
|
PKCS5_PBKDF2_HMAC_SHA1(password.c_str(), password.size(), |
||||||
|
reinterpret_cast<const unsigned char*>(salt.c_str()), salt.size(), iterations, |
||||||
|
key_size, reinterpret_cast<unsigned char*>(&key[0])); |
||||||
|
return key; |
||||||
|
} |
||||||
|
}; |
||||||
|
} |
||||||
|
#endif /* CRYPTO_HPP */ |
||||||
|
|
||||||
@ -0,0 +1,774 @@ |
|||||||
|
#ifndef SERVER_WS_HPP |
||||||
|
#define SERVER_WS_HPP |
||||||
|
|
||||||
|
#include "crypto.hpp" |
||||||
|
|
||||||
|
#include <boost/asio.hpp> |
||||||
|
#include <boost/asio/spawn.hpp> |
||||||
|
#include <boost/algorithm/string/predicate.hpp> |
||||||
|
#include <boost/functional/hash.hpp> |
||||||
|
|
||||||
|
#include <unordered_map> |
||||||
|
#include <thread> |
||||||
|
#include <mutex> |
||||||
|
#include <unordered_set> |
||||||
|
#include <list> |
||||||
|
#include <memory> |
||||||
|
#include <atomic> |
||||||
|
#include <iostream> |
||||||
|
|
||||||
|
#ifndef CASE_INSENSITIVE_EQUALS_AND_HASH |
||||||
|
#define CASE_INSENSITIVE_EQUALS_AND_HASH |
||||||
|
//Based on http://www.boost.org/doc/libs/1_60_0/doc/html/unordered/hash_equality.html
|
||||||
|
class case_insensitive_equals { |
||||||
|
public: |
||||||
|
bool operator()(const std::string &key1, const std::string &key2) const { |
||||||
|
return boost::algorithm::iequals(key1, key2); |
||||||
|
} |
||||||
|
}; |
||||||
|
class case_insensitive_hash { |
||||||
|
public: |
||||||
|
size_t operator()(const std::string &key) const { |
||||||
|
std::size_t seed=0; |
||||||
|
for(auto &c: key) |
||||||
|
boost::hash_combine(seed, std::tolower(c)); |
||||||
|
return seed; |
||||||
|
} |
||||||
|
}; |
||||||
|
#endif |
||||||
|
|
||||||
|
// Late 2017 TODO: remove the following checks and always use std::regex
|
||||||
|
#ifdef USE_BOOST_REGEX |
||||||
|
#include <boost/regex.hpp> |
||||||
|
#define REGEX_NS boost |
||||||
|
#else |
||||||
|
#include <regex> |
||||||
|
#define REGEX_NS std |
||||||
|
#endif |
||||||
|
|
||||||
|
// TODO when switching to c++14, use [[deprecated]] instead
|
||||||
|
#ifndef DEPRECATED |
||||||
|
#ifdef __GNUC__ |
||||||
|
#define DEPRECATED __attribute__((deprecated)) |
||||||
|
#elif defined(_MSC_VER) |
||||||
|
#define DEPRECATED __declspec(deprecated) |
||||||
|
#else |
||||||
|
#define DEPRECATED |
||||||
|
#endif |
||||||
|
#endif |
||||||
|
|
||||||
|
namespace SimpleWeb { |
||||||
|
// TODO: remove when onopen, onmessage, etc is removed:
|
||||||
|
#ifdef __GNUC__ |
||||||
|
#pragma GCC diagnostic push |
||||||
|
#pragma GCC diagnostic ignored "-Wdeprecated-declarations" |
||||||
|
#elif defined(_MSC_VER) |
||||||
|
#pragma warning(push) |
||||||
|
#pragma warning(disable: 4996) |
||||||
|
#endif |
||||||
|
|
||||||
|
template <class socket_type> |
||||||
|
class SocketServer; |
||||||
|
|
||||||
|
template <class socket_type> |
||||||
|
class SocketServerBase { |
||||||
|
public: |
||||||
|
virtual ~SocketServerBase() {} |
||||||
|
|
||||||
|
class SendStream : public std::ostream { |
||||||
|
friend class SocketServerBase<socket_type>; |
||||||
|
private: |
||||||
|
boost::asio::streambuf streambuf; |
||||||
|
public: |
||||||
|
SendStream(): std::ostream(&streambuf) {} |
||||||
|
size_t size() { |
||||||
|
return streambuf.size(); |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
class Connection { |
||||||
|
friend class SocketServerBase<socket_type>; |
||||||
|
friend class SocketServer<socket_type>; |
||||||
|
|
||||||
|
public: |
||||||
|
Connection(const std::shared_ptr<socket_type> &socket): socket(socket), strand(socket->get_io_service()), closed(false) {} |
||||||
|
|
||||||
|
std::string method, path, http_version; |
||||||
|
|
||||||
|
std::unordered_multimap<std::string, std::string, case_insensitive_hash, case_insensitive_equals> header; |
||||||
|
|
||||||
|
REGEX_NS::smatch path_match; |
||||||
|
|
||||||
|
std::string remote_endpoint_address; |
||||||
|
unsigned short remote_endpoint_port; |
||||||
|
|
||||||
|
private: |
||||||
|
Connection(socket_type *socket): socket(socket), strand(socket->get_io_service()), closed(false) {} |
||||||
|
|
||||||
|
class SendData { |
||||||
|
public: |
||||||
|
SendData(const std::shared_ptr<SendStream> &header_stream, const std::shared_ptr<SendStream> &message_stream, |
||||||
|
const std::function<void(const boost::system::error_code)> &callback) : |
||||||
|
header_stream(header_stream), message_stream(message_stream), callback(callback) {} |
||||||
|
std::shared_ptr<SendStream> header_stream; |
||||||
|
std::shared_ptr<SendStream> message_stream; |
||||||
|
std::function<void(const boost::system::error_code)> callback; |
||||||
|
}; |
||||||
|
|
||||||
|
std::shared_ptr<socket_type> socket; |
||||||
|
|
||||||
|
boost::asio::strand strand; |
||||||
|
|
||||||
|
std::list<SendData> send_queue; |
||||||
|
|
||||||
|
void send_from_queue(const std::shared_ptr<Connection> &connection) { |
||||||
|
strand.post([this, connection]() { |
||||||
|
boost::asio::async_write(*socket, send_queue.begin()->header_stream->streambuf, |
||||||
|
strand.wrap([this, connection](const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(!ec) { |
||||||
|
boost::asio::async_write(*socket, send_queue.begin()->message_stream->streambuf, |
||||||
|
strand.wrap([this, connection] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
auto send_queued=send_queue.begin(); |
||||||
|
if(send_queued->callback) |
||||||
|
send_queued->callback(ec); |
||||||
|
if(!ec) { |
||||||
|
send_queue.erase(send_queued); |
||||||
|
if(send_queue.size()>0) |
||||||
|
send_from_queue(connection); |
||||||
|
} |
||||||
|
else |
||||||
|
send_queue.clear(); |
||||||
|
})); |
||||||
|
} |
||||||
|
else { |
||||||
|
auto send_queued=send_queue.begin(); |
||||||
|
if(send_queued->callback) |
||||||
|
send_queued->callback(ec); |
||||||
|
send_queue.clear(); |
||||||
|
} |
||||||
|
})); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
std::atomic<bool> closed; |
||||||
|
|
||||||
|
std::unique_ptr<boost::asio::deadline_timer> timer_idle; |
||||||
|
|
||||||
|
void read_remote_endpoint_data() { |
||||||
|
try { |
||||||
|
remote_endpoint_address=socket->lowest_layer().remote_endpoint().address().to_string(); |
||||||
|
remote_endpoint_port=socket->lowest_layer().remote_endpoint().port(); |
||||||
|
} |
||||||
|
catch(...) {} |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
class Message : public std::istream { |
||||||
|
friend class SocketServerBase<socket_type>; |
||||||
|
|
||||||
|
public: |
||||||
|
unsigned char fin_rsv_opcode; |
||||||
|
size_t size() { |
||||||
|
return length; |
||||||
|
} |
||||||
|
std::string string() { |
||||||
|
std::stringstream ss; |
||||||
|
ss << rdbuf(); |
||||||
|
return ss.str(); |
||||||
|
} |
||||||
|
private: |
||||||
|
Message(): std::istream(&streambuf) {} |
||||||
|
size_t length; |
||||||
|
boost::asio::streambuf streambuf; |
||||||
|
}; |
||||||
|
|
||||||
|
class Endpoint { |
||||||
|
friend class SocketServerBase<socket_type>; |
||||||
|
private: |
||||||
|
std::unordered_set<std::shared_ptr<Connection> > connections; |
||||||
|
std::mutex connections_mutex; |
||||||
|
|
||||||
|
public: |
||||||
|
DEPRECATED std::function<void(std::shared_ptr<Connection>)> onopen; |
||||||
|
std::function<void(std::shared_ptr<Connection>)> on_open; |
||||||
|
DEPRECATED std::function<void(std::shared_ptr<Connection>, std::shared_ptr<Message>)> onmessage; |
||||||
|
std::function<void(std::shared_ptr<Connection>, std::shared_ptr<Message>)> on_message; |
||||||
|
DEPRECATED std::function<void(std::shared_ptr<Connection>, int, const std::string&)> onclose; |
||||||
|
std::function<void(std::shared_ptr<Connection>, int, const std::string&)> on_close; |
||||||
|
DEPRECATED std::function<void(std::shared_ptr<Connection>, const boost::system::error_code&)> onerror; |
||||||
|
std::function<void(std::shared_ptr<Connection>, const boost::system::error_code&)> on_error; |
||||||
|
|
||||||
|
std::unordered_set<std::shared_ptr<Connection> > get_connections() { |
||||||
|
std::lock_guard<std::mutex> lock(connections_mutex); |
||||||
|
auto copy=connections; |
||||||
|
return copy; |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
class Config { |
||||||
|
friend class SocketServerBase<socket_type>; |
||||||
|
private: |
||||||
|
Config(unsigned short port): port(port) {} |
||||||
|
public: |
||||||
|
/// Port number to use. Defaults to 80 for HTTP and 443 for HTTPS.
|
||||||
|
unsigned short port; |
||||||
|
/// Number of threads that the server will use when start() is called. Defaults to 1 thread.
|
||||||
|
size_t thread_pool_size=1; |
||||||
|
/// Timeout on request handling. Defaults to 5 seconds.
|
||||||
|
size_t timeout_request=5; |
||||||
|
/// Idle timeout. Defaults to no timeout.
|
||||||
|
size_t timeout_idle=0; |
||||||
|
/// IPv4 address in dotted decimal form or IPv6 address in hexadecimal notation.
|
||||||
|
/// If empty, the address will be any address.
|
||||||
|
std::string address; |
||||||
|
/// Set to false to avoid binding the socket to an address that is already in use. Defaults to true.
|
||||||
|
bool reuse_address=true; |
||||||
|
}; |
||||||
|
///Set before calling start().
|
||||||
|
Config config; |
||||||
|
|
||||||
|
private: |
||||||
|
class regex_orderable : public REGEX_NS::regex { |
||||||
|
std::string str; |
||||||
|
public: |
||||||
|
regex_orderable(const char *regex_cstr) : REGEX_NS::regex(regex_cstr), str(regex_cstr) {} |
||||||
|
regex_orderable(const std::string ®ex_str) : REGEX_NS::regex(regex_str), str(regex_str) {} |
||||||
|
bool operator<(const regex_orderable &rhs) const { |
||||||
|
return str<rhs.str; |
||||||
|
} |
||||||
|
}; |
||||||
|
public: |
||||||
|
/// Warning: do not add or remove endpoints after start() is called
|
||||||
|
std::map<regex_orderable, Endpoint> endpoint; |
||||||
|
|
||||||
|
virtual void start() { |
||||||
|
for(auto &endp: endpoint) { |
||||||
|
// TODO: remove when onopen, onmessage, etc is removed:
|
||||||
|
if(endp.second.onopen) |
||||||
|
endp.second.on_open=endp.second.onopen; |
||||||
|
if(endp.second.onmessage) |
||||||
|
endp.second.on_message=endp.second.onmessage; |
||||||
|
if(endp.second.onclose) |
||||||
|
endp.second.on_close=endp.second.onclose; |
||||||
|
if(endp.second.onerror) |
||||||
|
endp.second.on_error=endp.second.onerror; |
||||||
|
} |
||||||
|
|
||||||
|
if(!io_service) |
||||||
|
io_service=std::make_shared<boost::asio::io_service>(); |
||||||
|
|
||||||
|
if(io_service->stopped()) |
||||||
|
io_service->reset(); |
||||||
|
|
||||||
|
boost::asio::ip::tcp::endpoint endpoint; |
||||||
|
if(config.address.size()>0) |
||||||
|
endpoint=boost::asio::ip::tcp::endpoint(boost::asio::ip::address::from_string(config.address), config.port); |
||||||
|
else |
||||||
|
endpoint=boost::asio::ip::tcp::endpoint(boost::asio::ip::tcp::v4(), config.port); |
||||||
|
|
||||||
|
if(!acceptor) |
||||||
|
acceptor=std::unique_ptr<boost::asio::ip::tcp::acceptor>(new boost::asio::ip::tcp::acceptor(*io_service)); |
||||||
|
acceptor->open(endpoint.protocol()); |
||||||
|
acceptor->set_option(boost::asio::socket_base::reuse_address(config.reuse_address)); |
||||||
|
acceptor->bind(endpoint); |
||||||
|
acceptor->listen(); |
||||||
|
|
||||||
|
accept(); |
||||||
|
|
||||||
|
//If thread_pool_size>1, start m_io_service.run() in (thread_pool_size-1) threads for thread-pooling
|
||||||
|
threads.clear(); |
||||||
|
for(size_t c=1;c<config.thread_pool_size;c++) { |
||||||
|
threads.emplace_back([this](){ |
||||||
|
io_service->run(); |
||||||
|
}); |
||||||
|
} |
||||||
|
//Main thread
|
||||||
|
if(config.thread_pool_size>0) |
||||||
|
io_service->run(); |
||||||
|
|
||||||
|
//Wait for the rest of the threads, if any, to finish as well
|
||||||
|
for(auto& t: threads) { |
||||||
|
t.join(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void stop() { |
||||||
|
acceptor->close(); |
||||||
|
if(config.thread_pool_size>0) |
||||||
|
io_service->stop(); |
||||||
|
|
||||||
|
for(auto &pair: endpoint) { |
||||||
|
std::lock_guard<std::mutex> lock(pair.second.connections_mutex); |
||||||
|
for(auto &connection: pair.second.connections) { |
||||||
|
connection->socket->lowest_layer().shutdown(boost::asio::ip::tcp::socket::shutdown_both); |
||||||
|
connection->socket->lowest_layer().close(); |
||||||
|
} |
||||||
|
pair.second.connections.clear(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
///fin_rsv_opcode: 129=one fragment, text, 130=one fragment, binary, 136=close connection.
|
||||||
|
///See http://tools.ietf.org/html/rfc6455#section-5.2 for more information
|
||||||
|
void send(const std::shared_ptr<Connection> &connection, const std::shared_ptr<SendStream> &message_stream,
|
||||||
|
const std::function<void(const boost::system::error_code&)>& callback=nullptr,
|
||||||
|
unsigned char fin_rsv_opcode=129) const { |
||||||
|
if(fin_rsv_opcode!=136) |
||||||
|
timer_idle_reset(connection); |
||||||
|
|
||||||
|
auto header_stream=std::make_shared<SendStream>(); |
||||||
|
|
||||||
|
size_t length=message_stream->size(); |
||||||
|
|
||||||
|
header_stream->put(fin_rsv_opcode); |
||||||
|
//unmasked (first length byte<128)
|
||||||
|
if(length>=126) { |
||||||
|
int num_bytes; |
||||||
|
if(length>0xffff) { |
||||||
|
num_bytes=8; |
||||||
|
header_stream->put(127); |
||||||
|
} |
||||||
|
else { |
||||||
|
num_bytes=2; |
||||||
|
header_stream->put(126); |
||||||
|
} |
||||||
|
|
||||||
|
for(int c=num_bytes-1;c>=0;c--) { |
||||||
|
header_stream->put((static_cast<unsigned long long>(length) >> (8 * c)) % 256); |
||||||
|
} |
||||||
|
} |
||||||
|
else |
||||||
|
header_stream->put(static_cast<unsigned char>(length)); |
||||||
|
|
||||||
|
connection->strand.post([this, connection, header_stream, message_stream, callback]() { |
||||||
|
connection->send_queue.emplace_back(header_stream, message_stream, callback); |
||||||
|
if(connection->send_queue.size()==1) |
||||||
|
connection->send_from_queue(connection); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
void send_close(const std::shared_ptr<Connection> &connection, int status, const std::string& reason="", |
||||||
|
const std::function<void(const boost::system::error_code&)>& callback=nullptr) const { |
||||||
|
//Send close only once (in case close is initiated by server)
|
||||||
|
if(connection->closed) |
||||||
|
return; |
||||||
|
connection->closed=true; |
||||||
|
|
||||||
|
auto send_stream=std::make_shared<SendStream>(); |
||||||
|
|
||||||
|
send_stream->put(status>>8); |
||||||
|
send_stream->put(status%256); |
||||||
|
|
||||||
|
*send_stream << reason; |
||||||
|
|
||||||
|
//fin_rsv_opcode=136: message close
|
||||||
|
send(connection, send_stream, callback, 136); |
||||||
|
} |
||||||
|
|
||||||
|
std::unordered_set<std::shared_ptr<Connection> > get_connections() { |
||||||
|
std::unordered_set<std::shared_ptr<Connection> > all_connections; |
||||||
|
for(auto& e: endpoint) { |
||||||
|
std::lock_guard<std::mutex> lock(e.second.connections_mutex); |
||||||
|
all_connections.insert(e.second.connections.begin(), e.second.connections.end()); |
||||||
|
} |
||||||
|
return all_connections; |
||||||
|
} |
||||||
|
|
||||||
|
/**
|
||||||
|
* Upgrades a request, from for instance Simple-Web-Server, to a WebSocket connection. |
||||||
|
* The parameters are moved to the Connection object. |
||||||
|
* See also Server::on_upgrade in the Simple-Web-Server project. |
||||||
|
* The socket's io_service is used, thus running start() is not needed. |
||||||
|
* |
||||||
|
* Example use: |
||||||
|
* server.on_upgrade=[&socket_server] (auto socket, auto request) { |
||||||
|
* auto connection=std::make_shared<SimpleWeb::SocketServer<SimpleWeb::WS>::Connection>(socket); |
||||||
|
* connection->method=std::move(request->method); |
||||||
|
* connection->path=std::move(request->path); |
||||||
|
* connection->http_version=std::move(request->http_version); |
||||||
|
* connection->header=std::move(request->header); |
||||||
|
* connection->remote_endpoint_address=std::move(request->remote_endpoint_address); |
||||||
|
* connection->remote_endpoint_port=request->remote_endpoint_port; |
||||||
|
* socket_server.upgrade(connection); |
||||||
|
* } |
||||||
|
*/ |
||||||
|
void upgrade(const std::shared_ptr<Connection> &connection) { |
||||||
|
auto read_buffer=std::make_shared<boost::asio::streambuf>(); |
||||||
|
write_handshake(connection, read_buffer); |
||||||
|
} |
||||||
|
|
||||||
|
/// If you have your own boost::asio::io_service, store its pointer here before running start().
|
||||||
|
/// You might also want to set config.thread_pool_size to 0.
|
||||||
|
std::shared_ptr<boost::asio::io_service> io_service; |
||||||
|
protected: |
||||||
|
const std::string ws_magic_string="258EAFA5-E914-47DA-95CA-C5AB0DC85B11"; |
||||||
|
|
||||||
|
std::unique_ptr<boost::asio::ip::tcp::acceptor> acceptor; |
||||||
|
|
||||||
|
std::vector<std::thread> threads; |
||||||
|
|
||||||
|
SocketServerBase(unsigned short port) : config(port) {} |
||||||
|
|
||||||
|
virtual void accept()=0; |
||||||
|
|
||||||
|
std::shared_ptr<boost::asio::deadline_timer> get_timeout_timer(const std::shared_ptr<Connection> &connection, size_t seconds) { |
||||||
|
if(seconds==0) |
||||||
|
return nullptr; |
||||||
|
|
||||||
|
auto timer=std::make_shared<boost::asio::deadline_timer>(connection->socket->get_io_service()); |
||||||
|
timer->expires_from_now(boost::posix_time::seconds(static_cast<long>(seconds))); |
||||||
|
timer->async_wait([connection](const boost::system::error_code& ec){ |
||||||
|
if(!ec) { |
||||||
|
connection->socket->lowest_layer().shutdown(boost::asio::ip::tcp::socket::shutdown_both); |
||||||
|
connection->socket->lowest_layer().close(); |
||||||
|
} |
||||||
|
}); |
||||||
|
return timer; |
||||||
|
} |
||||||
|
|
||||||
|
void read_handshake(const std::shared_ptr<Connection> &connection) { |
||||||
|
connection->read_remote_endpoint_data(); |
||||||
|
|
||||||
|
//Create new read_buffer for async_read_until()
|
||||||
|
//Shared_ptr is used to pass temporary objects to the asynchronous functions
|
||||||
|
auto read_buffer=std::make_shared<boost::asio::streambuf>(); |
||||||
|
|
||||||
|
//Set timeout on the following boost::asio::async-read or write function
|
||||||
|
auto timer=get_timeout_timer(connection, config.timeout_request); |
||||||
|
|
||||||
|
boost::asio::async_read_until(*connection->socket, *read_buffer, "\r\n\r\n", |
||||||
|
[this, connection, read_buffer, timer] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(timer) |
||||||
|
timer->cancel(); |
||||||
|
if(!ec) { |
||||||
|
//Convert to istream to extract string-lines
|
||||||
|
std::istream stream(read_buffer.get()); |
||||||
|
|
||||||
|
parse_handshake(connection, stream); |
||||||
|
|
||||||
|
write_handshake(connection, read_buffer); |
||||||
|
} |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
void parse_handshake(const std::shared_ptr<Connection> &connection, std::istream& stream) const { |
||||||
|
std::string line; |
||||||
|
getline(stream, line); |
||||||
|
size_t method_end; |
||||||
|
if((method_end=line.find(' '))!=std::string::npos) { |
||||||
|
size_t path_end; |
||||||
|
if((path_end=line.find(' ', method_end+1))!=std::string::npos) { |
||||||
|
connection->method=line.substr(0, method_end); |
||||||
|
connection->path=line.substr(method_end+1, path_end-method_end-1); |
||||||
|
if((path_end+6)<line.size()) |
||||||
|
connection->http_version=line.substr(path_end+6, line.size()-(path_end+6)-1); |
||||||
|
else |
||||||
|
connection->http_version="1.1"; |
||||||
|
|
||||||
|
getline(stream, line); |
||||||
|
size_t param_end; |
||||||
|
while((param_end=line.find(':'))!=std::string::npos) { |
||||||
|
size_t value_start=param_end+1; |
||||||
|
if((value_start)<line.size()) { |
||||||
|
if(line[value_start]==' ') |
||||||
|
value_start++; |
||||||
|
if(value_start<line.size()) |
||||||
|
connection->header.emplace(line.substr(0, param_end), line.substr(value_start, line.size()-value_start-1)); |
||||||
|
} |
||||||
|
|
||||||
|
getline(stream, line); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void write_handshake(const std::shared_ptr<Connection> &connection, const std::shared_ptr<boost::asio::streambuf> &read_buffer) { |
||||||
|
//Find path- and method-match, and generate response
|
||||||
|
for(auto ®ex_endpoint: endpoint) { |
||||||
|
REGEX_NS::smatch path_match; |
||||||
|
if(REGEX_NS::regex_match(connection->path, path_match, regex_endpoint.first)) { |
||||||
|
auto write_buffer=std::make_shared<boost::asio::streambuf>(); |
||||||
|
std::ostream handshake(write_buffer.get()); |
||||||
|
|
||||||
|
if(generate_handshake(connection, handshake)) { |
||||||
|
connection->path_match=std::move(path_match); |
||||||
|
//Capture write_buffer in lambda so it is not destroyed before async_write is finished
|
||||||
|
boost::asio::async_write(*connection->socket, *write_buffer,
|
||||||
|
[this, connection, write_buffer, read_buffer, ®ex_endpoint] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(!ec) { |
||||||
|
connection_open(connection, regex_endpoint.second); |
||||||
|
read_message(connection, read_buffer, regex_endpoint.second); |
||||||
|
} |
||||||
|
else |
||||||
|
connection_error(connection, regex_endpoint.second, ec); |
||||||
|
}); |
||||||
|
} |
||||||
|
return; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
bool generate_handshake(const std::shared_ptr<Connection> &connection, std::ostream& handshake) const { |
||||||
|
auto header_it=connection->header.find("Sec-WebSocket-Key"); |
||||||
|
if(header_it==connection->header.end()) |
||||||
|
return false; |
||||||
|
|
||||||
|
auto sha1=Crypto::sha1(header_it->second+ws_magic_string); |
||||||
|
|
||||||
|
handshake << "HTTP/1.1 101 Web Socket Protocol Handshake\r\n"; |
||||||
|
handshake << "Upgrade: websocket\r\n"; |
||||||
|
handshake << "Connection: Upgrade\r\n"; |
||||||
|
handshake << "Sec-WebSocket-Accept: " << Crypto::Base64::encode(sha1) << "\r\n"; |
||||||
|
handshake << "\r\n"; |
||||||
|
|
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
void read_message(const std::shared_ptr<Connection> &connection, |
||||||
|
const std::shared_ptr<boost::asio::streambuf> &read_buffer, Endpoint& endpoint) const { |
||||||
|
boost::asio::async_read(*connection->socket, *read_buffer, boost::asio::transfer_exactly(2), |
||||||
|
[this, connection, read_buffer, &endpoint] |
||||||
|
(const boost::system::error_code& ec, size_t bytes_transferred) { |
||||||
|
if(!ec) { |
||||||
|
if(bytes_transferred==0) { //TODO: why does this happen sometimes?
|
||||||
|
read_message(connection, read_buffer, endpoint); |
||||||
|
return; |
||||||
|
} |
||||||
|
std::istream stream(read_buffer.get()); |
||||||
|
|
||||||
|
std::vector<unsigned char> first_bytes; |
||||||
|
first_bytes.resize(2); |
||||||
|
stream.read((char*)&first_bytes[0], 2); |
||||||
|
|
||||||
|
unsigned char fin_rsv_opcode=first_bytes[0]; |
||||||
|
|
||||||
|
//Close connection if unmasked message from client (protocol error)
|
||||||
|
if(first_bytes[1]<128) { |
||||||
|
const std::string reason("message from client not masked"); |
||||||
|
send_close(connection, 1002, reason, [this, connection](const boost::system::error_code& /*ec*/) {}); |
||||||
|
connection_close(connection, endpoint, 1002, reason); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
size_t length=(first_bytes[1]&127); |
||||||
|
|
||||||
|
if(length==126) { |
||||||
|
//2 next bytes is the size of content
|
||||||
|
boost::asio::async_read(*connection->socket, *read_buffer, boost::asio::transfer_exactly(2), |
||||||
|
[this, connection, read_buffer, &endpoint, fin_rsv_opcode] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(!ec) { |
||||||
|
std::istream stream(read_buffer.get()); |
||||||
|
|
||||||
|
std::vector<unsigned char> length_bytes; |
||||||
|
length_bytes.resize(2); |
||||||
|
stream.read((char*)&length_bytes[0], 2); |
||||||
|
|
||||||
|
size_t length=0; |
||||||
|
int num_bytes=2; |
||||||
|
for(int c=0;c<num_bytes;c++) |
||||||
|
length+=length_bytes[c]<<(8*(num_bytes-1-c)); |
||||||
|
|
||||||
|
read_message_content(connection, read_buffer, length, endpoint, fin_rsv_opcode); |
||||||
|
} |
||||||
|
else |
||||||
|
connection_error(connection, endpoint, ec); |
||||||
|
}); |
||||||
|
} |
||||||
|
else if(length==127) { |
||||||
|
//8 next bytes is the size of content
|
||||||
|
boost::asio::async_read(*connection->socket, *read_buffer, boost::asio::transfer_exactly(8), |
||||||
|
[this, connection, read_buffer, &endpoint, fin_rsv_opcode] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(!ec) { |
||||||
|
std::istream stream(read_buffer.get()); |
||||||
|
|
||||||
|
std::vector<unsigned char> length_bytes; |
||||||
|
length_bytes.resize(8); |
||||||
|
stream.read((char*)&length_bytes[0], 8); |
||||||
|
|
||||||
|
size_t length=0; |
||||||
|
int num_bytes=8; |
||||||
|
for(int c=0;c<num_bytes;c++) |
||||||
|
length+=length_bytes[c]<<(8*(num_bytes-1-c)); |
||||||
|
|
||||||
|
read_message_content(connection, read_buffer, length, endpoint, fin_rsv_opcode); |
||||||
|
} |
||||||
|
else |
||||||
|
connection_error(connection, endpoint, ec); |
||||||
|
}); |
||||||
|
} |
||||||
|
else |
||||||
|
read_message_content(connection, read_buffer, length, endpoint, fin_rsv_opcode); |
||||||
|
} |
||||||
|
else |
||||||
|
connection_error(connection, endpoint, ec); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
void read_message_content(const std::shared_ptr<Connection> &connection, const std::shared_ptr<boost::asio::streambuf> &read_buffer, |
||||||
|
size_t length, Endpoint& endpoint, unsigned char fin_rsv_opcode) const { |
||||||
|
boost::asio::async_read(*connection->socket, *read_buffer, boost::asio::transfer_exactly(4+length), |
||||||
|
[this, connection, read_buffer, length, &endpoint, fin_rsv_opcode] |
||||||
|
(const boost::system::error_code& ec, size_t /*bytes_transferred*/) { |
||||||
|
if(!ec) { |
||||||
|
std::istream raw_message_data(read_buffer.get()); |
||||||
|
|
||||||
|
//Read mask
|
||||||
|
std::vector<unsigned char> mask; |
||||||
|
mask.resize(4); |
||||||
|
raw_message_data.read((char*)&mask[0], 4); |
||||||
|
|
||||||
|
std::shared_ptr<Message> message(new Message()); |
||||||
|
message->length=length; |
||||||
|
message->fin_rsv_opcode=fin_rsv_opcode; |
||||||
|
|
||||||
|
std::ostream message_data_out_stream(&message->streambuf); |
||||||
|
for(size_t c=0;c<length;c++) { |
||||||
|
message_data_out_stream.put(raw_message_data.get()^mask[c%4]); |
||||||
|
} |
||||||
|
|
||||||
|
//If connection close
|
||||||
|
if((fin_rsv_opcode&0x0f)==8) { |
||||||
|
int status=0; |
||||||
|
if(length>=2) { |
||||||
|
unsigned char byte1=message->get(); |
||||||
|
unsigned char byte2=message->get(); |
||||||
|
status=(byte1<<8)+byte2; |
||||||
|
} |
||||||
|
|
||||||
|
auto reason=message->string(); |
||||||
|
send_close(connection, status, reason, [this, connection](const boost::system::error_code& /*ec*/) {}); |
||||||
|
connection_close(connection, endpoint, status, reason); |
||||||
|
return; |
||||||
|
} |
||||||
|
else { |
||||||
|
//If ping
|
||||||
|
if((fin_rsv_opcode&0x0f)==9) { |
||||||
|
//send pong
|
||||||
|
auto empty_send_stream=std::make_shared<SendStream>(); |
||||||
|
send(connection, empty_send_stream, nullptr, fin_rsv_opcode+1); |
||||||
|
} |
||||||
|
else if(endpoint.on_message) { |
||||||
|
timer_idle_reset(connection); |
||||||
|
endpoint.on_message(connection, message); |
||||||
|
} |
||||||
|
|
||||||
|
//Next message
|
||||||
|
read_message(connection, read_buffer, endpoint); |
||||||
|
} |
||||||
|
} |
||||||
|
else |
||||||
|
connection_error(connection, endpoint, ec); |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
void connection_open(const std::shared_ptr<Connection> &connection, Endpoint& endpoint) { |
||||||
|
timer_idle_init(connection); |
||||||
|
|
||||||
|
{ |
||||||
|
std::lock_guard<std::mutex> lock(endpoint.connections_mutex); |
||||||
|
endpoint.connections.insert(connection); |
||||||
|
} |
||||||
|
|
||||||
|
if(endpoint.on_open) |
||||||
|
endpoint.on_open(connection); |
||||||
|
} |
||||||
|
|
||||||
|
void connection_close(const std::shared_ptr<Connection> &connection, Endpoint& endpoint, int status, const std::string& reason) const { |
||||||
|
timer_idle_cancel(connection); |
||||||
|
|
||||||
|
{ |
||||||
|
std::lock_guard<std::mutex> lock(endpoint.connections_mutex); |
||||||
|
endpoint.connections.erase(connection); |
||||||
|
} |
||||||
|
|
||||||
|
if(endpoint.on_close) |
||||||
|
endpoint.on_close(connection, status, reason); |
||||||
|
} |
||||||
|
|
||||||
|
void connection_error(const std::shared_ptr<Connection> &connection, Endpoint& endpoint, const boost::system::error_code& ec) const { |
||||||
|
timer_idle_cancel(connection); |
||||||
|
|
||||||
|
{ |
||||||
|
std::lock_guard<std::mutex> lock(endpoint.connections_mutex); |
||||||
|
endpoint.connections.erase(connection); |
||||||
|
} |
||||||
|
|
||||||
|
if(endpoint.on_error) |
||||||
|
endpoint.on_error(connection, ec); |
||||||
|
} |
||||||
|
|
||||||
|
void timer_idle_init(const std::shared_ptr<Connection> &connection) { |
||||||
|
if(config.timeout_idle>0) { |
||||||
|
connection->timer_idle=std::unique_ptr<boost::asio::deadline_timer>(new boost::asio::deadline_timer(connection->socket->get_io_service())); |
||||||
|
connection->timer_idle->expires_from_now(boost::posix_time::seconds(static_cast<unsigned long>(config.timeout_idle))); |
||||||
|
timer_idle_expired_function(connection); |
||||||
|
} |
||||||
|
} |
||||||
|
void timer_idle_reset(const std::shared_ptr<Connection> &connection) const { |
||||||
|
if(config.timeout_idle>0 && connection->timer_idle->expires_from_now(boost::posix_time::seconds(static_cast<unsigned long>(config.timeout_idle)))>0) |
||||||
|
timer_idle_expired_function(connection); |
||||||
|
} |
||||||
|
void timer_idle_cancel(const std::shared_ptr<Connection> &connection) const { |
||||||
|
if(config.timeout_idle>0) |
||||||
|
connection->timer_idle->cancel(); |
||||||
|
} |
||||||
|
|
||||||
|
void timer_idle_expired_function(const std::shared_ptr<Connection> &connection) const { |
||||||
|
connection->timer_idle->async_wait([this, connection](const boost::system::error_code& ec){ |
||||||
|
if(!ec) |
||||||
|
send_close(connection, 1000, "idle timeout"); //1000=normal closure
|
||||||
|
}); |
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
template<class socket_type> |
||||||
|
class SocketServer : public SocketServerBase<socket_type> {}; |
||||||
|
|
||||||
|
typedef boost::asio::ip::tcp::socket WS; |
||||||
|
|
||||||
|
template<> |
||||||
|
class SocketServer<WS> : public SocketServerBase<WS> { |
||||||
|
public: |
||||||
|
DEPRECATED SocketServer(unsigned short port, size_t thread_pool_size=1, size_t timeout_request=5, size_t timeout_idle=0) :
|
||||||
|
SocketServer() { |
||||||
|
config.port=port; |
||||||
|
config.thread_pool_size=thread_pool_size; |
||||||
|
config.timeout_request=timeout_request; |
||||||
|
config.timeout_idle=timeout_idle; |
||||||
|
}; |
||||||
|
|
||||||
|
SocketServer() : SocketServerBase<WS>(80) {} |
||||||
|
|
||||||
|
protected: |
||||||
|
void accept() { |
||||||
|
//Create new socket for this connection (stored in Connection::socket)
|
||||||
|
//Shared_ptr is used to pass temporary objects to the asynchronous functions
|
||||||
|
std::shared_ptr<Connection> connection(new Connection(new WS(*io_service))); |
||||||
|
|
||||||
|
acceptor->async_accept(*connection->socket, [this, connection](const boost::system::error_code& ec) { |
||||||
|
//Immediately start accepting a new connection (if io_service hasn't been stopped)
|
||||||
|
if (ec != boost::asio::error::operation_aborted) |
||||||
|
accept(); |
||||||
|
|
||||||
|
if(!ec) { |
||||||
|
boost::asio::ip::tcp::no_delay option(true); |
||||||
|
connection->socket->set_option(option); |
||||||
|
|
||||||
|
read_handshake(connection); |
||||||
|
} |
||||||
|
}); |
||||||
|
} |
||||||
|
}; |
||||||
|
// TODO: remove when onopen, onmessage, etc is removed:
|
||||||
|
#ifdef __GNUC__ |
||||||
|
#pragma GCC diagnostic pop |
||||||
|
#elif defined(_MSC_VER) |
||||||
|
#pragma warning(pop) |
||||||
|
#endif |
||||||
|
} |
||||||
|
|
||||||
|
#endif /* SERVER_WS_HPP */ |
||||||
Loading…
Reference in new issue