Comparison

mod_pep_plus/pubsub.lib.lua @ 2801:cb2342cf3f3c

mod_pep_plus: Snapshot from Prosody trunk 910d3c3f60a6 including dependencies
author Kim Alvefur <zash@zash.se>
date Wed, 18 Oct 2017 09:56:29 +0200
comparison
equal deleted inserted replaced
2800:8d9aed6d1f87 2801:cb2342cf3f3c
1 local t_unpack = table.unpack or unpack; -- luacheck: ignore 113
2 local time_now = os.time;
3
4 local set = require "util.set";
5 local st = require "util.stanza";
6 local it = require "util.iterators";
7 local uuid_generate = require "util.uuid".generate;
8 local dataform = require"util.dataforms".new;
9
10 local xmlns_pubsub = "http://jabber.org/protocol/pubsub";
11 local xmlns_pubsub_errors = "http://jabber.org/protocol/pubsub#errors";
12 local xmlns_pubsub_owner = "http://jabber.org/protocol/pubsub#owner";
13
14 local _M = {};
15
16 local handlers = {};
17 _M.handlers = handlers;
18
19 local pubsub_errors = {
20 ["conflict"] = { "cancel", "conflict" };
21 ["invalid-jid"] = { "modify", "bad-request", nil, "invalid-jid" };
22 ["jid-required"] = { "modify", "bad-request", nil, "jid-required" };
23 ["nodeid-required"] = { "modify", "bad-request", nil, "nodeid-required" };
24 ["item-not-found"] = { "cancel", "item-not-found" };
25 ["not-subscribed"] = { "modify", "unexpected-request", nil, "not-subscribed" };
26 ["forbidden"] = { "auth", "forbidden" };
27 ["not-allowed"] = { "cancel", "not-allowed" };
28 };
29 local function pubsub_error_reply(stanza, error)
30 local e = pubsub_errors[error];
31 local reply = st.error_reply(stanza, t_unpack(e, 1, 3));
32 if e[4] then
33 reply:tag(e[4], { xmlns = xmlns_pubsub_errors }):up();
34 end
35 return reply;
36 end
37 _M.pubsub_error_reply = pubsub_error_reply;
38
39 local node_config_form = dataform {
40 {
41 type = "hidden";
42 name = "FORM_TYPE";
43 value = "http://jabber.org/protocol/pubsub#node_config";
44 };
45 {
46 type = "text-single";
47 name = "pubsub#max_items";
48 label = "Max # of items to persist";
49 };
50 {
51 type = "boolean";
52 name = "pubsub#persist_items";
53 label = "Persist items to storage";
54 };
55 };
56
57 local service_method_feature_map = {
58 add_subscription = { "subscribe" };
59 create = { "create-nodes", "instant-nodes", "item-ids", "create-and-configure" };
60 delete = { "delete-nodes" };
61 get_items = { "retrieve-items" };
62 get_subscriptions = { "retrieve-subscriptions" };
63 node_defaults = { "retrieve-default" };
64 publish = { "publish" };
65 purge = { "purge-nodes" };
66 retract = { "delete-items", "retract-items" };
67 set_node_config = { "config-node" };
68 };
69 local service_config_feature_map = {
70 autocreate_on_publish = { "auto-create" };
71 };
72
73 function _M.get_feature_set(service)
74 local supported_features = set.new();
75
76 for method, features in pairs(service_method_feature_map) do
77 if service[method] then
78 for _, feature in ipairs(features) do
79 if feature then
80 supported_features:add(feature);
81 end
82 end
83 end
84 end
85
86 for option, features in pairs(service_config_feature_map) do
87 if service.config[option] then
88 for _, feature in ipairs(features) do
89 if feature then
90 supported_features:add(feature);
91 end
92 end
93 end
94 end
95
96 for affiliation in pairs(service.config.capabilities) do
97 if affiliation ~= "none" and affiliation ~= "owner" then
98 supported_features:add(affiliation.."-affiliation");
99 end
100 end
101
102 return supported_features;
103 end
104
105 function _M.handle_pubsub_iq(event, service)
106 local origin, stanza = event.origin, event.stanza;
107 local pubsub_tag = stanza.tags[1];
108 local action = pubsub_tag.tags[1];
109 if not action then
110 return origin.send(st.error_reply(stanza, "cancel", "bad-request"));
111 end
112 local prefix = "";
113 if pubsub_tag.attr.xmlns == xmlns_pubsub_owner then
114 prefix = "owner_";
115 end
116 local handler = handlers[prefix..stanza.attr.type.."_"..action.name];
117 if handler then
118 handler(origin, stanza, action, service);
119 return true;
120 end
121 end
122
123 function handlers.get_items(origin, stanza, items, service)
124 local node = items.attr.node;
125 local item = items:get_child("item");
126 local item_id = item and item.attr.id;
127
128 if not node then
129 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
130 return true;
131 end
132 local ok, results = service:get_items(node, stanza.attr.from, item_id);
133 if not ok then
134 origin.send(pubsub_error_reply(stanza, results));
135 return true;
136 end
137
138 local data = st.stanza("items", { node = node });
139 for _, id in ipairs(results) do
140 data:add_child(results[id]);
141 end
142 local reply;
143 if data then
144 reply = st.reply(stanza)
145 :tag("pubsub", { xmlns = xmlns_pubsub })
146 :add_child(data);
147 else
148 reply = pubsub_error_reply(stanza, "item-not-found");
149 end
150 origin.send(reply);
151 return true;
152 end
153
154 function handlers.get_subscriptions(origin, stanza, subscriptions, service)
155 local node = subscriptions.attr.node;
156 local ok, ret = service:get_subscriptions(node, stanza.attr.from, stanza.attr.from);
157 if not ok then
158 origin.send(pubsub_error_reply(stanza, ret));
159 return true;
160 end
161 local reply = st.reply(stanza)
162 :tag("pubsub", { xmlns = xmlns_pubsub })
163 :tag("subscriptions");
164 for _, sub in ipairs(ret) do
165 reply:tag("subscription", { node = sub.node, jid = sub.jid, subscription = 'subscribed' }):up();
166 end
167 origin.send(reply);
168 return true;
169 end
170
171 function handlers.set_create(origin, stanza, create, service)
172 local node = create.attr.node;
173 local ok, ret, reply;
174 local config;
175 local configure = stanza.tags[1]:get_child("configure");
176 if configure then
177 local config_form = configure:get_child("x", "jabber:x:data");
178 if not config_form then
179 origin.send(st.error_reply(stanza, "modify", "bad-request", "Missing dataform"));
180 return true;
181 end
182 local form_data, err = node_config_form:data(config_form);
183 if not form_data then
184 origin.send(st.error_reply(stanza, "modify", "bad-request", err));
185 return true;
186 end
187 config = {
188 ["max_items"] = tonumber(form_data["pubsub#max_items"]);
189 ["persist_items"] = form_data["pubsub#persist_items"];
190 };
191 end
192 if node then
193 ok, ret = service:create(node, stanza.attr.from, config);
194 if ok then
195 reply = st.reply(stanza);
196 else
197 reply = pubsub_error_reply(stanza, ret);
198 end
199 else
200 repeat
201 node = uuid_generate();
202 ok, ret = service:create(node, stanza.attr.from, config);
203 until ok or ret ~= "conflict";
204 if ok then
205 reply = st.reply(stanza)
206 :tag("pubsub", { xmlns = xmlns_pubsub })
207 :tag("create", { node = node });
208 else
209 reply = pubsub_error_reply(stanza, ret);
210 end
211 end
212 origin.send(reply);
213 return true;
214 end
215
216 function handlers.owner_set_delete(origin, stanza, delete, service)
217 local node = delete.attr.node;
218
219 local reply;
220 if not node then
221 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
222 return true;
223 end
224 local ok, ret = service:delete(node, stanza.attr.from);
225 if ok then
226 reply = st.reply(stanza);
227 else
228 reply = pubsub_error_reply(stanza, ret);
229 end
230 origin.send(reply);
231 return true;
232 end
233
234 function handlers.set_subscribe(origin, stanza, subscribe, service)
235 local node, jid = subscribe.attr.node, subscribe.attr.jid;
236 if not (node and jid) then
237 origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
238 return true;
239 end
240 --[[
241 local options_tag, options = stanza.tags[1]:get_child("options"), nil;
242 if options_tag then
243 options = options_form:data(options_tag.tags[1]);
244 end
245 --]]
246 local options_tag, options; -- FIXME
247 local ok, ret = service:add_subscription(node, stanza.attr.from, jid, options);
248 local reply;
249 if ok then
250 reply = st.reply(stanza)
251 :tag("pubsub", { xmlns = xmlns_pubsub })
252 :tag("subscription", {
253 node = node,
254 jid = jid,
255 subscription = "subscribed"
256 }):up();
257 if options_tag then
258 reply:add_child(options_tag);
259 end
260 else
261 reply = pubsub_error_reply(stanza, ret);
262 end
263 origin.send(reply);
264 end
265
266 function handlers.set_unsubscribe(origin, stanza, unsubscribe, service)
267 local node, jid = unsubscribe.attr.node, unsubscribe.attr.jid;
268 if not (node and jid) then
269 origin.send(pubsub_error_reply(stanza, jid and "nodeid-required" or "invalid-jid"));
270 return true;
271 end
272 local ok, ret = service:remove_subscription(node, stanza.attr.from, jid);
273 local reply;
274 if ok then
275 reply = st.reply(stanza);
276 else
277 reply = pubsub_error_reply(stanza, ret);
278 end
279 origin.send(reply);
280 return true;
281 end
282
283 function handlers.set_publish(origin, stanza, publish, service)
284 local node = publish.attr.node;
285 if not node then
286 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
287 return true;
288 end
289 local item = publish:get_child("item");
290 local id = (item and item.attr.id);
291 if not id then
292 id = uuid_generate();
293 if item then
294 item.attr.id = id;
295 end
296 end
297 local ok, ret = service:publish(node, stanza.attr.from, id, item);
298 local reply;
299 if ok then
300 reply = st.reply(stanza)
301 :tag("pubsub", { xmlns = xmlns_pubsub })
302 :tag("publish", { node = node })
303 :tag("item", { id = id });
304 else
305 reply = pubsub_error_reply(stanza, ret);
306 end
307 origin.send(reply);
308 return true;
309 end
310
311 function handlers.set_retract(origin, stanza, retract, service)
312 local node, notify = retract.attr.node, retract.attr.notify;
313 notify = (notify == "1") or (notify == "true");
314 local item = retract:get_child("item");
315 local id = item and item.attr.id
316 if not (node and id) then
317 origin.send(pubsub_error_reply(stanza, node and "item-not-found" or "nodeid-required"));
318 return true;
319 end
320 local reply, notifier;
321 if notify then
322 notifier = st.stanza("retract", { id = id });
323 end
324 local ok, ret = service:retract(node, stanza.attr.from, id, notifier);
325 if ok then
326 reply = st.reply(stanza);
327 else
328 reply = pubsub_error_reply(stanza, ret);
329 end
330 origin.send(reply);
331 return true;
332 end
333
334 function handlers.owner_set_purge(origin, stanza, purge, service)
335 local node, notify = purge.attr.node, purge.attr.notify;
336 notify = (notify == "1") or (notify == "true");
337 local reply;
338 if not node then
339 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
340 return true;
341 end
342 local ok, ret = service:purge(node, stanza.attr.from, notify);
343 if ok then
344 reply = st.reply(stanza);
345 else
346 reply = pubsub_error_reply(stanza, ret);
347 end
348 origin.send(reply);
349 return true;
350 end
351
352 function handlers.owner_get_configure(origin, stanza, config, service)
353 local node = config.attr.node;
354 if not node then
355 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
356 return true;
357 end
358
359 if not service:may(node, stanza.attr.from, "configure") then
360 origin.send(pubsub_error_reply(stanza, "forbidden"));
361 return true;
362 end
363
364 local node_obj = service.nodes[node];
365 if not node_obj then
366 origin.send(pubsub_error_reply(stanza, "item-not-found"));
367 return true;
368 end
369
370 local node_config = node_obj.config;
371 local pubsub_form_data = {
372 ["pubsub#max_items"] = tostring(node_config["max_items"]);
373 ["pubsub#persist_items"] = node_config["persist_items"]
374 }
375 local reply = st.reply(stanza)
376 :tag("pubsub", { xmlns = xmlns_pubsub_owner })
377 :tag("configure", { node = node })
378 :add_child(node_config_form:form(pubsub_form_data));
379 origin.send(reply);
380 return true;
381 end
382
383 function handlers.owner_set_configure(origin, stanza, config, service)
384 local node = config.attr.node;
385 if not node then
386 origin.send(pubsub_error_reply(stanza, "nodeid-required"));
387 return true;
388 end
389 if not service:may(node, stanza.attr.from, "configure") then
390 origin.send(pubsub_error_reply(stanza, "forbidden"));
391 return true;
392 end
393 local config_form = config:get_child("x", "jabber:x:data");
394 if not config_form then
395 origin.send(st.error_reply(stanza, "modify", "bad-request", "Missing dataform"));
396 return true;
397 end
398 local form_data, err = node_config_form:data(config_form);
399 if not form_data then
400 origin.send(st.error_reply(stanza, "modify", "bad-request", err));
401 return true;
402 end
403 local new_config = {
404 ["max_items"] = tonumber(form_data["pubsub#max_items"]);
405 ["persist_items"] = form_data["pubsub#persist_items"];
406 };
407 local ok, err = service:set_node_config(node, stanza.attr.from, new_config);
408 if not ok then
409 origin.send(pubsub_error_reply(stanza, err));
410 return true;
411 end
412 origin.send(st.reply(stanza));
413 return true;
414 end
415
416 function handlers.owner_get_default(origin, stanza, default, service) -- luacheck: ignore 212/default
417 local pubsub_form_data = {
418 ["pubsub#max_items"] = tostring(service.node_defaults["max_items"]);
419 ["pubsub#persist_items"] = service.node_defaults["persist_items"]
420 }
421 local reply = st.reply(stanza)
422 :tag("pubsub", { xmlns = xmlns_pubsub_owner })
423 :tag("default")
424 :add_child(node_config_form:form(pubsub_form_data));
425 origin.send(reply);
426 return true;
427 end
428
429 local function create_encapsulating_item(id, payload)
430 local item = st.stanza("item", { id = id, xmlns = xmlns_pubsub });
431 item:add_child(payload);
432 return item;
433 end
434
435 local function archive_itemstore(archive, config, user, node)
436 module:log("debug", "Creation of itemstore for node %s with config %s", node, config);
437 local get_set = {};
438 function get_set:items() -- luacheck: ignore 212/self
439 local data, err = archive:find(user, {
440 limit = tonumber(config["pubsub#max_items"]);
441 reverse = true;
442 });
443 if not data then
444 module:log("error", "Unable to get items: %s", err);
445 return true;
446 end
447 module:log("debug", "Listed items %s", data);
448 return it.reverse(function()
449 local id, payload, when, publisher = data();
450 if id == nil then
451 return;
452 end
453 local item = create_encapsulating_item(id, payload, publisher);
454 return id, item;
455 end);
456 end
457 function get_set:get(key) -- luacheck: ignore 212/self
458 local data, err = archive:find(user, {
459 key = key;
460 -- Get the last item with that key, if the archive doesn't deduplicate
461 reverse = true,
462 limit = 1;
463 });
464 if not data then
465 module:log("error", "Unable to get item: %s", err);
466 return nil, err;
467 end
468 local id, payload, when, publisher = data();
469 module:log("debug", "Get item %s (published at %s by %s)", id, when, publisher);
470 if id == nil then
471 return nil;
472 end
473 return create_encapsulating_item(id, payload, publisher);
474 end
475 function get_set:set(key, value) -- luacheck: ignore 212/self
476 local data, err;
477 if value ~= nil then
478 local publisher = value.attr.publisher;
479 local payload = value.tags[1];
480 data, err = archive:append(user, key, payload, time_now(), publisher);
481 else
482 data, err = archive:delete(user, { key = key; });
483 end
484 if not data then
485 module:log("error", "Unable to set item: %s", err);
486 return nil, err;
487 end
488 return data;
489 end
490 function get_set:clear() -- luacheck: ignore 212/self
491 return archive:delete(user);
492 end
493 return setmetatable(get_set, archive);
494 end
495 _M.archive_itemstore = archive_itemstore;
496
497 return _M;