Comparison

mod_muc_moderation/mod_muc_moderation.lua @ 3897:3a96070f4a14

mod_muc_moderation: Initial commit of XEP-0425: Message Moderation
author Kim Alvefur <zash@zash.se>
date Sat, 22 Feb 2020 21:11:31 +0100
child 3900:06971a04216f
comparison
equal deleted inserted replaced
3896:987b203bb091 3897:3a96070f4a14
1 -- Imports
2 local dt = require "util.datetime";
3 local id = require "util.id";
4 local jid = require "util.jid";
5 local st = require "util.stanza";
6
7 -- Plugin dependencies
8 local mod_muc = module:depends "muc";
9
10 local muc_util = module:require "muc/util";
11 local valid_roles = muc_util.valid_roles;
12
13 local muc_log_archive = module:open_store("muc_log", "archive");
14
15 if not muc_log_archive.set then
16 module:log("warn", "Selected archive storage module does not support message replacement, no tombstones will be saved");
17 end
18
19 -- Namespaces
20 local xmlns_fasten = "urn:xmpp:fasten:0";
21 local xmlns_moderate = "urn:xmpp:message-moderate:0";
22 local xmlns_retract = "urn:xmpp:message-retract:0";
23
24 -- Discovering support
25 module:hook("muc-disco#info", function (event)
26 event.reply:tag("feature", { var = xmlns_moderate }):up();
27 end);
28
29 -- Main handling
30 module:hook("iq-set/bare/" .. xmlns_fasten .. ":apply-to", function (event)
31 local stanza, origin = event.stanza, event.origin;
32
33 -- Collect info we need
34 local apply_to = stanza.tags[1];
35 local moderate_tag = apply_to:get_child("moderate", xmlns_moderate);
36 if not moderate_tag then return end -- some other kind of fastening?
37
38 local reason = moderate_tag:get_child_text("reason");
39
40 local room_jid = stanza.attr.to;
41 local room_node = jid.split(room_jid);
42 local room = mod_muc.get_room_from_jid(room_jid);
43
44 local stanza_id = apply_to.attr.id;
45
46 -- Permissions
47 local actor = stanza.attr.from;
48 local actor_nick = room:get_occupant_jid(actor);
49 local affiliation = room:get_affiliation(actor);
50 local role = room:get_role(actor_nick) or room:get_default_role(affiliation);
51 if valid_roles[role or "none"] < valid_roles.moderator then
52 origin.send(st.error_reply(stanza, "auth", "forbidden", "Insufficient privileges"));
53 return true;
54 end
55
56 -- Original stanza to base tombstone on
57 local original, err;
58 if muc_log_archive.get then
59 original, err = muc_log_archive:get(room_node, stanza_id);
60 else
61 -- COMPAT missing :get API
62 err = "item-not-found";
63 for i, item in muc_log_archive:find(room_node, { key = stanza_id, limit = 1 }) do
64 if i == stanza_id then
65 original, err = item, nil;
66 end
67 end
68 end
69 if not original then
70 if err == "item-not-found" then
71 origin.send(st.error_reply(stanza, "modify", "item-not-found"));
72 else
73 origin.send(st.error_reply(stanza, "wait", "internal-server-error"));
74 end
75 return true;
76 end
77
78 -- Replacements
79 local tombstone = st.message({ from = original.attr.from, type = "groupchat", id = original.attr.id })
80 :tag("moderated", { xmlns = xmlns_moderate, by = actor_nick })
81 :tag("retracted", { xmlns = xmlns_retract, stamp = dt.datetime() }):up();
82
83 local announcement = st.message({ from = room_jid, type = "groupchat", id = id.medium(), })
84 :tag("apply-to", { xmlns = xmlns_fasten, id = stanza_id })
85 :tag("moderated", { xmlns = xmlns_moderate, by = actor_nick })
86 :tag("retract", { xmlns = xmlns_retract }):up();
87
88 if reason then
89 tombstone:text_tag("reason", reason);
90 announcement:text_tag("reason", reason);
91 end
92
93 if muc_log_archive.set then
94 -- Tombstone
95 local was_replaced = muc_log_archive:set(room_node, stanza_id, tombstone);
96 if not was_replaced then
97 origin.send(st.error_reply(stanza, "wait", "internal-server-error"));
98 return true;
99 end
100 end
101
102 -- Done, tell people about it
103 module:log("info", "Message with id '%s' in room %s moderated by %s, reason: %s", stanza_id, room_jid, actor, reason);
104 module:log("debug", ":broadcast(%s)", announcement);
105 room:broadcast(announcement);
106
107 origin.send(st.reply(stanza));
108 return true;
109 end);
110
111 module:hook("muc-message-is-historic", function (event)
112 -- Ensure moderation messages are stored
113 if event.stanza.attr.from == event.room.jid then
114 return event.stanza:get_child("apply-to", xmlns_fasten);
115 end
116 end, 1);