mirror of
https://github.com/matrix-construct/construct
synced 2025-01-04 11:54:25 +01:00
502 lines
9 KiB
C++
502 lines
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
|
|
{
|
|
"Matrix m.room.aliases"
|
|
};
|
|
|
|
const m::room::id::buf
|
|
alias_room_id
|
|
{
|
|
"alias", ircd::my_host()
|
|
};
|
|
|
|
const m::room
|
|
alias_room
|
|
{
|
|
alias_room_id
|
|
};
|
|
|
|
conf::item<seconds>
|
|
alias_cache_ttl
|
|
{
|
|
{ "name", "ircd.m.room.aliases.cache.ttl" },
|
|
{ "default", 604800L },
|
|
};
|
|
|
|
conf::item<seconds>
|
|
alias_fetch_timeout
|
|
{
|
|
{ "name", "ircd.m.room.aliases.fetch.timeout" },
|
|
{ "default", 10L },
|
|
};
|
|
|
|
//
|
|
// hook handlers
|
|
//
|
|
|
|
const m::hookfn<m::vm::eval &>
|
|
_create_alias_room
|
|
{
|
|
{
|
|
{ "_site", "vm.effect" },
|
|
{ "room_id", "!ircd" },
|
|
{ "type", "m.room.create" },
|
|
},
|
|
[](const m::event &, m::vm::eval &)
|
|
{
|
|
m::create(alias_room_id, m::me.user_id);
|
|
}
|
|
};
|
|
|
|
static void
|
|
_can_change_aliases(const m::event &event,
|
|
m::vm::eval &);
|
|
|
|
const m::hookfn<m::vm::eval &>
|
|
_can_change_aliases_hookfn
|
|
{
|
|
_can_change_aliases,
|
|
{
|
|
{ "_site", "vm.eval" },
|
|
{ "type", "m.room.aliases" },
|
|
}
|
|
};
|
|
|
|
void
|
|
_can_change_aliases(const m::event &event,
|
|
m::vm::eval &eval)
|
|
{
|
|
const m::room::id &room_id
|
|
{
|
|
at<"room_id"_>(event)
|
|
};
|
|
|
|
const string_view &state_key
|
|
{
|
|
at<"state_key"_>(event)
|
|
};
|
|
|
|
if(state_key != at<"origin"_>(event))
|
|
throw m::ACCESS_DENIED
|
|
{
|
|
"Cannot set aliases for host '%s' from origin '%s'",
|
|
state_key,
|
|
at<"origin"_>(event)
|
|
};
|
|
|
|
const json::array &aliases
|
|
{
|
|
at<"content"_>(event).get("aliases")
|
|
};
|
|
|
|
for(const json::string &alias_ : aliases)
|
|
{
|
|
const m::room::alias &alias{alias_};
|
|
if(at<"origin"_>(event) != alias.host())
|
|
throw m::ACCESS_DENIED
|
|
{
|
|
"Cannot set alias for host '%s' from origin '%s'",
|
|
alias.host(),
|
|
at<"origin"_>(event)
|
|
};
|
|
}
|
|
}
|
|
|
|
static void
|
|
_changed_aliases(const m::event &event,
|
|
m::vm::eval &);
|
|
|
|
const m::hookfn<m::vm::eval &>
|
|
_changed_aliases_hookfn
|
|
{
|
|
_changed_aliases,
|
|
{
|
|
{ "_site", "vm.effect" },
|
|
{ "type", "m.room.aliases" },
|
|
}
|
|
};
|
|
|
|
void
|
|
_changed_aliases(const m::event &event,
|
|
m::vm::eval &)
|
|
{
|
|
const m::room::id &room_id
|
|
{
|
|
at<"room_id"_>(event)
|
|
};
|
|
|
|
const json::array &aliases
|
|
{
|
|
at<"content"_>(event).get("aliases")
|
|
};
|
|
|
|
for(const json::string &alias : aliases) try
|
|
{
|
|
if(m::room::aliases::cache::has(alias))
|
|
continue;
|
|
|
|
m::room::aliases::cache::set(alias, room_id);
|
|
|
|
log::info
|
|
{
|
|
m::log, "Updated aliases of %s by %s in %s with %s",
|
|
string_view{room_id},
|
|
json::get<"sender"_>(event),
|
|
json::get<"event_id"_>(event),
|
|
string_view{alias},
|
|
};
|
|
}
|
|
catch(const std::exception &e)
|
|
{
|
|
log::error
|
|
{
|
|
m::log, "Updating aliases of %s by %s in %s with %s :%s",
|
|
string_view{room_id},
|
|
json::get<"sender"_>(event),
|
|
json::get<"event_id"_>(event),
|
|
string_view{alias},
|
|
e.what(),
|
|
};
|
|
}
|
|
}
|
|
|
|
//
|
|
// m::room::aliases
|
|
//
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::for_each(const m::room &room,
|
|
const string_view &server,
|
|
const closure_bool &closure)
|
|
{
|
|
const room::state state
|
|
{
|
|
room
|
|
};
|
|
|
|
assert(server);
|
|
const event::idx &event_idx
|
|
{
|
|
state.get(std::nothrow, "m.room.aliases", server)
|
|
};
|
|
|
|
if(!event_idx)
|
|
return true;
|
|
|
|
bool ret{true};
|
|
m::get(std::nothrow, event_idx, "content", [&closure, &ret]
|
|
(const json::object &content)
|
|
{
|
|
const json::array &aliases
|
|
{
|
|
content["aliases"]
|
|
};
|
|
|
|
for(auto it(begin(aliases)); it != end(aliases) && ret; ++it)
|
|
{
|
|
const json::string &alias(*it);
|
|
if(!valid(m::id::ROOM_ALIAS, alias))
|
|
continue;
|
|
|
|
if(!closure(alias))
|
|
ret = false;
|
|
}
|
|
});
|
|
|
|
return ret;
|
|
}
|
|
|
|
//
|
|
// m::room::aliases::cache
|
|
//
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::del(const alias &alias)
|
|
{
|
|
char swapbuf[m::id::room_alias::buf::SIZE];
|
|
const string_view &key
|
|
{
|
|
alias.swap(swapbuf)
|
|
};
|
|
|
|
const auto &event_idx
|
|
{
|
|
alias_room.get(std::nothrow, "ircd.room.alias", key)
|
|
};
|
|
|
|
if(!event_idx)
|
|
return false;
|
|
|
|
const auto event_id
|
|
{
|
|
m::event_id(event_idx, std::nothrow)
|
|
};
|
|
|
|
if(!event_id)
|
|
return false;
|
|
|
|
const auto ret
|
|
{
|
|
redact(alias_room, m::me.user_id, event_id, "deleted")
|
|
};
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::set(const alias &alias,
|
|
const id &id)
|
|
{
|
|
char swapbuf[m::id::room_alias::buf::SIZE];
|
|
const string_view &key
|
|
{
|
|
alias.swap(swapbuf)
|
|
};
|
|
|
|
const auto ret
|
|
{
|
|
send(alias_room, m::me.user_id, "ircd.room.alias", key,
|
|
{
|
|
{ "room_id", id }
|
|
})
|
|
};
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::get(std::nothrow_t,
|
|
const alias &alias,
|
|
const id::closure &closure)
|
|
{
|
|
char swapbuf[m::id::room_alias::buf::SIZE];
|
|
const string_view &key
|
|
{
|
|
alias.swap(swapbuf)
|
|
};
|
|
|
|
m::event::idx event_idx
|
|
{
|
|
alias_room.get(std::nothrow, "ircd.room.alias", key)
|
|
};
|
|
|
|
if(!event_idx)
|
|
{
|
|
if(my_host(alias.host()))
|
|
return false;
|
|
|
|
if(!fetch(std::nothrow, alias, alias.host()))
|
|
return false;
|
|
|
|
event_idx = alias_room.get(std::nothrow, "ircd.room.alias", key);
|
|
}
|
|
|
|
time_t ts;
|
|
if(!m::get(event_idx, "origin_server_ts", ts))
|
|
return false;
|
|
|
|
if(!my_host(alias.host()) && ircd::time() - ts > seconds(alias_cache_ttl).count())
|
|
{
|
|
if(!fetch(std::nothrow, alias, alias.host()))
|
|
return false;
|
|
|
|
event_idx = alias_room.get(std::nothrow, "ircd.room.alias", key);
|
|
}
|
|
|
|
if(!event_idx)
|
|
return false;
|
|
|
|
bool ret{false};
|
|
m::get(std::nothrow, event_idx, "content", [&closure, &ret]
|
|
(const json::object &content)
|
|
{
|
|
const json::string &room_id
|
|
{
|
|
content.get("room_id")
|
|
};
|
|
|
|
if(!empty(room_id))
|
|
{
|
|
ret = true;
|
|
closure(room_id);
|
|
}
|
|
});
|
|
|
|
return ret;
|
|
}
|
|
|
|
namespace ircd::m
|
|
{
|
|
thread_local char room_aliases_cache_fetch_hpbuf[384];
|
|
}
|
|
|
|
void
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::fetch(const alias &alias,
|
|
const net::hostport &hp)
|
|
try
|
|
{
|
|
const unique_buffer<mutable_buffer> buf
|
|
{
|
|
16_KiB
|
|
};
|
|
|
|
m::v1::query::opts opts;
|
|
opts.remote = hp;
|
|
|
|
m::v1::query::directory request
|
|
{
|
|
alias, buf, std::move(opts)
|
|
};
|
|
|
|
request.wait(seconds(alias_fetch_timeout));
|
|
const http::code &code
|
|
{
|
|
request.get()
|
|
};
|
|
|
|
const json::object &response
|
|
{
|
|
request
|
|
};
|
|
|
|
if(!response.has("room_id"))
|
|
throw m::NOT_FOUND
|
|
{
|
|
"Server '%s' does not know room_id for %s",
|
|
string(room_aliases_cache_fetch_hpbuf, hp),
|
|
string_view{alias},
|
|
};
|
|
|
|
const m::room::id &room_id
|
|
{
|
|
unquote(response["room_id"])
|
|
};
|
|
|
|
set(alias, room_id);
|
|
}
|
|
catch(const ctx::timeout &e)
|
|
{
|
|
throw m::error
|
|
{
|
|
http::GATEWAY_TIMEOUT, "M_ROOM_ALIAS_TIMEOUT",
|
|
"Server '%s' did not respond with a room_id for %s in time",
|
|
string(room_aliases_cache_fetch_hpbuf, hp),
|
|
string_view{alias},
|
|
};
|
|
}
|
|
catch(const server::unavailable &e)
|
|
{
|
|
throw m::error
|
|
{
|
|
http::BAD_GATEWAY, "M_ROOM_ALIAS_UNAVAILABLE",
|
|
"Server '%s' is not available to query a room_id for %s",
|
|
string(room_aliases_cache_fetch_hpbuf, hp),
|
|
string_view{alias},
|
|
};
|
|
}
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::has(const alias &alias)
|
|
{
|
|
char swapbuf[m::id::room_alias::buf::SIZE];
|
|
const string_view &key
|
|
{
|
|
alias.swap(swapbuf)
|
|
};
|
|
|
|
const auto &event_idx
|
|
{
|
|
alias_room.get(std::nothrow, "ircd.room.alias", key)
|
|
};
|
|
|
|
if(!event_idx)
|
|
return false;
|
|
|
|
time_t ts;
|
|
if(!m::get(event_idx, "origin_server_ts", ts))
|
|
return false;
|
|
|
|
if(ircd::time() - ts > seconds(alias_cache_ttl).count())
|
|
return false;
|
|
|
|
bool ret{false};
|
|
m::get(std::nothrow, event_idx, "content", [&ret]
|
|
(const json::object &content)
|
|
{
|
|
const json::string &room_id
|
|
{
|
|
content.get("room_id")
|
|
};
|
|
|
|
ret = !empty(room_id);
|
|
});
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool
|
|
IRCD_MODULE_EXPORT
|
|
ircd::m::room::aliases::cache::for_each(const string_view &server,
|
|
const closure_bool &closure)
|
|
{
|
|
const m::room::state state
|
|
{
|
|
alias_room
|
|
};
|
|
|
|
bool ret{true};
|
|
const m::room::state::closure_bool reclosure{[&server, &closure, &ret]
|
|
(const string_view &type, const string_view &state_key, const m::event::idx &event_idx)
|
|
{
|
|
thread_local char swapbuf[m::id::room_alias::buf::SIZE];
|
|
const alias &alias
|
|
{
|
|
m::id::unswap(state_key, swapbuf)
|
|
};
|
|
|
|
if(server && alias.host() != server)
|
|
return false;
|
|
|
|
time_t ts;
|
|
if(!m::get(event_idx, "origin_server_ts", ts))
|
|
return true;
|
|
|
|
if(ircd::time() - ts > seconds(alias_cache_ttl).count())
|
|
return true;
|
|
|
|
m::get(std::nothrow, event_idx, "content", [&closure, &ret, &alias]
|
|
(const json::object &content)
|
|
{
|
|
const json::string &room_id
|
|
{
|
|
content.get("room_id")
|
|
};
|
|
|
|
if(!empty(room_id))
|
|
ret = closure(alias, room_id);
|
|
});
|
|
|
|
return ret;
|
|
}};
|
|
|
|
state.for_each("ircd.room.alias", server, reclosure);
|
|
return ret;
|
|
}
|