2016-11-29 16:23:38 +01:00
|
|
|
/*
|
|
|
|
* charybdis: 21st Century IRC++d
|
|
|
|
*
|
|
|
|
* Copyright (C) 2016 Charybdis Development Team
|
|
|
|
* Copyright (C) 2016 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.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
|
|
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_HTTP_H
|
|
|
|
|
|
|
|
namespace ircd {
|
|
|
|
namespace http {
|
|
|
|
|
2017-03-11 02:46:25 +01:00
|
|
|
enum code
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 20:02:24 +01:00
|
|
|
CONTINUE = 100,
|
|
|
|
SWITCHING_PROTOCOLS = 101,
|
|
|
|
|
|
|
|
OK = 200,
|
|
|
|
CREATED = 201,
|
|
|
|
ACCEPTED = 202,
|
|
|
|
NON_AUTHORITATIVE_INFORMATION = 203,
|
|
|
|
NO_CONTENT = 204,
|
2017-04-07 01:54:20 +02:00
|
|
|
PARTIAL_CONTENT = 206,
|
|
|
|
|
|
|
|
MOVED_PERMANENTLY = 301,
|
|
|
|
FOUND = 302,
|
|
|
|
SEE_OTHER = 303,
|
|
|
|
NOT_MODIFIED = 304,
|
|
|
|
TEMPORARY_REDIRECT = 305,
|
|
|
|
PERMANENT_REDIRECT = 306,
|
2017-03-13 20:02:24 +01:00
|
|
|
|
|
|
|
BAD_REQUEST = 400,
|
|
|
|
UNAUTHORIZED = 401,
|
|
|
|
FORBIDDEN = 403,
|
|
|
|
NOT_FOUND = 404,
|
|
|
|
METHOD_NOT_ALLOWED = 405,
|
|
|
|
REQUEST_TIMEOUT = 408,
|
2017-04-03 05:59:18 +02:00
|
|
|
CONFLICT = 409,
|
|
|
|
REQUEST_URI_TOO_LONG = 414,
|
2017-03-13 20:02:24 +01:00
|
|
|
EXPECTATION_FAILED = 417,
|
2017-04-07 02:08:40 +02:00
|
|
|
IM_A_TEAPOT = 418,
|
2017-04-03 05:59:18 +02:00
|
|
|
UNPROCESSABLE_ENTITY = 422,
|
2017-03-13 20:02:24 +01:00
|
|
|
TOO_MANY_REQUESTS = 429,
|
|
|
|
REQUEST_HEADER_FIELDS_TOO_LARGE = 431,
|
|
|
|
|
|
|
|
INTERNAL_SERVER_ERROR = 500,
|
|
|
|
NOT_IMPLEMENTED = 501,
|
|
|
|
SERVICE_UNAVAILABLE = 503,
|
|
|
|
HTTP_VERSION_NOT_SUPPORTED = 505,
|
2017-04-03 05:59:18 +02:00
|
|
|
INSUFFICIENT_STORAGE = 507,
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-23 23:02:59 +01:00
|
|
|
extern std::map<code, string_view> reason;
|
2017-03-18 04:28:49 +01:00
|
|
|
enum code status(const string_view &);
|
2016-11-29 16:23:38 +01:00
|
|
|
|
|
|
|
struct error
|
|
|
|
:ircd::error
|
|
|
|
{
|
2017-03-11 02:46:25 +01:00
|
|
|
enum code code;
|
2016-11-29 16:23:38 +01:00
|
|
|
std::string content;
|
|
|
|
|
2017-03-18 04:28:49 +01:00
|
|
|
error(const enum code &, std::string content = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct line
|
|
|
|
:string_view
|
|
|
|
{
|
|
|
|
struct request;
|
|
|
|
struct response;
|
|
|
|
struct header;
|
|
|
|
|
|
|
|
using string_view::string_view;
|
2017-03-13 23:24:42 +01:00
|
|
|
line(parse::capstan &);
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct line::request
|
|
|
|
{
|
|
|
|
string_view method;
|
2017-04-03 06:00:12 +02:00
|
|
|
string_view path;
|
|
|
|
string_view query;
|
|
|
|
string_view fragment;
|
2016-11-29 16:23:38 +01:00
|
|
|
string_view version;
|
|
|
|
|
|
|
|
request(const line &);
|
|
|
|
request() = default;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct line::response
|
|
|
|
{
|
|
|
|
string_view version;
|
|
|
|
string_view status;
|
|
|
|
string_view reason;
|
|
|
|
|
|
|
|
response(const line &);
|
|
|
|
response() = default;
|
|
|
|
};
|
|
|
|
|
2017-04-03 06:00:12 +02:00
|
|
|
struct 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;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Query string is read as a complete string off the tape (into request.query)
|
|
|
|
// and not parsed further. To make queries into that string pun this class on it.
|
|
|
|
struct query::string
|
|
|
|
:string_view
|
|
|
|
{
|
|
|
|
void for_each(const std::function<void (const query &)> &) const;
|
|
|
|
bool until(const std::function<bool (const query &)> &) const;
|
|
|
|
|
|
|
|
string_view at(const string_view &key) const;
|
|
|
|
string_view operator[](const string_view &key) const;
|
|
|
|
|
|
|
|
using string_view::string_view;
|
|
|
|
};
|
|
|
|
|
2016-11-29 16:23:38 +01:00
|
|
|
struct line::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); }
|
|
|
|
|
|
|
|
using std::pair<string_view, string_view>::pair;
|
|
|
|
header(const line &);
|
|
|
|
header() = default;
|
|
|
|
};
|
|
|
|
|
2017-04-03 06:00:12 +02:00
|
|
|
// HTTP headers are read once off the tape and proffered to the closure.
|
2016-11-29 16:23:38 +01:00
|
|
|
struct headers
|
|
|
|
{
|
2017-03-13 22:07:58 +01:00
|
|
|
using closure = std::function<void (const line::header &)>;
|
2016-11-29 16:23:38 +01:00
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
headers(parse::capstan &, const closure & = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-04-03 06:00:12 +02:00
|
|
|
// Use the request::content / response::content wrappers. They ensure the proper amount
|
|
|
|
// of content is read and the tape is in the right position for the next request
|
|
|
|
// with exception safety.
|
2016-11-29 16:23:38 +01:00
|
|
|
struct content
|
2017-03-13 22:07:58 +01:00
|
|
|
:string_view
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 22:07:58 +01:00
|
|
|
IRCD_OVERLOAD(discard)
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
content(parse::capstan &, const size_t &length, discard_t);
|
|
|
|
content(parse::capstan &, const size_t &length);
|
2017-03-13 22:07:58 +01:00
|
|
|
content() = default;
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct response
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
|
|
|
struct head;
|
2017-03-13 22:07:58 +01:00
|
|
|
struct content;
|
|
|
|
|
|
|
|
using write_closure = std::function<void (const_buffers &)>;
|
|
|
|
using proffer = std::function<void (const head &)>;
|
|
|
|
|
|
|
|
response(const code &,
|
|
|
|
const string_view &content,
|
|
|
|
const write_closure &,
|
|
|
|
const std::initializer_list<line::header> & = {});
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
response(parse::capstan &,
|
2017-03-13 22:07:58 +01:00
|
|
|
content *const & = nullptr,
|
|
|
|
const proffer & = nullptr,
|
|
|
|
const headers::closure & = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct response::head
|
|
|
|
:line::response
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 22:07:58 +01:00
|
|
|
size_t content_length {0};
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
head(parse::capstan &pc, const headers::closure &c = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct response::content
|
|
|
|
:http::content
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 23:24:42 +01:00
|
|
|
content(parse::capstan &pc, const head &h, discard_t)
|
2017-03-13 22:07:58 +01:00
|
|
|
:http::content{pc, h.content_length, discard}
|
|
|
|
{}
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
content(parse::capstan &pc, const head &h)
|
2017-03-13 22:07:58 +01:00
|
|
|
:http::content{pc, h.content_length}
|
2016-11-29 16:23:38 +01:00
|
|
|
{}
|
2017-03-13 22:07:58 +01:00
|
|
|
|
|
|
|
content() = default;
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct request
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 22:07:58 +01:00
|
|
|
struct head;
|
|
|
|
struct content;
|
|
|
|
|
|
|
|
using write_closure = std::function<void (const_buffers &)>;
|
|
|
|
using proffer = std::function<void (const head &)>;
|
|
|
|
|
|
|
|
request(const string_view &host = {},
|
|
|
|
const string_view &method = "GET",
|
2017-04-03 06:00:12 +02:00
|
|
|
const string_view &path = "/",
|
|
|
|
const string_view &query = {},
|
2017-03-13 22:07:58 +01:00
|
|
|
const string_view &content = {},
|
|
|
|
const write_closure & = nullptr,
|
|
|
|
const std::initializer_list<line::header> & = {});
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
request(parse::capstan &,
|
2017-03-13 22:07:58 +01:00
|
|
|
content *const & = nullptr,
|
|
|
|
const write_closure & = nullptr,
|
|
|
|
const proffer & = nullptr,
|
|
|
|
const headers::closure & = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct request::head
|
|
|
|
:line::request
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 22:07:58 +01:00
|
|
|
string_view host;
|
|
|
|
string_view expect;
|
|
|
|
string_view te;
|
|
|
|
size_t content_length {0};
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
head(parse::capstan &pc, const headers::closure &c = {});
|
2016-11-29 16:23:38 +01:00
|
|
|
};
|
|
|
|
|
2017-03-13 22:07:58 +01:00
|
|
|
struct request::content
|
|
|
|
:http::content
|
2016-11-29 16:23:38 +01:00
|
|
|
{
|
2017-03-13 23:24:42 +01:00
|
|
|
content(parse::capstan &pc, const head &h, discard_t)
|
2017-03-13 22:07:58 +01:00
|
|
|
:http::content{pc, h.content_length, discard}
|
|
|
|
{}
|
|
|
|
|
2017-03-13 23:24:42 +01:00
|
|
|
content(parse::capstan &pc, const head &h)
|
2017-03-13 22:07:58 +01:00
|
|
|
:http::content{pc, h.content_length}
|
2016-11-29 16:23:38 +01:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace http
|
|
|
|
} // namespace ircd
|