Providing some human readable error messages and some fixes.
[prosody.git] / plugins / mod_roster.lua
1
2 local st = require "util.stanza"
3 local send = require "core.sessionmanager".send_to_session
4
5 local jid_split = require "util.jid".split;
6 local t_concat = table.concat;
7
8 local rm_remove_from_roster = require "core.rostermanager".remove_from_roster;
9 local rm_add_to_roster = require "core.rostermanager".add_to_roster;
10 local rm_roster_push = require "core.rostermanager".roster_push;
11
12 add_iq_handler("c2s", "jabber:iq:roster", 
13                 function (session, stanza)
14                         if stanza.tags[1].name == "query" then
15                                 if stanza.attr.type == "get" then
16                                         local roster = st.reply(stanza)
17                                                                 :query("jabber:iq:roster");
18                                         for jid in pairs(session.roster) do
19                                                 if jid ~= "pending" then
20                                                         roster:tag("item", {
21                                                                 jid = jid,
22                                                                 subscription = session.roster[jid].subscription,
23                                                                 ask = session.roster[jid].ask,
24                                                                 name = session.roster[jid].name,
25                                                         });
26                                                         for group in pairs(session.roster[jid].groups) do
27                                                                 roster:tag("group"):text(group):up();
28                                                         end
29                                                         roster:up(); -- move out from item
30                                                 end
31                                         end
32                                         send(session, roster);
33                                         session.interested = true; -- resource is interested in roster updates
34                                         return true;
35                                 elseif stanza.attr.type == "set" then
36                                         local query = stanza.tags[1];
37                                         if #query.tags == 1 and query.tags[1].name == "item"
38                                                         and query.tags[1].attr.xmlns == "jabber:iq:roster" and query.tags[1].attr.jid
39                                                         and query.tags[1].attr.jid ~= "pending" then
40                                                 local item = query.tags[1];
41                                                 local from_node, from_host = jid_split(stanza.attr.from);
42                                                 local node, host, resource = jid_split(item.attr.jid);
43                                                 if not resource then
44                                                         if item.attr.jid ~= from_node.."@"..from_host then
45                                                                 if item.attr.subscription == "remove" then
46                                                                         if session.roster[item.attr.jid] then
47                                                                                 local success, err_type, err_cond, err_msg = rm_remove_from_roster(session, item.attr.jid);
48                                                                                 if success then
49                                                                                         send(session, st.reply(stanza));
50                                                                                         rm_roster_push(from_node, from_host, item.attr.jid);
51                                                                                 else
52                                                                                         send(session, st.error_reply(stanza, err_type, err_cond, err_msg));
53                                                                                 end
54                                                                         else
55                                                                                 send(session, st.error_reply(stanza, "modify", "item-not-found"));
56                                                                         end
57                                                                 else
58                                                                         local r_item = {name = item.attr.name, groups = {}};
59                                                                         if r_item.name == "" then r_item.name = nil; end
60                                                                         if session.roster[item.attr.jid] then
61                                                                                 r_item.subscription = session.roster[item.attr.jid].subscription;
62                                                                                 r_item.ask = session.roster[item.attr.jid].ask;
63                                                                         else
64                                                                                 r_item.subscription = "none";
65                                                                         end
66                                                                         for _, child in ipairs(item) do 
67                                                                                 if child.name == "group" then
68                                                                                         local text = t_concat(child);
69                                                                                         if text and text ~= "" then
70                                                                                                 r_item.groups[text] = true;
71                                                                                         end
72                                                                                 end
73                                                                         end
74                                                                         local success, err_type, err_cond, err_msg = rm_add_to_roster(session, item.attr.jid, r_item);
75                                                                         if success then
76                                                                                 send(session, st.reply(stanza));
77                                                                                 rm_roster_push(from_node, from_host, item.attr.jid);
78                                                                         else
79                                                                                 send(session, st.error_reply(stanza, err_type, err_cond, err_msg));
80                                                                         end
81                                                                 end
82                                                         else
83                                                                 send(session, st.error_reply(stanza, "cancel", "not-allowed"));
84                                                         end
85                                                 else
86                                                         send(session, st.error_reply(stanza, "modify", "bad-request")); -- FIXME what's the correct error?
87                                                 end
88                                         else
89                                                 send(session, st.error_reply(stanza, "modify", "bad-request"));
90                                         end
91                                         return true;
92                                 end
93                         end
94                 end);