mirror of
https://github.com/oatpp/oatpp.git
synced 2025-03-31 18:30:22 +08:00
RequestHeadersReader added
This commit is contained in:
parent
919d4472c1
commit
619ad7461d
@ -367,6 +367,40 @@ std::shared_ptr<Protocol::Headers> Protocol::parseHeaders(oatpp::parser::Parsing
|
||||
|
||||
}
|
||||
|
||||
void Protocol::parseRequestStartingLineStruct(RequestStartingLineStruct& line,
|
||||
const std::shared_ptr<oatpp::base::StrBuffer>& headersText,
|
||||
oatpp::parser::ParsingCaret& caret,
|
||||
Status& error) {
|
||||
|
||||
oatpp::parser::ParsingCaret::Label methodLabel(caret);
|
||||
if(caret.findChar(' ')){
|
||||
line.method = oatpp::data::share::MemoryLabel(headersText, methodLabel.getData(), methodLabel.getSize());
|
||||
caret.inc();
|
||||
} else {
|
||||
error = Status::CODE_400;
|
||||
return;
|
||||
}
|
||||
|
||||
oatpp::parser::ParsingCaret::Label pathLabel(caret);
|
||||
if(caret.findChar(' ')){
|
||||
line.path = oatpp::data::share::MemoryLabel(headersText, pathLabel.getData(), pathLabel.getSize());
|
||||
caret.inc();
|
||||
} else {
|
||||
error = Status::CODE_400;
|
||||
return;
|
||||
}
|
||||
|
||||
oatpp::parser::ParsingCaret::Label protocolLabel(caret);
|
||||
if(caret.findRN()){
|
||||
line.protocol = oatpp::data::share::MemoryLabel(headersText, protocolLabel.getData(), protocolLabel.getSize());
|
||||
caret.skipRN();
|
||||
} else {
|
||||
error = Status::CODE_400;
|
||||
return;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void Protocol::parseOneHeaderLabel(HeadersLabels& headers,
|
||||
const std::shared_ptr<oatpp::base::StrBuffer>& headersText,
|
||||
oatpp::parser::ParsingCaret& caret,
|
||||
|
@ -291,6 +291,11 @@ public:
|
||||
static void parseOneHeader(Headers& headers, oatpp::parser::ParsingCaret& caret, Status& error);
|
||||
static std::shared_ptr<Headers> parseHeaders(oatpp::parser::ParsingCaret& caret, Status& error);
|
||||
|
||||
static void parseRequestStartingLineStruct(RequestStartingLineStruct& line,
|
||||
const std::shared_ptr<oatpp::base::StrBuffer>& headersText,
|
||||
oatpp::parser::ParsingCaret& caret,
|
||||
Status& error);
|
||||
|
||||
static void parseOneHeaderLabel(HeadersLabels& headers,
|
||||
const std::shared_ptr<oatpp::base::StrBuffer>& headersText,
|
||||
oatpp::parser::ParsingCaret& caret,
|
||||
|
@ -41,6 +41,9 @@ os::io::Library::v_size RequestHeadersReader::readHeadersSection(const std::shar
|
||||
v_int32 desiredToRead = m_bufferSize;
|
||||
if(progress + desiredToRead > m_maxHeadersSize) {
|
||||
desiredToRead = m_maxHeadersSize - progress;
|
||||
if(desiredToRead <= 0) {
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
res = connection->read(m_buffer, desiredToRead);
|
||||
@ -81,11 +84,101 @@ RequestHeadersReader::Result RequestHeadersReader::readHeaders(const std::shared
|
||||
auto headersText = buffer.toString();
|
||||
oatpp::parser::ParsingCaret caret (headersText);
|
||||
http::Status status;
|
||||
http::Protocol::parseHeadersLabels(result.headers, headersText.getPtr(), caret, status);
|
||||
http::Protocol::parseRequestStartingLineStruct(result.startingLine, headersText.getPtr(), caret, status);
|
||||
if(status.code == 0) {
|
||||
http::Protocol::parseHeadersLabels(result.headers, headersText.getPtr(), caret, status);
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
|
||||
}
|
||||
|
||||
|
||||
RequestHeadersReader::Action RequestHeadersReader::readHeadersAsync(oatpp::async::AbstractCoroutine* parentCoroutine,
|
||||
AsyncCallback callback,
|
||||
const std::shared_ptr<oatpp::data::stream::IOStream>& connection)
|
||||
{
|
||||
|
||||
class ReaderCoroutine : public oatpp::async::CoroutineWithResult<ReaderCoroutine, Result> {
|
||||
private:
|
||||
std::shared_ptr<oatpp::data::stream::IOStream> m_connection;
|
||||
p_char8 m_buffer;
|
||||
v_int32 m_bufferSize;
|
||||
v_int32 m_maxHeadersSize;
|
||||
v_word32 m_accumulator;
|
||||
v_int32 m_progress;
|
||||
RequestHeadersReader::Result m_result;
|
||||
oatpp::data::stream::ChunkedBuffer m_bufferStream;
|
||||
public:
|
||||
|
||||
ReaderCoroutine(const std::shared_ptr<oatpp::data::stream::IOStream>& connection,
|
||||
p_char8 buffer, v_int32 bufferSize, v_int32 maxHeadersSize)
|
||||
: m_connection(connection)
|
||||
, m_buffer(buffer)
|
||||
, m_bufferSize(bufferSize)
|
||||
, m_maxHeadersSize(maxHeadersSize)
|
||||
, m_accumulator(0)
|
||||
, m_progress(0)
|
||||
{}
|
||||
|
||||
Action act() override {
|
||||
|
||||
v_int32 desiredToRead = m_bufferSize;
|
||||
if(m_progress + desiredToRead > m_maxHeadersSize) {
|
||||
desiredToRead = m_maxHeadersSize - m_progress;
|
||||
if(desiredToRead <= 0) {
|
||||
return error("Headers section is too large");
|
||||
}
|
||||
}
|
||||
|
||||
auto res = m_connection->read(m_buffer, desiredToRead);
|
||||
if(res > 0) {
|
||||
m_bufferStream.write(m_buffer, res);
|
||||
|
||||
for(v_int32 i = 0; i < res; i ++) {
|
||||
m_accumulator <<= 8;
|
||||
m_accumulator |= m_buffer[i];
|
||||
if(m_accumulator == SECTION_END) {
|
||||
m_result.bufferPosStart = i + 1;
|
||||
m_result.bufferPosEnd = (v_int32) res;
|
||||
return yieldTo(&ReaderCoroutine::parseHeaders);
|
||||
}
|
||||
}
|
||||
|
||||
return waitRetry();
|
||||
|
||||
} else if(res == oatpp::data::stream::Errors::ERROR_IO_WAIT_RETRY || res == oatpp::data::stream::Errors::ERROR_IO_RETRY) {
|
||||
return waitRetry();
|
||||
} else {
|
||||
return abort();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Action parseHeaders() {
|
||||
|
||||
auto headersText = m_bufferStream.toString();
|
||||
oatpp::parser::ParsingCaret caret (headersText);
|
||||
http::Status status;
|
||||
http::Protocol::parseRequestStartingLineStruct(m_result.startingLine, headersText.getPtr(), caret, status);
|
||||
if(status.code == 0) {
|
||||
http::Protocol::parseHeadersLabels(m_result.headers, headersText.getPtr(), caret, status);
|
||||
if(status.code == 0) {
|
||||
return _return(m_result);
|
||||
} else {
|
||||
return error("error occurred while parsing headers");
|
||||
}
|
||||
} else {
|
||||
return error("can't parse starting line");
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
return parentCoroutine->startCoroutineForResult<ReaderCoroutine>(callback, connection, m_buffer, m_bufferSize, m_maxHeadersSize);
|
||||
|
||||
}
|
||||
|
||||
}}}}}
|
||||
|
@ -26,10 +26,15 @@
|
||||
#define oatpp_web_protocol_http_incoming_RequestHeadersReader_hpp
|
||||
|
||||
#include "oatpp/web/protocol/http/Http.hpp"
|
||||
#include "oatpp/core/async/Coroutine.hpp"
|
||||
|
||||
namespace oatpp { namespace web { namespace protocol { namespace http { namespace incoming {
|
||||
|
||||
class RequestHeadersReader {
|
||||
public:
|
||||
typedef oatpp::async::Action Action;
|
||||
private:
|
||||
static constexpr v_int32 SECTION_END = ('\r' << 24) | ('\n' << 16) | ('\r' << 8) | ('\n');
|
||||
public:
|
||||
|
||||
struct Result {
|
||||
@ -38,6 +43,9 @@ public:
|
||||
v_int32 bufferPosStart;
|
||||
v_int32 bufferPosEnd;
|
||||
};
|
||||
|
||||
public:
|
||||
typedef Action (oatpp::async::AbstractCoroutine::*AsyncCallback)(const Result&);
|
||||
private:
|
||||
os::io::Library::v_size readHeadersSection(const std::shared_ptr<oatpp::data::stream::IOStream>& connection,
|
||||
oatpp::data::stream::OutputStream* bufferStream,
|
||||
@ -49,6 +57,9 @@ private:
|
||||
public:
|
||||
|
||||
Result readHeaders(const std::shared_ptr<oatpp::data::stream::IOStream>& connection, http::Status& error);
|
||||
Action readHeadersAsync(oatpp::async::AbstractCoroutine* parentCoroutine,
|
||||
AsyncCallback callback,
|
||||
const std::shared_ptr<oatpp::data::stream::IOStream>& connection);
|
||||
|
||||
};
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user