mirror of
https://github.com/matrix-construct/construct
synced 2025-01-13 08:23:56 +01:00
433 lines
13 KiB
C++
433 lines
13 KiB
C++
// Matrix Construct
|
|
//
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
|
|
//
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
// full license for this software is available in the LICENSE file.
|
|
|
|
#pragma once
|
|
#define HAVE_IRCD_HTTP_H
|
|
|
|
/// HyperText TransPort: formal grammars & tools
|
|
namespace ircd::http
|
|
{
|
|
struct error;
|
|
struct line;
|
|
struct query;
|
|
struct header;
|
|
struct headers;
|
|
struct request;
|
|
struct response;
|
|
|
|
enum code :ushort;
|
|
string_view status(const enum code &) noexcept;
|
|
enum code status(const string_view &);
|
|
|
|
enum class category :uint8_t;
|
|
string_view category(const enum category &) noexcept;
|
|
enum category category(const enum code &) noexcept;
|
|
enum category category(const string_view &) noexcept;
|
|
enum log::level severity(const enum category &) noexcept;
|
|
|
|
void writeline(window_buffer &);
|
|
void writeline(window_buffer &, const window_buffer::closure &);
|
|
void write(window_buffer &, const header &);
|
|
void write(window_buffer &, const vector_view<const header> &);
|
|
size_t serialized(const vector_view<const header> &);
|
|
std::string strung(const vector_view<const header> &);
|
|
void writechunk(window_buffer &, const uint32_t &size);
|
|
const_buffer writechunk(const mutable_buffer &, const uint32_t &size);
|
|
bool has(const headers &, const string_view &key);
|
|
bool has(const vector_view<const header> &, const string_view &key);
|
|
}
|
|
|
|
/// HTTP Status classifications.
|
|
enum class ircd::http::category
|
|
:uint8_t
|
|
{
|
|
NONE = 0, ///< Sentinel
|
|
INFO = 1, ///< Informational (100-199)
|
|
SUCCESS = 2, ///< Successful (200-299)
|
|
REDIRECT = 3, ///< Redirectional (300-399)
|
|
ERROR = 4, ///< Client Error (400-499)
|
|
SERVER = 5, ///< Server Error (501-599)
|
|
INTERNAL = 6, ///< Internal Error (500 only)
|
|
UNKNOWN = 7, ///< Pair with default case in switch/tables
|
|
};
|
|
|
|
/// HTTP Status codes.
|
|
enum ircd::http::code
|
|
:ushort
|
|
{
|
|
CONTINUE = 100,
|
|
SWITCHING_PROTOCOLS = 101,
|
|
PROCESSING = 102,
|
|
EARLY_HINTS = 103,
|
|
|
|
OK = 200,
|
|
CREATED = 201,
|
|
ACCEPTED = 202,
|
|
NON_AUTHORITATIVE_INFORMATION = 203,
|
|
NO_CONTENT = 204,
|
|
PARTIAL_CONTENT = 206,
|
|
|
|
MULTIPLE_CHOICES = 300,
|
|
MOVED_PERMANENTLY = 301,
|
|
FOUND = 302,
|
|
SEE_OTHER = 303,
|
|
NOT_MODIFIED = 304,
|
|
USE_PROXY = 305,
|
|
SWITCH_PROXY = 306,
|
|
TEMPORARY_REDIRECT = 307,
|
|
PERMANENT_REDIRECT = 308,
|
|
|
|
BAD_REQUEST = 400,
|
|
UNAUTHORIZED = 401,
|
|
FORBIDDEN = 403,
|
|
NOT_FOUND = 404,
|
|
METHOD_NOT_ALLOWED = 405,
|
|
NOT_ACCEPTABLE = 406,
|
|
REQUEST_TIMEOUT = 408,
|
|
CONFLICT = 409,
|
|
GONE = 410,
|
|
LENGTH_REQUIRED = 411,
|
|
PAYLOAD_TOO_LARGE = 413,
|
|
REQUEST_URI_TOO_LONG = 414,
|
|
UNSUPPORTED_MEDIA_TYPE = 415,
|
|
RANGE_NOT_SATISFIABLE = 416,
|
|
EXPECTATION_FAILED = 417,
|
|
IM_A_TEAPOT = 418,
|
|
UNPROCESSABLE_ENTITY = 422,
|
|
PRECONDITION_REQUIRED = 428,
|
|
TOO_MANY_REQUESTS = 429,
|
|
REQUEST_HEADER_FIELDS_TOO_LARGE = 431,
|
|
|
|
INTERNAL_SERVER_ERROR = 500,
|
|
NOT_IMPLEMENTED = 501,
|
|
BAD_GATEWAY = 502,
|
|
SERVICE_UNAVAILABLE = 503,
|
|
GATEWAY_TIMEOUT = 504,
|
|
HTTP_VERSION_NOT_SUPPORTED = 505,
|
|
INSUFFICIENT_STORAGE = 507,
|
|
|
|
CLOUDFLARE_REFUSED = 521,
|
|
CLOUDFLARE_TIMEDOUT = 522,
|
|
CLOUDFLARE_UNREACHABLE = 523,
|
|
CLOUDFLARE_REQUEST_TIMEOUT = 524,
|
|
};
|
|
|
|
/// Root exception for HTTP.
|
|
struct ircd::http::error
|
|
:ircd::error
|
|
{
|
|
std::string content;
|
|
std::string headers;
|
|
enum code code {http::code(0)};
|
|
|
|
explicit operator bool() const { return code != http::code(0); }
|
|
bool operator!() const { return !bool(*this); }
|
|
|
|
error() = default;
|
|
error(const enum code &, std::string content = {}, std::string headers = {});
|
|
error(const enum code &, std::string content, const vector_view<const header> &);
|
|
template<class... args> error(const string_view &fmt, const enum code &, args&&...);
|
|
~error() noexcept;
|
|
};
|
|
|
|
/// Represents a single \r\n delimited line used in HTTP.
|
|
///
|
|
/// This object is just a string_view of that line. The actual data backing
|
|
/// that view is the responsibility of the user. This object is constructed
|
|
/// with an ircd::parse::capstan argument which is used by the formal grammar
|
|
/// in the constructor.
|
|
///
|
|
struct ircd::http::line
|
|
:string_view
|
|
{
|
|
struct request;
|
|
struct response;
|
|
|
|
static const string_view terminator; // "\r\n"
|
|
|
|
using string_view::string_view;
|
|
line(parse::capstan &);
|
|
};
|
|
|
|
/// Represents a 'request line' or the first line a client sends to a server.
|
|
///
|
|
/// This is a dual-use class. For HTTP clients, one may simply connect the
|
|
/// members to the proper strings and then pass this structure to a function
|
|
/// making a client request. For HTTP servers, pass an http::line to the ctor
|
|
/// and the formal grammar will set the members appropriately. The actual data
|
|
/// behind these members is the responsibility of the user.
|
|
///
|
|
struct ircd::http::line::request
|
|
{
|
|
string_view method;
|
|
string_view path;
|
|
string_view query;
|
|
string_view fragment;
|
|
string_view version;
|
|
|
|
operator string_view() const; // full view of line
|
|
|
|
request(const line &);
|
|
request() = default;
|
|
};
|
|
|
|
/// Represents a 'response line' or the first line a server sends to a client.
|
|
///
|
|
/// This is a dual-use class and symmetric to the http::line::request class.
|
|
/// Servers may set the members and then use this object to respond to a client
|
|
/// while clients should provide an http::line to the constructor which will
|
|
/// fill in the members.
|
|
///
|
|
struct ircd::http::line::response
|
|
{
|
|
string_view version;
|
|
string_view status;
|
|
string_view reason;
|
|
|
|
response(const line &);
|
|
response() = default;
|
|
};
|
|
|
|
/// Represents a single key/value pair in a query string.
|
|
///
|
|
/// This is used by the ircd::http::query::string object when parsing query
|
|
/// strings.
|
|
///
|
|
struct ircd::http::query
|
|
:std::pair<string_view, string_view>
|
|
{
|
|
struct string;
|
|
|
|
bool operator<(const string_view &s) const { return iless(first, s); }
|
|
bool operator==(const string_view &s) const { return iequals(first, s); }
|
|
|
|
using std::pair<string_view, string_view>::pair;
|
|
query() = default;
|
|
};
|
|
|
|
/// Tool for parsing an HTTP query string.
|
|
///
|
|
/// Query string is read as a complete string off the tape (into request.query)
|
|
/// and not parsed further. To make queries into that string use this class to
|
|
/// view it. Once this object is constructed by viewing the whole query string,
|
|
/// the member functions invoke the formal grammar to get individual key/value
|
|
/// pairs.
|
|
///
|
|
struct ircd::http::query::string
|
|
:string_view
|
|
{
|
|
using closure = std::function<bool (const query &)>;
|
|
|
|
bool for_each(const closure &) const;
|
|
bool for_each(const string_view &key, const closure &) const;
|
|
|
|
string_view _get(const string_view &key, size_t idx = 0) const;
|
|
template<class T = string_view> T get(const string_view &key, const T &def = {}, const size_t &idx = 0) const;
|
|
string_view operator[](const string_view &key) const;
|
|
|
|
string_view at(const string_view &key, const size_t &idx = 0) const;
|
|
template<class T> T at(const string_view &key, const size_t &idx = 0) const;
|
|
|
|
vector_view<string_view> array(const mutable_buffer &, const string_view &key, string_view *const &, const size_t &) const;
|
|
template<size_t MAX> vector_view<string_view> array(const mutable_buffer &, const string_view &key, string_view (&)[MAX]) const;
|
|
|
|
size_t count(const string_view &key) const;
|
|
bool has(const string_view &key) const;
|
|
|
|
using string_view::string_view;
|
|
};
|
|
|
|
/// Represents an HTTP header key/value pair.
|
|
///
|
|
/// This is a dual-use class. Those sending headers will simply fill in the
|
|
/// components of the std::pair. Those receiving headers can pass the ctor an
|
|
/// ircd::http::line which will construct the pair using the formal grammars.
|
|
///
|
|
struct ircd::http::header
|
|
:std::pair<string_view, string_view>
|
|
{
|
|
bool operator<(const string_view &s) const { return iless(first, s); }
|
|
bool operator==(const string_view &s) const { return iequals(first, s); }
|
|
bool operator!=(const string_view &s) const { return !operator==(s); }
|
|
|
|
using std::pair<string_view, string_view>::pair;
|
|
header(const line &);
|
|
header() = default;
|
|
};
|
|
|
|
/// This device allows parsing HTTP headers directly off the wire without state
|
|
///
|
|
/// The constructor of this object contains the grammar to read HTTP headers
|
|
/// from the capstan and then proffer them one by one to the provided closure,
|
|
/// that's all it does.
|
|
///
|
|
struct ircd::http::headers
|
|
:string_view
|
|
{
|
|
using closure = std::function<void (const header &)>;
|
|
using closure_bool = std::function<bool (const header &)>;
|
|
|
|
static const string_view terminator; // "\r\n\r\n"
|
|
|
|
bool for_each(const closure_bool &) const;
|
|
string_view operator[](const string_view &key) const;
|
|
string_view at(const string_view &key) const;
|
|
bool has(const string_view &key) const;
|
|
|
|
using string_view::string_view;
|
|
headers(parse::capstan &, closure_bool);
|
|
headers(parse::capstan &, const closure & = {});
|
|
headers() = default;
|
|
|
|
friend bool has(const headers &, const string_view &key);
|
|
friend bool has(const vector_view<const header> &, const string_view &key);
|
|
};
|
|
|
|
/// HTTP request suite. Functionality to send and receive requests.
|
|
///
|
|
struct ircd::http::request
|
|
{
|
|
struct head;
|
|
|
|
// compose a request into buffer
|
|
request(window_buffer &,
|
|
const string_view &host,
|
|
const string_view &method = "GET",
|
|
const string_view &uri = "/",
|
|
const size_t &content_length = 0,
|
|
const string_view &content_type = {},
|
|
const vector_view<const header> & = {},
|
|
const bool &termination = true);
|
|
};
|
|
|
|
/// Represents an HTTP request head. This is only for receiving requests.
|
|
///
|
|
struct ircd::http::request::head
|
|
:line::request
|
|
{
|
|
string_view host;
|
|
string_view expect;
|
|
string_view te;
|
|
string_view authorization;
|
|
string_view connection;
|
|
string_view content_type;
|
|
string_view user_agent;
|
|
string_view upgrade;
|
|
string_view range;
|
|
string_view if_range;
|
|
string_view forwarded_for;
|
|
size_t content_length {0};
|
|
|
|
string_view uri; // full view of (path, query, fragmet)
|
|
string_view headers; // full view of all headers
|
|
|
|
// full view of all head (request line and headers)
|
|
operator string_view() const;
|
|
|
|
head(parse::capstan &pc, const headers::closure &c = {});
|
|
head() = default;
|
|
};
|
|
|
|
/// HTTP response suite. Functionality to send and receive responses.
|
|
///
|
|
struct ircd::http::response
|
|
{
|
|
struct head;
|
|
struct chunk;
|
|
|
|
// compose a response into buffer
|
|
response(window_buffer &,
|
|
const enum code &,
|
|
const size_t &content_length = 0,
|
|
const string_view &content_type = {},
|
|
const http::headers &headers = {},
|
|
const vector_view<const header> & = {},
|
|
const bool &termination = true);
|
|
|
|
response() = default;
|
|
};
|
|
|
|
/// Represents an HTTP response head. This is for receiving responses only.
|
|
///
|
|
struct ircd::http::response::head
|
|
:line::response
|
|
{
|
|
string_view content_type;
|
|
size_t content_length {0};
|
|
string_view content_range;
|
|
string_view accept_range;
|
|
string_view transfer_encoding;
|
|
string_view server;
|
|
string_view location;
|
|
|
|
string_view headers;
|
|
|
|
head(parse::capstan &pc, const headers::closure &c);
|
|
head(parse::capstan &pc);
|
|
head() = default;
|
|
};
|
|
|
|
struct ircd::http::response::chunk
|
|
:line
|
|
{
|
|
uint32_t size {0};
|
|
|
|
chunk(parse::capstan &pc);
|
|
chunk() = default;
|
|
};
|
|
|
|
template<size_t MAX>
|
|
inline ircd::vector_view<ircd::string_view>
|
|
ircd::http::query::string::array(const mutable_buffer &buf,
|
|
const string_view &key,
|
|
string_view (&out)[MAX])
|
|
const
|
|
{
|
|
return array(buf, key, out, MAX);
|
|
}
|
|
|
|
template<class T>
|
|
inline T
|
|
ircd::http::query::string::get(const string_view &key,
|
|
const T &def,
|
|
const size_t &idx)
|
|
const try
|
|
{
|
|
const auto val(_get(key, idx));
|
|
return val? lex_cast<T>(val) : def;
|
|
}
|
|
catch(const bad_lex_cast &)
|
|
{
|
|
return def;
|
|
}
|
|
|
|
template<class T>
|
|
inline T
|
|
ircd::http::query::string::at(const string_view &key,
|
|
const size_t &idx)
|
|
const
|
|
{
|
|
return lex_cast<T>(at(key, idx));
|
|
}
|
|
|
|
template<class... args>
|
|
inline
|
|
ircd::http::error::error(const string_view &fmt,
|
|
const http::code &code,
|
|
args&&... a)
|
|
:error
|
|
{
|
|
code,
|
|
fmt::snstringf //TODO: XXX fmt::sstringf
|
|
{
|
|
3072, fmt, std::forward<args>(a)...
|
|
}
|
|
}
|
|
{}
|