mirror of
https://github.com/bjc/prosody.git
synced 2025-04-03 21:27:38 +03:00
Merge 0.9->0.10
This commit is contained in:
commit
27b4049481
136 changed files with 3493 additions and 2266 deletions
53
configure
vendored
53
configure
vendored
|
@ -96,32 +96,31 @@ do
|
|||
--ostype=*)
|
||||
OSTYPE="$value"
|
||||
OSTYPE_SET=yes
|
||||
if [ "$OSTYPE" = "debian" ]
|
||||
then LUA_SUFFIX="5.1";
|
||||
LUA_SUFFIX_SET=yes
|
||||
RUNWITH="lua5.1"
|
||||
LUA_INCDIR=/usr/include/lua5.1;
|
||||
LUA_INCDIR_SET=yes
|
||||
CFLAGS="$CFLAGS -D_GNU_SOURCE"
|
||||
fi
|
||||
if [ "$OSTYPE" = "macosx" ]
|
||||
then LUA_INCDIR=/usr/local/include;
|
||||
LUA_INCDIR_SET=yes
|
||||
LUA_LIBDIR=/usr/local/lib
|
||||
LUA_LIBDIR_SET=yes
|
||||
LDFLAGS="-bundle -undefined dynamic_lookup"
|
||||
fi
|
||||
if [ "$OSTYPE" = "linux" ]
|
||||
then LUA_INCDIR=/usr/local/include;
|
||||
if [ "$OSTYPE" = "debian" ]; then
|
||||
LUA_SUFFIX="5.1";
|
||||
LUA_SUFFIX_SET=yes
|
||||
RUNWITH="lua5.1"
|
||||
LUA_INCDIR=/usr/include/lua5.1;
|
||||
LUA_INCDIR_SET=yes
|
||||
CFLAGS="$CFLAGS -D_GNU_SOURCE"
|
||||
fi
|
||||
if [ "$OSTYPE" = "macosx" ]; then
|
||||
LUA_INCDIR=/usr/local/include;
|
||||
LUA_INCDIR_SET=yes
|
||||
LUA_LIBDIR=/usr/local/lib
|
||||
LUA_LIBDIR_SET=yes
|
||||
LDFLAGS="-bundle -undefined dynamic_lookup"
|
||||
fi
|
||||
if [ "$OSTYPE" = "linux" ]; then
|
||||
LUA_INCDIR=/usr/local/include;
|
||||
LUA_INCDIR_SET=yes
|
||||
LUA_LIBDIR=/usr/local/lib
|
||||
LUA_LIBDIR_SET=yes
|
||||
CFLAGS="-Wall -fPIC"
|
||||
CFLAGS="$CFLAGS -D_GNU_SOURCE"
|
||||
CFLAGS="-Wall -fPIC -D_GNU_SOURCE"
|
||||
LDFLAGS="-shared"
|
||||
fi
|
||||
if [ "$OSTYPE" = "freebsd" -o "$OSTYPE" = "openbsd" ]
|
||||
then LUA_INCDIR="/usr/local/include/lua51"
|
||||
fi
|
||||
if [ "$OSTYPE" = "freebsd" -o "$OSTYPE" = "openbsd" ]; then
|
||||
LUA_INCDIR="/usr/local/include/lua51"
|
||||
LUA_INCDIR_SET=yes
|
||||
CFLAGS="-Wall -fPIC -I/usr/local/include"
|
||||
LDFLAGS="-I/usr/local/include -L/usr/local/lib -shared"
|
||||
|
@ -129,10 +128,10 @@ do
|
|||
LUA_SUFFIX_SET=yes
|
||||
LUA_DIR=/usr/local
|
||||
LUA_DIR_SET=yes
|
||||
fi
|
||||
if [ "$OSTYPE" = "openbsd" ]
|
||||
then LUA_INCDIR="/usr/local/include";
|
||||
fi
|
||||
fi
|
||||
if [ "$OSTYPE" = "openbsd" ]; then
|
||||
LUA_INCDIR="/usr/local/include";
|
||||
fi
|
||||
;;
|
||||
--datadir=*)
|
||||
DATADIR="$value"
|
||||
|
@ -291,7 +290,7 @@ then
|
|||
IDNA_LIBS="$ICU_FLAGS"
|
||||
CFLAGS="$CFLAGS -DUSE_STRINGPREP_ICU"
|
||||
fi
|
||||
if [ "$IDN_LIBRARY" = "idn" ]
|
||||
if [ "$IDN_LIBRARY" = "idn" ]
|
||||
then
|
||||
IDNA_LIBS="-l$IDN_LIB"
|
||||
fi
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -12,6 +12,7 @@ local ssl = ssl;
|
|||
local ssl_newcontext = ssl and ssl.newcontext;
|
||||
|
||||
local tostring = tostring;
|
||||
local pairs = pairs;
|
||||
local type = type;
|
||||
local io_open = io.open;
|
||||
|
||||
|
@ -30,68 +31,82 @@ end
|
|||
module "certmanager"
|
||||
|
||||
-- Global SSL options if not overridden per-host
|
||||
local default_ssl_config = configmanager.get("*", "ssl");
|
||||
local default_capath = "/etc/ssl/certs";
|
||||
local default_verify = (ssl and ssl.x509 and { "peer", "client_once", }) or "none";
|
||||
local default_options = { "no_sslv2", "cipher_server_preference", luasec_has_noticket and "no_ticket" or nil };
|
||||
local default_verifyext = { "lsec_continue", "lsec_ignore_purpose" };
|
||||
local global_ssl_config = configmanager.get("*", "ssl");
|
||||
|
||||
local core_defaults = {
|
||||
capath = "/etc/ssl/certs";
|
||||
protocol = "sslv23";
|
||||
verify = (ssl and ssl.x509 and { "peer", "client_once", }) or "none";
|
||||
options = { "no_sslv2", "no_sslv3", "cipher_server_preference", luasec_has_noticket and "no_ticket" or nil };
|
||||
verifyext = { "lsec_continue", "lsec_ignore_purpose" };
|
||||
curve = "secp384r1";
|
||||
ciphers = "HIGH+kEDH:HIGH+kEECDH:HIGH:!PSK:!SRP:!3DES:!aNULL";
|
||||
}
|
||||
local path_options = { -- These we pass through resolve_path()
|
||||
key = true, certificate = true, cafile = true, capath = true, dhparam = true
|
||||
}
|
||||
|
||||
if ssl and not luasec_has_verifyext and ssl.x509 then
|
||||
-- COMPAT mw/luasec-hg
|
||||
for i=1,#default_verifyext do -- Remove lsec_ prefix
|
||||
default_verify[#default_verify+1] = default_verifyext[i]:sub(6);
|
||||
for i=1,#core_defaults.verifyext do -- Remove lsec_ prefix
|
||||
core_defaults.verify[#core_defaults.verify+1] = core_defaults.verifyext[i]:sub(6);
|
||||
end
|
||||
end
|
||||
if luasec_has_no_compression and configmanager.get("*", "ssl_compression") ~= true then
|
||||
default_options[#default_options+1] = "no_compression";
|
||||
end
|
||||
|
||||
if luasec_has_no_compression then -- Has no_compression? Then it has these too...
|
||||
default_options[#default_options+1] = "single_dh_use";
|
||||
default_options[#default_options+1] = "single_ecdh_use";
|
||||
core_defaults.options[#core_defaults.options+1] = "single_dh_use";
|
||||
core_defaults.options[#core_defaults.options+1] = "single_ecdh_use";
|
||||
if configmanager.get("*", "ssl_compression") ~= true then
|
||||
core_defaults.options[#core_defaults.options+1] = "no_compression";
|
||||
end
|
||||
end
|
||||
|
||||
function create_context(host, mode, user_ssl_config)
|
||||
user_ssl_config = user_ssl_config or default_ssl_config;
|
||||
user_ssl_config = user_ssl_config or {}
|
||||
user_ssl_config.mode = mode;
|
||||
|
||||
if not ssl then return nil, "LuaSec (required for encryption) was not found"; end
|
||||
if not user_ssl_config then return nil, "No SSL/TLS configuration present for "..host; end
|
||||
|
||||
local ssl_config = {
|
||||
mode = mode;
|
||||
protocol = user_ssl_config.protocol or "sslv23";
|
||||
key = resolve_path(config_path, user_ssl_config.key);
|
||||
password = user_ssl_config.password or function() log("error", "Encrypted certificate for %s requires 'ssl' 'password' to be set in config", host); end;
|
||||
certificate = resolve_path(config_path, user_ssl_config.certificate);
|
||||
capath = resolve_path(config_path, user_ssl_config.capath or default_capath);
|
||||
cafile = resolve_path(config_path, user_ssl_config.cafile);
|
||||
verify = user_ssl_config.verify or default_verify;
|
||||
verifyext = user_ssl_config.verifyext or default_verifyext;
|
||||
options = user_ssl_config.options or default_options;
|
||||
depth = user_ssl_config.depth;
|
||||
curve = user_ssl_config.curve or "secp384r1";
|
||||
ciphers = user_ssl_config.ciphers or "HIGH+kEDH:HIGH+kEECDH:HIGH:!PSK:!SRP:!3DES:!aNULL";
|
||||
dhparam = user_ssl_config.dhparam;
|
||||
};
|
||||
|
||||
if global_ssl_config then
|
||||
for option,default_value in pairs(global_ssl_config) do
|
||||
if not user_ssl_config[option] then
|
||||
user_ssl_config[option] = default_value;
|
||||
end
|
||||
end
|
||||
end
|
||||
for option,default_value in pairs(core_defaults) do
|
||||
if not user_ssl_config[option] then
|
||||
user_ssl_config[option] = default_value;
|
||||
end
|
||||
end
|
||||
user_ssl_config.password = user_ssl_config.password or function() log("error", "Encrypted certificate for %s requires 'ssl' 'password' to be set in config", host); end;
|
||||
for option in pairs(path_options) do
|
||||
if type(user_ssl_config[option]) == "string" then
|
||||
user_ssl_config[option] = resolve_path(config_path, user_ssl_config[option]);
|
||||
end
|
||||
end
|
||||
|
||||
if not user_ssl_config.key then return nil, "No key present in SSL/TLS configuration for "..host; end
|
||||
if not user_ssl_config.certificate then return nil, "No certificate present in SSL/TLS configuration for "..host; end
|
||||
|
||||
-- LuaSec expects dhparam to be a callback that takes two arguments.
|
||||
-- We ignore those because it is mostly used for having a separate
|
||||
-- set of params for EXPORT ciphers, which we don't have by default.
|
||||
if type(ssl_config.dhparam) == "string" then
|
||||
local f, err = io_open(resolve_path(config_path, ssl_config.dhparam));
|
||||
if type(user_ssl_config.dhparam) == "string" then
|
||||
local f, err = io_open(user_ssl_config.dhparam);
|
||||
if not f then return nil, "Could not open DH parameters: "..err end
|
||||
local dhparam = f:read("*a");
|
||||
f:close();
|
||||
ssl_config.dhparam = function() return dhparam; end
|
||||
user_ssl_config.dhparam = function() return dhparam; end
|
||||
end
|
||||
|
||||
local ctx, err = ssl_newcontext(ssl_config);
|
||||
local ctx, err = ssl_newcontext(user_ssl_config);
|
||||
|
||||
-- COMPAT: LuaSec 0.4.1 ignores the cipher list from the config, so we have to take
|
||||
-- care of it ourselves...
|
||||
if ctx and ssl_config.ciphers then
|
||||
-- COMPAT Older LuaSec ignores the cipher list from the config, so we have to take care
|
||||
-- of it ourselves (W/A for #x)
|
||||
if ctx and user_ssl_config.ciphers then
|
||||
local success;
|
||||
success, err = ssl.context.setcipher(ctx, ssl_config.ciphers);
|
||||
success, err = ssl.context.setcipher(ctx, user_ssl_config.ciphers);
|
||||
if not success then ctx = nil; end
|
||||
end
|
||||
|
||||
|
@ -100,9 +115,9 @@ function create_context(host, mode, user_ssl_config)
|
|||
local file = err:match("^error loading (.-) %(");
|
||||
if file then
|
||||
if file == "private key" then
|
||||
file = ssl_config.key or "your private key";
|
||||
file = user_ssl_config.key or "your private key";
|
||||
elseif file == "certificate" then
|
||||
file = ssl_config.certificate or "your certificate file";
|
||||
file = user_ssl_config.certificate or "your certificate file";
|
||||
end
|
||||
local reason = err:match("%((.+)%)$") or "some reason";
|
||||
if reason == "Permission denied" then
|
||||
|
@ -125,7 +140,7 @@ function create_context(host, mode, user_ssl_config)
|
|||
end
|
||||
|
||||
function reload_ssl_config()
|
||||
default_ssl_config = configmanager.get("*", "ssl");
|
||||
global_ssl_config = configmanager.get("*", "ssl");
|
||||
end
|
||||
|
||||
prosody.events.add_handler("config-reloaded", reload_ssl_config);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -73,7 +73,7 @@ do
|
|||
-- Some normalization
|
||||
parent_path = parent_path:gsub("%"..path_sep.."+$", "");
|
||||
path = path:gsub("^%.%"..path_sep.."+", "");
|
||||
|
||||
|
||||
local is_relative;
|
||||
if path_sep == "/" and path:sub(1,1) ~= "/" then
|
||||
is_relative = true;
|
||||
|
@ -85,7 +85,7 @@ do
|
|||
end
|
||||
end
|
||||
return path;
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- Helper function to convert a glob to a Lua pattern
|
||||
|
@ -167,7 +167,7 @@ do
|
|||
set(config, env.__currenthost or "*", k, v);
|
||||
end
|
||||
});
|
||||
|
||||
|
||||
rawset(env, "__currenthost", "*") -- Default is global
|
||||
function env.VirtualHost(name)
|
||||
if rawget(config, name) and rawget(config[name], "component_module") then
|
||||
|
@ -185,7 +185,7 @@ do
|
|||
end;
|
||||
end
|
||||
env.Host, env.host = env.VirtualHost, env.VirtualHost;
|
||||
|
||||
|
||||
function env.Component(name)
|
||||
if rawget(config, name) and rawget(config[name], "defined") and not rawget(config[name], "component_module") then
|
||||
error(format("Component %q clashes with previously defined Host %q, for services use a sub-domain like conference.%s",
|
||||
|
@ -201,7 +201,7 @@ do
|
|||
set(config, name or "*", option_name, option_value);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
return function (module)
|
||||
if type(module) == "string" then
|
||||
set(config, name, "component_module", module);
|
||||
|
@ -211,7 +211,7 @@ do
|
|||
end
|
||||
end
|
||||
env.component = env.Component;
|
||||
|
||||
|
||||
function env.Include(file)
|
||||
if file:match("[*?]") then
|
||||
local path_pos, glob = file:match("()([^"..path_sep.."]+)$");
|
||||
|
@ -240,26 +240,26 @@ do
|
|||
end
|
||||
end
|
||||
env.include = env.Include;
|
||||
|
||||
|
||||
function env.RunScript(file)
|
||||
return dofile(resolve_relative_path(config_file:gsub("[^"..path_sep.."]+$", ""), file));
|
||||
end
|
||||
|
||||
|
||||
local chunk, err = envload(data, "@"..config_file, env);
|
||||
|
||||
|
||||
if not chunk then
|
||||
return nil, err;
|
||||
end
|
||||
|
||||
|
||||
local ok, err = pcall(chunk);
|
||||
|
||||
|
||||
if not ok then
|
||||
return nil, err;
|
||||
end
|
||||
|
||||
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
end
|
||||
|
||||
return _M;
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -35,7 +35,7 @@ local hosts_loaded_once;
|
|||
local function load_enabled_hosts(config)
|
||||
local defined_hosts = config or configmanager.getconfig();
|
||||
local activated_any_host;
|
||||
|
||||
|
||||
for host, host_config in pairs(defined_hosts) do
|
||||
if host ~= "*" and host_config.enabled ~= false then
|
||||
if not host_config.component_module then
|
||||
|
@ -44,11 +44,11 @@ local function load_enabled_hosts(config)
|
|||
activate(host, host_config);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if not activated_any_host then
|
||||
log("error", "No active VirtualHost entries in the config file. This may cause unexpected behaviour as no modules will be loaded.");
|
||||
end
|
||||
|
||||
|
||||
prosody_events.fire_event("hosts-activated", defined_hosts);
|
||||
hosts_loaded_once = true;
|
||||
end
|
||||
|
@ -93,7 +93,7 @@ function activate(host, host_config)
|
|||
log("warn", "%s: Option '%s' has no effect for virtual hosts - put it in the server-wide section instead", host, option_name);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
log((hosts_loaded_once and "info") or "debug", "Activated host: %s", host);
|
||||
prosody_events.fire_event("host-activated", host);
|
||||
return true;
|
||||
|
@ -104,11 +104,11 @@ function deactivate(host, reason)
|
|||
if not host_session then return nil, "The host "..tostring(host).." is not activated"; end
|
||||
log("info", "Deactivating host: %s", host);
|
||||
prosody_events.fire_event("host-deactivating", { host = host, host_session = host_session, reason = reason });
|
||||
|
||||
|
||||
if type(reason) ~= "table" then
|
||||
reason = { condition = "host-gone", text = tostring(reason or "This server has stopped serving "..host) };
|
||||
end
|
||||
|
||||
|
||||
-- Disconnect local users, s2s connections
|
||||
-- TODO: These should move to mod_c2s and mod_s2s (how do they know they're being unloaded and not reloaded?)
|
||||
if host_session.sessions then
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -48,7 +48,7 @@ local function add_rule(sink_config)
|
|||
if sink_maker then
|
||||
-- Create sink
|
||||
local sink = sink_maker(sink_config);
|
||||
|
||||
|
||||
-- Set sink for all chosen levels
|
||||
for level in pairs(get_levels(sink_config.levels or logging_levels)) do
|
||||
logger.add_level_sink(level, sink);
|
||||
|
@ -63,7 +63,7 @@ end
|
|||
-- the log_sink_types table.
|
||||
function apply_sink_rules(sink_type)
|
||||
if type(logging_config) == "table" then
|
||||
|
||||
|
||||
for _, level in ipairs(logging_levels) do
|
||||
if type(logging_config[level]) == "string" then
|
||||
local value = logging_config[level];
|
||||
|
@ -82,7 +82,7 @@ function apply_sink_rules(sink_type)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
for _, sink_config in ipairs(logging_config) do
|
||||
if (type(sink_config) == "table" and sink_config.to == sink_type) then
|
||||
add_rule(sink_config);
|
||||
|
@ -128,7 +128,7 @@ function get_levels(criteria, set)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
for _, level in ipairs(criteria) do
|
||||
set[level] = true;
|
||||
end
|
||||
|
@ -138,12 +138,12 @@ end
|
|||
-- Initialize config, etc. --
|
||||
function reload_logging()
|
||||
local old_sink_types = {};
|
||||
|
||||
|
||||
for name, sink_maker in pairs(log_sink_types) do
|
||||
old_sink_types[name] = sink_maker;
|
||||
log_sink_types[name] = nil;
|
||||
end
|
||||
|
||||
|
||||
logger.reset();
|
||||
|
||||
local debug_mode = config.get("*", "debug");
|
||||
|
@ -155,12 +155,12 @@ function reload_logging()
|
|||
default_timestamp = "%b %d %H:%M:%S";
|
||||
|
||||
logging_config = config.get("*", "log") or default_logging;
|
||||
|
||||
|
||||
|
||||
|
||||
for name, sink_maker in pairs(old_sink_types) do
|
||||
log_sink_types[name] = sink_maker;
|
||||
end
|
||||
|
||||
|
||||
prosody.events.fire_event("logging-reloaded");
|
||||
end
|
||||
|
||||
|
@ -179,11 +179,11 @@ local sourcewidth = 20;
|
|||
|
||||
function log_sink_types.stdout(config)
|
||||
local timestamps = config.timestamps;
|
||||
|
||||
|
||||
if timestamps == true then
|
||||
timestamps = default_timestamp; -- Default format
|
||||
end
|
||||
|
||||
|
||||
return function (name, level, message, ...)
|
||||
sourcewidth = math_max(#name+2, sourcewidth);
|
||||
local namelen = #name;
|
||||
|
@ -200,7 +200,7 @@ end
|
|||
|
||||
do
|
||||
local do_pretty_printing = true;
|
||||
|
||||
|
||||
local logstyles = {};
|
||||
if do_pretty_printing then
|
||||
logstyles["info"] = getstyle("bold");
|
||||
|
@ -212,7 +212,7 @@ do
|
|||
if not do_pretty_printing then
|
||||
return log_sink_types.stdout(config);
|
||||
end
|
||||
|
||||
|
||||
local timestamps = config.timestamps;
|
||||
|
||||
if timestamps == true then
|
||||
|
@ -222,7 +222,7 @@ do
|
|||
return function (name, level, message, ...)
|
||||
sourcewidth = math_max(#name+2, sourcewidth);
|
||||
local namelen = #name;
|
||||
|
||||
|
||||
if timestamps then
|
||||
io_write(os_date(timestamps), " ");
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2008-2012 Matthew Wild
|
||||
-- Copyright (C) 2008-2012 Waqas Hussain
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -44,7 +44,7 @@ function api:get_host()
|
|||
end
|
||||
|
||||
function api:get_host_type()
|
||||
return self.host ~= "*" and hosts[self.host].type or nil;
|
||||
return (self.host == "*" and "global") or hosts[self.host].type or "local";
|
||||
end
|
||||
|
||||
function api:set_global()
|
||||
|
@ -74,7 +74,7 @@ end
|
|||
function api:has_identity(category, type, name)
|
||||
for _, id in ipairs(self:get_host_items("identity")) do
|
||||
if id.category == category and id.type == type and id.name == name then
|
||||
return true;
|
||||
return true;
|
||||
end
|
||||
end
|
||||
return false;
|
||||
|
@ -113,6 +113,10 @@ function api:hook_tag(xmlns, name, handler, priority)
|
|||
end
|
||||
api.hook_stanza = api.hook_tag; -- COMPAT w/pre-0.9
|
||||
|
||||
function api:unhook(event, handler)
|
||||
return self:unhook_object_event((hosts[self.host] or prosody).events, event, handler);
|
||||
end
|
||||
|
||||
function api:require(lib)
|
||||
local f, n = pluginloader.load_code(self.name, lib..".lib.lua", self.environment);
|
||||
if not f then
|
||||
|
@ -252,21 +256,21 @@ function api:get_option_array(name, ...)
|
|||
if value == nil then
|
||||
return nil;
|
||||
end
|
||||
|
||||
|
||||
if type(value) ~= "table" then
|
||||
return array{ value }; -- Assume any non-list is a single-item list
|
||||
end
|
||||
|
||||
|
||||
return array():append(value); -- Clone
|
||||
end
|
||||
|
||||
function api:get_option_set(name, ...)
|
||||
local value = self:get_option_array(name, ...);
|
||||
|
||||
|
||||
if value == nil then
|
||||
return nil;
|
||||
end
|
||||
|
||||
|
||||
return set.new(value);
|
||||
end
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -29,7 +29,7 @@ pcall = function(f, ...)
|
|||
return xpcall(function() return f(unpack(params, 1, n)) end, function(e) return tostring(e).."\n"..debug_traceback(); end);
|
||||
end
|
||||
|
||||
local autoload_modules = {"presence", "message", "iq", "offline", "c2s", "s2s"};
|
||||
local autoload_modules = {prosody.platform, "presence", "message", "iq", "offline", "c2s", "s2s"};
|
||||
local component_inheritable_modules = {"tls", "saslauth", "dialback", "iq", "s2s"};
|
||||
|
||||
-- We need this to let modules access the real global namespace
|
||||
|
@ -45,28 +45,28 @@ local modulemap = { ["*"] = {} };
|
|||
-- Load modules when a host is activated
|
||||
function load_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);
|
||||
|
||||
|
||||
-- COMPAT w/ pre 0.8
|
||||
if modules:contains("console") then
|
||||
log("error", "The mod_console plugin has been renamed to mod_admin_telnet. Please update your config.");
|
||||
modules:remove("console");
|
||||
modules:add("admin_telnet");
|
||||
end
|
||||
|
||||
|
||||
if component then
|
||||
load(host, component);
|
||||
end
|
||||
|
@ -84,18 +84,18 @@ end);
|
|||
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
|
||||
|
@ -117,11 +117,11 @@ local function do_load_module(host, module_name, state)
|
|||
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
|
||||
log("warn", "%s is already loaded for %s, so not loading again", module_name, host);
|
||||
return nil, "module-already-loaded";
|
||||
|
@ -147,7 +147,7 @@ local function do_load_module(host, module_name, state)
|
|||
end
|
||||
return nil, "global-module-already-loaded";
|
||||
end
|
||||
|
||||
|
||||
|
||||
|
||||
local _log = logger.init(host..":"..module_name);
|
||||
|
@ -158,7 +158,7 @@ local function do_load_module(host, module_name, state)
|
|||
|
||||
local pluginenv = setmetatable({ module = api_instance }, { __index = _G });
|
||||
api_instance.environment = pluginenv;
|
||||
|
||||
|
||||
local mod, err = pluginloader.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");
|
||||
|
|
|
@ -89,7 +89,7 @@ function activate(service_name)
|
|||
if not service_info then
|
||||
return nil, "Unknown service: "..service_name;
|
||||
end
|
||||
|
||||
|
||||
local listener = service_info.listener;
|
||||
|
||||
local config_prefix = (service_info.config_prefix or service_name).."_";
|
||||
|
@ -105,7 +105,7 @@ function activate(service_name)
|
|||
or listener.default_interface -- COMPAT w/pre0.9
|
||||
or default_interfaces
|
||||
bind_interfaces = set.new(type(bind_interfaces)~="table" and {bind_interfaces} or bind_interfaces);
|
||||
|
||||
|
||||
local bind_ports = config.get("*", config_prefix.."ports")
|
||||
or service_info.default_ports
|
||||
or {service_info.default_port
|
||||
|
@ -115,7 +115,7 @@ function activate(service_name)
|
|||
|
||||
local mode, ssl = listener.default_mode or default_mode;
|
||||
local hooked_ports = {};
|
||||
|
||||
|
||||
for interface in bind_interfaces do
|
||||
for port in bind_ports do
|
||||
local port_number = tonumber(port);
|
||||
|
@ -190,7 +190,7 @@ function register_service(service_name, service_info)
|
|||
log("error", "Failed to activate service '%s': %s", service_name, err or "unknown error");
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
fire_event("service-added", { name = service_name, service = service_info });
|
||||
return true;
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -100,7 +100,7 @@ function load_roster(username, host)
|
|||
log("warn", "roster for %s has a self-contact", jid);
|
||||
end
|
||||
if not err then
|
||||
hosts[host].events.fire_event("roster-load", username, host, roster);
|
||||
hosts[host].events.fire_event("roster-load", { username = username, host = host, roster = roster });
|
||||
end
|
||||
return roster, err;
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -70,14 +70,14 @@ end
|
|||
function destroy_session(session, reason)
|
||||
if session.destroyed then return; end
|
||||
(session.log or log)("debug", "Destroying "..tostring(session.direction).." session "..tostring(session.from_host).."->"..tostring(session.to_host)..(reason and (": "..reason) or ""));
|
||||
|
||||
|
||||
if session.direction == "outgoing" then
|
||||
hosts[session.from_host].s2sout[session.to_host] = nil;
|
||||
session:bounce_sendq(reason);
|
||||
elseif session.direction == "incoming" then
|
||||
incoming_s2s[session] = nil;
|
||||
end
|
||||
|
||||
|
||||
local event_data = { session = session, reason = reason };
|
||||
if session.type == "s2sout" then
|
||||
fire_event("s2sout-destroyed", event_data);
|
||||
|
@ -90,7 +90,7 @@ function destroy_session(session, reason)
|
|||
hosts[session.to_host].events.fire_event("s2sin-destroyed", event_data);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
retire_session(session, reason); -- Clean session until it is GC'd
|
||||
return true;
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -44,7 +44,7 @@ function new_session(conn)
|
|||
session.ip = conn:ip();
|
||||
local conn_name = "c2s"..tostring(session):match("[a-f0-9]+$");
|
||||
session.log = logger.init(conn_name);
|
||||
|
||||
|
||||
return session;
|
||||
end
|
||||
|
||||
|
@ -73,19 +73,19 @@ end
|
|||
function destroy_session(session, err)
|
||||
(session.log or log)("debug", "Destroying session for %s (%s@%s)%s", session.full_jid or "(unknown)", session.username or "(unknown)", session.host or "(unknown)", err and (": "..err) or "");
|
||||
if session.destroyed then return; end
|
||||
|
||||
|
||||
-- Remove session/resource from user's session list
|
||||
if session.full_jid then
|
||||
local host_session = hosts[session.host];
|
||||
|
||||
|
||||
-- Allow plugins to prevent session destruction
|
||||
if host_session.events.fire_event("pre-resource-unbind", {session=session, error=err}) then
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
host_session.sessions[session.username].sessions[session.resource] = nil;
|
||||
full_sessions[session.full_jid] = nil;
|
||||
|
||||
|
||||
if not next(host_session.sessions[session.username].sessions) then
|
||||
log("debug", "All resources of %s are now offline", session.username);
|
||||
host_session.sessions[session.username] = nil;
|
||||
|
@ -94,7 +94,7 @@ function destroy_session(session, err)
|
|||
|
||||
host_session.events.fire_event("resource-unbind", {session=session, error=err});
|
||||
end
|
||||
|
||||
|
||||
retire_session(session);
|
||||
end
|
||||
|
||||
|
@ -119,7 +119,7 @@ function bind_resource(session, resource)
|
|||
resource = resourceprep(resource);
|
||||
resource = resource ~= "" and resource or uuid_generate();
|
||||
--FIXME: Randomly-generated resources must be unique per-user, and never conflict with existing
|
||||
|
||||
|
||||
if not hosts[session.host].sessions[session.username] then
|
||||
local sessions = { sessions = {} };
|
||||
hosts[session.host].sessions[session.username] = sessions;
|
||||
|
@ -156,12 +156,12 @@ function bind_resource(session, resource)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
session.resource = resource;
|
||||
session.full_jid = session.username .. '@' .. session.host .. '/' .. resource;
|
||||
hosts[session.host].sessions[session.username].sessions[resource] = session;
|
||||
full_sessions[session.full_jid] = session;
|
||||
|
||||
|
||||
local err;
|
||||
session.roster, err = rm_load_roster(session.username, session.host);
|
||||
if err then
|
||||
|
@ -176,9 +176,9 @@ function bind_resource(session, resource)
|
|||
session.log("error", "Roster loading failed: %s", err);
|
||||
return nil, "cancel", "internal-server-error", "Error loading roster";
|
||||
end
|
||||
|
||||
|
||||
hosts[session.host].events.fire_event("resource-bind", {session=session});
|
||||
|
||||
|
||||
return true;
|
||||
end
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -196,7 +196,7 @@ function core_route_stanza(origin, stanza)
|
|||
-- Auto-detect origin if not specified
|
||||
origin = origin or hosts[from_host];
|
||||
if not origin then return false; end
|
||||
|
||||
|
||||
if hosts[host] then
|
||||
-- old stanza routing code removed
|
||||
core_post_stanza(origin, stanza);
|
||||
|
|
|
@ -37,7 +37,7 @@ function initialize_host(host)
|
|||
local item = event.item;
|
||||
stores_available:set(host, item.name, item);
|
||||
end);
|
||||
|
||||
|
||||
host_session.events.add_handler("item-removed/storage-provider", function (event)
|
||||
local item = event.item;
|
||||
stores_available:set(host, item.name, nil);
|
||||
|
@ -70,7 +70,7 @@ function get_driver(host, store)
|
|||
if not driver_name then
|
||||
driver_name = config.get(host, "default_storage") or "internal";
|
||||
end
|
||||
|
||||
|
||||
local driver = load_driver(host, driver_name);
|
||||
if not driver then
|
||||
log("warn", "Falling back to null driver for %s storage on %s", store, host);
|
||||
|
|
|
@ -10,7 +10,6 @@ local modulemanager = require "core.modulemanager";
|
|||
local log = require "util.logger".init("usermanager");
|
||||
local type = type;
|
||||
local ipairs = ipairs;
|
||||
local pairs = pairs;
|
||||
local jid_bare = require "util.jid".bare;
|
||||
local jid_prep = require "util.jid".prep;
|
||||
local config = require "core.configmanager";
|
||||
|
@ -39,7 +38,7 @@ local provider_mt = { __index = new_null_provider() };
|
|||
function initialize_host(host)
|
||||
local host_session = hosts[host];
|
||||
if host_session.type ~= "local" then return; end
|
||||
|
||||
|
||||
host_session.events.add_handler("item-added/auth-provider", function (event)
|
||||
local provider = event.item;
|
||||
local auth_provider = config.get(host, "authentication") or default_provider;
|
||||
|
@ -115,10 +114,10 @@ function is_admin(jid, host)
|
|||
local is_admin;
|
||||
jid = jid_bare(jid);
|
||||
host = host or "*";
|
||||
|
||||
|
||||
local host_admins = config.get(host, "admins");
|
||||
local global_admins = config.get("*", "admins");
|
||||
|
||||
|
||||
if host_admins and host_admins ~= global_admins then
|
||||
if type(host_admins) == "table" then
|
||||
for _,admin in ipairs(host_admins) do
|
||||
|
@ -131,7 +130,7 @@ function is_admin(jid, host)
|
|||
log("error", "Option 'admins' for host '%s' is not a list", host);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if not is_admin and global_admins then
|
||||
if type(global_admins) == "table" then
|
||||
for _,admin in ipairs(global_admins) do
|
||||
|
@ -144,7 +143,7 @@ function is_admin(jid, host)
|
|||
log("error", "Global option 'admins' is not a list");
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
-- Still not an admin, check with auth provider
|
||||
if not is_admin and host ~= "*" and hosts[host].users and hosts[host].users.is_admin then
|
||||
is_admin = hosts[host].users.is_admin(jid);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -61,7 +61,7 @@ local function parser(data, handlers, ns_separator)
|
|||
while #data == 0 do data = coroutine.yield(); end
|
||||
return data:sub(1,1);
|
||||
end
|
||||
|
||||
|
||||
local ns = { xml = "http://www.w3.org/XML/1998/namespace" };
|
||||
ns.__index = ns;
|
||||
local function apply_ns(name, dodefault)
|
||||
|
@ -100,7 +100,7 @@ local function parser(data, handlers, ns_separator)
|
|||
ns = getmetatable(ns);
|
||||
return tag;
|
||||
end
|
||||
|
||||
|
||||
while true do
|
||||
if peek() == "<" then
|
||||
local elem = read_until(">"):sub(2,-2);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -64,7 +64,7 @@ function new_async_socket(sock, resolver)
|
|||
if resolver.socketset[conn] == resolver.best_server and resolver.best_server == #servers then
|
||||
log("error", "Exhausted all %d configured DNS servers, next lookup will try %s again", #servers, servers[1]);
|
||||
end
|
||||
|
||||
|
||||
resolver:servfail(conn); -- Let the magic commence
|
||||
end
|
||||
end
|
||||
|
@ -72,7 +72,7 @@ function new_async_socket(sock, resolver)
|
|||
if not handler then
|
||||
return nil, err;
|
||||
end
|
||||
|
||||
|
||||
handler.settimeout = function () end
|
||||
handler.setsockname = function (_, ...) return sock:setsockname(...); end
|
||||
handler.setpeername = function (_, ...) peername = (...); local ret = sock:setpeername(...); _:set_send(dummy_send); return ret; end
|
||||
|
|
25
net/dns.lua
25
net/dns.lua
|
@ -14,6 +14,7 @@
|
|||
|
||||
local socket = require "socket";
|
||||
local timer = require "util.timer";
|
||||
local new_ip = require "util.ip".new_ip;
|
||||
|
||||
local _, windows = pcall(require, "util.windows");
|
||||
local is_windows = (_ and windows) or os.getenv("WINDIR");
|
||||
|
@ -597,11 +598,12 @@ function resolver:adddefaultnameservers() -- - - - - adddefaultnameservers
|
|||
if resolv_conf then
|
||||
for line in resolv_conf:lines() do
|
||||
line = line:gsub("#.*$", "")
|
||||
:match('^%s*nameserver%s+(.*)%s*$');
|
||||
:match('^%s*nameserver%s+([%x:%.]*)%s*$');
|
||||
if line then
|
||||
line:gsub("%f[%d.](%d+%.%d+%.%d+%.%d+)%f[^%d.]", function (address)
|
||||
self:addnameserver(address)
|
||||
end);
|
||||
local ip = new_ip(line);
|
||||
if ip then
|
||||
self:addnameserver(ip.addr);
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
@ -621,7 +623,12 @@ function resolver:getsocket(servernum) -- - - - - - - - - - - - - getsocket
|
|||
if sock then return sock; end
|
||||
|
||||
local err;
|
||||
sock, err = socket.udp();
|
||||
local peer = self.server[servernum];
|
||||
if peer:find(":") then
|
||||
sock, err = socket.udp6();
|
||||
else
|
||||
sock, err = socket.udp();
|
||||
end
|
||||
if sock and self.socket_wrapper then sock, err = self.socket_wrapper(sock, self); end
|
||||
if not sock then
|
||||
return nil, err;
|
||||
|
@ -629,7 +636,7 @@ function resolver:getsocket(servernum) -- - - - - - - - - - - - - getsocket
|
|||
sock:settimeout(0);
|
||||
-- todo: attempt to use a random port, fallback to 0
|
||||
sock:setsockname('*', 0);
|
||||
sock:setpeername(self.server[servernum], 53);
|
||||
sock:setpeername(peer, 53);
|
||||
self.socket[servernum] = sock;
|
||||
self.socketset[sock] = servernum;
|
||||
return sock;
|
||||
|
@ -746,7 +753,7 @@ function resolver:query(qname, qtype, qclass) -- - - - - - - - - - -- query
|
|||
return nil, err;
|
||||
end
|
||||
conn:send (o.packet)
|
||||
|
||||
|
||||
if timer and self.timeout then
|
||||
local num_servers = #self.server;
|
||||
local i = 1;
|
||||
|
@ -842,7 +849,7 @@ function resolver:receive(rset) -- - - - - - - - - - - - - - - - - receive
|
|||
-- retire the query
|
||||
local queries = self.active[response.header.id];
|
||||
queries[response.question.raw] = nil;
|
||||
|
||||
|
||||
if not next(queries) then self.active[response.header.id] = nil; end
|
||||
if not next(self.active) then self:closeall(); end
|
||||
|
||||
|
@ -857,7 +864,7 @@ function resolver:receive(rset) -- - - - - - - - - - - - - - - - - receive
|
|||
set(self.wanted, q.class, q.type, q.name, nil);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
30
net/http.lua
30
net/http.lua
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -37,7 +37,7 @@ function listener.onconnect(conn)
|
|||
if req.query then
|
||||
t_insert(request_line, 4, "?"..req.query);
|
||||
end
|
||||
|
||||
|
||||
conn:write(t_concat(request_line));
|
||||
local t = { [2] = ": ", [4] = "\r\n" };
|
||||
for k, v in pairs(req.headers) do
|
||||
|
@ -45,7 +45,7 @@ function listener.onconnect(conn)
|
|||
conn:write(t_concat(t));
|
||||
end
|
||||
conn:write("\r\n");
|
||||
|
||||
|
||||
if req.body then
|
||||
conn:write(req.body);
|
||||
end
|
||||
|
@ -81,12 +81,12 @@ local function request_reader(request, data, err)
|
|||
end
|
||||
destroy_request(request);
|
||||
end
|
||||
|
||||
|
||||
if not data then
|
||||
error_cb(err);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
local function success_cb(r)
|
||||
if request.callback then
|
||||
request.callback(r.body, r.code, r, request);
|
||||
|
@ -105,18 +105,18 @@ end
|
|||
local function handleerr(err) log("error", "Traceback[http]: %s", traceback(tostring(err), 2)); end
|
||||
function request(u, ex, callback)
|
||||
local req = url.parse(u);
|
||||
|
||||
|
||||
if not (req and req.host) then
|
||||
callback(nil, 0, req);
|
||||
return nil, "invalid-url";
|
||||
end
|
||||
|
||||
|
||||
if not req.path then
|
||||
req.path = "/";
|
||||
end
|
||||
|
||||
|
||||
local method, headers, body;
|
||||
|
||||
|
||||
local host, port = req.host, req.port;
|
||||
local host_header = host;
|
||||
if (port == "80" and req.scheme == "http")
|
||||
|
@ -130,7 +130,7 @@ function request(u, ex, callback)
|
|||
["Host"] = host_header;
|
||||
["User-Agent"] = "Prosody XMPP Server";
|
||||
};
|
||||
|
||||
|
||||
if req.userinfo then
|
||||
headers["Authorization"] = "Basic "..b64(req.userinfo);
|
||||
end
|
||||
|
@ -150,16 +150,16 @@ function request(u, ex, callback)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
-- Attach to request object
|
||||
req.method, req.headers, req.body = method, headers, body;
|
||||
|
||||
|
||||
local using_https = req.scheme == "https";
|
||||
if using_https and not ssl_available then
|
||||
error("SSL not available, unable to contact https URL");
|
||||
end
|
||||
local port_number = port and tonumber(port) or (using_https and 443 or 80);
|
||||
|
||||
|
||||
-- Connect the socket, and wrap it with net.server
|
||||
local conn = socket.tcp();
|
||||
conn:settimeout(10);
|
||||
|
@ -168,7 +168,7 @@ function request(u, ex, callback)
|
|||
callback(nil, 0, req);
|
||||
return nil, err;
|
||||
end
|
||||
|
||||
|
||||
local sslctx = false;
|
||||
if using_https then
|
||||
sslctx = ex and ex.sslctx or { mode = "client", protocol = "sslv23", options = { "no_sslv2" } };
|
||||
|
@ -176,7 +176,7 @@ function request(u, ex, callback)
|
|||
|
||||
req.handler, req.conn = assert(server.wrapclient(conn, host, port_number, listener, "*a", sslctx));
|
||||
req.write = function (...) return req.handler:write(...); end
|
||||
|
||||
|
||||
req.callback = function (content, code, request, response) log("debug", "Calling callback, status %s", code or "---"); return select(2, xpcall(function () return callback(content, code, request, response) end, handleerr)); end
|
||||
req.reader = request_reader;
|
||||
req.state = "status";
|
||||
|
|
|
@ -204,7 +204,7 @@ function handle_request(conn, request, finish_cb)
|
|||
err_code, err = 400, "Missing or invalid 'Host' header";
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if err then
|
||||
response.status_code = err_code;
|
||||
response:send(events.fire_event("http-error", { code = err_code, message = err }));
|
||||
|
@ -250,7 +250,7 @@ function _M.send_response(response, body)
|
|||
if response.finished then return; end
|
||||
response.finished = true;
|
||||
response.conn._http_open_response = nil;
|
||||
|
||||
|
||||
local status_line = "HTTP/"..response.request.httpversion.." "..(response.status or codes[response.status_code]);
|
||||
local headers = response.headers;
|
||||
body = body or response.body or "";
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -115,10 +115,10 @@ end )( )
|
|||
local interface_mt
|
||||
do
|
||||
interface_mt = {}; interface_mt.__index = interface_mt;
|
||||
|
||||
|
||||
local addevent = base.addevent
|
||||
local coroutine_wrap, coroutine_yield = coroutine.wrap,coroutine.yield
|
||||
|
||||
|
||||
-- Private methods
|
||||
function interface_mt:_position(new_position)
|
||||
self.position = new_position or self.position
|
||||
|
@ -127,7 +127,7 @@ do
|
|||
function interface_mt:_close()
|
||||
return self:_destroy();
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:_start_connection(plainssl) -- should be called from addclient
|
||||
local callback = function( event )
|
||||
if EV_TIMEOUT == event then -- timeout during connection
|
||||
|
@ -268,12 +268,12 @@ do
|
|||
interfacelist( "delete", self )
|
||||
return true
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:_lock(nointerface, noreading, nowriting) -- lock or unlock this interface or events
|
||||
self.nointerface, self.noreading, self.nowriting = nointerface, noreading, nowriting
|
||||
return nointerface, noreading, nowriting
|
||||
end
|
||||
|
||||
|
||||
--TODO: Deprecate
|
||||
function interface_mt:lock_read(switch)
|
||||
if switch then
|
||||
|
@ -300,7 +300,7 @@ do
|
|||
end
|
||||
return self._connections
|
||||
end
|
||||
|
||||
|
||||
-- Public methods
|
||||
function interface_mt:write(data)
|
||||
if self.nowriting then return nil, "locked" end
|
||||
|
@ -343,27 +343,27 @@ do
|
|||
return true
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:socket()
|
||||
return self.conn
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:server()
|
||||
return self._server or self;
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:port()
|
||||
return self._port
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:serverport()
|
||||
return self._serverport
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:ip()
|
||||
return self._ip
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:ssl()
|
||||
return self._usingssl
|
||||
end
|
||||
|
@ -372,15 +372,15 @@ do
|
|||
function interface_mt:type()
|
||||
return self._type or "client"
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:connections()
|
||||
return self._connections
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:address()
|
||||
return self.addr
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:set_sslctx(sslctx)
|
||||
self._sslctx = sslctx;
|
||||
if sslctx then
|
||||
|
@ -396,11 +396,11 @@ do
|
|||
end
|
||||
return self._pattern;
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:set_send(new_send)
|
||||
-- No-op, we always use the underlying connection's send
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:starttls(sslctx, call_onconnect)
|
||||
debug( "try to start ssl at client id:", self.id )
|
||||
local err
|
||||
|
@ -429,19 +429,20 @@ do
|
|||
self.starttls = false;
|
||||
return true
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:setoption(option, value)
|
||||
if self.conn.setoption then
|
||||
return self.conn:setoption(option, value);
|
||||
end
|
||||
return false, "setoption not implemented";
|
||||
end
|
||||
|
||||
|
||||
function interface_mt:setlistener(listener)
|
||||
self.onconnect, self.ondisconnect, self.onincoming, self.ontimeout, self.onstatus
|
||||
= listener.onconnect, listener.ondisconnect, listener.onincoming, listener.ontimeout, listener.onstatus;
|
||||
self.onconnect, self.ondisconnect, self.onincoming, self.ontimeout, self.onreadtimeout, self.onstatus
|
||||
= listener.onconnect, listener.ondisconnect, listener.onincoming,
|
||||
listener.ontimeout, listener.onreadtimeout, listener.onstatus;
|
||||
end
|
||||
|
||||
|
||||
-- Stub handlers
|
||||
function interface_mt:onconnect()
|
||||
end
|
||||
|
@ -451,6 +452,12 @@ do
|
|||
end
|
||||
function interface_mt:ontimeout()
|
||||
end
|
||||
function interface_mt:onreadtimeout()
|
||||
self.fatalerror = "timeout during receiving"
|
||||
debug( "connection failed:", self.fatalerror )
|
||||
self:_close()
|
||||
self.eventread = nil
|
||||
end
|
||||
function interface_mt:ondrain()
|
||||
end
|
||||
function interface_mt:onstatus()
|
||||
|
@ -478,6 +485,7 @@ do
|
|||
ondisconnect = listener.ondisconnect; -- will be called when client disconnects
|
||||
onincoming = listener.onincoming; -- will be called when client sends data
|
||||
ontimeout = listener.ontimeout; -- called when fatal socket timeout occurs
|
||||
onreadtimeout = listener.onreadtimeout; -- called when socket inactivity timeout occurs
|
||||
ondrain = listener.ondrain; -- called when writebuffer is empty
|
||||
onstatus = listener.onstatus; -- called for status changes (e.g. of SSL/TLS)
|
||||
eventread = false, eventwrite = false, eventclose = false,
|
||||
|
@ -492,7 +500,7 @@ do
|
|||
noreading = false, nowriting = false; -- locks of the read/writecallback
|
||||
startsslcallback = false; -- starting handshake callback
|
||||
position = false; -- position of client in interfacelist
|
||||
|
||||
|
||||
-- Properties
|
||||
_ip = ip, _port = port, _server = server, _pattern = pattern,
|
||||
_serverport = (server and server:port() or nil),
|
||||
|
@ -568,7 +576,7 @@ do
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
interface.readcallback = function( event ) -- called on read events
|
||||
--vdebug( "new client read event, id/ip/port:", tostring(interface.id), tostring(ip), tostring(port) )
|
||||
if interface.noreading or interface.fatalerror then -- leave this event
|
||||
|
@ -576,61 +584,56 @@ do
|
|||
interface.eventread = nil
|
||||
return -1
|
||||
end
|
||||
if EV_TIMEOUT == event then -- took too long to get some data from client -> disconnect
|
||||
interface.fatalerror = "timeout during receiving"
|
||||
debug( "connection failed:", interface.fatalerror )
|
||||
if EV_TIMEOUT == event and interface:onreadtimeout() ~= true then
|
||||
return -1 -- took too long to get some data from client -> disconnect
|
||||
end
|
||||
if interface._usingssl then -- handle luasec
|
||||
if interface.eventwritetimeout then -- ok, in the past writecallback was regged
|
||||
local ret = interface.writecallback( ) -- call it
|
||||
--vdebug( "tried to write in readcallback, result:", tostring(ret) )
|
||||
end
|
||||
if interface.eventreadtimeout then
|
||||
interface.eventreadtimeout:close( )
|
||||
interface.eventreadtimeout = nil
|
||||
end
|
||||
end
|
||||
local buffer, err, part = interface.conn:receive( interface._pattern ) -- receive buffer with "pattern"
|
||||
--vdebug( "read data:", tostring(buffer), "error:", tostring(err), "part:", tostring(part) )
|
||||
buffer = buffer or part
|
||||
if buffer and #buffer > cfg.MAX_READ_LENGTH then -- check buffer length
|
||||
interface.fatalerror = "receive buffer exceeded"
|
||||
debug( "fatal error:", interface.fatalerror )
|
||||
interface:_close()
|
||||
interface.eventread = nil
|
||||
return -1
|
||||
else -- can read
|
||||
if interface._usingssl then -- handle luasec
|
||||
if interface.eventwritetimeout then -- ok, in the past writecallback was regged
|
||||
local ret = interface.writecallback( ) -- call it
|
||||
--vdebug( "tried to write in readcallback, result:", tostring(ret) )
|
||||
end
|
||||
if err and ( err ~= "timeout" and err ~= "wantread" ) then
|
||||
if "wantwrite" == err then -- need to read on write event
|
||||
if not interface.eventwrite then -- register new write event if needed
|
||||
interface.eventwrite = addevent( base, interface.conn, EV_WRITE, interface.writecallback, cfg.WRITE_TIMEOUT )
|
||||
end
|
||||
if interface.eventreadtimeout then
|
||||
interface.eventreadtimeout:close( )
|
||||
interface.eventreadtimeout = nil
|
||||
end
|
||||
end
|
||||
local buffer, err, part = interface.conn:receive( interface._pattern ) -- receive buffer with "pattern"
|
||||
--vdebug( "read data:", tostring(buffer), "error:", tostring(err), "part:", tostring(part) )
|
||||
buffer = buffer or part
|
||||
if buffer and #buffer > cfg.MAX_READ_LENGTH then -- check buffer length
|
||||
interface.fatalerror = "receive buffer exceeded"
|
||||
debug( "fatal error:", interface.fatalerror )
|
||||
interface.eventreadtimeout = addevent( base, nil, EV_TIMEOUT,
|
||||
function( )
|
||||
interface:_close()
|
||||
end, cfg.READ_TIMEOUT
|
||||
)
|
||||
debug( "wantwrite during read attempt, reg it in writecallback but dont know what really happens next..." )
|
||||
-- to be honest i dont know what happens next, if it is allowed to first read, the write etc...
|
||||
else -- connection was closed or fatal error
|
||||
interface.fatalerror = err
|
||||
debug( "connection failed in read event:", interface.fatalerror )
|
||||
interface:_close()
|
||||
interface.eventread = nil
|
||||
return -1
|
||||
end
|
||||
if err and ( err ~= "timeout" and err ~= "wantread" ) then
|
||||
if "wantwrite" == err then -- need to read on write event
|
||||
if not interface.eventwrite then -- register new write event if needed
|
||||
interface.eventwrite = addevent( base, interface.conn, EV_WRITE, interface.writecallback, cfg.WRITE_TIMEOUT )
|
||||
end
|
||||
interface.eventreadtimeout = addevent( base, nil, EV_TIMEOUT,
|
||||
function( )
|
||||
interface:_close()
|
||||
end, cfg.READ_TIMEOUT
|
||||
)
|
||||
debug( "wantwrite during read attempt, reg it in writecallback but dont know what really happens next..." )
|
||||
-- to be honest i dont know what happens next, if it is allowed to first read, the write etc...
|
||||
else -- connection was closed or fatal error
|
||||
interface.fatalerror = err
|
||||
debug( "connection failed in read event:", interface.fatalerror )
|
||||
interface:_close()
|
||||
interface.eventread = nil
|
||||
return -1
|
||||
end
|
||||
else
|
||||
interface.onincoming( interface, buffer, err ) -- send new data to listener
|
||||
end
|
||||
if interface.noreading then
|
||||
interface.eventread = nil;
|
||||
return -1;
|
||||
end
|
||||
return EV_READ, cfg.READ_TIMEOUT
|
||||
else
|
||||
interface.onincoming( interface, buffer, err ) -- send new data to listener
|
||||
end
|
||||
if interface.noreading then
|
||||
interface.eventread = nil;
|
||||
return -1;
|
||||
end
|
||||
return EV_READ, cfg.READ_TIMEOUT
|
||||
end
|
||||
|
||||
client:settimeout( 0 ) -- set non blocking
|
||||
|
@ -646,7 +649,7 @@ do
|
|||
debug "creating server interface..."
|
||||
local interface = {
|
||||
_connections = 0;
|
||||
|
||||
|
||||
conn = server;
|
||||
onconnect = listener.onconnect; -- will be called when new client connected
|
||||
eventread = false; -- read event handler
|
||||
|
@ -654,7 +657,7 @@ do
|
|||
readcallback = false; -- read event callback
|
||||
fatalerror = false; -- error message
|
||||
nointerface = true; -- lock/unlock parameter
|
||||
|
||||
|
||||
_ip = addr, _port = port, _pattern = pattern,
|
||||
_sslctx = sslctx;
|
||||
}
|
||||
|
@ -693,12 +696,12 @@ do
|
|||
clientinterface:_start_session( true )
|
||||
end
|
||||
debug( "accepted incoming client connection from:", client_ip or "<unknown IP>", client_port or "<unknown port>", "to", port or "<unknown port>");
|
||||
|
||||
|
||||
client, err = server:accept() -- try to accept again
|
||||
end
|
||||
return EV_READ
|
||||
end
|
||||
|
||||
|
||||
server:settimeout( 0 )
|
||||
setmetatable(interface, interface_mt)
|
||||
interfacelist( "add", interface )
|
||||
|
@ -741,7 +744,7 @@ do
|
|||
return interface, client
|
||||
--function handleclient( client, ip, port, server, pattern, listener, _, sslctx ) -- creates an client interface
|
||||
end
|
||||
|
||||
|
||||
function addclient( addr, serverport, listener, pattern, localaddr, localport, sslcfg, startssl )
|
||||
local client, err = socket.tcp() -- creating new socket
|
||||
if not client then
|
||||
|
@ -832,14 +835,14 @@ end
|
|||
|
||||
local function link(sender, receiver, buffersize)
|
||||
local sender_locked;
|
||||
|
||||
|
||||
function receiver:ondrain()
|
||||
if sender_locked then
|
||||
sender:resume();
|
||||
sender_locked = nil;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
function sender:onincoming(data)
|
||||
receiver:write(data);
|
||||
if receiver.writebufferlen >= buffersize then
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
--
|
||||
--
|
||||
-- server.lua by blastbeat of the luadch project
|
||||
-- Re-used here under the MIT/X Consortium License
|
||||
--
|
||||
--
|
||||
-- Modifications (C) 2008-2010 Matthew Wild, Waqas Hussain
|
||||
--
|
||||
|
||||
|
@ -145,7 +145,7 @@ _tcpbacklog = 128 -- some kind of hint to the OS
|
|||
_maxsendlen = 51000 * 1024 -- max len of send buffer
|
||||
_maxreadlen = 25000 * 1024 -- max len of read buffer
|
||||
|
||||
_checkinterval = 1200000 -- interval in secs to check idle clients
|
||||
_checkinterval = 30 -- interval in secs to check idle clients
|
||||
_sendtimeout = 60000 -- allowed send idle time in secs
|
||||
_readtimeout = 6 * 60 * 60 -- allowed read idle time in secs
|
||||
|
||||
|
@ -284,6 +284,7 @@ wrapconnection = function( server, listeners, socket, ip, serverport, clientport
|
|||
local status = listeners.onstatus
|
||||
local disconnect = listeners.ondisconnect
|
||||
local drain = listeners.ondrain
|
||||
local onreadtimeout = listeners.onreadtimeout;
|
||||
|
||||
local bufferqueue = { } -- buffer array
|
||||
local bufferqueuelen = 0 -- end of buffer array
|
||||
|
@ -312,11 +313,14 @@ wrapconnection = function( server, listeners, socket, ip, serverport, clientport
|
|||
handler.disconnect = function( )
|
||||
return disconnect
|
||||
end
|
||||
handler.onreadtimeout = onreadtimeout;
|
||||
|
||||
handler.setlistener = function( self, listeners )
|
||||
dispatch = listeners.onincoming
|
||||
disconnect = listeners.ondisconnect
|
||||
status = listeners.onstatus
|
||||
drain = listeners.ondrain
|
||||
handler.onreadtimeout = listeners.onreadtimeout
|
||||
end
|
||||
handler.getstats = function( )
|
||||
return readtraffic, sendtraffic
|
||||
|
@ -608,7 +612,7 @@ wrapconnection = function( server, listeners, socket, ip, serverport, clientport
|
|||
shutdown = id
|
||||
_socketlist[ socket ] = handler
|
||||
_readlistlen = addsocket(_readlist, socket, _readlistlen)
|
||||
|
||||
|
||||
-- remove traces of the old socket
|
||||
_readlistlen = removesocket( _readlist, oldsocket, _readlistlen )
|
||||
_sendlistlen = removesocket( _sendlist, oldsocket, _sendlistlen )
|
||||
|
@ -696,7 +700,7 @@ local function link(sender, receiver, buffersize)
|
|||
sender_locked = nil;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
local _readbuffer = sender.readbuffer;
|
||||
function sender.readbuffer()
|
||||
_readbuffer();
|
||||
|
@ -864,16 +868,16 @@ loop = function(once) -- this is the main loop of the program
|
|||
_starttime = _currenttime
|
||||
for handler, timestamp in pairs( _writetimes ) do
|
||||
if os_difftime( _currenttime - timestamp ) > _sendtimeout then
|
||||
--_writetimes[ handler ] = nil
|
||||
handler.disconnect( )( handler, "send timeout" )
|
||||
handler:force_close() -- forced disconnect
|
||||
end
|
||||
end
|
||||
for handler, timestamp in pairs( _readtimes ) do
|
||||
if os_difftime( _currenttime - timestamp ) > _readtimeout then
|
||||
--_readtimes[ handler ] = nil
|
||||
handler.disconnect( )( handler, "read timeout" )
|
||||
handler:close( ) -- forced disconnect?
|
||||
if not(handler.onreadtimeout) or handler:onreadtimeout() ~= true then
|
||||
handler.disconnect( )( handler, "read timeout" )
|
||||
handler:close( ) -- forced disconnect?
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
@ -934,9 +938,9 @@ local addclient = function( address, port, listeners, pattern, sslctx )
|
|||
client:settimeout( 0 )
|
||||
_, err = client:connect( address, port )
|
||||
if err then -- try again
|
||||
local handler = wrapclient( client, address, port, listeners )
|
||||
return wrapclient( client, address, port, listeners, pattern, sslctx )
|
||||
else
|
||||
wrapconnection( nil, listeners, client, address, port, "clientport", pattern, sslctx )
|
||||
return wrapconnection( nil, listeners, client, address, port, "clientport", pattern, sslctx )
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -966,7 +970,7 @@ return {
|
|||
|
||||
addclient = addclient,
|
||||
wrapclient = wrapclient,
|
||||
|
||||
|
||||
loop = loop,
|
||||
link = link,
|
||||
step = step,
|
||||
|
|
|
@ -6,79 +6,81 @@
|
|||
--
|
||||
|
||||
local st = require "util.stanza";
|
||||
local keys = require "util.iterators".keys;
|
||||
local array_collect = require "util.array".collect;
|
||||
local is_admin = require "core.usermanager".is_admin;
|
||||
local jid_split = require "util.jid".split;
|
||||
local adhoc_handle_cmd = module:require "adhoc".handle_cmd;
|
||||
local xmlns_cmd = "http://jabber.org/protocol/commands";
|
||||
local xmlns_disco = "http://jabber.org/protocol/disco";
|
||||
local commands = {};
|
||||
|
||||
module:add_feature(xmlns_cmd);
|
||||
|
||||
module:hook("iq/host/"..xmlns_disco.."#info:query", function (event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if stanza.attr.type == "get" and node then
|
||||
if commands[node] then
|
||||
local privileged = is_admin(stanza.attr.from, stanza.attr.to);
|
||||
if (commands[node].permission == "admin" and privileged)
|
||||
or (commands[node].permission == "user") then
|
||||
reply = st.reply(stanza);
|
||||
reply:tag("query", { xmlns = xmlns_disco.."#info",
|
||||
node = node });
|
||||
reply:tag("identity", { name = commands[node].name,
|
||||
category = "automation", type = "command-node" }):up();
|
||||
reply:tag("feature", { var = xmlns_cmd }):up();
|
||||
reply:tag("feature", { var = "jabber:x:data" }):up();
|
||||
else
|
||||
reply = st.error_reply(stanza, "auth", "forbidden", "This item is not available to you");
|
||||
end
|
||||
origin.send(reply);
|
||||
return true;
|
||||
elseif node == xmlns_cmd then
|
||||
reply = st.reply(stanza);
|
||||
reply:tag("query", { xmlns = xmlns_disco.."#info",
|
||||
node = node });
|
||||
reply:tag("identity", { name = "Ad-Hoc Commands",
|
||||
category = "automation", type = "command-list" }):up();
|
||||
origin.send(reply);
|
||||
return true;
|
||||
|
||||
module:hook("host-disco-info-node", function (event)
|
||||
local stanza, origin, reply, node = event.stanza, event.origin, event.reply, event.node;
|
||||
if commands[node] then
|
||||
local from = stanza.attr.from;
|
||||
local privileged = is_admin(from, stanza.attr.to);
|
||||
local global_admin = is_admin(from);
|
||||
local username, hostname = jid_split(from);
|
||||
local command = commands[node];
|
||||
if (command.permission == "admin" and privileged)
|
||||
or (command.permission == "global_admin" and global_admin)
|
||||
or (command.permission == "local_user" and hostname == module.host)
|
||||
or (command.permission == "user") then
|
||||
reply:tag("identity", { name = command.name,
|
||||
category = "automation", type = "command-node" }):up();
|
||||
reply:tag("feature", { var = xmlns_cmd }):up();
|
||||
reply:tag("feature", { var = "jabber:x:data" }):up();
|
||||
event.exists = true;
|
||||
else
|
||||
return origin.send(st.error_reply(stanza, "auth", "forbidden", "This item is not available to you"));
|
||||
end
|
||||
elseif node == xmlns_cmd then
|
||||
reply:tag("identity", { name = "Ad-Hoc Commands",
|
||||
category = "automation", type = "command-list" }):up();
|
||||
event.exists = true;
|
||||
end
|
||||
end);
|
||||
|
||||
module:hook("iq/host/"..xmlns_disco.."#items:query", function (event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type == "get" and stanza.tags[1].attr.node
|
||||
and stanza.tags[1].attr.node == xmlns_cmd then
|
||||
local admin = is_admin(stanza.attr.from, stanza.attr.to);
|
||||
local global_admin = is_admin(stanza.attr.from);
|
||||
reply = st.reply(stanza);
|
||||
reply:tag("query", { xmlns = xmlns_disco.."#items",
|
||||
node = xmlns_cmd });
|
||||
for node, command in pairs(commands) do
|
||||
if (command.permission == "admin" and admin)
|
||||
or (command.permission == "global_admin" and global_admin)
|
||||
or (command.permission == "user") then
|
||||
reply:tag("item", { name = command.name,
|
||||
node = node, jid = module:get_host() });
|
||||
reply:up();
|
||||
end
|
||||
end
|
||||
origin.send(reply);
|
||||
return true;
|
||||
module:hook("host-disco-items-node", function (event)
|
||||
local stanza, origin, reply, node = event.stanza, event.origin, event.reply, event.node;
|
||||
if node ~= xmlns_cmd then
|
||||
return;
|
||||
end
|
||||
end, 500);
|
||||
|
||||
local from = stanza.attr.from;
|
||||
local admin = is_admin(from, stanza.attr.to);
|
||||
local global_admin = is_admin(from);
|
||||
local username, hostname = jid_split(from);
|
||||
local nodes = array_collect(keys(commands)):sort();
|
||||
for _, node in ipairs(nodes) do
|
||||
local command = commands[node];
|
||||
if (command.permission == "admin" and admin)
|
||||
or (command.permission == "global_admin" and global_admin)
|
||||
or (command.permission == "local_user" and hostname == module.host)
|
||||
or (command.permission == "user") then
|
||||
reply:tag("item", { name = command.name,
|
||||
node = node, jid = module:get_host() });
|
||||
reply:up();
|
||||
end
|
||||
end
|
||||
event.exists = true;
|
||||
end);
|
||||
|
||||
module:hook("iq/host/"..xmlns_cmd..":command", function (event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type == "set" then
|
||||
local node = stanza.tags[1].attr.node
|
||||
if commands[node] then
|
||||
local admin = is_admin(stanza.attr.from, stanza.attr.to);
|
||||
local global_admin = is_admin(stanza.attr.from);
|
||||
if (commands[node].permission == "admin" and not admin)
|
||||
or (commands[node].permission == "global_admin" and not global_admin) then
|
||||
local command = commands[node];
|
||||
if command then
|
||||
local from = stanza.attr.from;
|
||||
local admin = is_admin(from, stanza.attr.to);
|
||||
local global_admin = is_admin(from);
|
||||
local username, hostname = jid_split(from);
|
||||
if (command.permission == "admin" and not admin)
|
||||
or (command.permission == "global_admin" and not global_admin)
|
||||
or (command.permission == "local_user" and hostname ~= module.host) then
|
||||
origin.send(st.error_reply(stanza, "auth", "forbidden", "You don't have permission to execute this command"):up()
|
||||
:add_child(commands[node]:cmdtag("canceled")
|
||||
:tag("note", {type="error"}):text("You don't have permission to execute this command")));
|
||||
|
|
|
@ -489,7 +489,7 @@ local globally_reload_module_handler = adhoc_initial(globally_reload_module_layo
|
|||
for _, host in pairs(hosts) do
|
||||
loaded_modules:append(array(keys(host.modules)));
|
||||
end
|
||||
loaded_modules = array(keys(set.new(loaded_modules):items())):sort();
|
||||
loaded_modules = array(set.new(loaded_modules):items()):sort();
|
||||
return { module = loaded_modules };
|
||||
end, function(fields, err)
|
||||
local is_global = false;
|
||||
|
@ -631,7 +631,7 @@ local globally_unload_module_handler = adhoc_initial(globally_unload_module_layo
|
|||
for _, host in pairs(hosts) do
|
||||
loaded_modules:append(array(keys(host.modules)));
|
||||
end
|
||||
loaded_modules = array(keys(set.new(loaded_modules):items())):sort();
|
||||
loaded_modules = array(set.new(loaded_modules):items()):sort();
|
||||
return { module = loaded_modules };
|
||||
end, function(fields, err)
|
||||
local is_global = false;
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -17,7 +17,6 @@ local _G = _G;
|
|||
|
||||
local prosody = _G.prosody;
|
||||
local hosts = prosody.hosts;
|
||||
local incoming_s2s = prosody.incoming_s2s;
|
||||
|
||||
local console_listener = { default_port = 5582; default_mode = "*a"; interface = "127.0.0.1" };
|
||||
|
||||
|
@ -60,20 +59,20 @@ function console:new_session(conn)
|
|||
disconnect = function () conn:close(); end;
|
||||
};
|
||||
session.env = setmetatable({}, default_env_mt);
|
||||
|
||||
|
||||
-- Load up environment with helper objects
|
||||
for name, t in pairs(def_env) do
|
||||
if type(t) == "table" then
|
||||
session.env[name] = setmetatable({ session = session }, { __index = t });
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
return session;
|
||||
end
|
||||
|
||||
function console:process_line(session, line)
|
||||
local useglobalenv;
|
||||
|
||||
|
||||
if line:match("^>") then
|
||||
line = line:gsub("^>", "");
|
||||
useglobalenv = true;
|
||||
|
@ -87,9 +86,9 @@ function console:process_line(session, line)
|
|||
return;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
session.env._ = line;
|
||||
|
||||
|
||||
local chunkname = "=console";
|
||||
local env = (useglobalenv and redirect_output(_G, session)) or session.env or nil
|
||||
local chunk, err = envload("return "..line, chunkname, env);
|
||||
|
@ -103,20 +102,20 @@ function console:process_line(session, line)
|
|||
return;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
local ranok, taskok, message = pcall(chunk);
|
||||
|
||||
|
||||
if not (ranok or message or useglobalenv) and commands[line:lower()] then
|
||||
commands[line:lower()](session, line);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
if not ranok then
|
||||
session.print("Fatal error while running command, it did not complete");
|
||||
session.print("Error: "..taskok);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
if not message then
|
||||
session.print("Result: "..tostring(taskok));
|
||||
return;
|
||||
|
@ -125,7 +124,7 @@ function console:process_line(session, line)
|
|||
session.print("Message: "..tostring(message));
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
session.print("OK: "..tostring(message));
|
||||
end
|
||||
|
||||
|
@ -344,9 +343,9 @@ end
|
|||
|
||||
function def_env.module:load(name, hosts, config)
|
||||
local mm = require "modulemanager";
|
||||
|
||||
|
||||
hosts = get_hosts_set(hosts);
|
||||
|
||||
|
||||
-- Load the module for each host
|
||||
local ok, err, count, mod = true, nil, 0, nil;
|
||||
for host in hosts do
|
||||
|
@ -367,15 +366,15 @@ function def_env.module:load(name, hosts, config)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
return ok, (ok and "Module loaded onto "..count.." host"..(count ~= 1 and "s" or "")) or ("Last error: "..tostring(err));
|
||||
|
||||
return ok, (ok and "Module loaded onto "..count.." host"..(count ~= 1 and "s" or "")) or ("Last error: "..tostring(err));
|
||||
end
|
||||
|
||||
function def_env.module:unload(name, hosts)
|
||||
local mm = require "modulemanager";
|
||||
|
||||
hosts = get_hosts_set(hosts, name);
|
||||
|
||||
|
||||
-- Unload the module for each host
|
||||
local ok, err, count = true, nil, 0;
|
||||
for host in hosts do
|
||||
|
@ -433,7 +432,7 @@ function def_env.module:list(hosts)
|
|||
if type(hosts) ~= "table" then
|
||||
return false, "Please supply a host or a list of hosts you would like to see";
|
||||
end
|
||||
|
||||
|
||||
local print = self.session.print;
|
||||
for _, host in ipairs(hosts) do
|
||||
print((host == "*" and "Global" or host)..":");
|
||||
|
@ -483,6 +482,25 @@ end
|
|||
function def_env.hosts:add(name)
|
||||
end
|
||||
|
||||
local function session_flags(session, line)
|
||||
line = line or {};
|
||||
if session.cert_identity_status == "valid" then
|
||||
line[#line+1] = "(secure)";
|
||||
elseif session.secure then
|
||||
line[#line+1] = "(encrypted)";
|
||||
end
|
||||
if session.compressed then
|
||||
line[#line+1] = "(compressed)";
|
||||
end
|
||||
if session.smacks then
|
||||
line[#line+1] = "(sm)";
|
||||
end
|
||||
if session.ip and session.ip:match(":") then
|
||||
line[#line+1] = "(IPv6)";
|
||||
end
|
||||
return table.concat(line, " ");
|
||||
end
|
||||
|
||||
def_env.c2s = {};
|
||||
|
||||
local function show_c2s(callback)
|
||||
|
@ -501,7 +519,7 @@ function def_env.c2s:count(match_jid)
|
|||
show_c2s(function (jid, session)
|
||||
if (not match_jid) or jid:match(match_jid) then
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
end);
|
||||
return true, "Total: "..count.." clients";
|
||||
end
|
||||
|
@ -518,15 +536,10 @@ function def_env.c2s:show(match_jid)
|
|||
count = count + 1;
|
||||
local status, priority = "unavailable", tostring(session.priority or "-");
|
||||
if session.presence then
|
||||
status = session.presence:child_with_name("show");
|
||||
if status then
|
||||
status = status:get_text() or "[invalid!]";
|
||||
else
|
||||
status = "available";
|
||||
end
|
||||
status = session.presence:get_child_text("show") or "available";
|
||||
end
|
||||
print(" "..jid.." - "..status.."("..priority..")");
|
||||
end
|
||||
print(session_flags(session, { " "..jid.." - "..status.."("..priority..")" }));
|
||||
end
|
||||
end);
|
||||
return true, "Total: "..count.." clients";
|
||||
end
|
||||
|
@ -537,7 +550,7 @@ function def_env.c2s:show_insecure(match_jid)
|
|||
if ((not match_jid) or jid:match(match_jid)) and not session.secure then
|
||||
count = count + 1;
|
||||
print(jid);
|
||||
end
|
||||
end
|
||||
end);
|
||||
return true, "Total: "..count.." insecure client connections";
|
||||
end
|
||||
|
@ -548,7 +561,7 @@ function def_env.c2s:show_secure(match_jid)
|
|||
if ((not match_jid) or jid:match(match_jid)) and session.secure then
|
||||
count = count + 1;
|
||||
print(jid);
|
||||
end
|
||||
end
|
||||
end);
|
||||
return true, "Total: "..count.." secure client connections";
|
||||
end
|
||||
|
@ -564,96 +577,80 @@ function def_env.c2s:close(match_jid)
|
|||
return true, "Total: "..count.." sessions closed";
|
||||
end
|
||||
|
||||
local function session_flags(session, line)
|
||||
if session.cert_identity_status == "valid" then
|
||||
line[#line+1] = "(secure)";
|
||||
elseif session.secure then
|
||||
line[#line+1] = "(encrypted)";
|
||||
end
|
||||
if session.compressed then
|
||||
line[#line+1] = "(compressed)";
|
||||
end
|
||||
if session.smacks then
|
||||
line[#line+1] = "(sm)";
|
||||
end
|
||||
if session.conn and session.conn:ip():match(":") then
|
||||
line[#line+1] = "(IPv6)";
|
||||
end
|
||||
return table.concat(line, " ");
|
||||
end
|
||||
|
||||
def_env.s2s = {};
|
||||
function def_env.s2s:show(match_jid)
|
||||
local _print = self.session.print;
|
||||
local print = self.session.print;
|
||||
|
||||
|
||||
local count_in, count_out = 0,0;
|
||||
|
||||
for host, host_session in pairs(hosts) do
|
||||
print = function (...) _print(host); _print(...); print = _print; end
|
||||
for remotehost, session in pairs(host_session.s2sout) do
|
||||
if (not match_jid) or remotehost:match(match_jid) or host:match(match_jid) then
|
||||
count_out = count_out + 1;
|
||||
print(session_flags(session, {" ", host, "->", remotehost}));
|
||||
if session.sendq then
|
||||
print(" There are "..#session.sendq.." queued outgoing stanzas for this connection");
|
||||
end
|
||||
if session.type == "s2sout_unauthed" then
|
||||
if session.connecting then
|
||||
print(" Connection not yet established");
|
||||
if not session.srv_hosts then
|
||||
if not session.conn then
|
||||
print(" We do not yet have a DNS answer for this host's SRV records");
|
||||
else
|
||||
print(" This host has no SRV records, using A record instead");
|
||||
end
|
||||
elseif session.srv_choice then
|
||||
print(" We are on SRV record "..session.srv_choice.." of "..#session.srv_hosts);
|
||||
local srv_choice = session.srv_hosts[session.srv_choice];
|
||||
print(" Using "..(srv_choice.target or ".")..":"..(srv_choice.port or 5269));
|
||||
local s2s_list = { };
|
||||
|
||||
local s2s_sessions = module:shared"/*/s2s/sessions";
|
||||
for _, session in pairs(s2s_sessions) do
|
||||
local remotehost, localhost, direction;
|
||||
if session.direction == "outgoing" then
|
||||
direction = "->";
|
||||
count_out = count_out + 1;
|
||||
remotehost, localhost = session.to_host or "?", session.from_host or "?";
|
||||
else
|
||||
direction = "<-";
|
||||
count_in = count_in + 1;
|
||||
remotehost, localhost = session.from_host or "?", session.to_host or "?";
|
||||
end
|
||||
local sess_lines = { l = localhost, r = remotehost,
|
||||
session_flags(session, { "", direction, remotehost or "?",
|
||||
"["..session.type..tostring(session):match("[a-f0-9]*$").."]" })};
|
||||
|
||||
if (not match_jid) or remotehost:match(match_jid) or localhost:match(match_jid) then
|
||||
table.insert(s2s_list, sess_lines);
|
||||
local print = function (s) table.insert(sess_lines, " "..s); end
|
||||
if session.sendq then
|
||||
print("There are "..#session.sendq.." queued outgoing stanzas for this connection");
|
||||
end
|
||||
if session.type == "s2sout_unauthed" then
|
||||
if session.connecting then
|
||||
print("Connection not yet established");
|
||||
if not session.srv_hosts then
|
||||
if not session.conn then
|
||||
print("We do not yet have a DNS answer for this host's SRV records");
|
||||
else
|
||||
print("This host has no SRV records, using A record instead");
|
||||
end
|
||||
elseif session.notopen then
|
||||
print(" The <stream> has not yet been opened");
|
||||
elseif not session.dialback_key then
|
||||
print(" Dialback has not been initiated yet");
|
||||
elseif session.dialback_key then
|
||||
print(" Dialback has been requested, but no result received");
|
||||
elseif session.srv_choice then
|
||||
print("We are on SRV record "..session.srv_choice.." of "..#session.srv_hosts);
|
||||
local srv_choice = session.srv_hosts[session.srv_choice];
|
||||
print("Using "..(srv_choice.target or ".")..":"..(srv_choice.port or 5269));
|
||||
end
|
||||
elseif session.notopen then
|
||||
print("The <stream> has not yet been opened");
|
||||
elseif not session.dialback_key then
|
||||
print("Dialback has not been initiated yet");
|
||||
elseif session.dialback_key then
|
||||
print("Dialback has been requested, but no result received");
|
||||
end
|
||||
end
|
||||
end
|
||||
local subhost_filter = function (h)
|
||||
return (match_jid and h:match(match_jid));
|
||||
end
|
||||
for session in pairs(incoming_s2s) do
|
||||
if session.to_host == host and ((not match_jid) or host:match(match_jid)
|
||||
or (session.from_host and session.from_host:match(match_jid))
|
||||
-- Pft! is what I say to list comprehensions
|
||||
or (session.hosts and #array.collect(keys(session.hosts)):filter(subhost_filter)>0)) then
|
||||
count_in = count_in + 1;
|
||||
print(session_flags(session, {" ", host, "<-", session.from_host or "(unknown)"}));
|
||||
if session.type == "s2sin_unauthed" then
|
||||
print(" Connection not yet authenticated");
|
||||
end
|
||||
if session.type == "s2sin_unauthed" then
|
||||
print("Connection not yet authenticated");
|
||||
elseif session.type == "s2sin" then
|
||||
for name in pairs(session.hosts) do
|
||||
if name ~= session.from_host then
|
||||
print(" also hosts "..tostring(name));
|
||||
print("also hosts "..tostring(name));
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
print = _print;
|
||||
end
|
||||
|
||||
for session in pairs(incoming_s2s) do
|
||||
if not session.to_host and ((not match_jid) or session.from_host and session.from_host:match(match_jid)) then
|
||||
count_in = count_in + 1;
|
||||
print("Other incoming s2s connections");
|
||||
print(" (unknown) <- "..(session.from_host or "(unknown)"));
|
||||
end
|
||||
|
||||
-- Sort by local host, then remote host
|
||||
table.sort(s2s_list, function(a,b)
|
||||
if a.l == b.l then return a.r < b.r; end
|
||||
return a.l < b.l;
|
||||
end);
|
||||
local lasthost;
|
||||
for _, sess_lines in ipairs(s2s_list) do
|
||||
if sess_lines.l ~= lasthost then print(sess_lines.l); lasthost=sess_lines.l end
|
||||
for _, line in ipairs(sess_lines) do print(line); end
|
||||
end
|
||||
|
||||
return true, "Total: "..count_out.." outgoing, "..count_in.." incoming connections";
|
||||
end
|
||||
|
||||
|
@ -685,14 +682,9 @@ end
|
|||
function def_env.s2s:showcert(domain)
|
||||
local ser = require "util.serialization".serialize;
|
||||
local print = self.session.print;
|
||||
local domain_sessions = set.new(array.collect(keys(incoming_s2s)))
|
||||
/function(session) return session.from_host == domain and session or nil; end;
|
||||
for local_host in values(prosody.hosts) do
|
||||
local s2sout = local_host.s2sout;
|
||||
if s2sout and s2sout[domain] then
|
||||
domain_sessions:add(s2sout[domain]);
|
||||
end
|
||||
end
|
||||
local s2s_sessions = module:shared"/*/s2s/sessions";
|
||||
local domain_sessions = set.new(array.collect(values(s2s_sessions)))
|
||||
/function(session) return (session.to_host == domain or session.from_host == domain) and session or nil; end;
|
||||
local cert_set = {};
|
||||
for session in domain_sessions do
|
||||
local conn = session.conn;
|
||||
|
@ -731,18 +723,18 @@ function def_env.s2s:showcert(domain)
|
|||
local domain_certs = array.collect(values(cert_set));
|
||||
-- Phew. We now have a array of unique certificates presented by domain.
|
||||
local n_certs = #domain_certs;
|
||||
|
||||
|
||||
if n_certs == 0 then
|
||||
return "No certificates found for "..domain;
|
||||
end
|
||||
|
||||
|
||||
local function _capitalize_and_colon(byte)
|
||||
return string.upper(byte)..":";
|
||||
end
|
||||
local function pretty_fingerprint(hash)
|
||||
return hash:gsub("..", _capitalize_and_colon):sub(1, -2);
|
||||
end
|
||||
|
||||
|
||||
for cert_info in values(domain_certs) do
|
||||
local certs = cert_info.certs;
|
||||
local cert = certs[1];
|
||||
|
@ -783,76 +775,38 @@ end
|
|||
|
||||
function def_env.s2s:close(from, to)
|
||||
local print, count = self.session.print, 0;
|
||||
|
||||
if not (from and to) then
|
||||
local s2s_sessions = module:shared"/*/s2s/sessions";
|
||||
|
||||
local match_id;
|
||||
if from and not to then
|
||||
match_id, from = from;
|
||||
elseif not to then
|
||||
return false, "Syntax: s2s:close('from', 'to') - Closes all s2s sessions from 'from' to 'to'";
|
||||
elseif from == to then
|
||||
return false, "Both from and to are the same... you can't do that :)";
|
||||
end
|
||||
|
||||
if hosts[from] and not hosts[to] then
|
||||
-- Is an outgoing connection
|
||||
local session = hosts[from].s2sout[to];
|
||||
if not session then
|
||||
print("No outgoing connection from "..from.." to "..to)
|
||||
else
|
||||
|
||||
for _, session in pairs(s2s_sessions) do
|
||||
local id = session.type..tostring(session):match("[a-f0-9]+$");
|
||||
if (match_id and match_id == id)
|
||||
or (session.from_host == from and session.to_host == to) then
|
||||
print(("Closing connection from %s to %s [%s]"):format(session.from_host, session.to_host, id));
|
||||
(session.close or s2smanager.destroy_session)(session);
|
||||
count = count + 1;
|
||||
print("Closed outgoing session from "..from.." to "..to);
|
||||
count = count + 1 ;
|
||||
end
|
||||
elseif hosts[to] and not hosts[from] then
|
||||
-- Is an incoming connection
|
||||
for session in pairs(incoming_s2s) do
|
||||
if session.to_host == to and session.from_host == from then
|
||||
(session.close or s2smanager.destroy_session)(session);
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
|
||||
if count == 0 then
|
||||
print("No incoming connections from "..from.." to "..to);
|
||||
else
|
||||
print("Closed "..count.." incoming session"..((count == 1 and "") or "s").." from "..from.." to "..to);
|
||||
end
|
||||
elseif hosts[to] and hosts[from] then
|
||||
return false, "Both of the hostnames you specified are local, there are no s2s sessions to close";
|
||||
else
|
||||
return false, "Neither of the hostnames you specified are being used on this server";
|
||||
end
|
||||
|
||||
return true, "Closed "..count.." s2s session"..((count == 1 and "") or "s");
|
||||
end
|
||||
|
||||
function def_env.s2s:closeall(host)
|
||||
local count = 0;
|
||||
|
||||
if not host or type(host) ~= "string" then return false, "wrong syntax: please use s2s:closeall('hostname.tld')"; end
|
||||
if hosts[host] then
|
||||
for session in pairs(incoming_s2s) do
|
||||
if session.to_host == host then
|
||||
(session.close or s2smanager.destroy_session)(session);
|
||||
local s2s_sessions = module:shared"/*/s2s/sessions";
|
||||
for _,session in pairs(s2s_sessions) do
|
||||
if not host or session.from_host == host or session.to_host == host then
|
||||
session:close();
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
for _, session in pairs(hosts[host].s2sout) do
|
||||
(session.close or s2smanager.destroy_session)(session);
|
||||
count = count + 1;
|
||||
end
|
||||
else
|
||||
for session in pairs(incoming_s2s) do
|
||||
if session.from_host == host then
|
||||
(session.close or s2smanager.destroy_session)(session);
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
for _, h in pairs(hosts) do
|
||||
if h.s2sout[host] then
|
||||
(h.s2sout[host].close or s2smanager.destroy_session)(h.s2sout[host]);
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
if count == 0 then return false, "No sessions to close.";
|
||||
else return true, "Closed "..count.." s2s session"..((count == 1 and "") or "s"); end
|
||||
end
|
||||
|
@ -1076,12 +1030,12 @@ function printbanner(session)
|
|||
local option = module:get_option("console_banner");
|
||||
if option == nil or option == "full" or option == "graphic" then
|
||||
session.print [[
|
||||
____ \ / _
|
||||
| _ \ _ __ ___ ___ _-_ __| |_ _
|
||||
____ \ / _
|
||||
| _ \ _ __ ___ ___ _-_ __| |_ _
|
||||
| |_) | '__/ _ \/ __|/ _ \ / _` | | | |
|
||||
| __/| | | (_) \__ \ |_| | (_| | |_| |
|
||||
|_| |_| \___/|___/\___/ \__,_|\__, |
|
||||
A study in simplicity |___/
|
||||
A study in simplicity |___/
|
||||
|
||||
]]
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -39,22 +39,22 @@ end
|
|||
function handle_announcement(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
local node, host, resource = jid.split(stanza.attr.to);
|
||||
|
||||
|
||||
if resource ~= "announce/online" then
|
||||
return; -- Not an announcement
|
||||
end
|
||||
|
||||
|
||||
if not is_admin(stanza.attr.from) then
|
||||
-- Not an admin? Not allowed!
|
||||
module:log("warn", "Non-admin '%s' tried to send server announcement", stanza.attr.from);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
module:log("info", "Sending server announcement to all online users");
|
||||
local message = st.clone(stanza);
|
||||
message.attr.type = "headline";
|
||||
message.attr.from = host;
|
||||
|
||||
|
||||
local c = send_to_online(message, host);
|
||||
module:log("info", "Announcement sent to %d online users", c);
|
||||
return true;
|
||||
|
@ -83,9 +83,9 @@ function announce_handler(self, data, state)
|
|||
module:log("info", "Sending server announcement to all online users");
|
||||
local message = st.message({type = "headline"}, fields.announcement):up()
|
||||
:tag("subject"):text(fields.subject or "Announcement");
|
||||
|
||||
|
||||
local count = send_to_online(message, data.to);
|
||||
|
||||
|
||||
module:log("info", "Announcement sent to %d online users", count);
|
||||
return { status = "completed", info = ("Announcement sent to %d online users"):format(count) };
|
||||
else
|
||||
|
|
|
@ -7,12 +7,16 @@
|
|||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
||||
local log = require "util.logger".init("auth_internal_hashed");
|
||||
local max = math.max;
|
||||
|
||||
local getAuthenticationDatabaseSHA1 = require "util.sasl.scram".getAuthenticationDatabaseSHA1;
|
||||
local usermanager = require "core.usermanager";
|
||||
local generate_uuid = require "util.uuid".generate;
|
||||
local new_sasl = require "util.sasl".new;
|
||||
|
||||
local log = module._log;
|
||||
local host = module.host;
|
||||
|
||||
local accounts = module:open_store("accounts");
|
||||
|
||||
local to_hex;
|
||||
|
@ -37,14 +41,13 @@ end
|
|||
|
||||
|
||||
-- Default; can be set per-user
|
||||
local iteration_count = 4096;
|
||||
local default_iteration_count = 4096;
|
||||
|
||||
local host = module.host;
|
||||
-- define auth provider
|
||||
local provider = {};
|
||||
log("debug", "initializing internal_hashed authentication provider for host '%s'", host);
|
||||
|
||||
function provider.test_password(username, password)
|
||||
log("debug", "test password for user '%s'", username);
|
||||
local credentials = accounts:get(username) or {};
|
||||
|
||||
if credentials.password ~= nil and string.len(credentials.password) ~= 0 then
|
||||
|
@ -62,12 +65,12 @@ function provider.test_password(username, password)
|
|||
if credentials.iteration_count == nil or credentials.salt == nil or string.len(credentials.salt) == 0 then
|
||||
return nil, "Auth failed. Stored salt and iteration count information is not complete.";
|
||||
end
|
||||
|
||||
|
||||
local valid, stored_key, server_key = getAuthenticationDatabaseSHA1(password, credentials.salt, credentials.iteration_count);
|
||||
|
||||
|
||||
local stored_key_hex = to_hex(stored_key);
|
||||
local server_key_hex = to_hex(server_key);
|
||||
|
||||
|
||||
if valid and stored_key_hex == credentials.stored_key and server_key_hex == credentials.server_key then
|
||||
return true;
|
||||
else
|
||||
|
@ -76,14 +79,15 @@ function provider.test_password(username, password)
|
|||
end
|
||||
|
||||
function provider.set_password(username, password)
|
||||
log("debug", "set_password for username '%s'", username);
|
||||
local account = accounts:get(username);
|
||||
if account then
|
||||
account.salt = account.salt or generate_uuid();
|
||||
account.iteration_count = account.iteration_count or iteration_count;
|
||||
account.salt = generate_uuid();
|
||||
account.iteration_count = max(account.iteration_count or 0, default_iteration_count);
|
||||
local valid, stored_key, server_key = getAuthenticationDatabaseSHA1(password, account.salt, account.iteration_count);
|
||||
local stored_key_hex = to_hex(stored_key);
|
||||
local server_key_hex = to_hex(server_key);
|
||||
|
||||
|
||||
account.stored_key = stored_key_hex
|
||||
account.server_key = server_key_hex
|
||||
|
||||
|
@ -96,7 +100,7 @@ end
|
|||
function provider.user_exists(username)
|
||||
local account = accounts:get(username);
|
||||
if not account then
|
||||
log("debug", "account not found for username '%s' at host '%s'", username, host);
|
||||
log("debug", "account not found for username '%s'", username);
|
||||
return nil, "Auth failed. Invalid username";
|
||||
end
|
||||
return true;
|
||||
|
@ -111,10 +115,10 @@ function provider.create_user(username, password)
|
|||
return accounts:set(username, {});
|
||||
end
|
||||
local salt = generate_uuid();
|
||||
local valid, stored_key, server_key = getAuthenticationDatabaseSHA1(password, salt, iteration_count);
|
||||
local valid, stored_key, server_key = getAuthenticationDatabaseSHA1(password, salt, default_iteration_count);
|
||||
local stored_key_hex = to_hex(stored_key);
|
||||
local server_key_hex = to_hex(server_key);
|
||||
return accounts:set(username, {stored_key = stored_key_hex, server_key = server_key_hex, salt = salt, iteration_count = iteration_count});
|
||||
return accounts:set(username, {stored_key = stored_key_hex, server_key = server_key_hex, salt = salt, iteration_count = default_iteration_count});
|
||||
end
|
||||
|
||||
function provider.delete_user(username)
|
||||
|
@ -134,7 +138,7 @@ function provider.get_sasl_handler()
|
|||
credentials = accounts:get(username);
|
||||
if not credentials then return; end
|
||||
end
|
||||
|
||||
|
||||
local stored_key, server_key, iteration_count, salt = credentials.stored_key, credentials.server_key, credentials.iteration_count, credentials.salt;
|
||||
stored_key = stored_key and from_hex(stored_key);
|
||||
server_key = server_key and from_hex(server_key);
|
||||
|
@ -143,6 +147,6 @@ function provider.get_sasl_handler()
|
|||
};
|
||||
return new_sasl(host, testpass_authentication_profile);
|
||||
end
|
||||
|
||||
|
||||
module:provides("auth", provider);
|
||||
|
||||
|
|
|
@ -16,10 +16,9 @@ local accounts = module:open_store("accounts");
|
|||
|
||||
-- define auth provider
|
||||
local provider = {};
|
||||
log("debug", "initializing internal_plain authentication provider for host '%s'", host);
|
||||
|
||||
function provider.test_password(username, password)
|
||||
log("debug", "test password for user %s at host %s", username, host);
|
||||
log("debug", "test password for user '%s'", username);
|
||||
local credentials = accounts:get(username) or {};
|
||||
|
||||
if password == credentials.password then
|
||||
|
@ -30,11 +29,12 @@ function provider.test_password(username, password)
|
|||
end
|
||||
|
||||
function provider.get_password(username)
|
||||
log("debug", "get_password for username '%s' at host '%s'", username, host);
|
||||
log("debug", "get_password for username '%s'", username);
|
||||
return (accounts:get(username) or {}).password;
|
||||
end
|
||||
|
||||
function provider.set_password(username, password)
|
||||
log("debug", "set_password for username '%s'", username);
|
||||
local account = accounts:get(username);
|
||||
if account then
|
||||
account.password = password;
|
||||
|
@ -46,7 +46,7 @@ end
|
|||
function provider.user_exists(username)
|
||||
local account = accounts:get(username);
|
||||
if not account then
|
||||
log("debug", "account not found for username '%s' at host '%s'", username, host);
|
||||
log("debug", "account not found for username '%s'", username);
|
||||
return nil, "Auth failed. Invalid username";
|
||||
end
|
||||
return true;
|
||||
|
@ -76,6 +76,6 @@ function provider.get_sasl_handler()
|
|||
};
|
||||
return new_sasl(host, getpass_authentication_profile);
|
||||
end
|
||||
|
||||
|
||||
module:provides("auth", provider);
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -37,24 +37,10 @@ local BOSH_DEFAULT_REQUESTS = module:get_option_number("bosh_max_requests", 2);
|
|||
local bosh_max_wait = module:get_option_number("bosh_max_wait", 120);
|
||||
|
||||
local consider_bosh_secure = module:get_option_boolean("consider_bosh_secure");
|
||||
|
||||
local default_headers = { ["Content-Type"] = "text/xml; charset=utf-8" };
|
||||
|
||||
local cross_domain = module:get_option("cross_domain_bosh", false);
|
||||
if cross_domain then
|
||||
default_headers["Access-Control-Allow-Methods"] = "GET, POST, OPTIONS";
|
||||
default_headers["Access-Control-Allow-Headers"] = "Content-Type";
|
||||
default_headers["Access-Control-Max-Age"] = "7200";
|
||||
|
||||
if cross_domain == true then
|
||||
default_headers["Access-Control-Allow-Origin"] = "*";
|
||||
elseif type(cross_domain) == "table" then
|
||||
cross_domain = table.concat(cross_domain, ", ");
|
||||
end
|
||||
if type(cross_domain) == "string" then
|
||||
default_headers["Access-Control-Allow-Origin"] = cross_domain;
|
||||
end
|
||||
end
|
||||
if cross_domain == true then cross_domain = "*"; end
|
||||
if type(cross_domain) == "table" then cross_domain = table.concat(cross_domain, ", "); end
|
||||
|
||||
local trusted_proxies = module:get_option_set("trusted_proxies", {"127.0.0.1"})._items;
|
||||
|
||||
|
@ -79,7 +65,7 @@ local os_time = os.time;
|
|||
local sessions, inactive_sessions = module:shared("sessions", "inactive_sessions");
|
||||
|
||||
-- Used to respond to idle sessions (those with waiting requests)
|
||||
local waiting_requests = {};
|
||||
local waiting_requests = module:shared("waiting_requests");
|
||||
function on_destroy_request(request)
|
||||
log("debug", "Request destroyed: %s", tostring(request));
|
||||
waiting_requests[request] = nil;
|
||||
|
@ -92,7 +78,7 @@ function on_destroy_request(request)
|
|||
break;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
-- If this session now has no requests open, mark it as inactive
|
||||
local max_inactive = session.bosh_max_inactive;
|
||||
if max_inactive and #requests == 0 then
|
||||
|
@ -102,11 +88,20 @@ function on_destroy_request(request)
|
|||
end
|
||||
end
|
||||
|
||||
function handle_OPTIONS(request)
|
||||
local headers = {};
|
||||
for k,v in pairs(default_headers) do headers[k] = v; end
|
||||
headers["Content-Type"] = nil;
|
||||
return { headers = headers, body = "" };
|
||||
local function set_cross_domain_headers(response)
|
||||
local headers = response.headers;
|
||||
headers.access_control_allow_methods = "GET, POST, OPTIONS";
|
||||
headers.access_control_allow_headers = "Content-Type";
|
||||
headers.access_control_max_age = "7200";
|
||||
headers.access_control_allow_origin = cross_domain;
|
||||
return response;
|
||||
end
|
||||
|
||||
function handle_OPTIONS(event)
|
||||
if cross_domain and event.request.headers.origin then
|
||||
set_cross_domain_headers(event.response);
|
||||
end
|
||||
return "";
|
||||
end
|
||||
|
||||
function handle_POST(event)
|
||||
|
@ -119,14 +114,24 @@ function handle_POST(event)
|
|||
local context = { request = request, response = response, notopen = true };
|
||||
local stream = new_xmpp_stream(context, stream_callbacks);
|
||||
response.context = context;
|
||||
|
||||
|
||||
local headers = response.headers;
|
||||
headers.content_type = "text/xml; charset=utf-8";
|
||||
|
||||
if cross_domain and event.request.headers.origin then
|
||||
set_cross_domain_headers(response);
|
||||
end
|
||||
|
||||
-- stream:feed() calls the stream_callbacks, so all stanzas in
|
||||
-- the body are processed in this next line before it returns.
|
||||
-- In particular, the streamopened() stream callback is where
|
||||
-- much of the session logic happens, because it's where we first
|
||||
-- get to see the 'sid' of this request.
|
||||
stream:feed(body);
|
||||
|
||||
if not stream:feed(body) then
|
||||
module:log("warn", "Error parsing BOSH payload")
|
||||
return 400;
|
||||
end
|
||||
|
||||
-- Stanzas (if any) in the request have now been processed, and
|
||||
-- we take care of the high-level BOSH logic here, including
|
||||
-- giving a response or putting the request "on hold".
|
||||
|
@ -141,9 +146,6 @@ function handle_POST(event)
|
|||
local r = session.requests;
|
||||
log("debug", "Session %s has %d out of %d requests open", context.sid, #r, session.bosh_hold);
|
||||
log("debug", "and there are %d things in the send_buffer:", #session.send_buffer);
|
||||
for i, thing in ipairs(session.send_buffer) do
|
||||
log("debug", " %s", tostring(thing));
|
||||
end
|
||||
if #r > session.bosh_hold then
|
||||
-- We are holding too many requests, send what's in the buffer,
|
||||
log("debug", "We are holding too many requests, so...");
|
||||
|
@ -162,7 +164,7 @@ function handle_POST(event)
|
|||
session.send_buffer = {};
|
||||
session.send(resp);
|
||||
end
|
||||
|
||||
|
||||
if not response.finished then
|
||||
-- We're keeping this request open, to respond later
|
||||
log("debug", "Have nothing to say, so leaving request unanswered for now");
|
||||
|
@ -170,7 +172,7 @@ function handle_POST(event)
|
|||
waiting_requests[response] = os_time() + session.bosh_wait;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if session.bosh_terminate then
|
||||
session.log("debug", "Closing session with %d requests open", #session.requests);
|
||||
session:close();
|
||||
|
@ -179,6 +181,8 @@ function handle_POST(event)
|
|||
return true; -- Inform http server we shall reply later
|
||||
end
|
||||
end
|
||||
module:log("warn", "Unable to associate request with a session (incomplete request?)");
|
||||
return 400;
|
||||
end
|
||||
|
||||
|
||||
|
@ -188,10 +192,10 @@ local stream_xmlns_attr = { xmlns = "urn:ietf:params:xml:ns:xmpp-streams" };
|
|||
|
||||
local function bosh_close_stream(session, reason)
|
||||
(session.log or log)("info", "BOSH client disconnected");
|
||||
|
||||
|
||||
local close_reply = st.stanza("body", { xmlns = xmlns_bosh, type = "terminate",
|
||||
["xmlns:stream"] = xmlns_streams });
|
||||
|
||||
|
||||
|
||||
if reason then
|
||||
close_reply.attr.condition = "remote-stream-error";
|
||||
|
@ -217,10 +221,9 @@ local function bosh_close_stream(session, reason)
|
|||
|
||||
local response_body = tostring(close_reply);
|
||||
for _, held_request in ipairs(session.requests) do
|
||||
held_request.headers = default_headers;
|
||||
held_request:send(response_body);
|
||||
end
|
||||
sessions[session.sid] = nil;
|
||||
sessions[session.sid] = nil;
|
||||
inactive_sessions[session] = nil;
|
||||
sm_destroy_session(session);
|
||||
end
|
||||
|
@ -233,7 +236,7 @@ function stream_callbacks.streamopened(context, attr)
|
|||
if not sid then
|
||||
-- New session request
|
||||
context.notopen = nil; -- Signals that we accept this opening tag
|
||||
|
||||
|
||||
-- TODO: Sanity checks here (rid, to, known host, etc.)
|
||||
if not hosts[attr.to] then
|
||||
-- Unknown host
|
||||
|
@ -243,7 +246,7 @@ function stream_callbacks.streamopened(context, attr)
|
|||
response:send(tostring(close_reply));
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
-- New session
|
||||
sid = new_uuid();
|
||||
local session = {
|
||||
|
@ -256,9 +259,9 @@ function stream_callbacks.streamopened(context, attr)
|
|||
ip = get_ip_from_request(request);
|
||||
};
|
||||
sessions[sid] = session;
|
||||
|
||||
|
||||
local filter = initialize_filters(session);
|
||||
|
||||
|
||||
session.log("debug", "BOSH session created for request from %s", session.ip);
|
||||
log("info", "New BOSH session, assigned it sid '%s'", sid);
|
||||
|
||||
|
@ -279,7 +282,6 @@ function stream_callbacks.streamopened(context, attr)
|
|||
local oldest_request = r[1];
|
||||
if oldest_request and not session.bosh_processing then
|
||||
log("debug", "We have an open request, so sending on that");
|
||||
oldest_request.headers = default_headers;
|
||||
local body_attr = { xmlns = "http://jabber.org/protocol/httpbind",
|
||||
["xmlns:stream"] = "http://etherx.jabber.org/streams";
|
||||
type = session.bosh_terminate and "terminate" or nil;
|
||||
|
@ -306,17 +308,16 @@ function stream_callbacks.streamopened(context, attr)
|
|||
end
|
||||
request.sid = sid;
|
||||
end
|
||||
|
||||
|
||||
local session = sessions[sid];
|
||||
if not session then
|
||||
-- Unknown sid
|
||||
log("info", "Client tried to use sid '%s' which we don't know about", sid);
|
||||
response.headers = default_headers;
|
||||
response:send(tostring(st.stanza("body", { xmlns = xmlns_bosh, type = "terminate", condition = "item-not-found" })));
|
||||
context.notopen = nil;
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
if session.rid then
|
||||
local rid = tonumber(attr.rid);
|
||||
local diff = rid - session.rid;
|
||||
|
@ -333,7 +334,7 @@ function stream_callbacks.streamopened(context, attr)
|
|||
end
|
||||
session.rid = rid;
|
||||
end
|
||||
|
||||
|
||||
if attr.type == "terminate" then
|
||||
-- Client wants to end this session, which we'll do
|
||||
-- after processing any stanzas in this request
|
||||
|
@ -348,8 +349,7 @@ function stream_callbacks.streamopened(context, attr)
|
|||
if session.notopen then
|
||||
local features = st.stanza("stream:features");
|
||||
hosts[session.host].events.fire_event("stream-features", { origin = session, features = features });
|
||||
fire_event("stream-features", session, features);
|
||||
session.send(tostring(features));
|
||||
session.send(features);
|
||||
session.notopen = nil;
|
||||
end
|
||||
end
|
||||
|
@ -370,8 +370,8 @@ function stream_callbacks.handlestanza(context, stanza)
|
|||
end
|
||||
end
|
||||
|
||||
function stream_callbacks.streamclosed(request)
|
||||
local session = sessions[request.sid];
|
||||
function stream_callbacks.streamclosed(context)
|
||||
local session = sessions[context.sid];
|
||||
if session then
|
||||
session.bosh_processing = false;
|
||||
if #session.send_buffer > 0 then
|
||||
|
@ -384,12 +384,11 @@ function stream_callbacks.error(context, error)
|
|||
log("debug", "Error parsing BOSH request payload; %s", error);
|
||||
if not context.sid then
|
||||
local response = context.response;
|
||||
response.headers = default_headers;
|
||||
response.status_code = 400;
|
||||
response:send();
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
local session = sessions[context.sid];
|
||||
if error == "stream-error" then -- Remote stream error, we close normally
|
||||
session:close();
|
||||
|
@ -398,7 +397,7 @@ function stream_callbacks.error(context, error)
|
|||
end
|
||||
end
|
||||
|
||||
local dead_sessions = {};
|
||||
local dead_sessions = module:shared("dead_sessions");
|
||||
function on_timer()
|
||||
-- log("debug", "Checking for requests soon to timeout...");
|
||||
-- Identify requests timing out within the next few seconds
|
||||
|
@ -413,7 +412,7 @@ function on_timer()
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
now = now - 3;
|
||||
local n_dead_sessions = 0;
|
||||
for session, close_after in pairs(inactive_sessions) do
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -15,9 +15,10 @@ local sessionmanager = require "core.sessionmanager";
|
|||
local st = require "util.stanza";
|
||||
local sm_new_session, sm_destroy_session = sessionmanager.new_session, sessionmanager.destroy_session;
|
||||
local uuid_generate = require "util.uuid".generate;
|
||||
local runner = require "util.async".runner;
|
||||
|
||||
local xpcall, tostring, type = xpcall, tostring, type;
|
||||
local traceback = debug.traceback;
|
||||
local t_insert, t_remove = table.insert, table.remove;
|
||||
|
||||
local xmlns_xmpp_streams = "urn:ietf:params:xml:ns:xmpp-streams";
|
||||
|
||||
|
@ -31,12 +32,12 @@ local sessions = module:shared("sessions");
|
|||
local core_process_stanza = prosody.core_process_stanza;
|
||||
local hosts = prosody.hosts;
|
||||
|
||||
local stream_callbacks = { default_ns = "jabber:client", handlestanza = core_process_stanza };
|
||||
local stream_callbacks = { default_ns = "jabber:client" };
|
||||
local listener = {};
|
||||
local runner_callbacks = {};
|
||||
|
||||
--- Stream events handlers
|
||||
local stream_xmlns_attr = {xmlns='urn:ietf:params:xml:ns:xmpp-streams'};
|
||||
local default_stream_attr = { ["xmlns:stream"] = "http://etherx.jabber.org/streams", xmlns = stream_callbacks.default_ns, version = "1.0", id = "" };
|
||||
|
||||
function stream_callbacks.streamopened(session, attr)
|
||||
local send = session.send;
|
||||
|
@ -50,15 +51,13 @@ function stream_callbacks.streamopened(session, attr)
|
|||
session.streamid = uuid_generate();
|
||||
(session.log or session)("debug", "Client sent opening <stream:stream> to %s", session.host);
|
||||
|
||||
if not hosts[session.host] then
|
||||
if not hosts[session.host] or not hosts[session.host].modules.c2s then
|
||||
-- We don't serve this host...
|
||||
session:close{ condition = "host-unknown", text = "This server does not serve "..tostring(session.host)};
|
||||
return;
|
||||
end
|
||||
|
||||
send("<?xml version='1.0'?>"..st.stanza("stream:stream", {
|
||||
xmlns = 'jabber:client', ["xmlns:stream"] = 'http://etherx.jabber.org/streams';
|
||||
id = session.streamid, from = session.host, version = '1.0', ["xml:lang"] = 'en' }):top_tag());
|
||||
session:open_stream();
|
||||
|
||||
(session.log or log)("debug", "Sent reply <stream:stream> to client");
|
||||
session.notopen = nil;
|
||||
|
@ -67,20 +66,21 @@ function stream_callbacks.streamopened(session, attr)
|
|||
-- since we now have a new stream header, session is secured
|
||||
if session.secure == false then
|
||||
session.secure = true;
|
||||
session.encrypted = true;
|
||||
|
||||
-- Check if TLS compression is used
|
||||
local sock = session.conn:socket();
|
||||
if sock.info then
|
||||
session.compressed = sock:info"compression";
|
||||
elseif sock.compression then
|
||||
session.compressed = sock:compression(); --COMPAT mw/luasec-hg
|
||||
local info = sock:info();
|
||||
(session.log or log)("info", "Stream encrypted (%s with %s)", info.protocol, info.cipher);
|
||||
session.compressed = info.compression;
|
||||
else
|
||||
(session.log or log)("info", "Stream encrypted");
|
||||
session.compressed = sock.compression and sock:compression(); --COMPAT mw/luasec-hg
|
||||
end
|
||||
end
|
||||
|
||||
local features = st.stanza("stream:features");
|
||||
hosts[session.host].events.fire_event("stream-features", { origin = session, features = features });
|
||||
module:fire_event("stream-features", session, features);
|
||||
|
||||
send(features);
|
||||
end
|
||||
|
||||
|
@ -116,12 +116,9 @@ function stream_callbacks.error(session, error, data)
|
|||
end
|
||||
end
|
||||
|
||||
local function handleerr(err) log("error", "Traceback[c2s]: %s", traceback(tostring(err), 2)); end
|
||||
function stream_callbacks.handlestanza(session, stanza)
|
||||
stanza = session.filter("stanzas/in", stanza);
|
||||
if stanza then
|
||||
return xpcall(function () return core_process_stanza(session, stanza) end, handleerr);
|
||||
end
|
||||
session.thread:run(stanza);
|
||||
end
|
||||
|
||||
--- Session methods
|
||||
|
@ -129,8 +126,7 @@ local function session_close(session, reason)
|
|||
local log = session.log or log;
|
||||
if session.conn then
|
||||
if session.notopen then
|
||||
session.send("<?xml version='1.0'?>");
|
||||
session.send(st.stanza("stream:stream", default_stream_attr):top_tag());
|
||||
session:open_stream();
|
||||
end
|
||||
if reason then -- nil == no err, initiated by us, false == initiated by client
|
||||
local stream_error = st.stanza("stream:error");
|
||||
|
@ -153,12 +149,12 @@ local function session_close(session, reason)
|
|||
log("debug", "Disconnecting client, <stream:error> is: %s", stream_error);
|
||||
session.send(stream_error);
|
||||
end
|
||||
|
||||
|
||||
session.send("</stream:stream>");
|
||||
function session.send() return false; end
|
||||
|
||||
|
||||
local reason = (reason and (reason.name or reason.text or reason.condition)) or reason;
|
||||
session.log("info", "c2s stream for %s closed: %s", session.full_jid or ("<"..session.ip..">"), reason or "session closed");
|
||||
session.log("debug", "c2s stream for %s closed: %s", session.full_jid or ("<"..session.ip..">"), reason or "session closed");
|
||||
|
||||
-- Authenticated incoming stream may still be sending us stanzas, so wait for </stream:stream> from remote
|
||||
local conn = session.conn;
|
||||
|
@ -178,6 +174,19 @@ local function session_close(session, reason)
|
|||
end
|
||||
end
|
||||
|
||||
local function session_open_stream(session)
|
||||
local attr = {
|
||||
["xmlns:stream"] = 'http://etherx.jabber.org/streams',
|
||||
xmlns = stream_callbacks.default_ns,
|
||||
version = "1.0",
|
||||
["xml:lang"] = 'en',
|
||||
id = session.streamid or "",
|
||||
from = session.host
|
||||
};
|
||||
session.send("<?xml version='1.0'?>");
|
||||
session.send(st.stanza("stream:stream", attr):top_tag());
|
||||
end
|
||||
|
||||
module:hook_global("user-deleted", function(event)
|
||||
local username, host = event.username, event.host;
|
||||
local user = hosts[host].sessions[username];
|
||||
|
@ -188,16 +197,29 @@ module:hook_global("user-deleted", function(event)
|
|||
end
|
||||
end, 200);
|
||||
|
||||
function runner_callbacks:ready()
|
||||
self.data.conn:resume();
|
||||
end
|
||||
|
||||
function runner_callbacks:waiting()
|
||||
self.data.conn:pause();
|
||||
end
|
||||
|
||||
function runner_callbacks:error(err)
|
||||
(self.data.log or log)("error", "Traceback[c2s]: %s", err);
|
||||
end
|
||||
|
||||
--- Port listener
|
||||
function listener.onconnect(conn)
|
||||
local session = sm_new_session(conn);
|
||||
sessions[conn] = session;
|
||||
|
||||
|
||||
session.log("info", "Client connected");
|
||||
|
||||
|
||||
-- Client is using legacy SSL (otherwise mod_tls sets this flag)
|
||||
if conn:ssl() then
|
||||
session.secure = true;
|
||||
session.encrypted = true;
|
||||
|
||||
-- Check if TLS compression is used
|
||||
local sock = conn:socket();
|
||||
|
@ -207,34 +229,42 @@ function listener.onconnect(conn)
|
|||
session.compressed = sock:compression(); --COMPAT mw/luasec-hg
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if opt_keepalives then
|
||||
conn:setoption("keepalive", opt_keepalives);
|
||||
end
|
||||
|
||||
|
||||
session.open_stream = session_open_stream;
|
||||
session.close = session_close;
|
||||
|
||||
|
||||
local stream = new_xmpp_stream(session, stream_callbacks);
|
||||
session.stream = stream;
|
||||
session.notopen = true;
|
||||
|
||||
|
||||
function session.reset_stream()
|
||||
session.notopen = true;
|
||||
session.stream:reset();
|
||||
end
|
||||
|
||||
|
||||
session.thread = runner(function (stanza)
|
||||
core_process_stanza(session, stanza);
|
||||
end, runner_callbacks, session);
|
||||
|
||||
local filter = session.filter;
|
||||
function session.data(data)
|
||||
data = filter("bytes/in", data);
|
||||
-- Parse the data, which will store stanzas in session.pending_stanzas
|
||||
if data then
|
||||
local ok, err = stream:feed(data);
|
||||
if ok then return; end
|
||||
log("debug", "Received invalid XML (%s) %d bytes: %s", tostring(err), #data, data:sub(1, 300):gsub("[\r\n]+", " "):gsub("[%z\1-\31]", "_"));
|
||||
session:close("not-well-formed");
|
||||
data = filter("bytes/in", data);
|
||||
if data then
|
||||
local ok, err = stream:feed(data);
|
||||
if not ok then
|
||||
log("debug", "Received invalid XML (%s) %d bytes: %s", tostring(err), #data, data:sub(1, 300):gsub("[\r\n]+", " "):gsub("[%z\1-\31]", "_"));
|
||||
session:close("not-well-formed");
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if c2s_timeout then
|
||||
add_task(c2s_timeout, function ()
|
||||
if session.type == "c2s_unauthed" then
|
||||
|
@ -262,10 +292,27 @@ function listener.ondisconnect(conn, err)
|
|||
end
|
||||
end
|
||||
|
||||
function listener.onreadtimeout(conn)
|
||||
local session = sessions[conn];
|
||||
if session then
|
||||
return (hosts[session.host] or prosody).events.fire_event("c2s-read-timeout", { session = session });
|
||||
end
|
||||
end
|
||||
|
||||
local function keepalive(event)
|
||||
return event.session.send(' ');
|
||||
end
|
||||
|
||||
function listener.associate_session(conn, session)
|
||||
sessions[conn] = session;
|
||||
end
|
||||
|
||||
function module.add_host(module)
|
||||
module:hook("c2s-read-timeout", keepalive, -1);
|
||||
end
|
||||
|
||||
module:hook("c2s-read-timeout", keepalive, -1);
|
||||
|
||||
module:hook("server-stopping", function(event)
|
||||
local reason = event.reason;
|
||||
for _, session in pairs(sessions) do
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -33,7 +33,7 @@ function module.add_host(module)
|
|||
if module:get_host_type() ~= "component" then
|
||||
error("Don't load mod_component manually, it should be for a component, please see http://prosody.im/doc/components", 0);
|
||||
end
|
||||
|
||||
|
||||
local env = module.environment;
|
||||
env.connected = false;
|
||||
|
||||
|
@ -44,26 +44,26 @@ function module.add_host(module)
|
|||
send = nil;
|
||||
session.on_destroy = nil;
|
||||
end
|
||||
|
||||
|
||||
-- Handle authentication attempts by component
|
||||
local function handle_component_auth(event)
|
||||
local session, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
if session.type ~= "component_unauthed" then return; end
|
||||
|
||||
|
||||
if (not session.host) or #stanza.tags > 0 then
|
||||
(session.log or log)("warn", "Invalid component handshake for host: %s", session.host);
|
||||
session:close("not-authorized");
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
local secret = module:get_option("component_secret");
|
||||
if not secret then
|
||||
(session.log or log)("warn", "Component attempted to identify as %s, but component_secret is not set", session.host);
|
||||
session:close("not-authorized");
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
local supplied_token = t_concat(stanza);
|
||||
local calculated_token = sha1(session.streamid..secret, true);
|
||||
if supplied_token:lower() ~= calculated_token:lower() then
|
||||
|
@ -71,13 +71,13 @@ function module.add_host(module)
|
|||
session:close{ condition = "not-authorized", text = "Given token does not match calculated token" };
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
if env.connected then
|
||||
module:log("error", "Second component attempted to connect, denying connection");
|
||||
session:close{ condition = "conflict", text = "Component already connected" };
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
env.connected = true;
|
||||
send = session.send;
|
||||
session.on_destroy = on_destroy;
|
||||
|
@ -85,7 +85,7 @@ function module.add_host(module)
|
|||
session.type = "component";
|
||||
module:log("info", "External component successfully authenticated");
|
||||
session.send(st.stanza("handshake"));
|
||||
|
||||
|
||||
return true;
|
||||
end
|
||||
module:hook("stanza/jabber:component:accept:handshake", handle_component_auth, -1);
|
||||
|
@ -116,7 +116,7 @@ function module.add_host(module)
|
|||
end
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
module:hook("iq/bare", handle_stanza, -1);
|
||||
module:hook("message/bare", handle_stanza, -1);
|
||||
module:hook("presence/bare", handle_stanza, -1);
|
||||
|
@ -275,14 +275,14 @@ function listener.onconnect(conn)
|
|||
if opt_keepalives then
|
||||
conn:setoption("keepalive", opt_keepalives);
|
||||
end
|
||||
|
||||
|
||||
session.log("info", "Incoming Jabber component connection");
|
||||
|
||||
|
||||
local stream = new_xmpp_stream(session, stream_callbacks);
|
||||
session.stream = stream;
|
||||
|
||||
|
||||
session.notopen = true;
|
||||
|
||||
|
||||
function session.reset_stream()
|
||||
session.notopen = true;
|
||||
session.stream:reset();
|
||||
|
@ -294,7 +294,7 @@ function listener.onconnect(conn)
|
|||
module:log("debug", "Received invalid XML (%s) %d bytes: %s", tostring(err), #data, data:sub(1, 300):gsub("[\r\n]+", " "):gsub("[%z\1-\31]", "_"));
|
||||
session:close("not-well-formed");
|
||||
end
|
||||
|
||||
|
||||
session.dispatch_stanza = stream_callbacks.handlestanza;
|
||||
|
||||
sessions[conn] = session;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2009-2012 Tobias Markmann
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -103,7 +103,7 @@ local function setup_compression(session, deflate_stream)
|
|||
return;
|
||||
end
|
||||
return compressed;
|
||||
end);
|
||||
end);
|
||||
end
|
||||
|
||||
-- setup decompression for a stream
|
||||
|
@ -131,13 +131,13 @@ module:hook("stanza/http://jabber.org/protocol/compress:compressed", function(ev
|
|||
-- create deflate and inflate streams
|
||||
local deflate_stream = get_deflate_stream(session);
|
||||
if not deflate_stream then return true; end
|
||||
|
||||
|
||||
local inflate_stream = get_inflate_stream(session);
|
||||
if not inflate_stream then return true; end
|
||||
|
||||
|
||||
-- setup compression for session.w
|
||||
setup_compression(session, deflate_stream);
|
||||
|
||||
|
||||
-- setup decompression for session.data
|
||||
setup_decompression(session, inflate_stream);
|
||||
session:reset_stream();
|
||||
|
@ -158,29 +158,29 @@ module:hook("stanza/http://jabber.org/protocol/compress:compress", function(even
|
|||
session.log("debug", "Client tried to establish another compression layer.");
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
-- checking if the compression method is supported
|
||||
local method = stanza:child_with_name("method");
|
||||
method = method and (method[1] or "");
|
||||
if method == "zlib" then
|
||||
session.log("debug", "zlib compression enabled.");
|
||||
|
||||
|
||||
-- create deflate and inflate streams
|
||||
local deflate_stream = get_deflate_stream(session);
|
||||
if not deflate_stream then return true; end
|
||||
|
||||
|
||||
local inflate_stream = get_inflate_stream(session);
|
||||
if not inflate_stream then return true; end
|
||||
|
||||
|
||||
(session.sends2s or session.send)(st.stanza("compressed", {xmlns=xmlns_compression_protocol}));
|
||||
session:reset_stream();
|
||||
|
||||
|
||||
-- setup compression for session.w
|
||||
setup_compression(session, deflate_stream);
|
||||
|
||||
|
||||
-- setup decompression for session.data
|
||||
setup_decompression(session, inflate_stream);
|
||||
|
||||
|
||||
session.compressed = true;
|
||||
elseif method then
|
||||
session.log("debug", "%s compression selected, but we don't support it.", tostring(method));
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -26,7 +26,7 @@ function initiate_dialback(session)
|
|||
-- generate dialback key
|
||||
session.dialback_key = generate_dialback(session.streamid, session.to_host, session.from_host);
|
||||
session.sends2s(st.stanza("db:result", { from = session.from_host, to = session.to_host }):text(session.dialback_key));
|
||||
session.log("info", "sent dialback key on outgoing s2s stream");
|
||||
session.log("debug", "sent dialback key on outgoing s2s stream");
|
||||
end
|
||||
|
||||
function verify_dialback(id, to, from, key)
|
||||
|
@ -35,7 +35,7 @@ end
|
|||
|
||||
module:hook("stanza/jabber:server:dialback:verify", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
if origin.type == "s2sin_unauthed" or origin.type == "s2sin" then
|
||||
-- We are being asked to verify the key, to ensure it was generated by us
|
||||
origin.log("debug", "verifying that dialback key is ours...");
|
||||
|
@ -62,26 +62,26 @@ end);
|
|||
|
||||
module:hook("stanza/jabber:server:dialback:result", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
if origin.type == "s2sin_unauthed" or origin.type == "s2sin" then
|
||||
-- he wants to be identified through dialback
|
||||
-- We need to check the key with the Authoritative server
|
||||
local attr = stanza.attr;
|
||||
local to, from = nameprep(attr.to), nameprep(attr.from);
|
||||
|
||||
|
||||
if not hosts[to] then
|
||||
-- Not a host that we serve
|
||||
origin.log("info", "%s tried to connect to %s, which we don't serve", from, to);
|
||||
origin.log("warn", "%s tried to connect to %s, which we don't serve", from, to);
|
||||
origin:close("host-unknown");
|
||||
return true;
|
||||
elseif not from then
|
||||
origin:close("improper-addressing");
|
||||
end
|
||||
|
||||
|
||||
origin.hosts[from] = { dialback_key = stanza[1] };
|
||||
|
||||
|
||||
dialback_requests[from.."/"..origin.streamid] = origin;
|
||||
|
||||
|
||||
-- COMPAT: ejabberd, gmail and perhaps others do not always set 'to' and 'from'
|
||||
-- on streams. We fill in the session's to/from here instead.
|
||||
if not origin.from_host then
|
||||
|
@ -102,7 +102,7 @@ end);
|
|||
|
||||
module:hook("stanza/jabber:server:dialback:verify", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
if origin.type == "s2sout_unauthed" or origin.type == "s2sout" then
|
||||
local attr = stanza.attr;
|
||||
local dialback_verifying = dialback_requests[attr.from.."/"..(attr.id or "")];
|
||||
|
@ -131,10 +131,10 @@ end);
|
|||
|
||||
module:hook("stanza/jabber:server:dialback:result", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
if origin.type == "s2sout_unauthed" or origin.type == "s2sout" then
|
||||
-- Remote server is telling us whether we passed dialback
|
||||
|
||||
|
||||
local attr = stanza.attr;
|
||||
if not hosts[attr.to] then
|
||||
origin:close("host-unknown");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -32,7 +32,9 @@ do -- validate disco_items
|
|||
end
|
||||
end
|
||||
|
||||
module:add_identity("server", "im", module:get_option_string("name", "Prosody")); -- FIXME should be in the non-existing mod_router
|
||||
if module:get_host_type() == "local" then
|
||||
module:add_identity("server", "im", module:get_option_string("name", "Prosody")); -- FIXME should be in the non-existing mod_router
|
||||
end
|
||||
module:add_feature("http://jabber.org/protocol/disco#info");
|
||||
module:add_feature("http://jabber.org/protocol/disco#items");
|
||||
|
||||
|
@ -97,7 +99,18 @@ module:hook("iq/host/http://jabber.org/protocol/disco#info:query", function(even
|
|||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type ~= "get" then return; end
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if node and node ~= "" and node ~= "http://prosody.im#"..get_server_caps_hash() then return; end -- TODO fire event?
|
||||
if node and node ~= "" and node ~= "http://prosody.im#"..get_server_caps_hash() then
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#info', node=node});
|
||||
local event = { origin = origin, stanza = stanza, reply = reply, node = node, exists = false};
|
||||
local ret = module:fire_event("host-disco-info-node", event);
|
||||
if ret ~= nil then return ret; end
|
||||
if event.exists then
|
||||
origin.send(reply);
|
||||
else
|
||||
origin.send(st.error_reply(stanza, "cancel", "item-not-found", "Node does not exist"));
|
||||
end
|
||||
return true;
|
||||
end
|
||||
local reply_query = get_server_disco_info();
|
||||
reply_query.node = node;
|
||||
local reply = st.reply(stanza):add_child(reply_query);
|
||||
|
@ -108,9 +121,21 @@ module:hook("iq/host/http://jabber.org/protocol/disco#items:query", function(eve
|
|||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type ~= "get" then return; end
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if node and node ~= "" then return; end -- TODO fire event?
|
||||
|
||||
if node and node ~= "" then
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#items', node=node});
|
||||
local event = { origin = origin, stanza = stanza, reply = reply, node = node, exists = false};
|
||||
local ret = module:fire_event("host-disco-items-node", event);
|
||||
if ret ~= nil then return ret; end
|
||||
if event.exists then
|
||||
origin.send(reply);
|
||||
else
|
||||
origin.send(st.error_reply(stanza, "cancel", "item-not-found", "Node does not exist"));
|
||||
end
|
||||
return true;
|
||||
end
|
||||
local reply = st.reply(stanza):query("http://jabber.org/protocol/disco#items");
|
||||
local ret = module:fire_event("host-disco-items", { origin = origin, stanza = stanza, reply = reply });
|
||||
if ret ~= nil then return ret; end
|
||||
for jid, name in pairs(get_children(module.host)) do
|
||||
reply:tag("item", {jid = jid, name = name~=true and name or nil}):up();
|
||||
end
|
||||
|
@ -133,12 +158,24 @@ module:hook("iq/bare/http://jabber.org/protocol/disco#info:query", function(even
|
|||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type ~= "get" then return; end
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if node and node ~= "" then return; end -- TODO fire event?
|
||||
local username = jid_split(stanza.attr.to) or origin.username;
|
||||
if not stanza.attr.to or is_contact_subscribed(username, module.host, jid_bare(stanza.attr.from)) then
|
||||
if node and node ~= "" then
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#info', node=node});
|
||||
if not reply.attr.from then reply.attr.from = origin.username.."@"..origin.host; end -- COMPAT To satisfy Psi when querying own account
|
||||
local event = { origin = origin, stanza = stanza, reply = reply, node = node, exists = false};
|
||||
local ret = module:fire_event("account-disco-info-node", event);
|
||||
if ret ~= nil then return ret; end
|
||||
if event.exists then
|
||||
origin.send(reply);
|
||||
else
|
||||
origin.send(st.error_reply(stanza, "cancel", "item-not-found", "Node does not exist"));
|
||||
end
|
||||
return true;
|
||||
end
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#info'});
|
||||
if not reply.attr.from then reply.attr.from = origin.username.."@"..origin.host; end -- COMPAT To satisfy Psi when querying own account
|
||||
module:fire_event("account-disco-info", { origin = origin, stanza = reply });
|
||||
module:fire_event("account-disco-info", { origin = origin, reply = reply });
|
||||
origin.send(reply);
|
||||
return true;
|
||||
end
|
||||
|
@ -147,12 +184,24 @@ module:hook("iq/bare/http://jabber.org/protocol/disco#items:query", function(eve
|
|||
local origin, stanza = event.origin, event.stanza;
|
||||
if stanza.attr.type ~= "get" then return; end
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if node and node ~= "" then return; end -- TODO fire event?
|
||||
local username = jid_split(stanza.attr.to) or origin.username;
|
||||
if not stanza.attr.to or is_contact_subscribed(username, module.host, jid_bare(stanza.attr.from)) then
|
||||
if node and node ~= "" then
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#items', node=node});
|
||||
if not reply.attr.from then reply.attr.from = origin.username.."@"..origin.host; end -- COMPAT To satisfy Psi when querying own account
|
||||
local event = { origin = origin, stanza = stanza, reply = reply, node = node, exists = false};
|
||||
local ret = module:fire_event("account-disco-items-node", event);
|
||||
if ret ~= nil then return ret; end
|
||||
if event.exists then
|
||||
origin.send(reply);
|
||||
else
|
||||
origin.send(st.error_reply(stanza, "cancel", "item-not-found", "Node does not exist"));
|
||||
end
|
||||
return true;
|
||||
end
|
||||
local reply = st.reply(stanza):tag('query', {xmlns='http://jabber.org/protocol/disco#items'});
|
||||
if not reply.attr.from then reply.attr.from = origin.username.."@"..origin.host; end -- COMPAT To satisfy Psi when querying own account
|
||||
module:fire_event("account-disco-items", { origin = origin, stanza = reply });
|
||||
module:fire_event("account-disco-items", { origin = origin, stanza = stanza, reply = reply });
|
||||
origin.send(reply);
|
||||
return true;
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -17,11 +17,13 @@ local jid_prep = jid.prep;
|
|||
|
||||
local module_host = module:get_host();
|
||||
|
||||
function inject_roster_contacts(username, host, roster)
|
||||
function inject_roster_contacts(event)
|
||||
local username, host= event.username, event.host;
|
||||
--module:log("debug", "Injecting group members to roster");
|
||||
local bare_jid = username.."@"..host;
|
||||
if not members[bare_jid] and not members[false] then return; end -- Not a member of any groups
|
||||
|
||||
|
||||
local roster = event.roster;
|
||||
local function import_jids_to_roster(group_name)
|
||||
for jid in pairs(groups[group_name]) do
|
||||
-- Add them to roster
|
||||
|
@ -48,7 +50,7 @@ function inject_roster_contacts(username, host, roster)
|
|||
import_jids_to_roster(group_name);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
-- Import public groups
|
||||
if members[false] then
|
||||
for _, group_name in ipairs(members[false]) do
|
||||
|
@ -56,7 +58,7 @@ function inject_roster_contacts(username, host, roster)
|
|||
import_jids_to_roster(group_name);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if roster[false] then
|
||||
roster[false].version = true;
|
||||
end
|
||||
|
@ -82,10 +84,10 @@ end
|
|||
function module.load()
|
||||
groups_file = module:get_option_string("groups_file");
|
||||
if not groups_file then return; end
|
||||
|
||||
|
||||
module:hook("roster-load", inject_roster_contacts);
|
||||
datamanager.add_callback(remove_virtual_contacts);
|
||||
|
||||
|
||||
groups = { default = {} };
|
||||
members = { };
|
||||
local curr_group = "default";
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2008-2012 Matthew Wild
|
||||
-- Copyright (C) 2008-2012 Waqas Hussain
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -111,7 +111,7 @@ function module.add_host(module)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
local function http_app_removed(event)
|
||||
local app_handlers = apps[event.item.name];
|
||||
apps[event.item.name] = nil;
|
||||
|
@ -119,7 +119,7 @@ function module.add_host(module)
|
|||
module:unhook_object_event(server, event, handler);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
module:handle_items("http-provider", http_app_added, http_app_removed);
|
||||
|
||||
server.add_host(host);
|
||||
|
|
|
@ -53,7 +53,7 @@ local entities = {
|
|||
|
||||
local function tohtml(plain)
|
||||
return (plain:gsub("[<>&'\"\n]", entities));
|
||||
|
||||
|
||||
end
|
||||
|
||||
local function get_page(code, extra)
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -43,7 +43,7 @@ module:hook("stanza/iq/jabber:iq:auth:query", function(event)
|
|||
session.send(st.error_reply(stanza, "modify", "not-acceptable", "Encryption (SSL or TLS) is required to connect to this server"));
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
local username = stanza.tags[1]:child_with_name("username");
|
||||
local password = stanza.tags[1]:child_with_name("password");
|
||||
local resource = stanza.tags[1]:child_with_name("resource");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -17,7 +17,7 @@ local user_exists = require "core.usermanager".user_exists;
|
|||
|
||||
local function process_to_bare(bare, origin, stanza)
|
||||
local user = bare_sessions[bare];
|
||||
|
||||
|
||||
local t = stanza.attr.type;
|
||||
if t == "error" then
|
||||
-- discard
|
||||
|
@ -66,7 +66,7 @@ end
|
|||
module:hook("message/full", function(data)
|
||||
-- message to full JID recieved
|
||||
local origin, stanza = data.origin, data.stanza;
|
||||
|
||||
|
||||
local session = full_sessions[stanza.attr.to];
|
||||
if session and session.send(stanza) then
|
||||
return true;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
-- Copyright (C) 2008-2010 Matthew Wild
|
||||
-- Copyright (C) 2008-2010 Waqas Hussain
|
||||
-- Copyright (C) 2010 Jeff Mitchell
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2008-2009 Matthew Wild
|
||||
-- Copyright (C) 2008-2009 Waqas Hussain
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -24,11 +24,11 @@ module:hook("message/offline/handle", function(event)
|
|||
else
|
||||
node, host = origin.username, origin.host;
|
||||
end
|
||||
|
||||
|
||||
stanza.attr.stamp, stanza.attr.stamp_legacy = datetime.datetime(), datetime.legacy();
|
||||
local result = datamanager.list_append(node, host, "offline", st.preserialize(stanza));
|
||||
stanza.attr.stamp, stanza.attr.stamp_legacy = nil, nil;
|
||||
|
||||
|
||||
return result;
|
||||
end);
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -263,19 +263,19 @@ module:hook("iq-result/bare/disco", function(event)
|
|||
end);
|
||||
|
||||
module:hook("account-disco-info", function(event)
|
||||
local stanza = event.stanza;
|
||||
stanza:tag('identity', {category='pubsub', type='pep'}):up();
|
||||
stanza:tag('feature', {var='http://jabber.org/protocol/pubsub#publish'}):up();
|
||||
local reply = event.reply;
|
||||
reply:tag('identity', {category='pubsub', type='pep'}):up();
|
||||
reply:tag('feature', {var='http://jabber.org/protocol/pubsub#publish'}):up();
|
||||
end);
|
||||
|
||||
module:hook("account-disco-items", function(event)
|
||||
local stanza = event.stanza;
|
||||
local bare = stanza.attr.to;
|
||||
local reply = event.reply;
|
||||
local bare = reply.attr.to;
|
||||
local user_data = data[bare];
|
||||
|
||||
if user_data then
|
||||
for node, _ in pairs(user_data) do
|
||||
stanza:tag('item', {jid=bare, node=node}):up(); -- TODO we need to handle queries to these nodes
|
||||
reply:tag('item', {jid=bare, node=node}):up(); -- TODO we need to handle queries to these nodes
|
||||
end
|
||||
end
|
||||
end);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -11,14 +11,11 @@ local st = require "util.stanza";
|
|||
module:add_feature("urn:xmpp:ping");
|
||||
|
||||
local function ping_handler(event)
|
||||
if event.stanza.attr.type == "get" then
|
||||
event.origin.send(st.reply(event.stanza));
|
||||
return true;
|
||||
end
|
||||
return event.origin.send(st.reply(event.stanza));
|
||||
end
|
||||
|
||||
module:hook("iq/bare/urn:xmpp:ping:ping", ping_handler);
|
||||
module:hook("iq/host/urn:xmpp:ping:ping", ping_handler);
|
||||
module:hook("iq-get/bare/urn:xmpp:ping:ping", ping_handler);
|
||||
module:hook("iq-get/host/urn:xmpp:ping:ping", ping_handler);
|
||||
|
||||
-- Ad-hoc command
|
||||
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -128,7 +128,7 @@ function syslog_sink_maker(config)
|
|||
end
|
||||
require "core.loggingmanager".register_sink_type("syslog", syslog_sink_maker);
|
||||
|
||||
local daemonize = module:get_option("daemonize");
|
||||
local daemonize = module:get_option("daemonize", prosody.installed);
|
||||
if daemonize == nil then
|
||||
local no_daemonize = module:get_option("no_daemonize"); --COMPAT w/ 0.5
|
||||
daemonize = not no_daemonize;
|
||||
|
@ -183,7 +183,7 @@ if signal.signal then
|
|||
prosody.reload_config();
|
||||
prosody.reopen_logfiles();
|
||||
end);
|
||||
|
||||
|
||||
signal.signal("SIGINT", function ()
|
||||
module:log("info", "Received SIGINT");
|
||||
prosody.unlock_globals();
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -227,7 +227,7 @@ function handle_inbound_presence_subscriptions_and_probes(origin, stanza, from_b
|
|||
local st_from, st_to = stanza.attr.from, stanza.attr.to;
|
||||
stanza.attr.from, stanza.attr.to = from_bare, to_bare;
|
||||
log("debug", "inbound presence %s from %s for %s", stanza.attr.type, from_bare, to_bare);
|
||||
|
||||
|
||||
if stanza.attr.type == "probe" then
|
||||
local result, err = rostermanager.is_contact_subscribed(node, host, from_bare);
|
||||
if result then
|
||||
|
@ -312,7 +312,7 @@ module:hook("presence/bare", function(data)
|
|||
if t ~= nil and t ~= "unavailable" and t ~= "error" then -- check for subscriptions and probes sent to bare JID
|
||||
return handle_inbound_presence_subscriptions_and_probes(origin, stanza, jid_bare(stanza.attr.from), jid_bare(stanza.attr.to));
|
||||
end
|
||||
|
||||
|
||||
local user = bare_sessions[to];
|
||||
if user then
|
||||
for _, session in pairs(user.sessions) do
|
||||
|
@ -347,7 +347,7 @@ end);
|
|||
module:hook("presence/host", function(data)
|
||||
-- inbound presence to the host
|
||||
local stanza = data.stanza;
|
||||
|
||||
|
||||
local from_bare = jid_bare(stanza.attr.from);
|
||||
local t = stanza.attr.type;
|
||||
if t == "probe" then
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
-- Copyright (C) 2009-2010 Matthew Wild
|
||||
-- Copyright (C) 2009-2010 Waqas Hussain
|
||||
-- Copyright (C) 2009 Thilo Cestonaro
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -103,7 +103,7 @@ end
|
|||
|
||||
function createOrReplaceList (privacy_lists, origin, stanza, name, entries)
|
||||
local bare_jid = origin.username.."@"..origin.host;
|
||||
|
||||
|
||||
if privacy_lists.lists == nil then
|
||||
privacy_lists.lists = {};
|
||||
end
|
||||
|
@ -119,14 +119,14 @@ function createOrReplaceList (privacy_lists, origin, stanza, name, entries)
|
|||
if to_number(item.attr.order) == nil or to_number(item.attr.order) < 0 or orderCheck[item.attr.order] ~= nil then
|
||||
return {"modify", "bad-request", "Order attribute not valid."};
|
||||
end
|
||||
|
||||
|
||||
if item.attr.type ~= nil and item.attr.type ~= "jid" and item.attr.type ~= "subscription" and item.attr.type ~= "group" then
|
||||
return {"modify", "bad-request", "Type attribute not valid."};
|
||||
end
|
||||
|
||||
|
||||
local tmp = {};
|
||||
orderCheck[item.attr.order] = true;
|
||||
|
||||
|
||||
tmp["type"] = item.attr.type;
|
||||
tmp["value"] = item.attr.value;
|
||||
tmp["action"] = item.attr.action;
|
||||
|
@ -135,13 +135,13 @@ function createOrReplaceList (privacy_lists, origin, stanza, name, entries)
|
|||
tmp["presence-out"] = false;
|
||||
tmp["message"] = false;
|
||||
tmp["iq"] = false;
|
||||
|
||||
|
||||
if #item.tags > 0 then
|
||||
for _,tag in ipairs(item.tags) do
|
||||
tmp[tag.name] = true;
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if tmp.type == "subscription" then
|
||||
if tmp.value ~= "both" and
|
||||
tmp.value ~= "to" and
|
||||
|
@ -150,13 +150,13 @@ function createOrReplaceList (privacy_lists, origin, stanza, name, entries)
|
|||
return {"cancel", "bad-request", "Subscription value must be both, to, from or none."};
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
if tmp.action ~= "deny" and tmp.action ~= "allow" then
|
||||
return {"cancel", "bad-request", "Action must be either deny or allow."};
|
||||
end
|
||||
list.items[#list.items + 1] = tmp;
|
||||
end
|
||||
|
||||
|
||||
table.sort(list, function(a, b) return a.order < b.order; end);
|
||||
|
||||
origin.send(st.reply(stanza));
|
||||
|
@ -207,14 +207,14 @@ function getList(privacy_lists, origin, stanza, name)
|
|||
return {"cancel", "item-not-found", "Unknown list specified."};
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
origin.send(reply);
|
||||
return true;
|
||||
end
|
||||
|
||||
module:hook("iq/bare/jabber:iq:privacy:query", function(data)
|
||||
local origin, stanza = data.origin, data.stanza;
|
||||
|
||||
|
||||
if stanza.attr.to == nil then -- only service requests to own bare JID
|
||||
local query = stanza.tags[1]; -- the query element
|
||||
local valid = false;
|
||||
|
@ -285,12 +285,12 @@ function checkIfNeedToBeBlocked(e, session)
|
|||
local bare_jid = session.username.."@"..session.host;
|
||||
local to = stanza.attr.to or bare_jid;
|
||||
local from = stanza.attr.from;
|
||||
|
||||
|
||||
local is_to_user = bare_jid == jid_bare(to);
|
||||
local is_from_user = bare_jid == jid_bare(from);
|
||||
|
||||
|
||||
--module:log("debug", "stanza: %s, to: %s, from: %s", tostring(stanza.name), tostring(to), tostring(from));
|
||||
|
||||
|
||||
if privacy_lists.lists == nil or
|
||||
not (session.activePrivacyList or privacy_lists.default)
|
||||
then
|
||||
|
@ -300,7 +300,7 @@ function checkIfNeedToBeBlocked(e, session)
|
|||
--module:log("debug", "Not blocking communications between user's resources");
|
||||
return; -- from one of a user's resource to another => HANDS OFF!
|
||||
end
|
||||
|
||||
|
||||
local listname = session.activePrivacyList;
|
||||
if listname == nil then
|
||||
listname = privacy_lists.default; -- no active list selected, use default list
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
-- Copyright (C) 2008-2011 Matthew Wild
|
||||
-- Copyright (C) 2008-2011 Waqas Hussain
|
||||
-- Copyright (C) 2009 Thilo Cestonaro
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
@ -30,7 +30,7 @@ function listener.onincoming(conn, data)
|
|||
(conn == initiator and target or initiator):write(data);
|
||||
return;
|
||||
end -- FIXME server.link should be doing this?
|
||||
|
||||
|
||||
if not session.greeting_done then
|
||||
local nmethods = data:byte(2) or 0;
|
||||
if data:byte(1) == 0x05 and nmethods > 0 and #data == 2 + nmethods then -- check if we have all the data
|
||||
|
@ -90,7 +90,7 @@ end
|
|||
|
||||
function module.add_host(module)
|
||||
local host, name = module:get_host(), module:get_option_string("name", "SOCKS5 Bytestreams Service");
|
||||
|
||||
|
||||
local proxy_address = module:get_option("proxy65_address", host);
|
||||
local proxy_port = next(portmanager.get_active_services():search("proxy65", nil)[1] or {});
|
||||
local proxy_acl = module:get_option("proxy65_acl");
|
||||
|
@ -101,30 +101,13 @@ function module.add_host(module)
|
|||
module:log("warn", "proxy65_port is deprecated, please put proxy65_ports = { %d } into the global section instead", legacy_config);
|
||||
end
|
||||
|
||||
module:depends("disco");
|
||||
module:add_identity("proxy", "bytestreams", name);
|
||||
module:add_feature("http://jabber.org/protocol/bytestreams");
|
||||
|
||||
module:hook("iq-get/host/http://jabber.org/protocol/disco#info:query", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
if not stanza.tags[1].attr.node then
|
||||
origin.send(st.reply(stanza):query("http://jabber.org/protocol/disco#info")
|
||||
:tag("identity", {category='proxy', type='bytestreams', name=name}):up()
|
||||
:tag("feature", {var="http://jabber.org/protocol/bytestreams"}) );
|
||||
return true;
|
||||
end
|
||||
end, -1);
|
||||
|
||||
module:hook("iq-get/host/http://jabber.org/protocol/disco#items:query", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
if not stanza.tags[1].attr.node then
|
||||
origin.send(st.reply(stanza):query("http://jabber.org/protocol/disco#items"));
|
||||
return true;
|
||||
end
|
||||
end, -1);
|
||||
|
||||
|
||||
module:hook("iq-get/host/http://jabber.org/protocol/bytestreams:query", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
-- check ACL
|
||||
while proxy_acl and #proxy_acl > 0 do -- using 'while' instead of 'if' so we can break out of it
|
||||
local jid = stanza.attr.from;
|
||||
|
@ -137,22 +120,22 @@ function module.add_host(module)
|
|||
origin.send(st.error_reply(stanza, "auth", "forbidden"));
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
local sid = stanza.tags[1].attr.sid;
|
||||
origin.send(st.reply(stanza):tag("query", {xmlns="http://jabber.org/protocol/bytestreams", sid=sid})
|
||||
:tag("streamhost", {jid=host, host=proxy_address, port=proxy_port}));
|
||||
return true;
|
||||
end);
|
||||
|
||||
|
||||
module:hook("iq-set/host/http://jabber.org/protocol/bytestreams:query", function(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
||||
|
||||
local query = stanza.tags[1];
|
||||
local sid = query.attr.sid;
|
||||
local from = stanza.attr.from;
|
||||
local to = query:get_child_text("activate");
|
||||
local prepped_to = jid_prep(to);
|
||||
|
||||
|
||||
local info = "sid: "..tostring(sid)..", initiator: "..tostring(from)..", target: "..tostring(prepped_to or to);
|
||||
if prepped_to and sid then
|
||||
local sha = sha1(sid .. from .. prepped_to, true);
|
||||
|
|
|
@ -1,463 +0,0 @@
|
|||
local pubsub = require "util.pubsub";
|
||||
local st = require "util.stanza";
|
||||
local jid_bare = require "util.jid".bare;
|
||||
local uuid_generate = require "util.uuid".generate;
|
||||
local usermanager = require "core.usermanager";
|
||||
|
||||
local xmlns_pubsub = "http://jabber.org/protocol/pubsub";
|
||||
local xmlns_pubsub_errors = "http://jabber.org/protocol/pubsub#errors";
|
||||
local xmlns_pubsub_event = "http://jabber.org/protocol/pubsub#event";
|
||||
local xmlns_pubsub_owner = "http://jabber.org/protocol/pubsub#owner";
|
||||
|
||||
local autocreate_on_publish = module:get_option_boolean("autocreate_on_publish", false);
|
||||
local autocreate_on_subscribe = module:get_option_boolean("autocreate_on_subscribe", false);
|
||||
local pubsub_disco_name = module:get_option("name");
|
||||
if type(pubsub_disco_name) ~= "string" then pubsub_disco_name = "Prosody PubSub Service"; end
|
||||
|
||||
local service;
|
||||
|
||||
local handlers = {};
|
||||
|
||||
function handle_pubsub_iq(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
local pubsub = stanza.tags[1];
|
||||
local action = pubsub.tags[1];
|
||||
if not action then
|
||||
return origin.send(st.error_reply(stanza, "cancel", "bad-request"));
|
||||
end
|
||||
local handler = handlers[stanza.attr.type.."_"..action.name];
|
||||
if handler then
|
||||
handler(origin, stanza, action);
|
||||
return true;
|
||||
end
|
||||
end
|
||||
|
||||
local pubsub_errors = {
|
||||
["conflict"] = { "cancel", "conflict" };
|
||||
["invalid-jid"] = { "modify", "bad-request", nil, "invalid-jid" };
|
||||
["jid-required"] = { "modify", "bad-request", nil, "jid-required" };
|
||||
["nodeid-required"] = { "modify", "bad-request", nil, "nodeid-required" };
|
||||
["item-not-found"] = { "cancel", "item-not-found" };
|
||||
["not-subscribed"] = { "modify", "unexpected-request", nil, "not-subscribed" };
|
||||
["forbidden"] = { "cancel", "forbidden" };
|
||||
};
|
||||
function pubsub_error_reply(stanza, error)
|
||||
local e = pubsub_errors[error];
|
||||
local reply = st.error_reply(stanza, unpack(e, 1, 3));
|
||||
if e[4] then
|
||||
reply:tag(e[4], { xmlns = xmlns_pubsub_errors }):up();
|
||||
end
|
||||
return reply;
|
||||
end
|
||||
|
||||
function handlers.get_items(origin, stanza, items)
|
||||
local node = items.attr.node;
|
||||
local item = items:get_child("item");
|
||||
local id = item and item.attr.id;
|
||||
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, results = service:get_items(node, stanza.attr.from, id);
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, results));
|
||||
end
|
||||
|
||||
local data = st.stanza("items", { node = node });
|
||||
for _, entry in pairs(results) do
|
||||
data:add_child(entry);
|
||||
end
|
||||
local reply;
|
||||
if data then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:add_child(data);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, "item-not-found");
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.get_subscriptions(origin, stanza, subscriptions)
|
||||
local node = subscriptions.attr.node;
|
||||
local ok, ret = service:get_subscriptions(node, stanza.attr.from, stanza.attr.from);
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, ret));
|
||||
end
|
||||
local reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("subscriptions");
|
||||
for _, sub in ipairs(ret) do
|
||||
reply:tag("subscription", { node = sub.node, jid = sub.jid, subscription = 'subscribed' }):up();
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_create(origin, stanza, create)
|
||||
local node = create.attr.node;
|
||||
local ok, ret, reply;
|
||||
if node then
|
||||
ok, ret = service:create(node, stanza.attr.from);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
else
|
||||
repeat
|
||||
node = uuid_generate();
|
||||
ok, ret = service:create(node, stanza.attr.from);
|
||||
until ok or ret ~= "conflict";
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("create", { node = node });
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_delete(origin, stanza, delete)
|
||||
local node = delete.attr.node;
|
||||
|
||||
local reply, notifier;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, ret = service:delete(node, stanza.attr.from);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_subscribe(origin, stanza, subscribe)
|
||||
local node, jid = subscribe.attr.node, subscribe.attr.jid;
|
||||
if not (node and jid) then
|
||||
return origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
|
||||
end
|
||||
--[[
|
||||
local options_tag, options = stanza.tags[1]:get_child("options"), nil;
|
||||
if options_tag then
|
||||
options = options_form:data(options_tag.tags[1]);
|
||||
end
|
||||
--]]
|
||||
local options_tag, options; -- FIXME
|
||||
local ok, ret = service:add_subscription(node, stanza.attr.from, jid, options);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("subscription", {
|
||||
node = node,
|
||||
jid = jid,
|
||||
subscription = "subscribed"
|
||||
}):up();
|
||||
if options_tag then
|
||||
reply:add_child(options_tag);
|
||||
end
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_unsubscribe(origin, stanza, unsubscribe)
|
||||
local node, jid = unsubscribe.attr.node, unsubscribe.attr.jid;
|
||||
if not (node and jid) then
|
||||
return origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
|
||||
end
|
||||
local ok, ret = service:remove_subscription(node, stanza.attr.from, jid);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_publish(origin, stanza, publish)
|
||||
local node = publish.attr.node;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local item = publish:get_child("item");
|
||||
local id = (item and item.attr.id);
|
||||
if not id then
|
||||
id = uuid_generate();
|
||||
if item then
|
||||
item.attr.id = id;
|
||||
end
|
||||
end
|
||||
local ok, ret = service:publish(node, stanza.attr.from, id, item);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("publish", { node = node })
|
||||
:tag("item", { id = id });
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_retract(origin, stanza, retract)
|
||||
local node, notify = retract.attr.node, retract.attr.notify;
|
||||
notify = (notify == "1") or (notify == "true");
|
||||
local item = retract:get_child("item");
|
||||
local id = item and item.attr.id
|
||||
if not (node and id) then
|
||||
return origin.send(pubsub_error_reply(stanza, node and "item-not-found" or "nodeid-required"));
|
||||
end
|
||||
local reply, notifier;
|
||||
if notify then
|
||||
notifier = st.stanza("retract", { id = id });
|
||||
end
|
||||
local ok, ret = service:retract(node, stanza.attr.from, id, notifier);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_purge(origin, stanza, purge)
|
||||
local node, notify = purge.attr.node, purge.attr.notify;
|
||||
notify = (notify == "1") or (notify == "true");
|
||||
local reply;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, ret = service:purge(node, stanza.attr.from, notify);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function simple_broadcast(kind, node, jids, item)
|
||||
if item then
|
||||
item = st.clone(item);
|
||||
item.attr.xmlns = nil; -- Clear the pubsub namespace
|
||||
end
|
||||
local message = st.message({ from = module.host, type = "headline" })
|
||||
:tag("event", { xmlns = xmlns_pubsub_event })
|
||||
:tag(kind, { node = node })
|
||||
:add_child(item);
|
||||
for jid in pairs(jids) do
|
||||
module:log("debug", "Sending notification to %s", jid);
|
||||
message.attr.to = jid;
|
||||
module:send(message);
|
||||
end
|
||||
end
|
||||
|
||||
module:hook("iq/host/"..xmlns_pubsub..":pubsub", handle_pubsub_iq);
|
||||
module:hook("iq/host/"..xmlns_pubsub_owner..":pubsub", handle_pubsub_iq);
|
||||
|
||||
local disco_info;
|
||||
|
||||
local feature_map = {
|
||||
create = { "create-nodes", "instant-nodes", "item-ids" };
|
||||
retract = { "delete-items", "retract-items" };
|
||||
purge = { "purge-nodes" };
|
||||
publish = { "publish", autocreate_on_publish and "auto-create" };
|
||||
delete = { "delete-nodes" };
|
||||
get_items = { "retrieve-items" };
|
||||
add_subscription = { "subscribe" };
|
||||
get_subscriptions = { "retrieve-subscriptions" };
|
||||
};
|
||||
|
||||
local function add_disco_features_from_service(disco, service)
|
||||
for method, features in pairs(feature_map) do
|
||||
if service[method] then
|
||||
for _, feature in ipairs(features) do
|
||||
if feature then
|
||||
disco:tag("feature", { var = xmlns_pubsub.."#"..feature }):up();
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
for affiliation in pairs(service.config.capabilities) do
|
||||
if affiliation ~= "none" and affiliation ~= "owner" then
|
||||
disco:tag("feature", { var = xmlns_pubsub.."#"..affiliation.."-affiliation" }):up();
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
local function build_disco_info(service)
|
||||
local disco_info = st.stanza("query", { xmlns = "http://jabber.org/protocol/disco#info" })
|
||||
:tag("identity", { category = "pubsub", type = "service", name = pubsub_disco_name }):up()
|
||||
:tag("feature", { var = "http://jabber.org/protocol/pubsub" }):up();
|
||||
add_disco_features_from_service(disco_info, service);
|
||||
return disco_info;
|
||||
end
|
||||
|
||||
module:hook("iq-get/host/http://jabber.org/protocol/disco#info:query", function (event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if not node then
|
||||
return origin.send(st.reply(stanza):add_child(disco_info));
|
||||
else
|
||||
local ok, ret = service:get_nodes(stanza.attr.from);
|
||||
if ok and not ret[node] then
|
||||
ok, ret = false, "item-not-found";
|
||||
end
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, ret));
|
||||
end
|
||||
local reply = st.reply(stanza)
|
||||
:tag("query", { xmlns = "http://jabber.org/protocol/disco#info", node = node })
|
||||
:tag("identity", { category = "pubsub", type = "leaf" });
|
||||
return origin.send(reply);
|
||||
end
|
||||
end);
|
||||
|
||||
local function handle_disco_items_on_node(event)
|
||||
local stanza, origin = event.stanza, event.origin;
|
||||
local query = stanza.tags[1];
|
||||
local node = query.attr.node;
|
||||
local ok, ret = service:get_items(node, stanza.attr.from);
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, ret));
|
||||
end
|
||||
|
||||
local reply = st.reply(stanza)
|
||||
:tag("query", { xmlns = "http://jabber.org/protocol/disco#items", node = node });
|
||||
|
||||
for id, item in pairs(ret) do
|
||||
reply:tag("item", { jid = module.host, name = id }):up();
|
||||
end
|
||||
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
|
||||
module:hook("iq-get/host/http://jabber.org/protocol/disco#items:query", function (event)
|
||||
if event.stanza.tags[1].attr.node then
|
||||
return handle_disco_items_on_node(event);
|
||||
end
|
||||
local ok, ret = service:get_nodes(event.stanza.attr.from);
|
||||
if not ok then
|
||||
event.origin.send(pubsub_error_reply(event.stanza, ret));
|
||||
else
|
||||
local reply = st.reply(event.stanza)
|
||||
:tag("query", { xmlns = "http://jabber.org/protocol/disco#items" });
|
||||
for node, node_obj in pairs(ret) do
|
||||
reply:tag("item", { jid = module.host, node = node, name = node_obj.config.name }):up();
|
||||
end
|
||||
event.origin.send(reply);
|
||||
end
|
||||
return true;
|
||||
end);
|
||||
|
||||
local admin_aff = module:get_option_string("default_admin_affiliation", "owner");
|
||||
local function get_affiliation(jid)
|
||||
local bare_jid = jid_bare(jid);
|
||||
if bare_jid == module.host or usermanager.is_admin(bare_jid, module.host) then
|
||||
return admin_aff;
|
||||
end
|
||||
end
|
||||
|
||||
function set_service(new_service)
|
||||
service = new_service;
|
||||
module.environment.service = service;
|
||||
disco_info = build_disco_info(service);
|
||||
end
|
||||
|
||||
function module.save()
|
||||
return { service = service };
|
||||
end
|
||||
|
||||
function module.restore(data)
|
||||
set_service(data.service);
|
||||
end
|
||||
|
||||
set_service(pubsub.new({
|
||||
capabilities = {
|
||||
none = {
|
||||
create = false;
|
||||
publish = false;
|
||||
retract = false;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
subscribe_other = false;
|
||||
unsubscribe_other = false;
|
||||
get_subscription_other = false;
|
||||
get_subscriptions_other = false;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = false;
|
||||
};
|
||||
publisher = {
|
||||
create = false;
|
||||
publish = true;
|
||||
retract = true;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
subscribe_other = false;
|
||||
unsubscribe_other = false;
|
||||
get_subscription_other = false;
|
||||
get_subscriptions_other = false;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = false;
|
||||
};
|
||||
owner = {
|
||||
create = true;
|
||||
publish = true;
|
||||
retract = true;
|
||||
delete = true;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
|
||||
subscribe_other = true;
|
||||
unsubscribe_other = true;
|
||||
get_subscription_other = true;
|
||||
get_subscriptions_other = true;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = true;
|
||||
};
|
||||
};
|
||||
|
||||
autocreate_on_publish = autocreate_on_publish;
|
||||
autocreate_on_subscribe = autocreate_on_subscribe;
|
||||
|
||||
broadcaster = simple_broadcast;
|
||||
get_affiliation = get_affiliation;
|
||||
|
||||
normalize_jid = jid_bare;
|
||||
}));
|
226
plugins/mod_pubsub/mod_pubsub.lua
Normal file
226
plugins/mod_pubsub/mod_pubsub.lua
Normal file
|
@ -0,0 +1,226 @@
|
|||
local pubsub = require "util.pubsub";
|
||||
local st = require "util.stanza";
|
||||
local jid_bare = require "util.jid".bare;
|
||||
local usermanager = require "core.usermanager";
|
||||
|
||||
local xmlns_pubsub = "http://jabber.org/protocol/pubsub";
|
||||
local xmlns_pubsub_event = "http://jabber.org/protocol/pubsub#event";
|
||||
local xmlns_pubsub_owner = "http://jabber.org/protocol/pubsub#owner";
|
||||
|
||||
local autocreate_on_publish = module:get_option_boolean("autocreate_on_publish", false);
|
||||
local autocreate_on_subscribe = module:get_option_boolean("autocreate_on_subscribe", false);
|
||||
local pubsub_disco_name = module:get_option("name");
|
||||
if type(pubsub_disco_name) ~= "string" then pubsub_disco_name = "Prosody PubSub Service"; end
|
||||
|
||||
local service;
|
||||
|
||||
local lib_pubsub = module:require "pubsub";
|
||||
local handlers = lib_pubsub.handlers;
|
||||
local pubsub_error_reply = lib_pubsub.pubsub_error_reply;
|
||||
|
||||
module:depends("disco");
|
||||
module:add_identity("pubsub", "service", pubsub_disco_name);
|
||||
module:add_feature("http://jabber.org/protocol/pubsub");
|
||||
|
||||
function handle_pubsub_iq(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
local pubsub = stanza.tags[1];
|
||||
local action = pubsub.tags[1];
|
||||
if not action then
|
||||
return origin.send(st.error_reply(stanza, "cancel", "bad-request"));
|
||||
end
|
||||
local handler = handlers[stanza.attr.type.."_"..action.name];
|
||||
if handler then
|
||||
handler(origin, stanza, action, service);
|
||||
return true;
|
||||
end
|
||||
end
|
||||
|
||||
function simple_broadcast(kind, node, jids, item)
|
||||
if item then
|
||||
item = st.clone(item);
|
||||
item.attr.xmlns = nil; -- Clear the pubsub namespace
|
||||
end
|
||||
local message = st.message({ from = module.host, type = "headline" })
|
||||
:tag("event", { xmlns = xmlns_pubsub_event })
|
||||
:tag(kind, { node = node })
|
||||
:add_child(item);
|
||||
for jid in pairs(jids) do
|
||||
module:log("debug", "Sending notification to %s", jid);
|
||||
message.attr.to = jid;
|
||||
module:send(message);
|
||||
end
|
||||
end
|
||||
|
||||
module:hook("iq/host/"..xmlns_pubsub..":pubsub", handle_pubsub_iq);
|
||||
module:hook("iq/host/"..xmlns_pubsub_owner..":pubsub", handle_pubsub_iq);
|
||||
|
||||
local feature_map = {
|
||||
create = { "create-nodes", "instant-nodes", "item-ids" };
|
||||
retract = { "delete-items", "retract-items" };
|
||||
purge = { "purge-nodes" };
|
||||
publish = { "publish", autocreate_on_publish and "auto-create" };
|
||||
delete = { "delete-nodes" };
|
||||
get_items = { "retrieve-items" };
|
||||
add_subscription = { "subscribe" };
|
||||
get_subscriptions = { "retrieve-subscriptions" };
|
||||
};
|
||||
|
||||
local function add_disco_features_from_service(service)
|
||||
for method, features in pairs(feature_map) do
|
||||
if service[method] then
|
||||
for _, feature in ipairs(features) do
|
||||
if feature then
|
||||
module:add_feature(xmlns_pubsub.."#"..feature);
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
for affiliation in pairs(service.config.capabilities) do
|
||||
if affiliation ~= "none" and affiliation ~= "owner" then
|
||||
module:add_feature(xmlns_pubsub.."#"..affiliation.."-affiliation");
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
module:hook("host-disco-info-node", function (event)
|
||||
local stanza, origin, reply, node = event.stanza, event.origin, event.reply, event.node;
|
||||
local ok, ret = service:get_nodes(stanza.attr.from);
|
||||
if not ok or not ret[node] then
|
||||
return;
|
||||
end
|
||||
event.exists = true;
|
||||
reply:tag("identity", { category = "pubsub", type = "leaf" });
|
||||
end);
|
||||
|
||||
module:hook("host-disco-items-node", function (event)
|
||||
local stanza, origin, reply, node = event.stanza, event.origin, event.reply, event.node;
|
||||
local ok, ret = service:get_items(node, stanza.attr.from);
|
||||
if not ok then
|
||||
return;
|
||||
end
|
||||
|
||||
for id, item in pairs(ret) do
|
||||
reply:tag("item", { jid = module.host, name = id }):up();
|
||||
end
|
||||
event.exists = true;
|
||||
end);
|
||||
|
||||
|
||||
module:hook("host-disco-items", function (event)
|
||||
local stanza, origin, reply = event.stanza, event.origin, event.reply;
|
||||
local ok, ret = service:get_nodes(event.stanza.attr.from);
|
||||
if not ok then
|
||||
return;
|
||||
end
|
||||
for node, node_obj in pairs(ret) do
|
||||
reply:tag("item", { jid = module.host, node = node, name = node_obj.config.name }):up();
|
||||
end
|
||||
end);
|
||||
|
||||
local admin_aff = module:get_option_string("default_admin_affiliation", "owner");
|
||||
local function get_affiliation(jid)
|
||||
local bare_jid = jid_bare(jid);
|
||||
if bare_jid == module.host or usermanager.is_admin(bare_jid, module.host) then
|
||||
return admin_aff;
|
||||
end
|
||||
end
|
||||
|
||||
function set_service(new_service)
|
||||
service = new_service;
|
||||
module.environment.service = service;
|
||||
add_disco_features_from_service(service);
|
||||
end
|
||||
|
||||
function module.save()
|
||||
return { service = service };
|
||||
end
|
||||
|
||||
function module.restore(data)
|
||||
set_service(data.service);
|
||||
end
|
||||
|
||||
function module.load()
|
||||
if module.reloading then return; end
|
||||
|
||||
set_service(pubsub.new({
|
||||
capabilities = {
|
||||
none = {
|
||||
create = false;
|
||||
publish = false;
|
||||
retract = false;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
subscribe_other = false;
|
||||
unsubscribe_other = false;
|
||||
get_subscription_other = false;
|
||||
get_subscriptions_other = false;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = false;
|
||||
};
|
||||
publisher = {
|
||||
create = false;
|
||||
publish = true;
|
||||
retract = true;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
subscribe_other = false;
|
||||
unsubscribe_other = false;
|
||||
get_subscription_other = false;
|
||||
get_subscriptions_other = false;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = false;
|
||||
};
|
||||
owner = {
|
||||
create = true;
|
||||
publish = true;
|
||||
retract = true;
|
||||
delete = true;
|
||||
get_nodes = true;
|
||||
|
||||
subscribe = true;
|
||||
unsubscribe = true;
|
||||
get_subscription = true;
|
||||
get_subscriptions = true;
|
||||
get_items = true;
|
||||
|
||||
|
||||
subscribe_other = true;
|
||||
unsubscribe_other = true;
|
||||
get_subscription_other = true;
|
||||
get_subscriptions_other = true;
|
||||
|
||||
be_subscribed = true;
|
||||
be_unsubscribed = true;
|
||||
|
||||
set_affiliation = true;
|
||||
};
|
||||
};
|
||||
|
||||
autocreate_on_publish = autocreate_on_publish;
|
||||
autocreate_on_subscribe = autocreate_on_subscribe;
|
||||
|
||||
broadcaster = simple_broadcast;
|
||||
get_affiliation = get_affiliation;
|
||||
|
||||
normalize_jid = jid_bare;
|
||||
}));
|
||||
end
|
225
plugins/mod_pubsub/pubsub.lib.lua
Normal file
225
plugins/mod_pubsub/pubsub.lib.lua
Normal file
|
@ -0,0 +1,225 @@
|
|||
local st = require "util.stanza";
|
||||
local uuid_generate = require "util.uuid".generate;
|
||||
|
||||
local xmlns_pubsub = "http://jabber.org/protocol/pubsub";
|
||||
local xmlns_pubsub_errors = "http://jabber.org/protocol/pubsub#errors";
|
||||
|
||||
local _M = {};
|
||||
|
||||
local handlers = {};
|
||||
_M.handlers = handlers;
|
||||
|
||||
local pubsub_errors = {
|
||||
["conflict"] = { "cancel", "conflict" };
|
||||
["invalid-jid"] = { "modify", "bad-request", nil, "invalid-jid" };
|
||||
["jid-required"] = { "modify", "bad-request", nil, "jid-required" };
|
||||
["nodeid-required"] = { "modify", "bad-request", nil, "nodeid-required" };
|
||||
["item-not-found"] = { "cancel", "item-not-found" };
|
||||
["not-subscribed"] = { "modify", "unexpected-request", nil, "not-subscribed" };
|
||||
["forbidden"] = { "cancel", "forbidden" };
|
||||
};
|
||||
local function pubsub_error_reply(stanza, error)
|
||||
local e = pubsub_errors[error];
|
||||
local reply = st.error_reply(stanza, unpack(e, 1, 3));
|
||||
if e[4] then
|
||||
reply:tag(e[4], { xmlns = xmlns_pubsub_errors }):up();
|
||||
end
|
||||
return reply;
|
||||
end
|
||||
_M.pubsub_error_reply = pubsub_error_reply;
|
||||
|
||||
function handlers.get_items(origin, stanza, items, service)
|
||||
local node = items.attr.node;
|
||||
local item = items:get_child("item");
|
||||
local id = item and item.attr.id;
|
||||
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, results = service:get_items(node, stanza.attr.from, id);
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, results));
|
||||
end
|
||||
|
||||
local data = st.stanza("items", { node = node });
|
||||
for _, entry in pairs(results) do
|
||||
data:add_child(entry);
|
||||
end
|
||||
local reply;
|
||||
if data then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:add_child(data);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, "item-not-found");
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.get_subscriptions(origin, stanza, subscriptions, service)
|
||||
local node = subscriptions.attr.node;
|
||||
local ok, ret = service:get_subscriptions(node, stanza.attr.from, stanza.attr.from);
|
||||
if not ok then
|
||||
return origin.send(pubsub_error_reply(stanza, ret));
|
||||
end
|
||||
local reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("subscriptions");
|
||||
for _, sub in ipairs(ret) do
|
||||
reply:tag("subscription", { node = sub.node, jid = sub.jid, subscription = 'subscribed' }):up();
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_create(origin, stanza, create, service)
|
||||
local node = create.attr.node;
|
||||
local ok, ret, reply;
|
||||
if node then
|
||||
ok, ret = service:create(node, stanza.attr.from);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
else
|
||||
repeat
|
||||
node = uuid_generate();
|
||||
ok, ret = service:create(node, stanza.attr.from);
|
||||
until ok or ret ~= "conflict";
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("create", { node = node });
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_delete(origin, stanza, delete, service)
|
||||
local node = delete.attr.node;
|
||||
|
||||
local reply, notifier;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, ret = service:delete(node, stanza.attr.from);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_subscribe(origin, stanza, subscribe, service)
|
||||
local node, jid = subscribe.attr.node, subscribe.attr.jid;
|
||||
if not (node and jid) then
|
||||
return origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
|
||||
end
|
||||
--[[
|
||||
local options_tag, options = stanza.tags[1]:get_child("options"), nil;
|
||||
if options_tag then
|
||||
options = options_form:data(options_tag.tags[1]);
|
||||
end
|
||||
--]]
|
||||
local options_tag, options; -- FIXME
|
||||
local ok, ret = service:add_subscription(node, stanza.attr.from, jid, options);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("subscription", {
|
||||
node = node,
|
||||
jid = jid,
|
||||
subscription = "subscribed"
|
||||
}):up();
|
||||
if options_tag then
|
||||
reply:add_child(options_tag);
|
||||
end
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_unsubscribe(origin, stanza, unsubscribe, service)
|
||||
local node, jid = unsubscribe.attr.node, unsubscribe.attr.jid;
|
||||
if not (node and jid) then
|
||||
return origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
|
||||
end
|
||||
local ok, ret = service:remove_subscription(node, stanza.attr.from, jid);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_publish(origin, stanza, publish, service)
|
||||
local node = publish.attr.node;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local item = publish:get_child("item");
|
||||
local id = (item and item.attr.id);
|
||||
if not id then
|
||||
id = uuid_generate();
|
||||
if item then
|
||||
item.attr.id = id;
|
||||
end
|
||||
end
|
||||
local ok, ret = service:publish(node, stanza.attr.from, id, item);
|
||||
local reply;
|
||||
if ok then
|
||||
reply = st.reply(stanza)
|
||||
:tag("pubsub", { xmlns = xmlns_pubsub })
|
||||
:tag("publish", { node = node })
|
||||
:tag("item", { id = id });
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_retract(origin, stanza, retract, service)
|
||||
local node, notify = retract.attr.node, retract.attr.notify;
|
||||
notify = (notify == "1") or (notify == "true");
|
||||
local item = retract:get_child("item");
|
||||
local id = item and item.attr.id
|
||||
if not (node and id) then
|
||||
return origin.send(pubsub_error_reply(stanza, node and "item-not-found" or "nodeid-required"));
|
||||
end
|
||||
local reply, notifier;
|
||||
if notify then
|
||||
notifier = st.stanza("retract", { id = id });
|
||||
end
|
||||
local ok, ret = service:retract(node, stanza.attr.from, id, notifier);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
function handlers.set_purge(origin, stanza, purge, service)
|
||||
local node, notify = purge.attr.node, purge.attr.notify;
|
||||
notify = (notify == "1") or (notify == "true");
|
||||
local reply;
|
||||
if not node then
|
||||
return origin.send(pubsub_error_reply(stanza, "nodeid-required"));
|
||||
end
|
||||
local ok, ret = service:purge(node, stanza.attr.from, notify);
|
||||
if ok then
|
||||
reply = st.reply(stanza);
|
||||
else
|
||||
reply = pubsub_error_reply(stanza, ret);
|
||||
end
|
||||
return origin.send(reply);
|
||||
end
|
||||
|
||||
return _M;
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -72,7 +72,7 @@ module:add_feature("jabber:iq:register");
|
|||
|
||||
local register_stream_feature = st.stanza("register", {xmlns="http://jabber.org/features/iq-register"}):up();
|
||||
module:hook("stream-features", function(event)
|
||||
local session, features = event.origin, event.features;
|
||||
local session, features = event.origin, event.features;
|
||||
|
||||
-- Advertise registration to unauthorized clients only.
|
||||
if not(allow_registration) or session.type ~= "c2s_unauthed" then
|
||||
|
@ -102,21 +102,21 @@ local function handle_registration_stanza(event)
|
|||
session.send(st.reply(stanza));
|
||||
return old_session_close(session, ...);
|
||||
end
|
||||
|
||||
|
||||
local ok, err = usermanager_delete_user(username, host);
|
||||
|
||||
|
||||
if not ok then
|
||||
module:log("debug", "Removing user account %s@%s failed: %s", username, host, err);
|
||||
session.close = old_session_close;
|
||||
session.send(st.error_reply(stanza, "cancel", "service-unavailable", err));
|
||||
return true;
|
||||
end
|
||||
|
||||
|
||||
module:log("info", "User removed their account: %s@%s", username, host);
|
||||
module:fire_event("user-deregistered", { username = username, host = host, source = "mod_register", session = session });
|
||||
else
|
||||
local username = nodeprep(query:get_child("username"):get_text());
|
||||
local password = query:get_child("password"):get_text();
|
||||
local username = nodeprep(query:get_child_text("username"));
|
||||
local password = query:get_child_text("password");
|
||||
if username and password then
|
||||
if username == session.username then
|
||||
if usermanager_set_password(username, password, session.host) then
|
||||
|
@ -170,13 +170,10 @@ local function parse_response(query)
|
|||
end
|
||||
|
||||
local recent_ips = {};
|
||||
local min_seconds_between_registrations = module:get_option("min_seconds_between_registrations");
|
||||
local whitelist_only = module:get_option("whitelist_registration_only");
|
||||
local whitelisted_ips = module:get_option("registration_whitelist") or { "127.0.0.1" };
|
||||
local blacklisted_ips = module:get_option("registration_blacklist") or {};
|
||||
|
||||
for _, ip in ipairs(whitelisted_ips) do whitelisted_ips[ip] = true; end
|
||||
for _, ip in ipairs(blacklisted_ips) do blacklisted_ips[ip] = true; end
|
||||
local min_seconds_between_registrations = module:get_option_number("min_seconds_between_registrations");
|
||||
local whitelist_only = module:get_option_boolean("whitelist_registration_only");
|
||||
local whitelisted_ips = module:get_option_set("registration_whitelist", { "127.0.0.1" })._items;
|
||||
local blacklisted_ips = module:get_option_set("registration_blacklist", {})._items;
|
||||
|
||||
module:hook("stanza/iq/jabber:iq:register:query", function(event)
|
||||
local session, stanza = event.origin, event.stanza;
|
||||
|
@ -209,7 +206,7 @@ module:hook("stanza/iq/jabber:iq:register:query", function(event)
|
|||
else
|
||||
local ip = recent_ips[session.ip];
|
||||
ip.count = ip.count + 1;
|
||||
|
||||
|
||||
if os_time() - ip.time < min_seconds_between_registrations then
|
||||
ip.time = os_time();
|
||||
session.send(st.error_reply(stanza, "wait", "not-acceptable"));
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -36,10 +36,10 @@ module:hook("iq/self/jabber:iq:roster:query", function(event)
|
|||
|
||||
if stanza.attr.type == "get" then
|
||||
local roster = st.reply(stanza);
|
||||
|
||||
|
||||
local client_ver = tonumber(stanza.tags[1].attr.ver);
|
||||
local server_ver = tonumber(session.roster[false].version or 1);
|
||||
|
||||
|
||||
if not (client_ver and server_ver) or client_ver ~= server_ver then
|
||||
roster:query("jabber:iq:roster");
|
||||
-- Client does not support versioning, or has stale roster
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -135,6 +135,12 @@ function route_to_new_session(event)
|
|||
return true;
|
||||
end
|
||||
|
||||
local function keepalive(event)
|
||||
return event.session.sends2s(' ');
|
||||
end
|
||||
|
||||
module:hook("s2s-read-timeout", keepalive, -1);
|
||||
|
||||
function module.add_host(module)
|
||||
if module:get_option_boolean("disallow_s2s", false) then
|
||||
module:log("warn", "The 'disallow_s2s' config option is deprecated, please see http://prosody.im/doc/s2s#disabling");
|
||||
|
@ -143,15 +149,16 @@ function module.add_host(module)
|
|||
module:hook("route/remote", route_to_existing_session, -1);
|
||||
module:hook("route/remote", route_to_new_session, -10);
|
||||
module:hook("s2s-authenticated", make_authenticated, -1);
|
||||
module:hook("s2s-read-timeout", keepalive, -1);
|
||||
end
|
||||
|
||||
-- Stream is authorised, and ready for normal stanzas
|
||||
function mark_connected(session)
|
||||
local sendq, send = session.sendq, session.sends2s;
|
||||
|
||||
|
||||
local from, to = session.from_host, session.to_host;
|
||||
|
||||
session.log("info", "%s s2s connection %s->%s complete", session.direction, from, to);
|
||||
|
||||
session.log("info", "%s s2s connection %s->%s complete", session.direction:gsub("^.", string.upper), from, to);
|
||||
|
||||
local event_data = { session = session };
|
||||
if session.type == "s2sout" then
|
||||
|
@ -166,7 +173,7 @@ function mark_connected(session)
|
|||
fire_global_event("s2sin-established", event_data);
|
||||
hosts[to].events.fire_event("s2sin-established", event_data);
|
||||
end
|
||||
|
||||
|
||||
if session.direction == "outgoing" then
|
||||
if sendq then
|
||||
session.log("debug", "sending %d queued stanzas across new outgoing connection to %s", #sendq, session.to_host);
|
||||
|
@ -176,7 +183,7 @@ function mark_connected(session)
|
|||
end
|
||||
session.sendq = nil;
|
||||
end
|
||||
|
||||
|
||||
session.ip_hosts = nil;
|
||||
session.srv_hosts = nil;
|
||||
end
|
||||
|
@ -211,9 +218,9 @@ function make_authenticated(event)
|
|||
return false;
|
||||
end
|
||||
session.log("debug", "connection %s->%s is now authenticated for %s", session.from_host, session.to_host, host);
|
||||
|
||||
|
||||
mark_connected(session);
|
||||
|
||||
|
||||
return true;
|
||||
end
|
||||
|
||||
|
@ -270,25 +277,28 @@ local xmlns_xmpp_streams = "urn:ietf:params:xml:ns:xmpp-streams";
|
|||
|
||||
function stream_callbacks.streamopened(session, attr)
|
||||
local send = session.sends2s;
|
||||
|
||||
|
||||
session.version = tonumber(attr.version) or 0;
|
||||
|
||||
|
||||
-- TODO: Rename session.secure to session.encrypted
|
||||
if session.secure == false then
|
||||
session.secure = true;
|
||||
session.encrypted = true;
|
||||
|
||||
-- Check if TLS compression is used
|
||||
local sock = session.conn:socket();
|
||||
if sock.info then
|
||||
session.compressed = sock:info"compression";
|
||||
elseif sock.compression then
|
||||
session.compressed = sock:compression(); --COMPAT mw/luasec-hg
|
||||
local info = sock:info();
|
||||
(session.log or log)("info", "Stream encrypted (%s with %s)", info.protocol, info.cipher);
|
||||
session.compressed = info.compression;
|
||||
else
|
||||
(session.log or log)("info", "Stream encrypted");
|
||||
session.compressed = sock.compression and sock:compression(); --COMPAT mw/luasec-hg
|
||||
end
|
||||
end
|
||||
|
||||
if session.direction == "incoming" then
|
||||
-- Send a reply stream header
|
||||
|
||||
|
||||
-- Validate to/from
|
||||
local to, from = nameprep(attr.to), nameprep(attr.from);
|
||||
if not to and attr.to then -- COMPAT: Some servers do not reliably set 'to' (especially on stream restarts)
|
||||
|
@ -299,7 +309,7 @@ function stream_callbacks.streamopened(session, attr)
|
|||
session:close({ condition = "improper-addressing", text = "Invalid 'from' address" });
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
-- Set session.[from/to]_host if they have not been set already and if
|
||||
-- this session isn't already authenticated
|
||||
if session.type == "s2sin_unauthed" and from and not session.from_host then
|
||||
|
@ -314,10 +324,10 @@ function stream_callbacks.streamopened(session, attr)
|
|||
session:close({ condition = "improper-addressing", text = "New stream 'to' attribute does not match original" });
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
-- For convenience we'll put the sanitised values into these variables
|
||||
to, from = session.to_host, session.from_host;
|
||||
|
||||
|
||||
session.streamid = uuid_gen();
|
||||
(session.log or log)("debug", "Incoming s2s received %s", st.stanza("stream:stream", attr):top_tag());
|
||||
if to then
|
||||
|
@ -352,13 +362,13 @@ function stream_callbacks.streamopened(session, attr)
|
|||
session:open_stream(session.to_host, session.from_host)
|
||||
if session.version >= 1.0 then
|
||||
local features = st.stanza("stream:features");
|
||||
|
||||
|
||||
if to then
|
||||
hosts[to].events.fire_event("s2s-stream-features", { origin = session, features = features });
|
||||
else
|
||||
(session.log or log)("warn", "No 'to' on stream header from %s means we can't offer any features", from or session.ip or "unknown host");
|
||||
end
|
||||
|
||||
|
||||
log("debug", "Sending stream features: %s", tostring(features));
|
||||
send(features);
|
||||
end
|
||||
|
@ -386,7 +396,7 @@ function stream_callbacks.streamopened(session, attr)
|
|||
end
|
||||
end
|
||||
session.send_buffer = nil;
|
||||
|
||||
|
||||
-- If server is pre-1.0, don't wait for features, just do dialback
|
||||
if session.version < 1.0 then
|
||||
if not session.dialback_verifying then
|
||||
|
@ -479,10 +489,10 @@ local function session_close(session, reason, remote_reason)
|
|||
|
||||
session.sends2s("</stream:stream>");
|
||||
function session.sends2s() return false; end
|
||||
|
||||
|
||||
local reason = remote_reason or (reason and (reason.text or reason.condition)) or reason;
|
||||
session.log("info", "%s s2s stream %s->%s closed: %s", session.direction, session.from_host or "(unknown host)", session.to_host or "(unknown host)", reason or "stream closed");
|
||||
|
||||
session.log("info", "%s s2s stream %s->%s closed: %s", session.direction:gsub("^.", string.upper), session.from_host or "(unknown host)", session.to_host or "(unknown host)", reason or "stream closed");
|
||||
|
||||
-- Authenticated incoming stream may still be sending us stanzas, so wait for </stream:stream> from remote
|
||||
local conn = session.conn;
|
||||
if reason == nil and not session.notopen and session.type == "s2sin" then
|
||||
|
@ -522,16 +532,16 @@ end
|
|||
local function initialize_session(session)
|
||||
local stream = new_xmpp_stream(session, stream_callbacks);
|
||||
session.stream = stream;
|
||||
|
||||
|
||||
session.notopen = true;
|
||||
|
||||
|
||||
function session.reset_stream()
|
||||
session.notopen = true;
|
||||
session.stream:reset();
|
||||
end
|
||||
|
||||
session.open_stream = session_open_stream;
|
||||
|
||||
|
||||
local filter = session.filter;
|
||||
function session.data(data)
|
||||
data = filter("bytes/in", data);
|
||||
|
@ -586,11 +596,12 @@ function listener.onconnect(conn)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
initialize_session(session);
|
||||
else -- Outgoing session connected
|
||||
session:open_stream(session.from_host, session.to_host);
|
||||
end
|
||||
session.ip = conn:ip();
|
||||
end
|
||||
|
||||
function listener.onincoming(conn, data)
|
||||
|
@ -599,7 +610,7 @@ function listener.onincoming(conn, data)
|
|||
session.data(data);
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
function listener.onstatus(conn, status)
|
||||
if status == "ssl-handshake-complete" then
|
||||
local session = sessions[conn];
|
||||
|
@ -617,7 +628,6 @@ function listener.ondisconnect(conn, err)
|
|||
if err and session.direction == "outgoing" and session.notopen then
|
||||
(session.log or log)("debug", "s2s connection attempt failed: %s", err);
|
||||
if s2sout.attempt_connection(session, err) then
|
||||
(session.log or log)("debug", "...so we're going to try another target");
|
||||
return; -- Session lives for now
|
||||
end
|
||||
end
|
||||
|
@ -626,6 +636,13 @@ function listener.ondisconnect(conn, err)
|
|||
end
|
||||
end
|
||||
|
||||
function listener.onreadtimeout(conn)
|
||||
local session = sessions[conn];
|
||||
if session then
|
||||
return (hosts[session.host] or prosody).events.fire_event("s2s-read-timeout", { session = session });
|
||||
end
|
||||
end
|
||||
|
||||
function listener.register_outgoing(conn, session)
|
||||
session.direction = "outgoing";
|
||||
sessions[conn] = session;
|
||||
|
@ -641,7 +658,7 @@ function check_auth_policy(event)
|
|||
elseif must_secure and insecure_domains[host] then
|
||||
must_secure = false;
|
||||
end
|
||||
|
||||
|
||||
if must_secure and (session.cert_chain_status ~= "valid" or session.cert_identity_status ~= "valid") then
|
||||
module:log("warn", "Forbidding insecure connection to/from %s", host or session.ip or "(unknown host)");
|
||||
if session.direction == "incoming" then
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -47,14 +47,14 @@ end
|
|||
function s2sout.initiate_connection(host_session)
|
||||
initialize_filters(host_session);
|
||||
host_session.version = 1;
|
||||
|
||||
|
||||
-- Kick the connection attempting machine into life
|
||||
if not s2sout.attempt_connection(host_session) then
|
||||
-- Intentionally not returning here, the
|
||||
-- session is needed, connected or not
|
||||
s2s_destroy_session(host_session);
|
||||
end
|
||||
|
||||
|
||||
if not host_session.sends2s then
|
||||
-- A sends2s which buffers data (until the stream is opened)
|
||||
-- note that data in this buffer will be sent before the stream is authed
|
||||
|
@ -75,11 +75,11 @@ end
|
|||
function s2sout.attempt_connection(host_session, err)
|
||||
local to_host = host_session.to_host;
|
||||
local connect_host, connect_port = to_host and idna_to_ascii(to_host), 5269;
|
||||
|
||||
|
||||
if not connect_host then
|
||||
return false;
|
||||
end
|
||||
|
||||
|
||||
if not err then -- This is our first attempt
|
||||
log("debug", "First attempt to connect to %s, starting with SRV lookup...", to_host);
|
||||
host_session.connecting = true;
|
||||
|
@ -100,7 +100,7 @@ function s2sout.attempt_connection(host_session, err)
|
|||
return;
|
||||
end
|
||||
t_sort(srv_hosts, compare_srv_priorities);
|
||||
|
||||
|
||||
local srv_choice = srv_hosts[1];
|
||||
host_session.srv_choice = 1;
|
||||
if srv_choice then
|
||||
|
@ -119,7 +119,7 @@ function s2sout.attempt_connection(host_session, err)
|
|||
end
|
||||
end
|
||||
end, "_xmpp-server._tcp."..connect_host..".", "SRV");
|
||||
|
||||
|
||||
return true; -- Attempt in progress
|
||||
elseif host_session.ip_hosts then
|
||||
return s2sout.try_connect(host_session, connect_host, connect_port, err);
|
||||
|
@ -129,11 +129,11 @@ function s2sout.attempt_connection(host_session, err)
|
|||
connect_host, connect_port = srv_choice.target or to_host, srv_choice.port or connect_port;
|
||||
host_session.log("info", "Connection failed (%s). Attempt #%d: This time to %s:%d", tostring(err), host_session.srv_choice, connect_host, connect_port);
|
||||
else
|
||||
host_session.log("info", "Out of connection options, can't connect to %s", tostring(host_session.to_host));
|
||||
host_session.log("info", "Failed in all attempts to connect to %s", tostring(host_session.to_host));
|
||||
-- We're out of options
|
||||
return false;
|
||||
end
|
||||
|
||||
|
||||
if not (connect_host and connect_port) then
|
||||
-- Likely we couldn't resolve DNS
|
||||
log("warn", "Hmm, we're without a host (%s) and port (%s) to connect to for %s, giving up :(", tostring(connect_host), tostring(connect_port), tostring(to_host));
|
||||
|
@ -265,11 +265,12 @@ function s2sout.try_connect(host_session, connect_host, connect_port, err)
|
|||
end
|
||||
|
||||
function s2sout.make_connect(host_session, connect_host, connect_port)
|
||||
(host_session.log or log)("info", "Beginning new connection attempt to %s ([%s]:%d)", host_session.to_host, connect_host.addr, connect_port);
|
||||
(host_session.log or log)("debug", "Beginning new connection attempt to %s ([%s]:%d)", host_session.to_host, connect_host.addr, connect_port);
|
||||
|
||||
-- Reset secure flag in case this is another
|
||||
-- connection attempt after a failed STARTTLS
|
||||
host_session.secure = nil;
|
||||
host_session.encrypted = nil;
|
||||
|
||||
local conn, handler;
|
||||
local proto = connect_host.proto;
|
||||
|
@ -280,7 +281,7 @@ function s2sout.make_connect(host_session, connect_host, connect_port)
|
|||
else
|
||||
handler = "Unsupported protocol: "..tostring(proto);
|
||||
end
|
||||
|
||||
|
||||
if not conn then
|
||||
log("warn", "Failed to create outgoing connection, system error: %s", handler);
|
||||
return false, handler;
|
||||
|
@ -292,10 +293,10 @@ function s2sout.make_connect(host_session, connect_host, connect_port)
|
|||
log("warn", "s2s connect() to %s (%s:%d) failed: %s", host_session.to_host, connect_host.addr, connect_port, err);
|
||||
return false, err;
|
||||
end
|
||||
|
||||
|
||||
conn = wrapclient(conn, connect_host.addr, connect_port, s2s_listener, "*a");
|
||||
host_session.conn = conn;
|
||||
|
||||
|
||||
local filter = initialize_filters(host_session);
|
||||
local w, log = conn.write, host_session.log;
|
||||
host_session.sends2s = function (t)
|
||||
|
@ -310,11 +311,11 @@ function s2sout.make_connect(host_session, connect_host, connect_port)
|
|||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
-- Register this outgoing connection so that xmppserver_listener knows about it
|
||||
-- otherwise it will assume it is a new incoming connection
|
||||
s2s_listener.register_outgoing(conn, host_session);
|
||||
|
||||
|
||||
log("debug", "Connection attempt in progress...");
|
||||
return true;
|
||||
end
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -242,6 +242,16 @@ module:hook("stream-features", function(event)
|
|||
return;
|
||||
end
|
||||
origin.sasl_handler = usermanager_get_sasl_handler(module.host, origin);
|
||||
if origin.encrypted then
|
||||
-- check wether LuaSec has the nifty binding to the function needed for tls-unique
|
||||
-- FIXME: would be nice to have this check only once and not for every socket
|
||||
if origin.conn:socket().getpeerfinished and origin.sasl_handler.add_cb_handler then
|
||||
origin.sasl_handler:add_cb_handler("tls-unique", function(self)
|
||||
return self.userdata:getpeerfinished();
|
||||
end);
|
||||
origin.sasl_handler["userdata"] = origin.conn:socket();
|
||||
end
|
||||
end
|
||||
local mechanisms = st.stanza("mechanisms", mechanisms_attr);
|
||||
for mechanism in pairs(origin.sasl_handler:mechanisms()) do
|
||||
if mechanism ~= "PLAIN" or origin.secure or allow_unencrypted_plain_auth then
|
||||
|
|
|
@ -93,7 +93,7 @@ local function create_table()
|
|||
elseif params.driver == "MySQL" then
|
||||
create_sql = create_sql:gsub("`value` TEXT", "`value` MEDIUMTEXT");
|
||||
end
|
||||
|
||||
|
||||
local stmt, err = connection:prepare(create_sql);
|
||||
if stmt then
|
||||
local ok = stmt:execute();
|
||||
|
@ -159,18 +159,18 @@ do -- process options to get a db connection
|
|||
end
|
||||
|
||||
params = params or { driver = "SQLite3" };
|
||||
|
||||
|
||||
if params.driver == "SQLite3" then
|
||||
params.database = resolve_relative_path(prosody.paths.data or ".", params.database or "prosody.sqlite");
|
||||
end
|
||||
|
||||
|
||||
assert(params.driver and params.database, "Both the SQL driver and the database need to be specified");
|
||||
|
||||
dburi = db2uri(params);
|
||||
connection = connections[dburi];
|
||||
|
||||
|
||||
assert(connect());
|
||||
|
||||
|
||||
-- Automatically create table, ignore failure (table probably already exists)
|
||||
create_table();
|
||||
end
|
||||
|
@ -209,7 +209,7 @@ local function dosql(sql, ...)
|
|||
local ok, err = stmt:execute(...);
|
||||
if not ok and not test_connection() then error("connection failed"); end
|
||||
if not ok then return nil, err; end
|
||||
|
||||
|
||||
return stmt;
|
||||
end
|
||||
local function getsql(sql, ...)
|
||||
|
@ -236,7 +236,7 @@ end
|
|||
local function keyval_store_get()
|
||||
local stmt, err = getsql("SELECT * FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=?");
|
||||
if not stmt then return rollback(nil, err); end
|
||||
|
||||
|
||||
local haveany;
|
||||
local result = {};
|
||||
for row in stmt:rows(true) do
|
||||
|
@ -256,7 +256,7 @@ end
|
|||
local function keyval_store_set(data)
|
||||
local affected, err = setsql("DELETE FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=?");
|
||||
if not affected then return rollback(affected, err); end
|
||||
|
||||
|
||||
if data and next(data) ~= nil then
|
||||
local extradata = {};
|
||||
for key, value in pairs(data) do
|
||||
|
@ -314,7 +314,7 @@ end
|
|||
local function map_store_get(key)
|
||||
local stmt, err = getsql("SELECT * FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=? AND `key`=?", key or "");
|
||||
if not stmt then return rollback(nil, err); end
|
||||
|
||||
|
||||
local haveany;
|
||||
local result = {};
|
||||
for row in stmt:rows(true) do
|
||||
|
@ -334,7 +334,7 @@ end
|
|||
local function map_store_set(key, data)
|
||||
local affected, err = setsql("DELETE FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=? AND `key`=?", key or "");
|
||||
if not affected then return rollback(affected, err); end
|
||||
|
||||
|
||||
if data and next(data) ~= nil then
|
||||
if type(key) == "string" and key ~= "" then
|
||||
local t, value = serialize(data);
|
||||
|
@ -365,15 +365,15 @@ local list_store = {};
|
|||
list_store.__index = list_store;
|
||||
function list_store:scan(username, from, to, jid, typ)
|
||||
user,store = username,self.store;
|
||||
|
||||
|
||||
local cols = {"from", "to", "jid", "typ"};
|
||||
local vals = { from , to , jid , typ };
|
||||
local stmt, err;
|
||||
local query = "SELECT * FROM `prosodyarchive` WHERE `host`=? AND `user`=? AND `store`=?";
|
||||
|
||||
|
||||
query = query.." ORDER BY time";
|
||||
--local stmt, err = getsql("SELECT * FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=? AND `key`=?", key or "");
|
||||
|
||||
|
||||
return nil, "not-implemented"
|
||||
end
|
||||
|
||||
|
|
377
plugins/mod_storage_sql2.lua
Normal file
377
plugins/mod_storage_sql2.lua
Normal file
|
@ -0,0 +1,377 @@
|
|||
|
||||
local json = require "util.json";
|
||||
local xml_parse = require "util.xml".parse;
|
||||
local uuid = require "util.uuid";
|
||||
local resolve_relative_path = require "core.configmanager".resolve_relative_path;
|
||||
|
||||
local stanza_mt = require"util.stanza".stanza_mt;
|
||||
local getmetatable = getmetatable;
|
||||
local t_concat = table.concat;
|
||||
local function is_stanza(x) return getmetatable(x) == stanza_mt; end
|
||||
|
||||
local noop = function() end
|
||||
local unpack = unpack
|
||||
local function iterator(result)
|
||||
return function(result)
|
||||
local row = result();
|
||||
if row ~= nil then
|
||||
return unpack(row);
|
||||
end
|
||||
end, result, nil;
|
||||
end
|
||||
|
||||
local mod_sql = module:require("sql");
|
||||
local params = module:get_option("sql");
|
||||
|
||||
local engine; -- TODO create engine
|
||||
|
||||
local function create_table()
|
||||
local Table,Column,Index = mod_sql.Table,mod_sql.Column,mod_sql.Index;
|
||||
|
||||
local ProsodyTable = Table {
|
||||
name="prosody";
|
||||
Column { name="host", type="TEXT", nullable=false };
|
||||
Column { name="user", type="TEXT", nullable=false };
|
||||
Column { name="store", type="TEXT", nullable=false };
|
||||
Column { name="key", type="TEXT", nullable=false };
|
||||
Column { name="type", type="TEXT", nullable=false };
|
||||
Column { name="value", type="MEDIUMTEXT", nullable=false };
|
||||
Index { name="prosody_index", "host", "user", "store", "key" };
|
||||
};
|
||||
engine:transaction(function()
|
||||
ProsodyTable:create(engine);
|
||||
end);
|
||||
|
||||
local ProsodyArchiveTable = Table {
|
||||
name="prosodyarchive";
|
||||
Column { name="sort_id", type="INTEGER", primary_key=true, auto_increment=true };
|
||||
Column { name="host", type="TEXT", nullable=false };
|
||||
Column { name="user", type="TEXT", nullable=false };
|
||||
Column { name="store", type="TEXT", nullable=false };
|
||||
Column { name="key", type="TEXT", nullable=false }; -- item id
|
||||
Column { name="when", type="INTEGER", nullable=false }; -- timestamp
|
||||
Column { name="with", type="TEXT", nullable=false }; -- related id
|
||||
Column { name="type", type="TEXT", nullable=false };
|
||||
Column { name="value", type="MEDIUMTEXT", nullable=false };
|
||||
Index { name="prosodyarchive_index", unique = true, "host", "user", "store", "key" };
|
||||
};
|
||||
engine:transaction(function()
|
||||
ProsodyArchiveTable:create(engine);
|
||||
end);
|
||||
end
|
||||
|
||||
local function upgrade_table()
|
||||
if params.driver == "MySQL" then
|
||||
local success,err = engine:transaction(function()
|
||||
local result = engine:execute("SHOW COLUMNS FROM prosody WHERE Field='value' and Type='text'");
|
||||
if result:rowcount() > 0 then
|
||||
module:log("info", "Upgrading database schema...");
|
||||
engine:execute("ALTER TABLE prosody MODIFY COLUMN `value` MEDIUMTEXT");
|
||||
module:log("info", "Database table automatically upgraded");
|
||||
end
|
||||
return true;
|
||||
end);
|
||||
if not success then
|
||||
module:log("error", "Failed to check/upgrade database schema (%s), please see "
|
||||
.."http://prosody.im/doc/mysql for help",
|
||||
err or "unknown error");
|
||||
return false;
|
||||
end
|
||||
-- COMPAT w/pre-0.9: Upgrade tables to UTF-8 if not already
|
||||
local check_encoding_query = "SELECT `COLUMN_NAME`,`COLUMN_TYPE` FROM `information_schema`.`columns` WHERE `TABLE_NAME`='prosody' AND ( `CHARACTER_SET_NAME`!='utf8' OR `COLLATION_NAME`!='utf8_bin' );";
|
||||
success,err = engine:transaction(function()
|
||||
local result = engine:execute(check_encoding_query);
|
||||
local n_bad_columns = result:rowcount();
|
||||
if n_bad_columns > 0 then
|
||||
module:log("warn", "Found %d columns in prosody table requiring encoding change, updating now...", n_bad_columns);
|
||||
local fix_column_query1 = "ALTER TABLE `prosody` CHANGE `%s` `%s` BLOB;";
|
||||
local fix_column_query2 = "ALTER TABLE `prosody` CHANGE `%s` `%s` %s CHARACTER SET 'utf8' COLLATE 'utf8_bin';";
|
||||
for row in result:rows() do
|
||||
local column_name, column_type = unpack(row);
|
||||
engine:execute(fix_column_query1:format(column_name, column_name));
|
||||
engine:execute(fix_column_query2:format(column_name, column_name, column_type));
|
||||
end
|
||||
module:log("info", "Database encoding upgrade complete!");
|
||||
end
|
||||
end);
|
||||
success,err = engine:transaction(function() return engine:execute(check_encoding_query); end);
|
||||
if not success then
|
||||
module:log("error", "Failed to check/upgrade database encoding: %s", err or "unknown error");
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
do -- process options to get a db connection
|
||||
params = params or { driver = "SQLite3" };
|
||||
|
||||
if params.driver == "SQLite3" then
|
||||
params.database = resolve_relative_path(prosody.paths.data or ".", params.database or "prosody.sqlite");
|
||||
end
|
||||
|
||||
assert(params.driver and params.database, "Both the SQL driver and the database need to be specified");
|
||||
|
||||
--local dburi = db2uri(params);
|
||||
engine = mod_sql:create_engine(params);
|
||||
|
||||
engine:set_encoding();
|
||||
|
||||
if module:get_option("sql_manage_tables", true) then
|
||||
-- Automatically create table, ignore failure (table probably already exists)
|
||||
create_table();
|
||||
-- Encoding mess
|
||||
upgrade_table();
|
||||
end
|
||||
end
|
||||
|
||||
local function serialize(value)
|
||||
local t = type(value);
|
||||
if t == "string" or t == "boolean" or t == "number" then
|
||||
return t, tostring(value);
|
||||
elseif is_stanza(value) then
|
||||
return "xml", tostring(value);
|
||||
elseif t == "table" then
|
||||
local value,err = json.encode(value);
|
||||
if value then return "json", value; end
|
||||
return nil, err;
|
||||
end
|
||||
return nil, "Unhandled value type: "..t;
|
||||
end
|
||||
local function deserialize(t, value)
|
||||
if t == "string" then return value;
|
||||
elseif t == "boolean" then
|
||||
if value == "true" then return true;
|
||||
elseif value == "false" then return false; end
|
||||
elseif t == "number" then return tonumber(value);
|
||||
elseif t == "json" then
|
||||
return json.decode(value);
|
||||
elseif t == "xml" then
|
||||
return xml_parse(value);
|
||||
end
|
||||
end
|
||||
|
||||
local host = module.host;
|
||||
local user, store;
|
||||
|
||||
local function keyval_store_get()
|
||||
local haveany;
|
||||
local result = {};
|
||||
for row in engine:select("SELECT `key`,`type`,`value` FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=?", host, user or "", store) do
|
||||
haveany = true;
|
||||
local k = row[1];
|
||||
local v = deserialize(row[2], row[3]);
|
||||
if k and v then
|
||||
if k ~= "" then result[k] = v; elseif type(v) == "table" then
|
||||
for a,b in pairs(v) do
|
||||
result[a] = b;
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
if haveany then
|
||||
return result;
|
||||
end
|
||||
end
|
||||
local function keyval_store_set(data)
|
||||
engine:delete("DELETE FROM `prosody` WHERE `host`=? AND `user`=? AND `store`=?", host, user or "", store);
|
||||
|
||||
if data and next(data) ~= nil then
|
||||
local extradata = {};
|
||||
for key, value in pairs(data) do
|
||||
if type(key) == "string" and key ~= "" then
|
||||
local t, value = serialize(value);
|
||||
assert(t, value);
|
||||
engine:insert("INSERT INTO `prosody` (`host`,`user`,`store`,`key`,`type`,`value`) VALUES (?,?,?,?,?,?)", host, user or "", store, key, t, value);
|
||||
else
|
||||
extradata[key] = value;
|
||||
end
|
||||
end
|
||||
if next(extradata) ~= nil then
|
||||
local t, extradata = serialize(extradata);
|
||||
assert(t, extradata);
|
||||
engine:insert("INSERT INTO `prosody` (`host`,`user`,`store`,`key`,`type`,`value`) VALUES (?,?,?,?,?,?)", host, user or "", store, "", t, extradata);
|
||||
end
|
||||
end
|
||||
return true;
|
||||
end
|
||||
|
||||
local keyval_store = {};
|
||||
keyval_store.__index = keyval_store;
|
||||
function keyval_store:get(username)
|
||||
user,store = username,self.store;
|
||||
return select(2, engine:transaction(keyval_store_get));
|
||||
end
|
||||
function keyval_store:set(username, data)
|
||||
user,store = username,self.store;
|
||||
return engine:transaction(function()
|
||||
return keyval_store_set(data);
|
||||
end);
|
||||
end
|
||||
function keyval_store:users()
|
||||
local ok, result = engine:transaction(function()
|
||||
return engine:select("SELECT DISTINCT `user` FROM `prosody` WHERE `host`=? AND `store`=?", host, self.store);
|
||||
end);
|
||||
if not ok then return ok, result end
|
||||
return iterator(result);
|
||||
end
|
||||
|
||||
local archive_store = {}
|
||||
archive_store.__index = archive_store
|
||||
function archive_store:append(username, key, when, with, value)
|
||||
if value == nil then -- COMPAT early versions
|
||||
when, with, value, key = key, when, with, value
|
||||
end
|
||||
local user,store = username,self.store;
|
||||
return engine:transaction(function()
|
||||
if key then
|
||||
engine:delete("DELETE FROM `prosodyarchive` WHERE `host`=? AND `user`=? AND `store`=? AND `key`=?", host, user or "", store, key);
|
||||
else
|
||||
key = uuid.generate();
|
||||
end
|
||||
local t, value = serialize(value);
|
||||
engine:insert("INSERT INTO `prosodyarchive` (`host`, `user`, `store`, `when`, `with`, `key`, `type`, `value`) VALUES (?,?,?,?,?,?,?,?)", host, user or "", store, when, with, key, t, value);
|
||||
return key;
|
||||
end);
|
||||
end
|
||||
|
||||
-- Helpers for building the WHERE clause
|
||||
local function archive_where(query, args, where)
|
||||
-- Time range, inclusive
|
||||
if query.start then
|
||||
args[#args+1] = query.start
|
||||
where[#where+1] = "`when` >= ?"
|
||||
end
|
||||
|
||||
if query["end"] then
|
||||
args[#args+1] = query["end"];
|
||||
if query.start then
|
||||
where[#where] = "`when` BETWEEN ? AND ?" -- is this inclusive?
|
||||
else
|
||||
where[#where+1] = "`when` <= ?"
|
||||
end
|
||||
end
|
||||
|
||||
-- Related name
|
||||
if query.with then
|
||||
where[#where+1] = "`with` = ?";
|
||||
args[#args+1] = query.with
|
||||
end
|
||||
|
||||
-- Unique id
|
||||
if query.key then
|
||||
where[#where+1] = "`key` = ?";
|
||||
args[#args+1] = query.key
|
||||
end
|
||||
end
|
||||
local function archive_where_id_range(query, args, where)
|
||||
local args_len = #args
|
||||
-- Before or after specific item, exclusive
|
||||
if query.after then -- keys better be unique!
|
||||
where[#where+1] = "`sort_id` > (SELECT `sort_id` FROM `prosodyarchive` WHERE `key` = ? AND `host` = ? AND `user` = ? AND `store` = ? LIMIT 1)"
|
||||
args[args_len+1], args[args_len+2], args[args_len+3], args[args_len+4] = query.after, args[1], args[2], args[3];
|
||||
args_len = args_len + 4
|
||||
end
|
||||
if query.before then
|
||||
where[#where+1] = "`sort_id` < (SELECT `sort_id` FROM `prosodyarchive` WHERE `key` = ? AND `host` = ? AND `user` = ? AND `store` = ? LIMIT 1)"
|
||||
args[args_len+1], args[args_len+2], args[args_len+3], args[args_len+4] = query.before, args[1], args[2], args[3];
|
||||
end
|
||||
end
|
||||
|
||||
function archive_store:find(username, query)
|
||||
query = query or {};
|
||||
local user,store = username,self.store;
|
||||
local total;
|
||||
local ok, result = engine:transaction(function()
|
||||
local sql_query = "SELECT `key`, `type`, `value`, `when` FROM `prosodyarchive` WHERE %s ORDER BY `sort_id` %s%s;";
|
||||
local args = { host, user or "", store, };
|
||||
local where = { "`host` = ?", "`user` = ?", "`store` = ?", };
|
||||
|
||||
archive_where(query, args, where);
|
||||
|
||||
-- Total matching
|
||||
if query.total then
|
||||
local stats = engine:select(sql_query:gsub("^(SELECT).-(FROM)", "%1 COUNT(*) %2"):format(t_concat(where, " AND "), "DESC", ""), unpack(args));
|
||||
if stats then
|
||||
local _total = stats()
|
||||
total = _total and _total[1];
|
||||
end
|
||||
if query.limit == 0 then -- Skip the real query
|
||||
return noop, total;
|
||||
end
|
||||
end
|
||||
|
||||
archive_where_id_range(query, args, where);
|
||||
|
||||
if query.limit then
|
||||
args[#args+1] = query.limit;
|
||||
end
|
||||
|
||||
sql_query = sql_query:format(t_concat(where, " AND "), query.reverse and "DESC" or "ASC", query.limit and " LIMIT ?" or "");
|
||||
module:log("debug", sql_query);
|
||||
return engine:select(sql_query, unpack(args));
|
||||
end);
|
||||
if not ok then return ok, result end
|
||||
return function()
|
||||
local row = result();
|
||||
if row ~= nil then
|
||||
return row[1], deserialize(row[2], row[3]), row[4];
|
||||
end
|
||||
end, total;
|
||||
end
|
||||
|
||||
function archive_store:delete(username, query)
|
||||
query = query or {};
|
||||
local user,store = username,self.store;
|
||||
return engine:transaction(function()
|
||||
local sql_query = "DELETE FROM `prosodyarchive` WHERE %s;";
|
||||
local args = { host, user or "", store, };
|
||||
local where = { "`host` = ?", "`user` = ?", "`store` = ?", };
|
||||
if user == true then
|
||||
table.remove(args, 2);
|
||||
table.remove(where, 2);
|
||||
end
|
||||
archive_where(query, args, where);
|
||||
archive_where_id_range(query, args, where);
|
||||
sql_query = sql_query:format(t_concat(where, " AND "));
|
||||
module:log("debug", sql_query);
|
||||
return engine:delete(sql_query, unpack(args));
|
||||
end);
|
||||
end
|
||||
|
||||
local stores = {
|
||||
keyval = keyval_store;
|
||||
archive = archive_store;
|
||||
};
|
||||
|
||||
local driver = {};
|
||||
|
||||
function driver:open(store, typ)
|
||||
local store_mt = stores[typ or "keyval"];
|
||||
if store_mt then
|
||||
return setmetatable({ store = store }, store_mt);
|
||||
end
|
||||
return nil, "unsupported-store";
|
||||
end
|
||||
|
||||
function driver:stores(username)
|
||||
local sql = "SELECT DISTINCT `store` FROM `prosody` WHERE `host`=? AND `user`" ..
|
||||
(username == true and "!=?" or "=?");
|
||||
if username == true or not username then
|
||||
username = "";
|
||||
end
|
||||
local ok, result = engine:transaction(function()
|
||||
return engine:select(sql, host, username);
|
||||
end);
|
||||
if not ok then return ok, result end
|
||||
return iterator(result);
|
||||
end
|
||||
|
||||
function driver:purge(username)
|
||||
return engine:transaction(function()
|
||||
local stmt,err = engine:delete("DELETE FROM `prosody` WHERE `host`=? AND `user`=?", host, username);
|
||||
return true,err;
|
||||
end);
|
||||
end
|
||||
|
||||
module:provides("storage", driver);
|
||||
|
||||
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -29,20 +29,47 @@ local s2s_feature = st.stanza("starttls", starttls_attr);
|
|||
if c2s_require_encryption then c2s_feature:tag("required"):up(); end
|
||||
if s2s_require_encryption then s2s_feature:tag("required"):up(); end
|
||||
|
||||
local global_ssl_ctx = prosody.global_ssl_ctx;
|
||||
|
||||
local hosts = prosody.hosts;
|
||||
local host = hosts[module.host];
|
||||
|
||||
local function can_do_tls(session)
|
||||
if session.type == "c2s_unauthed" then
|
||||
return session.conn.starttls and host.ssl_ctx_in;
|
||||
elseif session.type == "s2sin_unauthed" and allow_s2s_tls then
|
||||
return session.conn.starttls and host.ssl_ctx_in;
|
||||
elseif session.direction == "outgoing" and allow_s2s_tls then
|
||||
return session.conn.starttls and host.ssl_ctx;
|
||||
local ssl_ctx_c2s, ssl_ctx_s2sout, ssl_ctx_s2sin;
|
||||
do
|
||||
local function get_ssl_cfg(typ)
|
||||
local cfg_key = (typ and typ.."_" or "").."ssl";
|
||||
local ssl_config = config.rawget(module.host, cfg_key);
|
||||
if not ssl_config then
|
||||
local base_host = module.host:match("%.(.*)");
|
||||
ssl_config = config.get(base_host, cfg_key);
|
||||
end
|
||||
return ssl_config or typ and get_ssl_cfg();
|
||||
end
|
||||
return false;
|
||||
|
||||
local ssl_config, err = get_ssl_cfg("c2s");
|
||||
ssl_ctx_c2s, err = create_context(host.host, "server", ssl_config); -- for incoming client connections
|
||||
if err then module:log("error", "Error creating context for c2s: %s", err); end
|
||||
|
||||
ssl_config = get_ssl_cfg("s2s");
|
||||
ssl_ctx_s2sin, err = create_context(host.host, "server", ssl_config); -- for incoming server connections
|
||||
ssl_ctx_s2sout = create_context(host.host, "client", ssl_config); -- for outgoing server connections
|
||||
if err then module:log("error", "Error creating context for s2s: %s", err); end -- Both would have the same issue
|
||||
end
|
||||
|
||||
local function can_do_tls(session)
|
||||
if not session.conn.starttls then
|
||||
return false;
|
||||
elseif session.ssl_ctx then
|
||||
return true;
|
||||
end
|
||||
if session.type == "c2s_unauthed" then
|
||||
session.ssl_ctx = ssl_ctx_c2s;
|
||||
elseif session.type == "s2sin_unauthed" and allow_s2s_tls then
|
||||
session.ssl_ctx = ssl_ctx_s2sin;
|
||||
elseif session.direction == "outgoing" and allow_s2s_tls then
|
||||
session.ssl_ctx = ssl_ctx_s2sout;
|
||||
else
|
||||
return false;
|
||||
end
|
||||
return session.ssl_ctx;
|
||||
end
|
||||
|
||||
-- Hook <starttls/>
|
||||
|
@ -51,9 +78,7 @@ module:hook("stanza/urn:ietf:params:xml:ns:xmpp-tls:starttls", function(event)
|
|||
if can_do_tls(origin) then
|
||||
(origin.sends2s or origin.send)(starttls_proceed);
|
||||
origin:reset_stream();
|
||||
local host = origin.to_host or origin.host;
|
||||
local ssl_ctx = host and hosts[host].ssl_ctx_in or global_ssl_ctx;
|
||||
origin.conn:starttls(ssl_ctx);
|
||||
origin.conn:starttls(origin.ssl_ctx);
|
||||
origin.log("debug", "TLS negotiation started for %s...", origin.type);
|
||||
origin.secure = false;
|
||||
else
|
||||
|
@ -91,30 +116,7 @@ end, 500);
|
|||
module:hook_stanza(xmlns_starttls, "proceed", function (session, stanza)
|
||||
module:log("debug", "Proceeding with TLS on s2sout...");
|
||||
session:reset_stream();
|
||||
local ssl_ctx = session.from_host and hosts[session.from_host].ssl_ctx or global_ssl_ctx;
|
||||
session.conn:starttls(ssl_ctx);
|
||||
session.conn:starttls(session.ssl_ctx);
|
||||
session.secure = false;
|
||||
return true;
|
||||
end);
|
||||
|
||||
local function assert_log(ret, err)
|
||||
if not ret then
|
||||
module:log("error", "Unable to initialize TLS: %s", err);
|
||||
end
|
||||
return ret;
|
||||
end
|
||||
|
||||
function module.load()
|
||||
local ssl_config = config.rawget(module.host, "ssl");
|
||||
if not ssl_config then
|
||||
local base_host = module.host:match("%.(.*)");
|
||||
ssl_config = config.get(base_host, "ssl");
|
||||
end
|
||||
host.ssl_ctx = assert_log(create_context(host.host, "client", ssl_config)); -- for outgoing connections
|
||||
host.ssl_ctx_in = assert_log(create_context(host.host, "server", ssl_config)); -- for incoming connections
|
||||
end
|
||||
|
||||
function module.unload()
|
||||
host.ssl_ctx = nil;
|
||||
host.ssl_ctx_in = nil;
|
||||
end
|
||||
|
|
4
plugins/mod_unknown.lua
Normal file
4
plugins/mod_unknown.lua
Normal file
|
@ -0,0 +1,4 @@
|
|||
-- Unknown platform stub
|
||||
module:set_global();
|
||||
|
||||
-- TODO Do things that make sense if we don't know about the platform
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
4
plugins/mod_windows.lua
Normal file
4
plugins/mod_windows.lua
Normal file
|
@ -0,0 +1,4 @@
|
|||
-- Windows platform stub
|
||||
module:set_global();
|
||||
|
||||
-- TODO Add Windows-specific things here
|
|
@ -1,11 +1,12 @@
|
|||
-- 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";
|
||||
|
||||
if module:get_host_type() ~= "component" then
|
||||
error("MUC should be loaded as a component, please see http://prosody.im/doc/components", 0);
|
||||
|
@ -16,12 +17,15 @@ local muc_name = module:get_option("name");
|
|||
if type(muc_name) ~= "string" then muc_name = "Prosody Chatrooms"; end
|
||||
local restrict_room_creation = module:get_option("restrict_room_creation");
|
||||
if restrict_room_creation then
|
||||
if restrict_room_creation == true then
|
||||
if restrict_room_creation == true then
|
||||
restrict_room_creation = "admin";
|
||||
elseif restrict_room_creation ~= "admin" and restrict_room_creation ~= "local" then
|
||||
restrict_room_creation = nil;
|
||||
end
|
||||
end
|
||||
local lock_rooms = module:get_option_boolean("muc_room_locking", false);
|
||||
local lock_room_timeout = module:get_option_number("muc_room_lock_timeout", 300);
|
||||
|
||||
local muclib = module:require "muc";
|
||||
local muc_new_room = muclib.new_room;
|
||||
local jid_split = require "util.jid".split;
|
||||
|
@ -40,12 +44,17 @@ local room_configs = module:open_store("config");
|
|||
-- Configurable options
|
||||
muclib.set_max_history_length(module:get_option_number("max_history_messages"));
|
||||
|
||||
module:depends("disco");
|
||||
module:add_identity("conference", "text", muc_name);
|
||||
module:add_feature("http://jabber.org/protocol/muc");
|
||||
|
||||
local function is_admin(jid)
|
||||
return um_is_admin(jid, module.host);
|
||||
end
|
||||
|
||||
local _set_affiliation = muc_new_room.room_mt.set_affiliation;
|
||||
local _get_affiliation = muc_new_room.room_mt.get_affiliation;
|
||||
room_mt = muclib.room_mt; -- Yes, global.
|
||||
local _set_affiliation = room_mt.set_affiliation;
|
||||
local _get_affiliation = room_mt.get_affiliation;
|
||||
function muclib.room_mt:get_affiliation(jid)
|
||||
if is_admin(jid) then return "owner"; end
|
||||
return _get_affiliation(self, jid);
|
||||
|
@ -83,6 +92,16 @@ function create_room(jid)
|
|||
room.route_stanza = room_route_stanza;
|
||||
room.save = room_save;
|
||||
rooms[jid] = room;
|
||||
if lock_rooms then
|
||||
room.locked = true;
|
||||
if lock_room_timeout and lock_room_timeout > 0 then
|
||||
module:add_timer(lock_room_timeout, function ()
|
||||
if room.locked then
|
||||
room:destroy(); -- Not unlocked in time
|
||||
end
|
||||
end);
|
||||
end
|
||||
end
|
||||
module:fire_event("muc-room-created", { room = room });
|
||||
return room;
|
||||
end
|
||||
|
@ -107,20 +126,15 @@ local host_room = muc_new_room(muc_host);
|
|||
host_room.route_stanza = room_route_stanza;
|
||||
host_room.save = room_save;
|
||||
|
||||
local function get_disco_info(stanza)
|
||||
return st.iq({type='result', id=stanza.attr.id, from=muc_host, to=stanza.attr.from}):query("http://jabber.org/protocol/disco#info")
|
||||
:tag("identity", {category='conference', type='text', name=muc_name}):up()
|
||||
:tag("feature", {var="http://jabber.org/protocol/muc"}); -- TODO cache disco reply
|
||||
end
|
||||
local function get_disco_items(stanza)
|
||||
local reply = st.iq({type='result', id=stanza.attr.id, from=muc_host, to=stanza.attr.from}):query("http://jabber.org/protocol/disco#items");
|
||||
module:hook("host-disco-items", function(event)
|
||||
local reply = event.reply;
|
||||
module:log("debug", "host-disco-items called");
|
||||
for jid, room in pairs(rooms) do
|
||||
if not room:is_hidden() then
|
||||
if not room:get_hidden() then
|
||||
reply:tag("item", {jid=jid, name=room:get_name()}):up();
|
||||
end
|
||||
end
|
||||
return reply; -- TODO cache disco reply
|
||||
end
|
||||
end);
|
||||
|
||||
local function handle_to_domain(event)
|
||||
local origin, stanza = event.origin, event.stanza;
|
||||
|
@ -129,11 +143,7 @@ local function handle_to_domain(event)
|
|||
if stanza.name == "iq" and type == "get" then
|
||||
local xmlns = stanza.tags[1].attr.xmlns;
|
||||
local node = stanza.tags[1].attr.node;
|
||||
if xmlns == "http://jabber.org/protocol/disco#info" and not node then
|
||||
origin.send(get_disco_info(stanza));
|
||||
elseif xmlns == "http://jabber.org/protocol/disco#items" and not node then
|
||||
origin.send(get_disco_items(stanza));
|
||||
elseif xmlns == "http://jabber.org/protocol/muc#unique" then
|
||||
if xmlns == "http://jabber.org/protocol/muc#unique" then
|
||||
origin.send(st.reply(stanza):tag("unique", {xmlns = xmlns}):text(uuid_gen())); -- FIXME Random UUIDs can theoretically have collisions
|
||||
else
|
||||
origin.send(st.error_reply(stanza, "cancel", "service-unavailable")); -- TODO disco/etc
|
||||
|
@ -220,7 +230,8 @@ function shutdown_component()
|
|||
if not saved then
|
||||
local stanza = st.presence({type = "unavailable"})
|
||||
:tag("x", {xmlns = "http://jabber.org/protocol/muc#user"})
|
||||
:tag("item", { affiliation='none', role='none' }):up();
|
||||
:tag("item", { affiliation='none', role='none' }):up()
|
||||
:tag("status", { code = "332"}):up();
|
||||
for roomjid, room in pairs(rooms) do
|
||||
shutdown_room(room, stanza);
|
||||
end
|
||||
|
@ -229,3 +240,39 @@ function shutdown_component()
|
|||
end
|
||||
module.unload = shutdown_component;
|
||||
module:hook_global("server-stopping", shutdown_component);
|
||||
|
||||
-- Ad-hoc commands
|
||||
module:depends("adhoc")
|
||||
local t_concat = table.concat;
|
||||
local keys = require "util.iterators".keys;
|
||||
local adhoc_new = module:require "adhoc".new;
|
||||
local adhoc_initial = require "util.adhoc".new_initial_data_form;
|
||||
local dataforms_new = require "util.dataforms".new;
|
||||
|
||||
local destroy_rooms_layout = dataforms_new {
|
||||
title = "Destroy rooms";
|
||||
instructions = "Select the rooms to destroy";
|
||||
|
||||
{ name = "FORM_TYPE", type = "hidden", value = "http://prosody.im/protocol/muc#destroy" };
|
||||
{ name = "rooms", type = "list-multi", required = true, label = "Rooms to destroy:"};
|
||||
};
|
||||
|
||||
local destroy_rooms_handler = adhoc_initial(destroy_rooms_layout, function()
|
||||
return { rooms = array.collect(keys(rooms)):sort() };
|
||||
end, function(fields, errors)
|
||||
if errors then
|
||||
local errmsg = {};
|
||||
for name, err in pairs(errors) do
|
||||
errmsg[#errmsg + 1] = name .. ": " .. err;
|
||||
end
|
||||
return { status = "completed", error = { message = t_concat(errmsg, "\n") } };
|
||||
end
|
||||
for _, room in ipairs(fields.rooms) do
|
||||
rooms[room]:destroy();
|
||||
rooms[room] = nil;
|
||||
end
|
||||
return { status = "completed", info = "The following rooms were destroyed:\n"..t_concat(fields.rooms, "\n") };
|
||||
end);
|
||||
local destroy_rooms_desc = adhoc_new("Destroy Rooms", "http://prosody.im/protocol/muc#destroy", destroy_rooms_handler, "admin");
|
||||
|
||||
module:provides("adhoc", destroy_rooms_desc);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -27,28 +27,16 @@ local muc_domain = nil; --module:get_host();
|
|||
local default_history_length, max_history_length = 20, math.huge;
|
||||
|
||||
------------
|
||||
local function filter_xmlns_from_array(array, filters)
|
||||
local count = 0;
|
||||
for i=#array,1,-1 do
|
||||
local attr = array[i].attr;
|
||||
if filters[attr and attr.xmlns] then
|
||||
t_remove(array, i);
|
||||
count = count + 1;
|
||||
end
|
||||
end
|
||||
return count;
|
||||
end
|
||||
local function filter_xmlns_from_stanza(stanza, filters)
|
||||
if filters then
|
||||
if filter_xmlns_from_array(stanza.tags, filters) ~= 0 then
|
||||
return stanza, filter_xmlns_from_array(stanza, filters);
|
||||
end
|
||||
end
|
||||
return stanza, 0;
|
||||
end
|
||||
local presence_filters = {["http://jabber.org/protocol/muc"]=true;["http://jabber.org/protocol/muc#user"]=true};
|
||||
local function presence_filter(tag)
|
||||
if presence_filters[tag.attr.xmlns] then
|
||||
return nil;
|
||||
end
|
||||
return tag;
|
||||
end
|
||||
|
||||
local function get_filtered_presence(stanza)
|
||||
return filter_xmlns_from_stanza(st.clone(stanza):reset(), presence_filters);
|
||||
return st.clone(stanza):maptags(presence_filter);
|
||||
end
|
||||
local kickable_error_conditions = {
|
||||
["gone"] = true;
|
||||
|
@ -72,17 +60,6 @@ local function is_kickable_error(stanza)
|
|||
local cond = get_error_condition(stanza);
|
||||
return kickable_error_conditions[cond] and cond;
|
||||
end
|
||||
local function getUsingPath(stanza, path, getText)
|
||||
local tag = stanza;
|
||||
for _, name in ipairs(path) do
|
||||
if type(tag) ~= 'table' then return; end
|
||||
tag = tag:child_with_name(name);
|
||||
end
|
||||
if tag and getText then tag = table.concat(tag); end
|
||||
return tag;
|
||||
end
|
||||
local function getTag(stanza, path) return getUsingPath(stanza, path); end
|
||||
local function getText(stanza, path) return getUsingPath(stanza, path, true); end
|
||||
-----------
|
||||
|
||||
local room_mt = {};
|
||||
|
@ -98,8 +75,8 @@ function room_mt:get_default_role(affiliation)
|
|||
elseif affiliation == "member" then
|
||||
return "participant";
|
||||
elseif not affiliation then
|
||||
if not self:is_members_only() then
|
||||
return self:is_moderated() and "visitor" or "participant";
|
||||
if not self:get_members_only() then
|
||||
return self:get_moderated() and "visitor" or "participant";
|
||||
end
|
||||
end
|
||||
end
|
||||
|
@ -130,18 +107,21 @@ function room_mt:broadcast_message(stanza, historic)
|
|||
end
|
||||
stanza.attr.to = to;
|
||||
if historic then -- add to history
|
||||
local history = self._data['history'];
|
||||
if not history then history = {}; self._data['history'] = history; end
|
||||
stanza = st.clone(stanza);
|
||||
stanza.attr.to = "";
|
||||
local stamp = datetime.datetime();
|
||||
stanza:tag("delay", {xmlns = "urn:xmpp:delay", from = muc_domain, stamp = stamp}):up(); -- XEP-0203
|
||||
stanza:tag("x", {xmlns = "jabber:x:delay", from = muc_domain, stamp = datetime.legacy()}):up(); -- XEP-0091 (deprecated)
|
||||
local entry = { stanza = stanza, stamp = stamp };
|
||||
t_insert(history, entry);
|
||||
while #history > (self._data.history_length or default_history_length) do t_remove(history, 1) end
|
||||
return self:save_to_history(stanza)
|
||||
end
|
||||
end
|
||||
function room_mt:save_to_history(stanza)
|
||||
local history = self._data['history'];
|
||||
if not history then history = {}; self._data['history'] = history; end
|
||||
stanza = st.clone(stanza);
|
||||
stanza.attr.to = "";
|
||||
local stamp = datetime.datetime();
|
||||
stanza:tag("delay", {xmlns = "urn:xmpp:delay", from = muc_domain, stamp = stamp}):up(); -- XEP-0203
|
||||
stanza:tag("x", {xmlns = "jabber:x:delay", from = muc_domain, stamp = datetime.legacy()}):up(); -- XEP-0091 (deprecated)
|
||||
local entry = { stanza = stanza, stamp = stamp };
|
||||
t_insert(history, entry);
|
||||
while #history > (self._data.history_length or default_history_length) do t_remove(history, 1) end
|
||||
end
|
||||
function room_mt:broadcast_except_nick(stanza, nick)
|
||||
for rnick, occupant in pairs(self._occupants) do
|
||||
if rnick ~= nick then
|
||||
|
@ -170,10 +150,10 @@ function room_mt:send_history(to, stanza)
|
|||
if history then
|
||||
local x_tag = stanza and stanza:get_child("x", "http://jabber.org/protocol/muc");
|
||||
local history_tag = x_tag and x_tag:get_child("history", "http://jabber.org/protocol/muc");
|
||||
|
||||
|
||||
local maxchars = history_tag and tonumber(history_tag.attr.maxchars);
|
||||
if maxchars then maxchars = math.floor(maxchars); end
|
||||
|
||||
|
||||
local maxstanzas = math.floor(history_tag and tonumber(history_tag.attr.maxstanzas) or #history);
|
||||
if not history_tag then maxstanzas = 20; end
|
||||
|
||||
|
@ -186,7 +166,7 @@ function room_mt:send_history(to, stanza)
|
|||
|
||||
local n = 0;
|
||||
local charcount = 0;
|
||||
|
||||
|
||||
for i=#history,1,-1 do
|
||||
local entry = history[i];
|
||||
if maxchars then
|
||||
|
@ -207,6 +187,8 @@ function room_mt:send_history(to, stanza)
|
|||
self:_route_stanza(msg);
|
||||
end
|
||||
end
|
||||
end
|
||||
function room_mt:send_subject(to)
|
||||
if self._data['subject'] then
|
||||
self:_route_stanza(st.message({type='groupchat', from=self._data['subject_from'] or self.jid, to=to}):tag("subject"):text(self._data['subject']));
|
||||
end
|
||||
|
@ -218,10 +200,10 @@ function room_mt:get_disco_info(stanza)
|
|||
:tag("identity", {category="conference", type="text", name=self:get_name()}):up()
|
||||
:tag("feature", {var="http://jabber.org/protocol/muc"}):up()
|
||||
:tag("feature", {var=self:get_password() and "muc_passwordprotected" or "muc_unsecured"}):up()
|
||||
:tag("feature", {var=self:is_moderated() and "muc_moderated" or "muc_unmoderated"}):up()
|
||||
:tag("feature", {var=self:is_members_only() and "muc_membersonly" or "muc_open"}):up()
|
||||
:tag("feature", {var=self:is_persistent() and "muc_persistent" or "muc_temporary"}):up()
|
||||
:tag("feature", {var=self:is_hidden() and "muc_hidden" or "muc_public"}):up()
|
||||
:tag("feature", {var=self:get_moderated() and "muc_moderated" or "muc_unmoderated"}):up()
|
||||
:tag("feature", {var=self:get_members_only() and "muc_membersonly" or "muc_open"}):up()
|
||||
:tag("feature", {var=self:get_persistent() and "muc_persistent" or "muc_temporary"}):up()
|
||||
:tag("feature", {var=self:get_hidden() and "muc_hidden" or "muc_public"}):up()
|
||||
:tag("feature", {var=self._data.whois ~= "anyone" and "muc_semianonymous" or "muc_nonanonymous"}):up()
|
||||
:add_child(dataform.new({
|
||||
{ name = "FORM_TYPE", type = "hidden", value = "http://jabber.org/protocol/muc#roominfo" },
|
||||
|
@ -238,7 +220,6 @@ function room_mt:get_disco_items(stanza)
|
|||
return reply;
|
||||
end
|
||||
function room_mt:set_subject(current_nick, subject)
|
||||
-- TODO check nick's authority
|
||||
if subject == "" then subject = nil; end
|
||||
self._data['subject'] = subject;
|
||||
self._data['subject_from'] = current_nick;
|
||||
|
@ -296,7 +277,7 @@ function room_mt:set_moderated(moderated)
|
|||
if self.save then self:save(true); end
|
||||
end
|
||||
end
|
||||
function room_mt:is_moderated()
|
||||
function room_mt:get_moderated()
|
||||
return self._data.moderated;
|
||||
end
|
||||
function room_mt:set_members_only(members_only)
|
||||
|
@ -306,7 +287,7 @@ function room_mt:set_members_only(members_only)
|
|||
if self.save then self:save(true); end
|
||||
end
|
||||
end
|
||||
function room_mt:is_members_only()
|
||||
function room_mt:get_members_only()
|
||||
return self._data.members_only;
|
||||
end
|
||||
function room_mt:set_persistent(persistent)
|
||||
|
@ -316,7 +297,7 @@ function room_mt:set_persistent(persistent)
|
|||
if self.save then self:save(true); end
|
||||
end
|
||||
end
|
||||
function room_mt:is_persistent()
|
||||
function room_mt:get_persistent()
|
||||
return self._data.persistent;
|
||||
end
|
||||
function room_mt:set_hidden(hidden)
|
||||
|
@ -326,9 +307,15 @@ function room_mt:set_hidden(hidden)
|
|||
if self.save then self:save(true); end
|
||||
end
|
||||
end
|
||||
function room_mt:is_hidden()
|
||||
function room_mt:get_hidden()
|
||||
return self._data.hidden;
|
||||
end
|
||||
function room_mt:get_public()
|
||||
return not self:get_hidden();
|
||||
end
|
||||
function room_mt:set_public(public)
|
||||
return self:set_hidden(not public);
|
||||
end
|
||||
function room_mt:set_changesubject(changesubject)
|
||||
changesubject = changesubject and true or nil;
|
||||
if self._data.changesubject ~= changesubject then
|
||||
|
@ -351,12 +338,25 @@ function room_mt:set_historylength(length)
|
|||
end
|
||||
|
||||
|
||||
local valid_whois = { moderators = true, anyone = true };
|
||||
|
||||
function room_mt:set_whois(whois)
|
||||
if valid_whois[whois] and self._data.whois ~= whois then
|
||||
self._data.whois = whois;
|
||||
if self.save then self:save(true); end
|
||||
end
|
||||
end
|
||||
|
||||
function room_mt:get_whois()
|
||||
return self._data.whois;
|
||||
end
|
||||
|
||||
local function construct_stanza_id(room, stanza)
|
||||
local from_jid, to_nick = stanza.attr.from, stanza.attr.to;
|
||||
local from_nick = room._jid_nick[from_jid];
|
||||
local occupant = room._occupants[to_nick];
|
||||
local to_jid = occupant.jid;
|
||||
|
||||
|
||||
return from_nick, to_jid, base64.encode(to_jid.."\0"..stanza.attr.id.."\0"..md5(from_jid));
|
||||
end
|
||||
local function deconstruct_stanza_id(room, stanza)
|
||||
|
@ -485,6 +485,12 @@ function room_mt:handle_to_occupant(origin, stanza) -- PM, vCards, etc
|
|||
log("debug", "%s joining as %s", from, to);
|
||||
if not next(self._affiliations) then -- new room, no owners
|
||||
self._affiliations[jid_bare(from)] = "owner";
|
||||
if self.locked and not stanza:get_child("x", "http://jabber.org/protocol/muc") then
|
||||
self.locked = nil; -- Older groupchat protocol doesn't lock
|
||||
end
|
||||
elseif self.locked then -- Deny entry
|
||||
origin.send(st.error_reply(stanza, "cancel", "item-not-found"));
|
||||
return;
|
||||
end
|
||||
local affiliation = self:get_affiliation(from);
|
||||
local role = self:get_default_role(affiliation)
|
||||
|
@ -506,9 +512,13 @@ function room_mt:handle_to_occupant(origin, stanza) -- PM, vCards, etc
|
|||
if self._data.whois == 'anyone' then
|
||||
pr:tag("status", {code='100'}):up();
|
||||
end
|
||||
if self.locked then
|
||||
pr:tag("status", {code='201'}):up();
|
||||
end
|
||||
pr.attr.to = from;
|
||||
self:_route_stanza(pr);
|
||||
self:send_history(from, stanza);
|
||||
self:send_subject(from);
|
||||
elseif not affiliation then -- registration required for entering members-only room
|
||||
local reply = st.error_reply(stanza, "auth", "registration-required"):up();
|
||||
reply.tags[1].attr.code = "407";
|
||||
|
@ -560,6 +570,7 @@ function room_mt:handle_to_occupant(origin, stanza) -- PM, vCards, etc
|
|||
end
|
||||
stanza.attr.from, stanza.attr.to, stanza.attr.id = from, to, id;
|
||||
else -- message
|
||||
stanza:tag("x", { xmlns = "http://jabber.org/protocol/muc#user" }):up();
|
||||
stanza.attr.from = current_nick;
|
||||
for jid in pairs(o_data.sessions) do
|
||||
stanza.attr.to = jid;
|
||||
|
@ -575,11 +586,11 @@ end
|
|||
|
||||
function room_mt:send_form(origin, stanza)
|
||||
origin.send(st.reply(stanza):query("http://jabber.org/protocol/muc#owner")
|
||||
:add_child(self:get_form_layout():form())
|
||||
:add_child(self:get_form_layout(stanza.attr.from):form())
|
||||
);
|
||||
end
|
||||
|
||||
function room_mt:get_form_layout()
|
||||
function room_mt:get_form_layout(actor)
|
||||
local form = dataform.new({
|
||||
title = "Configuration for "..self.jid,
|
||||
instructions = "Complete and submit this form to configure the room.",
|
||||
|
@ -604,13 +615,13 @@ function room_mt:get_form_layout()
|
|||
name = 'muc#roomconfig_persistentroom',
|
||||
type = 'boolean',
|
||||
label = 'Make Room Persistent?',
|
||||
value = self:is_persistent()
|
||||
value = self:get_persistent()
|
||||
},
|
||||
{
|
||||
name = 'muc#roomconfig_publicroom',
|
||||
type = 'boolean',
|
||||
label = 'Make Room Publicly Searchable?',
|
||||
value = not self:is_hidden()
|
||||
value = not self:get_hidden()
|
||||
},
|
||||
{
|
||||
name = 'muc#roomconfig_changesubject',
|
||||
|
@ -637,13 +648,13 @@ function room_mt:get_form_layout()
|
|||
name = 'muc#roomconfig_moderatedroom',
|
||||
type = 'boolean',
|
||||
label = 'Make Room Moderated?',
|
||||
value = self:is_moderated()
|
||||
value = self:get_moderated()
|
||||
},
|
||||
{
|
||||
name = 'muc#roomconfig_membersonly',
|
||||
type = 'boolean',
|
||||
label = 'Make Room Members-Only?',
|
||||
value = self:is_members_only()
|
||||
value = self:get_members_only()
|
||||
},
|
||||
{
|
||||
name = 'muc#roomconfig_historylength',
|
||||
|
@ -652,14 +663,9 @@ function room_mt:get_form_layout()
|
|||
value = tostring(self:get_historylength())
|
||||
}
|
||||
});
|
||||
return module:fire_event("muc-config-form", { room = self, form = form }) or form;
|
||||
return module:fire_event("muc-config-form", { room = self, actor = actor, form = form }) or form;
|
||||
end
|
||||
|
||||
local valid_whois = {
|
||||
moderators = true,
|
||||
anyone = true,
|
||||
}
|
||||
|
||||
function room_mt:process_form(origin, stanza)
|
||||
local query = stanza.tags[1];
|
||||
local form;
|
||||
|
@ -668,84 +674,50 @@ function room_mt:process_form(origin, stanza)
|
|||
if form.attr.type == "cancel" then origin.send(st.reply(stanza)); return; end
|
||||
if form.attr.type ~= "submit" then origin.send(st.error_reply(stanza, "cancel", "bad-request", "Not a submitted form")); return; end
|
||||
|
||||
local fields = self:get_form_layout():data(form);
|
||||
local fields = self:get_form_layout(stanza.attr.from):data(form);
|
||||
if fields.FORM_TYPE ~= "http://jabber.org/protocol/muc#roomconfig" then origin.send(st.error_reply(stanza, "cancel", "bad-request", "Form is not of type room configuration")); return; end
|
||||
|
||||
local dirty = false
|
||||
|
||||
local event = { room = self, fields = fields, changed = dirty };
|
||||
local changed = {};
|
||||
|
||||
local function handle_option(name, field, allowed)
|
||||
local new = fields[field];
|
||||
if new == nil then return; end
|
||||
if allowed and not allowed[new] then return; end
|
||||
if new == self["get_"..name](self) then return; end
|
||||
changed[name] = true;
|
||||
self["set_"..name](self, new);
|
||||
end
|
||||
|
||||
local event = { room = self, fields = fields, changed = changed, stanza = stanza, origin = origin, update_option = handle_option };
|
||||
module:fire_event("muc-config-submitted", event);
|
||||
dirty = event.changed or dirty;
|
||||
|
||||
local name = fields['muc#roomconfig_roomname'];
|
||||
if name ~= self:get_name() then
|
||||
self:set_name(name);
|
||||
end
|
||||
|
||||
local description = fields['muc#roomconfig_roomdesc'];
|
||||
if description ~= self:get_description() then
|
||||
self:set_description(description);
|
||||
end
|
||||
|
||||
local persistent = fields['muc#roomconfig_persistentroom'];
|
||||
dirty = dirty or (self:is_persistent() ~= persistent)
|
||||
module:log("debug", "persistent=%s", tostring(persistent));
|
||||
|
||||
local moderated = fields['muc#roomconfig_moderatedroom'];
|
||||
dirty = dirty or (self:is_moderated() ~= moderated)
|
||||
module:log("debug", "moderated=%s", tostring(moderated));
|
||||
|
||||
local membersonly = fields['muc#roomconfig_membersonly'];
|
||||
dirty = dirty or (self:is_members_only() ~= membersonly)
|
||||
module:log("debug", "membersonly=%s", tostring(membersonly));
|
||||
|
||||
local public = fields['muc#roomconfig_publicroom'];
|
||||
dirty = dirty or (self:is_hidden() ~= (not public and true or nil))
|
||||
|
||||
local changesubject = fields['muc#roomconfig_changesubject'];
|
||||
dirty = dirty or (self:get_changesubject() ~= (not changesubject and true or nil))
|
||||
module:log('debug', 'changesubject=%s', changesubject and "true" or "false")
|
||||
|
||||
local historylength = tonumber(fields['muc#roomconfig_historylength']);
|
||||
dirty = dirty or (historylength and (self:get_historylength() ~= historylength));
|
||||
module:log('debug', 'historylength=%s', historylength)
|
||||
|
||||
|
||||
local whois = fields['muc#roomconfig_whois'];
|
||||
if not valid_whois[whois] then
|
||||
origin.send(st.error_reply(stanza, 'cancel', 'bad-request', "Invalid value for 'whois'"));
|
||||
return;
|
||||
end
|
||||
local whois_changed = self._data.whois ~= whois
|
||||
self._data.whois = whois
|
||||
module:log('debug', 'whois=%s', whois)
|
||||
|
||||
local password = fields['muc#roomconfig_roomsecret'];
|
||||
if self:get_password() ~= password then
|
||||
self:set_password(password);
|
||||
end
|
||||
self:set_moderated(moderated);
|
||||
self:set_members_only(membersonly);
|
||||
self:set_persistent(persistent);
|
||||
self:set_hidden(not public);
|
||||
self:set_changesubject(changesubject);
|
||||
self:set_historylength(historylength);
|
||||
handle_option("name", "muc#roomconfig_roomname");
|
||||
handle_option("description", "muc#roomconfig_roomdesc");
|
||||
handle_option("persistent", "muc#roomconfig_persistentroom");
|
||||
handle_option("moderated", "muc#roomconfig_moderatedroom");
|
||||
handle_option("members_only", "muc#roomconfig_membersonly");
|
||||
handle_option("public", "muc#roomconfig_publicroom");
|
||||
handle_option("changesubject", "muc#roomconfig_changesubject");
|
||||
handle_option("historylength", "muc#roomconfig_historylength");
|
||||
handle_option("whois", "muc#roomconfig_whois", valid_whois);
|
||||
handle_option("password", "muc#roomconfig_roomsecret");
|
||||
|
||||
if self.save then self:save(true); end
|
||||
if self.locked then
|
||||
module:fire_event("muc-room-unlocked", { room = self });
|
||||
self.locked = nil;
|
||||
end
|
||||
origin.send(st.reply(stanza));
|
||||
|
||||
if dirty or whois_changed then
|
||||
if next(changed) then
|
||||
local msg = st.message({type='groupchat', from=self.jid})
|
||||
:tag('x', {xmlns='http://jabber.org/protocol/muc#user'}):up()
|
||||
|
||||
if dirty then
|
||||
msg.tags[1]:tag('status', {code = '104'}):up();
|
||||
end
|
||||
if whois_changed then
|
||||
local code = (whois == 'moderators') and "173" or "172";
|
||||
:tag('status', {code = '104'}):up();
|
||||
if changed.whois then
|
||||
local code = (self:get_whois() == 'moderators') and "173" or "172";
|
||||
msg.tags[1]:tag('status', {code = code}):up();
|
||||
end
|
||||
|
||||
self:broadcast_message(msg, false)
|
||||
end
|
||||
end
|
||||
|
@ -881,7 +853,7 @@ function room_mt:handle_to_room(origin, stanza) -- presence changes and groupcha
|
|||
origin.send(st.error_reply(stanza, "cancel", "service-unavailable"));
|
||||
end
|
||||
elseif stanza.name == "message" and type == "groupchat" then
|
||||
local from, to = stanza.attr.from, stanza.attr.to;
|
||||
local from = stanza.attr.from;
|
||||
local current_nick = self._jid_nick[from];
|
||||
local occupant = self._occupants[current_nick];
|
||||
if not occupant then -- not in room
|
||||
|
@ -891,11 +863,11 @@ function room_mt:handle_to_room(origin, stanza) -- presence changes and groupcha
|
|||
else
|
||||
local from = stanza.attr.from;
|
||||
stanza.attr.from = current_nick;
|
||||
local subject = getText(stanza, {"subject"});
|
||||
local subject = stanza:get_child_text("subject");
|
||||
if subject then
|
||||
if occupant.role == "moderator" or
|
||||
( self._data.changesubject and occupant.role == "participant" ) then -- and participant
|
||||
self:set_subject(current_nick, subject); -- TODO use broadcast_message_stanza
|
||||
self:set_subject(current_nick, subject);
|
||||
else
|
||||
stanza.attr.from = from;
|
||||
origin.send(st.error_reply(stanza, "auth", "forbidden"));
|
||||
|
@ -943,7 +915,7 @@ function room_mt:handle_to_room(origin, stanza) -- presence changes and groupcha
|
|||
:tag('body') -- Add a plain message for clients which don't support invites
|
||||
:text(_from..' invited you to the room '.._to..(_reason and (' ('.._reason..')') or ""))
|
||||
:up();
|
||||
if self:is_members_only() and not self:get_affiliation(_invitee) then
|
||||
if self:get_members_only() and not self:get_affiliation(_invitee) then
|
||||
log("debug", "%s invited %s into members only room %s, granting membership", _from, _invitee, _to);
|
||||
self:set_affiliation(_from, _invitee, "member", nil, "Invited by " .. self._jid_nick[_from])
|
||||
end
|
||||
|
|
|
@ -1,97 +0,0 @@
|
|||
|
||||
-- Basic SQL driver
|
||||
-- This driver stores data as simple key-values
|
||||
|
||||
local ser = require "util.serialization".serialize;
|
||||
local envload = require "util.envload".envload;
|
||||
local deser = function(data)
|
||||
module:log("debug", "deser: %s", tostring(data));
|
||||
if not data then return nil; end
|
||||
local f = envload("return "..data, nil, {});
|
||||
if not f then return nil; end
|
||||
local s, d = pcall(f);
|
||||
if not s then return nil; end
|
||||
return d;
|
||||
end;
|
||||
|
||||
local driver = {};
|
||||
driver.__index = driver;
|
||||
|
||||
driver.item_table = "item";
|
||||
driver.list_table = "list";
|
||||
|
||||
function driver:prepare(sql)
|
||||
module:log("debug", "query: %s", sql);
|
||||
local err;
|
||||
if not self.sqlcache then self.sqlcache = {}; end
|
||||
local r = self.sqlcache[sql];
|
||||
if r then return r; end
|
||||
r, err = self.connection:prepare(sql);
|
||||
if not r then error("Unable to prepare SQL statement: "..err); end
|
||||
self.sqlcache[sql] = r;
|
||||
return r;
|
||||
end
|
||||
|
||||
function driver:load(username, host, datastore)
|
||||
local select = self:prepare("select data from "..self.item_table.." where username=? and host=? and datastore=?");
|
||||
select:execute(username, host, datastore);
|
||||
local row = select:fetch();
|
||||
return row and deser(row[1]) or nil;
|
||||
end
|
||||
|
||||
function driver:store(username, host, datastore, data)
|
||||
if not data or next(data) == nil then
|
||||
local delete = self:prepare("delete from "..self.item_table.." where username=? and host=? and datastore=?");
|
||||
delete:execute(username, host, datastore);
|
||||
return true;
|
||||
else
|
||||
local d = self:load(username, host, datastore);
|
||||
if d then -- update
|
||||
local update = self:prepare("update "..self.item_table.." set data=? where username=? and host=? and datastore=?");
|
||||
return update:execute(ser(data), username, host, datastore);
|
||||
else -- insert
|
||||
local insert = self:prepare("insert into "..self.item_table.." values (?, ?, ?, ?)");
|
||||
return insert:execute(username, host, datastore, ser(data));
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
function driver:list_append(username, host, datastore, data)
|
||||
if not data then return; end
|
||||
local insert = self:prepare("insert into "..self.list_table.." values (?, ?, ?, ?)");
|
||||
return insert:execute(username, host, datastore, ser(data));
|
||||
end
|
||||
|
||||
function driver:list_store(username, host, datastore, data)
|
||||
-- remove existing data
|
||||
local delete = self:prepare("delete from "..self.list_table.." where username=? and host=? and datastore=?");
|
||||
delete:execute(username, host, datastore);
|
||||
if data and next(data) ~= nil then
|
||||
-- add data
|
||||
for _, d in ipairs(data) do
|
||||
self:list_append(username, host, datastore, ser(d));
|
||||
end
|
||||
end
|
||||
return true;
|
||||
end
|
||||
|
||||
function driver:list_load(username, host, datastore)
|
||||
local select = self:prepare("select data from "..self.list_table.." where username=? and host=? and datastore=?");
|
||||
select:execute(username, host, datastore);
|
||||
local r = {};
|
||||
for row in select:rows() do
|
||||
table.insert(r, deser(row[1]));
|
||||
end
|
||||
return r;
|
||||
end
|
||||
|
||||
local _M = {};
|
||||
function _M.new(dbtype, dbname, ...)
|
||||
local d = {};
|
||||
setmetatable(d, driver);
|
||||
local dbh = get_database(dbtype, dbname, ...);
|
||||
--d:set_connection(dbh);
|
||||
d.connection = dbh;
|
||||
return d;
|
||||
end
|
||||
return _M;
|
6
prosody
6
prosody
|
@ -265,12 +265,6 @@ function init_global_state()
|
|||
prosody.events.fire_event("server-stopping", {reason = reason});
|
||||
server.setquitting(true);
|
||||
end
|
||||
|
||||
-- Load SSL settings from config, and create a ctx table
|
||||
local certmanager = require "core.certmanager";
|
||||
local global_ssl_ctx = certmanager.create_context("*", "server");
|
||||
prosody.global_ssl_ctx = global_ssl_ctx;
|
||||
|
||||
end
|
||||
|
||||
function read_version()
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
-- website at http://prosody.im/doc/configure
|
||||
--
|
||||
-- Tip: You can check that the syntax of this file is correct
|
||||
-- when you have finished by running: luac -p prosody.cfg.lua
|
||||
-- when you have finished by running: prosodyctl check config
|
||||
-- If there are any errors, it will let you know what and where
|
||||
-- they are, otherwise it will keep quiet.
|
||||
--
|
||||
|
@ -24,7 +24,7 @@ admins = { }
|
|||
|
||||
-- Enable use of libevent for better performance under high load
|
||||
-- For more information see: http://prosody.im/doc/libevent
|
||||
--use_libevent = true;
|
||||
--use_libevent = true
|
||||
|
||||
-- This is the list of modules Prosody will load on startup.
|
||||
-- It looks for mod_modulename.lua in the plugins folder, so make sure that exists too.
|
||||
|
@ -63,14 +63,13 @@ modules_enabled = {
|
|||
--"http_files"; -- Serve static files from a directory over HTTP
|
||||
|
||||
-- Other specific functionality
|
||||
--"posix"; -- POSIX functionality, sends server to background, enables syslog, etc.
|
||||
--"groups"; -- Shared roster support
|
||||
--"announce"; -- Send announcement to all online users
|
||||
--"welcome"; -- Welcome users who register accounts
|
||||
--"watchregistrations"; -- Alert admins of registrations
|
||||
--"motd"; -- Send a message to users when they log in
|
||||
--"legacyauth"; -- Legacy authentication. Only used by some old clients and bots.
|
||||
};
|
||||
}
|
||||
|
||||
-- These modules are auto-loaded, but should you want
|
||||
-- to disable them then uncomment them here:
|
||||
|
@ -78,11 +77,12 @@ modules_disabled = {
|
|||
-- "offline"; -- Store offline messages
|
||||
-- "c2s"; -- Handle client connections
|
||||
-- "s2s"; -- Handle server-to-server connections
|
||||
};
|
||||
-- "posix"; -- POSIX functionality, sends server to background, enables syslog, etc.
|
||||
}
|
||||
|
||||
-- Disable account creation by default, for security
|
||||
-- For more information see http://prosody.im/doc/creating_accounts
|
||||
allow_registration = false;
|
||||
allow_registration = false
|
||||
|
||||
-- These are the SSL/TLS-related settings. If you don't want
|
||||
-- to use SSL/TLS, you may comment or remove this
|
||||
|
@ -94,7 +94,7 @@ ssl = {
|
|||
-- Force clients to use encrypted connections? This option will
|
||||
-- prevent clients from authenticating unless they are using encryption.
|
||||
|
||||
c2s_require_encryption = false
|
||||
c2s_require_encryption = true
|
||||
|
||||
-- Force certificate authentication for server-to-server connections?
|
||||
-- This provides ideal security, but requires servers you communicate
|
||||
|
|
336
prosodyctl
336
prosodyctl
|
@ -274,11 +274,12 @@ local commands = {};
|
|||
local command = arg[1];
|
||||
|
||||
function commands.adduser(arg)
|
||||
local jid_split = require "util.jid".split;
|
||||
if not arg[1] or arg[1] == "--help" then
|
||||
show_usage([[adduser JID]], [[Create the specified user account in Prosody]]);
|
||||
return 1;
|
||||
end
|
||||
local user, host = arg[1]:match("([^@]+)@(.+)");
|
||||
local user, host = jid_split(arg[1]);
|
||||
if not user and host then
|
||||
show_message [[Failed to understand JID, please supply the JID you want to create]]
|
||||
show_usage [[adduser user@host]]
|
||||
|
@ -313,11 +314,12 @@ function commands.adduser(arg)
|
|||
end
|
||||
|
||||
function commands.passwd(arg)
|
||||
local jid_split = require "util.jid".split;
|
||||
if not arg[1] or arg[1] == "--help" then
|
||||
show_usage([[passwd JID]], [[Set the password for the specified user account in Prosody]]);
|
||||
return 1;
|
||||
end
|
||||
local user, host = arg[1]:match("([^@]+)@(.+)");
|
||||
local user, host = jid_split(arg[1]);
|
||||
if not user and host then
|
||||
show_message [[Failed to understand JID, please supply the JID you want to set the password for]]
|
||||
show_usage [[passwd user@host]]
|
||||
|
@ -352,11 +354,12 @@ function commands.passwd(arg)
|
|||
end
|
||||
|
||||
function commands.deluser(arg)
|
||||
local jid_split = require "util.jid".split;
|
||||
if not arg[1] or arg[1] == "--help" then
|
||||
show_usage([[deluser JID]], [[Permanently remove the specified user account from Prosody]]);
|
||||
return 1;
|
||||
end
|
||||
local user, host = arg[1]:match("([^@]+)@(.+)");
|
||||
local user, host = jid_split(arg[1]);
|
||||
if not user and host then
|
||||
show_message [[Failed to understand JID, please supply the JID you want to set the password for]]
|
||||
show_usage [[passwd user@host]]
|
||||
|
@ -781,6 +784,333 @@ function commands.cert(arg)
|
|||
show_usage("cert config|request|generate|key", "Helpers for generating X.509 certificates and keys.")
|
||||
end
|
||||
|
||||
function commands.check(arg)
|
||||
if arg[1] == "--help" then
|
||||
show_usage([[check]], [[Perform basic checks on your Prosody installation]]);
|
||||
return 1;
|
||||
end
|
||||
local what = table.remove(arg, 1);
|
||||
local array, set = require "util.array", require "util.set";
|
||||
local it = require "util.iterators";
|
||||
local ok = true;
|
||||
if not what or what == "config" then
|
||||
print("Checking config...");
|
||||
local known_global_options = set.new({
|
||||
"pidfile", "log", "plugin_paths", "prosody_user", "prosody_group", "daemonize",
|
||||
"umask", "prosodyctl_timeout", "use_ipv6", "use_libevent", "network_settings"
|
||||
});
|
||||
local config = config.getconfig();
|
||||
-- Check that we have any global options (caused by putting a host at the top)
|
||||
if it.count(it.filter("log", pairs(config["*"]))) == 0 then
|
||||
ok = false;
|
||||
print("");
|
||||
print(" No global options defined. Perhaps you have put a host definition at the top")
|
||||
print(" of the config file? They should be at the bottom, see http://prosody.im/doc/configure#overview");
|
||||
end
|
||||
-- Check for global options under hosts
|
||||
local global_options = set.new(it.to_array(it.keys(config["*"])));
|
||||
for host, options in it.filter("*", pairs(config)) do
|
||||
local host_options = set.new(it.to_array(it.keys(options)));
|
||||
local misplaced_options = set.intersection(host_options, known_global_options);
|
||||
for name in pairs(options) do
|
||||
if name:match("^interfaces?")
|
||||
or name:match("_ports?$") or name:match("_interfaces?$")
|
||||
or name:match("_ssl$") then
|
||||
misplaced_options:add(name);
|
||||
end
|
||||
end
|
||||
if not misplaced_options:empty() then
|
||||
ok = false;
|
||||
print("");
|
||||
local n = it.count(misplaced_options);
|
||||
print(" You have "..n.." option"..(n>1 and "s " or " ").."set under "..host.." that should be");
|
||||
print(" in the global section of the config file, above any VirtualHost or Component definitions,")
|
||||
print(" see http://prosody.im/doc/configure#overview for more information.")
|
||||
print("");
|
||||
print(" You need to move the following option"..(n>1 and "s" or "")..": "..table.concat(it.to_array(misplaced_options), ", "));
|
||||
end
|
||||
local subdomain = host:match("^[^.]+");
|
||||
if not(host_options:contains("component_module")) and (subdomain == "jabber" or subdomain == "xmpp"
|
||||
or subdomain == "chat" or subdomain == "im") then
|
||||
print("");
|
||||
print(" Suggestion: If "..host.. " is a new host with no real users yet, consider renaming it now to");
|
||||
print(" "..host:gsub("^[^.]+%.", "")..". You can use SRV records to redirect XMPP clients and servers to "..host..".");
|
||||
print(" For more information see: http://prosody.im/doc/dns");
|
||||
end
|
||||
end
|
||||
|
||||
print("Done.\n");
|
||||
end
|
||||
if not what or what == "dns" then
|
||||
local dns = require "net.dns";
|
||||
local idna = require "util.encodings".idna;
|
||||
local ip = require "util.ip";
|
||||
local c2s_ports = set.new(config.get("*", "c2s_ports") or {5222});
|
||||
local s2s_ports = set.new(config.get("*", "s2s_ports") or {5269});
|
||||
|
||||
local c2s_srv_required, s2s_srv_required;
|
||||
if not c2s_ports:contains(5222) then
|
||||
c2s_srv_required = true;
|
||||
end
|
||||
if not s2s_ports:contains(5269) then
|
||||
s2s_srv_required = true;
|
||||
end
|
||||
|
||||
local problem_hosts = set.new();
|
||||
|
||||
local external_addresses, internal_addresses = set.new(), set.new();
|
||||
|
||||
local fqdn = socket.dns.tohostname(socket.dns.gethostname());
|
||||
if fqdn then
|
||||
local res = dns.lookup(idna.to_ascii(fqdn), "A");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
external_addresses:add(record.a);
|
||||
end
|
||||
end
|
||||
local res = dns.lookup(idna.to_ascii(fqdn), "AAAA");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
external_addresses:add(record.aaaa);
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
local local_addresses = require"util.net".local_addresses() or {};
|
||||
|
||||
for addr in it.values(local_addresses) do
|
||||
if not ip.new_ip(addr).private then
|
||||
external_addresses:add(addr);
|
||||
else
|
||||
internal_addresses:add(addr);
|
||||
end
|
||||
end
|
||||
|
||||
if external_addresses:empty() then
|
||||
print("");
|
||||
print(" Failed to determine the external addresses of this server. Checks may be inaccurate.");
|
||||
c2s_srv_required, s2s_srv_required = true, true;
|
||||
end
|
||||
|
||||
local v6_supported = not not socket.tcp6;
|
||||
|
||||
for host, host_options in it.filter("*", pairs(config.getconfig())) do
|
||||
local all_targets_ok, some_targets_ok = true, false;
|
||||
|
||||
local is_component = not not host_options.component_module;
|
||||
print("Checking DNS for "..(is_component and "component" or "host").." "..host.."...");
|
||||
local target_hosts = set.new();
|
||||
if not is_component then
|
||||
local res = dns.lookup("_xmpp-client._tcp."..idna.to_ascii(host)..".", "SRV");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
target_hosts:add(record.srv.target);
|
||||
if not c2s_ports:contains(record.srv.port) then
|
||||
print(" SRV target "..record.srv.target.." contains unknown client port: "..record.srv.port);
|
||||
end
|
||||
end
|
||||
else
|
||||
if c2s_srv_required then
|
||||
print(" No _xmpp-client SRV record found for "..host..", but it looks like you need one.");
|
||||
all_targst_ok = false;
|
||||
else
|
||||
target_hosts:add(host);
|
||||
end
|
||||
end
|
||||
end
|
||||
local res = dns.lookup("_xmpp-server._tcp."..idna.to_ascii(host)..".", "SRV");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
target_hosts:add(record.srv.target);
|
||||
if not s2s_ports:contains(record.srv.port) then
|
||||
print(" SRV target "..record.srv.target.." contains unknown server port: "..record.srv.port);
|
||||
end
|
||||
end
|
||||
else
|
||||
if s2s_srv_required then
|
||||
print(" No _xmpp-server SRV record found for "..host..", but it looks like you need one.");
|
||||
all_targets_ok = false;
|
||||
else
|
||||
target_hosts:add(host);
|
||||
end
|
||||
end
|
||||
if target_hosts:empty() then
|
||||
target_hosts:add(host);
|
||||
end
|
||||
|
||||
if target_hosts:contains("localhost") then
|
||||
print(" Target 'localhost' cannot be accessed from other servers");
|
||||
target_hosts:remove("localhost");
|
||||
end
|
||||
|
||||
local modules = set.new(it.to_array(it.values(host_options.modules_enabled)))
|
||||
+ set.new(it.to_array(it.values(config.get("*", "modules_enabled"))))
|
||||
+ set.new({ config.get(host, "component_module") });
|
||||
|
||||
if modules:contains("proxy65") then
|
||||
local proxy65_target = config.get(host, "proxy65_address") or host;
|
||||
local A, AAAA = dns.lookup(idna.to_ascii(proxy65_target), "A"), dns.lookup(idna.to_ascii(proxy65_target), "AAAA");
|
||||
local prob = {};
|
||||
if not A then
|
||||
table.insert(prob, "A");
|
||||
end
|
||||
if v6_supported and not AAAA then
|
||||
table.insert(prob, "AAAA");
|
||||
end
|
||||
if #prob > 0 then
|
||||
print(" File transfer proxy "..proxy65_target.." has no "..table.concat(prob, "/").." record. Create one or set 'proxy65_address' to the correct host/IP.");
|
||||
end
|
||||
end
|
||||
|
||||
for host in target_hosts do
|
||||
local host_ok_v4, host_ok_v6;
|
||||
local res = dns.lookup(idna.to_ascii(host), "A");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
if external_addresses:contains(record.a) then
|
||||
some_targets_ok = true;
|
||||
host_ok_v4 = true;
|
||||
elseif internal_addresses:contains(record.a) then
|
||||
host_ok_v4 = true;
|
||||
some_targets_ok = true;
|
||||
print(" "..host.." A record points to internal address, external connections might fail");
|
||||
else
|
||||
print(" "..host.." A record points to unknown address "..record.a);
|
||||
all_targets_ok = false;
|
||||
end
|
||||
end
|
||||
end
|
||||
local res = dns.lookup(idna.to_ascii(host), "AAAA");
|
||||
if res then
|
||||
for _, record in ipairs(res) do
|
||||
if external_addresses:contains(record.aaaa) then
|
||||
some_targets_ok = true;
|
||||
host_ok_v6 = true;
|
||||
elseif internal_addresses:contains(record.aaaa) then
|
||||
host_ok_v6 = true;
|
||||
some_targets_ok = true;
|
||||
print(" "..host.." AAAA record points to internal address, external connections might fail");
|
||||
else
|
||||
print(" "..host.." AAAA record points to unknown address "..record.aaaa);
|
||||
all_targets_ok = false;
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
local bad_protos = {}
|
||||
if not host_ok_v4 then
|
||||
table.insert(bad_protos, "IPv4");
|
||||
end
|
||||
if not host_ok_v6 then
|
||||
table.insert(bad_protos, "IPv6");
|
||||
end
|
||||
if #bad_protos > 0 then
|
||||
print(" Host "..host.." does not seem to resolve to this server ("..table.concat(bad_protos, "/")..")");
|
||||
end
|
||||
if host_ok_v6 and not v6_supported then
|
||||
print(" Host "..host.." has AAAA records, but your version of LuaSocket does not support IPv6.");
|
||||
print(" Please see http://prosody.im/doc/ipv6 for more information.");
|
||||
end
|
||||
end
|
||||
if not all_targets_ok then
|
||||
print(" "..(some_targets_ok and "Only some" or "No").." targets for "..host.." appear to resolve to this server.");
|
||||
if is_component then
|
||||
print(" DNS records are necessary if you want users on other servers to access this component.");
|
||||
end
|
||||
problem_hosts:add(host);
|
||||
end
|
||||
print("");
|
||||
end
|
||||
if not problem_hosts:empty() then
|
||||
print("");
|
||||
print("For more information about DNS configuration please see http://prosody.im/doc/dns");
|
||||
print("");
|
||||
ok = false;
|
||||
end
|
||||
end
|
||||
if not what or what == "certs" then
|
||||
local cert_ok;
|
||||
print"Checking certificates..."
|
||||
local x509_verify_identity = require"util.x509".verify_identity;
|
||||
local ssl = dependencies.softreq"ssl";
|
||||
-- local datetime_parse = require"util.datetime".parse_x509;
|
||||
local load_cert = ssl and ssl.x509 and ssl.x509.load;
|
||||
-- or ssl.cert_from_pem
|
||||
if not ssl then
|
||||
print("LuaSec not available, can't perform certificate checks")
|
||||
if what == "certs" then cert_ok = false end
|
||||
elseif not load_cert then
|
||||
print("This version of LuaSec (" .. ssl._VERSION .. ") does not support certificate checking");
|
||||
cert_ok = false
|
||||
else
|
||||
for host in pairs(hosts) do
|
||||
if host ~= "*" then -- Should check global certs too.
|
||||
print("Checking certificate for "..host);
|
||||
-- First, let's find out what certificate this host uses.
|
||||
local ssl_config = config.rawget(host, "ssl");
|
||||
if not ssl_config then
|
||||
local base_host = host:match("%.(.*)");
|
||||
ssl_config = config.get(base_host, "ssl");
|
||||
end
|
||||
if not ssl_config then
|
||||
print(" No 'ssl' option defined for "..host)
|
||||
cert_ok = false
|
||||
elseif not ssl_config.certificate then
|
||||
print(" No 'certificate' set in ssl option for "..host)
|
||||
cert_ok = false
|
||||
elseif not ssl_config.key then
|
||||
print(" No 'key' set in ssl option for "..host)
|
||||
cert_ok = false
|
||||
else
|
||||
local key, err = io.open(ssl_config.key); -- Permissions check only
|
||||
if not key then
|
||||
print(" Could not open "..ssl_config.key..": "..err);
|
||||
cert_ok = false
|
||||
else
|
||||
key:close();
|
||||
end
|
||||
local cert_fh, err = io.open(ssl_config.certificate); -- Load the file.
|
||||
if not cert_fh then
|
||||
print(" Could not open "..ssl_config.certificate..": "..err);
|
||||
cert_ok = false
|
||||
else
|
||||
print(" Certificate: "..ssl_config.certificate)
|
||||
local cert = load_cert(cert_fh:read"*a"); cert_fh = cert_fh:close();
|
||||
if not cert:validat(os.time()) then
|
||||
print(" Certificate has expired.")
|
||||
cert_ok = false
|
||||
end
|
||||
if config.get(host, "component_module") == nil
|
||||
and not x509_verify_identity(host, "_xmpp-client", cert) then
|
||||
print(" Not vaild for client connections to "..host..".")
|
||||
cert_ok = false
|
||||
end
|
||||
if (not (config.get(name, "anonymous_login")
|
||||
or config.get(name, "authentication") == "anonymous"))
|
||||
and not x509_verify_identity(host, "_xmpp-client", cert) then
|
||||
print(" Not vaild for server-to-server connections to "..host..".")
|
||||
cert_ok = false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
if cert_ok == false then
|
||||
print("")
|
||||
print("For more information about certificates please see http://prosody.im/doc/certificates");
|
||||
ok = false
|
||||
end
|
||||
end
|
||||
print("")
|
||||
end
|
||||
if not ok then
|
||||
print("Problems found, see above.");
|
||||
else
|
||||
print("All checks passed, congratulations!");
|
||||
end
|
||||
return ok and 0 or 2;
|
||||
end
|
||||
|
||||
---------------------
|
||||
|
||||
if command and command:match("^mod_") then -- Is a command in a module
|
||||
|
|
|
@ -18,7 +18,7 @@ function test_value(value, returns)
|
|||
assert(module:get_option_number("opt") == returns.number, "number doesn't match");
|
||||
assert(module:get_option_string("opt") == returns.string, "string doesn't match");
|
||||
assert(module:get_option_boolean("opt") == returns.boolean, "boolean doesn't match");
|
||||
|
||||
|
||||
if type(returns.array) == "table" then
|
||||
local target_array, returned_array = returns.array, module:get_option_array("opt");
|
||||
assert(#target_array == #returned_array, "array length doesn't match");
|
||||
|
@ -28,7 +28,7 @@ function test_value(value, returns)
|
|||
else
|
||||
assert(module:get_option_array("opt") == returns.array, "array is returned (not nil)");
|
||||
end
|
||||
|
||||
|
||||
if type(returns.set) == "table" then
|
||||
local target_items, returned_items = set.new(returns.set), module:get_option_set("opt");
|
||||
assert(target_items == returned_items, "set doesn't match");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -12,15 +12,15 @@ function run_all_tests()
|
|||
package.loaded["net.connlisteners"] = { get = function () return {} end };
|
||||
dotest "util.jid"
|
||||
dotest "util.multitable"
|
||||
dotest "util.rfc3484"
|
||||
dotest "net.http"
|
||||
dotest "core.modulemanager"
|
||||
dotest "util.rfc6724"
|
||||
dotest "util.http"
|
||||
dotest "core.stanza_router"
|
||||
dotest "core.s2smanager"
|
||||
dotest "core.configmanager"
|
||||
dotest "util.ip"
|
||||
dotest "util.stanza"
|
||||
dotest "util.sasl.scram"
|
||||
|
||||
|
||||
dosingletest("test_sasl.lua", "latin1toutf8");
|
||||
end
|
||||
|
||||
|
@ -87,12 +87,12 @@ function dosingletest(testname, fname)
|
|||
print("WARNING: ", "Failed to initialise tests for "..testname, err);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
if type(tests[fname]) ~= "function" then
|
||||
error(testname.." has no test '"..fname.."'", 0);
|
||||
end
|
||||
|
||||
|
||||
|
||||
|
||||
local line_hook, line_info = new_line_coverage_monitor(testname);
|
||||
debug.sethook(line_hook, "l")
|
||||
local success, ret = pcall(tests[fname]);
|
||||
|
@ -134,17 +134,23 @@ function dotest(unitname)
|
|||
print("WARNING: ", "Failed to load module: "..unitname, err);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
local oldmodule, old_M = _fakeG.module, _fakeG._M;
|
||||
_fakeG.module = function () _M = _G end
|
||||
_fakeG.module = function () _M = unit end
|
||||
setfenv(chunk, unit);
|
||||
local success, err = pcall(chunk);
|
||||
local success, ret = pcall(chunk);
|
||||
_fakeG.module, _fakeG._M = oldmodule, old_M;
|
||||
if not success then
|
||||
print("WARNING: ", "Failed to initialise module: "..unitname, err);
|
||||
return;
|
||||
end
|
||||
|
||||
|
||||
if type(ret) == "table" then
|
||||
for k,v in pairs(ret) do
|
||||
unit[k] = v;
|
||||
end
|
||||
end
|
||||
|
||||
for name, f in pairs(unit) do
|
||||
local test = rawget(tests, name);
|
||||
if type(f) ~= "function" then
|
||||
|
@ -191,11 +197,11 @@ end
|
|||
function new_line_coverage_monitor(file)
|
||||
local lines_hit, funcs_hit = {}, {};
|
||||
local total_lines, covered_lines = 0, 0;
|
||||
|
||||
|
||||
for line in io.lines(file) do
|
||||
total_lines = total_lines + 1;
|
||||
end
|
||||
|
||||
|
||||
return function (event, line) -- Line hook
|
||||
if not lines_hit[line] then
|
||||
local info = debug.getinfo(2, "fSL")
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -9,27 +9,23 @@
|
|||
|
||||
|
||||
function get(get, config)
|
||||
config.set("example.com", "test", "testkey", 123);
|
||||
assert_equal(get("example.com", "test", "testkey"), 123, "Retrieving a set key");
|
||||
config.set("example.com", "testkey", 123);
|
||||
assert_equal(get("example.com", "testkey"), 123, "Retrieving a set key");
|
||||
|
||||
config.set("*", "testkey1", 321);
|
||||
assert_equal(get("*", "testkey1"), 321, "Retrieving a set global key");
|
||||
assert_equal(get("example.com", "testkey1"), 321, "Retrieving a set key of undefined host, of which only a globally set one exists");
|
||||
|
||||
config.set("example.com", ""); -- Creates example.com host in config
|
||||
assert_equal(get("example.com", "testkey1"), 321, "Retrieving a set key, of which only a globally set one exists");
|
||||
|
||||
config.set("*", "test", "testkey1", 321);
|
||||
assert_equal(get("*", "test", "testkey1"), 321, "Retrieving a set global key");
|
||||
assert_equal(get("example.com", "test", "testkey1"), 321, "Retrieving a set key of undefined host, of which only a globally set one exists");
|
||||
|
||||
config.set("example.com", "test", ""); -- Creates example.com host in config
|
||||
assert_equal(get("example.com", "test", "testkey1"), 321, "Retrieving a set key, of which only a globally set one exists");
|
||||
|
||||
assert_equal(get(), nil, "No parameters to get()");
|
||||
assert_equal(get("undefined host"), nil, "Getting for undefined host");
|
||||
assert_equal(get("undefined host", "undefined section"), nil, "Getting for undefined host & section");
|
||||
assert_equal(get("undefined host", "undefined section", "undefined key"), nil, "Getting for undefined host & section & key");
|
||||
|
||||
assert_equal(get("example.com", "undefined section", "testkey"), nil, "Defined host, undefined section");
|
||||
assert_equal(get("undefined host", "undefined key"), nil, "Getting for undefined host & key");
|
||||
end
|
||||
|
||||
function set(set, u)
|
||||
assert_equal(set("*"), false, "Set with no section/key");
|
||||
assert_equal(set("*", "set_test"), false, "Set with no key");
|
||||
assert_equal(set("*"), false, "Set with no key");
|
||||
|
||||
assert_equal(set("*", "set_test", "testkey"), true, "Setting a nil global value");
|
||||
assert_equal(set("*", "set_test", "testkey", 123), true, "Setting a global value");
|
||||
|
|
|
@ -1,48 +0,0 @@
|
|||
-- 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 config = require "core.configmanager";
|
||||
local helpers = require "util.helpers";
|
||||
local set = require "util.set";
|
||||
|
||||
function load_modules_for_host(load_modules_for_host, mm)
|
||||
local test_num = 0;
|
||||
local function test_load(global_modules_enabled, global_modules_disabled, host_modules_enabled, host_modules_disabled, expected_modules)
|
||||
test_num = test_num + 1;
|
||||
-- Prepare
|
||||
hosts = { ["example.com"] = {} };
|
||||
config.set("*", "core", "modules_enabled", global_modules_enabled);
|
||||
config.set("*", "core", "modules_disabled", global_modules_disabled);
|
||||
config.set("example.com", "core", "modules_enabled", host_modules_enabled);
|
||||
config.set("example.com", "core", "modules_disabled", host_modules_disabled);
|
||||
|
||||
expected_modules = set.new(expected_modules);
|
||||
expected_modules:add_list(helpers.get_upvalue(load_modules_for_host, "autoload_modules"));
|
||||
|
||||
local loaded_modules = set.new();
|
||||
function mm.load(host, module)
|
||||
assert_equal(host, "example.com", test_num..": Host isn't example.com but "..tostring(host));
|
||||
assert_equal(expected_modules:contains(module), true, test_num..": Loading unexpected module '"..tostring(module).."'");
|
||||
loaded_modules:add(module);
|
||||
end
|
||||
load_modules_for_host("example.com");
|
||||
assert_equal((expected_modules - loaded_modules):empty(), true, test_num..": Not all modules loaded: "..tostring(expected_modules - loaded_modules));
|
||||
end
|
||||
|
||||
test_load({ "one", "two", "three" }, nil, nil, nil, { "one", "two", "three" });
|
||||
test_load({ "one", "two", "three" }, {}, nil, nil, { "one", "two", "three" });
|
||||
test_load({ "one", "two", "three" }, { "two" }, nil, nil, { "one", "three" });
|
||||
test_load({ "one", "two", "three" }, { "three" }, nil, nil, { "one", "two" });
|
||||
test_load({ "one", "two", "three" }, nil, nil, { "three" }, { "one", "two" });
|
||||
test_load({ "one", "two", "three" }, nil, { "three" }, { "three" }, { "one", "two", "three" });
|
||||
|
||||
test_load({ "one", "two" }, nil, { "three" }, nil, { "one", "two", "three" });
|
||||
test_load({ "one", "two", "three" }, nil, { "three" }, nil, { "one", "two", "three" });
|
||||
test_load({ "one", "two", "three" }, { "three" }, { "three" }, nil, { "one", "two", "three" });
|
||||
test_load({ "one", "two" }, { "three" }, { "three" }, nil, { "one", "two", "three" });
|
||||
end
|
|
@ -1,11 +1,14 @@
|
|||
-- 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.
|
||||
--
|
||||
|
||||
env = {
|
||||
prosody = { events = require "util.events".new() };
|
||||
};
|
||||
|
||||
function compare_srv_priorities(csp)
|
||||
local r1 = { priority = 10, weight = 0 }
|
||||
|
@ -13,7 +16,7 @@ function compare_srv_priorities(csp)
|
|||
local r3 = { priority = 1000, weight = 2 }
|
||||
local r4 = { priority = 1000, weight = 2 }
|
||||
local r5 = { priority = 1000, weight = 5 }
|
||||
|
||||
|
||||
assert_equal(csp(r1, r1), false);
|
||||
assert_equal(csp(r1, r2), true);
|
||||
assert_equal(csp(r1, r3), true);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -14,7 +14,7 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local s2sin_session = { from_host = "remotehost", to_host = "localhost", type = "s2sin", hosts = { ["remotehost"] = { authed = true } } }
|
||||
local local_host_session = { host = "localhost", type = "local", s2sout = { ["remotehost"] = s2sout_session } }
|
||||
local local_user_session = { username = "user", host = "localhost", resource = "resource", full_jid = "user@localhost/resource", type = "c2s" }
|
||||
|
||||
|
||||
_G.prosody.hosts["localhost"] = local_host_session;
|
||||
_G.prosody.full_sessions["user@localhost/resource"] = local_user_session;
|
||||
_G.prosody.bare_sessions["user@localhost"] = { sessions = { resource = local_user_session } };
|
||||
|
@ -23,15 +23,15 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_message_full_jid()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("message", { to = "user@localhost/resource", type = "chat" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_post_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of routed stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "routed stanza is not correct one: "..p_stanza:pretty_print());
|
||||
target_routed = true;
|
||||
end
|
||||
|
||||
|
||||
env.hosts = hosts;
|
||||
env.prosody = { hosts = hosts };
|
||||
setfenv(core_process_stanza, env);
|
||||
|
@ -42,9 +42,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_message_bare_jid()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("message", { to = "user@localhost", type = "chat" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_post_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of routed stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "routed stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -60,9 +60,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_message_no_to()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("message", { type = "chat" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_handled;
|
||||
|
||||
|
||||
function env.core_post_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -78,9 +78,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_message_to_remote_bare()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("message", { to = "user@remotehost", type = "chat" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_route_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -88,7 +88,7 @@ function core_process_stanza(core_process_stanza, u)
|
|||
end
|
||||
|
||||
function env.core_post_stanza(...) env.core_route_stanza(...); end
|
||||
|
||||
|
||||
env.hosts = hosts;
|
||||
setfenv(core_process_stanza, env);
|
||||
assert_equal(core_process_stanza(local_user_session, msg), nil, "core_process_stanza returned incorrect value");
|
||||
|
@ -98,9 +98,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_message_to_remote_server()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("message", { to = "remotehost", type = "chat" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_route_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -123,9 +123,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_iq_to_remote_server()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("iq", { to = "remotehost", type = "get", id = "id" }):tag("body"):text("Hello world");
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_route_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -145,9 +145,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_iq_error_to_local_user()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("iq", { to = "user@localhost/resource", from = "user@remotehost", type = "error", id = "id" }):tag("error", { type = 'cancel' }):tag("item-not-found", { xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' });
|
||||
|
||||
|
||||
local target_routed;
|
||||
|
||||
|
||||
function env.core_route_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, s2sin_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -167,9 +167,9 @@ function core_process_stanza(core_process_stanza, u)
|
|||
local function test_iq_to_local_bare()
|
||||
local env = testlib_new_env();
|
||||
local msg = stanza.stanza("iq", { to = "user@localhost", from = "user@localhost", type = "get", id = "id" }):tag("ping", { xmlns = "urn:xmpp:ping:0" });
|
||||
|
||||
|
||||
local target_handled;
|
||||
|
||||
|
||||
function env.core_post_stanza(p_origin, p_stanza)
|
||||
assert_equal(p_origin, local_user_session, "origin of handled stanza is not correct");
|
||||
assert_equal(p_stanza, msg, "handled stanza is not correct one: "..p_stanza:pretty_print());
|
||||
|
@ -209,11 +209,11 @@ function core_route_stanza(core_route_stanza)
|
|||
local msg2 = stanza.stanza("iq", { to = "user@localhost/foo", from = "user@localhost", type = "error" }):tag("ping", { xmlns = "urn:xmpp:ping:0" });
|
||||
--package.loaded["core.usermanager"] = { user_exists = function (user, host) print("RAR!") return true or user == "user" and host == "localhost" and true; end };
|
||||
local target_handled, target_replied;
|
||||
|
||||
|
||||
function env.core_post_stanza(p_origin, p_stanza)
|
||||
target_handled = true;
|
||||
end
|
||||
|
||||
|
||||
function local_user_session.send(data)
|
||||
--print("Replying with: ", tostring(data));
|
||||
--print(debug.traceback())
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -30,7 +30,7 @@ function latin1toutf8()
|
|||
local function assert_utf8(latin, utf8)
|
||||
assert_equal(_latin1toutf8(latin), utf8, "Incorrect UTF8 from Latin1: "..tostring(latin));
|
||||
end
|
||||
|
||||
|
||||
assert_utf8("", "")
|
||||
assert_utf8("test", "test")
|
||||
assert_utf8(nil, nil)
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
89
tests/test_util_ip.lua
Normal file
89
tests/test_util_ip.lua
Normal file
|
@ -0,0 +1,89 @@
|
|||
|
||||
function match(match, _M)
|
||||
local _ = _M.new_ip;
|
||||
local ip = _"10.20.30.40";
|
||||
assert_equal(match(ip, _"10.0.0.0", 8), true);
|
||||
assert_equal(match(ip, _"10.0.0.0", 16), false);
|
||||
assert_equal(match(ip, _"10.0.0.0", 24), false);
|
||||
assert_equal(match(ip, _"10.0.0.0", 32), false);
|
||||
|
||||
assert_equal(match(ip, _"10.20.0.0", 8), true);
|
||||
assert_equal(match(ip, _"10.20.0.0", 16), true);
|
||||
assert_equal(match(ip, _"10.20.0.0", 24), false);
|
||||
assert_equal(match(ip, _"10.20.0.0", 32), false);
|
||||
|
||||
assert_equal(match(ip, _"0.0.0.0", 32), false);
|
||||
assert_equal(match(ip, _"0.0.0.0", 0), true);
|
||||
assert_equal(match(ip, _"0.0.0.0"), false);
|
||||
|
||||
assert_equal(match(ip, _"10.0.0.0", 255), false, "excessive number of bits");
|
||||
assert_equal(match(ip, _"10.0.0.0", -8), true, "negative number of bits");
|
||||
assert_equal(match(ip, _"10.0.0.0", -32), true, "negative number of bits");
|
||||
assert_equal(match(ip, _"10.0.0.0", 0), true, "zero bits");
|
||||
assert_equal(match(ip, _"10.0.0.0"), false, "no specified number of bits (differing ip)");
|
||||
assert_equal(match(ip, _"10.20.30.40"), true, "no specified number of bits (same ip)");
|
||||
|
||||
assert_equal(match(_"127.0.0.1", _"127.0.0.1"), true, "simple ip");
|
||||
|
||||
assert_equal(match(_"8.8.8.8", _"8.8.0.0", 16), true);
|
||||
assert_equal(match(_"8.8.4.4", _"8.8.0.0", 16), true);
|
||||
end
|
||||
|
||||
function parse_cidr(parse_cidr, _M)
|
||||
local new_ip = _M.new_ip;
|
||||
|
||||
assert_equal(new_ip"0.0.0.0", new_ip"0.0.0.0")
|
||||
|
||||
local function assert_cidr(cidr, ip, bits)
|
||||
local parsed_ip, parsed_bits = parse_cidr(cidr);
|
||||
assert_equal(new_ip(ip), parsed_ip, cidr.." parsed ip is "..ip);
|
||||
assert_equal(bits, parsed_bits, cidr.." parsed bits is "..tostring(bits));
|
||||
end
|
||||
assert_cidr("0.0.0.0", "0.0.0.0", nil);
|
||||
assert_cidr("127.0.0.1", "127.0.0.1", nil);
|
||||
assert_cidr("127.0.0.1/0", "127.0.0.1", 0);
|
||||
assert_cidr("127.0.0.1/8", "127.0.0.1", 8);
|
||||
assert_cidr("127.0.0.1/32", "127.0.0.1", 32);
|
||||
assert_cidr("127.0.0.1/256", "127.0.0.1", 256);
|
||||
assert_cidr("::/48", "::", 48);
|
||||
end
|
||||
|
||||
function new_ip(new_ip)
|
||||
local v4, v6 = "IPv4", "IPv6";
|
||||
local function assert_proto(s, proto)
|
||||
local ip = new_ip(s);
|
||||
if proto then
|
||||
assert_equal(ip and ip.proto, proto, "protocol is correct for "..("%q"):format(s));
|
||||
else
|
||||
assert_equal(ip, nil, "address is invalid");
|
||||
end
|
||||
end
|
||||
assert_proto("127.0.0.1", v4);
|
||||
assert_proto("::1", v6);
|
||||
assert_proto("", nil);
|
||||
assert_proto("abc", nil);
|
||||
assert_proto(" ", nil);
|
||||
end
|
||||
|
||||
function commonPrefixLength(cpl, _M)
|
||||
local new_ip = _M.new_ip;
|
||||
local function assert_cpl6(a, b, len, v4)
|
||||
local ipa, ipb = new_ip(a), new_ip(b);
|
||||
if v4 then len = len+96; end
|
||||
assert_equal(cpl(ipa, ipb), len, "common prefix length of "..a.." and "..b.." is "..len);
|
||||
assert_equal(cpl(ipb, ipa), len, "common prefix length of "..b.." and "..a.." is "..len);
|
||||
end
|
||||
local function assert_cpl4(a, b, len)
|
||||
return assert_cpl6(a, b, len, "IPv4");
|
||||
end
|
||||
assert_cpl4("0.0.0.0", "0.0.0.0", 32);
|
||||
assert_cpl4("255.255.255.255", "0.0.0.0", 0);
|
||||
assert_cpl4("255.255.255.255", "255.255.0.0", 16);
|
||||
assert_cpl4("255.255.255.255", "255.255.255.255", 32);
|
||||
assert_cpl4("255.255.255.255", "255.255.255.255", 32);
|
||||
|
||||
assert_cpl6("::1", "::1", 128);
|
||||
assert_cpl6("abcd::1", "abcd::1", 128);
|
||||
assert_cpl6("abcd::abcd", "abcd::", 112);
|
||||
assert_cpl6("abcd::abcd", "abcd::abcd:abcd", 96);
|
||||
end
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -41,7 +41,7 @@ function get(get, multitable)
|
|||
end
|
||||
|
||||
mt = multitable.new();
|
||||
|
||||
|
||||
local trigger1, trigger2, trigger3 = {}, {}, {};
|
||||
local item1, item2, item3 = {}, {}, {};
|
||||
|
||||
|
@ -51,12 +51,12 @@ function get(get, multitable)
|
|||
mt:add(1, 2, 3, item1);
|
||||
|
||||
assert_has_all("Has item1 for 1, 2, 3", mt:get(1, 2, 3), item1);
|
||||
|
||||
|
||||
-- Doesn't support nil
|
||||
--[[ mt:add(nil, item1);
|
||||
mt:add(nil, item2);
|
||||
mt:add(nil, item3);
|
||||
|
||||
|
||||
assert_has_all("Has all items with (nil)", mt:get(nil), item1, item2, item3);
|
||||
]]
|
||||
end
|
||||
|
|
|
@ -1,51 +0,0 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2011 Florian Zeitz
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
||||
function source(source)
|
||||
local new_ip = require"util.ip".new_ip;
|
||||
assert_equal(source(new_ip("2001::1", "IPv6"), {new_ip("3ffe::1", "IPv6"), new_ip("fe80::1", "IPv6")}).addr, "3ffe::1", "prefer appropriate scope");
|
||||
assert_equal(source(new_ip("2001::1", "IPv6"), {new_ip("fe80::1", "IPv6"), new_ip("fec0::1", "IPv6")}).addr, "fec0::1", "prefer appropriate scope");
|
||||
assert_equal(source(new_ip("fec0::1", "IPv6"), {new_ip("fe80::1", "IPv6"), new_ip("2001::1", "IPv6")}).addr, "2001::1", "prefer appropriate scope");
|
||||
assert_equal(source(new_ip("ff05::1", "IPv6"), {new_ip("fe80::1", "IPv6"), new_ip("fec0::1", "IPv6"), new_ip("2001::1", "IPv6")}).addr, "fec0::1", "prefer appropriate scope");
|
||||
assert_equal(source(new_ip("2001::1", "IPv6"), {new_ip("2001::1", "IPv6"), new_ip("2002::1", "IPv6")}).addr, "2001::1", "prefer same address");
|
||||
assert_equal(source(new_ip("fec0::1", "IPv6"), {new_ip("fec0::2", "IPv6"), new_ip("2001::1", "IPv6")}).addr, "fec0::2", "prefer appropriate scope");
|
||||
assert_equal(source(new_ip("2001::1", "IPv6"), {new_ip("2001::2", "IPv6"), new_ip("3ffe::2", "IPv6")}).addr, "2001::2", "longest matching prefix");
|
||||
assert_equal(source(new_ip("2002:836b:2179::1", "IPv6"), {new_ip("2002:836b:2179::d5e3:7953:13eb:22e8", "IPv6"), new_ip("2001::2", "IPv6")}).addr, "2002:836b:2179::d5e3:7953:13eb:22e8", "prefer matching label");
|
||||
end
|
||||
|
||||
function destination(dest)
|
||||
local order;
|
||||
local new_ip = require"util.ip".new_ip;
|
||||
order = dest({new_ip("2001::1", "IPv6"), new_ip("131.107.65.121", "IPv4")}, {new_ip("2001::2", "IPv6"), new_ip("fe80::1", "IPv6"), new_ip("169.254.13.78", "IPv4")})
|
||||
assert_equal(order[1].addr, "2001::1", "prefer matching scope");
|
||||
assert_equal(order[2].addr, "131.107.65.121", "prefer matching scope")
|
||||
|
||||
order = dest({new_ip("2001::1", "IPv6"), new_ip("131.107.65.121", "IPv4")}, {new_ip("fe80::1", "IPv6"), new_ip("131.107.65.117", "IPv4")})
|
||||
assert_equal(order[1].addr, "131.107.65.121", "prefer matching scope")
|
||||
assert_equal(order[2].addr, "2001::1", "prefer matching scope")
|
||||
|
||||
order = dest({new_ip("2001::1", "IPv6"), new_ip("10.1.2.3", "IPv4")}, {new_ip("2001::2", "IPv6"), new_ip("fe80::1", "IPv6"), new_ip("10.1.2.4", "IPv4")})
|
||||
assert_equal(order[1].addr, "2001::1", "prefer higher precedence");
|
||||
assert_equal(order[2].addr, "10.1.2.3", "prefer higher precedence");
|
||||
|
||||
order = dest({new_ip("2001::1", "IPv6"), new_ip("fec0::1", "IPv6"), new_ip("fe80::1", "IPv6")}, {new_ip("2001::2", "IPv6"), new_ip("fec0::1", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "fe80::1", "prefer smaller scope");
|
||||
assert_equal(order[2].addr, "fec0::1", "prefer smaller scope");
|
||||
assert_equal(order[3].addr, "2001::1", "prefer smaller scope");
|
||||
|
||||
order = dest({new_ip("2001::1", "IPv6"), new_ip("3ffe::1", "IPv6")}, {new_ip("2001::2", "IPv6"), new_ip("3f44::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001::1", "longest matching prefix");
|
||||
assert_equal(order[2].addr, "3ffe::1", "longest matching prefix");
|
||||
|
||||
order = dest({new_ip("2002:836b:4179::1", "IPv6"), new_ip("2001::1", "IPv6")}, {new_ip("2002:836b:4179::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2002:836b:4179::1", "prefer matching label");
|
||||
assert_equal(order[2].addr, "2001::1", "prefer matching label");
|
||||
|
||||
order = dest({new_ip("2002:836b:4179::1", "IPv6"), new_ip("2001::1", "IPv6")}, {new_ip("2002:836b:4179::2", "IPv6"), new_ip("2001::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001::1", "prefer higher precedence");
|
||||
assert_equal(order[2].addr, "2002:836b:4179::1", "prefer higher precedence");
|
||||
end
|
97
tests/test_util_rfc6724.lua
Normal file
97
tests/test_util_rfc6724.lua
Normal file
|
@ -0,0 +1,97 @@
|
|||
-- Prosody IM
|
||||
-- Copyright (C) 2011-2013 Florian Zeitz
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
||||
function source(source)
|
||||
local new_ip = require"util.ip".new_ip;
|
||||
assert_equal(source(new_ip("2001:db8:1::1", "IPv6"),
|
||||
{new_ip("2001:db8:3::1", "IPv6"), new_ip("fe80::1", "IPv6")}).addr,
|
||||
"2001:db8:3::1",
|
||||
"prefer appropriate scope");
|
||||
assert_equal(source(new_ip("ff05::1", "IPv6"),
|
||||
{new_ip("2001:db8:3::1", "IPv6"), new_ip("fe80::1", "IPv6")}).addr,
|
||||
"2001:db8:3::1",
|
||||
"prefer appropriate scope");
|
||||
assert_equal(source(new_ip("2001:db8:1::1", "IPv6"),
|
||||
{new_ip("2001:db8:1::1", "IPv6"), new_ip("2001:db8:2::1", "IPv6")}).addr,
|
||||
"2001:db8:1::1",
|
||||
"prefer same address"); -- "2001:db8:1::1" should be marked "deprecated" here, we don't handle that right now
|
||||
assert_equal(source(new_ip("fe80::1", "IPv6"),
|
||||
{new_ip("fe80::2", "IPv6"), new_ip("2001:db8:1::1", "IPv6")}).addr,
|
||||
"fe80::2",
|
||||
"prefer appropriate scope"); -- "fe80::2" should be marked "deprecated" here, we don't handle that right now
|
||||
assert_equal(source(new_ip("2001:db8:1::1", "IPv6"),
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("2001:db8:3::2", "IPv6")}).addr,
|
||||
"2001:db8:1::2",
|
||||
"longest matching prefix");
|
||||
--[[ "2001:db8:1::2" should be a care-of address and "2001:db8:3::2" a home address, we can't handle this and would fail
|
||||
assert_equal(source(new_ip("2001:db8:1::1", "IPv6"),
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("2001:db8:3::2", "IPv6")}).addr,
|
||||
"2001:db8:3::2",
|
||||
"prefer home address");
|
||||
]]
|
||||
assert_equal(source(new_ip("2002:c633:6401::1", "IPv6"),
|
||||
{new_ip("2002:c633:6401::d5e3:7953:13eb:22e8", "IPv6"), new_ip("2001:db8:1::2", "IPv6")}).addr,
|
||||
"2002:c633:6401::d5e3:7953:13eb:22e8",
|
||||
"prefer matching label"); -- "2002:c633:6401::d5e3:7953:13eb:22e8" should be marked "temporary" here, we don't handle that right now
|
||||
assert_equal(source(new_ip("2001:db8:1::d5e3:0:0:1", "IPv6"),
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("2001:db8:1::d5e3:7953:13eb:22e8", "IPv6")}).addr,
|
||||
"2001:db8:1::d5e3:7953:13eb:22e8",
|
||||
"prefer temporary address") -- "2001:db8:1::2" should be marked "public" and "2001:db8:1::d5e3:7953:13eb:22e8" should be marked "temporary" here, we don't handle that right now
|
||||
end
|
||||
|
||||
function destination(dest)
|
||||
local order;
|
||||
local new_ip = require"util.ip".new_ip;
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("198.51.100.121", "IPv4")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("fe80::1", "IPv6"), new_ip("169.254.13.78", "IPv4")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "prefer matching scope");
|
||||
assert_equal(order[2].addr, "198.51.100.121", "prefer matching scope");
|
||||
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("198.51.100.121", "IPv4")},
|
||||
{new_ip("fe80::1", "IPv6"), new_ip("198.51.100.117", "IPv4")})
|
||||
assert_equal(order[1].addr, "198.51.100.121", "prefer matching scope");
|
||||
assert_equal(order[2].addr, "2001:db8:1::1", "prefer matching scope");
|
||||
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("10.1.2.3", "IPv4")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("fe80::1", "IPv6"), new_ip("10.1.2.4", "IPv4")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "prefer higher precedence");
|
||||
assert_equal(order[2].addr, "10.1.2.3", "prefer higher precedence");
|
||||
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("fe80::1", "IPv6")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "fe80::1", "prefer smaller scope");
|
||||
assert_equal(order[2].addr, "2001:db8:1::1", "prefer smaller scope");
|
||||
|
||||
--[[ "2001:db8:1::2" and "fe80::2" should be marked "care-of address", while "2001:db8:3::1" should be marked "home address", we can't currently handle this and would fail the test
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("fe80::1", "IPv6")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("2001:db8:3::1", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "prefer home address");
|
||||
assert_equal(order[2].addr, "fe80::1", "prefer home address");
|
||||
]]
|
||||
|
||||
--[[ "fe80::2" should be marked "deprecated", we can't currently handle this and would fail the test
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("fe80::1", "IPv6")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "avoid deprecated addresses");
|
||||
assert_equal(order[2].addr, "fe80::1", "avoid deprecated addresses");
|
||||
]]
|
||||
|
||||
order = dest({new_ip("2001:db8:1::1", "IPv6"), new_ip("2001:db8:3ffe::1", "IPv6")},
|
||||
{new_ip("2001:db8:1::2", "IPv6"), new_ip("2001:db8:3f44::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "longest matching prefix");
|
||||
assert_equal(order[2].addr, "2001:db8:3ffe::1", "longest matching prefix");
|
||||
|
||||
order = dest({new_ip("2002:c633:6401::1", "IPv6"), new_ip("2001:db8:1::1", "IPv6")},
|
||||
{new_ip("2002:c633:6401::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2002:c633:6401::1", "prefer matching label");
|
||||
assert_equal(order[2].addr, "2001:db8:1::1", "prefer matching label");
|
||||
|
||||
order = dest({new_ip("2002:c633:6401::1", "IPv6"), new_ip("2001:db8:1::1", "IPv6")},
|
||||
{new_ip("2002:c633:6401::2", "IPv6"), new_ip("2001:db8:1::2", "IPv6"), new_ip("fe80::2", "IPv6")})
|
||||
assert_equal(order[1].addr, "2001:db8:1::1", "prefer higher precedence");
|
||||
assert_equal(order[2].addr, "2002:c633:6401::1", "prefer higher precedence");
|
||||
end
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
@ -18,7 +18,7 @@ end
|
|||
|
||||
function deserialize(deserialize, st)
|
||||
local stanza = st.stanza("message", { a = "a" });
|
||||
|
||||
|
||||
local stanza2 = deserialize(st.preserialize(stanza));
|
||||
assert_is(stanza2 and stanza.name, "deserialize returns a stanza");
|
||||
assert_table(stanza2.attr, "Deserialized stanza has attributes");
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
-- 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.
|
||||
--
|
||||
|
|
|
@ -5,242 +5,242 @@ do
|
|||
|
||||
|
||||
local _parse_sql_actions = { [0] =
|
||||
0, 1, 0, 1, 1, 2, 0, 2, 2, 0, 9, 2, 0, 10, 2, 0, 11, 2, 0, 13,
|
||||
2, 1, 2, 2, 1, 6, 3, 0, 3, 4, 3, 0, 3, 5, 3, 0, 3, 7, 3, 0,
|
||||
0, 1, 0, 1, 1, 2, 0, 2, 2, 0, 9, 2, 0, 10, 2, 0, 11, 2, 0, 13,
|
||||
2, 1, 2, 2, 1, 6, 3, 0, 3, 4, 3, 0, 3, 5, 3, 0, 3, 7, 3, 0,
|
||||
3, 8, 3, 0, 3, 12, 4, 0, 2, 3, 7, 4, 0, 3, 8, 11
|
||||
};
|
||||
|
||||
local _parse_sql_trans_keys = { [0] =
|
||||
0, 0, 45, 45, 10, 10, 42, 42, 10, 42, 10, 47, 82, 82,
|
||||
69, 69, 65, 65, 84, 84, 69, 69, 32, 32, 68, 84, 65,
|
||||
65, 84, 84, 65, 65, 66, 66, 65, 65, 83, 83, 69, 69,
|
||||
9, 47, 9, 96, 45, 45, 10, 10, 42, 42, 10, 42, 10, 47,
|
||||
10, 96, 10, 96, 9, 47, 9, 59, 45, 45, 10, 10, 42,
|
||||
42, 10, 42, 10, 47, 65, 65, 66, 66, 76, 76, 69, 69,
|
||||
32, 32, 73, 96, 70, 70, 32, 32, 78, 78, 79, 79, 84, 84,
|
||||
32, 32, 69, 69, 88, 88, 73, 73, 83, 83, 84, 84, 83,
|
||||
83, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 40, 40,
|
||||
10, 10, 32, 41, 32, 32, 75, 96, 69, 69, 89, 89, 32, 32,
|
||||
96, 96, 10, 96, 10, 96, 10, 10, 82, 82, 73, 73, 77,
|
||||
77, 65, 65, 82, 82, 89, 89, 32, 32, 75, 75, 69, 69,
|
||||
89, 89, 32, 32, 78, 78, 73, 73, 81, 81, 85, 85, 69, 69,
|
||||
32, 32, 75, 75, 10, 96, 10, 96, 10, 10, 10, 59, 10,
|
||||
59, 82, 82, 79, 79, 80, 80, 32, 32, 84, 84, 65, 65,
|
||||
66, 66, 76, 76, 69, 69, 32, 32, 73, 73, 70, 70, 32, 32,
|
||||
69, 69, 88, 88, 73, 73, 83, 83, 84, 84, 83, 83, 32,
|
||||
32, 96, 96, 10, 96, 10, 96, 59, 59, 78, 78, 83, 83,
|
||||
69, 69, 82, 82, 84, 84, 32, 32, 73, 73, 78, 78, 84, 84,
|
||||
79, 79, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 40,
|
||||
86, 10, 41, 32, 32, 86, 86, 65, 65, 76, 76, 85, 85,
|
||||
69, 69, 83, 83, 32, 32, 40, 40, 39, 78, 10, 92, 10, 92,
|
||||
41, 44, 44, 59, 32, 78, 48, 57, 41, 57, 48, 57, 41,
|
||||
57, 85, 85, 76, 76, 76, 76, 34, 116, 79, 79, 67, 67,
|
||||
75, 75, 32, 32, 84, 84, 65, 65, 66, 66, 76, 76, 69, 69,
|
||||
83, 83, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 87,
|
||||
87, 82, 82, 73, 73, 84, 84, 69, 69, 69, 69, 84, 84,
|
||||
32, 32, 10, 59, 10, 59, 78, 83, 76, 76, 79, 79, 67, 67,
|
||||
75, 75, 32, 32, 84, 84, 65, 65, 66, 66, 76, 76, 69,
|
||||
0, 0, 45, 45, 10, 10, 42, 42, 10, 42, 10, 47, 82, 82,
|
||||
69, 69, 65, 65, 84, 84, 69, 69, 32, 32, 68, 84, 65,
|
||||
65, 84, 84, 65, 65, 66, 66, 65, 65, 83, 83, 69, 69,
|
||||
9, 47, 9, 96, 45, 45, 10, 10, 42, 42, 10, 42, 10, 47,
|
||||
10, 96, 10, 96, 9, 47, 9, 59, 45, 45, 10, 10, 42,
|
||||
42, 10, 42, 10, 47, 65, 65, 66, 66, 76, 76, 69, 69,
|
||||
32, 32, 73, 96, 70, 70, 32, 32, 78, 78, 79, 79, 84, 84,
|
||||
32, 32, 69, 69, 88, 88, 73, 73, 83, 83, 84, 84, 83,
|
||||
83, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 40, 40,
|
||||
10, 10, 32, 41, 32, 32, 75, 96, 69, 69, 89, 89, 32, 32,
|
||||
96, 96, 10, 96, 10, 96, 10, 10, 82, 82, 73, 73, 77,
|
||||
77, 65, 65, 82, 82, 89, 89, 32, 32, 75, 75, 69, 69,
|
||||
89, 89, 32, 32, 78, 78, 73, 73, 81, 81, 85, 85, 69, 69,
|
||||
32, 32, 75, 75, 10, 96, 10, 96, 10, 10, 10, 59, 10,
|
||||
59, 82, 82, 79, 79, 80, 80, 32, 32, 84, 84, 65, 65,
|
||||
66, 66, 76, 76, 69, 69, 32, 32, 73, 73, 70, 70, 32, 32,
|
||||
69, 69, 88, 88, 73, 73, 83, 83, 84, 84, 83, 83, 32,
|
||||
32, 96, 96, 10, 96, 10, 96, 59, 59, 78, 78, 83, 83,
|
||||
69, 69, 82, 82, 84, 84, 32, 32, 73, 73, 78, 78, 84, 84,
|
||||
79, 79, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 40,
|
||||
86, 10, 41, 32, 32, 86, 86, 65, 65, 76, 76, 85, 85,
|
||||
69, 69, 83, 83, 32, 32, 40, 40, 39, 78, 10, 92, 10, 92,
|
||||
41, 44, 44, 59, 32, 78, 48, 57, 41, 57, 48, 57, 41,
|
||||
57, 85, 85, 76, 76, 76, 76, 34, 116, 79, 79, 67, 67,
|
||||
75, 75, 32, 32, 84, 84, 65, 65, 66, 66, 76, 76, 69, 69,
|
||||
83, 83, 32, 32, 96, 96, 10, 96, 10, 96, 32, 32, 87,
|
||||
87, 82, 82, 73, 73, 84, 84, 69, 69, 69, 69, 84, 84,
|
||||
32, 32, 10, 59, 10, 59, 78, 83, 76, 76, 79, 79, 67, 67,
|
||||
75, 75, 32, 32, 84, 84, 65, 65, 66, 66, 76, 76, 69,
|
||||
69, 83, 83, 69, 69, 9, 85, 0
|
||||
};
|
||||
|
||||
local _parse_sql_key_spans = { [0] =
|
||||
0, 1, 1, 1, 33, 38, 1, 1, 1, 1, 1, 1, 17, 1, 1, 1, 1, 1, 1, 1,
|
||||
39, 88, 1, 1, 1, 33, 38, 87, 87, 39, 51, 1, 1, 1, 33, 38, 1, 1, 1, 1,
|
||||
1, 24, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1,
|
||||
1, 10, 1, 22, 1, 1, 1, 1, 87, 87, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 50, 50, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 47, 32, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 40, 83, 83, 4, 16, 47, 10, 17, 10, 17, 1, 1, 1, 83, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
0, 1, 1, 1, 33, 38, 1, 1, 1, 1, 1, 1, 17, 1, 1, 1, 1, 1, 1, 1,
|
||||
39, 88, 1, 1, 1, 33, 38, 87, 87, 39, 51, 1, 1, 1, 33, 38, 1, 1, 1, 1,
|
||||
1, 24, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1,
|
||||
1, 10, 1, 22, 1, 1, 1, 1, 87, 87, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 50, 50, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 47, 32, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 40, 83, 83, 4, 16, 47, 10, 17, 10, 17, 1, 1, 1, 83, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 87, 87, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 50, 50, 6, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 77
|
||||
};
|
||||
|
||||
local _parse_sql_index_offsets = { [0] =
|
||||
0, 0, 2, 4, 6, 40, 79, 81, 83, 85, 87, 89, 91, 109, 111, 113, 115, 117, 119, 121,
|
||||
123, 163, 252, 254, 256, 258, 292, 331, 419, 507, 547, 599, 601, 603, 605, 639, 678, 680, 682, 684,
|
||||
686, 688, 713, 715, 717, 719, 721, 723, 725, 727, 729, 731, 733, 735, 737, 739, 741, 829, 917, 919,
|
||||
921, 923, 934, 936, 959, 961, 963, 965, 967, 1055, 1143, 1145, 1147, 1149, 1151, 1153, 1155, 1157, 1159, 1161,
|
||||
1163, 1165, 1167, 1169, 1171, 1173, 1175, 1177, 1179, 1181, 1269, 1357, 1359, 1410, 1461, 1463, 1465, 1467, 1469, 1471,
|
||||
1473, 1475, 1477, 1479, 1481, 1483, 1485, 1487, 1489, 1491, 1493, 1495, 1497, 1499, 1501, 1503, 1591, 1679, 1681, 1683,
|
||||
1685, 1687, 1689, 1691, 1693, 1695, 1697, 1699, 1701, 1703, 1705, 1793, 1881, 1883, 1931, 1964, 1966, 1968, 1970, 1972,
|
||||
1974, 1976, 1978, 1980, 1982, 2023, 2107, 2191, 2196, 2213, 2261, 2272, 2290, 2301, 2319, 2321, 2323, 2325, 2409, 2411,
|
||||
2413, 2415, 2417, 2419, 2421, 2423, 2425, 2427, 2429, 2431, 2433, 2521, 2609, 2611, 2613, 2615, 2617, 2619, 2621, 2623,
|
||||
0, 0, 2, 4, 6, 40, 79, 81, 83, 85, 87, 89, 91, 109, 111, 113, 115, 117, 119, 121,
|
||||
123, 163, 252, 254, 256, 258, 292, 331, 419, 507, 547, 599, 601, 603, 605, 639, 678, 680, 682, 684,
|
||||
686, 688, 713, 715, 717, 719, 721, 723, 725, 727, 729, 731, 733, 735, 737, 739, 741, 829, 917, 919,
|
||||
921, 923, 934, 936, 959, 961, 963, 965, 967, 1055, 1143, 1145, 1147, 1149, 1151, 1153, 1155, 1157, 1159, 1161,
|
||||
1163, 1165, 1167, 1169, 1171, 1173, 1175, 1177, 1179, 1181, 1269, 1357, 1359, 1410, 1461, 1463, 1465, 1467, 1469, 1471,
|
||||
1473, 1475, 1477, 1479, 1481, 1483, 1485, 1487, 1489, 1491, 1493, 1495, 1497, 1499, 1501, 1503, 1591, 1679, 1681, 1683,
|
||||
1685, 1687, 1689, 1691, 1693, 1695, 1697, 1699, 1701, 1703, 1705, 1793, 1881, 1883, 1931, 1964, 1966, 1968, 1970, 1972,
|
||||
1974, 1976, 1978, 1980, 1982, 2023, 2107, 2191, 2196, 2213, 2261, 2272, 2290, 2301, 2319, 2321, 2323, 2325, 2409, 2411,
|
||||
2413, 2415, 2417, 2419, 2421, 2423, 2425, 2427, 2429, 2431, 2433, 2521, 2609, 2611, 2613, 2615, 2617, 2619, 2621, 2623,
|
||||
2625, 2627, 2678, 2729, 2736, 2738, 2740, 2742, 2744, 2746, 2748, 2750, 2752, 2754, 2756, 2758, 2760
|
||||
};
|
||||
|
||||
local _parse_sql_indicies = { [0] =
|
||||
0, 1, 2, 0, 3, 1, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 5, 3,
|
||||
4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 5, 3, 3, 3, 3, 6, 3, 7,
|
||||
1, 8, 1, 9, 1, 10, 1, 11, 1, 12, 1, 13, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 14, 1, 15, 1, 16, 1, 17, 1, 18, 1, 19, 1, 20,
|
||||
1, 21, 1, 22, 23, 22, 22, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 24,
|
||||
1, 25, 1, 22, 23, 22, 22, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 24,
|
||||
1, 25, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 26, 1, 27, 1, 23, 27, 28, 1, 29, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 30, 28, 29, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 30, 28, 28, 28, 28, 22, 28, 32, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 1, 31, 32,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 33, 31, 34, 35, 34, 34, 34, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 34, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 36, 1, 37, 1, 34, 35, 34, 34, 34, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 34, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 36, 1, 37, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 6, 1, 38,
|
||||
1, 35, 38, 39, 1, 40, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 41, 39, 40,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 41, 39, 39, 39, 39, 34, 39, 42, 1,
|
||||
43, 1, 44, 1, 45, 1, 46, 1, 47, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 48, 1, 49, 1, 50, 1, 51, 1, 52,
|
||||
1, 53, 1, 54, 1, 55, 1, 56, 1, 57, 1, 58, 1, 59, 1, 60, 1, 61, 1, 48,
|
||||
1, 63, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 1, 62, 65, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 66, 64, 67, 1, 68,
|
||||
1, 69, 1, 70, 1, 1, 1, 1, 1, 1, 1, 1, 71, 1, 72, 1, 73, 1, 1, 1,
|
||||
1, 74, 1, 1, 1, 1, 75, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 76, 1, 77,
|
||||
1, 78, 1, 79, 1, 80, 1, 82, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 1, 81, 82, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 83, 81, 69, 83, 84, 1, 85, 1, 86, 1, 87, 1, 88, 1, 89, 1, 90, 1, 91,
|
||||
1, 92, 1, 93, 1, 83, 1, 94, 1, 95, 1, 96, 1, 97, 1, 98, 1, 99, 1, 73,
|
||||
1, 101, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 1, 100, 103, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 104, 102, 105, 83, 106,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 107, 71, 108, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 107,
|
||||
71, 109, 1, 110, 1, 111, 1, 112, 1, 113, 1, 114, 1, 115, 1, 116, 1, 117, 1, 118,
|
||||
1, 119, 1, 120, 1, 121, 1, 122, 1, 123, 1, 124, 1, 125, 1, 126, 1, 127, 1, 128,
|
||||
1, 129, 1, 131, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 1, 130, 131, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 132, 130, 6,
|
||||
1, 133, 1, 134, 1, 135, 1, 136, 1, 137, 1, 138, 1, 139, 1, 140, 1, 141, 1, 142,
|
||||
1, 143, 1, 144, 1, 146, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 1, 145, 148, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 149,
|
||||
147, 150, 1, 151, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 152, 1, 153, 151, 151, 151, 151, 151, 151, 151, 151,
|
||||
151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151,
|
||||
151, 151, 154, 151, 155, 1, 152, 1, 156, 1, 157, 1, 158, 1, 159, 1, 160, 1, 161, 1,
|
||||
162, 1, 163, 1, 1, 1, 1, 1, 164, 1, 1, 165, 165, 165, 165, 165, 165, 165, 165, 165,
|
||||
165, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 166, 1, 168, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 169, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 170, 167, 172, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 173, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 174, 171, 175, 1, 1, 176, 1, 161, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 177, 1, 178, 1, 1, 1, 1, 1, 1,
|
||||
163, 1, 1, 1, 1, 1, 164, 1, 1, 165, 165, 165, 165, 165, 165, 165, 165, 165, 165, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 166,
|
||||
1, 179, 179, 179, 179, 179, 179, 179, 179, 179, 179, 1, 180, 1, 1, 181, 1, 182, 1, 179,
|
||||
179, 179, 179, 179, 179, 179, 179, 179, 179, 1, 183, 183, 183, 183, 183, 183, 183, 183, 183, 183,
|
||||
1, 180, 1, 1, 181, 1, 1, 1, 183, 183, 183, 183, 183, 183, 183, 183, 183, 183, 1, 184,
|
||||
1, 185, 1, 186, 1, 171, 1, 1, 171, 1, 171, 1, 1, 1, 1, 1, 1, 1, 1, 171,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 171, 1, 171, 1, 1, 171, 1, 1, 171, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 171, 1, 1, 1, 171, 1, 171, 1, 187, 1, 188, 1, 189, 1, 190, 1, 191, 1, 192,
|
||||
1, 193, 1, 194, 1, 195, 1, 196, 1, 197, 1, 198, 1, 200, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 1,
|
||||
199, 200, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 201, 199, 202, 1, 203, 1, 204, 1, 205, 1, 206, 1, 132,
|
||||
1, 207, 1, 208, 1, 209, 1, 210, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 211, 209, 2, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 211, 209, 212, 1, 1, 1, 1, 213, 1, 214, 1, 215, 1,
|
||||
216, 1, 217, 1, 218, 1, 219, 1, 220, 1, 221, 1, 222, 1, 223, 1, 132, 1, 127, 1,
|
||||
6, 2, 6, 6, 6, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 6, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 224, 1, 225, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 6, 1, 1, 1, 1, 1, 1, 1, 226, 227,
|
||||
0, 1, 2, 0, 3, 1, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 5, 3,
|
||||
4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,
|
||||
3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 5, 3, 3, 3, 3, 6, 3, 7,
|
||||
1, 8, 1, 9, 1, 10, 1, 11, 1, 12, 1, 13, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 14, 1, 15, 1, 16, 1, 17, 1, 18, 1, 19, 1, 20,
|
||||
1, 21, 1, 22, 23, 22, 22, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 24,
|
||||
1, 25, 1, 22, 23, 22, 22, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 22, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 24,
|
||||
1, 25, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 26, 1, 27, 1, 23, 27, 28, 1, 29, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 30, 28, 29, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28, 28,
|
||||
28, 28, 28, 28, 30, 28, 28, 28, 28, 22, 28, 32, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 1, 31, 32,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31, 31,
|
||||
31, 31, 31, 31, 31, 33, 31, 34, 35, 34, 34, 34, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 34, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 36, 1, 37, 1, 34, 35, 34, 34, 34, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 34, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 36, 1, 37, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 6, 1, 38,
|
||||
1, 35, 38, 39, 1, 40, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 41, 39, 40,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39,
|
||||
39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 39, 41, 39, 39, 39, 39, 34, 39, 42, 1,
|
||||
43, 1, 44, 1, 45, 1, 46, 1, 47, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 48, 1, 49, 1, 50, 1, 51, 1, 52,
|
||||
1, 53, 1, 54, 1, 55, 1, 56, 1, 57, 1, 58, 1, 59, 1, 60, 1, 61, 1, 48,
|
||||
1, 63, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62, 62,
|
||||
62, 62, 62, 62, 62, 62, 62, 1, 62, 65, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64,
|
||||
64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 66, 64, 67, 1, 68,
|
||||
1, 69, 1, 70, 1, 1, 1, 1, 1, 1, 1, 1, 71, 1, 72, 1, 73, 1, 1, 1,
|
||||
1, 74, 1, 1, 1, 1, 75, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 76, 1, 77,
|
||||
1, 78, 1, 79, 1, 80, 1, 82, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 1, 81, 82, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81, 81,
|
||||
81, 83, 81, 69, 83, 84, 1, 85, 1, 86, 1, 87, 1, 88, 1, 89, 1, 90, 1, 91,
|
||||
1, 92, 1, 93, 1, 83, 1, 94, 1, 95, 1, 96, 1, 97, 1, 98, 1, 99, 1, 73,
|
||||
1, 101, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100, 100,
|
||||
100, 100, 100, 100, 100, 100, 100, 1, 100, 103, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102,
|
||||
102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 102, 104, 102, 105, 83, 106,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 107, 71, 108, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71,
|
||||
71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 71, 107,
|
||||
71, 109, 1, 110, 1, 111, 1, 112, 1, 113, 1, 114, 1, 115, 1, 116, 1, 117, 1, 118,
|
||||
1, 119, 1, 120, 1, 121, 1, 122, 1, 123, 1, 124, 1, 125, 1, 126, 1, 127, 1, 128,
|
||||
1, 129, 1, 131, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 1, 130, 131, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130,
|
||||
130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 130, 132, 130, 6,
|
||||
1, 133, 1, 134, 1, 135, 1, 136, 1, 137, 1, 138, 1, 139, 1, 140, 1, 141, 1, 142,
|
||||
1, 143, 1, 144, 1, 146, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145,
|
||||
145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 145, 1, 145, 148, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147,
|
||||
147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 147, 149,
|
||||
147, 150, 1, 151, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 152, 1, 153, 151, 151, 151, 151, 151, 151, 151, 151,
|
||||
151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151, 151,
|
||||
151, 151, 154, 151, 155, 1, 152, 1, 156, 1, 157, 1, 158, 1, 159, 1, 160, 1, 161, 1,
|
||||
162, 1, 163, 1, 1, 1, 1, 1, 164, 1, 1, 165, 165, 165, 165, 165, 165, 165, 165, 165,
|
||||
165, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 166, 1, 168, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 169, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167, 167,
|
||||
167, 167, 167, 167, 167, 170, 167, 172, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 173, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171, 171,
|
||||
171, 171, 171, 171, 171, 171, 171, 171, 171, 174, 171, 175, 1, 1, 176, 1, 161, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 177, 1, 178, 1, 1, 1, 1, 1, 1,
|
||||
163, 1, 1, 1, 1, 1, 164, 1, 1, 165, 165, 165, 165, 165, 165, 165, 165, 165, 165, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 166,
|
||||
1, 179, 179, 179, 179, 179, 179, 179, 179, 179, 179, 1, 180, 1, 1, 181, 1, 182, 1, 179,
|
||||
179, 179, 179, 179, 179, 179, 179, 179, 179, 1, 183, 183, 183, 183, 183, 183, 183, 183, 183, 183,
|
||||
1, 180, 1, 1, 181, 1, 1, 1, 183, 183, 183, 183, 183, 183, 183, 183, 183, 183, 1, 184,
|
||||
1, 185, 1, 186, 1, 171, 1, 1, 171, 1, 171, 1, 1, 1, 1, 1, 1, 1, 1, 171,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 171, 1, 171, 1, 1, 171, 1, 1, 171, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 171, 1, 1, 1, 171, 1, 171, 1, 187, 1, 188, 1, 189, 1, 190, 1, 191, 1, 192,
|
||||
1, 193, 1, 194, 1, 195, 1, 196, 1, 197, 1, 198, 1, 200, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 1,
|
||||
199, 200, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199, 199,
|
||||
199, 199, 199, 199, 199, 199, 199, 201, 199, 202, 1, 203, 1, 204, 1, 205, 1, 206, 1, 132,
|
||||
1, 207, 1, 208, 1, 209, 1, 210, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 211, 209, 2, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209, 209,
|
||||
209, 209, 209, 209, 209, 209, 209, 211, 209, 212, 1, 1, 1, 1, 213, 1, 214, 1, 215, 1,
|
||||
216, 1, 217, 1, 218, 1, 219, 1, 220, 1, 221, 1, 222, 1, 223, 1, 132, 1, 127, 1,
|
||||
6, 2, 6, 6, 6, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 6, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 224, 1, 225, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 6, 1, 1, 1, 1, 1, 1, 1, 226, 227,
|
||||
1, 1, 1, 1, 228, 1, 1, 229, 1, 1, 1, 1, 1, 1, 230, 1, 231, 1, 0
|
||||
};
|
||||
|
||||
local _parse_sql_trans_targs = { [0] =
|
||||
2, 0, 196, 4, 4, 5, 196, 7, 8, 9, 10, 11, 12, 13, 36, 14, 15, 16, 17, 18,
|
||||
19, 20, 21, 21, 22, 24, 27, 23, 25, 25, 26, 28, 28, 29, 30, 30, 31, 33, 32, 34,
|
||||
34, 35, 37, 38, 39, 40, 41, 42, 56, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53,
|
||||
54, 55, 57, 57, 57, 57, 58, 59, 60, 61, 62, 92, 63, 64, 71, 82, 89, 65, 66, 67,
|
||||
68, 69, 69, 70, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 83, 84, 85, 86, 87, 88,
|
||||
90, 90, 90, 90, 91, 70, 92, 93, 196, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104, 105,
|
||||
106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 116, 117, 119, 120, 121, 122, 123, 124, 125,
|
||||
126, 127, 128, 129, 130, 131, 131, 131, 131, 132, 133, 134, 137, 134, 135, 136, 138, 139, 140, 141,
|
||||
142, 143, 144, 145, 150, 151, 154, 146, 146, 147, 157, 146, 146, 147, 157, 148, 149, 196, 144, 151,
|
||||
148, 149, 152, 153, 155, 156, 147, 159, 160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171,
|
||||
171, 172, 173, 174, 175, 176, 177, 179, 180, 181, 181, 182, 184, 195, 185, 186, 187, 188, 189, 190,
|
||||
2, 0, 196, 4, 4, 5, 196, 7, 8, 9, 10, 11, 12, 13, 36, 14, 15, 16, 17, 18,
|
||||
19, 20, 21, 21, 22, 24, 27, 23, 25, 25, 26, 28, 28, 29, 30, 30, 31, 33, 32, 34,
|
||||
34, 35, 37, 38, 39, 40, 41, 42, 56, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53,
|
||||
54, 55, 57, 57, 57, 57, 58, 59, 60, 61, 62, 92, 63, 64, 71, 82, 89, 65, 66, 67,
|
||||
68, 69, 69, 70, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 83, 84, 85, 86, 87, 88,
|
||||
90, 90, 90, 90, 91, 70, 92, 93, 196, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104, 105,
|
||||
106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 116, 117, 119, 120, 121, 122, 123, 124, 125,
|
||||
126, 127, 128, 129, 130, 131, 131, 131, 131, 132, 133, 134, 137, 134, 135, 136, 138, 139, 140, 141,
|
||||
142, 143, 144, 145, 150, 151, 154, 146, 146, 147, 157, 146, 146, 147, 157, 148, 149, 196, 144, 151,
|
||||
148, 149, 152, 153, 155, 156, 147, 159, 160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171,
|
||||
171, 172, 173, 174, 175, 176, 177, 179, 180, 181, 181, 182, 184, 195, 185, 186, 187, 188, 189, 190,
|
||||
191, 192, 193, 194, 1, 3, 6, 94, 118, 158, 178, 183
|
||||
};
|
||||
|
||||
local _parse_sql_trans_actions = { [0] =
|
||||
1, 0, 3, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 3, 1, 1, 1, 1, 1, 3, 1, 1, 3, 1, 1, 3, 1, 1, 1, 1,
|
||||
3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 5, 20, 1, 3, 30, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
5, 20, 1, 3, 26, 3, 3, 1, 23, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 5, 20, 1, 3, 42, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 11, 1, 5, 5, 1, 5, 20, 46, 5, 1, 3, 34, 1, 14, 1, 17, 1, 1,
|
||||
51, 38, 1, 1, 1, 1, 8, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 0, 3, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 3, 1, 1, 1, 1, 1, 3, 1, 1, 3, 1, 1, 3, 1, 1, 1, 1,
|
||||
3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 5, 20, 1, 3, 30, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
5, 20, 1, 3, 26, 3, 3, 1, 23, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 5, 20, 1, 3, 42, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 11, 1, 5, 5, 1, 5, 20, 46, 5, 1, 3, 34, 1, 14, 1, 17, 1, 1,
|
||||
51, 38, 1, 1, 1, 1, 8, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
|
||||
};
|
||||
|
||||
|
@ -277,7 +277,7 @@ function parse_sql(data, h)
|
|||
local mark, token;
|
||||
local table_name, columns, value_lists, value_list, value_count;
|
||||
|
||||
|
||||
|
||||
cs = parse_sql_start;
|
||||
|
||||
-- ragel flat exec
|
||||
|
@ -322,10 +322,10 @@ function parse_sql(data, h)
|
|||
_inds = _parse_sql_index_offsets[cs];
|
||||
_slen = _parse_sql_key_spans[cs];
|
||||
|
||||
if _slen > 0 and
|
||||
_parse_sql_trans_keys[_keys] <= data:byte(p) and
|
||||
data:byte(p) <= _parse_sql_trans_keys[_keys + 1] then
|
||||
_trans = _parse_sql_indicies[ _inds + data:byte(p) - _parse_sql_trans_keys[_keys] ];
|
||||
if _slen > 0 and
|
||||
_parse_sql_trans_keys[_keys] <= data:byte(p) and
|
||||
data:byte(p) <= _parse_sql_trans_keys[_keys + 1] then
|
||||
_trans = _parse_sql_indicies[ _inds + data:byte(p) - _parse_sql_trans_keys[_keys] ];
|
||||
else _trans =_parse_sql_indicies[ _inds + _slen ]; end
|
||||
|
||||
cs = _parse_sql_trans_targs[_trans];
|
||||
|
@ -364,7 +364,7 @@ function parse_sql(data, h)
|
|||
h.create(table_name, columns); -- ACTION
|
||||
elseif _tempval == 7 then --4 FROM_STATE_ACTION_SWITCH
|
||||
-- line 65 "sql.rl" -- end of line directive
|
||||
|
||||
|
||||
value_count = value_count + 1; value_list[value_count] = token:gsub("\\.", _sql_unescapes);
|
||||
-- ACTION
|
||||
elseif _tempval == 8 then --4 FROM_STATE_ACTION_SWITCH
|
||||
|
@ -392,7 +392,7 @@ function parse_sql(data, h)
|
|||
end
|
||||
|
||||
if _trigger_goto then _continue = true; break; end
|
||||
end -- endif
|
||||
end -- endif
|
||||
|
||||
if _goto_level <= _again then
|
||||
if cs == 0 then
|
||||
|
|
|
@ -24,7 +24,7 @@ local function create_table(connection, params)
|
|||
elseif params.driver == "MySQL" then
|
||||
create_sql = create_sql:gsub("`value` TEXT", "`value` MEDIUMTEXT");
|
||||
end
|
||||
|
||||
|
||||
local stmt = connection:prepare(create_sql);
|
||||
if stmt then
|
||||
local ok = stmt:execute();
|
||||
|
|
|
@ -115,7 +115,7 @@ if have_err then
|
|||
print("");
|
||||
os.exit(1);
|
||||
end
|
||||
|
||||
|
||||
local itype = config[from_store].type;
|
||||
local otype = config[to_store].type;
|
||||
local reader = require("migrator."..itype).reader(config[from_store]);
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
#!/usr/bin/env lua
|
||||
-- Prosody IM
|
||||
-- Copyright (C) 2008-2009 Waqas Hussain
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
-- Copyright (C) 2008-2009 Matthew Wild
|
||||
-- Copyright (C) 2008-2009 Waqas Hussain
|
||||
-- Copyright (C) 2010 Stefan Gehn
|
||||
--
|
||||
--
|
||||
-- This project is MIT/X11 licensed. Please see the
|
||||
-- COPYING file in the source package for more information.
|
||||
--
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue