2018-02-03 18:22:01 -08:00
|
|
|
// 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.
|
2016-11-29 07:23:38 -08:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_RESOURCE_H
|
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
namespace ircd
|
|
|
|
{
|
2018-02-17 15:44:53 -08:00
|
|
|
struct client;
|
2017-08-28 14:51:22 -07:00
|
|
|
struct resource;
|
|
|
|
}
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
struct ircd::resource
|
2016-11-29 07:23:38 -08:00
|
|
|
{
|
|
|
|
IRCD_EXCEPTION(ircd::error, error)
|
|
|
|
|
2017-12-12 13:26:39 -07:00
|
|
|
enum flag :uint;
|
|
|
|
struct opts;
|
2016-11-29 07:23:38 -08:00
|
|
|
struct method;
|
|
|
|
struct request;
|
|
|
|
struct response;
|
|
|
|
|
2017-08-23 15:06:14 -06:00
|
|
|
static std::map<string_view, resource *, iless> resources;
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2017-08-23 15:06:14 -06:00
|
|
|
string_view path;
|
|
|
|
string_view description;
|
2017-12-12 13:26:39 -07:00
|
|
|
enum flag flags;
|
2017-08-23 15:06:14 -06:00
|
|
|
std::map<string_view, method *> methods;
|
|
|
|
unique_const_iterator<decltype(resources)> resources_it;
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2018-03-11 10:17:37 -07:00
|
|
|
string_view allow_methods_list(const mutable_buffer &buf);
|
|
|
|
|
2017-08-23 15:06:14 -06:00
|
|
|
private:
|
|
|
|
virtual void handle_request(client &, method &, resource::request &);
|
2017-03-20 19:30:07 -07:00
|
|
|
|
2016-11-29 07:23:38 -08:00
|
|
|
public:
|
2017-08-23 15:06:14 -06:00
|
|
|
method &operator[](const string_view &path);
|
2018-01-11 18:45:25 -08:00
|
|
|
|
2018-02-12 11:58:40 -08:00
|
|
|
void operator()(client &, const http::request::head &, const string_view &content_partial);
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2017-12-12 13:26:39 -07:00
|
|
|
resource(const string_view &path, const opts &);
|
|
|
|
resource(const string_view &path);
|
2017-08-23 15:06:14 -06:00
|
|
|
resource() = default;
|
2016-11-29 07:23:38 -08:00
|
|
|
virtual ~resource() noexcept;
|
2017-08-23 15:06:14 -06:00
|
|
|
|
2018-04-23 19:11:11 -07:00
|
|
|
static resource &find(const string_view &path);
|
2016-11-29 07:23:38 -08:00
|
|
|
};
|
|
|
|
|
2017-12-12 13:26:39 -07:00
|
|
|
enum ircd::resource::flag
|
|
|
|
:uint
|
|
|
|
{
|
|
|
|
DIRECTORY = 0x01,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::resource::opts
|
|
|
|
{
|
|
|
|
/// developer's literal description of the resource
|
|
|
|
string_view description
|
|
|
|
{
|
|
|
|
"no description"
|
|
|
|
};
|
|
|
|
|
|
|
|
/// flags for the resource
|
|
|
|
flag flags
|
|
|
|
{
|
|
|
|
flag(0)
|
|
|
|
};
|
|
|
|
|
|
|
|
/// parameter count limits (DIRECTORY only)
|
|
|
|
std::pair<short, short> parc
|
|
|
|
{
|
|
|
|
0, // minimum params
|
|
|
|
15 // maximum params
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
struct ircd::resource::request
|
2017-08-23 15:47:21 -06:00
|
|
|
:json::object
|
2016-11-29 07:23:38 -08:00
|
|
|
{
|
2017-09-21 02:11:07 -07:00
|
|
|
template<class> struct object;
|
2017-08-28 14:51:22 -07:00
|
|
|
|
2018-02-17 15:44:53 -08:00
|
|
|
http::request::head head;
|
2018-01-11 18:45:25 -08:00
|
|
|
string_view content;
|
2017-08-23 15:06:14 -06:00
|
|
|
http::query::string query;
|
2018-04-02 22:58:12 -07:00
|
|
|
string_view origin;
|
2018-02-10 22:07:06 -08:00
|
|
|
string_view access_token;
|
2017-10-11 20:52:33 -07:00
|
|
|
vector_view<string_view> parv;
|
2018-02-21 17:41:28 -08:00
|
|
|
string_view param[8];
|
2018-04-22 16:22:26 -07:00
|
|
|
m::user::id::buf user_id;
|
2017-03-20 19:30:07 -07:00
|
|
|
|
2018-01-11 18:45:25 -08:00
|
|
|
request(const http::request::head &head,
|
2018-02-17 15:44:53 -08:00
|
|
|
const string_view &content)
|
|
|
|
:json::object{content}
|
|
|
|
,head{head}
|
|
|
|
,content{content}
|
2018-02-21 17:41:28 -08:00
|
|
|
,query{this->head.query}
|
2018-02-17 15:44:53 -08:00
|
|
|
{}
|
|
|
|
|
|
|
|
request() = default;
|
2017-03-20 19:30:07 -07:00
|
|
|
};
|
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
template<class tuple>
|
2017-09-21 02:11:07 -07:00
|
|
|
struct ircd::resource::request::object
|
2017-08-28 14:51:22 -07:00
|
|
|
:tuple
|
|
|
|
{
|
2017-09-21 02:11:07 -07:00
|
|
|
resource::request &r;
|
|
|
|
const http::request::head &head;
|
2018-01-11 18:45:25 -08:00
|
|
|
const string_view &content;
|
2017-09-21 02:11:07 -07:00
|
|
|
const http::query::string &query;
|
2018-04-02 22:58:12 -07:00
|
|
|
const decltype(r.origin) &origin;
|
2017-12-12 13:26:39 -07:00
|
|
|
const decltype(r.user_id) &user_id;
|
2018-02-10 22:07:06 -08:00
|
|
|
const decltype(r.access_token) &access_token;
|
2017-10-11 20:52:33 -07:00
|
|
|
const vector_view<string_view> &parv;
|
2017-09-21 02:11:07 -07:00
|
|
|
const json::object &body;
|
|
|
|
|
|
|
|
object(resource::request &r)
|
|
|
|
:tuple{r}
|
|
|
|
,r{r}
|
|
|
|
,head{r.head}
|
|
|
|
,content{r.content}
|
|
|
|
,query{r.query}
|
2018-04-02 22:58:12 -07:00
|
|
|
,origin{r.origin}
|
2017-12-12 13:26:39 -07:00
|
|
|
,user_id{r.user_id}
|
2018-02-10 22:07:06 -08:00
|
|
|
,access_token{r.access_token}
|
2017-10-11 20:52:33 -07:00
|
|
|
,parv{r.parv}
|
2017-09-21 02:11:07 -07:00
|
|
|
,body{r}
|
|
|
|
{}
|
2017-08-28 14:51:22 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ircd::resource::response
|
2017-03-20 19:30:07 -07:00
|
|
|
{
|
2018-04-13 23:19:43 -07:00
|
|
|
struct chunked;
|
|
|
|
|
2018-03-09 13:15:32 -08:00
|
|
|
response(client &, const http::code &, const string_view &content_type, const size_t &content_length, const string_view &headers = {});
|
2017-12-24 14:25:09 -07:00
|
|
|
response(client &, const string_view &str, const string_view &content_type, const http::code &, const vector_view<const http::header> &);
|
|
|
|
response(client &, const string_view &str, const string_view &content_type, const http::code & = http::OK, const string_view &headers = {});
|
2017-09-15 12:07:07 -07:00
|
|
|
response(client &, const json::object &str, const http::code & = http::OK);
|
2017-09-24 17:00:05 -07:00
|
|
|
response(client &, const json::array &str, const http::code & = http::OK);
|
2017-09-15 12:07:07 -07:00
|
|
|
response(client &, const json::members & = {}, const http::code & = http::OK);
|
2017-09-24 17:00:05 -07:00
|
|
|
response(client &, const json::value &, const http::code & = http::OK);
|
2017-09-09 12:20:00 -07:00
|
|
|
response(client &, const json::iov &, const http::code & = http::OK);
|
2017-09-12 13:34:21 -07:00
|
|
|
response(client &, const http::code &, const json::members &);
|
2017-09-24 17:00:05 -07:00
|
|
|
response(client &, const http::code &, const json::value &);
|
2017-09-09 12:20:00 -07:00
|
|
|
response(client &, const http::code &, const json::iov &);
|
|
|
|
response(client &, const http::code &);
|
2017-03-20 19:30:07 -07:00
|
|
|
response() = default;
|
2016-11-29 07:23:38 -08:00
|
|
|
};
|
|
|
|
|
2018-04-13 23:19:43 -07:00
|
|
|
struct ircd::resource::response::chunked
|
|
|
|
:resource::response
|
|
|
|
{
|
|
|
|
client *c {nullptr};
|
|
|
|
|
|
|
|
size_t write(const const_buffer &chunk);
|
|
|
|
bool finish();
|
|
|
|
|
|
|
|
chunked(client &, const http::code &, const string_view &content_type, const string_view &headers = {});
|
|
|
|
chunked(client &, const http::code &, const string_view &content_type, const vector_view<const http::header> &);
|
|
|
|
chunked(client &, const http::code &, const vector_view<const http::header> &);
|
|
|
|
chunked(client &, const http::code &);
|
|
|
|
chunked(const chunked &) = delete;
|
|
|
|
chunked(chunked &&) noexcept;
|
|
|
|
chunked() = default;
|
|
|
|
~chunked() noexcept;
|
|
|
|
};
|
|
|
|
|
2017-08-28 14:51:22 -07:00
|
|
|
struct ircd::resource::method
|
2016-11-29 07:23:38 -08:00
|
|
|
{
|
2018-01-11 18:45:25 -08:00
|
|
|
using handler = std::function<response (client &, request &)>;
|
|
|
|
|
2017-08-23 15:06:14 -06:00
|
|
|
enum flag
|
|
|
|
{
|
2018-02-17 14:32:11 -08:00
|
|
|
REQUIRES_AUTH = 0x01,
|
|
|
|
RATE_LIMITED = 0x02,
|
|
|
|
VERIFY_ORIGIN = 0x04,
|
|
|
|
CONTENT_DISCRETION = 0x08,
|
2017-08-23 15:06:14 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
struct opts
|
|
|
|
{
|
2018-01-11 18:45:25 -08:00
|
|
|
flag flags {(flag)0};
|
2017-08-23 15:06:14 -06:00
|
|
|
|
2018-04-15 16:42:13 -07:00
|
|
|
/// Timeout specific to this resource.
|
|
|
|
seconds timeout {30s};
|
|
|
|
|
2018-01-11 18:45:25 -08:00
|
|
|
/// The maximum size of the Content-Length for this method. Anything
|
|
|
|
/// larger will be summarily rejected with a 413.
|
|
|
|
size_t payload_max {128_KiB};
|
2018-04-15 16:42:13 -07:00
|
|
|
|
2018-01-11 18:45:25 -08:00
|
|
|
};
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2017-08-23 15:06:14 -06:00
|
|
|
string_view name;
|
2016-11-29 07:23:38 -08:00
|
|
|
struct resource *resource;
|
2017-08-23 15:06:14 -06:00
|
|
|
handler function;
|
2018-01-11 18:45:25 -08:00
|
|
|
struct opts opts;
|
2017-08-23 15:06:14 -06:00
|
|
|
unique_const_iterator<decltype(resource::methods)> methods_it;
|
2016-11-29 07:23:38 -08:00
|
|
|
|
|
|
|
public:
|
2017-08-23 15:06:14 -06:00
|
|
|
virtual response operator()(client &, request &);
|
2016-11-29 07:23:38 -08:00
|
|
|
|
2018-01-11 18:45:25 -08:00
|
|
|
method(struct resource &, const string_view &name, const handler &, const struct opts &);
|
|
|
|
method(struct resource &, const string_view &name, const handler &);
|
2017-08-23 15:06:14 -06:00
|
|
|
virtual ~method() noexcept;
|
2016-11-29 07:23:38 -08:00
|
|
|
};
|