mirror of
https://gitcode.com/GitHub_Trending/ji/jitsi-meet.git
synced 2025-12-30 11:22:31 +00:00
145 lines
5.7 KiB
Lua
145 lines
5.7 KiB
Lua
--- This module removes identity information from presence stanzas when the
|
|
--- hideDisplayNameForGuests options are enabled
|
|
--- for a room.
|
|
|
|
--- To be enabled under the main muc component
|
|
local filters = require 'util.filters';
|
|
local st = require 'util.stanza';
|
|
|
|
local util = module:require 'util';
|
|
local filter_identity_from_presence = util.filter_identity_from_presence;
|
|
local get_room_by_name_and_subdomain = util.get_room_by_name_and_subdomain;
|
|
local is_admin = util.is_admin;
|
|
local ends_with = util.ends_with;
|
|
local internal_room_jid_match_rewrite = util.internal_room_jid_match_rewrite;
|
|
local NICK_NS = 'http://jabber.org/protocol/nick';
|
|
local DISPLAY_NAME_NS = 'http://jitsi.org/protocol/display-name';
|
|
|
|
local sessions = prosody.full_sessions;
|
|
|
|
local ignore_jwt_name = module:get_option_boolean('ignore_jwt_name', false);
|
|
|
|
-- we need to get the shared resource for joining moderators, as participants are marked as moderators
|
|
-- after joining which is after the filter for stanza/out, but we need to know will this participant be a moderator
|
|
local joining_moderator_participants = module:shared('moderators/joining_moderator_participants');
|
|
|
|
--- Filter presence sent to non-moderator members of a room when the hideDisplayNameForGuests option is set.
|
|
function filter_stanza_out(stanza, session)
|
|
if stanza.name ~= 'presence' or stanza.attr.type == 'error'
|
|
or stanza.attr.type == 'unavailable' or ends_with(stanza.attr.from, '/focus') then
|
|
return stanza;
|
|
end
|
|
|
|
local room = get_room_by_name_and_subdomain(session.jitsi_web_query_room, session.jitsi_web_query_prefix);
|
|
local shouldFilter = false;
|
|
|
|
if room and room._data.hideDisplayNameForGuests == true then
|
|
local occupant = room:get_occupant_by_real_jid(stanza.attr.to);
|
|
-- don't touch self-presence
|
|
if occupant and stanza.attr.from ~= internal_room_jid_match_rewrite(occupant.nick) then
|
|
shouldFilter = occupant.role ~= 'moderator' and not joining_moderator_participants[occupant.bare_jid];
|
|
end
|
|
end
|
|
|
|
if shouldFilter then
|
|
return filter_identity_from_presence(stanza);
|
|
else
|
|
return stanza;
|
|
end
|
|
end
|
|
|
|
-- When a participant has a name-readonly feature enabled, we need to ensure that the
|
|
-- nick element in the presence stanza is set to the user's name, if it exists.
|
|
-- If the user does not have a name, we remove the nick element.
|
|
-- This is to ensure that the name is not changed by the user, as it is read-only.
|
|
function filter_stanza_in(stanza, session)
|
|
if not session or not session.jitsi_meet_context_features
|
|
or session.jitsi_meet_context_features['name-readonly'] ~= true then
|
|
-- if the name-readonly feature is not set, there is nothing we need to do
|
|
return stanza;
|
|
end
|
|
|
|
if stanza.name ~= 'presence' or stanza.attr.type == 'error'
|
|
or stanza.attr.type == 'unavailable' or ends_with(stanza.attr.from, '/focus') then
|
|
return stanza;
|
|
end
|
|
|
|
-- if the user does not have a name in token and name is readonly, remove any nick element
|
|
if not session.jitsi_meet_context_user or not session.jitsi_meet_context_user.name then
|
|
stanza:remove_children('nick', NICK_NS);
|
|
return stanza;
|
|
end
|
|
|
|
local nick_element = stanza:get_child('nick', NICK_NS);
|
|
|
|
if nick_element and nick_element:get_text() ~= session.jitsi_meet_context_user.name then
|
|
stanza:remove_children('nick', NICK_NS);
|
|
stanza:tag('nick', { xmlns = NICK_NS }):text(session.jitsi_meet_context_user.name):up();
|
|
end
|
|
|
|
return stanza;
|
|
end
|
|
|
|
-- 'muc-add-history' is called in 'muc-broadcast-message' with priority 0
|
|
-- we want to clean up message after we have processed it for history, but do it for all messages
|
|
-- not only those that are added to history (as someone can forge a message and add 'no-store')
|
|
module:hook('muc-broadcast-message', function(event)
|
|
-- any message that is about to be delivered, clear up nick and display-name elements
|
|
event.stanza:remove_children('nick', NICK_NS);
|
|
event.stanza:remove_children('display-name', DISPLAY_NAME_NS);
|
|
end, -1);
|
|
|
|
module:hook('muc-add-history', function(event)
|
|
local room, stanza = event.room, event.stanza;
|
|
local name;
|
|
local source;
|
|
|
|
local occupant = room:get_occupant_by_nick(stanza.attr.from);
|
|
|
|
if occupant then
|
|
local session = sessions[occupant.jid];
|
|
if session and session.jitsi_meet_context_user then
|
|
if ignore_jwt_name then
|
|
name = occupant:get_presence():get_child_text('nick', NICK_NS);
|
|
else
|
|
name = session.jitsi_meet_context_user.name;
|
|
end
|
|
|
|
source = 'token';
|
|
else
|
|
name = occupant:get_presence():get_child_text('nick', NICK_NS);
|
|
source = 'guest';
|
|
end
|
|
else
|
|
-- if no occupant found it is a message from visitor let's check display-name source
|
|
local display_name_element = stanza:get_child('display-name', DISPLAY_NAME_NS);
|
|
if display_name_element and display_name_element.attr.source == 'visitor' then
|
|
name = display_name_element:get_text();
|
|
source = display_name_element.attr.source;
|
|
end
|
|
end
|
|
|
|
if name then
|
|
-- clone the stanza, so only history has the display_name extension
|
|
event.stanza = st.clone(stanza);
|
|
|
|
event.stanza:tag('display-name', {
|
|
xmlns = DISPLAY_NAME_NS,
|
|
source = source
|
|
}):text(name):up();
|
|
end
|
|
end);
|
|
|
|
function filter_session(session)
|
|
filters.add_filter(session, 'stanzas/out', filter_stanza_out, -100);
|
|
filters.add_filter(session, 'stanzas/in', filter_stanza_in, -100);
|
|
end
|
|
|
|
function module.load()
|
|
filters.add_filter_hook(filter_session);
|
|
end
|
|
|
|
function module.unload()
|
|
filters.remove_filter_hook(filter_session);
|
|
end
|