mirror of
https://github.com/fluencelabs/redis
synced 2025-04-03 16:21:03 +00:00
Cluster: implement redis-trib fix for uncovered slots.
This commit is contained in:
parent
b58796f520
commit
04ae459bc2
@ -425,6 +425,7 @@ class RedisTrib
|
|||||||
def nodes_with_keys_in_slot(slot)
|
def nodes_with_keys_in_slot(slot)
|
||||||
nodes = []
|
nodes = []
|
||||||
@nodes.each{|n|
|
@nodes.each{|n|
|
||||||
|
next if n.has_flag?("slave")
|
||||||
nodes << n if n.r.cluster("getkeysinslot",slot,1).length > 0
|
nodes << n if n.r.cluster("getkeysinslot",slot,1).length > 0
|
||||||
}
|
}
|
||||||
nodes
|
nodes
|
||||||
@ -443,7 +444,7 @@ class RedisTrib
|
|||||||
not_covered.each{|slot|
|
not_covered.each{|slot|
|
||||||
nodes = nodes_with_keys_in_slot(slot)
|
nodes = nodes_with_keys_in_slot(slot)
|
||||||
slots[slot] = nodes
|
slots[slot] = nodes
|
||||||
xputs "Slot #{slot} has keys in #{nodes.length} nodes: #{nodes.join}"
|
xputs "Slot #{slot} has keys in #{nodes.length} nodes: #{nodes.join(", ")}"
|
||||||
}
|
}
|
||||||
|
|
||||||
none = slots.select {|k,v| v.length == 0}
|
none = slots.select {|k,v| v.length == 0}
|
||||||
@ -479,14 +480,20 @@ class RedisTrib
|
|||||||
xputs multi.keys.join(",")
|
xputs multi.keys.join(",")
|
||||||
yes_or_die "Fix these slots by moving keys into a single node?"
|
yes_or_die "Fix these slots by moving keys into a single node?"
|
||||||
multi.each{|slot,nodes|
|
multi.each{|slot,nodes|
|
||||||
xputs ">>> Covering slot #{slot} moving keys to #{nodes[0]}"
|
target = get_node_with_most_keys_in_slot(nodes,slot)
|
||||||
# TODO
|
xputs ">>> Covering slot #{slot} moving keys to #{target}"
|
||||||
# 1) Set all nodes as "MIGRATING" for this slot, so that we
|
|
||||||
# can access keys in the hash slot using ASKING.
|
target.r.cluster('addslots',slot)
|
||||||
# 2) Move everything to node[0]
|
target.r.cluster('setslot',slot,'stable')
|
||||||
# 3) Clear MIGRATING from nodes, and ADDSLOTS the slot to
|
nodes.each{|src|
|
||||||
# node[0].
|
next if src == target
|
||||||
raise "TODO: Work in progress"
|
# Set the source node in 'importing' state (even if we will
|
||||||
|
# actually migrate keys away) in order to avoid receiving
|
||||||
|
# redirections for MIGRATE.
|
||||||
|
src.r.cluster('setslot',slot,'importing',target.info[:name])
|
||||||
|
move_slot(src,target,slot,:dots=>true,:fix=>true,:cold=>true)
|
||||||
|
src.r.cluster('setslot',slot,'stable')
|
||||||
|
}
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -501,6 +508,22 @@ class RedisTrib
|
|||||||
nil
|
nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Return the node, among 'nodes' with the greatest number of keys
|
||||||
|
# in the specified slot.
|
||||||
|
def get_node_with_most_keys_in_slot(nodes,slot)
|
||||||
|
best = nil
|
||||||
|
best_numkeys = 0
|
||||||
|
@nodes.each{|n|
|
||||||
|
next if n.has_flag?("slave")
|
||||||
|
numkeys = n.r.cluster("countkeysinslot",slot)
|
||||||
|
if numkeys > best_numkeys || best == nil
|
||||||
|
best = n
|
||||||
|
best_numkeys = numkeys
|
||||||
|
end
|
||||||
|
}
|
||||||
|
return best
|
||||||
|
end
|
||||||
|
|
||||||
# Slot 'slot' was found to be in importing or migrating state in one or
|
# Slot 'slot' was found to be in importing or migrating state in one or
|
||||||
# more nodes. This function fixes this condition by migrating keys where
|
# more nodes. This function fixes this condition by migrating keys where
|
||||||
# it seems more sensible.
|
# it seems more sensible.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user