1 Star 0 Fork 0

nhky / swoole-src

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
swoole_websocket_server.cc 35.61 KB
一键复制 编辑 原始数据 按行查看 历史
韩天峰 提交于 2020-09-15 15:35 . Optimize code (#3661)
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921
/*
+----------------------------------------------------------------------+
| Swoole |
+----------------------------------------------------------------------+
| This source file is subject to version 2.0 of the Apache license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.apache.org/licenses/LICENSE-2.0.html |
| If you did not receive a copy of the Apache2.0 license and are unable|
| to obtain it through the world-wide-web, please send a note to |
| license@swoole.com so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
| Author: Tianfeng Han <mikan.tenny@gmail.com> |
+----------------------------------------------------------------------+
*/
#include "php_swoole_http_server.h"
#include <iostream>
SW_EXTERN_C_BEGIN
#include "ext/standard/url.h"
#include "ext/standard/sha1.h"
#include "ext/standard/php_var.h"
#include "ext/standard/php_string.h"
#include "ext/date/php_date.h"
#include "main/php_variables.h"
SW_EXTERN_C_END
#include "swoole_base64.h"
#include "thirdparty/swoole_http_parser.h"
using swoole::Server;
using swoole::Connection;
using swoole::ListenPort;
using swoole::String;
using swoole::coroutine::Socket;
using http_request = swoole::http::Request;
using http_response = swoole::http::Response;
using http_context = swoole::http::Context;
zend_class_entry *swoole_websocket_server_ce;
static zend_object_handlers swoole_websocket_server_handlers;
zend_class_entry *swoole_websocket_frame_ce;
static zend_object_handlers swoole_websocket_frame_handlers;
static zend_class_entry *swoole_websocket_closeframe_ce;
static zend_object_handlers swoole_websocket_closeframe_handlers;
SW_EXTERN_C_BEGIN
static PHP_METHOD(swoole_websocket_server, push);
static PHP_METHOD(swoole_websocket_server, isEstablished);
static PHP_METHOD(swoole_websocket_server, pack);
static PHP_METHOD(swoole_websocket_server, unpack);
static PHP_METHOD(swoole_websocket_server, disconnect);
static PHP_METHOD(swoole_websocket_frame, __toString);
SW_EXTERN_C_END
// clang-format off
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_server_push, 0, 0, 2)
ZEND_ARG_INFO(0, fd)
ZEND_ARG_INFO(0, data)
ZEND_ARG_INFO(0, opcode)
ZEND_ARG_INFO(0, flags)
ZEND_END_ARG_INFO()
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_server_disconnect, 0, 0, 1)
ZEND_ARG_INFO(0, fd)
ZEND_ARG_INFO(0, code)
ZEND_ARG_INFO(0, reason)
ZEND_END_ARG_INFO()
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_server_pack, 0, 0, 1)
ZEND_ARG_INFO(0, data)
ZEND_ARG_INFO(0, opcode)
ZEND_ARG_INFO(0, flags)
ZEND_END_ARG_INFO()
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_server_unpack, 0, 0, 1)
ZEND_ARG_INFO(0, data)
ZEND_END_ARG_INFO()
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_server_isEstablished, 0, 0, 1)
ZEND_ARG_INFO(0, fd)
ZEND_END_ARG_INFO()
ZEND_BEGIN_ARG_INFO_EX(arginfo_swoole_websocket_frame_void, 0, 0, 0)
ZEND_END_ARG_INFO()
const zend_function_entry swoole_websocket_server_methods[] =
{
PHP_ME(swoole_websocket_server, push, arginfo_swoole_websocket_server_push, ZEND_ACC_PUBLIC)
PHP_ME(swoole_websocket_server, disconnect, arginfo_swoole_websocket_server_disconnect, ZEND_ACC_PUBLIC)
PHP_ME(swoole_websocket_server, isEstablished, arginfo_swoole_websocket_server_isEstablished, ZEND_ACC_PUBLIC)
PHP_ME(swoole_websocket_server, pack, arginfo_swoole_websocket_server_pack, ZEND_ACC_PUBLIC | ZEND_ACC_STATIC)
PHP_ME(swoole_websocket_server, unpack, arginfo_swoole_websocket_server_unpack, ZEND_ACC_PUBLIC | ZEND_ACC_STATIC)
PHP_FE_END
};
const zend_function_entry swoole_websocket_frame_methods[] =
{
PHP_ME(swoole_websocket_frame, __toString, arginfo_swoole_websocket_frame_void, ZEND_ACC_PUBLIC)
PHP_ME(swoole_websocket_server, pack, arginfo_swoole_websocket_server_pack, ZEND_ACC_PUBLIC | ZEND_ACC_STATIC)
PHP_ME(swoole_websocket_server, unpack, arginfo_swoole_websocket_server_unpack, ZEND_ACC_PUBLIC | ZEND_ACC_STATIC)
PHP_FE_END
};
// clang-format on
#ifdef SW_HAVE_ZLIB
static bool websocket_message_compress(String *buffer, const char *data, size_t length, int level);
static bool websocket_message_uncompress(String *buffer, const char *in, size_t in_len);
#endif
static void php_swoole_websocket_construct_frame(zval *zframe, zend_long opcode, zval *zpayload, uint8_t flags) {
if (opcode == WEBSOCKET_OPCODE_CLOSE) {
const char *payload = Z_STRVAL_P(zpayload);
size_t payload_length = Z_STRLEN_P(zpayload);
object_init_ex(zframe, swoole_websocket_closeframe_ce);
if (payload_length >= SW_WEBSOCKET_CLOSE_CODE_LEN) {
// WebSocket Close code
zend_update_property_long(swoole_websocket_closeframe_ce,
SW_Z8_OBJ_P(zframe),
ZEND_STRL("code"),
(payload[0] << 8) ^ (payload[1] & 0xFF));
if (payload_length > SW_WEBSOCKET_CLOSE_CODE_LEN) {
// WebSocket Close reason message
zend_update_property_stringl(swoole_websocket_closeframe_ce,
SW_Z8_OBJ_P(zframe),
ZEND_STRL("reason"),
payload + SW_WEBSOCKET_CLOSE_CODE_LEN,
payload_length - SW_WEBSOCKET_CLOSE_CODE_LEN);
}
}
} else {
object_init_ex(zframe, swoole_websocket_frame_ce);
zend_update_property(swoole_websocket_frame_ce, SW_Z8_OBJ_P(zframe), ZEND_STRL("data"), zpayload);
}
zend_update_property_long(swoole_websocket_frame_ce, SW_Z8_OBJ_P(zframe), ZEND_STRL("opcode"), opcode);
zend_update_property_long(swoole_websocket_frame_ce, SW_Z8_OBJ_P(zframe), ZEND_STRL("flags"), flags);
/* BC */
zend_update_property_bool(
swoole_websocket_frame_ce, SW_Z8_OBJ_P(zframe), ZEND_STRL("finish"), flags & SW_WEBSOCKET_FLAG_FIN);
}
void php_swoole_websocket_frame_unpack_ex(String *data, zval *zframe, uchar uncompress) {
swWebSocket_frame frame;
zval zpayload;
uint8_t flags;
if (data->length < sizeof(frame.header)) {
swoole_set_last_error(SW_ERROR_PROTOCOL_ERROR);
ZVAL_FALSE(zframe);
return;
}
swWebSocket_decode(&frame, data);
flags = swWebSocket_get_flags(&frame);
#ifdef SW_HAVE_ZLIB
if (uncompress && frame.header.RSV1) {
swoole_zlib_buffer->clear();
if (!websocket_message_uncompress(swoole_zlib_buffer, frame.payload, frame.payload_length)) {
swoole_set_last_error(SW_ERROR_PROTOCOL_ERROR);
ZVAL_FALSE(zframe);
return;
}
frame.payload = swoole_zlib_buffer->str;
frame.payload_length = swoole_zlib_buffer->length;
flags ^= (SW_WEBSOCKET_FLAG_RSV1 | SW_WEBSOCKET_FLAG_COMPRESS);
}
#endif
/* TODO: optimize memory copy */
ZVAL_STRINGL(&zpayload, frame.payload, frame.payload_length);
php_swoole_websocket_construct_frame(zframe, frame.header.OPCODE, &zpayload, flags);
zval_ptr_dtor(&zpayload);
}
void php_swoole_websocket_frame_unpack(String *data, zval *zframe) {
return php_swoole_websocket_frame_unpack_ex(data, zframe, 0);
}
static sw_inline int php_swoole_websocket_frame_pack_ex(String *buffer,
zval *zdata,
zend_long opcode,
zend_long code,
uint8_t flags,
zend_bool mask,
zend_bool allow_compress) {
char *data = nullptr;
size_t length = 0;
if (sw_unlikely(opcode > SW_WEBSOCKET_OPCODE_MAX)) {
php_swoole_fatal_error(E_WARNING, "the maximum value of opcode is %d", SW_WEBSOCKET_OPCODE_MAX);
return SW_ERR;
}
zend::String str_zdata;
if (zdata && !ZVAL_IS_NULL(zdata)) {
str_zdata = zdata;
data = str_zdata.val();
length = str_zdata.len();
}
if (mask) {
flags |= SW_WEBSOCKET_FLAG_MASK;
}
#ifdef SW_HAVE_ZLIB
if (flags & SW_WEBSOCKET_FLAG_COMPRESS) {
if (!allow_compress) {
flags ^= SW_WEBSOCKET_FLAG_COMPRESS;
} else if (length > 0) {
swoole_zlib_buffer->clear();
if (websocket_message_compress(swoole_zlib_buffer, data, length, Z_DEFAULT_COMPRESSION)) {
data = swoole_zlib_buffer->str;
length = swoole_zlib_buffer->length;
flags |= SW_WEBSOCKET_FLAG_RSV1;
}
}
}
#endif
switch (opcode) {
case WEBSOCKET_OPCODE_CLOSE:
return swWebSocket_pack_close_frame(buffer, code, data, length, flags);
default:
swWebSocket_encode(buffer, data, length, opcode, flags);
}
return SW_OK;
}
int php_swoole_websocket_frame_pack_ex(
String *buffer, zval *zdata, zend_long opcode, uint8_t flags, zend_bool mask, zend_bool allow_compress) {
return php_swoole_websocket_frame_pack_ex(
buffer, zdata, opcode, WEBSOCKET_CLOSE_NORMAL, flags, mask, allow_compress);
}
int php_swoole_websocket_frame_object_pack_ex(String *buffer, zval *zdata, zend_bool mask, zend_bool allow_compress) {
zval *zframe = zdata;
zend_long opcode = WEBSOCKET_OPCODE_TEXT;
zend_long code = WEBSOCKET_CLOSE_NORMAL;
zend_long flags = SW_WEBSOCKET_FLAG_FIN;
zval *ztmp = nullptr;
zdata = nullptr;
if ((ztmp = sw_zend_read_property_ex(swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_OPCODE), 0))) {
opcode = zval_get_long(ztmp);
}
if (opcode == WEBSOCKET_OPCODE_CLOSE) {
if ((ztmp = sw_zend_read_property_not_null_ex(
swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_CODE), 1))) {
code = zval_get_long(ztmp);
}
if ((ztmp = sw_zend_read_property_not_null_ex(
swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_REASON), 1))) {
zdata = ztmp;
}
}
if (!zdata &&
(ztmp = sw_zend_read_property_ex(swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_DATA), 0))) {
zdata = ztmp;
}
if ((ztmp = sw_zend_read_property_ex(swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_FLAGS), 0))) {
flags = zval_get_long(ztmp) & SW_WEBSOCKET_FLAGS_ALL;
}
if ((ztmp = sw_zend_read_property_not_null_ex(
swoole_websocket_frame_ce, zframe, SW_ZSTR_KNOWN(SW_ZEND_STR_FINISH), 0))) {
if (zval_is_true(ztmp)) {
flags |= SW_WEBSOCKET_FLAG_FIN;
} else {
flags &= ~SW_WEBSOCKET_FLAG_FIN;
}
}
return php_swoole_websocket_frame_pack_ex(
buffer, zdata, opcode, code, flags & SW_WEBSOCKET_FLAGS_ALL, mask, allow_compress);
}
void swoole_websocket_onOpen(Server *serv, http_context *ctx) {
Connection *conn = serv->get_connection_by_session_id(ctx->fd);
if (!conn) {
swoole_error_log(SW_LOG_NOTICE, SW_ERROR_SESSION_CLOSED, "session[%d] is closed", ctx->fd);
return;
}
zend_fcall_info_cache *fci_cache = php_swoole_server_get_fci_cache(serv, conn->server_fd, SW_SERVER_CB_onOpen);
if (fci_cache) {
zval args[2];
args[0] = *((zval *) serv->ptr2);
args[1] = *ctx->request.zobject;
if (UNEXPECTED(!zend::function::call(fci_cache, 2, args, nullptr, SwooleG.enable_coroutine))) {
php_swoole_error(E_WARNING, "%s->onOpen handler error", ZSTR_VAL(swoole_websocket_server_ce->name));
serv->close(ctx->fd, 0);
}
}
}
/**
* default onRequest callback
*/
void swoole_websocket_onRequest(http_context *ctx) {
const char *bad_request = "HTTP/1.1 400 Bad Request\r\n"
"Connection: close\r\n"
"Content-Type: text/html; charset=UTF-8\r\n"
"Cache-Control: must-revalidate,no-cache,no-store\r\n"
"Content-Length: 83\r\n"
"Server: " SW_HTTP_SERVER_SOFTWARE "\r\n\r\n"
"<html><body><h2>HTTP 400 Bad Request</h2><hr><i>Powered by Swoole</i></body></html>";
ctx->send(ctx, (char *) bad_request, strlen(bad_request));
ctx->end = 1;
ctx->close(ctx);
}
void php_swoole_sha1(const char *str, int _len, unsigned char *digest) {
PHP_SHA1_CTX context;
PHP_SHA1Init(&context);
PHP_SHA1Update(&context, (unsigned char *) str, _len);
PHP_SHA1Final(digest, &context);
}
bool swoole_websocket_handshake(http_context *ctx) {
char sec_buf[128];
zval *header = ctx->request.zheader;
HashTable *ht = Z_ARRVAL_P(header);
zval *pData;
zval retval;
if (!(pData = zend_hash_str_find(ht, ZEND_STRL("sec-websocket-key")))) {
_bad_request:
ctx->response.status = SW_HTTP_BAD_REQUEST;
swoole_http_response_end(ctx, nullptr, &retval);
return false;
}
zend::String str_pData(pData);
if (str_pData.len() != BASE64_ENCODE_OUT_SIZE(SW_WEBSOCKET_SEC_KEY_LEN)) {
goto _bad_request;
}
char sha1_str[20];
// sec_websocket_accept
memcpy(sec_buf, str_pData.val(), str_pData.len());
memcpy(sec_buf + str_pData.len(), SW_WEBSOCKET_GUID, sizeof(SW_WEBSOCKET_GUID) - 1);
// sha1 sec_websocket_accept
php_swoole_sha1(sec_buf, str_pData.len() + sizeof(SW_WEBSOCKET_GUID) - 1, (unsigned char *) sha1_str);
// base64 encode
int sec_len = swBase64_encode((unsigned char *) sha1_str, sizeof(sha1_str), sec_buf);
swoole_http_response_set_header(ctx, ZEND_STRL("Upgrade"), ZEND_STRL("websocket"), false);
swoole_http_response_set_header(ctx, ZEND_STRL("Connection"), ZEND_STRL("Upgrade"), false);
swoole_http_response_set_header(ctx, ZEND_STRL("Sec-WebSocket-Accept"), sec_buf, sec_len, false);
swoole_http_response_set_header(ctx, ZEND_STRL("Sec-WebSocket-Version"), ZEND_STRL(SW_WEBSOCKET_VERSION), false);
#ifdef SW_HAVE_ZLIB
bool enable_websocket_compression = true;
bool websocket_compression = false;
#endif
Server *serv = nullptr;
Connection *conn = nullptr;
if (!ctx->co_socket) {
serv = (Server *) ctx->private_data;
conn = serv->get_connection_by_session_id(ctx->fd);
if (!conn) {
swoole_error_log(SW_LOG_NOTICE, SW_ERROR_SESSION_CLOSED, "session[%d] is closed", ctx->fd);
return false;
}
#ifdef SW_HAVE_ZLIB
enable_websocket_compression = serv->websocket_compression;
#endif
}
#ifdef SW_HAVE_ZLIB
else {
enable_websocket_compression = ctx->websocket_compression;
}
#endif
#ifdef SW_HAVE_ZLIB
if (enable_websocket_compression && (pData = zend_hash_str_find(ht, ZEND_STRL("sec-websocket-extensions"))) &&
Z_TYPE_P(pData) == IS_STRING) {
std::string value(Z_STRVAL_P(pData), Z_STRLEN_P(pData));
if (value.substr(0, value.find_first_of(';')) == "permessage-deflate") {
websocket_compression = true;
swoole_http_response_set_header(
ctx, ZEND_STRL("Sec-Websocket-Extensions"), ZEND_STRL(SW_WEBSOCKET_EXTENSION_DEFLATE), false);
}
}
#endif
if (conn) {
conn->websocket_status = WEBSOCKET_STATUS_ACTIVE;
ListenPort *port = serv->get_port_by_server_fd(conn->server_fd);
if (port && !port->websocket_subprotocol.empty()) {
swoole_http_response_set_header(ctx,
ZEND_STRL("Sec-WebSocket-Protocol"),
port->websocket_subprotocol.c_str(),
port->websocket_subprotocol.length(),
false);
}
#ifdef SW_HAVE_ZLIB
ctx->websocket_compression = conn->websocket_compression = websocket_compression;
#endif
} else {
Socket *sock = (Socket *) ctx->private_data;
sock->open_length_check = 1;
sock->protocol.package_length_size = SW_WEBSOCKET_HEADER_LEN;
sock->protocol.package_length_offset = 0;
sock->protocol.package_body_offset = 0;
sock->protocol.get_package_length = swWebSocket_get_package_length;
#ifdef SW_HAVE_ZLIB
ctx->websocket_compression = websocket_compression;
#endif
}
ctx->response.status = SW_HTTP_SWITCHING_PROTOCOLS;
ctx->upgrade = 1;
swoole_http_response_end(ctx, nullptr, &retval);
return Z_TYPE(retval) == IS_TRUE;
}
#ifdef SW_HAVE_ZLIB
static bool websocket_message_uncompress(String *buffer, const char *in, size_t in_len) {
z_stream zstream;
int status;
bool ret = false;
memset(&zstream, 0, sizeof(zstream));
zstream.zalloc = php_zlib_alloc;
zstream.zfree = php_zlib_free;
// gzip_stream.total_out = 0;
status = inflateInit2(&zstream, SW_ZLIB_ENCODING_RAW);
if (status != Z_OK) {
swWarn("inflateInit2() failed by %s", zError(status));
return false;
}
zstream.next_in = (Bytef *) in;
zstream.avail_in = in_len;
zstream.total_in = 0;
while (1) {
zstream.avail_out = buffer->size - buffer->length;
zstream.next_out = (Bytef *) (buffer->str + buffer->length);
status = inflate(&zstream, Z_SYNC_FLUSH);
if (status >= 0) {
buffer->length = zstream.total_out;
}
if (status == Z_STREAM_END || (status == Z_OK && zstream.avail_in == 0)) {
ret = true;
break;
}
if (status != Z_OK) {
break;
}
if (buffer->length + (SW_BUFFER_SIZE_STD / 2) >= buffer->size) {
if (!buffer->extend()) {
status = Z_MEM_ERROR;
break;
}
}
}
inflateEnd(&zstream);
if (!ret) {
swWarn("inflate() failed, Error: %s[%d]", zError(status), status);
return false;
}
return true;
}
static bool websocket_message_compress(String *buffer, const char *data, size_t length, int level) {
// ==== ZLIB ====
if (level == Z_NO_COMPRESSION) {
level = Z_DEFAULT_COMPRESSION;
} else if (level > Z_BEST_COMPRESSION) {
level = Z_BEST_COMPRESSION;
}
z_stream zstream = {};
int status;
zstream.zalloc = php_zlib_alloc;
zstream.zfree = php_zlib_free;
status = deflateInit2(&zstream, level, Z_DEFLATED, SW_ZLIB_ENCODING_RAW, MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY);
if (status != Z_OK) {
swWarn("deflateInit2() failed, Error: [%d]", status);
return false;
}
zstream.next_in = (Bytef *) data;
zstream.avail_in = length;
zstream.next_out = (Bytef *) buffer->str;
size_t max_length = deflateBound(&zstream, length);
if (max_length > buffer->size) {
if (!buffer->extend(max_length)) {
return false;
}
}
size_t bytes_written = 0;
uchar in_sync_flush;
int result;
do {
size_t write_remaining;
if (zstream.avail_out == 0) {
size_t write_position;
zstream.avail_out = max_length;
write_position = buffer->length;
buffer->length = max_length;
zstream.next_out = (Bytef *) buffer->str + write_position;
/* Use a fixed value for buffer increments */
max_length = 4096;
}
write_remaining = buffer->length - bytes_written;
in_sync_flush = zstream.avail_in == 0;
result = deflate(&zstream, in_sync_flush ? Z_SYNC_FLUSH : Z_NO_FLUSH);
bytes_written += write_remaining - zstream.avail_out;
} while (result == Z_OK);
deflateEnd(&zstream);
if (result != Z_BUF_ERROR || bytes_written < 4) {
swWarn("Failed to compress outgoing frame");
return false;
}
if (status != Z_OK) {
swWarn("deflate() failed, Error: [%d]", status);
return false;
}
buffer->length = bytes_written - 4;
return true;
}
#endif
int swoole_websocket_onMessage(Server *serv, swRecvData *req) {
int fd = req->info.fd;
uchar flags = 0;
zend_long opcode = 0;
auto port = serv->get_port_by_session_id(fd);
if (!port) {
return SW_ERR;
}
zval zdata;
char frame_header[2];
memcpy(frame_header, &req->info.ext_flags, sizeof(frame_header));
php_swoole_get_recv_data(serv, &zdata, req);
// frame info has already decoded in swWebSocket_dispatch_frame
flags = frame_header[0];
opcode = frame_header[1];
if ((opcode == WEBSOCKET_OPCODE_CLOSE && !port->open_websocket_close_frame) ||
(opcode == WEBSOCKET_OPCODE_PING && !port->open_websocket_ping_frame) ||
(opcode == WEBSOCKET_OPCODE_PONG && !port->open_websocket_pong_frame)) {
if (opcode == WEBSOCKET_OPCODE_PING) {
String send_frame = {};
char buf[SW_WEBSOCKET_HEADER_LEN + SW_WEBSOCKET_CLOSE_CODE_LEN + SW_WEBSOCKET_CLOSE_REASON_MAX_LEN];
send_frame.str = buf;
send_frame.size = sizeof(buf);
swWebSocket_encode(&send_frame, req->data, req->info.len, WEBSOCKET_OPCODE_PONG, SW_WEBSOCKET_FLAG_FIN);
serv->send(fd, send_frame.str, send_frame.length);
}
zval_ptr_dtor(&zdata);
return SW_OK;
}
#ifdef SW_HAVE_ZLIB
/**
* RFC 7692
*/
if (serv->websocket_compression && (flags & SW_WEBSOCKET_FLAG_RSV1)) {
swoole_zlib_buffer->clear();
if (!websocket_message_uncompress(swoole_zlib_buffer, Z_STRVAL(zdata), Z_STRLEN(zdata))) {
zval_ptr_dtor(&zdata);
return SW_OK;
}
zval_ptr_dtor(&zdata);
ZVAL_STRINGL(&zdata, swoole_zlib_buffer->str, swoole_zlib_buffer->length);
flags ^= (SW_WEBSOCKET_FLAG_RSV1 | SW_WEBSOCKET_FLAG_COMPRESS);
}
#endif
zend_fcall_info_cache *fci_cache =
php_swoole_server_get_fci_cache(serv, req->info.server_fd, SW_SERVER_CB_onMessage);
zval args[2];
args[0] = *(zval *) serv->ptr2;
php_swoole_websocket_construct_frame(&args[1], opcode, &zdata, flags);
zend_update_property_long(swoole_websocket_frame_ce, SW_Z8_OBJ_P(&args[1]), ZEND_STRL("fd"), fd);
if (UNEXPECTED(!zend::function::call(fci_cache, 2, args, nullptr, SwooleG.enable_coroutine))) {
php_swoole_error(E_WARNING, "%s->onMessage handler error", ZSTR_VAL(swoole_websocket_server_ce->name));
serv->close(fd, false);
}
zval_ptr_dtor(&zdata);
zval_ptr_dtor(&args[1]);
return SW_OK;
}
int swoole_websocket_onHandshake(Server *serv, swListenPort *port, http_context *ctx) {
int fd = ctx->fd;
bool success = swoole_websocket_handshake(ctx);
if (success) {
swoole_websocket_onOpen(serv, ctx);
} else {
serv->close(fd, true);
}
return SW_OK;
}
void php_swoole_websocket_server_minit(int module_number) {
SW_INIT_CLASS_ENTRY_EX(swoole_websocket_server,
"Swoole\\WebSocket\\Server",
"swoole_websocket_server",
nullptr,
swoole_websocket_server_methods,
swoole_http_server);
SW_SET_CLASS_SERIALIZABLE(swoole_websocket_server, zend_class_serialize_deny, zend_class_unserialize_deny);
SW_SET_CLASS_CLONEABLE(swoole_websocket_server, sw_zend_class_clone_deny);
SW_SET_CLASS_UNSET_PROPERTY_HANDLER(swoole_websocket_server, sw_zend_class_unset_property_deny);
SW_INIT_CLASS_ENTRY(swoole_websocket_frame,
"Swoole\\WebSocket\\Frame",
"swoole_websocket_frame",
nullptr,
swoole_websocket_frame_methods);
zend_declare_property_long(swoole_websocket_frame_ce, ZEND_STRL("fd"), 0, ZEND_ACC_PUBLIC);
zend_declare_property_string(swoole_websocket_frame_ce, ZEND_STRL("data"), "", ZEND_ACC_PUBLIC);
zend_declare_property_long(swoole_websocket_frame_ce, ZEND_STRL("opcode"), WEBSOCKET_OPCODE_TEXT, ZEND_ACC_PUBLIC);
zend_declare_property_long(swoole_websocket_frame_ce, ZEND_STRL("flags"), SW_WEBSOCKET_FLAG_FIN, ZEND_ACC_PUBLIC);
zend_declare_property_null(swoole_websocket_frame_ce, ZEND_STRL("finish"), ZEND_ACC_PUBLIC);
SW_INIT_CLASS_ENTRY_EX(swoole_websocket_closeframe,
"Swoole\\WebSocket\\CloseFrame",
"swoole_websocket_closeframe",
nullptr,
nullptr,
swoole_websocket_frame);
zend_declare_property_long(
swoole_websocket_closeframe_ce, ZEND_STRL("opcode"), WEBSOCKET_OPCODE_CLOSE, ZEND_ACC_PUBLIC);
zend_declare_property_long(
swoole_websocket_closeframe_ce, ZEND_STRL("code"), WEBSOCKET_CLOSE_NORMAL, ZEND_ACC_PUBLIC);
zend_declare_property_string(swoole_websocket_closeframe_ce, ZEND_STRL("reason"), "", ZEND_ACC_PUBLIC);
/* {{{ swoole namespace */
// status
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_STATUS_CONNECTION", WEBSOCKET_STATUS_CONNECTION);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_STATUS_HANDSHAKE", WEBSOCKET_STATUS_HANDSHAKE);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_STATUS_ACTIVE", WEBSOCKET_STATUS_ACTIVE);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_STATUS_CLOSING", WEBSOCKET_STATUS_CLOSING);
// all opcodes
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_CONTINUATION", WEBSOCKET_OPCODE_CONTINUATION);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_TEXT", WEBSOCKET_OPCODE_TEXT);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_BINARY", WEBSOCKET_OPCODE_BINARY);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_CLOSE", WEBSOCKET_OPCODE_CLOSE);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_PING", WEBSOCKET_OPCODE_PING);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_OPCODE_PONG", WEBSOCKET_OPCODE_PONG);
// flags
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_FIN", SW_WEBSOCKET_FLAG_FIN);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_RSV1", SW_WEBSOCKET_FLAG_RSV1);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_RSV2", SW_WEBSOCKET_FLAG_RSV2);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_RSV3", SW_WEBSOCKET_FLAG_RSV3);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_MASK", SW_WEBSOCKET_FLAG_MASK);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_FLAG_COMPRESS", SW_WEBSOCKET_FLAG_COMPRESS);
// close error
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_NORMAL", WEBSOCKET_CLOSE_NORMAL);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_GOING_AWAY", WEBSOCKET_CLOSE_GOING_AWAY);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_PROTOCOL_ERROR", WEBSOCKET_CLOSE_PROTOCOL_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_DATA_ERROR", WEBSOCKET_CLOSE_DATA_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_STATUS_ERROR", WEBSOCKET_CLOSE_STATUS_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_ABNORMAL", WEBSOCKET_CLOSE_ABNORMAL);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_MESSAGE_ERROR", WEBSOCKET_CLOSE_MESSAGE_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_POLICY_ERROR", WEBSOCKET_CLOSE_POLICY_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_MESSAGE_TOO_BIG", WEBSOCKET_CLOSE_MESSAGE_TOO_BIG);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_EXTENSION_MISSING", WEBSOCKET_CLOSE_EXTENSION_MISSING);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_SERVER_ERROR", WEBSOCKET_CLOSE_SERVER_ERROR);
SW_REGISTER_LONG_CONSTANT("SWOOLE_WEBSOCKET_CLOSE_TLS", WEBSOCKET_CLOSE_TLS);
/* swoole namespace }}} */
/* BC */
// status
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_STATUS_CONNECTION", WEBSOCKET_STATUS_CONNECTION);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_STATUS_HANDSHAKE", WEBSOCKET_STATUS_HANDSHAKE);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_STATUS_FRAME", WEBSOCKET_STATUS_ACTIVE);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_STATUS_ACTIVE", WEBSOCKET_STATUS_ACTIVE);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_STATUS_CLOSING", WEBSOCKET_STATUS_CLOSING);
// all opcodes
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_CONTINUATION", WEBSOCKET_OPCODE_CONTINUATION);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_TEXT", WEBSOCKET_OPCODE_TEXT);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_BINARY", WEBSOCKET_OPCODE_BINARY);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_CLOSE", WEBSOCKET_OPCODE_CLOSE);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_PING", WEBSOCKET_OPCODE_PING);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_OPCODE_PONG", WEBSOCKET_OPCODE_PONG);
// close error
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_NORMAL", WEBSOCKET_CLOSE_NORMAL);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_GOING_AWAY", WEBSOCKET_CLOSE_GOING_AWAY);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_PROTOCOL_ERROR", WEBSOCKET_CLOSE_PROTOCOL_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_DATA_ERROR", WEBSOCKET_CLOSE_DATA_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_STATUS_ERROR", WEBSOCKET_CLOSE_STATUS_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_ABNORMAL", WEBSOCKET_CLOSE_ABNORMAL);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_MESSAGE_ERROR", WEBSOCKET_CLOSE_MESSAGE_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_POLICY_ERROR", WEBSOCKET_CLOSE_POLICY_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_MESSAGE_TOO_BIG", WEBSOCKET_CLOSE_MESSAGE_TOO_BIG);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_EXTENSION_MISSING", WEBSOCKET_CLOSE_EXTENSION_MISSING);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_SERVER_ERROR", WEBSOCKET_CLOSE_SERVER_ERROR);
SW_REGISTER_LONG_CONSTANT("WEBSOCKET_CLOSE_TLS", WEBSOCKET_CLOSE_TLS);
}
static sw_inline bool swoole_websocket_server_push(Server *serv, int fd, String *buffer) {
if (sw_unlikely(fd <= 0)) {
php_swoole_fatal_error(E_WARNING, "fd[%d] is invalid", fd);
return false;
}
Connection *conn = serv->get_connection_by_session_id(fd);
if (!conn || conn->websocket_status < WEBSOCKET_STATUS_HANDSHAKE) {
swoole_set_last_error(SW_ERROR_WEBSOCKET_UNCONNECTED);
php_swoole_fatal_error(
E_WARNING, "the connected client of connection[%d] is not a websocket client or closed", (int) fd);
return false;
}
bool ret = serv->send(fd, buffer->str, buffer->length);
if (!ret && swoole_get_last_error() == SW_ERROR_OUTPUT_SEND_YIELD) {
zval _return_value;
zval *return_value = &_return_value;
zval _yield_data;
ZVAL_STRINGL(&_yield_data, buffer->str, buffer->length);
ZVAL_FALSE(return_value);
php_swoole_server_send_yield(serv, fd, &_yield_data, return_value);
ret = Z_BVAL_P(return_value);
}
return ret;
}
static sw_inline bool swoole_websocket_server_close(Server *serv, int fd, String *buffer, bool real_close) {
bool ret = swoole_websocket_server_push(serv, fd, buffer);
if (!ret || !real_close) {
return ret;
}
Connection *conn = serv->get_connection_by_session_id(fd);
if (conn) {
// Change status immediately to avoid double close
conn->websocket_status = WEBSOCKET_STATUS_CLOSING;
// Server close connection immediately
return serv->close(fd, false);
} else {
return false;
}
}
static PHP_METHOD(swoole_websocket_server, disconnect) {
Server *serv = php_swoole_server_get_and_check_server(ZEND_THIS);
if (sw_unlikely(!serv->is_started())) {
php_swoole_fatal_error(E_WARNING, "server is not running");
RETURN_FALSE;
}
zend_long fd = 0;
zend_long code = WEBSOCKET_CLOSE_NORMAL;
char *data = nullptr;
size_t length = 0;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "l|ls", &fd, &code, &data, &length) == FAILURE) {
RETURN_FALSE;
}
swoole_http_buffer->clear();
if (swWebSocket_pack_close_frame(swoole_http_buffer, code, data, length, 0) < 0) {
RETURN_FALSE;
}
RETURN_BOOL(swoole_websocket_server_close(serv, fd, swoole_http_buffer, 1));
}
static PHP_METHOD(swoole_websocket_server, push) {
Server *serv = php_swoole_server_get_and_check_server(ZEND_THIS);
if (sw_unlikely(!serv->is_started())) {
php_swoole_fatal_error(E_WARNING, "server is not running");
RETURN_FALSE;
}
zend_long fd = 0;
zval *zdata = nullptr;
zend_long opcode = WEBSOCKET_OPCODE_TEXT;
zval *zflags = nullptr;
zend_long flags = SW_WEBSOCKET_FLAG_FIN;
#ifdef SW_HAVE_ZLIB
zend_bool allow_compress = 0;
#endif
ZEND_PARSE_PARAMETERS_START(2, 4)
Z_PARAM_LONG(fd)
Z_PARAM_ZVAL(zdata)
Z_PARAM_OPTIONAL
Z_PARAM_LONG(opcode)
Z_PARAM_ZVAL_EX(zflags, 1, 0)
ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
if (zflags != nullptr) {
flags = zval_get_long(zflags);
}
#ifdef SW_HAVE_ZLIB
Connection *conn = serv->get_connection_verify(fd);
if (!conn) {
RETURN_FALSE;
}
allow_compress = conn->websocket_compression;
#endif
swoole_http_buffer->clear();
if (php_swoole_websocket_frame_is_object(zdata)) {
if (php_swoole_websocket_frame_object_pack(swoole_http_buffer, zdata, 0, allow_compress) < 0) {
RETURN_FALSE;
}
} else {
if (php_swoole_websocket_frame_pack(
swoole_http_buffer, zdata, opcode, flags & SW_WEBSOCKET_FLAGS_ALL, 0, allow_compress) < 0) {
RETURN_FALSE;
}
}
switch (opcode) {
case WEBSOCKET_OPCODE_CLOSE:
RETURN_BOOL(swoole_websocket_server_close(serv, fd, swoole_http_buffer, flags & SW_WEBSOCKET_FLAG_FIN));
break;
default:
RETURN_BOOL(swoole_websocket_server_push(serv, fd, swoole_http_buffer));
}
}
static PHP_METHOD(swoole_websocket_server, pack) {
String *buffer = SwooleTG.buffer_stack;
zval *zdata;
zend_long opcode = WEBSOCKET_OPCODE_TEXT;
zval *zflags = nullptr;
zend_long flags = SW_WEBSOCKET_FLAG_FIN;
ZEND_PARSE_PARAMETERS_START(1, 3)
Z_PARAM_ZVAL(zdata)
Z_PARAM_OPTIONAL
Z_PARAM_LONG(opcode)
Z_PARAM_ZVAL_EX(zflags, 1, 0)
ZEND_PARSE_PARAMETERS_END_EX(RETURN_FALSE);
if (zflags != nullptr) {
flags = zval_get_long(zflags);
}
buffer->clear();
if (php_swoole_websocket_frame_is_object(zdata)) {
if (php_swoole_websocket_frame_object_pack(buffer, zdata, 0, 1) < 0) {
RETURN_EMPTY_STRING();
}
} else {
if (php_swoole_websocket_frame_pack(buffer, zdata, opcode, flags & SW_WEBSOCKET_FLAGS_ALL, 0, 1) < 0) {
RETURN_EMPTY_STRING();
}
}
RETURN_STRINGL(buffer->str, buffer->length);
}
static PHP_METHOD(swoole_websocket_frame, __toString) {
String *buffer = SwooleTG.buffer_stack;
buffer->clear();
if (php_swoole_websocket_frame_object_pack(buffer, ZEND_THIS, 0, 1) < 0) {
RETURN_EMPTY_STRING();
}
RETURN_STRINGL(buffer->str, buffer->length);
}
static PHP_METHOD(swoole_websocket_server, unpack) {
String buffer = {};
if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &buffer.str, &buffer.length) == FAILURE) {
RETURN_FALSE;
}
php_swoole_websocket_frame_unpack(&buffer, return_value);
}
static PHP_METHOD(swoole_websocket_server, isEstablished) {
Server *serv = php_swoole_server_get_and_check_server(ZEND_THIS);
if (sw_unlikely(!serv->is_started())) {
php_swoole_fatal_error(E_WARNING, "server is not running");
RETURN_FALSE;
}
zend_long fd;
if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &fd) == FAILURE) {
RETURN_FALSE;
}
Connection *conn = serv->get_connection_by_session_id(fd);
// not isEstablished
if (!conn || conn->active == 0 || conn->closed || conn->websocket_status < WEBSOCKET_STATUS_ACTIVE) {
RETURN_FALSE;
} else {
RETURN_TRUE;
}
}
1
https://gitee.com/nhky/swoole-src.git
git@gitee.com:nhky/swoole-src.git
nhky
swoole-src
swoole-src
master

搜索帮助

53164aa7 5694891 3bd8fe86 5694891