construct/modules/client/rooms/initialsync.cc

302 lines
5.6 KiB
C++

// 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;
static void
get__initialsync_remote(client &client,
const m::resource::request &request,
const m::room &room);
static void
get__initialsync_local(client &client,
const m::resource::request &request,
const m::room &room,
const m::user &user,
json::stack::object &out);
static conf::item<size_t>
initial_backfill
{
{ "name", "ircd.client.rooms.initialsync.backfill" },
{ "default", 20L },
};
static conf::item<size_t>
buffer_size
{
{ "name", "ircd.client.rooms.initialsync.buffer_size" },
{ "default", long(128_KiB) },
};
static conf::item<size_t>
flush_hiwat
{
{ "name", "ircd.client.rooms.initialsync.flush.hiwat" },
{ "default", long(32_KiB) },
};
m::resource::response
get__initialsync(client &client,
const m::resource::request &request,
const m::room::id &room_id)
{
const m::room room
{
room_id
};
if(!exists(room))
{
if(!my(room))
get__initialsync_remote(client, request, room);
else
throw m::NOT_FOUND
{
"room_id '%s' does not exist.", string_view{room_id}
};
}
m::resource::response::chunked response
{
client, http::OK, buffer_size
};
json::stack out
{
response.buf, response.flusher(), size_t(flush_hiwat)
};
json::stack::object top
{
out
};
get__initialsync_local(client, request, room, request.user_id, top);
return response;
}
void
get__initialsync_local(client &client,
const m::resource::request &request,
const m::room &room,
const m::user &user,
json::stack::object &out)
{
const m::user::room user_room
{
user
};
const auto room_depth
{
m::depth(std::nothrow, room)
};
char membership_buf[m::room::MEMBERSHIP_MAX_SIZE];
json::stack::member
{
out, "membership", membership(membership_buf, room, request.user_id)
};
json::stack::member
{
out, "visibility", m::rooms::summary::has(room)? "public"_sv: "private"_sv
};
const m::user::room_account_data room_account_data
{
user, room
};
json::stack::array account_data
{
out, "account_data"
};
room_account_data.for_each([&account_data]
(const string_view &type, const json::object &content)
{
json::stack::object object
{
account_data
};
json::stack::member
{
object, "type", type
};
json::stack::member
{
object, "content", content
};
return true;
});
const m::user::room_tags room_tags
{
user, room
};
json::stack::object tag
{
account_data
};
json::stack::member
{
tag, "type", "m.tag"
};
json::stack::object tag_content
{
tag, "content"
};
json::stack::object tags
{
tag_content, "tags"
};
room_tags.for_each([&tags]
(const string_view &type, const json::object &content)
{
json::stack::member
{
tags, type, content
};
return true;
});
tags.~object();
tag_content.~object();
tag.~object();
account_data.~array();
json::stack::array state
{
out, "state"
};
m::event::fetch state_event;
m::room::state{room}.for_each([&]
(const auto &type, const auto &state_key, const m::event::idx &event_idx)
{
if(!seek(std::nothrow, state_event, event_idx))
return true;
if(!visible(state_event, user.user_id))
return true;
m::event::append
{
state, state_event,
{
.event_idx = event_idx,
.user_id = user.user_id,
.user_room_id = user_room.room_id,
.room_depth = room_depth,
.query_txnid = false,
}
};
return true;
});
state.~array();
m::room::events it{room};
json::stack::object messages
{
out, "messages"
};
if(it)
json::stack::member
{
messages, "start", m::event_id(it.event_idx())
};
const size_t limit
{
std::min(request.query.get("limit", size_t(initial_backfill)), 256UL)
};
// seek down first to give events in chronological order.
for(size_t i(0); it && i <= limit; --it, ++i);
if(it)
json::stack::member
{
messages, "end", m::event_id(it.event_idx())
};
json::stack::array chunk
{
messages, "chunk"
};
m::event::fetch event;
for(; it; ++it)
{
const auto &event_idx(it.event_idx());
if(!seek(std::nothrow, event, event_idx))
continue;
if(!visible(event, user.user_id))
continue;
m::event::append
{
chunk, event,
{
.event_idx = event_idx,
.user_id = user.user_id,
.user_room_id = user_room.room_id,
.room_depth = room_depth,
.query_txnid = true,
}
};
}
}
void
get__initialsync_remote(client &client,
const m::resource::request &request,
const m::room &room)
{
const m::room::origins origins{room};
char server_buf[384];
const string_view server
{
origins.random(server_buf, [](const string_view &origin)
{
return !my_host(origin);
})
};
const auto remote
{
server?: room.room_id.host()
};
const auto head
{
m::room::head::fetch::one(room, remote, request.user_id)
};
m::room room_{room};
room_.event_id = head;
//m::fetch::state_ids(room_);
}