forked from mailru/tntlua
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmrimoff.lua
99 lines (83 loc) · 2.5 KB
/
mrimoff.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
-- mrimoff.lua
--
-- Stored procedures for Mail.Ru Agent offline messages storage.
--
--
-- index0 - { userid, msgid } (TREE, unique)
--
local function get_key_cardinality_and_max_msgid(index_no, ...)
local key = ...
-- TODO: optimize (replace by calculation cardinality of a key in the index)
local data = { box.select(0, index_no, key) }
local n = #data
if n == 0 then
return 0, 0
end
return n, box.unpack('i', data[n][1])
end
--
-- Add offline message @msg for user @userid.
-- If storage already contains @limit messages do nothing.
-- Returns tuple { "number of messages for this user", "id for added message" }
-- "id for added message" == 0 if no message was added.
--
function mrim_add(userid, msg)
-- client sends integers encoded as BER-strings
userid = box.unpack('i', userid)
local limit = 1000
while true do
local n_msgs, max_msgid = get_key_cardinality_and_max_msgid(0, userid)
if n_msgs >= limit then
return { box.pack('i', n_msgs), box.pack('i', 0) }
end
local msgid = max_msgid + 1
local status, result = pcall(box.insert, 0, userid, msgid, msg)
if status then
return { box.pack('i', n_msgs + 1), box.pack('i', msgid) }
else
--exception
box.fiber.sleep(0.001)
end
end
end
--
-- Delete offline message with id @msgid for user @userid.
-- Returns flag of deletion success.
--
function mrim_del(userid, msgid)
-- client sends integers encoded as BER-strings
userid = box.unpack('i', userid)
msgid = box.unpack('i', msgid)
local del = box.delete(0, userid, msgid)
if del ~= nil then
return box.pack('i', 1)
end
return box.pack('i', 0)
end
--
-- Delete all offline messages for user @userid.
-- Returns flag of deletion success.
--
function mrim_del_all(userid)
-- client sends integers encoded as BER-strings
userid = box.unpack('i', userid)
local msgs = { box.select(0, 0, userid) }
for _, msg in pairs(msgs) do
box.delete(0, msg[0], msg[1])
end
return box.pack('i', #msgs)
end
--
-- Get no more then @limit messages for user @userid
-- sorted ascending by addition order.
-- Returns tuple { "total number of messages for user" }
-- followed by tuples with requested user messages.
--
function mrim_get(userid, limit)
-- client sends integers encoded as BER-strings
userid = box.unpack('i', userid)
limit = box.unpack('i', limit)
-- TODO: use one select request for calculation of n_msgs and getting no more then @limit msgs
local n_msgs, _ = get_key_cardinality_and_max_msgid(0, userid)
return box.pack('i', n_msgs), box.select_limit(0, 0, 0, limit, userid)
end