0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-22 20:40:00 +01:00
construct/modules/client/rooms/invite.cc

314 lines
5.5 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.
#include "rooms.h"
namespace ircd::m
{
static event::id::buf invite_remote(const event &);
static void on_invite_remote(const event &, vm::eval &);
extern conf::item<seconds> invite_remote_timeout;
extern hookfn<vm::eval &> invite_remote_hook;
}
decltype(ircd::m::invite_remote_timeout)
ircd::m::invite_remote_timeout
{
{ "name", "ircd.client.rooms.invite.remote.timeout" },
{ "default", 30L },
};
decltype(ircd::m::invite_remote_hook)
ircd::m::invite_remote_hook
{
on_invite_remote,
{
{ "_site", "vm.issue" },
{ "type", "m.room.member" },
{ "membership", "invite" },
}
};
ircd::m::resource::response
post__invite(ircd::client &client,
const ircd::m::resource::request &request,
const ircd::m::room::id &room_id)
{
using namespace ircd;
const m::user::id &target
{
unquote(request.at("user_id"))
};
const m::user::id &sender
{
request.user_id
};
const json::string &id_server
{
request["id_server"]
};
const json::string &id_access_token
{
request["id_access_token"]
};
const json::string &medium
{
request["medium"]
};
const json::string &address
{
request["address"]
};
const auto event_id
{
m::invite(room_id, target, sender)
};
return m::resource::response
{
client, http::OK
};
}
void
ircd::m::on_invite_remote(const event &event,
vm::eval &eval)
{
const m::room::id &room_id
{
at<"room_id"_>(event)
};
const m::user::id &target
{
at<"state_key"_>(event)
};
const auto target_host
{
target.host()
};
if(m::my_host(target_host))
return;
// Host already signed event elsehow.
for(const auto &[host, sig] : json::get<"signatures"_>(event))
if(host == target_host)
return;
const m::room::origins origins
{
room_id
};
if(origins.has(target_host))
return;
const auto eid
{
invite_remote(event)
};
log::info
{
m::log, "Invite %s to %s by %s completed with %s",
string_view{target},
string_view{room_id},
json::get<"sender"_>(event),
string_view{eid},
};
}
ircd::m::event::id::buf
ircd::m::invite_remote(const event &event)
try
{
const auto &event_id
{
event.event_id
};
const m::room::id &room_id
{
at<"room_id"_>(event)
};
const m::user::id &target
{
at<"state_key"_>(event)
};
assert(!my(target));
const unique_buffer<mutable_buffer> bufs
{
148_KiB
};
json::stack out{bufs};
json::stack::object top{out};
{
char versionbuf[32];
json::stack::member room_version
{
top, "room_version", json::value
{
m::version(versionbuf, room_id, std::nothrow), json::STRING
}
};
}
{
json::stack::object _event
{
top, "event"
};
_event.append(event);
}
{
json::stack::array invite_room_state
{
top, "invite_room_state"
};
const auto append
{
[&invite_room_state](const m::event &event)
{
invite_room_state.append(event);
}
};
const m::room::state state
{
room_id
};
state.get(std::nothrow, "m.room.create", "", append);
state.get(std::nothrow, "m.room.power_levels", "", append);
state.get(std::nothrow, "m.room.join_rules", "", append);
state.get(std::nothrow, "m.room.history_visibility", "", append);
state.get(std::nothrow, "m.room.aliases", my_host(), append);
state.get(std::nothrow, "m.room.canonical_alias", "", append);
state.get(std::nothrow, "m.room.avatar", "", append);
state.get(std::nothrow, "m.room.name", "", append);
state.get(std::nothrow, "m.room.member", at<"sender"_>(event), append);
}
top.~object();
const string_view &proto
{
out.completed()
};
const mutable_buffer buf
{
bufs + size(proto)
};
m::fed::invite2::opts opts;
opts.remote = target.host();
m::fed::invite2 request
{
room_id, event_id, proto, buf, std::move(opts)
};
log::debug
{
m::log, "Sending invite %s to %s",
string_view{event.event_id},
target.host(),
};
http::code rcode; try
{
request.wait(seconds(invite_remote_timeout));
rcode = request.get();
}
catch(const http::error &e)
{
log::error
{
m::log, "Invite %s to %s :%s :%s",
string_view{event.event_id},
target.host(),
e.what(),
e.content,
};
throw;
}
const json::object response
{
request
};
m::event::id::buf revent_id;
const m::event &revent
{
revent_id, response.at("event")
};
if(!verify(revent, target.host()))
throw m::error
{
http::UNAUTHORIZED, "M_INVITE_UNSIGNED",
"Invitee's host '%s' did not sign the invite.",
target.host()
};
if(!verify(revent, my_host()))
throw m::error
{
http::FORBIDDEN, "M_INVITE_MODIFIED",
"Invite event no longer verified by our signature."
};
log::logf
{
m::log, log::level::DEBUG,
"Invite %s in %s accepted by '%s'",
string_view{event.event_id},
string_view{room_id},
string_view{target.host()},
};
m::vm::opts vmopts;
vmopts.infolog_accept = true;
vmopts.unique = false;
m::vm::eval eval
{
revent, vmopts
};
return eval.event_id;
}
catch(const std::exception &e)
{
log::error
{
m::log, "Invite remote %s :%s",
string_view{event.event_id},
e.what(),
};
throw;
}