mirror of
https://github.com/matrix-construct/construct
synced 2025-01-15 17:16:49 +01:00
163 lines
3.3 KiB
C++
163 lines
3.3 KiB
C++
// Matrix Construct
|
|
//
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
// Copyright (C) 2016-2019 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.
|
|
|
|
namespace ircd::m
|
|
{
|
|
static bool need_bootstrap(const room::id &);
|
|
}
|
|
|
|
ircd::m::event::id::buf
|
|
ircd::m::join(const room::alias &room_alias,
|
|
const user::id &user_id)
|
|
{
|
|
if(unlikely(!my(user_id)))
|
|
throw panic
|
|
{
|
|
"Can only join my users."
|
|
};
|
|
|
|
const room::id::buf room_id
|
|
{
|
|
m::room_id(room_alias)
|
|
};
|
|
|
|
if(need_bootstrap(room_id))
|
|
{
|
|
m::event::id::buf ret;
|
|
m::room::bootstrap
|
|
{
|
|
ret, room_id, user_id, room_alias.host()
|
|
};
|
|
|
|
return ret;
|
|
}
|
|
|
|
const m::room room
|
|
{
|
|
room_id
|
|
};
|
|
|
|
return m::join(room, user_id);
|
|
}
|
|
|
|
ircd::m::event::id::buf
|
|
ircd::m::join(const m::room &room,
|
|
const m::id::user &user_id)
|
|
{
|
|
if(unlikely(!my(user_id)))
|
|
throw panic
|
|
{
|
|
"Can only join my users."
|
|
};
|
|
|
|
// Branch for when nothing is known about the room.
|
|
if(need_bootstrap(room))
|
|
{
|
|
// The bootstrap condcts a blocking make_join and issues a join
|
|
// event, returning the event_id; afterward asynchronously it will
|
|
// attempt a send_join, and then process the room events.
|
|
m::event::id::buf ret;
|
|
m::room::bootstrap
|
|
{
|
|
ret, room.room_id, user_id, room.room_id.host() //TODO: host
|
|
};
|
|
|
|
return ret;
|
|
}
|
|
|
|
if(membership(room, user_id, "join"))
|
|
{
|
|
const auto &event_idx
|
|
{
|
|
room.get(std::nothrow, "m.room.member", user_id)
|
|
};
|
|
|
|
const event::id::buf event_id
|
|
{
|
|
event_idx?
|
|
m::event_id(std::nothrow, event_idx):
|
|
event::id::buf{}
|
|
};
|
|
|
|
//TODO: check duplicate content
|
|
//if(event_id)
|
|
// return event_id;
|
|
}
|
|
|
|
json::iov event;
|
|
json::iov content;
|
|
const json::iov::push push[]
|
|
{
|
|
{ event, { "type", "m.room.member" }},
|
|
{ event, { "sender", user_id }},
|
|
{ event, { "state_key", user_id }},
|
|
{ content, { "membership", "join" }},
|
|
};
|
|
|
|
const m::user user{user_id};
|
|
const m::user::profile profile{user};
|
|
|
|
char displayname_buf[256];
|
|
const string_view displayname
|
|
{
|
|
profile.get(displayname_buf, "displayname")
|
|
};
|
|
|
|
char avatar_url_buf[256];
|
|
const string_view avatar_url
|
|
{
|
|
profile.get(avatar_url_buf, "avatar_url")
|
|
};
|
|
|
|
const json::iov::add _displayname
|
|
{
|
|
content, !empty(displayname),
|
|
{
|
|
"displayname", [&displayname]() -> json::value
|
|
{
|
|
return displayname;
|
|
}
|
|
}
|
|
};
|
|
|
|
const json::iov::add _avatar_url
|
|
{
|
|
content, !empty(avatar_url),
|
|
{
|
|
"avatar_url", [&avatar_url]() -> json::value
|
|
{
|
|
return avatar_url;
|
|
}
|
|
}
|
|
};
|
|
|
|
return commit(room, event, content);
|
|
}
|
|
|
|
bool
|
|
ircd::m::need_bootstrap(const room::id &room_id)
|
|
{
|
|
// No bootstrap for my rooms
|
|
//TODO: issue for clustering
|
|
if(my(room_id))
|
|
return false;
|
|
|
|
// We have nothing for the room
|
|
if(!exists(room_id))
|
|
return true;
|
|
|
|
// No users are currently joined from this server;
|
|
//TODO: bootstrap shouldn't have to be used to re-sync a room where we have
|
|
//TODO: some partial state, so this condition should be eliminated.
|
|
if(local_joined(room_id) == 0)
|
|
return true;
|
|
|
|
return false;
|
|
}
|