123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161 |
- -- Maps MUC JIDs like room1@muc.foo.example.com to JIDs like [foo]room1@muc.example.com
- -- Must be loaded on the client host in Prosody
-
- -- It is recommended to set muc_mapper_domain_base to the main domain being served (example.com)
-
- local jid = require "util.jid";
-
- local filters = require "util.filters";
-
- local muc_domain_prefix = module:get_option_string("muc_mapper_domain_prefix", "conference");
-
- local muc_domain_base = module:get_option_string("muc_mapper_domain_base");
- if not muc_domain_base then
- module:log("warn", "No 'muc_mapper_domain_base' option set, disabling muc_mapper plugin inactive");
- return
- end
-
- -- The "real" MUC domain that we are proxying to
- local muc_domain = module:get_option_string("muc_mapper_domain", muc_domain_prefix.."."..muc_domain_base);
-
- local escaped_muc_domain_base = muc_domain_base:gsub("%p", "%%%1");
- local escaped_muc_domain_prefix = muc_domain_prefix:gsub("%p", "%%%1");
- -- The pattern used to extract the target subdomain (e.g. extract 'foo' from 'foo.muc.example.com')
- local target_subdomain_pattern = "^"..escaped_muc_domain_prefix..".([^%.]+)%."..escaped_muc_domain_base;
-
- -- table to store all incoming iqs without roomname in it, like discoinfo to the muc compoent
- local roomless_iqs = {};
-
- if not muc_domain then
- module:log("warn", "No 'muc_mapper_domain' option set, disabling muc_mapper plugin inactive");
- return
- end
-
-
- -- Utility function to check and convert a room JID from virtual room1@muc.foo.example.com to real [foo]room1@muc.example.com
- local function match_rewrite_to_jid(room_jid, stanza)
- local node, host, resource = jid.split(room_jid);
- local target_subdomain = host and host:match(target_subdomain_pattern);
- if not target_subdomain then
- module:log("debug", "No need to rewrite out 'to' %s", room_jid);
- return room_jid;
- end
- -- Ok, rewrite room_jid address to new format
- local new_node, new_host, new_resource;
- if node then
- new_node, new_host, new_resource = "["..target_subdomain.."]"..node, muc_domain, resource;
- else
- module:log("debug", "No room name provided so rewriting only host 'to' %s", room_jid);
- new_host, new_resource = muc_domain, resource;
-
- if (stanza.attr and stanza.attr.id) then
- roomless_iqs[stanza.attr.id] = stanza.attr.to;
- end
- end
- room_jid = jid.join(new_node, new_host, new_resource);
- module:log("debug", "Rewrote to %s", room_jid);
- return room_jid
- end
-
- -- Utility function to check and convert a room JID from real [foo]room1@muc.example.com to virtual room1@muc.foo.example.com
- local function match_rewrite_from_jid(room_jid, stanza)
- local node, host, resource = jid.split(room_jid);
- if host ~= muc_domain or not node then
- module:log("debug", "No need to rewrite %s (not from the MUC host) %s, %s", room_jid, stanza.attr.id, roomless_iqs[stanza.attr.id]);
-
- if (stanza.attr and stanza.attr.id and roomless_iqs[stanza.attr.id]) then
- local result = roomless_iqs[stanza.attr.id];
- roomless_iqs[stanza.attr.id] = nil;
- return result;
- end
-
- return room_jid;
- end
- local target_subdomain, target_node = node:match("^%[([^%]]+)%](.+)$");
- if not (target_node and target_subdomain) then
- module:log("debug", "Not rewriting... unexpected node format: %s", node);
- return room_jid;
- end
- -- Ok, rewrite room_jid address to pretty format
- local new_node, new_host, new_resource = target_node, muc_domain_prefix..".".. target_subdomain.."."..muc_domain_base, resource;
- room_jid = jid.join(new_node, new_host, new_resource);
- module:log("debug", "Rewrote to %s", room_jid);
- return room_jid
- end
-
-
- -- We must filter stanzas in order to hook in to all incoming and outgoing messaging which skips the stanza routers
- function filter_stanza(stanza)
- if stanza.name == "message" or stanza.name == "iq" or stanza.name == "presence" then
- module:log("debug", "Filtering stanza type %s to %s from %s",stanza.name,stanza.attr.to,stanza.attr.from);
- if stanza.name == "iq" then
- local conf = stanza:get_child('conference')
- if conf then
- module:log("debug", "Filtering stanza conference %s to %s from %s",conf.attr.room,stanza.attr.to,stanza.attr.from);
- conf.attr.room = match_rewrite_to_jid(conf.attr.room, stanza)
- end
- end
- if stanza.attr.to then
- stanza.attr.to = match_rewrite_to_jid(stanza.attr.to, stanza)
- end
- if stanza.attr.from then
- stanza.attr.from = match_rewrite_from_jid(stanza.attr.from, stanza)
- end
- end
- return stanza;
- end
-
- function filter_session(session)
- module:log("warn", "Session filters applied");
- -- filters.add_filter(session, "stanzas/in", filter_stanza_in);
- filters.add_filter(session, "stanzas/out", filter_stanza);
- end
-
- function module.load()
- if module.reloading then
- module:log("debug", "Reloading MUC mapper!");
- else
- module:log("debug", "First load of MUC mapper!");
- end
- filters.add_filter_hook(filter_session);
- end
-
- function module.unload()
- filters.remove_filter_hook(filter_session);
- end
-
-
- local function outgoing_stanza_rewriter(event)
- local stanza = event.stanza;
- if stanza.attr.to then
- stanza.attr.to = match_rewrite_to_jid(stanza.attr.to, stanza)
- end
- end
-
- local function incoming_stanza_rewriter(event)
- local stanza = event.stanza;
- if stanza.attr.from then
- stanza.attr.from = match_rewrite_from_jid(stanza.attr.from, stanza)
- end
- end
-
- -- The stanza rewriters helper functions are attached for all stanza router hooks
- local function hook_all_stanzas(handler, host_module, event_prefix)
- for _, stanza_type in ipairs({ "message", "presence", "iq" }) do
- for _, jid_type in ipairs({ "host", "bare", "full" }) do
- host_module:hook((event_prefix or "")..stanza_type.."/"..jid_type, handler);
- end
- end
- end
-
- function add_host(host)
- module:log("info", "Loading mod_muc_domain_mapper for host %s!", host);
- local host_module = module:context(host);
- hook_all_stanzas(incoming_stanza_rewriter, host_module);
- hook_all_stanzas(outgoing_stanza_rewriter, host_module, "pre-");
- end
-
- prosody.events.add_handler("host-activated", add_host);
- for host in pairs(prosody.hosts) do
- add_host(host);
- end
|