182 lines
5.9 KiB
C++
182 lines
5.9 KiB
C++
/*
|
||
* Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
|
||
*
|
||
* This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
|
||
*
|
||
* Use of this source code is governed by MIT-like license that can be found in the
|
||
* LICENSE file in the root of the source tree. All contributing project authors
|
||
* may be found in the AUTHORS file in the root of the source tree.
|
||
*/
|
||
|
||
#ifndef ZLMEDIAKIT_WEBSOCKETSPLITTER_H
|
||
#define ZLMEDIAKIT_WEBSOCKETSPLITTER_H
|
||
|
||
#include <cstdint>
|
||
#include <string>
|
||
#include <vector>
|
||
#include <memory>
|
||
#include "Network/Buffer.h"
|
||
|
||
// websocket组合包最大不得超过4MB(防止内存爆炸) [AUTO-TRANSLATED:99c11a1d]
|
||
// websocket combined package size must not exceed 4MB (to prevent memory explosion)
|
||
#define MAX_WS_PACKET (4 * 1024 * 1024)
|
||
|
||
namespace mediakit {
|
||
|
||
class WebSocketHeader {
|
||
public:
|
||
using Ptr = std::shared_ptr<WebSocketHeader>;
|
||
typedef enum {
|
||
CONTINUATION = 0x0,
|
||
TEXT = 0x1,
|
||
BINARY = 0x2,
|
||
RSV3 = 0x3,
|
||
RSV4 = 0x4,
|
||
RSV5 = 0x5,
|
||
RSV6 = 0x6,
|
||
RSV7 = 0x7,
|
||
CLOSE = 0x8,
|
||
PING = 0x9,
|
||
PONG = 0xA,
|
||
CONTROL_RSVB = 0xB,
|
||
CONTROL_RSVC = 0xC,
|
||
CONTROL_RSVD = 0xD,
|
||
CONTROL_RSVE = 0xE,
|
||
CONTROL_RSVF = 0xF
|
||
} Type;
|
||
public:
|
||
|
||
WebSocketHeader() : _mask(4){
|
||
// 获取_mask内部buffer的内存地址,该内存是malloc开辟的,地址为随机 [AUTO-TRANSLATED:9406f0b6]
|
||
// Get the memory address of the internal buffer of _mask, the memory is allocated by malloc, and the address is random
|
||
uint64_t ptr = (uint64_t)(&_mask[0]);
|
||
// 根据内存地址设置掩码随机数 [AUTO-TRANSLATED:47881295]
|
||
// Set the mask random number according to the memory address
|
||
_mask.assign((uint8_t*)(&ptr), (uint8_t*)(&ptr) + 4);
|
||
}
|
||
|
||
virtual ~WebSocketHeader() = default;
|
||
|
||
public:
|
||
bool _fin;
|
||
uint8_t _reserved;
|
||
Type _opcode;
|
||
bool _mask_flag;
|
||
size_t _payload_len;
|
||
std::vector<uint8_t > _mask;
|
||
};
|
||
|
||
// websocket协议收到的字符串类型缓存,用户协议层获取该数据传输的方式 [AUTO-TRANSLATED:a66e0177]
|
||
// String type cache received by the websocket protocol, the way the user protocol layer obtains this data transmission
|
||
class WebSocketBuffer : public toolkit::BufferString {
|
||
public:
|
||
using Ptr = std::shared_ptr<WebSocketBuffer>;
|
||
|
||
template<typename ...ARGS>
|
||
WebSocketBuffer(WebSocketHeader::Type headType, bool fin, ARGS &&...args)
|
||
: toolkit::BufferString(std::forward<ARGS>(args)...), _fin(fin), _head_type(headType){}
|
||
|
||
WebSocketHeader::Type headType() const { return _head_type; }
|
||
|
||
bool isFinished() const { return _fin; };
|
||
|
||
private:
|
||
bool _fin;
|
||
WebSocketHeader::Type _head_type;
|
||
};
|
||
|
||
class WebSocketSplitter : public WebSocketHeader{
|
||
public:
|
||
/**
|
||
* 输入数据以便解包webSocket数据以及处理粘包问题
|
||
* 可能触发onWebSocketDecodeHeader和onWebSocketDecodePayload回调
|
||
* @param data 需要解包的数据,可能是不完整的包或多个包
|
||
* @param len 数据长度
|
||
* Input data to unpack webSocket data and handle sticky packet problems
|
||
* May trigger onWebSocketDecodeHeader and onWebSocketDecodePayload callbacks
|
||
* @param data Data to be unpacked, may be incomplete packets or multiple packets
|
||
* @param len Data length
|
||
|
||
* [AUTO-TRANSLATED:e5f2c2c6]
|
||
*/
|
||
void decode(uint8_t *data, size_t len);
|
||
|
||
/**
|
||
* 编码一个数据包
|
||
* 将触发2次onWebSocketEncodeData回调
|
||
* @param header 数据头
|
||
* @param buffer 负载数据
|
||
* Encode a data packet
|
||
* Will trigger 2 onWebSocketEncodeData callbacks
|
||
* @param header Data header
|
||
* @param buffer Payload data
|
||
|
||
* [AUTO-TRANSLATED:f308e552]
|
||
*/
|
||
void encode(const WebSocketHeader &header,const toolkit::Buffer::Ptr &buffer);
|
||
|
||
protected:
|
||
/**
|
||
* 收到一个webSocket数据包包头,后续将继续触发onWebSocketDecodePayload回调
|
||
* @param header 数据包头
|
||
* Receive a webSocket data packet header, and will continue to trigger onWebSocketDecodePayload callback
|
||
* @param header Data packet header
|
||
|
||
* [AUTO-TRANSLATED:7bc6b7c6]
|
||
*/
|
||
virtual void onWebSocketDecodeHeader(const WebSocketHeader &header) {};
|
||
|
||
/**
|
||
* 收到webSocket数据包负载
|
||
* @param header 数据包包头
|
||
* @param ptr 负载数据指针
|
||
* @param len 负载数据长度
|
||
* @param recved 已接收数据长度(包含本次数据长度),等于header._payload_len时则接受完毕
|
||
* Receive webSocket data packet payload
|
||
* @param header Data packet header
|
||
* @param ptr Payload data pointer
|
||
* @param len Payload data length
|
||
* @param recved Received data length (including the length of this data), equals header._payload_len when the reception is complete
|
||
|
||
* [AUTO-TRANSLATED:ca056d2e]
|
||
*/
|
||
virtual void onWebSocketDecodePayload(const WebSocketHeader &header, const uint8_t *ptr, size_t len, size_t recved) {};
|
||
|
||
/**
|
||
* 接收到完整的一个webSocket数据包后回调
|
||
* @param header 数据包包头
|
||
* Callback after receiving a complete webSocket data packet
|
||
* @param header Data packet header
|
||
|
||
* [AUTO-TRANSLATED:f506a7c5]
|
||
*/
|
||
virtual void onWebSocketDecodeComplete(const WebSocketHeader &header) {};
|
||
|
||
/**
|
||
* websocket数据编码回调
|
||
* @param ptr 数据指针
|
||
* @param len 数据指针长度
|
||
* websocket data encoding callback
|
||
* @param ptr Data pointer
|
||
* @param len Data pointer length
|
||
|
||
|
||
* [AUTO-TRANSLATED:7c940c67]
|
||
*/
|
||
virtual void onWebSocketEncodeData(toolkit::Buffer::Ptr buffer){};
|
||
|
||
private:
|
||
void onPayloadData(uint8_t *data, size_t len);
|
||
|
||
private:
|
||
bool _got_header = false;
|
||
int _mask_offset = 0;
|
||
size_t _payload_offset = 0;
|
||
std::string _remain_data;
|
||
};
|
||
|
||
} /* namespace mediakit */
|
||
|
||
|
||
#endif //ZLMEDIAKIT_WEBSOCKETSPLITTER_H
|