// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 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. The // full license for this software is available in the LICENSE file. // Fetch unit state namespace ircd::m::fetch { struct request; // m/fetch.h struct evaltab; static bool operator<(const request &a, const request &b) noexcept; static bool operator<(const request &a, const string_view &b) noexcept; static bool operator<(const string_view &a, const request &b) noexcept; extern ctx::dock dock; extern std::set> requests; extern std::multimap rooms; extern std::deque complete; extern ctx::context eval_context; extern ctx::context request_context; extern hookfn hook; extern conf::item auth_timeout; extern conf::item timeout; extern conf::item enable; static bool timedout(const request &, const time_t &now); static string_view select_origin(request &, const string_view &); static string_view select_random_origin(request &); static void finish(request &); static void retry(request &); static bool start(request &, m::v1::event::opts &); static bool start(request &); static bool handle(request &); template static bool submit(const event::id &, const room::id &, const size_t &bufsz = 8_KiB, args&&...); static void eval_handle(const decltype(requests)::iterator &); static void eval_handle(); static void eval_worker(); static void request_handle(const decltype(requests)::iterator &); static void request_handle(); static size_t request_cleanup(); static void request_worker(); static void hook_handle_prev(const event &, vm::eval &, evaltab &, const room &); static void hook_handle_auth(const event &, vm::eval &, evaltab &, const room &); static void hook_handle(const event &, vm::eval &); static void init(); static void fini(); } struct ircd::m::fetch::evaltab { size_t auth_count {0}; size_t auth_exists {0}; size_t prev_count {0}; size_t prev_exists {0}; size_t prev_fetching {0}; size_t prev_fetched {0}; }; template bool ircd::m::fetch::submit(const m::event::id &event_id, const m::room::id &room_id, const size_t &bufsz, args&&... a) { auto it { requests.lower_bound(string_view(event_id)) }; if(it != end(requests) && it->event_id == event_id) { assert(it->room_id == room_id); return false; } else try { it = requests.emplace_hint(it, room_id, event_id, bufsz, std::forward(a)...); auto &request(const_cast(*it)); while(!start(request)) request.origin = {}; return true; } catch(const std::exception &e) { log::error { m::log, "Failed to start any fetch for %s in %s :%s", string_view{event_id}, string_view{room_id}, e.what(), }; requests.erase(it); return false; }; }