mirror of
https://github.com/matrix-construct/construct
synced 2024-11-25 16:22:35 +01:00
modules/federation/federation: Split apart feds head; add feds perspective.
This commit is contained in:
parent
af74329e4a
commit
b00da0f95f
1 changed files with 166 additions and 59 deletions
|
@ -20,6 +20,8 @@ namespace ircd::m::feds
|
||||||
{
|
{
|
||||||
struct version;
|
struct version;
|
||||||
struct state;
|
struct state;
|
||||||
|
struct head;
|
||||||
|
struct perspective;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ircd::m::feds::version
|
struct ircd::m::feds::version
|
||||||
|
@ -131,7 +133,7 @@ struct ircd::m::feds::state
|
||||||
opts.dynamic = true;
|
opts.dynamic = true;
|
||||||
opts.ids_only = true;
|
opts.ids_only = true;
|
||||||
opts.event_id = event_id;
|
opts.event_id = event_id;
|
||||||
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
opts.remote = string_view{strlcpy{this->origin, origin}};
|
||||||
return m::v1::state{room_id, mutable_buffer{buf}, std::move(opts)};
|
return m::v1::state{room_id, mutable_buffer{buf}, std::move(opts)};
|
||||||
}()}
|
}()}
|
||||||
{}
|
{}
|
||||||
|
@ -293,30 +295,29 @@ feds__event(const m::event::id &event_id, std::ostream &out)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void
|
struct ircd::m::feds::head
|
||||||
feds__head(const m::room::id &room_id,
|
:m::v1::make_join
|
||||||
const m::user::id &user_id,
|
|
||||||
std::ostream &out)
|
|
||||||
{
|
{
|
||||||
struct req
|
char origin[256];
|
||||||
:m::v1::make_join
|
char buf[16_KiB];
|
||||||
|
|
||||||
|
head(const m::room::id &room_id, const m::user::id &user_id, const string_view &origin)
|
||||||
|
:m::v1::make_join{[this, &room_id, &user_id, &origin]
|
||||||
{
|
{
|
||||||
char origin[256];
|
m::v1::make_join::opts opts;
|
||||||
char buf[16_KiB];
|
opts.dynamic = false;
|
||||||
|
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
||||||
|
return m::v1::make_join{room_id, user_id, mutable_buffer{buf}, std::move(opts)};
|
||||||
|
}()}
|
||||||
|
{}
|
||||||
|
};
|
||||||
|
|
||||||
req(const m::room::id &room_id, const m::user::id &user_id, const string_view &origin)
|
std::list<m::feds::head>
|
||||||
:m::v1::make_join{[this, &room_id, &user_id, &origin]
|
feds__head(const m::room::id &room_id,
|
||||||
{
|
const m::user::id &user_id)
|
||||||
m::v1::make_join::opts opts;
|
{
|
||||||
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
std::list<m::feds::head> reqs;
|
||||||
return m::v1::make_join{room_id, user_id, mutable_buffer{buf}, std::move(opts)};
|
m::room::origins{room_id}.for_each([&reqs, &room_id, &user_id]
|
||||||
}()}
|
|
||||||
{}
|
|
||||||
};
|
|
||||||
|
|
||||||
std::list<req> reqs;
|
|
||||||
const m::room::origins origins{room_id};
|
|
||||||
origins.for_each([&out, &room_id, &user_id, &reqs]
|
|
||||||
(const string_view &origin)
|
(const string_view &origin)
|
||||||
{
|
{
|
||||||
const auto emsg
|
const auto emsg
|
||||||
|
@ -324,59 +325,65 @@ feds__head(const m::room::id &room_id,
|
||||||
ircd::server::errmsg(origin)
|
ircd::server::errmsg(origin)
|
||||||
};
|
};
|
||||||
|
|
||||||
if(emsg)
|
if(!emsg) try
|
||||||
{
|
|
||||||
out << "! " << origin << " " << emsg << std::endl;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
{
|
||||||
reqs.emplace_back(room_id, user_id, origin);
|
reqs.emplace_back(room_id, user_id, origin);
|
||||||
}
|
}
|
||||||
catch(const std::exception &e)
|
catch(const std::exception &)
|
||||||
{
|
{
|
||||||
out << "! " << origin << " " << e.what() << std::endl;
|
return;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
auto all
|
return std::move(reqs);
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C" void
|
||||||
|
feds__head(const m::room::id &room_id,
|
||||||
|
const m::user::id &user_id,
|
||||||
|
const milliseconds &timeout,
|
||||||
|
const m::feds::state::closure &closure)
|
||||||
|
{
|
||||||
|
auto reqs
|
||||||
{
|
{
|
||||||
ctx::when_all(begin(reqs), end(reqs))
|
feds__head(room_id, user_id)
|
||||||
};
|
};
|
||||||
|
|
||||||
all.wait(30s, std::nothrow);
|
auto when
|
||||||
|
|
||||||
for(auto &req : reqs) try
|
|
||||||
{
|
{
|
||||||
if(req.wait(1ms, std::nothrow))
|
now<steady_point>() + timeout
|
||||||
|
};
|
||||||
|
|
||||||
|
while(!reqs.empty())
|
||||||
|
{
|
||||||
|
auto next
|
||||||
|
{
|
||||||
|
ctx::when_any(begin(reqs), end(reqs))
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!next.wait_until(when, std::nothrow))
|
||||||
|
break;
|
||||||
|
|
||||||
|
const auto it
|
||||||
|
{
|
||||||
|
next.get()
|
||||||
|
};
|
||||||
|
|
||||||
|
auto &req{*it}; try
|
||||||
{
|
{
|
||||||
const auto code{req.get()};
|
const auto code{req.get()};
|
||||||
const json::object &response{req};
|
const json::object &response{req};
|
||||||
const json::object &event{response.at("event")};
|
const json::object &event{response.at("event")};
|
||||||
const json::array prev_events
|
if(!closure(req.origin, {}, event))
|
||||||
{
|
break;
|
||||||
event.at("prev_events")
|
|
||||||
};
|
|
||||||
|
|
||||||
out << "+ " << std::setw(40) << std::left << req.origin;
|
|
||||||
out << " " << event.at("depth");
|
|
||||||
for(const json::array prev_event : prev_events)
|
|
||||||
{
|
|
||||||
const auto &prev_event_id
|
|
||||||
{
|
|
||||||
unquote(prev_event.at(0))
|
|
||||||
};
|
|
||||||
|
|
||||||
out << " " << string_view{prev_event_id};
|
|
||||||
};
|
|
||||||
out << std::endl;
|
|
||||||
}
|
}
|
||||||
else cancel(req);
|
catch(const std::exception &)
|
||||||
}
|
{
|
||||||
catch(const std::exception &e)
|
if(!closure(req.origin, std::current_exception(), {}))
|
||||||
{
|
break;
|
||||||
out << "- " << req.origin << " " << e.what() << std::endl;
|
}
|
||||||
|
|
||||||
|
reqs.erase(it);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -477,3 +484,103 @@ feds__backfill(const m::room::id &room_id,
|
||||||
out << "| " << req.origin << std::endl;
|
out << "| " << req.origin << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct ircd::m::feds::perspective
|
||||||
|
:m::v1::key::query
|
||||||
|
{
|
||||||
|
using closure = std::function<bool (const string_view &, std::exception_ptr, const json::array &)>;
|
||||||
|
|
||||||
|
char origin[256];
|
||||||
|
char buf[24_KiB];
|
||||||
|
|
||||||
|
perspective(const string_view &origin,
|
||||||
|
const m::v1::key::server_key &server_key)
|
||||||
|
:m::v1::key::query{[&]
|
||||||
|
{
|
||||||
|
m::v1::key::opts opts;
|
||||||
|
opts.dynamic = false;
|
||||||
|
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
||||||
|
return m::v1::key::query
|
||||||
|
{
|
||||||
|
{&server_key, 1}, mutable_buffer{buf}, std::move(opts)
|
||||||
|
};
|
||||||
|
}()}
|
||||||
|
{}
|
||||||
|
|
||||||
|
perspective(perspective &&) = delete;
|
||||||
|
perspective(const perspective &) = delete;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::list<m::feds::perspective>
|
||||||
|
feds__perspective(const m::room::id &room_id,
|
||||||
|
const m::v1::key::server_key &server_key)
|
||||||
|
{
|
||||||
|
std::list<m::feds::perspective> reqs;
|
||||||
|
m::room::origins{room_id}.for_each([&reqs, &server_key]
|
||||||
|
(const string_view &origin)
|
||||||
|
{
|
||||||
|
const auto emsg
|
||||||
|
{
|
||||||
|
ircd::server::errmsg(origin)
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!emsg) try
|
||||||
|
{
|
||||||
|
reqs.emplace_back(origin, server_key);
|
||||||
|
}
|
||||||
|
catch(const std::exception &)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return std::move(reqs);
|
||||||
|
}
|
||||||
|
|
||||||
|
extern "C" void
|
||||||
|
feds__perspective(const m::room::id &room_id,
|
||||||
|
const m::v1::key::server_key &server_key, // pair<server_name, key_id>
|
||||||
|
const milliseconds &timeout,
|
||||||
|
const m::feds::perspective::closure &closure)
|
||||||
|
{
|
||||||
|
auto reqs
|
||||||
|
{
|
||||||
|
feds__perspective(room_id, server_key)
|
||||||
|
};
|
||||||
|
|
||||||
|
auto when
|
||||||
|
{
|
||||||
|
now<steady_point>() + timeout
|
||||||
|
};
|
||||||
|
|
||||||
|
while(!reqs.empty())
|
||||||
|
{
|
||||||
|
auto next
|
||||||
|
{
|
||||||
|
ctx::when_any(begin(reqs), end(reqs))
|
||||||
|
};
|
||||||
|
|
||||||
|
if(!next.wait_until(when, std::nothrow))
|
||||||
|
break;
|
||||||
|
|
||||||
|
const auto it
|
||||||
|
{
|
||||||
|
next.get()
|
||||||
|
};
|
||||||
|
|
||||||
|
auto &req{*it}; try
|
||||||
|
{
|
||||||
|
const auto code{req.get()};
|
||||||
|
const json::array &response{req};
|
||||||
|
if(!closure(req.origin, {}, response))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
catch(const std::exception &)
|
||||||
|
{
|
||||||
|
if(!closure(req.origin, std::current_exception(), {}))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
reqs.erase(it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue