0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2024-12-28 00:14:07 +01:00

modules/client/rooms/relations: Improve w/ chunked encoding; visibility checks; event append. (#126)

This commit is contained in:
Jason Volk 2019-08-14 22:09:39 -07:00
parent 7420bf1156
commit 08120bb194

View file

@ -12,6 +12,15 @@
using namespace ircd; using namespace ircd;
static void
relations_chunk(client &client,
const resource::request &request,
const m::room::id &room_id,
const m::event::id &event_id,
const string_view &rel_type,
const string_view &type,
json::stack::array &chunk);
resource::response resource::response
get__relations(client &client, get__relations(client &client,
const resource::request &request, const resource::request &request,
@ -20,7 +29,7 @@ get__relations(client &client,
if(!m::exists(room_id)) if(!m::exists(room_id))
throw m::NOT_FOUND throw m::NOT_FOUND
{ {
"Cannot take a report about %s which is not found.", "Cannot find relations in %s which is not found.",
string_view{room_id} string_view{room_id}
}; };
@ -42,114 +51,141 @@ get__relations(client &client,
string_view{event_id} string_view{event_id}
}; };
if(request.parv.size() < 4) // Get the rel_type path parameter.
// Note: Not requiring path parameter; empty will mean all types.
char rel_type_buf[m::event::TYPE_MAX_SIZE];
if((false) && request.parv.size() < 4)
throw m::NEED_MORE_PARAMS throw m::NEED_MORE_PARAMS
{ {
"relation first path parameter required" "relation rel_type path parameter required"
}; };
if(request.parv.size() < 5) const string_view &rel_type
{
url::decode(rel_type_buf, request.parv[3])
};
// Get the alleged type path parameter.
// Note: Not requiring this path parameter either; it is not clear yet
// what this parameter actually means.
char type_buf[m::event::TYPE_MAX_SIZE];
if((false) && request.parv.size() < 5)
throw m::NEED_MORE_PARAMS throw m::NEED_MORE_PARAMS
{ {
"relation second path parameter required" "relation ?type? path parameter required"
}; };
char rbuf[2][256]; const string_view &type
const string_view r[2]
{ {
url::decode(rbuf[0], request.parv[3]), url::decode(type_buf, request.parv[4])
url::decode(rbuf[1], request.parv[4]),
}; };
const unique_buffer<mutable_buffer> buf resource::response::chunked response
{ {
96_KiB client, http::OK
};
json::stack out
{
response.buf, response.flusher()
}; };
json::stack out{buf};
json::stack::object top json::stack::object top
{ {
out out
}; };
m::event::idx next_idx
{
index(event_id, std::nothrow)
};
json::stack::array chunk json::stack::array chunk
{ {
top, "chunk" top, "chunk"
}; };
const m::event::fetch event relations_chunk(client, request, room_id, event_id, rel_type, type, chunk);
return response;
}
void
relations_chunk(client &client,
const resource::request &request,
const m::room::id &room_id,
const m::event::id &event_id,
const string_view &rel_type,
const string_view &type,
json::stack::array &chunk)
try
{
const auto append{[&chunk, &request]
(const auto &event_idx, const m::event &event)
{ {
next_idx, std::nothrow m::event::append::opts opts;
opts.event_idx = &event_idx;
opts.user_id = &request.user_id;
opts.query_txnid = false;
m::event::append
{
chunk, event, opts
};
}};
m::event::idx event_idx
{
index(event_id, std::nothrow)
}; };
if(event.valid) m::event::fetch event
chunk.append(event);
const auto each_ref{[&r, &next_idx, &chunk]
(const m::event::idx &event_idx, const m::dbs::ref &)
{
const m::event::fetch event
{ {
event_idx, std::nothrow event_idx, std::nothrow
}; };
if(!event.valid) const auto each_ref{[&event, &append, &rel_type, &request]
(const m::event::idx &event_idx, const m::dbs::ref &)
{
if(!seek(event, event_idx, std::nothrow))
return true; return true;
const json::object &m_relates_to const json::object &m_relates_to
{ {
json::get<"content"_>(event).get("m.relates_to") at<"content"_>(event).get("m.relates_to")
}; };
const json::string &rel_type const json::string &_rel_type
{ {
m_relates_to.get("rel_type") m_relates_to.get("rel_type")
}; };
if(rel_type != r[0]) if(_rel_type != rel_type)
return true; return true;
chunk.append(event); if(!visible(event, request.user_id))
next_idx = event_idx; return true;
return false;
append(event_idx, event);
return true;
}}; }};
while(next_idx) try if(!event.valid)
{ return;
const m::event::refs refs
{
next_idx
};
if(refs.for_each(m::dbs::ref::M_RELATES, each_ref)) if(!visible(event, request.user_id))
break; return;
}
catch(const std::exception &e) // Send the original event
{ append(event_idx, event);
// Send all the referencees
const m::event::refs refs{event_idx};
refs.for_each(m::dbs::ref::M_RELATES, each_ref);
}
catch(const std::exception &e)
{
log::error log::error
{ {
m::log, "relation trace from %s on %ld :%s", m::log, "relations in %s for %s rel_type:%s type:%s by %s :%s",
string_view{room_id},
string_view{event_id}, string_view{event_id},
next_idx, rel_type,
e.what() type,
}; string_view{request.user_id},
e.what(),
break;
}
chunk.~array();
top.~object();
return resource::response
{
client, json::object
{
out.completed()
}
}; };
} }