2019-07-03 19:58:13 -07:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
using namespace ircd;
|
|
|
|
|
2020-03-03 17:45:58 -08:00
|
|
|
// federation_invite (weak)
|
|
|
|
extern conf::item<milliseconds>
|
|
|
|
stream_cross_sleeptime;
|
|
|
|
|
2019-07-05 19:00:41 -07:00
|
|
|
static void
|
|
|
|
process(client &,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &,
|
2019-07-05 19:00:41 -07:00
|
|
|
const m::event &);
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
static m::resource::response
|
2019-07-03 19:58:13 -07:00
|
|
|
put__invite(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request);
|
2019-07-03 19:58:13 -07:00
|
|
|
|
|
|
|
mapi::header
|
|
|
|
IRCD_MODULE
|
|
|
|
{
|
|
|
|
"Federation 12 :Inviting to a room (v2)"
|
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource
|
2019-07-03 19:58:13 -07:00
|
|
|
invite_resource
|
|
|
|
{
|
|
|
|
"/_matrix/federation/v2/invite/",
|
|
|
|
{
|
|
|
|
"Inviting to a room",
|
|
|
|
resource::DIRECTORY
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource::method
|
2019-07-03 19:58:13 -07:00
|
|
|
method_put
|
|
|
|
{
|
|
|
|
invite_resource, "PUT", put__invite,
|
|
|
|
{
|
|
|
|
method_put.VERIFY_ORIGIN
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource::response
|
2019-07-03 19:58:13 -07:00
|
|
|
put__invite(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request)
|
2019-07-03 19:58:13 -07:00
|
|
|
{
|
|
|
|
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
|
|
|
|
{
|
|
|
|
"event_id path parameter required"
|
|
|
|
};
|
|
|
|
|
|
|
|
m::event::id::buf event_id
|
|
|
|
{
|
|
|
|
url::decode(event_id, request.parv[1])
|
|
|
|
};
|
|
|
|
|
|
|
|
const json::string &room_version
|
|
|
|
{
|
|
|
|
request.get("room_version", "1")
|
|
|
|
};
|
|
|
|
|
2019-07-06 00:09:08 -07:00
|
|
|
m::event event
|
2019-07-03 19:58:13 -07:00
|
|
|
{
|
2019-07-05 19:45:02 -07:00
|
|
|
request["event"], event_id
|
2019-07-03 19:58:13 -07:00
|
|
|
};
|
|
|
|
|
2019-07-10 00:25:42 -07:00
|
|
|
if(!json::get<"event_id"_>(event))
|
|
|
|
if(room_version == "1" || room_version == "2")
|
|
|
|
json::get<"event_id"_>(event) = event_id;
|
2019-07-03 19:58:13 -07:00
|
|
|
|
2019-07-10 00:25:42 -07:00
|
|
|
if(!check_id(event, room_version))
|
2019-07-03 19:58:13 -07:00
|
|
|
throw m::BAD_REQUEST
|
|
|
|
{
|
2019-07-10 00:25:42 -07:00
|
|
|
"Claimed event_id %s is incorrect.",
|
2019-07-03 19:58:13 -07:00
|
|
|
string_view{event_id},
|
|
|
|
};
|
|
|
|
|
|
|
|
if(at<"room_id"_>(event) != room_id)
|
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::NOT_MODIFIED, "M_MISMATCH_ROOM_ID",
|
|
|
|
"ID of room in request body %s does not match path param %s",
|
|
|
|
string_view{at<"room_id"_>(event)},
|
|
|
|
string_view{room_id},
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:44:02 -07:00
|
|
|
if(at<"type"_>(event) != "m.room.member")
|
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::NOT_MODIFIED, "M_INVALID_TYPE",
|
|
|
|
"event.type must be m.room.member"
|
|
|
|
};
|
|
|
|
|
|
|
|
if(unquote(at<"content"_>(event).at("membership")) != "invite")
|
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::NOT_MODIFIED, "M_INVALID_CONTENT_MEMBERSHIP",
|
|
|
|
"event.content.membership must be invite."
|
|
|
|
};
|
|
|
|
|
2020-04-12 14:55:54 -07:00
|
|
|
if(at<"origin"_>(event) != request.node_id)
|
2019-07-05 18:44:02 -07:00
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::FORBIDDEN, "M_INVALID_ORIGIN",
|
|
|
|
"event.origin must be you."
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::user::id &sender
|
|
|
|
{
|
|
|
|
at<"sender"_>(event)
|
|
|
|
};
|
|
|
|
|
2020-04-12 14:55:54 -07:00
|
|
|
if(sender.host() != request.node_id)
|
2019-07-05 18:44:02 -07:00
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::FORBIDDEN, "M_INVALID_ORIGIN",
|
|
|
|
"event.sender must be your user."
|
|
|
|
};
|
|
|
|
|
|
|
|
const m::user::id &target
|
|
|
|
{
|
|
|
|
at<"state_key"_>(event)
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!my_host(target.host()))
|
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::FORBIDDEN, "M_INVALID_STATE_KEY",
|
|
|
|
"event.state_key must be my user."
|
|
|
|
};
|
|
|
|
|
|
|
|
m::event::conforms non_conforms;
|
|
|
|
const m::event::conforms report
|
|
|
|
{
|
|
|
|
event, non_conforms.report
|
|
|
|
};
|
|
|
|
|
|
|
|
if(!report.clean())
|
|
|
|
throw m::error
|
|
|
|
{
|
|
|
|
http::NOT_MODIFIED, "M_INVALID_EVENT",
|
|
|
|
"Proffered event has the following problems: %s",
|
|
|
|
string(report)
|
|
|
|
};
|
|
|
|
|
|
|
|
// May conduct disk IO to check ACL
|
2019-07-03 19:58:13 -07:00
|
|
|
if(m::room::server_acl::enable_write)
|
|
|
|
if(!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."
|
|
|
|
};
|
|
|
|
|
2019-07-05 18:44:02 -07:00
|
|
|
// May conduct network IO to fetch node's key; disk IO to fetch node's key
|
|
|
|
if(!verify(event, request.node_id))
|
|
|
|
throw m::ACCESS_DENIED
|
|
|
|
{
|
|
|
|
"Invite event fails verification for %s",
|
|
|
|
string_view{request.node_id},
|
|
|
|
};
|
2019-07-03 19:58:13 -07:00
|
|
|
|
|
|
|
thread_local char sigs[4_KiB];
|
2019-07-06 00:09:08 -07:00
|
|
|
m::event signed_event
|
2019-07-03 19:58:13 -07:00
|
|
|
{
|
2020-02-23 22:44:14 -08:00
|
|
|
signatures(sigs, event, target.host())
|
2019-07-03 19:58:13 -07:00
|
|
|
};
|
|
|
|
|
2019-07-06 00:09:08 -07:00
|
|
|
signed_event.event_id = event_id;
|
2019-07-05 19:00:41 -07:00
|
|
|
const json::strung signed_json
|
2019-07-03 19:58:13 -07:00
|
|
|
{
|
|
|
|
signed_event
|
|
|
|
};
|
|
|
|
|
|
|
|
// Send back the signed event first before eval. If we eval the signed
|
|
|
|
// event first: the effects will occur before the inviting server has
|
|
|
|
// the signed event returned from us; they might not consider the user
|
|
|
|
// invited yet, causing trouble for the eval effects. That may actually
|
|
|
|
// still happen due to the two separate TCP connections being uncoordinated
|
|
|
|
// (one for this request, and another when m::eval effects connect to them
|
|
|
|
// and make any requests). But either way if this call fails then we will
|
|
|
|
// lose the invite but that may not be such a bad thing.
|
2019-09-28 16:12:07 -07:00
|
|
|
m::resource::response response
|
2019-07-03 19:58:13 -07:00
|
|
|
{
|
2019-07-06 00:09:08 -07:00
|
|
|
client, json::members
|
|
|
|
{
|
|
|
|
{ "event", json::object{signed_json} }
|
|
|
|
}
|
2019-07-03 19:58:13 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
// Synapse needs time to process our response otherwise our eval below may
|
|
|
|
// complete before this response arrives for them and is processed.
|
2020-03-03 17:45:58 -08:00
|
|
|
ctx::sleep(milliseconds(stream_cross_sleeptime));
|
2019-07-03 19:58:13 -07:00
|
|
|
|
2019-07-05 19:00:41 -07:00
|
|
|
// Post processing, does not throw.
|
2019-07-13 22:41:20 -07:00
|
|
|
process(client, request, signed_event);
|
2019-07-05 19:00:41 -07:00
|
|
|
|
|
|
|
// note: returning a resource response is a symbolic/indicator action to
|
|
|
|
// the caller and has no real effect at the point of return.
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
process(client &client,
|
2019-09-28 16:12:07 -07:00
|
|
|
const m::resource::request &request,
|
2019-07-05 19:00:41 -07:00
|
|
|
const m::event &event)
|
|
|
|
try
|
|
|
|
{
|
2019-07-03 19:58:13 -07:00
|
|
|
// Eval the dual-signed invite event. This will write it locally. This will
|
|
|
|
// also try to sync the room as best as possible. The invitee will then be
|
|
|
|
// presented with this invite request in their rooms list.
|
|
|
|
m::vm::opts vmopts;
|
2020-04-12 14:55:54 -07:00
|
|
|
vmopts.node_id = request.node_id;
|
2019-07-03 19:58:13 -07:00
|
|
|
|
|
|
|
// Synapse may 403 a fetch of the prev_event of the invite event.
|
2020-05-11 20:24:54 -07:00
|
|
|
vmopts.phase.set(m::vm::phase::FETCH_PREV, false);
|
2021-02-07 11:15:29 -08:00
|
|
|
vmopts.phase.set(m::vm::phase::EMPTION, false);
|
2019-07-03 19:58:13 -07:00
|
|
|
|
2019-07-10 00:26:25 -07:00
|
|
|
// Don't throw an exception for a re-evaluation; this will just be a no-op
|
2019-07-06 00:09:08 -07:00
|
|
|
vmopts.nothrows |= m::vm::fault::EXISTS;
|
2019-08-17 11:46:54 -07:00
|
|
|
vmopts.room_version = unquote(request.get("room_version", "1"));
|
2019-07-03 19:58:13 -07:00
|
|
|
|
|
|
|
m::vm::eval
|
|
|
|
{
|
2019-07-05 19:00:41 -07:00
|
|
|
event, vmopts
|
|
|
|
};
|
|
|
|
}
|
|
|
|
catch(const std::exception &e)
|
|
|
|
{
|
|
|
|
log::error
|
|
|
|
{
|
|
|
|
m::log, "Processing invite from:%s to:%s :%s",
|
|
|
|
json::get<"sender"_>(event),
|
|
|
|
json::get<"state_key"_>(event),
|
|
|
|
e.what(),
|
|
|
|
};
|
2019-07-03 19:58:13 -07:00
|
|
|
|
2019-07-05 19:00:41 -07:00
|
|
|
return;
|
2019-07-03 19:58:13 -07:00
|
|
|
}
|