-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathZSPLIT.lua
41 lines (33 loc) · 1.42 KB
/
ZSPLIT.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
-- @desc: Split a ZSET in two via round-robin
-- @usage: redis-cli eval "$(cat ZSPLIT)" 2 <originalZset> <newZset>
-- @return: list of moved members
local function ZSPLIT(zset1, zset2)
local zset1size = redis.call("ZCARD", zset1)
if zset1size == 0 then return {} end
local members = redis.call("ZREVRANGEBYSCORE", zset1, "+INF", "-INF", "WITHSCORES")
local tmpMembers = {}
local tmpScores = {}
for idx,val in ipairs(members) do
if idx % 2 == 0 then table.insert(tmpScores, val)
else table.insert(tmpMembers, val) end
if idx % 4 == 0 then
redis.call("ZREM", zset1, tmpMembers[#tmpMembers])
redis.call("ZADD", zset2, tonumber(tmpScores[#tmpScores]), tmpMembers[#tmpMembers])
end
end
return tmpMembers
end
--[[ @TEST
redis.call("DEL", "test_zsplit1", "test_zsplit2")
redis.call("ZADD", "test_zsplit1", 20, "num1", 30, "num2", 40, "num3", 50, "num4")
local movedMembers = ZSPLIT("test_zsplit1", "test_zsplit2")
assert(movedMembers[1] == "num4")
assert(redis.call("ZCARD", "test_zsplit1") == 2)
assert(redis.call("ZCARD", "test_zsplit2") == 2)
assert(redis.call("ZREVRANGE", "test_zsplit1", 0, 1)[1] == "num4")
assert(redis.call("ZREVRANGE", "test_zsplit1", 0, 1)[2] == "num2")
assert(redis.call("ZREVRANGE", "test_zsplit2", 0, 1)[1] == "num3")
assert(redis.call("ZREVRANGE", "test_zsplit2", 0, 1)[2] == "num1")
redis.call("DEL", "test_zsplit1", "test_zsplit2")
--]]
return ZSPLIT(KEYS[1], KEYS[2])