2018-09-27 23:56:48 +02:00
|
|
|
/*
|
|
|
|
* IXWebSocketTransport.h
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2017-2018 Machine Zone, Inc. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
//
|
|
|
|
// Adapted from https://github.com/dhbaird/easywsclient
|
|
|
|
//
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <functional>
|
|
|
|
#include <memory>
|
|
|
|
#include <mutex>
|
2018-10-01 23:46:11 +02:00
|
|
|
#include <atomic>
|
2019-02-21 03:59:07 +01:00
|
|
|
#include <list>
|
2018-09-27 23:56:48 +02:00
|
|
|
|
2018-11-10 03:23:49 +01:00
|
|
|
#include "IXWebSocketSendInfo.h"
|
2018-11-07 23:54:44 +01:00
|
|
|
#include "IXWebSocketPerMessageDeflate.h"
|
2018-11-10 03:23:49 +01:00
|
|
|
#include "IXWebSocketPerMessageDeflateOptions.h"
|
|
|
|
#include "IXWebSocketHttpHeaders.h"
|
2019-01-02 16:45:07 +01:00
|
|
|
#include "IXCancellationRequest.h"
|
2019-01-03 05:07:54 +01:00
|
|
|
#include "IXWebSocketHandshake.h"
|
2019-02-21 03:59:07 +01:00
|
|
|
#include "IXProgressCallback.h"
|
2019-05-16 21:46:53 +02:00
|
|
|
#include "IXWebSocketCloseConstants.h"
|
2018-11-07 23:54:44 +01:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
namespace ix
|
2018-09-27 23:56:48 +02:00
|
|
|
{
|
|
|
|
class Socket;
|
|
|
|
|
2019-03-22 22:22:58 +01:00
|
|
|
enum class SendMessageKind
|
|
|
|
{
|
|
|
|
Text,
|
|
|
|
Binary,
|
|
|
|
Ping
|
|
|
|
};
|
|
|
|
|
2018-09-27 23:56:48 +02:00
|
|
|
class WebSocketTransport
|
|
|
|
{
|
|
|
|
public:
|
2019-05-11 23:22:06 +02:00
|
|
|
enum class ReadyState
|
2018-09-27 23:56:48 +02:00
|
|
|
{
|
|
|
|
CLOSING,
|
|
|
|
CLOSED,
|
|
|
|
CONNECTING,
|
|
|
|
OPEN
|
|
|
|
};
|
|
|
|
|
2019-05-11 23:22:06 +02:00
|
|
|
enum class MessageKind
|
2018-10-25 21:01:47 +02:00
|
|
|
{
|
|
|
|
MSG,
|
|
|
|
PING,
|
2019-03-11 19:12:43 +01:00
|
|
|
PONG,
|
|
|
|
FRAGMENT
|
2018-10-25 21:01:47 +02:00
|
|
|
};
|
|
|
|
|
2019-05-11 23:22:06 +02:00
|
|
|
enum class PollResult
|
2019-05-09 18:21:05 +02:00
|
|
|
{
|
2019-05-11 23:22:06 +02:00
|
|
|
Succeeded,
|
|
|
|
AbnormalClose
|
2019-05-09 18:21:05 +02:00
|
|
|
};
|
|
|
|
|
2018-10-25 21:01:47 +02:00
|
|
|
using OnMessageCallback = std::function<void(const std::string&,
|
2018-11-10 03:23:49 +01:00
|
|
|
size_t,
|
2018-11-15 00:52:28 +01:00
|
|
|
bool,
|
2018-10-25 21:01:47 +02:00
|
|
|
MessageKind)>;
|
2018-10-26 03:51:19 +02:00
|
|
|
using OnCloseCallback = std::function<void(uint16_t,
|
2018-11-10 03:23:49 +01:00
|
|
|
const std::string&,
|
2019-04-23 13:31:55 +02:00
|
|
|
size_t,
|
|
|
|
bool)>;
|
2018-09-27 23:56:48 +02:00
|
|
|
|
|
|
|
WebSocketTransport();
|
|
|
|
~WebSocketTransport();
|
|
|
|
|
2019-01-24 21:42:49 +01:00
|
|
|
void configure(const WebSocketPerMessageDeflateOptions& perMessageDeflateOptions,
|
2019-04-18 18:24:16 +02:00
|
|
|
bool enablePong,
|
2019-04-20 01:57:38 +02:00
|
|
|
int pingIntervalSecs,
|
|
|
|
int pingTimeoutSecs);
|
2018-09-27 23:56:48 +02:00
|
|
|
|
2019-01-04 03:33:08 +01:00
|
|
|
WebSocketInitResult connectToUrl(const std::string& url, // Client
|
|
|
|
int timeoutSecs);
|
|
|
|
WebSocketInitResult connectToSocket(int fd, // Server
|
|
|
|
int timeoutSecs);
|
2018-12-30 06:53:33 +01:00
|
|
|
|
2019-05-11 23:22:06 +02:00
|
|
|
PollResult poll();
|
2019-02-21 03:59:07 +01:00
|
|
|
WebSocketSendInfo sendBinary(const std::string& message,
|
|
|
|
const OnProgressCallback& onProgressCallback);
|
2019-03-22 22:22:58 +01:00
|
|
|
WebSocketSendInfo sendText(const std::string& message,
|
|
|
|
const OnProgressCallback& onProgressCallback);
|
2018-11-10 03:23:49 +01:00
|
|
|
WebSocketSendInfo sendPing(const std::string& message);
|
2019-04-16 18:55:12 +02:00
|
|
|
|
2019-05-16 21:46:53 +02:00
|
|
|
void close(uint16_t code = WebSocketCloseConstants::kNormalClosureCode,
|
|
|
|
const std::string& reason = WebSocketCloseConstants::kNormalClosureMessage,
|
2019-04-23 13:31:55 +02:00
|
|
|
size_t closeWireSize = 0,
|
|
|
|
bool remote = false);
|
2019-04-16 18:55:12 +02:00
|
|
|
|
2019-05-16 00:18:27 +02:00
|
|
|
void closeSocket();
|
|
|
|
ssize_t send();
|
|
|
|
|
2019-05-11 23:22:06 +02:00
|
|
|
ReadyState getReadyState() const;
|
|
|
|
void setReadyState(ReadyState readyState);
|
2018-10-26 03:51:19 +02:00
|
|
|
void setOnCloseCallback(const OnCloseCallback& onCloseCallback);
|
2019-05-11 23:22:06 +02:00
|
|
|
void dispatch(PollResult pollResult,
|
2019-05-09 18:21:05 +02:00
|
|
|
const OnMessageCallback& onMessageCallback);
|
2019-03-14 07:09:45 +01:00
|
|
|
size_t bufferedAmount() const;
|
2018-09-27 23:56:48 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
std::string _url;
|
|
|
|
|
|
|
|
struct wsheader_type {
|
|
|
|
unsigned header_size;
|
|
|
|
bool fin;
|
2018-11-07 20:45:17 +01:00
|
|
|
bool rsv1;
|
2018-09-27 23:56:48 +02:00
|
|
|
bool mask;
|
|
|
|
enum opcode_type {
|
|
|
|
CONTINUATION = 0x0,
|
2019-05-11 23:22:06 +02:00
|
|
|
TEXT_FRAME = 0x1,
|
2018-09-27 23:56:48 +02:00
|
|
|
BINARY_FRAME = 0x2,
|
2019-05-11 23:22:06 +02:00
|
|
|
CLOSE = 8,
|
|
|
|
PING = 9,
|
|
|
|
PONG = 0xa,
|
2018-09-27 23:56:48 +02:00
|
|
|
} opcode;
|
|
|
|
int N0;
|
|
|
|
uint64_t N;
|
|
|
|
uint8_t masking_key[4];
|
|
|
|
};
|
|
|
|
|
2019-03-22 17:53:56 +01:00
|
|
|
// Tells whether we should mask the data we send.
|
|
|
|
// client should mask but server should not
|
2019-04-29 17:16:30 +02:00
|
|
|
std::atomic<bool> _useMask;
|
2019-03-22 17:53:56 +01:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
// Buffer for reading from our socket. That buffer is never resized.
|
|
|
|
std::vector<uint8_t> _readbuf;
|
|
|
|
|
|
|
|
// Contains all messages that were fetched in the last socket read.
|
|
|
|
// This could be a mix of control messages (Close, Ping, etc...) and
|
|
|
|
// data messages. That buffer
|
2018-09-27 23:56:48 +02:00
|
|
|
std::vector<uint8_t> _rxbuf;
|
2019-02-21 03:59:07 +01:00
|
|
|
|
|
|
|
// Contains all messages that are waiting to be sent
|
2018-09-27 23:56:48 +02:00
|
|
|
std::vector<uint8_t> _txbuf;
|
|
|
|
mutable std::mutex _txbufMutex;
|
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
// Hold fragments for multi-fragments messages in a list. We support receiving very large
|
|
|
|
// messages (tested messages up to 700M) and we cannot put them in a single
|
|
|
|
// buffer that is resized, as this operation can be slow when a buffer has its
|
|
|
|
// size increased 2 fold, while appending to a list has a fixed cost.
|
|
|
|
std::list<std::vector<uint8_t>> _chunks;
|
|
|
|
|
|
|
|
// Fragments are 32K long
|
|
|
|
static constexpr size_t kChunkSize = 1 << 15;
|
|
|
|
|
|
|
|
// Underlying TCP socket
|
2018-09-27 23:56:48 +02:00
|
|
|
std::shared_ptr<Socket> _socket;
|
2019-05-16 04:19:13 +02:00
|
|
|
std::mutex _socketMutex;
|
2018-09-27 23:56:48 +02:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
// Hold the state of the connection (OPEN, CLOSED, etc...)
|
2019-05-11 23:22:06 +02:00
|
|
|
std::atomic<ReadyState> _readyState;
|
2018-09-27 23:56:48 +02:00
|
|
|
|
2018-10-26 03:51:19 +02:00
|
|
|
OnCloseCallback _onCloseCallback;
|
|
|
|
uint16_t _closeCode;
|
|
|
|
std::string _closeReason;
|
2018-11-10 03:23:49 +01:00
|
|
|
size_t _closeWireSize;
|
2019-04-23 13:31:55 +02:00
|
|
|
bool _closeRemote;
|
2018-10-26 03:51:19 +02:00
|
|
|
mutable std::mutex _closeDataMutex;
|
2018-09-27 23:56:48 +02:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
// Data used for Per Message Deflate compression (with zlib)
|
2018-11-07 23:54:44 +01:00
|
|
|
WebSocketPerMessageDeflate _perMessageDeflate;
|
2018-11-10 03:23:49 +01:00
|
|
|
WebSocketPerMessageDeflateOptions _perMessageDeflateOptions;
|
|
|
|
std::atomic<bool> _enablePerMessageDeflate;
|
2018-11-07 23:54:44 +01:00
|
|
|
|
2018-12-15 01:28:11 +01:00
|
|
|
// Used to cancel dns lookup + socket connect + http upgrade
|
|
|
|
std::atomic<bool> _requestInitCancellation;
|
2019-05-16 01:50:00 +02:00
|
|
|
|
2019-05-09 18:21:05 +02:00
|
|
|
mutable std::mutex _closingTimePointMutex;
|
|
|
|
std::chrono::time_point<std::chrono::steady_clock>_closingTimePoint;
|
|
|
|
static const int kClosingMaximumWaitingDelayInMs;
|
2019-02-21 03:59:07 +01:00
|
|
|
|
2019-04-18 19:02:31 +02:00
|
|
|
// Constants for dealing with closing conneections
|
2019-04-19 18:16:25 +02:00
|
|
|
static const uint16_t kInternalErrorCode;
|
|
|
|
static const uint16_t kAbnormalCloseCode;
|
2019-04-23 13:31:55 +02:00
|
|
|
static const uint16_t kProtocolErrorCode;
|
2019-05-09 18:21:05 +02:00
|
|
|
static const uint16_t kNoStatusCodeErrorCode;
|
2019-04-20 01:57:38 +02:00
|
|
|
static const std::string kInternalErrorMessage;
|
|
|
|
static const std::string kAbnormalCloseMessage;
|
|
|
|
static const std::string kPingTimeoutMessage;
|
2019-04-23 13:31:55 +02:00
|
|
|
static const std::string kProtocolErrorMessage;
|
2019-05-09 18:21:05 +02:00
|
|
|
static const std::string kNoStatusCodeErrorMessage;
|
2019-04-20 01:57:38 +02:00
|
|
|
|
2019-04-18 18:24:16 +02:00
|
|
|
// enable auto response to ping
|
2019-05-10 00:06:05 +02:00
|
|
|
std::atomic<bool> _enablePong;
|
2019-04-18 18:24:16 +02:00
|
|
|
static const bool kDefaultEnablePong;
|
|
|
|
|
2019-04-18 19:02:31 +02:00
|
|
|
// Optional ping and pong timeout
|
2019-04-20 01:57:38 +02:00
|
|
|
// if both ping interval and timeout are set (> 0),
|
|
|
|
// then use GCD of these value to wait for the lowest time
|
2019-04-18 18:24:16 +02:00
|
|
|
int _pingIntervalSecs;
|
|
|
|
int _pingTimeoutSecs;
|
2019-04-20 01:57:38 +02:00
|
|
|
int _pingIntervalOrTimeoutGCDSecs;
|
2019-04-18 19:02:31 +02:00
|
|
|
|
2019-04-18 18:24:16 +02:00
|
|
|
static const int kDefaultPingIntervalSecs;
|
|
|
|
static const int kDefaultPingTimeoutSecs;
|
2019-04-20 01:57:38 +02:00
|
|
|
static const std::string kPingMessage;
|
|
|
|
|
2019-05-09 18:21:05 +02:00
|
|
|
// Record time step for ping/ ping timeout to ensure we wait for the right left duration
|
|
|
|
std::chrono::time_point<std::chrono::steady_clock> _nextGCDTimePoint;
|
|
|
|
|
2019-04-20 01:57:38 +02:00
|
|
|
// We record when ping are being sent so that we can know when to send the next one
|
2019-04-18 19:02:31 +02:00
|
|
|
// We also record when pong are being sent as a reply to pings, to close the connections
|
|
|
|
// if no pong were received sufficiently fast.
|
2019-04-18 18:24:16 +02:00
|
|
|
mutable std::mutex _lastSendPingTimePointMutex;
|
|
|
|
mutable std::mutex _lastReceivePongTimePointMutex;
|
|
|
|
std::chrono::time_point<std::chrono::steady_clock> _lastSendPingTimePoint;
|
|
|
|
std::chrono::time_point<std::chrono::steady_clock> _lastReceivePongTimePoint;
|
|
|
|
|
2019-04-18 19:02:31 +02:00
|
|
|
// If this function returns true, it is time to send a new ping
|
2019-04-18 18:24:16 +02:00
|
|
|
bool pingIntervalExceeded();
|
2019-04-20 01:57:38 +02:00
|
|
|
|
2019-04-18 18:24:16 +02:00
|
|
|
// No PONG data was received through the socket for longer than ping timeout delay
|
|
|
|
bool pingTimeoutExceeded();
|
2018-12-15 01:28:11 +01:00
|
|
|
|
2019-05-09 18:21:05 +02:00
|
|
|
// after calling close(), if no CLOSE frame answer is received back from the remote, we should close the connexion
|
|
|
|
bool closingDelayExceeded();
|
|
|
|
|
2019-05-13 18:08:46 +02:00
|
|
|
void initTimePointsAndGCDAfterConnect();
|
|
|
|
|
2019-05-09 18:21:05 +02:00
|
|
|
void sendCloseFrame(uint16_t code, const std::string& reason);
|
|
|
|
|
|
|
|
void closeSocketAndSwitchToClosedState(uint16_t code,
|
|
|
|
const std::string& reason,
|
|
|
|
size_t closeWireSize,
|
|
|
|
bool remote);
|
|
|
|
|
2018-09-27 23:56:48 +02:00
|
|
|
void sendOnSocket();
|
2019-02-21 03:59:07 +01:00
|
|
|
WebSocketSendInfo sendData(wsheader_type::opcode_type type,
|
2018-11-12 18:00:55 +01:00
|
|
|
const std::string& message,
|
2019-02-21 03:59:07 +01:00
|
|
|
bool compress,
|
|
|
|
const OnProgressCallback& onProgressCallback = nullptr);
|
|
|
|
|
|
|
|
void sendFragment(wsheader_type::opcode_type type,
|
|
|
|
bool fin,
|
|
|
|
std::string::const_iterator begin,
|
|
|
|
std::string::const_iterator end,
|
|
|
|
bool compress);
|
2018-11-10 03:23:49 +01:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
void emitMessage(MessageKind messageKind,
|
2018-11-10 03:23:49 +01:00
|
|
|
const std::string& message,
|
|
|
|
const wsheader_type& ws,
|
|
|
|
const OnMessageCallback& onMessageCallback);
|
2018-09-27 23:56:48 +02:00
|
|
|
|
|
|
|
bool isSendBufferEmpty() const;
|
|
|
|
void appendToSendBuffer(const std::vector<uint8_t>& header,
|
|
|
|
std::string::const_iterator begin,
|
|
|
|
std::string::const_iterator end,
|
|
|
|
uint64_t message_size,
|
|
|
|
uint8_t masking_key[4]);
|
|
|
|
|
|
|
|
unsigned getRandomUnsigned();
|
2018-10-26 03:51:19 +02:00
|
|
|
void unmaskReceiveBuffer(const wsheader_type& ws);
|
2019-02-21 03:59:07 +01:00
|
|
|
|
|
|
|
std::string getMergedChunks() const;
|
2018-09-27 23:56:48 +02:00
|
|
|
};
|
|
|
|
}
|