1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
|
-- Prosody IM
-- Copyright (C) 2008-2010 Matthew Wild
-- Copyright (C) 2008-2010 Waqas Hussain
--
-- This project is MIT/X11 licensed. Please see the
-- COPYING file in the source package for more information.
--
local array = require "util.array";
local logger = require "util.logger";
local log = logger.init("modulemanager");
local config = require "core.configmanager";
local pluginloader = require "util.pluginloader";
local envload = require "util.envload";
local set = require "util.set";
local core_features = require "core.features".available;
local new_multitable = require "util.multitable".new;
local api = require "core.moduleapi"; -- Module API container
local prosody = prosody;
local hosts = prosody.hosts;
local xpcall = require "util.xpcall".xpcall;
local debug_traceback = debug.traceback;
local setmetatable, rawget = setmetatable, rawget;
local ipairs, pairs, type, t_insert = ipairs, pairs, type, table.insert;
local lua_version = _VERSION:match("5%.%d$");
local autoload_modules = {
prosody.platform,
"presence",
"message",
"iq",
"offline",
"c2s",
"s2s",
"s2s_auth_certs",
};
local component_inheritable_modules = {
"tls",
"saslauth",
"dialback",
"iq",
"s2s",
"s2s_bidi",
"smacks",
"server_contact_info",
};
-- We need this to let modules access the real global namespace
local _G = _G;
local _ENV = nil;
-- luacheck: std none
local loader = pluginloader.init({
load_filter_cb = function (path, content)
local metadata = {};
for line in content:gmatch("([^\r\n]+)\r?\n") do
local key, value = line:match("^%-%-%% *([%w_]+): *(.+)$");
if key then
value = value:gsub("%s+$", "");
metadata[key] = value;
end
end
if metadata.conflicts_core_features then
local conflicts_core_features = set.new(array.collect(metadata.conflicts_core_features:gmatch("[^, ]+")));
local conflicted_features = set.intersection(conflicts_core_features, core_features);
if not conflicted_features:empty() then
log("warn", "Not loading module, due to conflicting features '%s': %s", conflicted_features, path);
return; -- Don't load this module
end
end
if metadata.requires_core_features then
local required_features = set.new(array.collect(metadata.requires_core_features:gmatch("[^, ]+")));
local missing_features = required_features - core_features;
if not missing_features:empty() then
log("warn", "Not loading module, due to missing features '%s': %s", missing_features, path);
return; -- Don't load this module
end
end
return path, content, metadata;
end;
});
local load_modules_for_host, load, unload, reload, get_module, get_items;
local get_modules, is_loaded, module_has_method, call_module_method;
-- [host] = { [module] = module_env }
local modulemap = { ["*"] = {} };
-- Get the list of modules to be loaded on a host
local function get_modules_for_host(host)
local component = config.get(host, "component_module");
local global_modules_enabled = config.get("*", "modules_enabled");
local global_modules_disabled = config.get("*", "modules_disabled");
local host_modules_enabled = config.get(host, "modules_enabled");
local host_modules_disabled = config.get(host, "modules_disabled");
if host_modules_enabled == global_modules_enabled then host_modules_enabled = nil; end
if host_modules_disabled == global_modules_disabled then host_modules_disabled = nil; end
local global_modules = set.new(autoload_modules) + set.new(global_modules_enabled) - set.new(global_modules_disabled);
if component then
global_modules = set.intersection(set.new(component_inheritable_modules), global_modules);
end
local modules = (global_modules + set.new(host_modules_enabled)) - set.new(host_modules_disabled);
if modules:contains("vcard") and modules:contains("vcard_legacy") then
log("error", "The mod_vcard_legacy plugin replaces mod_vcard but both are enabled. Please update your config.");
modules:remove("vcard");
end
return modules, component;
end
-- Load modules when a host is activated
function load_modules_for_host(host)
local modules, component_module = get_modules_for_host(host);
-- Ensure component module is loaded first
if component_module then
load(host, component_module);
end
for module in modules do
load(host, module);
end
end
prosody.events.add_handler("host-activated", load_modules_for_host);
prosody.events.add_handler("host-deactivated", function (host)
modulemap[host] = nil;
end);
--- Private helpers ---
local function do_unload_module(host, name)
local mod = get_module(host, name);
if not mod then return nil, "module-not-loaded"; end
if module_has_method(mod, "unload") then
local ok, err = call_module_method(mod, "unload");
if (not ok) and err then
log("warn", "Non-fatal error unloading module '%s' on '%s': %s", name, host, err);
end
end
for object, event, handler in mod.module.event_handlers:iter(nil, nil, nil) do
object.remove_handler(event, handler);
end
if mod.module.items then -- remove items
local events = (host == "*" and prosody.events) or hosts[host].events;
for key,t in pairs(mod.module.items) do
for i = #t,1,-1 do
local value = t[i];
t[i] = nil;
events.fire_event("item-removed/"..key, {source = mod.module, item = value});
end
end
end
mod.module.loaded = false;
modulemap[host][name] = nil;
return true;
end
local function do_load_module(host, module_name, state)
if not (host and module_name) then
return nil, "insufficient-parameters";
elseif not hosts[host] and host ~= "*"then
return nil, "unknown-host";
end
if not modulemap[host] then
modulemap[host] = hosts[host].modules;
end
if modulemap[host][module_name] then
if not modulemap["*"][module_name] then
log("debug", "%s is already loaded for %s, so not loading again", module_name, host);
end
return nil, "module-already-loaded";
elseif modulemap["*"][module_name] then
local mod = modulemap["*"][module_name];
if module_has_method(mod, "add_host") then
local _log = logger.init(host..":"..module_name);
local host_module_api = setmetatable({
host = host, event_handlers = new_multitable(), items = {};
_log = _log, log = function (self, ...) return _log(...); end; --luacheck: ignore 212/self
},{
__index = modulemap["*"][module_name].module;
});
local host_module = setmetatable({ module = host_module_api }, { __index = mod });
host_module_api.environment = host_module;
modulemap[host][module_name] = host_module;
local ok, result, module_err = call_module_method(mod, "add_host", host_module_api);
if not ok or result == false then
modulemap[host][module_name] = nil;
return nil, ok and module_err or result;
end
return host_module;
end
return nil, "global-module-already-loaded";
end
local _log = logger.init(host..":"..module_name);
local api_instance = setmetatable({ name = module_name, host = host,
_log = _log, log = function (self, ...) return _log(...); end, --luacheck: ignore 212/self
event_handlers = new_multitable(), reloading = not not state,
saved_state = state~=true and state or nil }
, { __index = api });
local pluginenv = setmetatable({ module = api_instance }, { __index = _G });
api_instance.environment = pluginenv;
local mod, err = loader:load_code(module_name, nil, pluginenv);
if not mod then
log("error", "Unable to load module '%s': %s", module_name or "nil", err or "nil");
api_instance:set_status("error", "Failed to load (see log)");
return nil, err;
end
api_instance.path = err;
local custom_plugins = prosody.paths.installer;
if custom_plugins and err:sub(1, #custom_plugins+1) == custom_plugins.."/" then
-- Stage 1: Make it work (you are here)
-- Stage 2: Make it less hacky (TODO)
local manifest = {};
local luarocks_path = custom_plugins.."/lib/luarocks/rocks-"..lua_version;
local manifest_filename = luarocks_path.."/manifest";
local load_manifest, err = envload.envloadfile(manifest_filename, manifest);
if not load_manifest then
-- COMPAT Luarocks 2.x
log("debug", "Could not load LuaRocks 3.x manifest, trying 2.x", err);
luarocks_path = custom_plugins.."/lib/luarocks/rocks";
manifest_filename = luarocks_path.."/manifest";
load_manifest, err = envload.envloadfile(manifest_filename, manifest);
end
if not load_manifest then
log("error", "Could not load manifest of installed plugins: %s", err, load_manifest);
else
local ok, err = xpcall(load_manifest, debug_traceback);
if not ok then
log("error", "Could not load manifest of installed plugins: %s", err);
elseif type(manifest.modules) ~= "table" then
log("debug", "Expected 'table' but manifest.modules = %q", manifest.modules);
log("error", "Can't look up resource path for mod_%s because '%s' does not appear to be a LuaRocks manifest", module_name, manifest_filename);
else
local versions = manifest.modules["mod_"..module_name];
if type(versions) == "table" and versions[1] then
-- Not going to deal with multiple installed versions
api_instance.resource_path = luarocks_path.."/"..versions[1];
else
log("debug", "mod_%s does not appear in the installation manifest", module_name);
end
end
end
end
modulemap[host][module_name] = pluginenv;
local ok, err = xpcall(mod, debug_traceback);
if ok then
-- Call module's "load"
if module_has_method(pluginenv, "load") then
ok, err = call_module_method(pluginenv, "load");
if not ok then
log("warn", "Error loading module '%s' on '%s': %s", module_name, host, err or "nil");
api_instance:set_status("warn", "Error during load (see log)");
end
end
api_instance.reloading, api_instance.saved_state = nil, nil;
if api_instance.host == "*" then
if not api_instance.global then -- COMPAT w/pre-0.9
if host ~= "*" then
log("warn", "mod_%s: Setting module.host = '*' deprecated, call module:set_global() instead", module_name);
end
api_instance:set_global();
end
modulemap[host][module_name] = nil;
modulemap[api_instance.host][module_name] = pluginenv;
if host ~= api_instance.host and module_has_method(pluginenv, "add_host") then
-- Now load the module again onto the host it was originally being loaded on
ok, err = do_load_module(host, module_name);
end
end
end
if not ok then
modulemap[api_instance.host][module_name] = nil;
log("error", "Error initializing module '%s' on '%s': %s", module_name, host, err or "nil");
api_instance:set_status("warn", "Error during load (see log)");
else
api_instance:set_status("core", "Loaded", false);
end
return ok and pluginenv, err;
end
local function do_reload_module(host, name)
local mod = get_module(host, name);
if not mod then return nil, "module-not-loaded"; end
local _mod, err = loader:load_code(name); -- checking for syntax errors
if not _mod then
log("error", "Unable to load module '%s': %s", name or "nil", err or "nil");
return nil, err;
end
local saved;
if module_has_method(mod, "save") then
-- FIXME What goes in 'err' here?
local ok, ret, err = call_module_method(mod, "save"); -- luacheck: ignore 211/err
if ok then
saved = ret;
else
log("warn", "Error saving module '%s:%s' state: %s", host, name, ret);
if not config.get(host, "force_module_reload") then
log("warn", "Aborting reload due to error, set force_module_reload to ignore this");
return nil, "save-state-failed";
else
log("warn", "Continuing with reload (using the force)");
end
end
end
mod.module.reloading = true;
do_unload_module(host, name);
local ok, err = do_load_module(host, name, saved or true);
if ok then
mod = get_module(host, name);
if module_has_method(mod, "restore") then
local ok, err = call_module_method(mod, "restore", saved or {})
if (not ok) and err then
log("warn", "Error restoring module '%s' from '%s': %s", name, host, err);
end
end
end
return ok and mod, err;
end
--- Public API ---
-- Load a module and fire module-loaded event
function load(host, name)
local mod, err = do_load_module(host, name);
if mod then
(hosts[mod.module.host] or prosody).events.fire_event("module-loaded", { module = name, host = mod.module.host });
end
return mod, err;
end
-- Unload a module and fire module-unloaded
function unload(host, name)
local ok, err = do_unload_module(host, name);
if ok then
(hosts[host] or prosody).events.fire_event("module-unloaded", { module = name, host = host });
end
return ok, err;
end
function reload(host, name)
local mod, err = do_reload_module(host, name);
if mod then
modulemap[host][name].module.reloading = true;
(hosts[host] or prosody).events.fire_event("module-reloaded", { module = name, host = host });
mod.module.reloading = nil;
elseif not is_loaded(host, name) then
(hosts[host] or prosody).events.fire_event("module-unloaded", { module = name, host = host });
end
return mod, err;
end
function get_module(host, name)
return modulemap[host] and modulemap[host][name];
end
function get_items(key, host)
local result = {};
local modules = modulemap[host];
if not key or not host or not modules then return nil; end
for _, module in pairs(modules) do
local mod = module.module;
if mod.items and mod.items[key] then
for _, value in ipairs(mod.items[key]) do
t_insert(result, value);
end
end
end
return result;
end
function get_modules(host)
return modulemap[host];
end
function is_loaded(host, name)
return modulemap[host] and modulemap[host][name] and true;
end
function module_has_method(module, method)
return type(rawget(module.module, method)) == "function";
end
function call_module_method(module, method, ...)
local f = rawget(module.module, method);
if type(f) == "function" then
return xpcall(f, debug_traceback, ...);
else
return false, "no-such-method";
end
end
return {
get_modules_for_host = get_modules_for_host;
load_modules_for_host = load_modules_for_host;
load = load;
unload = unload;
reload = reload;
get_module = get_module;
get_items = get_items;
get_modules = get_modules;
is_loaded = is_loaded;
module_has_method = module_has_method;
call_module_method = call_module_method;
loader = loader;
};
|