2023-12-31 22:23:25 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
|
|
|
Copyright (c) 2019-present Axmol Engine contributors (see AUTHORS.md).
|
|
|
|
|
|
|
|
https://axmolengine.github.io/
|
|
|
|
|
|
|
|
Permission is
|
|
|
|
* hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files
|
|
|
|
* (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use,
|
|
|
|
* copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to
|
|
|
|
* whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this
|
|
|
|
* permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS
|
|
|
|
* PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT
|
|
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
|
|
* ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
|
|
|
|
* ****************************************************************************/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#if !defined(__EMSCRIPTEN__)
|
|
|
|
|
|
|
|
# include <string>
|
|
|
|
# include <vector>
|
|
|
|
# include <mutex>
|
|
|
|
# include <memory> // for std::shared_ptr
|
|
|
|
# include <atomic>
|
|
|
|
# include <condition_variable>
|
2023-12-31 23:00:13 +08:00
|
|
|
# include <future>
|
2023-12-31 22:23:25 +08:00
|
|
|
|
|
|
|
# include "platform/PlatformMacros.h"
|
|
|
|
# include "platform/StdC.h"
|
|
|
|
# include "base/Scheduler.h"
|
|
|
|
# include "base/ConcurrentDeque.h"
|
|
|
|
# include "yasio/yasio_fwd.hpp"
|
|
|
|
# include "yasio/byte_buffer.hpp"
|
|
|
|
# include "network/Uri.h"
|
|
|
|
# include "llhttp.h"
|
|
|
|
# include "websocket_parser.h"
|
|
|
|
|
|
|
|
NS_AX_BEGIN
|
|
|
|
|
|
|
|
class EventListenerCustom;
|
|
|
|
|
|
|
|
namespace network
|
|
|
|
{
|
|
|
|
|
|
|
|
struct WebSocketProtocol;
|
|
|
|
|
|
|
|
/**
|
2024-01-02 19:10:51 +08:00
|
|
|
* WebSocket implementation using yasio.
|
|
|
|
* Please note that all public methods of WebSocket have to be invoked on Axmol Thread.
|
2023-12-31 22:23:25 +08:00
|
|
|
*/
|
|
|
|
class AX_DLL WebSocket
|
|
|
|
{
|
|
|
|
friend struct WebSocketProtocol;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Constructor of WebSocket.
|
|
|
|
*
|
|
|
|
* @js ctor
|
|
|
|
*/
|
|
|
|
WebSocket();
|
|
|
|
/**
|
|
|
|
* Destructor of WebSocket.
|
|
|
|
*
|
|
|
|
* @js NA
|
|
|
|
* @lua NA
|
|
|
|
*/
|
|
|
|
virtual ~WebSocket();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ErrorCode enum used to represent the error in the websocket.
|
|
|
|
*/
|
|
|
|
enum class ErrorCode
|
|
|
|
{
|
|
|
|
OK,
|
|
|
|
TIME_OUT, /** < value 0 */
|
|
|
|
CONNECTION_FAILURE, /** < value 1 */
|
|
|
|
NO_SEC_ACCEPT,
|
|
|
|
VERIFY_SEC_ACCEPT_FAILURE,
|
|
|
|
UPGRADE_FAILURE
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* State enum used to represent the Websocket state.
|
|
|
|
*/
|
|
|
|
enum class State
|
|
|
|
{
|
|
|
|
CONNECTING, /** < value 0 */
|
|
|
|
OPEN, /** < value 1 */
|
|
|
|
CLOSING, /** < value 2 */
|
|
|
|
CLOSED, /** < value 3 */
|
|
|
|
};
|
|
|
|
|
2024-05-03 22:15:08 +08:00
|
|
|
class Event : public ax::Object
|
2023-12-31 22:23:25 +08:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
enum class Type
|
|
|
|
{
|
|
|
|
ON_OPEN,
|
|
|
|
ON_ERROR,
|
|
|
|
ON_CLOSE,
|
|
|
|
ON_MESSAGE,
|
|
|
|
};
|
|
|
|
Type getType() const { return _type; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
Type _type;
|
|
|
|
};
|
|
|
|
|
|
|
|
class WsOpenEvent : public Event
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
WsOpenEvent() { this->_type = Type::ON_OPEN; }
|
|
|
|
};
|
|
|
|
|
|
|
|
class CloseEvent : public Event
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CloseEvent() { this->_type = Type::ON_CLOSE; }
|
|
|
|
};
|
|
|
|
|
|
|
|
class ErrorEvent : public Event
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ErrorEvent(ErrorCode errorCode) : _errorCode(errorCode) { this->_type = Type::ON_ERROR; }
|
|
|
|
|
|
|
|
ErrorCode getErrorCode() const { return _errorCode; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
ErrorCode _errorCode;
|
|
|
|
};
|
|
|
|
|
|
|
|
class MessageEvent : public Event
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
MessageEvent(yasio::sbyte_buffer&& message, bool isBinary) : _message(std::move(message)), _isBinary(isBinary)
|
|
|
|
{
|
|
|
|
this->_type = Type::ON_MESSAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
yasio::sbyte_buffer& getMessage() { return _message; }
|
|
|
|
const yasio::sbyte_buffer& getMessage() const { return _message; }
|
|
|
|
bool isBinary() const { return _isBinary; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
yasio::sbyte_buffer _message;
|
|
|
|
bool _isBinary;
|
|
|
|
/* TODO:
|
|
|
|
issued;
|
|
|
|
bool isBinary;
|
|
|
|
void* ext;
|
|
|
|
*/
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Data structure for message view
|
|
|
|
*/
|
|
|
|
struct Data
|
|
|
|
{
|
|
|
|
Data() : bytes(nullptr), len(0), issued(0), isBinary(false), ext(nullptr) {}
|
|
|
|
Data(MessageEvent* event)
|
|
|
|
: bytes(event->getMessage().data())
|
|
|
|
, len(event->getMessage().size())
|
|
|
|
, isBinary(event->isBinary())
|
|
|
|
, issued(0)
|
|
|
|
, ext(nullptr)
|
|
|
|
{}
|
|
|
|
const char* bytes;
|
|
|
|
size_t len, issued;
|
|
|
|
bool isBinary;
|
|
|
|
void* ext;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The delegate class is used to process websocket events.
|
|
|
|
*
|
|
|
|
* The most member function are pure virtual functions,they should be implemented the in subclass.
|
|
|
|
* @lua NA
|
|
|
|
*/
|
|
|
|
class Delegate
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/** Destructor of Delegate. */
|
|
|
|
virtual ~Delegate() {}
|
|
|
|
/**
|
|
|
|
* This function to be called after the client connection complete a handshake with the remote server.
|
|
|
|
* This means that the WebSocket connection is ready to send and receive data.
|
|
|
|
*
|
|
|
|
* @param ws The WebSocket object connected
|
|
|
|
*/
|
|
|
|
virtual void onOpen(WebSocket* ws) = 0;
|
|
|
|
/**
|
|
|
|
* This function to be called when data has appeared from the server for the client connection.
|
|
|
|
*
|
|
|
|
* @param ws The WebSocket object connected.
|
|
|
|
* @param data Data object for message.
|
|
|
|
*/
|
|
|
|
virtual void onMessage(WebSocket* ws, const Data& data) = 0;
|
|
|
|
/**
|
|
|
|
* When the WebSocket object connected wants to close or the protocol won't get used at all and current
|
|
|
|
* _readyState is State::CLOSING,this function is to be called.
|
|
|
|
*
|
|
|
|
* @param ws The WebSocket object connected.
|
|
|
|
*/
|
|
|
|
virtual void onClose(WebSocket* ws) = 0;
|
|
|
|
/**
|
|
|
|
* This function is to be called in the following cases:
|
|
|
|
* 1. client connection is failed.
|
|
|
|
* 2. the request client connection has been unable to complete a handshake with the remote server.
|
|
|
|
* 3. the protocol won't get used at all after this callback and current _readyState is State::CONNECTING.
|
|
|
|
* 4. when a socket descriptor needs to be removed from an external polling array. in is again the struct
|
|
|
|
* libwebsocket_pollargs containing the fd member to be removed. If you are using the internal polling loop, you
|
|
|
|
* can just ignore it and current _readyState is State::CONNECTING.
|
|
|
|
*
|
|
|
|
* @param ws The WebSocket object connected.
|
|
|
|
* @param error WebSocket::ErrorCode enum,would be ErrorCode::TIME_OUT or ErrorCode::CONNECTION_FAILURE.
|
|
|
|
*/
|
|
|
|
virtual void onError(WebSocket* ws, const ErrorCode& error) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The initialized method for websocket.
|
|
|
|
* It needs to be invoked right after websocket instance is allocated.
|
|
|
|
* @param delegate The delegate which want to receive event from websocket.
|
|
|
|
* @param url The URL of websocket server.
|
|
|
|
* @param caFilePath The ca file path for wss connection
|
2024-01-02 19:10:51 +08:00
|
|
|
* @param protocols Comma-separated list of sub-protocols that agree with websocket server
|
2023-12-31 22:23:25 +08:00
|
|
|
* @return true: Success, false: Failure.
|
|
|
|
* @lua NA
|
|
|
|
*/
|
|
|
|
bool open(Delegate* delegate,
|
|
|
|
std::string_view url,
|
|
|
|
std::string_view caFilePath = "",
|
2024-01-02 19:10:51 +08:00
|
|
|
std::string_view protocols = "");
|
2023-12-31 22:23:25 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sends string data to websocket server.
|
|
|
|
*
|
|
|
|
* @param message string data.
|
|
|
|
* @lua sendstring
|
|
|
|
*/
|
|
|
|
void send(std::string_view message);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sends binary data to websocket server.
|
|
|
|
*
|
2024-01-02 19:10:51 +08:00
|
|
|
* @param data binary data.
|
2023-12-31 22:23:25 +08:00
|
|
|
* @param len the size of binary string data.
|
|
|
|
* @lua sendstring
|
|
|
|
*/
|
|
|
|
void send(const void* data, unsigned int len);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Closes the connection to server synchronously.
|
|
|
|
* @note It's a synchronous method, it will not return until websocket thread exits.
|
|
|
|
*/
|
|
|
|
void close();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Closes the connection to server asynchronously.
|
|
|
|
* @note It's an asynchronous method, it just notifies websocket thread to exit and returns directly,
|
|
|
|
* If using 'closeAsync' to close websocket connection,
|
|
|
|
* be careful of not using destructed variables in the callback of 'onClose'.
|
|
|
|
*/
|
|
|
|
void closeAsync();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Gets current state of connection.
|
|
|
|
* @return State the state value could be State::CONNECTING, State::OPEN, State::CLOSING or State::CLOSED
|
|
|
|
*/
|
|
|
|
State getReadyState() const { return _state; };
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Gets the URL of websocket connection.
|
|
|
|
*/
|
2024-01-02 19:10:51 +08:00
|
|
|
std::string_view getUrl() const { return _url; }
|
2023-12-31 22:23:25 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Gets the protocol selected by websocket server.
|
|
|
|
*/
|
2024-01-02 19:10:51 +08:00
|
|
|
std::string_view getProtocol() const { return _selectedProtocol; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set custom-defined headers.
|
|
|
|
*
|
|
|
|
* @param headers The string vector of custom-defined headers.
|
|
|
|
*/
|
|
|
|
void setHeaders(const std::vector<std::string>& headers) { _headers = headers; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get custom headers.
|
|
|
|
*
|
|
|
|
* @return std::vector<std::string> the string vector of custom-defined headers.
|
|
|
|
*/
|
|
|
|
const std::vector<std::string>& getHeaders() const { return _headers; }
|
2023-12-31 22:23:25 +08:00
|
|
|
|
|
|
|
protected:
|
|
|
|
void purgePendingEvents();
|
|
|
|
void dispatchEvents();
|
|
|
|
|
|
|
|
void setupParsers();
|
|
|
|
void generateHandshakeSecKey();
|
|
|
|
void handleNetworkEvent(yasio::io_event* event);
|
|
|
|
|
|
|
|
void do_handshake(const char* d, size_t n)
|
|
|
|
{
|
|
|
|
enum llhttp_errno err = llhttp_execute(&_context, d, n);
|
|
|
|
if (err != HPE_OK)
|
|
|
|
{
|
|
|
|
_handshakeFinished = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int on_header_field(llhttp_t* context, const char* at, size_t length)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
thiz->_currentHeader.insert(thiz->_currentHeader.end(), at, at + length);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int on_header_field_complete(llhttp_t* context)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
std::transform(thiz->_currentHeader.begin(), thiz->_currentHeader.end(), thiz->_currentHeader.begin(),
|
|
|
|
::tolower);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int on_header_value(llhttp_t* context, const char* at, size_t length)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
thiz->_currentHeaderValue.insert(thiz->_currentHeaderValue.end(), at, at + length);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int on_header_value_complete(llhttp_t* context)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
thiz->_responseHeaders.emplace(std::move(thiz->_currentHeader), std::move(thiz->_currentHeaderValue));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int on_body(llhttp_t* context, const char* at, size_t length)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
thiz->_responseData.insert(thiz->_responseData.end(), at, at + length);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int on_complete(llhttp_t* context)
|
|
|
|
{
|
|
|
|
auto thiz = (WebSocket*)context->data;
|
|
|
|
thiz->_responseCode = context->status_code;
|
|
|
|
thiz->_handshakeFinished = true;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int on_frame_header(websocket_parser*);
|
|
|
|
static int on_frame_body(websocket_parser*, const char* at, size_t length);
|
|
|
|
static int on_frame_end(websocket_parser*);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
using ResponseHeaderMap = std::multimap<std::string, std::string>;
|
|
|
|
|
|
|
|
Scheduler* _scheduler;
|
|
|
|
|
|
|
|
std::shared_ptr<std::atomic<bool>> _isDestroyed;
|
|
|
|
|
|
|
|
// for close sync
|
|
|
|
std::shared_ptr<std::promise<int>> _syncCloseState;
|
|
|
|
|
|
|
|
std::string _url;
|
|
|
|
Uri _requestUri;
|
|
|
|
std::string _selectedProtocol;
|
|
|
|
std::string _caFilePath;
|
|
|
|
std::string _handshakeSecKey;
|
|
|
|
std::string _verifySecKey;
|
|
|
|
Delegate* _delegate{};
|
|
|
|
yasio::io_service* _service;
|
|
|
|
yasio::transport_handle_t _transport{};
|
|
|
|
yasio::highp_timer* _heartbeatTimer;
|
|
|
|
State _state = State::CLOSED;
|
|
|
|
bool _handshakeFinished = false;
|
|
|
|
|
|
|
|
// for parse handshake response
|
|
|
|
llhttp_t _context;
|
|
|
|
llhttp_settings_t _contextSettings;
|
|
|
|
|
|
|
|
// for parse websocket frame
|
|
|
|
websocket_parser _wsParser;
|
|
|
|
websocket_parser_settings _wsParserSettings;
|
|
|
|
|
|
|
|
enum class FrameState
|
|
|
|
{
|
|
|
|
BEGIN,
|
|
|
|
HEADER,
|
|
|
|
BODY,
|
|
|
|
END,
|
|
|
|
FIN,
|
|
|
|
};
|
|
|
|
FrameState _frameState = FrameState::BEGIN;
|
|
|
|
int _opcode = 0;
|
|
|
|
|
|
|
|
std::string _currentHeader;
|
|
|
|
std::string _currentHeaderValue;
|
|
|
|
ResponseHeaderMap _responseHeaders; /// the returned raw header data. You can also dump it as a string
|
|
|
|
std::string _responseData; /// the handshake repsonse message when fail
|
|
|
|
|
|
|
|
/// the http status code returned from server, e.g. 200, 404, refer to
|
|
|
|
/// https://developer.mozilla.org/en-US/docs/Web/HTTP/Status
|
|
|
|
int _responseCode = -1;
|
|
|
|
|
|
|
|
// for receiveData
|
|
|
|
yasio::sbyte_buffer _receivedData;
|
|
|
|
std::recursive_mutex _receivedDataMtx;
|
|
|
|
|
|
|
|
EventListenerCustom* _resetDirectorListener;
|
|
|
|
|
|
|
|
ConcurrentDeque<Event*> _eventQueue;
|
2024-01-02 19:10:51 +08:00
|
|
|
std::vector<std::string> _headers; /// custom headers
|
|
|
|
std::string _protocols;
|
2023-12-31 22:23:25 +08:00
|
|
|
};
|
|
|
|
} // namespace network
|
|
|
|
|
|
|
|
NS_AX_END
|
|
|
|
#else
|
|
|
|
# include "network/WebSocket-wasm.h"
|
|
|
|
#endif
|