2018-04-28 03:02:42 +02: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.
|
|
|
|
|
|
|
|
using namespace ircd;
|
|
|
|
|
|
|
|
mapi::header
|
|
|
|
IRCD_MODULE
|
|
|
|
{
|
|
|
|
"Federation :General Library and Utils"
|
|
|
|
};
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
namespace ircd::m::feds
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
template<class T> struct request;
|
2018-04-30 14:43:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
template<class T>
|
|
|
|
struct ircd::m::feds::request
|
|
|
|
:T
|
2018-04-30 14:43:56 +02:00
|
|
|
{
|
|
|
|
char origin[256];
|
2019-04-11 14:29:59 +02:00
|
|
|
char buf[8_KiB];
|
2018-04-30 14:43:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
request(const std::function<T (request &)> &);
|
|
|
|
request(request &&) = delete;
|
|
|
|
request(const request &) = delete;
|
2018-04-30 14:43:56 +02:00
|
|
|
};
|
2018-04-28 03:02:42 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
template<class T>
|
|
|
|
ircd::m::feds::request<T>::request(const std::function<T (request &)> &closure)
|
|
|
|
:T{closure(*this)}
|
|
|
|
{}
|
|
|
|
|
|
|
|
bool
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::feds::version(const opts &opts,
|
|
|
|
const closure &closure)
|
2018-04-30 14:43:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
static const auto make_request{[]
|
|
|
|
(auto &request, const auto &origin)
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
m::v1::version::opts opts;
|
|
|
|
opts.dynamic = false;
|
|
|
|
opts.remote = string_view
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
request.origin, strlcpy{request.origin, origin}
|
2018-04-30 14:30:50 +02:00
|
|
|
};
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
return m::v1::version
|
|
|
|
{
|
|
|
|
request.buf, std::move(opts)
|
|
|
|
};
|
|
|
|
}};
|
|
|
|
|
|
|
|
std::list<m::feds::request<m::v1::version>> reqs;
|
|
|
|
m::room::origins{opts.room_id}.for_each([&reqs]
|
|
|
|
(const string_view &origin)
|
|
|
|
{
|
|
|
|
if(!server::errmsg(origin)) try
|
2018-04-30 14:30:50 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
reqs.emplace_back([&origin]
|
|
|
|
(auto &request)
|
|
|
|
{
|
|
|
|
return make_request(request, origin);
|
|
|
|
});
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
2018-04-30 14:43:56 +02:00
|
|
|
catch(const std::exception &)
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
const auto when
|
2018-04-30 14:43:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
now<steady_point>() + opts.timeout
|
2018-04-28 03:02:42 +02:00
|
|
|
};
|
|
|
|
|
2018-05-05 12:22:20 +02:00
|
|
|
while(!reqs.empty())
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2018-05-05 12:22:20 +02:00
|
|
|
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
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
|
|
|
const auto code{req.get()};
|
|
|
|
const json::object &response{req};
|
2019-04-11 14:29:59 +02:00
|
|
|
if(!closure({req.origin, {}, response}))
|
|
|
|
return false;
|
2018-05-05 12:22:20 +02:00
|
|
|
}
|
|
|
|
catch(const std::exception &)
|
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
const ctx::exception_handler eptr;
|
|
|
|
if(!closure({req.origin, eptr}))
|
|
|
|
return false;
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
2018-05-05 12:22:20 +02:00
|
|
|
|
|
|
|
reqs.erase(it);
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
2019-04-11 14:29:59 +02:00
|
|
|
|
|
|
|
return true;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
bool
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::feds::state(const opts &opts,
|
|
|
|
const closure &closure)
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
const auto make_request{[&opts]
|
|
|
|
(auto &request, const auto &origin)
|
|
|
|
{
|
|
|
|
m::v1::state::opts v1opts;
|
|
|
|
v1opts.dynamic = true;
|
|
|
|
v1opts.ids_only = opts.ids;
|
|
|
|
v1opts.event_id = opts.event_id;
|
|
|
|
v1opts.remote = string_view
|
|
|
|
{
|
|
|
|
strlcpy{request.origin, origin}
|
|
|
|
};
|
2018-05-05 12:23:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
return m::v1::state
|
|
|
|
{
|
|
|
|
opts.room_id, request.buf, std::move(v1opts)
|
|
|
|
};
|
|
|
|
}};
|
2018-05-05 12:23:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
std::list<m::feds::request<m::v1::state>> reqs;
|
|
|
|
m::room::origins{opts.room_id}.for_each([&reqs, &make_request]
|
|
|
|
(const string_view &origin)
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
if(!server::errmsg(origin)) try
|
|
|
|
{
|
|
|
|
reqs.emplace_back([&origin, &make_request]
|
|
|
|
(auto &request)
|
|
|
|
{
|
|
|
|
return make_request(request, origin);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
catch(const std::exception &)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
2018-05-05 12:23:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
const auto when
|
|
|
|
{
|
|
|
|
now<steady_point>() + opts.timeout
|
|
|
|
};
|
2018-05-05 12:23:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
while(!reqs.empty())
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
auto next
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
ctx::when_any(begin(reqs), end(reqs))
|
2018-05-05 12:23:56 +02:00
|
|
|
};
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
if(!next.wait_until(when, std::nothrow))
|
|
|
|
break;
|
|
|
|
|
|
|
|
const auto it
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
next.get()
|
|
|
|
};
|
|
|
|
|
|
|
|
auto &req{*it}; try
|
|
|
|
{
|
|
|
|
const auto code{req.get()};
|
|
|
|
const json::object &response{req};
|
|
|
|
if(!closure({req.origin, {}, response}))
|
|
|
|
return false;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
catch(const std::exception &)
|
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
const ctx::exception_handler eptr;
|
|
|
|
if(!closure({req.origin, eptr}))
|
|
|
|
return false;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
reqs.erase(it);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
bool
|
|
|
|
IRCD_MODULE_EXPORT
|
|
|
|
ircd::m::feds::head(const opts &opts,
|
|
|
|
const closure &closure)
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
const auto make_request{[&opts]
|
|
|
|
(auto &request, const auto &origin)
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
m::v1::make_join::opts v1opts;
|
|
|
|
v1opts.remote = string_view
|
|
|
|
{
|
|
|
|
strlcpy{request.origin, origin}
|
|
|
|
};
|
2018-05-05 12:23:56 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
return m::v1::make_join
|
|
|
|
{
|
|
|
|
opts.room_id, opts.user_id, request.buf, std::move(v1opts)
|
|
|
|
};
|
|
|
|
}};
|
|
|
|
|
|
|
|
std::list<m::feds::request<m::v1::make_join>> reqs;
|
|
|
|
m::room::origins{opts.room_id}.for_each([&reqs, &make_request]
|
|
|
|
(const string_view &origin)
|
2018-05-05 12:23:56 +02:00
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
if(!server::errmsg(origin)) try
|
|
|
|
{
|
|
|
|
reqs.emplace_back([&origin, &make_request]
|
|
|
|
(auto &request)
|
|
|
|
{
|
|
|
|
return make_request(request, origin);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
catch(const std::exception &)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const auto when
|
|
|
|
{
|
|
|
|
now<steady_point>() + opts.timeout
|
2018-05-05 12:23:56 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
while(!reqs.empty())
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2018-05-05 12:23:56 +02:00
|
|
|
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::object &response{req};
|
2019-04-11 14:29:59 +02:00
|
|
|
if(!closure({req.origin, {}, response}))
|
|
|
|
return false;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
catch(const std::exception &)
|
|
|
|
{
|
2019-04-11 14:29:59 +02:00
|
|
|
const ctx::exception_handler eptr;
|
|
|
|
if(!closure({req.origin, eptr}))
|
|
|
|
return false;
|
2018-05-05 12:23:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
reqs.erase(it);
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
2019-04-11 14:29:59 +02:00
|
|
|
|
|
|
|
return true;
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void
|
|
|
|
feds__event(const m::event::id &event_id, std::ostream &out)
|
|
|
|
{
|
|
|
|
const m::room::id::buf room_id{[&event_id]
|
|
|
|
{
|
|
|
|
const m::event::fetch event
|
|
|
|
{
|
|
|
|
event_id
|
|
|
|
};
|
|
|
|
|
|
|
|
return m::room::id::buf{at<"room_id"_>(event)};
|
|
|
|
}()};
|
|
|
|
|
|
|
|
struct req
|
|
|
|
:m::v1::event
|
|
|
|
{
|
|
|
|
char origin[256];
|
|
|
|
char buf[96_KiB];
|
|
|
|
|
2018-04-30 14:30:50 +02:00
|
|
|
req(const m::event::id &event_id, const string_view &origin)
|
|
|
|
:m::v1::event{[&]
|
|
|
|
{
|
|
|
|
m::v1::event::opts opts;
|
|
|
|
opts.dynamic = false;
|
|
|
|
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
|
|
|
return m::v1::event{event_id, mutable_buffer{buf}, std::move(opts)};
|
|
|
|
}()}
|
2018-04-28 03:02:42 +02:00
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::list<req> reqs;
|
|
|
|
const m::room::origins origins{room_id};
|
|
|
|
origins.for_each([&out, &event_id, &reqs]
|
|
|
|
(const string_view &origin)
|
|
|
|
{
|
2018-04-30 14:30:50 +02:00
|
|
|
const auto emsg
|
2018-04-28 03:02:42 +02:00
|
|
|
{
|
2018-04-30 14:30:50 +02:00
|
|
|
ircd::server::errmsg(origin)
|
|
|
|
};
|
|
|
|
|
|
|
|
if(emsg)
|
|
|
|
{
|
|
|
|
out << "! " << origin << " " << emsg << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
reqs.emplace_back(event_id, origin);
|
2018-04-28 03:02:42 +02:00
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
out << "! " << origin << " " << e.what() << std::endl;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
auto all
|
|
|
|
{
|
|
|
|
ctx::when_all(begin(reqs), end(reqs))
|
|
|
|
};
|
|
|
|
|
|
|
|
all.wait(30s, std::nothrow);
|
|
|
|
|
|
|
|
for(auto &req : reqs) try
|
|
|
|
{
|
|
|
|
if(req.wait(1ms, std::nothrow))
|
|
|
|
{
|
|
|
|
const auto code{req.get()};
|
|
|
|
out << "+ " << req.origin << " " << http::status(code) << std::endl;
|
|
|
|
}
|
|
|
|
else cancel(req);
|
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
out << "- " << req.origin << " " << e.what() << std::endl;
|
|
|
|
}
|
|
|
|
}
|
2018-04-30 14:31:38 +02:00
|
|
|
|
2018-04-30 17:51:03 +02:00
|
|
|
extern "C" void
|
|
|
|
feds__backfill(const m::room::id &room_id,
|
|
|
|
const m::event::id &event_id,
|
|
|
|
const size_t &limit,
|
|
|
|
std::ostream &out)
|
|
|
|
{
|
|
|
|
struct req
|
|
|
|
:m::v1::backfill
|
|
|
|
{
|
|
|
|
char origin[256];
|
|
|
|
char buf[16_KiB];
|
|
|
|
|
|
|
|
req(const m::room::id &room_id, const m::event::id &event_id, const string_view &origin, const size_t &limit)
|
|
|
|
:m::v1::backfill{[&]
|
|
|
|
{
|
|
|
|
m::v1::backfill::opts opts;
|
|
|
|
opts.remote = string_view{this->origin, strlcpy(this->origin, origin)};
|
|
|
|
opts.event_id = event_id;
|
|
|
|
opts.limit = limit;
|
|
|
|
opts.dynamic = true;
|
|
|
|
return m::v1::backfill{room_id, mutable_buffer{buf}, std::move(opts)};
|
|
|
|
}()}
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::list<req> reqs;
|
|
|
|
const m::room::origins origins{room_id};
|
|
|
|
origins.for_each([&out, &room_id, &event_id, &limit, &reqs]
|
|
|
|
(const string_view &origin)
|
|
|
|
{
|
|
|
|
const auto emsg
|
|
|
|
{
|
|
|
|
ircd::server::errmsg(origin)
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!emsg) try
|
|
|
|
{
|
|
|
|
reqs.emplace_back(room_id, event_id, origin, limit);
|
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
auto all
|
|
|
|
{
|
|
|
|
ctx::when_all(begin(reqs), end(reqs))
|
|
|
|
};
|
|
|
|
|
|
|
|
all.wait(30s, std::nothrow);
|
|
|
|
|
|
|
|
std::map<string_view, std::set<const req *>> grid;
|
|
|
|
|
|
|
|
for(auto &req : reqs) try
|
|
|
|
{
|
|
|
|
if(req.wait(1ms, std::nothrow))
|
|
|
|
{
|
|
|
|
const auto code{req.get()};
|
|
|
|
const json::object &response{req};
|
|
|
|
const json::array pdus
|
|
|
|
{
|
|
|
|
response.at("pdus")
|
|
|
|
};
|
|
|
|
|
|
|
|
for(const json::object &pdu : pdus)
|
|
|
|
{
|
|
|
|
const auto &event_id
|
|
|
|
{
|
|
|
|
unquote(pdu.at("event_id"))
|
|
|
|
};
|
|
|
|
|
|
|
|
grid[event_id].emplace(&req);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else cancel(req);
|
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
out << "- " << req.origin << " " << e.what() << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t i(0);
|
|
|
|
for(const auto &p : grid)
|
|
|
|
out << i++ << " " << p.first << std::endl;
|
|
|
|
|
|
|
|
for(size_t j(0); j < i; ++j)
|
2018-05-01 07:04:34 +02:00
|
|
|
out << "| " << std::left << std::setw(2) << j;
|
2018-04-30 17:51:03 +02:00
|
|
|
out << "|" << std::endl;
|
|
|
|
|
|
|
|
for(const auto &req : reqs)
|
|
|
|
{
|
|
|
|
for(const auto &p : grid)
|
|
|
|
out << "| " << (p.second.count(&req)? '+' : ' ') << " ";
|
|
|
|
out << "| " << req.origin << std::endl;
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 04:19:22 +02:00
|
|
|
|
2019-04-11 14:29:59 +02:00
|
|
|
namespace ircd::m::feds::v1
|
|
|
|
{
|
|
|
|
struct perspective;
|
|
|
|
}
|
|
|
|
|
2018-08-26 04:40:51 +02:00
|
|
|
struct ircd::m::feds::v1::perspective
|
2018-05-14 04:19:22 +02:00
|
|
|
: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;
|
|
|
|
};
|
|
|
|
|
2018-08-26 04:40:51 +02:00
|
|
|
std::list<m::feds::v1::perspective>
|
2018-05-14 04:19:22 +02:00
|
|
|
feds__perspective(const m::room::id &room_id,
|
|
|
|
const m::v1::key::server_key &server_key)
|
|
|
|
{
|
2018-08-26 04:40:51 +02:00
|
|
|
std::list<m::feds::v1::perspective> reqs;
|
2018-05-14 04:19:22 +02:00
|
|
|
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,
|
2018-08-26 04:40:51 +02:00
|
|
|
const m::feds::v1::perspective::closure &closure)
|
2018-05-14 04:19:22 +02:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|