2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
42#include "web/http/Parser.h"
44#include "core/socket/stream/SocketContext.h"
45#include "web/http/decoder/Chunked.h"
46#include "web/http/decoder/HTTP10Response.h"
47#include "web/http/decoder/Identity.h"
49#ifndef DOXYGEN_SHOULD_SKIP_THIS
51#include "web/http/http_utils.h"
55#include <system_error>
66 unsigned long long value = 0;
69 const char* first = s.data();
70 const char* last = s.data() + s.size();
72 const auto [ptr, ec] = std::from_chars(first, last, value, 10);
74 if (ec == std::errc{} && ptr == last && value <=
static_cast<
unsigned long long>(std::numeric_limits<std::size_t>::max())) {
75 out =
static_cast<std::size_t>(value);
84 bool hasChunked =
false;
86 if (headers.contains(
"Transfer-Encoding")) {
87 const std::string& encoding = headers[
"Transfer-Encoding"];
98 : hTTPCompliance(compliance)
119 delete contentDecoder;
128 std::size_t consumed = 0;
158 std::size_t consumed = 0;
167 if (ch ==
'\r' || ch ==
'\n') {
205 if (
headers.contains(
"Trailer")) {
206 std::string trailers =
headers[
"Trailer"];
208 while (!trailers.empty()) {
209 std::string trailerField;
213 trailerField.clear();
217 }
else if (
headers.contains(
"Content-Length")) {
218 std::size_t length = 0;
232 if (
headers.contains(
"Transfer-Encoding")) {
233 const std::string& encoding =
headers[
"Transfer-Encoding"];
249 if (
headers.contains(
"Content-Encoding")) {
250 const std::string& encoding =
headers[
"Content-Encoding"];
271 const std::size_t consumed = contentDecoder
->read();
298 headers.insert(trailer.begin(), trailer.end());
306 return static_cast<
enum Parser::
HTTPCompliance>(
static_cast<
unsigned short>(c1) |
static_cast<
unsigned short>(c2));
310 return static_cast<
enum Parser::
HTTPCompliance>(
static_cast<
unsigned short>(c1) &
static_cast<
unsigned short>(c2));
std::size_t readFromPeer(char *chunk, std::size_t chunklen) const final
virtual std::size_t read()=0
std::vector< char > && getContent()
friend enum HTTPCompliance operator&(const enum HTTPCompliance &c1, const enum HTTPCompliance &c2)
virtual void parseError(int code, const std::string &reason)=0
Parser(core::socket::stream::SocketContext *socketContext, const enum HTTPCompliance &compliance=HTTPCompliance::RFC2616|HTTPCompliance::RFC7230)
static const std::regex httpVersionRegex
friend enum HTTPCompliance operator|(const enum HTTPCompliance &c1, const enum HTTPCompliance &c2)
std::size_t contentLength
virtual void parseStartLine(const std::string &line)=0
std::size_t contentLengthRead
web::http::decoder::Fields headerDecoder
core::socket::stream::SocketContext * socketContext
CiStringMap< std::string > headers
std::size_t readContent()
std::list< web::http::ContentDecoder * > decoderQueue
virtual void parsingFinished()=0
TransferEncoding transferEncoding
std::size_t readTrailer()
std::vector< char > content
std::size_t readStartLine()
virtual void analyzeHeader()
std::set< std::string > trailerFieldsExpected
web::http::decoder::Fields trailerDecoder
Chunked(const core::socket::stream::SocketContext *socketContext)
std::string getErrorReason()
Fields(core::socket::stream::SocketContext *socketContext, std::set< std::string > fieldsExpected={})
void setFieldsExpected(std::set< std::string > fieldsExpected)
web::http::CiStringMap< std::string > && getHeader()
HTTP10Response(const core::socket::stream::SocketContext *socketContext)
Identity(const core::socket::stream::SocketContext *socketContext, std::size_t contentLengthExpected)
std::pair< std::string, std::string > str_split(const std::string &base, char c_middle)
std::string & str_trimm(std::string &text)
bool ciContains(const std::string &str1, const std::string &str2)
static bool transferEncodingHasChunked(CiStringMap< std::string > &headers)
static bool parseContentLengthStrict(const std::string &s, std::size_t &out)