mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-28 05:41:02 +00:00
859 lines
23 KiB
C++
859 lines
23 KiB
C++
|
/****************************************************************************
|
||
|
Copyright (c) 2019 Xiamen Yaji Software Co., Ltd.
|
||
|
|
||
|
http://www.cocos.com
|
||
|
|
||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
of this software and associated engine source code (the "Software"), a limited,
|
||
|
worldwide, royalty-free, non-assignable, revocable and non-exclusive license
|
||
|
to use Cocos Creator solely to develop games on your target platforms. You shall
|
||
|
not use Cocos Creator software for developing other software or tools that's
|
||
|
used for developing games. You are not granted to publish, distribute,
|
||
|
sublicense, and/or sell copies of Cocos Creator.
|
||
|
|
||
|
The software or tools in this License Agreement are licensed, not sold.
|
||
|
Xiamen Yaji Software Co., Ltd. reserves all rights not expressly granted to you.
|
||
|
|
||
|
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.
|
||
|
****************************************************************************/
|
||
|
#include "base/ccConfig.h"
|
||
|
|
||
|
#if (USE_SOCKET > 0) && (USE_WEBSOCKET_SERVER > 0)
|
||
|
|
||
|
#include <iostream>
|
||
|
#include <cassert>
|
||
|
#include <atomic>
|
||
|
|
||
|
#include "WebSocketServer.h"
|
||
|
#include "platform/CCApplication.h"
|
||
|
#include "base/CCScheduler.h"
|
||
|
|
||
|
#define MAX_MSG_PAYLOAD 2048
|
||
|
#define SEND_BUFF 1024
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
std::atomic_int32_t _aliveServer{0}; //debug info
|
||
|
|
||
|
struct lws_protocols protocols[] = {
|
||
|
{
|
||
|
"", //protocol name
|
||
|
cocos2d::network::WebSocketServer::_websocketServerCallback,
|
||
|
sizeof(int),
|
||
|
MAX_MSG_PAYLOAD
|
||
|
},
|
||
|
{
|
||
|
nullptr, nullptr, 0
|
||
|
}
|
||
|
};
|
||
|
|
||
|
const struct lws_extension exts[] = {
|
||
|
{
|
||
|
"permessage-deflate",
|
||
|
lws_extension_callback_pm_deflate,
|
||
|
"permessage-deflate; client_on_context_takeover; client_max_window_bits"
|
||
|
},
|
||
|
{
|
||
|
"deflate-frame",
|
||
|
lws_extension_callback_pm_deflate,
|
||
|
"deflate-frame"
|
||
|
},
|
||
|
{
|
||
|
nullptr, nullptr, nullptr
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct AsyncTaskData {
|
||
|
std::mutex mtx;
|
||
|
std::list<std::function<void()> > tasks;
|
||
|
};
|
||
|
|
||
|
// run in server thread loop
|
||
|
void flush_tasks_in_server_loop_cb(uv_async_t* asyn)
|
||
|
{
|
||
|
AsyncTaskData* data = (AsyncTaskData*)asyn->data;
|
||
|
std::lock_guard<std::mutex> guard(data->mtx);
|
||
|
while (!data->tasks.empty())
|
||
|
{
|
||
|
// fetch task, run task
|
||
|
data->tasks.front()();
|
||
|
// drop task
|
||
|
data->tasks.pop_front();
|
||
|
}
|
||
|
|
||
|
}
|
||
|
void init_libuv_async_handle(uv_loop_t* loop, uv_async_t* async)
|
||
|
{
|
||
|
memset(async, 0, sizeof(uv_async_t));
|
||
|
uv_async_init(loop, async, flush_tasks_in_server_loop_cb);
|
||
|
async->data = new AsyncTaskData();
|
||
|
}
|
||
|
|
||
|
// run in game thread, dispatch runnable object into server loop
|
||
|
void schedule_task_into_server_thread_task_queue(uv_async_t* asyn, std::function<void()> func)
|
||
|
{
|
||
|
|
||
|
AsyncTaskData* data = (AsyncTaskData*)asyn->data;
|
||
|
{
|
||
|
std::lock_guard<std::mutex> guard(data->mtx);
|
||
|
data->tasks.emplace_back(func);
|
||
|
}
|
||
|
//notify server thread to invoke `flush_tasks_in_server_loop_cb()`
|
||
|
uv_async_send(asyn);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
namespace cocos2d {
|
||
|
namespace network {
|
||
|
|
||
|
#define RUN_IN_GAMETHREAD(task) \
|
||
|
do { \
|
||
|
cocos2d::Application::getInstance()->getScheduler()->performFunctionInCocosThread([=]() { \
|
||
|
task; \
|
||
|
});\
|
||
|
} while(0)
|
||
|
|
||
|
#define DISPATCH_CALLBACK_IN_GAMETHREAD() do {\
|
||
|
data->setCallback([callback](const std::string& msg) { \
|
||
|
auto wrapper = [callback, msg]() {callback(msg); }; \
|
||
|
RUN_IN_GAMETHREAD(wrapper()); \
|
||
|
}); \
|
||
|
}while(0)
|
||
|
|
||
|
#define RUN_IN_SERVERTHREAD(task) do { \
|
||
|
schedule_task_into_server_thread_task_queue(&_async, [=](){ \
|
||
|
task; \
|
||
|
}); \
|
||
|
} while(0)
|
||
|
|
||
|
//#define LOGE() CCLOG("WSS: %s", __FUNCTION__)
|
||
|
#define LOGE()
|
||
|
|
||
|
|
||
|
|
||
|
DataFrame::DataFrame(const std::string& data) :_isBinary(false)
|
||
|
{
|
||
|
_underlyingData.resize(data.size() + LWS_PRE);
|
||
|
memcpy(getData(), data.c_str(), data.length());
|
||
|
}
|
||
|
|
||
|
DataFrame::DataFrame(const void* data, int len, bool isBinary) :_isBinary(isBinary)
|
||
|
{
|
||
|
_underlyingData.resize(len + LWS_PRE);
|
||
|
memcpy(getData(), data, len);
|
||
|
}
|
||
|
|
||
|
DataFrame::~DataFrame()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void DataFrame::append(unsigned char* p, int len)
|
||
|
{
|
||
|
_underlyingData.insert(_underlyingData.end(), p, p + len);
|
||
|
}
|
||
|
|
||
|
int DataFrame::slice(unsigned char** p, int len) {
|
||
|
*p = getData() + _consumed;
|
||
|
if (_consumed + len > size()) {
|
||
|
return size() - _consumed;
|
||
|
}
|
||
|
return len;
|
||
|
}
|
||
|
|
||
|
int DataFrame::consume(int len) {
|
||
|
_consumed = len + _consumed > size() ? size() : len + _consumed;
|
||
|
return _consumed;
|
||
|
}
|
||
|
|
||
|
int DataFrame::remain() const {
|
||
|
return size() - _consumed;
|
||
|
}
|
||
|
|
||
|
|
||
|
std::string DataFrame::toString()
|
||
|
{
|
||
|
return std::string((char*)getData(), size());
|
||
|
}
|
||
|
|
||
|
WebSocketServer::WebSocketServer()
|
||
|
{
|
||
|
_aliveServer.fetch_add(1);
|
||
|
}
|
||
|
|
||
|
|
||
|
WebSocketServer::~WebSocketServer()
|
||
|
{
|
||
|
_aliveServer.fetch_sub(1);
|
||
|
destroyContext();
|
||
|
}
|
||
|
|
||
|
bool WebSocketServer::close(std::function<void(const std::string & errorMsg)> callback)
|
||
|
{
|
||
|
_onclose_cb = callback;
|
||
|
if(_ctx)
|
||
|
lws_libuv_stop(_ctx);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::closeAsync(std::function<void(const std::string & errorMsg)> callback)
|
||
|
{
|
||
|
RUN_IN_SERVERTHREAD(this->close(callback));
|
||
|
}
|
||
|
|
||
|
|
||
|
void WebSocketServer::listen(std::shared_ptr<WebSocketServer> server, int port, const std::string& host, std::function<void(const std::string & errorMsg)> callback)
|
||
|
{
|
||
|
auto tryLock = server->_serverLock.try_lock();
|
||
|
if (!tryLock) {
|
||
|
CCLOG("websocketserver is already running!");
|
||
|
if (callback) {
|
||
|
RUN_IN_GAMETHREAD(callback("Error: Server is already running!"));
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
server->_serverState = ServerThreadState::RUNNING;
|
||
|
//lws_set_log_level(-1, nullptr);
|
||
|
|
||
|
if (server->_ctx) {
|
||
|
server->destroyContext();
|
||
|
if (callback) {
|
||
|
RUN_IN_GAMETHREAD(callback("Error: lws_context already created!"));
|
||
|
}
|
||
|
RUN_IN_GAMETHREAD(if (server->_onerror) server->_onerror("websocket listen error!"));
|
||
|
server->_serverState = ServerThreadState::ST_ERROR;
|
||
|
server->_serverLock.unlock();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
server->_host = host;
|
||
|
|
||
|
struct lws_context_creation_info info;
|
||
|
memset(&info, 0, sizeof(info));
|
||
|
info.port = port;
|
||
|
info.iface = server->_host.empty() ? nullptr : server->_host.c_str();
|
||
|
info.protocols = protocols;
|
||
|
info.gid = -1;
|
||
|
info.uid = -1;
|
||
|
info.extensions = exts;
|
||
|
info.options = LWS_SERVER_OPTION_VALIDATE_UTF8
|
||
|
|LWS_SERVER_OPTION_LIBUV
|
||
|
|LWS_SERVER_OPTION_SKIP_SERVER_CANONICAL_NAME
|
||
|
;
|
||
|
info.timeout_secs = 60; //
|
||
|
info.max_http_header_pool = 1;
|
||
|
info.user = server.get();
|
||
|
|
||
|
server->_ctx = lws_create_context(&info);
|
||
|
|
||
|
if (!server->_ctx) {
|
||
|
if (callback) {
|
||
|
RUN_IN_GAMETHREAD(callback("Error: Failed to create lws_context!"));
|
||
|
}
|
||
|
RUN_IN_GAMETHREAD(if (server->_onerror)server->_onerror("websocket listen error!"));
|
||
|
server->_serverState = ServerThreadState::ST_ERROR;
|
||
|
server->_serverLock.unlock();
|
||
|
return;
|
||
|
}
|
||
|
uv_loop_t* loop = nullptr;
|
||
|
if (lws_uv_initloop(server->_ctx, loop, 0)) {
|
||
|
if (callback) {
|
||
|
RUN_IN_GAMETHREAD(callback("Error: Failed to create libuv loop!"));
|
||
|
}
|
||
|
RUN_IN_GAMETHREAD(if (server->_onerror)server->_onerror("websocket listen error, failed to create libuv loop!"));
|
||
|
server->_serverState = ServerThreadState::ST_ERROR;
|
||
|
server->_serverLock.unlock();
|
||
|
server->destroyContext();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
loop = lws_uv_getloop(server->_ctx, 0);
|
||
|
init_libuv_async_handle(loop, &server->_async);
|
||
|
RUN_IN_GAMETHREAD(if(server->_onlistening)server->_onlistening(""));
|
||
|
RUN_IN_GAMETHREAD(if(server->_onbegin)server->_onbegin());
|
||
|
RUN_IN_GAMETHREAD(if(callback)callback(""));
|
||
|
|
||
|
lws_libuv_run(server->_ctx, 0);
|
||
|
uv_close((uv_handle_t*)&server->_async, nullptr);
|
||
|
|
||
|
RUN_IN_GAMETHREAD(if(server->_onclose)server->_onclose(""));
|
||
|
RUN_IN_GAMETHREAD(if(server->_onclose_cb)server->_onclose_cb(""));
|
||
|
RUN_IN_GAMETHREAD(if(server->_onend)server->_onend());
|
||
|
server->_serverState = ServerThreadState::STOPPED;
|
||
|
server->destroyContext();
|
||
|
server->_serverLock.unlock();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::listenAsync(std::shared_ptr<WebSocketServer>& server,int port, const std::string& host, std::function<void(const std::string & errorMsg)> callback)
|
||
|
{
|
||
|
std::thread([=]() {
|
||
|
WebSocketServer::listen(server, port, host, callback);
|
||
|
}).detach();
|
||
|
}
|
||
|
|
||
|
|
||
|
std::vector<std::shared_ptr<WebSocketServerConnection>> WebSocketServer::getConnections() const
|
||
|
{
|
||
|
std::lock_guard<std::mutex> guard(_connsMtx);
|
||
|
std::vector<std::shared_ptr<WebSocketServerConnection> > ret;
|
||
|
for (auto itr : _conns) {
|
||
|
ret.emplace_back(itr.second);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::onCreateClient(struct lws* wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = std::make_shared<WebSocketServerConnection>(wsi);
|
||
|
//char ip[221] = { 0 };
|
||
|
//char addr[221] = { 0 };
|
||
|
//lws_get_peer_addresses(wsi, lws_get_socket_fd(wsi), ip, 220, addr, 220);
|
||
|
//lws_get_peer_simple(wsi, ip, 220);
|
||
|
{
|
||
|
std::lock_guard<std::mutex> guard(_connsMtx);
|
||
|
_conns.emplace(wsi, conn);
|
||
|
}
|
||
|
RUN_IN_GAMETHREAD(if(_onconnection) _onconnection(conn));
|
||
|
conn->onConnected();
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::onDestroyClient(struct lws* wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
if( conn ){
|
||
|
conn->onDestroyClient();
|
||
|
}
|
||
|
std::lock_guard<std::mutex> guard(_connsMtx);
|
||
|
_conns.erase(wsi);
|
||
|
}
|
||
|
void WebSocketServer::onCloseClient(struct lws* wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
if (conn) {
|
||
|
conn->onClientCloseInit();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::onCloseClientInit(struct lws* wsi, void* in, int len)
|
||
|
{
|
||
|
LOGE();
|
||
|
int16_t code;
|
||
|
char* msg = nullptr;
|
||
|
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
|
||
|
if (conn && len > 2) {
|
||
|
code = ntohs(*(int16_t*)in);
|
||
|
msg = (char*)in + sizeof(code);
|
||
|
std::string cp(msg, len - sizeof(code));
|
||
|
conn->onClientCloseInit(code, cp);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::onClientReceive(struct lws* wsi, void* in, int len)
|
||
|
{
|
||
|
LOGE();
|
||
|
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
if (conn)
|
||
|
{
|
||
|
conn->onDataReceive(in, len);
|
||
|
}
|
||
|
}
|
||
|
int WebSocketServer::onServerWritable(struct lws* wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
if (conn)
|
||
|
{
|
||
|
return conn->onDrainData();
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::onClientHTTP(struct lws* wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<WebSocketServerConnection> conn = findConnection(wsi);
|
||
|
if(conn)
|
||
|
{
|
||
|
conn->onHTTP();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
std::shared_ptr<WebSocketServerConnection> WebSocketServer::findConnection(struct lws *wsi) {
|
||
|
|
||
|
std::shared_ptr<WebSocketServerConnection> conn;
|
||
|
{
|
||
|
std::lock_guard<std::mutex> guard(_connsMtx);
|
||
|
auto itr = _conns.find(wsi);
|
||
|
if (itr != _conns.end()) {
|
||
|
conn = itr->second;
|
||
|
}
|
||
|
}
|
||
|
return conn;
|
||
|
}
|
||
|
|
||
|
void WebSocketServer::destroyContext()
|
||
|
{
|
||
|
if (_ctx) {
|
||
|
lws_context_destroy(_ctx);
|
||
|
lws_context_destroy2(_ctx);
|
||
|
_ctx = nullptr;
|
||
|
}
|
||
|
if (_async.data) {
|
||
|
delete (AsyncTaskData*)_async.data;
|
||
|
_async.data = nullptr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebSocketServerConnection::WebSocketServerConnection(struct lws* wsi) : _wsi(wsi)
|
||
|
{
|
||
|
LOGE();
|
||
|
uv_loop_t* loop = lws_uv_getloop(lws_get_context(wsi), 0);
|
||
|
init_libuv_async_handle(loop, &_async);
|
||
|
}
|
||
|
|
||
|
|
||
|
WebSocketServerConnection::~WebSocketServerConnection()
|
||
|
{
|
||
|
LOGE();
|
||
|
if (_async.data) {
|
||
|
delete (AsyncTaskData*)_async.data;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
bool WebSocketServerConnection::send(std::shared_ptr<DataFrame> data)
|
||
|
{
|
||
|
_sendQueue.emplace_back(data);
|
||
|
onDrainData();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
void WebSocketServerConnection::sendTextAsync(const std::string& text, std::function<void(const std::string&)> callback)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<DataFrame> data = std::make_shared<DataFrame>(text);
|
||
|
if (callback) {
|
||
|
DISPATCH_CALLBACK_IN_GAMETHREAD();
|
||
|
}
|
||
|
RUN_IN_SERVERTHREAD(this->send(data));
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::sendBinaryAsync(const void* in, size_t len, std::function<void(const std::string&)> callback)
|
||
|
{
|
||
|
LOGE();
|
||
|
std::shared_ptr<DataFrame> data = std::make_shared<DataFrame>(in, len);
|
||
|
if (callback) {
|
||
|
DISPATCH_CALLBACK_IN_GAMETHREAD();
|
||
|
}
|
||
|
RUN_IN_SERVERTHREAD(this->send(data));
|
||
|
}
|
||
|
|
||
|
bool WebSocketServerConnection::close(int code, std::string message)
|
||
|
{
|
||
|
LOGE();
|
||
|
if (!_wsi) return false;
|
||
|
_readyState = ReadyState::CLOSING;
|
||
|
_closeReason = message;
|
||
|
_closeCode = code;
|
||
|
onClientCloseInit();
|
||
|
//trigger callback to return -1 which indicates connection closed
|
||
|
lws_callback_on_writable(_wsi);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::closeAsync(int code, std::string message)
|
||
|
{
|
||
|
LOGE();
|
||
|
RUN_IN_SERVERTHREAD(this->close(code, message));
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onConnected()
|
||
|
{
|
||
|
_readyState = ReadyState::OPEN;
|
||
|
RUN_IN_GAMETHREAD(if(_onconnect)_onconnect());
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onDataReceive(void* in, int len)
|
||
|
{
|
||
|
LOGE();
|
||
|
|
||
|
bool isFinal = (bool)lws_is_final_fragment(_wsi);
|
||
|
bool isBinary = (bool)lws_frame_is_binary(_wsi);
|
||
|
|
||
|
if (!_prevPkg) {
|
||
|
_prevPkg = std::make_shared<DataFrame>(in, len, isBinary);
|
||
|
}
|
||
|
else {
|
||
|
_prevPkg->append((unsigned char*)in, len);
|
||
|
}
|
||
|
|
||
|
if (isFinal) {
|
||
|
//trigger event
|
||
|
std::shared_ptr<DataFrame> fullpkg = _prevPkg;
|
||
|
if (isBinary)
|
||
|
{
|
||
|
RUN_IN_GAMETHREAD(if(_onbinary)_onbinary(fullpkg));
|
||
|
}
|
||
|
if (!isBinary)
|
||
|
{
|
||
|
RUN_IN_GAMETHREAD(if(_ontext)_ontext(fullpkg));
|
||
|
}
|
||
|
|
||
|
RUN_IN_GAMETHREAD(if(_ondata)_ondata(fullpkg));
|
||
|
|
||
|
_prevPkg.reset();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int WebSocketServerConnection::onDrainData()
|
||
|
{
|
||
|
LOGE();
|
||
|
if (!_wsi) return -1;
|
||
|
if (_closed) return -1;
|
||
|
if (_readyState == ReadyState::CLOSING) {
|
||
|
return -1;
|
||
|
}
|
||
|
if (_readyState != ReadyState::OPEN) return 0;
|
||
|
unsigned char* p = nullptr;
|
||
|
int send_len = 0;
|
||
|
int finish_len = 0;
|
||
|
int flags = 0;
|
||
|
|
||
|
std::vector<char> buff(SEND_BUFF + LWS_PRE);
|
||
|
|
||
|
if (!_sendQueue.empty())
|
||
|
{
|
||
|
std::shared_ptr<DataFrame> frag = _sendQueue.front();
|
||
|
|
||
|
send_len = frag->slice(&p, SEND_BUFF);
|
||
|
|
||
|
if(frag->isFront())
|
||
|
{
|
||
|
if (frag->isBinary())
|
||
|
{
|
||
|
flags |= LWS_WRITE_BINARY;
|
||
|
}
|
||
|
if (frag->isString())
|
||
|
{
|
||
|
flags |= LWS_WRITE_TEXT;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (frag->remain() != send_len)
|
||
|
{
|
||
|
// remain bytes > 0
|
||
|
// not FIN
|
||
|
flags |= LWS_WRITE_NO_FIN;
|
||
|
}
|
||
|
|
||
|
if (!frag->isFront()) {
|
||
|
flags |= LWS_WRITE_CONTINUATION;
|
||
|
}
|
||
|
|
||
|
finish_len = lws_write(_wsi, p, send_len, (lws_write_protocol)flags);
|
||
|
|
||
|
if (finish_len == 0)
|
||
|
{
|
||
|
frag->onFinish("Connection Closed");
|
||
|
return -1;
|
||
|
}
|
||
|
else if (finish_len < 0)
|
||
|
{
|
||
|
frag->onFinish("Send Error!");
|
||
|
return -1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
frag->consume(finish_len);
|
||
|
}
|
||
|
|
||
|
if (frag->remain() == 0) {
|
||
|
frag->onFinish("");
|
||
|
_sendQueue.pop_front();
|
||
|
}
|
||
|
lws_callback_on_writable(_wsi);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onHTTP()
|
||
|
{
|
||
|
if (!_wsi) return;
|
||
|
|
||
|
_headers.clear();
|
||
|
|
||
|
int n = 0, len;
|
||
|
std::vector<char> buf(256);
|
||
|
const char* c;
|
||
|
do {
|
||
|
|
||
|
lws_token_indexes idx = static_cast<lws_token_indexes>(n);
|
||
|
c = (const char*)lws_token_to_string(idx);
|
||
|
if (!c) {
|
||
|
n++;
|
||
|
break;
|
||
|
}
|
||
|
len = lws_hdr_total_length(_wsi, idx);
|
||
|
if (!len) {
|
||
|
n++;
|
||
|
continue;
|
||
|
}
|
||
|
else if (len + 1 > buf.size())
|
||
|
{
|
||
|
buf.resize(len + 1);
|
||
|
}
|
||
|
lws_hdr_copy(_wsi, buf.data(), buf.size(), idx);
|
||
|
buf[len] = '\0';
|
||
|
_headers.emplace(std::string(c), std::string(buf.data()));
|
||
|
n++;
|
||
|
} while (c);
|
||
|
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onClientCloseInit(int code, const std::string& msg)
|
||
|
{
|
||
|
_closeCode = code;
|
||
|
_closeReason = msg;
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onClientCloseInit()
|
||
|
{
|
||
|
if (_closed) return;
|
||
|
lws_close_reason(_wsi, (lws_close_status)_closeCode, (unsigned char*)_closeReason.c_str(), _closeReason.length());
|
||
|
_closed = true;
|
||
|
}
|
||
|
|
||
|
void WebSocketServerConnection::onDestroyClient()
|
||
|
{
|
||
|
_readyState = ReadyState::CLOSED;
|
||
|
//on wsi destroied
|
||
|
if (_wsi)
|
||
|
{
|
||
|
RUN_IN_GAMETHREAD(if(_onclose)_onclose(_closeCode, _closeReason));
|
||
|
RUN_IN_GAMETHREAD(if(_onend) _onend());
|
||
|
uv_close((uv_handle_t*)&_async, nullptr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
std::vector<std::string> WebSocketServerConnection::getProtocols() {
|
||
|
std::vector<std::string> ret;
|
||
|
if (_wsi) {
|
||
|
//TODO cause abort
|
||
|
//const struct lws_protocols* protos = lws_get_protocol(_wsi);
|
||
|
//while (protos && protos->name != nullptr)
|
||
|
//{
|
||
|
// ret.emplace_back(protos->name);
|
||
|
// protos++;
|
||
|
//}
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
std::map<std::string, std::string> WebSocketServerConnection::getHeaders()
|
||
|
{
|
||
|
if (!_wsi) return {};
|
||
|
return _headers;
|
||
|
}
|
||
|
|
||
|
|
||
|
int WebSocketServer::_websocketServerCallback(struct lws* wsi, enum lws_callback_reasons reason,
|
||
|
void* user, void* in, size_t len)
|
||
|
{
|
||
|
int ret = 0;
|
||
|
WebSocketServer* server = nullptr;
|
||
|
lws_context* ctx = nullptr;
|
||
|
|
||
|
if (wsi)
|
||
|
ctx = lws_get_context(wsi);
|
||
|
if (ctx)
|
||
|
server = static_cast<WebSocketServer*>(lws_context_user(ctx));
|
||
|
|
||
|
if (!server) {
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
switch (reason)
|
||
|
{
|
||
|
case LWS_CALLBACK_ESTABLISHED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_CONNECTION_ERROR:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_FILTER_PRE_ESTABLISH:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_ESTABLISHED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLOSED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLOSED_HTTP:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RECEIVE:
|
||
|
server->onClientReceive(wsi, in, len);
|
||
|
break;
|
||
|
case LWS_CALLBACK_RECEIVE_PONG:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_RECEIVE:
|
||
|
server->onClientReceive(wsi, in, len);
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_RECEIVE_PONG:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_WRITEABLE:
|
||
|
//ret = server->onClientWritable(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_SERVER_WRITEABLE:
|
||
|
ret = server->onServerWritable(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_BODY:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_BODY_COMPLETION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_FILE_COMPLETION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_WRITEABLE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_FILTER_NETWORK_CONNECTION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_FILTER_HTTP_CONNECTION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_SERVER_NEW_CLIENT_INSTANTIATED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_FILTER_PROTOCOL_CONNECTION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_OPENSSL_LOAD_EXTRA_CLIENT_VERIFY_CERTS:
|
||
|
break;
|
||
|
case LWS_CALLBACK_OPENSSL_LOAD_EXTRA_SERVER_VERIFY_CERTS:
|
||
|
break;
|
||
|
case LWS_CALLBACK_OPENSSL_PERFORM_CLIENT_CERT_VERIFICATION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_APPEND_HANDSHAKE_HEADER:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CONFIRM_EXTENSION_OKAY:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_CONFIRM_EXTENSION_SUPPORTED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_PROTOCOL_INIT:
|
||
|
break;
|
||
|
case LWS_CALLBACK_PROTOCOL_DESTROY:
|
||
|
break;
|
||
|
case LWS_CALLBACK_WSI_CREATE:
|
||
|
server->onCreateClient(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_WSI_DESTROY:
|
||
|
server->onDestroyClient(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_GET_THREAD_ID:
|
||
|
break;
|
||
|
case LWS_CALLBACK_ADD_POLL_FD:
|
||
|
break;
|
||
|
case LWS_CALLBACK_DEL_POLL_FD:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CHANGE_MODE_POLL_FD:
|
||
|
break;
|
||
|
case LWS_CALLBACK_LOCK_POLL:
|
||
|
break;
|
||
|
case LWS_CALLBACK_UNLOCK_POLL:
|
||
|
break;
|
||
|
case LWS_CALLBACK_OPENSSL_CONTEXT_REQUIRES_PRIVATE_KEY:
|
||
|
break;
|
||
|
case LWS_CALLBACK_WS_PEER_INITIATED_CLOSE:
|
||
|
server->onCloseClientInit(wsi, in, len);
|
||
|
break;
|
||
|
case LWS_CALLBACK_WS_EXT_DEFAULTS:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CGI:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CGI_TERMINATED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CGI_STDIN_DATA:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CGI_STDIN_COMPLETED:
|
||
|
break;
|
||
|
case LWS_CALLBACK_ESTABLISHED_CLIENT_HTTP:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLOSED_CLIENT_HTTP:
|
||
|
server->onCloseClient(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_RECEIVE_CLIENT_HTTP:
|
||
|
break;
|
||
|
case LWS_CALLBACK_COMPLETED_CLIENT_HTTP:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RECEIVE_CLIENT_HTTP_READ:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_BIND_PROTOCOL:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_DROP_PROTOCOL:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CHECK_ACCESS_RIGHTS:
|
||
|
break;
|
||
|
case LWS_CALLBACK_PROCESS_HTML:
|
||
|
break;
|
||
|
case LWS_CALLBACK_ADD_HEADERS:
|
||
|
server->onClientHTTP(wsi);
|
||
|
break;
|
||
|
case LWS_CALLBACK_SESSION_INFO:
|
||
|
break;
|
||
|
case LWS_CALLBACK_GS_EVENT:
|
||
|
break;
|
||
|
case LWS_CALLBACK_HTTP_PMO:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CLIENT_HTTP_WRITEABLE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_OPENSSL_PERFORM_SERVER_CERT_VERIFICATION:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_RX:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_CLOSE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_WRITEABLE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_ADOPT:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_ADOPT_FILE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_RX_FILE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_WRITEABLE_FILE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_RAW_CLOSE_FILE:
|
||
|
break;
|
||
|
case LWS_CALLBACK_SSL_INFO:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CHILD_WRITE_VIA_PARENT:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CHILD_CLOSING:
|
||
|
break;
|
||
|
case LWS_CALLBACK_CGI_PROCESS_ATTACH:
|
||
|
break;
|
||
|
case LWS_CALLBACK_USER:
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
} // namespace network
|
||
|
} // namespace cocos2d
|
||
|
|
||
|
#endif
|