Software / code / prosody
Comparison
plugins/mod_storage_sql_ejabberd.lua @ 3646:66c5fb109e64
Rename storage/mod_ejabberd to mod_storage_sql_ejabberd. Also rename configuration option. Untested.
| author | Matthew Wild <mwild1@gmail.com> |
|---|---|
| date | Sat, 27 Nov 2010 19:49:29 +0000 |
| parent | 3429:plugins/storage/mod_ejabberd.lua@8cdb0179371a |
comparison
equal
deleted
inserted
replaced
| 3645:951f4752ef31 | 3646:66c5fb109e64 |
|---|---|
| 1 | |
| 2 local setmetatable = setmetatable; | |
| 3 local error = error; | |
| 4 local unpack = unpack; | |
| 5 local module = module; | |
| 6 local tostring = tostring; | |
| 7 local pairs, next = pairs, next; | |
| 8 local prosody = prosody; | |
| 9 local assert = assert; | |
| 10 local require = require; | |
| 11 local st = require "util.stanza"; | |
| 12 local DBI = require "DBI"; | |
| 13 | |
| 14 -- connect to db | |
| 15 local params = module:get_option("sql_ejabberd") or error("No sql_ejabberd config option"); | |
| 16 local database; | |
| 17 do | |
| 18 module:log("debug", "Opening database: %s", "dbi:"..params.driver..":"..params.database); | |
| 19 prosody.unlock_globals(); | |
| 20 local dbh, err = DBI.Connect( | |
| 21 params.driver, params.database, | |
| 22 params.username, params.password, | |
| 23 params.host, params.port | |
| 24 ); | |
| 25 prosody.lock_globals(); | |
| 26 assert(dbh, err); | |
| 27 dbh:autocommit(true); | |
| 28 database = dbh; | |
| 29 end | |
| 30 | |
| 31 -- initialize db | |
| 32 local ejabberd_init = module:require("ejabberd_init"); | |
| 33 ejabberd_init.init(database); | |
| 34 | |
| 35 local sqlcache = {}; | |
| 36 local function prepare(sql) | |
| 37 module:log("debug", "query: %s", sql); | |
| 38 local err; | |
| 39 local r = sqlcache[sql]; | |
| 40 if not r then | |
| 41 r, err = database:prepare(sql); | |
| 42 if not r then error("Unable to prepare SQL statement: "..err); end | |
| 43 sqlcache[sql] = r; | |
| 44 end | |
| 45 return r; | |
| 46 end | |
| 47 | |
| 48 local _parse_xml = module:require("xmlparse"); | |
| 49 local function parse_xml(str) | |
| 50 local s = _parse_xml(str); | |
| 51 if s and not s.gsub then | |
| 52 return st.preserialize(s); | |
| 53 end | |
| 54 end | |
| 55 local function unparse_xml(s) | |
| 56 return tostring(st.deserialize(s)); | |
| 57 end | |
| 58 | |
| 59 | |
| 60 local handlers = {}; | |
| 61 | |
| 62 handlers.accounts = { | |
| 63 get = function(self, user) | |
| 64 local select = self:query("select password from users where username=? and host=?", user, self.host); | |
| 65 local row = select and select:fetch(); | |
| 66 if row then return { password = row[1] }; end | |
| 67 end; | |
| 68 set = function(self, user, data) | |
| 69 if data and data.password then | |
| 70 return self:modify("update users set password=? where username=? and host=?", data.password, user, self.host) | |
| 71 or self:modify("insert into users (username, host, password) values (?, ?, ?)", user, self.host, data.password); | |
| 72 else | |
| 73 return self:modify("delete from users where username=? and host=?", user, self.host); | |
| 74 end | |
| 75 end; | |
| 76 }; | |
| 77 handlers.vcard = { | |
| 78 get = function(self, user) | |
| 79 local select = self:query("select vcard from vcard where username=? and host=?", user, self.host); | |
| 80 local row = select and select:fetch(); | |
| 81 if row then return parse_xml(row[1]); end | |
| 82 end; | |
| 83 set = function(self, user, data) | |
| 84 if data then | |
| 85 data = unparse_xml(data); | |
| 86 return self:modify("update vcard set vcard=? where username=? and host=?", data, user, self.host) | |
| 87 or self:modify("insert into vcard (username, host, vcard) values (?, ?, ?)", user, self.host, data); | |
| 88 else | |
| 89 return self:modify("delete from vcard where username=? and host=?", user, self.host); | |
| 90 end | |
| 91 end; | |
| 92 }; | |
| 93 handlers.private = { | |
| 94 get = function(self, user) | |
| 95 local select = self:query("select namespace,data from private_storage where username=? and host=?", user, self.host); | |
| 96 if select then | |
| 97 local data = {}; | |
| 98 for row in select:rows() do | |
| 99 data[row[1]] = parse_xml(row[2]); | |
| 100 end | |
| 101 return data; | |
| 102 end | |
| 103 end; | |
| 104 set = function(self, user, data) | |
| 105 if data then | |
| 106 self:modify("delete from private_storage where username=? and host=?", user, self.host); | |
| 107 for namespace,text in pairs(data) do | |
| 108 self:modify("insert into private_storage (username, host, namespace, data) values (?, ?, ?, ?)", user, self.host, namespace, unparse_xml(text)); | |
| 109 end | |
| 110 return true; | |
| 111 else | |
| 112 return self:modify("delete from private_storage where username=? and host=?", user, self.host); | |
| 113 end | |
| 114 end; | |
| 115 -- TODO map_set, map_get | |
| 116 }; | |
| 117 local subscription_map = { N = "none", B = "both", F = "from", T = "to" }; | |
| 118 local subscription_map_reverse = { none = "N", both = "B", from = "F", to = "T" }; | |
| 119 handlers.roster = { | |
| 120 get = function(self, user) | |
| 121 local select = self:query("select jid,nick,subscription,ask,server,subscribe,type from rosterusers where username=?", user); | |
| 122 if select then | |
| 123 local roster = { pending = {} }; | |
| 124 for row in select:rows() do | |
| 125 local jid,nick,subscription,ask,server,subscribe,typ = unpack(row); | |
| 126 local item = { groups = {} }; | |
| 127 if nick == "" then nick = nil; end | |
| 128 item.nick = nick; | |
| 129 item.subscription = subscription_map[subscription]; | |
| 130 if ask == "N" then ask = nil; | |
| 131 elseif ask == "O" then ask = "subscribe" | |
| 132 elseif ask == "I" then roster.pending[jid] = true; ask = nil; | |
| 133 elseif ask == "B" then roster.pending[jid] = true; ask = "subscribe"; | |
| 134 else module:log("debug", "bad roster_item.ask: %s", ask); ask = nil; end | |
| 135 item.ask = ask; | |
| 136 roster[jid] = item; | |
| 137 end | |
| 138 | |
| 139 select = self:query("select jid,grp from rostergroups where username=?", user); | |
| 140 if select then | |
| 141 for row in select:rows() do | |
| 142 local jid,grp = unpack(row); | |
| 143 if roster[jid] then roster[jid].groups[grp] = true; end | |
| 144 end | |
| 145 end | |
| 146 select = self:query("select version from roster_version where username=?", user); | |
| 147 local row = select and select:fetch(); | |
| 148 if row then | |
| 149 roster[false] = { version = row[1]; }; | |
| 150 end | |
| 151 return roster; | |
| 152 end | |
| 153 end; | |
| 154 set = function(self, user, data) | |
| 155 if data and next(data) ~= nil then | |
| 156 self:modify("delete from rosterusers where username=?", user); | |
| 157 self:modify("delete from rostergroups where username=?", user); | |
| 158 self:modify("delete from roster_version where username=?", user); | |
| 159 local done = {}; | |
| 160 local pending = data.pending or {}; | |
| 161 for jid,item in pairs(data) do | |
| 162 if jid and jid ~= "pending" then | |
| 163 local subscription = subscription_map_reverse[item.subscription]; | |
| 164 local ask; | |
| 165 if pending[jid] then | |
| 166 if item.ask then ask = "B"; else ask = "I"; end | |
| 167 else | |
| 168 if item.ask then ask = "O"; else ask = "N"; end | |
| 169 end | |
| 170 local r = self:modify("insert into rosterusers (username,jid,nick,subscription,ask,askmessage,server,subscribe) values (?, ?, ?, ?, ?, '', '', '')", user, jid, item.nick or "", subscription, ask); | |
| 171 if not r then module:log("debug", "--- :( %s", tostring(r)); end | |
| 172 done[jid] = true; | |
| 173 for group in pairs(item.groups) do | |
| 174 self:modify("insert into rostergroups (username,jid,grp) values (?, ?, ?)", user, jid, group); | |
| 175 end | |
| 176 end | |
| 177 end | |
| 178 for jid in pairs(pending) do | |
| 179 if not done[jid] then | |
| 180 self:modify("insert into rosterusers (username,jid,nick,subscription,ask,askmessage,server,subscribe) values (?, ?, ?, ?, ?. ''. ''. '')", user, jid, "", "N", "I"); | |
| 181 end | |
| 182 end | |
| 183 local version = data[false] and data[false].version; | |
| 184 if version then | |
| 185 self:modify("insert into roster_version (username,version) values (?, ?)", user, version); | |
| 186 end | |
| 187 return true; | |
| 188 else | |
| 189 self:modify("delete from rosterusers where username=?", user); | |
| 190 self:modify("delete from rostergroups where username=?", user); | |
| 191 self:modify("delete from roster_version where username=?", user); | |
| 192 end | |
| 193 end; | |
| 194 }; | |
| 195 | |
| 196 ----------------------------- | |
| 197 local driver = {}; | |
| 198 driver.__index = driver; | |
| 199 | |
| 200 function driver:query(sql, ...) | |
| 201 local stmt,err = prepare(sql); | |
| 202 if not stmt then | |
| 203 module:log("error", "Failed to prepare SQL [[%s]], error: %s", sql, err); | |
| 204 return nil, err; | |
| 205 end | |
| 206 local ok, err = stmt:execute(...); | |
| 207 if not ok then | |
| 208 module:log("error", "Failed to execute SQL [[%s]], error: %s", sql, err); | |
| 209 return nil, err; | |
| 210 end | |
| 211 return stmt; | |
| 212 end | |
| 213 function driver:modify(sql, ...) | |
| 214 local stmt, err = self:query(sql, ...); | |
| 215 if stmt and stmt:affected() > 0 then return stmt; end | |
| 216 return nil, err; | |
| 217 end | |
| 218 | |
| 219 function driver:open(datastore, typ) | |
| 220 local instance = setmetatable({ host = module.host, datastore = datastore }, self); | |
| 221 local handler = handlers[datastore]; | |
| 222 if not handler then return nil; end | |
| 223 for key,val in pairs(handler) do | |
| 224 instance[key] = val; | |
| 225 end | |
| 226 if instance.init then instance:init(); end | |
| 227 return instance; | |
| 228 end | |
| 229 | |
| 230 ----------------------------- | |
| 231 | |
| 232 module:add_item("data-driver", driver); |