util.hashring: Support associating arbitrary data with nodes

In this API, a 'node' is always a simple text string. Sometimes the caller may
have a more complex structure representing a node, but the hash ring is really
only concerned with the node's name.

This API change allows :add_nodes() to take a table of `node_name = value`
pairs, as well as the simple array of node names previously accepted.

The 'value' of the selected node is returned as a new second result from
:get_node().

If no value is passed when a node is added, it defaults to `true` (as before,
but this was never previously exposed).
This commit is contained in:
Matthew Wild 2022-12-02 20:32:36 +00:00
parent d33904f7e9
commit ae84717255
2 changed files with 30 additions and 9 deletions

View file

@ -83,4 +83,11 @@ describe("util.hashring", function ()
end end
end); end);
it("should support values associated with nodes", function ()
local r = hashring.new(128, sha256);
r:add_node("node1", { a = 1 });
local node, value = r:get_node("foo");
assert.is_equal("node1", node);
assert.same({ a = 1 }, value);
end);
end); end);

View file

@ -1,3 +1,5 @@
local it = require "util.iterators";
local function generate_ring(nodes, num_replicas, hash) local function generate_ring(nodes, num_replicas, hash)
local new_ring = {}; local new_ring = {};
for _, node_name in ipairs(nodes) do for _, node_name in ipairs(nodes) do
@ -28,18 +30,22 @@ local function new(num_replicas, hash_function)
return setmetatable({ nodes = {}, num_replicas = num_replicas, hash = hash_function }, hashring_mt); return setmetatable({ nodes = {}, num_replicas = num_replicas, hash = hash_function }, hashring_mt);
end; end;
function hashring_methods:add_node(name) function hashring_methods:add_node(name, value)
self.ring = nil; self.ring = nil;
self.nodes[name] = true; self.nodes[name] = value == nil and true or value;
table.insert(self.nodes, name); table.insert(self.nodes, name);
return true; return true;
end end
function hashring_methods:add_nodes(nodes) function hashring_methods:add_nodes(nodes)
self.ring = nil; self.ring = nil;
for _, node_name in ipairs(nodes) do local iter = pairs;
if not self.nodes[node_name] then if nodes[1] then -- simple array?
self.nodes[node_name] = true; iter = it.values;
end
for node_name, node_value in iter(nodes) do
if self.nodes[node_name] == nil then
self.nodes[node_name] = node_value == nil and true or node_value;
table.insert(self.nodes, node_name); table.insert(self.nodes, node_name);
end end
end end
@ -48,7 +54,7 @@ end
function hashring_methods:remove_node(node_name) function hashring_methods:remove_node(node_name)
self.ring = nil; self.ring = nil;
if self.nodes[node_name] then if self.nodes[node_name] ~= nil then
for i, stored_node_name in ipairs(self.nodes) do for i, stored_node_name in ipairs(self.nodes) do
if node_name == stored_node_name then if node_name == stored_node_name then
self.nodes[node_name] = nil; self.nodes[node_name] = nil;
@ -69,18 +75,26 @@ end
function hashring_methods:clone() function hashring_methods:clone()
local clone_hashring = new(self.num_replicas, self.hash); local clone_hashring = new(self.num_replicas, self.hash);
clone_hashring:add_nodes(self.nodes); for node_name, node_value in pairs(self.nodes) do
clone_hashring.nodes[node_name] = node_value;
end
clone_hashring.ring = nil;
return clone_hashring; return clone_hashring;
end end
function hashring_methods:get_node(key) function hashring_methods:get_node(key)
local node;
local key_hash = self.hash(key); local key_hash = self.hash(key);
for _, replica_hash in ipairs(self.ring) do for _, replica_hash in ipairs(self.ring) do
if key_hash < replica_hash then if key_hash < replica_hash then
return self.ring[replica_hash]; node = self.ring[replica_hash];
break;
end end
end end
return self.ring[self.ring[1]]; if not node then
node = self.ring[self.ring[1]];
end
return node, self.nodes[node];
end end
return { return {