mirror of
https://github.com/matrix-construct/construct
synced 2025-01-22 12:30:00 +01:00
205 lines
3.9 KiB
C++
205 lines
3.9 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 :Send join event"
|
|
};
|
|
|
|
const string_view
|
|
send_join_description
|
|
{R"(
|
|
|
|
Inject a join event into a room originating from a server without any joined
|
|
users in that room.
|
|
|
|
)"};
|
|
|
|
resource
|
|
send_join_resource
|
|
{
|
|
"/_matrix/federation/v1/send_join/",
|
|
{
|
|
send_join_description,
|
|
resource::DIRECTORY
|
|
}
|
|
};
|
|
|
|
resource::response
|
|
put__send_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(request.parv[0], room_id)
|
|
};
|
|
|
|
if(!my_host(room_id.host()))
|
|
throw m::error
|
|
{
|
|
http::FORBIDDEN, "M_INVALID_ROOM_ID",
|
|
"Can only send_join for rooms on my host '%s'",
|
|
my_host()
|
|
};
|
|
|
|
if(request.parv.size() < 2)
|
|
throw m::NEED_MORE_PARAMS
|
|
{
|
|
"event_id path parameter required"
|
|
};
|
|
|
|
m::event::id::buf event_id
|
|
{
|
|
url::decode(request.parv[1], event_id)
|
|
};
|
|
|
|
const m::event event
|
|
{
|
|
request
|
|
};
|
|
|
|
if(at<"event_id"_>(event) != event_id)
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_MISMATCH_EVENT_ID",
|
|
"ID of event in request body does not match path parameter."
|
|
};
|
|
|
|
if(at<"room_id"_>(event) != room_id)
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_MISMATCH_ROOM_ID",
|
|
"ID of room in request body does not match path parameter."
|
|
};
|
|
|
|
if(json::get<"type"_>(event) != "m.room.member")
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_INVALID_TYPE",
|
|
"Event type must be m.room.member"
|
|
};
|
|
|
|
if(json::get<"membership"_>(event) && json::get<"membership"_>(event) != "join")
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_INVALID_MEMBERSHIP",
|
|
"Event membership state must be 'join'."
|
|
};
|
|
|
|
if(unquote(json::get<"content"_>(event).get("membership")) != "join")
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_INVALID_CONTENT_MEMBERSHIP",
|
|
"Event content.membership state must be 'join'."
|
|
};
|
|
|
|
if(json::get<"origin"_>(event) != request.origin)
|
|
throw m::error
|
|
{
|
|
http::NOT_MODIFIED, "M_MISMATCH_ORIGIN",
|
|
"Event origin must be you."
|
|
};
|
|
|
|
m::vm::opts vmopts;
|
|
vmopts.non_conform.set(m::event::conforms::MISSING_PREV_STATE);
|
|
vmopts.non_conform.set(m::event::conforms::MISSING_MEMBERSHIP);
|
|
m::vm::eval eval
|
|
{
|
|
event, vmopts
|
|
};
|
|
|
|
const m::room::state state
|
|
{
|
|
room_id
|
|
};
|
|
|
|
//TODO: direct to socket
|
|
const unique_buffer<mutable_buffer> buf{2_MiB}; //TODO: XXX
|
|
json::stack out{buf};
|
|
{
|
|
json::stack::array top
|
|
{
|
|
out
|
|
};
|
|
|
|
// First element is the 200
|
|
top.append(json::value(200L));
|
|
|
|
// Second element is the object
|
|
json::stack::object data{top};
|
|
{
|
|
json::stack::member auth_chain_m
|
|
{
|
|
data, "auth_chain"
|
|
};
|
|
|
|
json::stack::array auth_chain_a
|
|
{
|
|
auth_chain_m
|
|
};
|
|
|
|
const auto appender{[&auth_chain_a]
|
|
(const m::event &event)
|
|
{
|
|
auth_chain_a.append(event);
|
|
}};
|
|
|
|
state.get(std::nothrow, "m.room.create", "", appender);
|
|
state.get(std::nothrow, "m.room.power_levels", "", appender);
|
|
state.get(std::nothrow, "m.room.join_rules", "", appender);
|
|
state.get(std::nothrow, "m.room.member", at<"state_key"_>(event), appender);
|
|
}
|
|
|
|
{
|
|
json::stack::member state_m
|
|
{
|
|
data, "state"
|
|
};
|
|
|
|
json::stack::array state_a
|
|
{
|
|
state_m
|
|
};
|
|
|
|
state.for_each([&state_a]
|
|
(const m::event &event)
|
|
{
|
|
state_a.append(event);
|
|
});
|
|
|
|
}
|
|
}
|
|
|
|
return resource::response
|
|
{
|
|
client, json::array
|
|
{
|
|
out.completed()
|
|
}
|
|
};
|
|
}
|
|
|
|
resource::method
|
|
method_put
|
|
{
|
|
send_join_resource, "PUT", put__send_join,
|
|
{
|
|
method_put.VERIFY_ORIGIN
|
|
}
|
|
};
|