0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-13 16:33:53 +01:00
construct/modules/m_room_create.cc

181 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.
namespace ircd::m
{
static void auth_room_create(const event &, room::auth::hookdata &);
extern hookfn<room::auth::hookdata &> auth_room_create_hookfn;
static void created_room(const event &, vm::eval &);
extern hookfn<vm::eval &> created_room_hookfn;
}
ircd::mapi::header
IRCD_MODULE
{
"Matrix m.room.create"
};
//
// an effect of room created
//
decltype(ircd::m::created_room_hookfn)
ircd::m::created_room_hookfn
{
created_room,
{
{ "_site", "vm.effect" },
{ "type", "m.room.create" },
}
};
void
ircd::m::created_room(const m::event &event,
m::vm::eval &)
try
{
const m::user::id &sender
{
at<"sender"_>(event)
};
const auto &level
{
my(sender) && sender != me()?
log::INFO:
log::DEBUG
};
if(RB_DEBUG_LEVEL || level != log::DEBUG)
log::logf
{
m::log, level, "Created room %s with %s by %s",
json::get<"room_id"_>(event),
json::get<"sender"_>(event),
string_view{event.event_id},
};
}
catch(const std::exception &e)
{
log::error
{
m::log, "Effect of creating room %s with %s by %s :%s",
json::get<"room_id"_>(event),
string_view{event.event_id},
json::get<"sender"_>(event),
e.what()
};
}
//
// auth handler
//
decltype(ircd::m::auth_room_create_hookfn)
ircd::m::auth_room_create_hookfn
{
auth_room_create,
{
{ "_site", "room.auth" },
{ "type", "m.room.create" },
}
};
void
ircd::m::auth_room_create(const event &event,
room::auth::hookdata &data)
{
using FAIL = m::room::auth::FAIL;
using conforms = m::event::conforms;
// 1. If type is m.room.create:
assert(json::get<"type"_>(event) == "m.room.create");
// a. If it has any previous events, reject.
if(count(data.prev))
throw FAIL
{
"m.room.create has previous events."
};
// b. If the domain of the room_id does not match the domain of the
// sender, reject.
if(room::id(at<"room_id"_>(event)).host() != user::id(at<"sender"_>(event)).host())
throw FAIL
{
"m.room.create room_id domain does not match sender domain."
};
// c. If content.room_version is present and is not a recognised
// version, reject.
if(json::get<"content"_>(event).has("room_version"))
{
const json::string &claim_version
{
json::get<"content"_>(event).get("room_version", "1"_sv)
};
const string_view &id_version
{
event.event_id.version()
};
if(claim_version == "1" || claim_version == "2")
{
// When the claimed version is 1 or 2 we don't actually
// care if the event_id is version 1, 3 or 4 etc; the server
// has eliminated use of the event_id hostpart in all rooms.
//if(id_version != "1")
// throw FAIL
// {
// "m.room.create room_version not 1"
// };
}
else if(claim_version == "3")
{
if(id_version != "3")
throw FAIL
{
"m.room.create room_version not 3"
};
}
else if(claim_version == "4")
{
if(id_version != "4")
throw FAIL
{
"m.room.create room_version not 4"
};
}
else
{
// Note that id_version reports "4" even for version "5"
// and beyond room versions. When a room version requires
// a new ID version these branches must be updated.
if(id_version != "4")
throw FAIL
{
"m.room.create room_version not 4"
};
}
}
// d. If content has no creator field, reject.
if(empty(json::get<"content"_>(event).get("creator")))
throw FAIL
{
"m.room.create content.creator is missing."
};
// e. Otherwise, allow.
data.allow = true;
}