2024-02-13 10:26:29 +00:00
|
|
|
-- mod_muc_slow_mode
|
|
|
|
--
|
2024-05-23 10:18:17 +00:00
|
|
|
-- SPDX-FileCopyrightText: 2024 John Livingston <https://www.john-livingston.fr/>
|
|
|
|
-- SPDX-License-Identifier: AGPL-3.0-only
|
2024-02-13 10:26:29 +00:00
|
|
|
--
|
|
|
|
-- This file is AGPL-v3 licensed.
|
|
|
|
-- Please see the Peertube livechat plugin copyright information.
|
|
|
|
-- https://livingston.frama.io/peertube-plugin-livechat/credits/
|
|
|
|
--
|
|
|
|
-- Implements: XEP-????: MUC Slow Mode (XEP to come).
|
|
|
|
--
|
|
|
|
-- Imports
|
2024-02-13 15:45:12 +00:00
|
|
|
local st = require "util.stanza";
|
|
|
|
local jid_bare = require "util.jid".bare;
|
|
|
|
local gettime = require 'socket'.gettime;
|
2024-02-13 10:26:29 +00:00
|
|
|
|
|
|
|
-- Plugin dependencies
|
|
|
|
local mod_muc = module:depends "muc";
|
|
|
|
|
2024-02-13 15:45:12 +00:00
|
|
|
local muc_util = module:require "muc/util";
|
|
|
|
local valid_roles = muc_util.valid_roles;
|
|
|
|
|
|
|
|
-- Namespaces
|
|
|
|
local xmlns_muc = "http://jabber.org/protocol/muc";
|
|
|
|
|
2024-02-14 08:26:01 +00:00
|
|
|
-- Options
|
|
|
|
|
|
|
|
-- form_position: the position in the room config form (this value will be passed as priority for the "muc-config-form" hook).
|
|
|
|
-- Depending on your application, it is possible that the slow mode is more important than other fields (for example for a video streaming service).
|
|
|
|
-- So there is an option to change this.
|
|
|
|
-- By default, field will be between muc#roomconfig_changesubject and muc#roomconfig_moderatedroom
|
2024-02-16 14:16:44 +00:00
|
|
|
local form_position = module:get_option_number("slow_mode_duration_form_position") or 80-2;
|
2024-02-14 08:26:01 +00:00
|
|
|
|
2024-02-13 10:26:29 +00:00
|
|
|
-- Getter/Setter
|
2024-02-16 14:16:44 +00:00
|
|
|
local function get_slow_mode_duration(room)
|
|
|
|
return room._data.slow_mode_duration or 0;
|
2024-02-13 10:26:29 +00:00
|
|
|
end
|
|
|
|
|
2024-02-16 14:16:44 +00:00
|
|
|
local function set_slow_mode_duration(room, duration)
|
|
|
|
if duration then
|
|
|
|
duration = assert(tonumber(duration), "Slow mode duration is not a valid number");
|
2024-02-14 13:00:36 +00:00
|
|
|
end
|
2024-02-16 14:16:44 +00:00
|
|
|
if duration and duration < 0 then
|
|
|
|
duration = 0;
|
2024-02-13 10:26:29 +00:00
|
|
|
end
|
2024-02-14 10:43:57 +00:00
|
|
|
|
2024-02-16 14:16:44 +00:00
|
|
|
if get_slow_mode_duration(room) == duration then return false; end
|
2024-02-14 10:43:57 +00:00
|
|
|
|
2024-02-16 14:16:44 +00:00
|
|
|
room._data.slow_mode_duration = duration;
|
2024-02-14 13:00:36 +00:00
|
|
|
return true;
|
2024-02-13 10:26:29 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Discovering support
|
|
|
|
local function add_disco_form(event)
|
2024-02-14 10:43:57 +00:00
|
|
|
table.insert(event.form, {
|
2024-02-16 14:16:44 +00:00
|
|
|
name = "muc#roominfo_slow_mode_duration";
|
2024-02-13 10:26:29 +00:00
|
|
|
value = "";
|
|
|
|
});
|
2024-02-16 14:16:44 +00:00
|
|
|
event.formdata["muc#roominfo_slow_mode_duration"] = get_slow_mode_duration(event.room);
|
2024-02-13 10:26:29 +00:00
|
|
|
end
|
|
|
|
|
2024-02-14 10:23:42 +00:00
|
|
|
module:hook("muc-disco#info", add_disco_form);
|
2024-02-13 10:26:29 +00:00
|
|
|
|
|
|
|
-- Config form declaration
|
|
|
|
local function add_form_option(event)
|
|
|
|
table.insert(event.form, {
|
2024-02-16 14:16:44 +00:00
|
|
|
name = "muc#roomconfig_slow_mode_duration";
|
2024-02-13 10:26:29 +00:00
|
|
|
type = "text-single";
|
2024-02-14 13:00:36 +00:00
|
|
|
datatype = "xs:integer";
|
2024-02-23 14:50:43 +00:00
|
|
|
range_min = 0;
|
2024-02-16 14:16:44 +00:00
|
|
|
label = "Slow Mode (0=disabled, any positive integer= users can send a message every X seconds.)";
|
|
|
|
-- desc = "";
|
|
|
|
value = get_slow_mode_duration(event.room);
|
2024-02-13 10:26:29 +00:00
|
|
|
});
|
|
|
|
end
|
|
|
|
|
2024-02-16 14:16:44 +00:00
|
|
|
module:hook("muc-config-submitted/muc#roomconfig_slow_mode_duration", function(event)
|
|
|
|
if set_slow_mode_duration(event.room, event.value) then
|
2024-02-13 10:26:29 +00:00
|
|
|
-- status 104 = configuration change: Inform occupants that a non-privacy-related room configuration change has occurred
|
2024-02-14 13:00:36 +00:00
|
|
|
event.status_codes["104"] = true;
|
|
|
|
end
|
2024-02-13 10:26:29 +00:00
|
|
|
end);
|
|
|
|
|
2024-02-14 08:26:01 +00:00
|
|
|
module:hook("muc-config-form", add_form_option, form_position);
|
2024-02-13 15:45:12 +00:00
|
|
|
|
|
|
|
-- handling groupchat messages
|
|
|
|
function handle_groupchat(event)
|
|
|
|
local origin, stanza = event.origin, event.stanza;
|
|
|
|
local room = event.room;
|
|
|
|
|
2024-02-22 09:19:51 +00:00
|
|
|
-- only consider messages with body (ie: ignore chatstate and other non-text xmpp messages)
|
|
|
|
local body = stanza:get_child_text("body")
|
|
|
|
if not body or #body < 1 then
|
|
|
|
-- module:log("debug", "No body, message accepted");
|
|
|
|
return;
|
|
|
|
end
|
|
|
|
|
2024-02-16 14:16:44 +00:00
|
|
|
local duration = get_slow_mode_duration(room) or 0;
|
|
|
|
if duration <= 0 then
|
2024-02-13 15:45:12 +00:00
|
|
|
-- no slow mode for this room
|
|
|
|
-- module:log("debug", "No slow mode for this room");
|
|
|
|
return;
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Checking user's permissions (moderators are not subject to slow mode)
|
2024-02-14 13:00:36 +00:00
|
|
|
local actor = stanza.attr.from;
|
|
|
|
local actor_nick = room:get_occupant_jid(actor);
|
2024-02-13 15:45:12 +00:00
|
|
|
local actor_jid = jid_bare(actor);
|
|
|
|
-- Only checking role, not affiliation (slow mode only applies on users currently connected to the room)
|
|
|
|
local role = room:get_role(actor_nick);
|
2024-02-14 13:00:36 +00:00
|
|
|
if valid_roles[role or "none"] >= valid_roles.moderator then
|
2024-02-13 15:45:12 +00:00
|
|
|
-- user bypasses the slow mode.
|
|
|
|
-- module:log("debug", "User is moderator, bypassing slow mode");
|
2024-02-14 13:00:36 +00:00
|
|
|
return;
|
|
|
|
end
|
2024-02-13 15:45:12 +00:00
|
|
|
|
|
|
|
if not room.slow_mode_last_messages then
|
|
|
|
-- We store last message time for each users in room.slow_mode_last_messages:
|
|
|
|
-- * key: bare jid (without the nickname)
|
|
|
|
-- * value: last message timestamp
|
|
|
|
-- If room is cleared from memory, these data are lost. But should not be an issue.
|
|
|
|
-- For now, i don't clean slow_mode_last_messages, it should not use too much memory.
|
|
|
|
-- module:log("debug", "Initializing slow_mode_last_messages for the room.");
|
|
|
|
room.slow_mode_last_messages = {};
|
|
|
|
end
|
|
|
|
|
|
|
|
local now = gettime();
|
|
|
|
local previous = room.slow_mode_last_messages[actor_jid];
|
|
|
|
-- module:log(
|
|
|
|
-- "debug",
|
2024-02-16 14:16:44 +00:00
|
|
|
-- "Last message for user %s was at %s, now is %s, duration is %s, now - previous is %s",
|
2024-02-13 15:45:12 +00:00
|
|
|
-- actor_jid,
|
|
|
|
-- previous or 0,
|
|
|
|
-- now,
|
2024-02-16 14:16:44 +00:00
|
|
|
-- duration,
|
2024-02-13 15:45:12 +00:00
|
|
|
-- (now - (previous or 0))
|
|
|
|
-- );
|
2024-02-16 14:16:44 +00:00
|
|
|
if ((not previous) or (now - previous > duration)) then
|
2024-02-13 15:45:12 +00:00
|
|
|
-- module:log("debug", "Message accepted");
|
|
|
|
room.slow_mode_last_messages[actor_jid] = now;
|
|
|
|
return;
|
|
|
|
end
|
|
|
|
|
|
|
|
module:log("debug", "Bouncing message for user %s", actor_nick);
|
|
|
|
local reply = st.error_reply(
|
|
|
|
stanza,
|
|
|
|
-- error_type = 'wait' (see descriptions in RFC 6120 https://xmpp.org/rfcs/rfc6120.html#stanzas-error-syntax)
|
|
|
|
"wait",
|
|
|
|
-- error_condition = 'policy-violation' (see RFC 6120 Defined Error Conditions https://xmpp.org/rfcs/rfc6120.html#stanzas-error-conditions)
|
|
|
|
"policy-violation",
|
2024-02-16 14:16:44 +00:00
|
|
|
"You have exceeded the limit imposed by the slow mode in this room. You have to wait " .. duration .. " seconds between messages. Please try again later"
|
2024-02-13 15:45:12 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
-- Note: following commented lines were inspired by mod_muc_limits, but it seems it is not required.
|
|
|
|
-- if body then
|
|
|
|
-- reply:up():tag("body"):text(body):up();
|
|
|
|
-- end
|
|
|
|
-- local x = stanza:get_child("x", xmlns_muc);
|
|
|
|
-- if x then
|
|
|
|
-- reply:add_child(st.clone(x));
|
|
|
|
-- end
|
|
|
|
|
|
|
|
origin.send(reply);
|
|
|
|
return true; -- stoping propagation
|
|
|
|
end
|
|
|
|
|
|
|
|
module:hook("muc-occupant-groupchat", handle_groupchat);
|