2#include <boost/algorithm/string/predicate.hpp>
3#include <boost/array.hpp>
4#include "crow/socket_adaptors.h"
5#include "crow/http_request.h"
6#include "crow/TinySHA1.hpp"
12 enum class WebSocketReadState
24 virtual void send_binary(
const std::string& msg) = 0;
25 virtual void send_text(
const std::string& msg) = 0;
26 virtual void send_ping(
const std::string& msg) = 0;
27 virtual void send_pong(
const std::string& msg) = 0;
28 virtual void close(
const std::string& msg =
"quit") = 0;
31 void userdata(
void* u) { userdata_ = u; }
32 void* userdata() {
return userdata_; }
58 template <
typename Adaptor>
73 : adaptor_(std::move(adaptor)), open_handler_(std::move(open_handler)), message_handler_(std::move(message_handler)), close_handler_(std::move(close_handler)), error_handler_(std::move(error_handler))
74 , accept_handler_(std::move(accept_handler))
76 if (!boost::iequals(req.get_header_value(
"upgrade"),
"websocket"))
85 if (!accept_handler_(req))
95 std::string magic = req.get_header_value(
"Sec-WebSocket-Key") +
"258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
97 s.processBytes(magic.data(), magic.size());
99 s.getDigestBytes(digest);
100 start(crow::utility::base64encode((
char*)digest, 20));
104 template<
typename CompletionHandler>
107 adaptor_.get_io_service().dispatch(handler);
111 template<
typename CompletionHandler>
112 void post(CompletionHandler handler)
114 adaptor_.get_io_service().post(handler);
125 write_buffers_.emplace_back(std::move(header));
126 write_buffers_.emplace_back(msg);
139 write_buffers_.emplace_back(std::move(header));
140 write_buffers_.emplace_back(msg);
150 write_buffers_.emplace_back(std::move(header));
151 write_buffers_.emplace_back(msg);
161 write_buffers_.emplace_back(std::move(header));
162 write_buffers_.emplace_back(msg);
171 void close(
const std::string& msg)
override
174 has_sent_close_ =
true;
175 if (has_recv_close_ && !is_close_handler_called_)
177 is_close_handler_called_ =
true;
179 close_handler_(*
this, msg);
182 write_buffers_.emplace_back(std::move(header));
183 write_buffers_.emplace_back(msg);
193 char buf[2+8] =
"\x80\x00";
197 buf[1] +=
static_cast<char>(size);
200 else if (size < 0x10000)
203 *(uint16_t*)(buf+2) = htons(
static_cast<uint16_t
>(size));
209 *
reinterpret_cast<uint64_t*
>(buf+2) = ((1==htonl(1)) ?
static_cast<uint64_t
>(size) : (
static_cast<uint64_t
>(htonl((size) & 0xFFFFFFFF)) << 32) | htonl(
static_cast<uint64_t
>(size) >> 32));
210 return {buf, buf+10};
220 static std::string header =
"HTTP/1.1 101 Switching Protocols\r\n"
221 "Upgrade: websocket\r\n"
222 "Connection: Upgrade\r\n"
223 "Sec-WebSocket-Accept: ";
224 static std::string crlf =
"\r\n";
225 write_buffers_.emplace_back(header);
226 write_buffers_.emplace_back(std::move(hello));
227 write_buffers_.emplace_back(crlf);
228 write_buffers_.emplace_back(crlf);
231 open_handler_(*
this);
247 case WebSocketReadState::MiniHeader:
251 adaptor_.socket().async_read_some(boost::asio::buffer(&mini_header_, 2),
252 [
this](
const boost::system::error_code& ec, std::size_t
253#ifdef CROW_ENABLE_DEBUG
260 mini_header_ = ntohs(mini_header_);
261#ifdef CROW_ENABLE_DEBUG
263 if (!ec && bytes_transferred != 2)
265 throw std::runtime_error(
"WebSocket:MiniHeader:async_read fail:asio bug?");
271 if ((mini_header_ & 0x80) == 0x80)
274 if ((mini_header_ & 0x7f) == 127)
276 state_ = WebSocketReadState::Len64;
278 else if ((mini_header_ & 0x7f) == 126)
280 state_ = WebSocketReadState::Len16;
284 remaining_length_ = mini_header_ & 0x7f;
285 state_ = WebSocketReadState::Mask;
291 close_connection_ =
true;
294 error_handler_(*
this);
300 case WebSocketReadState::Len16:
302 remaining_length_ = 0;
303 remaining_length16_ = 0;
304 boost::asio::async_read(adaptor_.socket(), boost::asio::buffer(&remaining_length16_, 2),
305 [
this](
const boost::system::error_code& ec, std::size_t
306#ifdef CROW_ENABLE_DEBUG
312 remaining_length16_ = ntohs(remaining_length16_);
313 remaining_length_ = remaining_length16_;
314#ifdef CROW_ENABLE_DEBUG
315 if (!ec && bytes_transferred != 2)
317 throw std::runtime_error(
"WebSocket:Len16:async_read fail:asio bug?");
323 state_ = WebSocketReadState::Mask;
328 close_connection_ =
true;
331 error_handler_(*
this);
337 case WebSocketReadState::Len64:
339 boost::asio::async_read(adaptor_.socket(), boost::asio::buffer(&remaining_length_, 8),
340 [
this](
const boost::system::error_code& ec, std::size_t
341#ifdef CROW_ENABLE_DEBUG
347 remaining_length_ = ((1==ntohl(1)) ? (remaining_length_) : (static_cast<uint64_t>(ntohl((remaining_length_) & 0xFFFFFFFF)) << 32) | ntohl((remaining_length_) >> 32));
348#ifdef CROW_ENABLE_DEBUG
349 if (!ec && bytes_transferred != 8)
351 throw std::runtime_error(
"WebSocket:Len16:async_read fail:asio bug?");
357 state_ = WebSocketReadState::Mask;
362 close_connection_ =
true;
365 error_handler_(*
this);
371 case WebSocketReadState::Mask:
374 boost::asio::async_read(adaptor_.socket(), boost::asio::buffer((
char*)&mask_, 4),
375 [
this](
const boost::system::error_code& ec, std::size_t
376#ifdef CROW_ENABLE_DEBUG
382#ifdef CROW_ENABLE_DEBUG
383 if (!ec && bytes_transferred != 4)
385 throw std::runtime_error(
"WebSocket:Mask:async_read fail:asio bug?");
391 state_ = WebSocketReadState::Payload;
396 close_connection_ =
true;
398 error_handler_(*
this);
405 state_ = WebSocketReadState::Payload;
409 case WebSocketReadState::Payload:
411 auto to_read =
static_cast<std::uint64_t
>(buffer_.size());
412 if (remaining_length_ < to_read)
413 to_read = remaining_length_;
414 adaptor_.socket().async_read_some(boost::asio::buffer(buffer_,
static_cast<std::size_t
>(to_read)),
415 [
this](
const boost::system::error_code& ec, std::size_t bytes_transferred)
421 fragment_.insert(fragment_.end(), buffer_.begin(), buffer_.begin() + bytes_transferred);
422 remaining_length_ -= bytes_transferred;
423 if (remaining_length_ == 0)
426 state_ = WebSocketReadState::MiniHeader;
434 close_connection_ =
true;
436 error_handler_(*
this);
448 return mini_header_ & 0x8000;
454 return (mini_header_ & 0x0f00) >> 8;
465 for(
decltype(fragment_.length()) i = 0; i < fragment_.length(); i ++)
467 fragment_[i] ^= ((
char*)&mask_)[i%4];
474 message_ += fragment_;
477 if (message_handler_)
478 message_handler_(*
this, message_, is_binary_);
486 message_ += fragment_;
489 if (message_handler_)
490 message_handler_(*
this, message_, is_binary_);
498 message_ += fragment_;
501 if (message_handler_)
502 message_handler_(*
this, message_, is_binary_);
509 has_recv_close_ =
true;
510 if (!has_sent_close_)
517 close_connection_ =
true;
518 if (!is_close_handler_called_)
521 close_handler_(*
this, fragment_);
522 is_close_handler_called_ =
true;
530 send_pong(fragment_);
535 pong_received_ =
true;
549 if (sending_buffers_.empty())
551 sending_buffers_.swap(write_buffers_);
552 std::vector<boost::asio::const_buffer> buffers;
553 buffers.reserve(sending_buffers_.size());
554 for(
auto& s:sending_buffers_)
556 buffers.emplace_back(boost::asio::buffer(s));
558 boost::asio::async_write(adaptor_.socket(), buffers,
559 [&](
const boost::system::error_code& ec, std::size_t )
561 sending_buffers_.clear();
562 if (!ec && !close_connection_)
564 if (!write_buffers_.empty())
567 close_connection_ = true;
571 close_connection_ = true;
582 if (!is_close_handler_called_)
584 close_handler_(*
this,
"uncleanly");
585 if (sending_buffers_.empty() && !is_reading)
591 std::vector<std::string> sending_buffers_;
592 std::vector<std::string> write_buffers_;
594 boost::array<char, 4096> buffer_;
596 std::string message_;
597 std::string fragment_;
598 WebSocketReadState state_{WebSocketReadState::MiniHeader};
599 uint16_t remaining_length16_{0};
600 uint64_t remaining_length_{0};
601 bool close_connection_{
false};
602 bool is_reading{
false};
603 bool has_mask_{
false};
605 uint16_t mini_header_;
606 bool has_sent_close_{
false};
607 bool has_recv_close_{
false};
608 bool error_occured_{
false};
609 bool pong_received_{
false};
610 bool is_close_handler_called_{
false};
A websocket connection.
Definition: websocket.h:60
void check_destroy()
Destroy the Connection.
Definition: websocket.h:579
void close(const std::string &msg) override
Send a close signal.
Definition: websocket.h:171
void start(std::string &&hello)
Send the HTTP upgrade response.
Definition: websocket.h:218
void handle_fragment()
Process the payload fragment.
Definition: websocket.h:461
int opcode()
Extract the opcode from the header.
Definition: websocket.h:452
void send_ping(const std::string &msg) override
Send a "Ping" message.
Definition: websocket.h:121
void do_write()
Send the buffers' data through the socket.
Definition: websocket.h:547
void dispatch(CompletionHandler handler)
Send data through the socket.
Definition: websocket.h:105
Connection(const crow::request &req, Adaptor &&adaptor, std::function< void(crow::websocket::connection &)> open_handler, std::function< void(crow::websocket::connection &, const std::string &, bool)> message_handler, std::function< void(crow::websocket::connection &, const std::string &)> close_handler, std::function< void(crow::websocket::connection &)> error_handler, std::function< bool(const crow::request &)> accept_handler)
Constructor for a connection.
Definition: websocket.h:67
void send_text(const std::string &msg) override
Send a plaintext message.
Definition: websocket.h:157
void post(CompletionHandler handler)
Send data through the socket and return immediately.
Definition: websocket.h:112
std::string build_header(int opcode, size_t size)
Generate the websocket headers using an opcode and the message size (in bytes).
Definition: websocket.h:191
void do_read()
Read a websocket message.
Definition: websocket.h:242
void send_binary(const std::string &msg) override
Send a binary encoded message.
Definition: websocket.h:146
void send_pong(const std::string &msg) override
Send a "Pong" message.
Definition: websocket.h:135
bool is_FIN()
Check if the FIN bit is set.
Definition: websocket.h:446
An HTTP request.
Definition: http_request.h:27
A base class for websocket connection.
Definition: websocket.h:23