Software / code / prosody-modules
File
mod_muc_moderation_delay/delay.lib.lua @ 6319:63ef69b2f046
mod_http_oauth2: Assume Prosody 13.0+ roles are available
Per the README, 0.12 is not supported, so we should not need to worry
about this. Plus it is assumed to be present elsewhere and that would
throw errors.
| author | Kim Alvefur <zash@zash.se> |
|---|---|
| date | Wed, 02 Jul 2025 16:15:32 +0200 |
| parent | 5990:a86720654fb9 |
line wrap: on
line source
-- SPDX-FileCopyrightText: 2024 John Livingston <https://www.john-livingston.fr/> -- SPDX-License-Identifier: AGPL-3.0-only local st = require "util.stanza"; local timer = require "util.timer"; local get_time = require "util.time".now; local get_moderation_delay = module:require("config").get_moderation_delay; local muc_util = module:require "muc/util"; local valid_roles = muc_util.valid_roles; local moderation_delay_tag = "moderation-delay"; local xmlns_fasten = "urn:xmpp:fasten:0"; local xmlns_moderated_0 = "urn:xmpp:message-moderate:0"; local xmlns_retract_0 = "urn:xmpp:message-retract:0"; local xmlns_moderated_1 = "urn:xmpp:message-moderate:1"; local xmlns_retract_1 = "urn:xmpp:message-retract:1"; local xmlns_st_id = "urn:xmpp:sid:0"; local queued_stanza_id_timers = {}; -- tests if a stanza is a retractation message. local function is_retractation_for_stanza_id(stanza) -- XEP 0425 was revised in 2023. For now, mod_muc_moderation uses the previous version. -- But we will make the code compatible with both. local apply_to = stanza:get_child("apply-to", xmlns_fasten); if apply_to and apply_to.attr.id then local moderated = apply_to:get_child("moderated", xmlns_moderated_0); if moderated then local retract = moderated:get_child("retract", xmlns_retract_0); if retract then return apply_to.attr.id; end end end local moderated = stanza:get_child("moderated", xmlns_moderated_1); if moderated then if moderated:get_child("retract", xmlns_retract_1) then return moderated.attr.id; end end return nil; end -- handler for muc-broadcast-message local function handle_broadcast_message(event) local room, stanza = event.room, event.stanza; local delay = get_moderation_delay(room); if delay == nil then -- feature disabled on the room, go for it. return; end -- only delay groupchat messages with body. if stanza.attr.type ~= "groupchat" then return; end -- detect retractations: local retracted_stanza_id = is_retractation_for_stanza_id(stanza); if retracted_stanza_id then module:log("debug", "Got a retractation message for %s", retracted_stanza_id); if queued_stanza_id_timers[retracted_stanza_id] then module:log("info", "Got a retractation message, for message %s that is currently waiting for broadcast. Cancelling.", retracted_stanza_id); timer.stop(queued_stanza_id_timers[retracted_stanza_id]); queued_stanza_id_timers[retracted_stanza_id] = nil; -- and we continue... end end if not stanza:get_child("body") then -- Dont want to delay message without body. -- This is usually messages like "xxx is typing", or any other service message. -- This also should concern retractation messages. -- Clients that will receive retractation messages for message they never got, should just drop them. And that's ok. return; end local stanza_id = nil; -- message stanza id... can be nil! local stanza_id_child = stanza:get_child("stanza-id", xmlns_st_id); if not stanza_id_child then -- this can happen when muc is not archived! -- in such case, message retractation is not possible. -- so, this is a normal use case, and we should handle it properly. else stanza_id = stanza_id_child.attr.id; end local id = stanza.attr.id; if not id then -- message should always have an id, but just in case... module:log("warn", "Message has no id, wont delay it."); return; end -- Message must be delayed, except for: -- * room moderators -- * the user that sent the message (if they don't get the echo quickly, their clients could have weird behaviours) module:log("debug", "Message %s / %s must be delayed by %i seconds, sending first broadcast wave.", id, stanza_id, delay); local moderator_role_value = valid_roles["moderator"]; local cloned_stanza = st.clone(stanza); -- we must clone, to send a copy for the second wave. -- first of all, if the initiator occupant is not moderator, me must send to them. -- (delaying the echo message could have some quircks in some xmpp clients) if stanza.attr.from then local from_occupant = room:get_occupant_by_nick(stanza.attr.from); if from_occupant and valid_roles[from_occupant.role or "none"] < moderator_role_value then module:log("debug", "Message %s / %s must be sent separatly to it initiator %s.", id, stanza_id, delay, stanza.attr.from); room:route_to_occupant(from_occupant, stanza); end end -- adding a tag, so that moderators can know that this message is delayed. stanza:tag(moderation_delay_tag, { delay = "" .. delay; waiting = string.format("%i", math.floor(get_time() + delay)); }):up(); -- then, sending to moderators (and only moderators): room:broadcast(stanza, function (nick, occupant) if valid_roles[occupant.role or "none"] >= moderator_role_value then return true; end return false; end); local task = timer.add_task(delay, function () module:log("debug", "Message %s has been delayed, sending to remaining participants.", id); room:broadcast(cloned_stanza, function (nick, occupant) if valid_roles[occupant.role or "none"] >= moderator_role_value then return false; end if nick == stanza.attr.from then -- we already sent it to them (because they are moderator, or because we sent them separately) return false; end return true; end); end); if stanza_id then -- store it, so we can stop timer if there is a retractation. queued_stanza_id_timers[stanza_id] = task; end return true; -- stop the default broadcast_message processing. end return { handle_broadcast_message = handle_broadcast_message; };