mirror of
https://github.com/matrix-construct/construct
synced 2024-12-27 07:54:05 +01:00
645ce4d8e2
fixes #47.
192 lines
3.2 KiB
C++
192 lines
3.2 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;
|
|
|
|
mapi::header
|
|
IRCD_MODULE
|
|
{
|
|
"Federation :Request a prototype for creating a join event."
|
|
};
|
|
|
|
const string_view
|
|
make_join_description
|
|
{R"(
|
|
|
|
Sends a partial event to the remote with enough information for them to
|
|
create a join event 'in the blind' for one of their users.
|
|
|
|
)"};
|
|
|
|
resource
|
|
make_join_resource
|
|
{
|
|
"/_matrix/federation/v1/make_join/",
|
|
{
|
|
make_join_description,
|
|
resource::DIRECTORY
|
|
}
|
|
};
|
|
|
|
resource::response
|
|
get__make_join(client &client,
|
|
const resource::request &request)
|
|
{
|
|
if(request.parv.size() < 1)
|
|
throw m::NEED_MORE_PARAMS
|
|
{
|
|
"room_id path parameter required"
|
|
};
|
|
|
|
m::room::id::buf room_id
|
|
{
|
|
url::decode(room_id, request.parv[0])
|
|
};
|
|
|
|
if(request.parv.size() < 2)
|
|
throw m::NEED_MORE_PARAMS
|
|
{
|
|
"user_id path parameter required"
|
|
};
|
|
|
|
m::user::id::buf user_id
|
|
{
|
|
url::decode(user_id, request.parv[1])
|
|
};
|
|
|
|
if(user_id.host() != request.origin)
|
|
throw m::ACCESS_DENIED
|
|
{
|
|
"You are not permitted to spoof users on other hosts."
|
|
};
|
|
|
|
const m::room room
|
|
{
|
|
room_id
|
|
};
|
|
|
|
if(!exists(room))
|
|
throw m::NOT_FOUND
|
|
{
|
|
"Room %s is not known here.",
|
|
string_view{room_id}
|
|
};
|
|
|
|
if(m::room::server_acl::enable_read && !m::room::server_acl::check(room_id, request.node_id))
|
|
throw m::ACCESS_DENIED
|
|
{
|
|
"You are not permitted by the room's server access control list."
|
|
};
|
|
|
|
if(!room.visible(user_id))
|
|
throw m::ACCESS_DENIED
|
|
{
|
|
"You are not permitted to view the room at this event."
|
|
};
|
|
|
|
const unique_buffer<mutable_buffer> buf
|
|
{
|
|
8_KiB
|
|
};
|
|
|
|
json::stack out{buf};
|
|
json::stack::object top{out};
|
|
json::stack::object event
|
|
{
|
|
top, "event"
|
|
};
|
|
|
|
{
|
|
const m::room::auth auth{room};
|
|
json::stack::array auth_events
|
|
{
|
|
event, "auth_events"
|
|
};
|
|
|
|
static const string_view types[]
|
|
{
|
|
"m.room.create",
|
|
"m.room.join_rules",
|
|
"m.room.power_levels",
|
|
"m.room.member",
|
|
};
|
|
|
|
auth.make_refs(auth_events, types, user_id);
|
|
}
|
|
|
|
json::stack::member
|
|
{
|
|
event, "content", R"({"membership":"join"})"
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "depth", json::value(m::depth(room) + 1L)
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "origin", request.origin
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "origin_server_ts", json::value(time<milliseconds>())
|
|
};
|
|
|
|
{
|
|
const m::room::head head{room};
|
|
json::stack::array prev_events
|
|
{
|
|
event, "prev_events"
|
|
};
|
|
|
|
head.make_refs(prev_events, 32, true);
|
|
}
|
|
|
|
json::stack::member
|
|
{
|
|
event, "room_id", room.room_id
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "sender", user_id
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "state_key", user_id
|
|
};
|
|
|
|
json::stack::member
|
|
{
|
|
event, "type", "m.room.member"
|
|
};
|
|
|
|
event.~object();
|
|
top.~object();
|
|
return resource::response
|
|
{
|
|
client, json::object
|
|
{
|
|
out.completed()
|
|
}
|
|
};
|
|
}
|
|
|
|
resource::method
|
|
method_get
|
|
{
|
|
make_join_resource, "GET", get__make_join,
|
|
{
|
|
method_get.VERIFY_ORIGIN
|
|
}
|
|
};
|