Software / code / prosody
Comparison
plugins/storage/ejabberdstore.lib.lua @ 2678:c5882e2e12b5
mod_storage, plus a bit of SQL and XML.
| author | Waqas Hussain <waqas20@gmail.com> |
|---|---|
| date | Fri, 19 Feb 2010 22:32:28 +0500 |
comparison
equal
deleted
inserted
replaced
| 2677:467568f1117d | 2678:c5882e2e12b5 |
|---|---|
| 1 | |
| 2 local handlers = {}; | |
| 3 | |
| 4 handlers.accounts = { | |
| 5 get = function(self, user) | |
| 6 local select = self:query("select password from users where username=?", user); | |
| 7 local row = select and select:fetch(); | |
| 8 if row then return { password = row[1] }; end | |
| 9 end; | |
| 10 set = function(self, user, data) | |
| 11 if data and data.password then | |
| 12 return self:modify("update users set password=? where username=?", data.password, user) | |
| 13 or self:modify("insert into users (username, password) values (?, ?)", user, data.password); | |
| 14 else | |
| 15 return self:modify("delete from users where username=?", user); | |
| 16 end | |
| 17 end; | |
| 18 }; | |
| 19 handlers.vcard = { | |
| 20 get = function(self, user) | |
| 21 local select = self:query("select vcard from vcard where username=?", user); | |
| 22 local row = select and select:fetch(); | |
| 23 if row then return parse_xml(row[1]); end | |
| 24 end; | |
| 25 set = function(self, user, data) | |
| 26 if data then | |
| 27 data = unparse_xml(data); | |
| 28 return self:modify("update vcard set vcard=? where username=?", data, user) | |
| 29 or self:modify("insert into vcard (username, vcard) values (?, ?)", user, data); | |
| 30 else | |
| 31 return self:modify("delete from vcard where username=?", user); | |
| 32 end | |
| 33 end; | |
| 34 }; | |
| 35 handlers.private = { | |
| 36 get = function(self, user) | |
| 37 local select = self:query("select namespace,data from private_storage where username=?", user); | |
| 38 if select then | |
| 39 local data = {}; | |
| 40 for row in select:rows() do | |
| 41 data[row[1]] = parse_xml(row[2]); | |
| 42 end | |
| 43 return data; | |
| 44 end | |
| 45 end; | |
| 46 set = function(self, user, data) | |
| 47 if data then | |
| 48 self:modify("delete from private_storage where username=?", user); | |
| 49 for namespace,text in pairs(data) do | |
| 50 self:modify("insert into private_storage (username, namespace, data) values (?, ?, ?)", user, namespace, unparse_xml(text)); | |
| 51 end | |
| 52 return true; | |
| 53 else | |
| 54 return self:modify("delete from private_storage where username=?", user); | |
| 55 end | |
| 56 end; | |
| 57 -- TODO map_set, map_get | |
| 58 }; | |
| 59 local subscription_map = { N = "none", B = "both", F = "from", T = "to" }; | |
| 60 local subscription_map_reverse = { none = "N", both = "B", from = "F", to = "T" }; | |
| 61 handlers.roster = { | |
| 62 get = function(self, user) | |
| 63 local select = self:query("select jid,nick,subscription,ask,server,subscribe,type from rosterusers where username=?", user); | |
| 64 if select then | |
| 65 local roster = { pending = {} }; | |
| 66 for row in select:rows() do | |
| 67 local jid,nick,subscription,ask,server,subscribe,typ = unpack(row); | |
| 68 local item = { groups = {} }; | |
| 69 if nick == "" then nick = nil; end | |
| 70 item.nick = nick; | |
| 71 item.subscription = subscription_map[subscription]; | |
| 72 if ask == "N" then ask = nil; | |
| 73 elseif ask == "O" then ask = "subscribe" | |
| 74 elseif ask == "I" then roster.pending[jid] = true; ask = nil; | |
| 75 elseif ask == "B" then roster.pending[jid] = true; ask = "subscribe"; | |
| 76 else module:log("debug", "bad roster_item.ask: %s", ask); ask = nil; end | |
| 77 item.ask = ask; | |
| 78 roster[jid] = item; | |
| 79 end | |
| 80 | |
| 81 select = self:query("select jid,grp from rostergroups where username=?", user); | |
| 82 if select then | |
| 83 for row in select:rows() do | |
| 84 local jid,grp = unpack(rows); | |
| 85 if roster[jid] then roster[jid].groups[grp] = true; end | |
| 86 end | |
| 87 end | |
| 88 select = self:query("select version from roster_version where username=?", user); | |
| 89 local row = select and select:fetch(); | |
| 90 if row then | |
| 91 roster[false] = { version = row[1]; }; | |
| 92 end | |
| 93 return roster; | |
| 94 end | |
| 95 end; | |
| 96 set = function(self, user, data) | |
| 97 if data and next(data) ~= nil then | |
| 98 self:modify("delete from rosterusers where username=?", user); | |
| 99 self:modify("delete from rostergroups where username=?", user); | |
| 100 self:modify("delete from roster_version where username=?", user); | |
| 101 local done = {}; | |
| 102 local pending = data.pending or {}; | |
| 103 for jid,item in pairs(data) do | |
| 104 if jid and jid ~= "pending" then | |
| 105 local subscription = subscription_map_reverse[item.subscription]; | |
| 106 local ask; | |
| 107 if pending[jid] then | |
| 108 if item.ask then ask = "B"; else ask = "I"; end | |
| 109 else | |
| 110 if item.ask then ask = "O"; else ask = "N"; end | |
| 111 end | |
| 112 local r = self:modify("insert into rosterusers (username,jid,nick,subscription,ask,askmessage,server,subscribe) values (?, ?, ?, ?, ?, '', '', '')", user, jid, item.nick or "", subscription, ask); | |
| 113 if not r then module:log("debug", "--- :( %s", tostring(r)); end | |
| 114 done[jid] = true; | |
| 115 for group in pairs(item.groups) do | |
| 116 self:modify("insert into rostergroups (username,jid,grp) values (?, ?, ?)", user, jid, group); | |
| 117 end | |
| 118 end | |
| 119 end | |
| 120 for jid in pairs(pending) do | |
| 121 if not done[jid] then | |
| 122 self:modify("insert into rosterusers (username,jid,nick,subscription,ask,askmessage,server,subscribe) values (?, ?, ?, ?, ?. ''. ''. '')", user, jid, "", "N", "I"); | |
| 123 end | |
| 124 end | |
| 125 local version = data[false] and data[false].version; | |
| 126 if version then | |
| 127 self:modify("insert into roster_version (username,version) values (?, ?)", user, version); | |
| 128 end | |
| 129 return true; | |
| 130 else | |
| 131 self:modify("delete from rosterusers where username=?", user); | |
| 132 self:modify("delete from rostergroups where username=?", user); | |
| 133 self:modify("delete from roster_version where username=?", user); | |
| 134 end | |
| 135 end; | |
| 136 }; | |
| 137 | |
| 138 ----------------------------- | |
| 139 local driver = {}; | |
| 140 driver.__index = driver; | |
| 141 | |
| 142 function driver:prepare(sql) | |
| 143 module:log("debug", "query: %s", sql); | |
| 144 local err; | |
| 145 if not self.sqlcache then self.sqlcache = {}; end | |
| 146 local r = self.sqlcache[sql]; | |
| 147 if r then return r; end | |
| 148 r, err = self.database:prepare(sql); | |
| 149 if not r then error("Unable to prepare SQL statement: "..err); end | |
| 150 self.sqlcache[sql] = r; | |
| 151 return r; | |
| 152 end | |
| 153 | |
| 154 function driver:query(sql, ...) | |
| 155 local stmt = self:prepare(sql); | |
| 156 if stmt:execute(...) then return stmt; end | |
| 157 end | |
| 158 function driver:modify(sql, ...) | |
| 159 local stmt = self:query(sql, ...); | |
| 160 if stmt and stmt:affected() > 0 then return stmt; end | |
| 161 end | |
| 162 | |
| 163 function driver:open(host, datastore, typ) | |
| 164 local cache_key = host.." "..datastore; | |
| 165 if self.ds_cache[cache_key] then return self.ds_cache[cache_key]; end | |
| 166 local instance = setmetatable({}, self); | |
| 167 instance.host = host; | |
| 168 instance.datastore = datastore; | |
| 169 local handler = handlers[datastore]; | |
| 170 if not handler then return nil; end | |
| 171 for key,val in pairs(handler) do | |
| 172 instance[key] = val; | |
| 173 end | |
| 174 if instance.init then instance:init(); end | |
| 175 self.ds_cache[cache_key] = instance; | |
| 176 return instance; | |
| 177 end | |
| 178 | |
| 179 ----------------------------- | |
| 180 local _M = {}; | |
| 181 | |
| 182 function _M.new(dbtype, dbname, ...) | |
| 183 local instance = setmetatable({}, driver); | |
| 184 instance.__index = instance; | |
| 185 instance.database = get_database(dbtype, dbname, ...); | |
| 186 instance.ds_cache = {}; | |
| 187 return instance; | |
| 188 end | |
| 189 | |
| 190 return _M; |