/* * Copyright (C) 2016 Charybdis Development Team * Copyright (C) 2016 Jason Volk * * 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_RESOURCE_H namespace ircd { struct resource; } struct ircd::resource { IRCD_EXCEPTION(ircd::error, error) enum flag :uint; struct opts; struct method; struct request; struct response; static std::map resources; string_view path; string_view description; enum flag flags; std::map methods; unique_const_iterator resources_it; private: virtual void handle_request(client &, method &, resource::request &); public: method &operator[](const string_view &path); void operator()(client &, struct client::request &, const http::request::head &); resource(const string_view &path, const opts &); resource(const string_view &path); resource() = default; virtual ~resource() noexcept; static resource &find(string_view path); }; 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 parc { 0, // minimum params 15 // maximum params }; }; struct ircd::resource::request :json::object { template struct object; const http::request::head &head; string_view content; http::query::string query; string_view user_id; //m::user::id::buf user_id; //TODO: bleeding vector_view parv; request(const http::request::head &head, const string_view &content, http::query::string query, const vector_view &parv); }; template struct ircd::resource::request::object :tuple { resource::request &r; const http::request::head &head; const string_view &content; const http::query::string &query; const decltype(r.user_id) &user_id; const vector_view &parv; const json::object &body; object(resource::request &r) :tuple{r} ,r{r} ,head{r.head} ,content{r.content} ,query{r.query} ,user_id{r.user_id} ,parv{r.parv} ,body{r} {} }; struct ircd::resource::response { response(client &, const string_view &str, const string_view &content_type, const http::code &, const vector_view &); response(client &, const string_view &str, const string_view &content_type, const http::code & = http::OK, const string_view &headers = {}); response(client &, const json::object &str, const http::code & = http::OK); response(client &, const json::array &str, const http::code & = http::OK); response(client &, const json::members & = {}, const http::code & = http::OK); response(client &, const json::value &, const http::code & = http::OK); response(client &, const json::iov &, const http::code & = http::OK); response(client &, const http::code &, const json::members &); response(client &, const http::code &, const json::value &); response(client &, const http::code &, const json::iov &); response(client &, const http::code &); response() = default; }; struct ircd::resource::method { using handler = std::function; enum flag { REQUIRES_AUTH = 0x01, RATE_LIMITED = 0x02, VERIFY_ORIGIN = 0x04, }; struct opts { flag flags {(flag)0}; /// 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}; }; string_view name; struct resource *resource; handler function; struct opts opts; unique_const_iterator methods_it; public: virtual response operator()(client &, request &); method(struct resource &, const string_view &name, const handler &, const struct opts &); method(struct resource &, const string_view &name, const handler &); virtual ~method() noexcept; };