blob: 681afa11f6f31b338134d9c97b7193ba02010a5e [file] [log] [blame] [edit]
#pragma once
#include <boost/spirit/home/x3.hpp>
#include <algorithm>
#include <cctype>
#include <iomanip>
#include <ostream>
#include <ranges>
#include <span>
#include <string>
#include <string_view>
#include <vector>
namespace http_helpers
{
enum class ContentType
{
NoMatch,
ANY, // Accepts: */*
CBOR,
HTML,
JSON,
OctetStream,
EventStream,
};
inline ContentType getContentType(std::string_view contentTypeHeader)
{
using boost::spirit::x3::char_;
using boost::spirit::x3::lit;
using boost::spirit::x3::no_case;
using boost::spirit::x3::omit;
using boost::spirit::x3::parse;
using boost::spirit::x3::space;
using boost::spirit::x3::symbols;
using boost::spirit::x3::uint_;
const symbols<ContentType> knownMimeType{
{"application/cbor", ContentType::CBOR},
{"application/json", ContentType::JSON},
{"application/octet-stream", ContentType::OctetStream},
{"text/event-stream", ContentType::EventStream},
{"text/html", ContentType::HTML}};
ContentType ct = ContentType::NoMatch;
auto typeCharset = +(char_("a-zA-Z0-9.+-"));
auto parameters =
*(lit(';') >> *space >> typeCharset >> lit("=") >> typeCharset);
auto parser = no_case[knownMimeType] >> omit[parameters];
std::string_view::iterator begin = contentTypeHeader.begin();
if (!parse(begin, contentTypeHeader.end(), parser, ct))
{
return ContentType::NoMatch;
}
if (begin != contentTypeHeader.end())
{
return ContentType::NoMatch;
}
return ct;
}
inline ContentType getPreferredContentType(
std::string_view acceptsHeader, std::span<const ContentType> preferredOrder)
{
using boost::spirit::x3::char_;
using boost::spirit::x3::lit;
using boost::spirit::x3::no_case;
using boost::spirit::x3::omit;
using boost::spirit::x3::parse;
using boost::spirit::x3::space;
using boost::spirit::x3::symbols;
using boost::spirit::x3::uint_;
const symbols<ContentType> knownMimeType{
{"application/cbor", ContentType::CBOR},
{"application/json", ContentType::JSON},
{"application/octet-stream", ContentType::OctetStream},
{"text/html", ContentType::HTML},
{"text/event-stream", ContentType::EventStream},
{"*/*", ContentType::ANY}};
std::vector<ContentType> ct;
auto typeCharset = +(char_("a-zA-Z0-9.+-"));
auto parameters = *(lit(';') >> typeCharset >> lit("=") >> typeCharset);
auto mimeType = no_case[knownMimeType] |
omit[+typeCharset >> lit('/') >> +typeCharset];
auto parser = +(mimeType >> omit[parameters >> -char_(',') >> *space]);
if (!parse(acceptsHeader.begin(), acceptsHeader.end(), parser, ct))
{
return ContentType::NoMatch;
}
for (const ContentType parsedType : ct)
{
if (parsedType == ContentType::ANY)
{
return parsedType;
}
auto it = std::ranges::find(preferredOrder, parsedType);
if (it != preferredOrder.end())
{
return *it;
}
}
return ContentType::NoMatch;
}
inline bool isContentTypeAllowed(std::string_view header, ContentType type,
bool allowWildcard)
{
auto types = std::to_array({type});
ContentType allowed = getPreferredContentType(header, types);
if (allowed == ContentType::ANY)
{
return allowWildcard;
}
return type == allowed;
}
enum class Encoding
{
ParseError,
NoMatch,
UnencodedBytes,
GZIP,
ZSTD,
ANY, // represents *. Never returned. Only used for string matching
};
inline Encoding
getPreferredEncoding(std::string_view acceptEncoding,
const std::span<const Encoding> availableEncodings)
{
if (acceptEncoding.empty())
{
return Encoding::UnencodedBytes;
}
using boost::spirit::x3::char_;
using boost::spirit::x3::lit;
using boost::spirit::x3::omit;
using boost::spirit::x3::parse;
using boost::spirit::x3::space;
using boost::spirit::x3::symbols;
using boost::spirit::x3::uint_;
const symbols<Encoding> knownAcceptEncoding{{"gzip", Encoding::GZIP},
{"zstd", Encoding::ZSTD},
{"*", Encoding::ANY}};
std::vector<Encoding> ct;
auto parameters = *(lit(';') >> lit("q=") >> uint_ >> -(lit('.') >> uint_));
auto typeCharset = char_("a-zA-Z.+-");
auto encodeType = knownAcceptEncoding | omit[+typeCharset];
auto parser = +(encodeType >> omit[parameters >> -char_(',') >> *space]);
if (!parse(acceptEncoding.begin(), acceptEncoding.end(), parser, ct))
{
return Encoding::ParseError;
}
for (const Encoding parsedType : ct)
{
if (parsedType == Encoding::ANY)
{
if (!availableEncodings.empty())
{
return *availableEncodings.begin();
}
}
auto it = std::ranges::find(availableEncodings, parsedType);
if (it != availableEncodings.end())
{
return *it;
}
}
// Fall back to raw bytes if it was allowed
auto it = std::ranges::find(availableEncodings, Encoding::UnencodedBytes);
if (it != availableEncodings.end())
{
return *it;
}
return Encoding::NoMatch;
}
} // namespace http_helpers