mirror of
https://github.com/matrix-construct/construct
synced 2025-01-20 03:21:54 +01:00
268 lines
4.7 KiB
C++
268 lines
4.7 KiB
C++
// The Construct
|
|
//
|
|
// Copyright (C) The Construct Developers, Authors & Contributors
|
|
// Copyright (C) 2016-2020 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.
|
|
|
|
bool
|
|
ircd::m::user::pushrules::del(const path &path)
|
|
const
|
|
{
|
|
const auto &[scope, kind, ruleid]
|
|
{
|
|
path
|
|
};
|
|
|
|
char typebuf[event::TYPE_MAX_SIZE];
|
|
const string_view &type
|
|
{
|
|
push::make_type(typebuf, path)
|
|
};
|
|
|
|
const m::user::room user_room
|
|
{
|
|
user
|
|
};
|
|
|
|
const event::idx &event_idx
|
|
{
|
|
user_room.get(std::nothrow, type, ruleid)
|
|
};
|
|
|
|
const m::event::id::buf event_id
|
|
{
|
|
m::event_id(std::nothrow, event_idx)
|
|
};
|
|
|
|
if(!event_id)
|
|
return false;
|
|
|
|
const auto redact_id
|
|
{
|
|
m::redact(user_room, user, event_id, "deleted")
|
|
};
|
|
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::pushrules::set(const path &path,
|
|
const json::object &content)
|
|
const
|
|
{
|
|
const auto &[scope, kind, ruleid]
|
|
{
|
|
path
|
|
};
|
|
|
|
const push::rule rule
|
|
{
|
|
content
|
|
};
|
|
|
|
char typebuf[event::TYPE_MAX_SIZE];
|
|
const string_view &type
|
|
{
|
|
push::make_type(typebuf, path)
|
|
};
|
|
|
|
const m::user::room user_room
|
|
{
|
|
user
|
|
};
|
|
|
|
const auto rule_event_id
|
|
{
|
|
m::send(user_room, user, type, ruleid, content)
|
|
};
|
|
|
|
return true;
|
|
}
|
|
|
|
void
|
|
ircd::m::user::pushrules::get(const path &path,
|
|
const closure &closure)
|
|
const
|
|
{
|
|
const auto &[scope, kind, ruleid]
|
|
{
|
|
path
|
|
};
|
|
|
|
if(!get(std::nothrow, path, closure))
|
|
throw m::NOT_FOUND
|
|
{
|
|
"push rule (%s,%s,%s) for user %s not found",
|
|
scope,
|
|
kind,
|
|
ruleid,
|
|
string_view{user.user_id}
|
|
};
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::pushrules::get(std::nothrow_t,
|
|
const path &path,
|
|
const closure &closure)
|
|
const
|
|
{
|
|
const auto &[scope, kind, ruleid]
|
|
{
|
|
path
|
|
};
|
|
|
|
char typebuf[event::TYPE_MAX_SIZE];
|
|
const string_view &type
|
|
{
|
|
push::make_type(typebuf, path)
|
|
};
|
|
|
|
const m::user::room user_room
|
|
{
|
|
user
|
|
};
|
|
|
|
const event::idx &event_idx
|
|
{
|
|
user_room.get(std::nothrow, type, ruleid)
|
|
};
|
|
|
|
const bool user_rule_found
|
|
{
|
|
m::get(std::nothrow, event_idx, "content", [&path, &closure, &event_idx]
|
|
(const json::object &content)
|
|
{
|
|
closure(event_idx, path, content);
|
|
})
|
|
};
|
|
|
|
// Allow user-set rule found with the same id as a server-default to
|
|
// always take priority.
|
|
if(user_rule_found)
|
|
return true;
|
|
|
|
// Present the default rule to the closure.
|
|
if(likely(scope == "global"))
|
|
{
|
|
const auto &rules
|
|
{
|
|
push::rules::defaults.get<json::array>(kind)
|
|
};
|
|
|
|
for(const json::object rule : rules)
|
|
{
|
|
const json::string _ruleid
|
|
{
|
|
rule["rule_id"]
|
|
};
|
|
|
|
if(_ruleid != ruleid)
|
|
continue;
|
|
|
|
closure(0UL, path, rule);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Nothing found
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::pushrules::for_each(const closure_bool &closure)
|
|
const
|
|
{
|
|
return for_each(path{}, closure);
|
|
}
|
|
|
|
bool
|
|
ircd::m::user::pushrules::for_each(const path &path,
|
|
const closure_bool &closure)
|
|
const
|
|
{
|
|
const auto &[scope, kind, ruleid]
|
|
{
|
|
path
|
|
};
|
|
|
|
// If the path contains a ruleid there's at most one item to iterate...
|
|
if(ruleid)
|
|
{
|
|
get(std::nothrow, path, closure);
|
|
return true;
|
|
}
|
|
|
|
const user::room user_room
|
|
{
|
|
user
|
|
};
|
|
|
|
const room::state state
|
|
{
|
|
user_room
|
|
};
|
|
|
|
// Present the default rules to the closure
|
|
if(!scope || scope == "global")
|
|
{
|
|
for(const auto &_kind : json::keys<decltype(push::rules::defaults)>())
|
|
{
|
|
if(kind && kind != _kind)
|
|
continue;
|
|
|
|
for(const json::object rule : push::rules::defaults.at<json::array>(_kind))
|
|
{
|
|
const json::string _ruleid
|
|
{
|
|
rule["rule_id"]
|
|
};
|
|
|
|
const push::path path
|
|
{
|
|
"global", _kind, _ruleid
|
|
};
|
|
|
|
char typebuf[event::TYPE_MAX_SIZE];
|
|
const string_view &type
|
|
{
|
|
push::make_type(typebuf, path)
|
|
};
|
|
|
|
// Check if the user set a rule with the same path/id as a
|
|
// server-default rule; if so, we want their rule to take
|
|
// priority over the this server-default below.
|
|
if(state.has(type, _ruleid))
|
|
continue;
|
|
|
|
// Present the server-default rule to the iteration.
|
|
if(!closure(0UL, path, rule))
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
char typebuf[event::TYPE_MAX_SIZE];
|
|
const room::state::type_prefix type
|
|
{
|
|
push::make_type(typebuf, path)
|
|
};
|
|
|
|
return state.for_each(type, [&closure]
|
|
(const string_view &type, const string_view &state_key, const m::event::idx &event_idx)
|
|
{
|
|
const auto path
|
|
{
|
|
push::make_path(type, state_key)
|
|
};
|
|
|
|
return m::query<bool>(std::nothrow, event_idx, "content", true, [&]
|
|
(const json::object &content)
|
|
{
|
|
return closure(event_idx, path, content);
|
|
});
|
|
});
|
|
}
|