182 lines
6.2 KiB
C++
182 lines
6.2 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.
|
|||
|
*/
|
|||
|
|
|||
|
#include "HttpRequestSplitter.h"
|
|||
|
#include "Util/logger.h"
|
|||
|
#include "Util/util.h"
|
|||
|
using namespace toolkit;
|
|||
|
using namespace std;
|
|||
|
|
|||
|
// 协议解析最大缓存4兆数据 [AUTO-TRANSLATED:75159526]
|
|||
|
// Protocol parsing maximum cache 4MB data
|
|||
|
static constexpr size_t kMaxCacheSize = 4 * 1024 * 1024;
|
|||
|
|
|||
|
namespace mediakit {
|
|||
|
|
|||
|
void HttpRequestSplitter::input(const char *data,size_t len) {
|
|||
|
{
|
|||
|
auto size = remainDataSize();
|
|||
|
if (size > _max_cache_size) {
|
|||
|
// 缓存太多数据无法处理则上抛异常 [AUTO-TRANSLATED:30e48e9e]
|
|||
|
// If too much data is cached and cannot be processed, throw an exception
|
|||
|
reset();
|
|||
|
throw std::out_of_range("remain data size is too huge, now cleared:" + to_string(size));
|
|||
|
}
|
|||
|
}
|
|||
|
const char *ptr = data;
|
|||
|
if(!_remain_data.empty()){
|
|||
|
_remain_data.append(data,len);
|
|||
|
data = ptr = _remain_data.data();
|
|||
|
len = _remain_data.size();
|
|||
|
}
|
|||
|
|
|||
|
splitPacket:
|
|||
|
|
|||
|
/*确保ptr最后一个字节是0,防止strstr越界
|
|||
|
*由于ZLToolKit确保内存最后一个字节是保留未使用字节并置0,
|
|||
|
*所以此处可以不用再次置0
|
|||
|
*但是上层数据可能来自其他渠道,保险起见还是置0
|
|||
|
*Ensure the last byte of ptr is 0 to prevent strstr from going out of bounds
|
|||
|
* Since ZLToolKit ensures that the last byte of memory is a reserved unused byte and set to 0,
|
|||
|
* so there is no need to set it to 0 again here
|
|||
|
* But the upper layer data may come from other channels, so it is better to set it to 0 for safety
|
|||
|
|
|||
|
* [AUTO-TRANSLATED:28ff47a5]
|
|||
|
*/
|
|||
|
|
|||
|
char &tail_ref = ((char *) ptr)[len];
|
|||
|
char tail_tmp = tail_ref;
|
|||
|
tail_ref = 0;
|
|||
|
|
|||
|
// 数据按照请求头处理 [AUTO-TRANSLATED:e7a0dbb4]
|
|||
|
// Data is processed according to the request header
|
|||
|
const char *index = nullptr;
|
|||
|
_remain_data_size = len;
|
|||
|
while (_content_len == 0 && _remain_data_size > 0 && (index = onSearchPacketTail(ptr,_remain_data_size)) != nullptr) {
|
|||
|
if (index == ptr) {
|
|||
|
break;
|
|||
|
}
|
|||
|
if (index < ptr || index > ptr + _remain_data_size) {
|
|||
|
throw std::out_of_range("上层分包逻辑异常");
|
|||
|
}
|
|||
|
// _content_len == 0,这是请求头 [AUTO-TRANSLATED:32af637b]
|
|||
|
// _content_len == 0, this is the request header
|
|||
|
const char *header_ptr = ptr;
|
|||
|
ssize_t header_size = index - ptr;
|
|||
|
ptr = index;
|
|||
|
_remain_data_size = len - (ptr - data);
|
|||
|
_content_len = onRecvHeader(header_ptr, header_size);
|
|||
|
}
|
|||
|
|
|||
|
/*
|
|||
|
* 恢复末尾字节
|
|||
|
* 移动到这来,目的是防止HttpRequestSplitter::reset()导致内存失效
|
|||
|
/*
|
|||
|
* Restore the last byte
|
|||
|
* Move it here to prevent HttpRequestSplitter::reset() from causing memory failure
|
|||
|
|
|||
|
* [AUTO-TRANSLATED:9c3e0597]
|
|||
|
*/
|
|||
|
tail_ref = tail_tmp;
|
|||
|
|
|||
|
if(_remain_data_size <= 0){
|
|||
|
// 没有剩余数据,清空缓存 [AUTO-TRANSLATED:16613daa]
|
|||
|
// No remaining data, clear the cache
|
|||
|
_remain_data.clear();
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if(_content_len == 0){
|
|||
|
// 尚未找到http头,缓存定位到剩余数据部分 [AUTO-TRANSLATED:7a9d6205]
|
|||
|
// HTTP header not found yet, cache is located at the remaining data part
|
|||
|
_remain_data.assign(ptr,_remain_data_size);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// 已经找到http头了 [AUTO-TRANSLATED:df166db7]
|
|||
|
// HTTP header has been found
|
|||
|
if(_content_len > 0){
|
|||
|
// 数据按照固定长度content处理 [AUTO-TRANSLATED:7272b7e7]
|
|||
|
// Data is processed according to fixed length content
|
|||
|
if(_remain_data_size < (size_t)_content_len){
|
|||
|
// 数据不够,缓存定位到剩余数据部分 [AUTO-TRANSLATED:61c32f5c]
|
|||
|
// Insufficient data, cache is located at the remaining data part
|
|||
|
_remain_data.assign(ptr, _remain_data_size);
|
|||
|
return;
|
|||
|
}
|
|||
|
// 收到content数据,并且接收content完毕 [AUTO-TRANSLATED:0342dc0e]
|
|||
|
// Content data received and content reception completed
|
|||
|
onRecvContent(ptr,_content_len);
|
|||
|
|
|||
|
_remain_data_size -= _content_len;
|
|||
|
ptr += _content_len;
|
|||
|
// content处理完毕,后面数据当做请求头处理 [AUTO-TRANSLATED:d268dfe4]
|
|||
|
// Content processing completed, subsequent data is treated as request header
|
|||
|
_content_len = 0;
|
|||
|
|
|||
|
if(_remain_data_size > 0){
|
|||
|
// 还有数据没有处理完毕 [AUTO-TRANSLATED:1cac6727]
|
|||
|
// There is still data that has not been processed
|
|||
|
_remain_data.assign(ptr,_remain_data_size);
|
|||
|
data = ptr = (char *)_remain_data.data();
|
|||
|
len = _remain_data.size();
|
|||
|
goto splitPacket;
|
|||
|
}
|
|||
|
_remain_data.clear();
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// _content_len < 0;数据按照不固定长度content处理 [AUTO-TRANSLATED:68d6a4d0]
|
|||
|
// _content_len < 0; Data is processed according to variable length content
|
|||
|
onRecvContent(ptr,_remain_data_size);//消费掉所有剩余数据
|
|||
|
_remain_data.clear();
|
|||
|
}
|
|||
|
|
|||
|
void HttpRequestSplitter::setContentLen(ssize_t content_len) {
|
|||
|
_content_len = content_len;
|
|||
|
}
|
|||
|
|
|||
|
void HttpRequestSplitter::reset() {
|
|||
|
_content_len = 0;
|
|||
|
_remain_data_size = 0;
|
|||
|
_remain_data.clear();
|
|||
|
}
|
|||
|
|
|||
|
const char *HttpRequestSplitter::onSearchPacketTail(const char *data,size_t len) {
|
|||
|
auto pos = strstr(data,"\r\n\r\n");
|
|||
|
if(pos == nullptr){
|
|||
|
return nullptr;
|
|||
|
}
|
|||
|
return pos + 4;
|
|||
|
}
|
|||
|
|
|||
|
size_t HttpRequestSplitter::remainDataSize() {
|
|||
|
return _remain_data_size;
|
|||
|
}
|
|||
|
|
|||
|
const char *HttpRequestSplitter::remainData() const {
|
|||
|
return _remain_data.data();
|
|||
|
}
|
|||
|
|
|||
|
void HttpRequestSplitter::setMaxCacheSize(size_t max_cache_size) {
|
|||
|
if (!max_cache_size) {
|
|||
|
max_cache_size = kMaxCacheSize;
|
|||
|
}
|
|||
|
_max_cache_size = max_cache_size;
|
|||
|
}
|
|||
|
|
|||
|
HttpRequestSplitter::HttpRequestSplitter() {
|
|||
|
setMaxCacheSize(0);
|
|||
|
}
|
|||
|
|
|||
|
} /* namespace mediakit */
|
|||
|
|