Merge 0.9->0.10
[prosody.git] / util / jid.lua
1 -- Prosody IM
2 -- Copyright (C) 2008-2010 Matthew Wild
3 -- Copyright (C) 2008-2010 Waqas Hussain
4 --
5 -- This project is MIT/X11 licensed. Please see the
6 -- COPYING file in the source package for more information.
7 --
8
9
10
11 local match, sub = string.match, string.sub;
12 local nodeprep = require "util.encodings".stringprep.nodeprep;
13 local nameprep = require "util.encodings".stringprep.nameprep;
14 local resourceprep = require "util.encodings".stringprep.resourceprep;
15
16 local escapes = {
17         [" "] = "\\20"; ['"'] = "\\22";
18         ["&"] = "\\26"; ["'"] = "\\27";
19         ["/"] = "\\2f"; [":"] = "\\3a";
20         ["<"] = "\\3c"; [">"] = "\\3e";
21         ["@"] = "\\40"; ["\\"] = "\\5c";
22 };
23 local unescapes = {};
24 for k,v in pairs(escapes) do unescapes[v] = k; end
25
26 module "jid"
27
28 local function _split(jid)
29         if not jid then return; end
30         local node, nodepos = match(jid, "^([^@/]+)@()");
31         local host, hostpos = match(jid, "^([^@/]+)()", nodepos)
32         if node and not host then return nil, nil, nil; end
33         local resource = match(jid, "^/(.+)$", hostpos);
34         if (not host) or ((not resource) and #jid >= hostpos) then return nil, nil, nil; end
35         return node, host, resource;
36 end
37 split = _split;
38
39 function bare(jid)
40         local node, host = _split(jid);
41         if node and host then
42                 return node.."@"..host;
43         end
44         return host;
45 end
46
47 local function _prepped_split(jid)
48         local node, host, resource = _split(jid);
49         if host then
50                 if sub(host, -1, -1) == "." then -- Strip empty root label
51                         host = sub(host, 1, -2);
52                 end
53                 host = nameprep(host);
54                 if not host then return; end
55                 if node then
56                         node = nodeprep(node);
57                         if not node then return; end
58                 end
59                 if resource then
60                         resource = resourceprep(resource);
61                         if not resource then return; end
62                 end
63                 return node, host, resource;
64         end
65 end
66 prepped_split = _prepped_split;
67
68 function prep(jid)
69         local node, host, resource = _prepped_split(jid);
70         if host then
71                 if node then
72                         host = node .. "@" .. host;
73                 end
74                 if resource then
75                         host = host .. "/" .. resource;
76                 end
77         end
78         return host;
79 end
80
81 function join(node, host, resource)
82         if node and host and resource then
83                 return node.."@"..host.."/"..resource;
84         elseif node and host then
85                 return node.."@"..host;
86         elseif host and resource then
87                 return host.."/"..resource;
88         elseif host then
89                 return host;
90         end
91         return nil; -- Invalid JID
92 end
93
94 function compare(jid, acl)
95         -- compare jid to single acl rule
96         -- TODO compare to table of rules?
97         local jid_node, jid_host, jid_resource = _split(jid);
98         local acl_node, acl_host, acl_resource = _split(acl);
99         if ((acl_node ~= nil and acl_node == jid_node) or acl_node == nil) and
100                 ((acl_host ~= nil and acl_host == jid_host) or acl_host == nil) and
101                 ((acl_resource ~= nil and acl_resource == jid_resource) or acl_resource == nil) then
102                 return true
103         end
104         return false
105 end
106
107 function escape(s) return s and (s:gsub(".", escapes)); end
108 function unescape(s) return s and (s:gsub("\\%x%x", unescapes)); end
109
110 return _M;