blob: 1ab1ff7f989e410ac0d5dd4a89cca941ba28c5b3 [file] [log] [blame]
#ifndef THIRD_PARTY_GBMCWEB_INCLUDE_HTTP_UTILITY_H_
#define THIRD_PARTY_GBMCWEB_INCLUDE_HTTP_UTILITY_H_
#include <algorithm>
#include <array>
#include <cctype>
#include <cstddef>
#include <cstdint>
#include <iomanip>
#include <ios>
#include <span> // NOLINT
#include <sstream>
#include <string>
#include <string_view>
#include "boost/algorithm/string/classification.hpp" // NOLINT
#include "boost/algorithm/string/constants.hpp" // NOLINT
#include "boost/iterator/iterator_facade.hpp" // NOLINT
#include "boost/type_index/type_index_facade.hpp" // NOLINT
// IWYU pragma: no_include <ctype.h>
namespace http_helpers {
enum class ContentType : std::uint8_t {
NoMatch,
ANY, // Accepts: */*
CBOR,
HTML,
JSON,
OctetStream,
};
struct ContentTypePair {
std::string_view contentTypeString; // NOLINT
ContentType contentTypeEnum; // NOLINT
};
constexpr std::array<ContentTypePair, 4> contentTypes{{
{"application/cbor", ContentType::CBOR},
{"application/json", ContentType::JSON},
{"application/octet-stream", ContentType::OctetStream},
{"text/html", ContentType::HTML},
}};
inline ContentType getPreferedContentType(
std::string_view header, std::span<const ContentType> preferedOrder) {
size_t last_index = 0;
while (last_index < header.size() + 1) {
size_t index = header.find(',', last_index);
if (index == std::string_view::npos) {
index = header.size();
}
std::string_view encoding = header.substr(last_index, index);
if (!header.empty()) {
header.remove_prefix(1);
}
last_index = index + 1;
// ignore any q-factor weighting (;q=)
std::size_t separator = encoding.find(";q=");
if (separator != std::string_view::npos) {
encoding = encoding.substr(0, separator);
}
// If the client allows any encoding, given them the first one on the
// servers list
if (encoding == "*/*") {
return ContentType::ANY;
}
const auto* known_content_type =
std::find_if(contentTypes.begin(), contentTypes.end(),
[encoding](const ContentTypePair& pair) {
return pair.contentTypeString == encoding;
});
if (known_content_type == contentTypes.end()) {
// not able to find content type in list
continue;
}
// Not one of the types requested
if (std::find(preferedOrder.begin(), preferedOrder.end(),
known_content_type->contentTypeEnum) == preferedOrder.end()) {
continue;
}
return known_content_type->contentTypeEnum;
}
return ContentType::NoMatch;
}
inline bool isContentTypeAllowed(std::string_view header, ContentType type,
bool allowWildcard) {
auto types = std::to_array({type});
ContentType allowed = getPreferedContentType(header, types);
if (allowed == ContentType::ANY) {
return allowWildcard;
}
return type == allowed;
}
inline std::string urlEncode(std::string_view value) {
std::ostringstream escaped;
escaped.fill('0');
escaped << std::hex;
for (const char c : value) {
// Keep alphanumeric and other accepted characters intact
if ((isalnum(c) != 0) || c == '-' || c == '_' || c == '.' || c == '~') {
escaped << c;
continue;
}
// Any other characters are percent-encoded
escaped << std::uppercase;
escaped << '%' << std::setw(2)
<< static_cast<int>(static_cast<unsigned char>(c));
escaped << std::nouppercase;
}
return escaped.str();
}
} // namespace http_helpers
#endif // THIRD_PARTY_GBMCWEB_INCLUDE_HTTP_UTILITY_H_