-
Notifications
You must be signed in to change notification settings - Fork 12
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Ring Client support #5
Comments
Hi @szuecs, We currently don't support the ring sharding because we already support the standard Cluster. I actually don't understand why to use the ring sharding over the standard Cluster. But I will still be grateful to accept a PR of ring client implementation. |
The problem is that with cluster you create a safe setup but not a scalable one. We need a non safe scalable system. https://redis.uptrace.dev/guide/ring.html#keys-distribution |
The standard cluster doesn't need to have replicas either. You can have a cluster that consists of only primary nodes.
That is an infrastructure failure that users should take very seriously. Normally, no one would like to run applications on such an unstable infrastructure. I believe even the users of the ring don't want to lose half of their nodes. So treating the ability to survive in such a case as a selling point seems unwise to me. But, since the ring is so simple, I am not against having a ring implementation in this library. Here are some tips if someone wants to implement it:
ring := valkey.NewRing(valkey.RingOption{
Shards: map[string]valkey.Client{
"shard1": client1,
"shard2": client2,
}
})
cmd := client.B().Get().Key(key).Build()
cmd.Slot() |
Sure, I just read the spec for "cluster" and tried to think about all possible advantages to answer your question. I think the biggest advantage is simplicity and more stable key distribution. Also in case of auto scaling crc16 mod N will move around all keys and as far as I understand rendezvous is a generalization of consistent hash so it should also have more stable shard selection in case of scaling attempts.
Thanks makes a lot of sense to me.
I will check autoscaling featureWhat do you think about vnode hashing featureFor 3. we would also need to expose an interface to override the hashing function similar to https://github.com/redis/go-redis/blob/b64d9deef330a51cfbd3e0425b6c26b27c1ee370/ring.go#L66 Maybe I should create separate issues for autoscaling feature and vnode hashing feature. What do you think about this? |
Yes, the value returned from
No need. We can address this in this issue. It looks like we need an interface to pick a shard, for example: type RingShardPicker func(cmd valkey.Completed, activeShards []string) string
No blocker, but I would prefer this one: func (*Ring) AddShard(shard string, client valkey.Client)
func (*Ring) DelShard(shard string) The reason is that I want to keep the implementation simple and avoid constructing valkey.Client in the implementation. |
As far as I see Slot(), just returns the already calculated |
We don't need to use the cmd := client.B().Get().Key(key).Build() // the crc16 is calculated here.
cmd.Slot() // we just need to retrieve it by using Slot(). I think we could implement the rendezvous hash like this: func RendezvousRingShardPicker(cmd valkey.Completed, activeShards []string) string {
slot := cmd.Slot()
sort.Slice(activeShards, func(i, j int) bool {
return (slot ^ hash(activeShards[i])) < (slot ^ hash(activeShards[j]))
})
return activeShards[0]
} |
I just reconsidered the To have spaces for performance optimization, we might still need an interface like this: type RingShardPicker interface {
AddShard(shard string)
DelShard(shard string)
Pick(cmd valkey.Completed) (shard string)
} This interface allows implementations to have their optimized structure for faster lookup. |
My thinking was more to replace crc16 by rendezvous. As far as I know double hashing is not very good, because it won't have the properties of the one hash function that you want to use. cmd := ringClient.B().Get().Key(key).Build() // rendezvous(key) is calculated here, maybe via Pick() from RingShardPicker
cmd.Slot() // does not change as before use the cached value
The interface seems to make a lot of sense, thanks! |
Hi @szuecs,
We can't change the crc16 calculation. The underlying If you want to re-calculate a hash completely, you can calculate it from the The rendezvous implementation used by go-redis also uses XOR to combine two hashes which, I believe, is similar to the RendezvousRingShardPicker I previously proposed. |
@rueian yes I see, thanks you are likely right. I will go with your proposed way. |
I would like to have a ring client support similar to go-redis ring.
It basically enables you to use shards to scale out redis.
There are a couple of feature that we would need in this ring client, that is likely good to know in advance:
In our case we use it for rate limits at the ingress layer, so we have a lot operations per second that will exceed a single redis process and require redis shards.
One key feature for us would be also to be able to update the shards like in https://github.com/redis/go-redis/blob/b64d9deef330a51cfbd3e0425b6c26b27c1ee370/ring.go#L538-L540 (redis/go-redis@6f7f800).
In some cases we need even to built a vnode layer on top to spread operations to more shards https://github.com/zalando/skipper/blob/master/net/redisclient.go#L218 , so it would be great if we could have an option to set also the ConsistentHash function to be used. In general rendezvous hash was the best option in my experience.
The text was updated successfully, but these errors were encountered: