// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2019 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_M_FETCH_H /// Event Fetcher (remote). /// /// This is a federation network interface to find and retrieve datas from /// remote parties serially. It operates by querying servers in a room until /// one server can provide a satisfying response. The exact method for /// determining who to contact, when and how is encapsulated internally for /// further development, but it is primarily stochastic. This is liable to be /// optimized with further development of selection algorithms and hinting. /// All viable servers in a room are exhausted before an error is the result. /// /// This is an asynchronous promise/future based interface. The result package /// is delivered by a ctx::future. Note that while fetch::start() is not /// intended to yield the ircd::ctx, though it is possible in rare cases. /// /// Due to the composition of multiple operations performed internally, result /// future has no real timeout control over the operation as a whole. While it /// can always go out of scope for an effective cancelation, internal conf::item /// are used to timeout failures after a deterministic `timeout * servers`. /// This means the user is not required to wait_for() or wait_until() on the /// future unless they want a stricter timeout; that may miss a valid response /// for a rare piece of data held by a minority of servers. /// /// Alternatively, m::feds is another federation network interface geared to /// conducting a parallel request to every server in a room; this conducts a /// serial request to every server in a room (and stopping when satisfied). /// namespace ircd::m::fetch { struct init; struct opts; struct result; struct request; enum class op :uint8_t; // Observers string_view reflect(const op &); bool for_each(const std::function<bool (request &)> &); bool exists(const opts &); size_t count(); // Primary operations ctx::future<result> start(opts); } enum class ircd::m::fetch::op :uint8_t { noop, auth, event, backfill, }; struct ircd::m::fetch::opts { /// Operation to perform. fetch::op op {op::noop}; /// room::id apropos. Many federation requests require a room_id, but /// nevertheless a room_id is still used by this unit as a pool of servers. room::id room_id; /// event::id apropos. For op::event operations this is being sought, but /// for others it may be required as a reference point. If not supplied and /// required, we'll try to use the top head from any room_id. event::id event_id; /// The principal allocation size. This is passed up the stack to m::fed, /// server::request and ends up containing the request head and content, /// and response head. The response content is usually dynamically /// allocated and that buffer is the one which ends up in result. Note /// that sufficiently large values here may allow for eliding the content /// allocation based on the following formula: >= 16_KiB + (64_KiB * limit) /// where 16_KiB is [current server default] for headers and 64_KiB is /// m::event::MAX_SIZE. size_t bufsz {0}; /// Name of a remote server which will be queried first; if failure, /// the normal room_id based operation is the fallback. If the room /// is not known to us, it would be best to set this. string_view hint; /// Limit the number of servers to be contacted for this operation. Zero /// is automatic / unlimited. Note that setting this value to 1 in /// conjunction with a hint is analogous to just making an m::fed request. size_t attempt_limit {0}; // // special options // /// If the op makes use of a spec limit parameter that can be controlled /// by the user here. The default of 0 will be replaced by some internal /// configured limit like 8 or 16 etc. size_t backfill_limit {0}; }; struct ircd::m::fetch::result { /// Backing buffer for any data pointed to by this result. shared_buffer<mutable_buffer> buf; /// The backing buffer may contain other data ahead of the response /// content; in any case this points to a view of the response content. /// User access to response content should be via a json conversion rather /// than this reference. string_view content; /// JSON result conversion. Note that developers should not let the result /// instance go out of scope by making this conversion. explicit operator json::object() const; explicit operator json::array() const; }; /// Fetch entity state. DO NOT CONSTRUCT. This is an internal structure but we /// expose it here for examination, statistics and hacking since it has no /// non-standard symbols; this is simpler than creating some accessor suite. /// Instances of this object are created and managed internally by the m::fetch /// unit after a fetch::start() is called. This definition is not required to /// operate the m::fetch interface as a user. struct ircd::m::fetch::request { using is_transparent = void; /// Copy of the user's request options. Note that the backing of strings in /// opts was changed to point at this structure; allowing safe access. fetch::opts opts; /// Time the first attempt was made; this value is not modified so it can /// be used to measure the total time of all attempts. system_point started; /// Time the last attempt was started system_point last; /// Time the request entered the finished state. This being non-zero /// indicates a finished state; may be difficult to observe. system_point finished; /// State for failed attempts; the names of servers which failed are /// stored here. Failure here means the request succeeded but the server /// did not provide a satisfying response. Appearing in this list prevents /// a server from being selected for the next attempt. std::set<std::string, std::less<>> attempted; /// Reference to the current server being attempted. This string is placed /// in the attempted set at the start of an attempt. string_view origin; /// HTTP heads and scratch buffer for server::request unique_buffer<mutable_buffer> buf; /// Our future for the server::request. Since we make std::unique_ptr<server::request> future; /// Promise for our user's future of this request. ctx::promise<result> promise; /// Error pointer state for an attempt. This is cleared each attempt. std::exception_ptr eptr; /// Buffer backing for opts m::event::id::buf event_id; m::room::id::buf room_id; /// Internal request(const fetch::opts &); request(request &&) = delete; request(const request &) = delete; request &operator=(request &&) = delete; request &operator=(const request &) = delete; ~request() noexcept; }; /// Internally held struct ircd::m::fetch::init { init(), ~init() noexcept; }; inline ircd::m::fetch::result::operator json::array() const { return content; } inline ircd::m::fetch::result::operator json::object() const { return content; }