redis/ring_test.go

194 lines
4.8 KiB
Go
Raw Normal View History

2015-05-25 16:22:27 +03:00
package redis_test
import (
"crypto/rand"
2015-05-25 16:22:27 +03:00
"fmt"
"time"
2017-02-18 17:42:34 +03:00
"github.com/go-redis/redis"
2015-05-25 16:22:27 +03:00
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
)
var _ = Describe("Redis Ring", func() {
2016-08-09 16:32:08 +03:00
const heartbeat = 100 * time.Millisecond
2015-05-25 16:22:27 +03:00
var ring *redis.Ring
setRingKeys := func() {
for i := 0; i < 100; i++ {
err := ring.Set(fmt.Sprintf("key%d", i), "value", 0).Err()
Expect(err).NotTo(HaveOccurred())
}
}
BeforeEach(func() {
opt := redisRingOptions()
opt.HeartbeatFrequency = heartbeat
ring = redis.NewRing(opt)
2015-05-25 16:22:27 +03:00
err := ring.ForEachShard(func(cl *redis.Client) error {
return cl.FlushDb().Err()
})
Expect(err).NotTo(HaveOccurred())
2015-05-25 16:22:27 +03:00
})
AfterEach(func() {
Expect(ring.Close()).NotTo(HaveOccurred())
})
2016-10-09 14:12:32 +03:00
It("distributes keys", func() {
2015-05-25 16:22:27 +03:00
setRingKeys()
// Both shards should have some keys now.
Expect(ringShard1.Info().Val()).To(ContainSubstring("keys=57"))
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
})
2016-10-09 14:12:32 +03:00
It("distributes keys when using EVAL", func() {
script := redis.NewScript(`
local r = redis.call('SET', KEYS[1], ARGV[1])
return r
`)
var key string
for i := 0; i < 100; i++ {
key = fmt.Sprintf("key%d", i)
err := script.Run(ring, []string{key}, "value").Err()
Expect(err).NotTo(HaveOccurred())
}
Expect(ringShard1.Info().Val()).To(ContainSubstring("keys=57"))
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
})
2015-12-22 12:44:49 +03:00
It("uses single shard when one of the shards is down", func() {
2015-05-25 16:22:27 +03:00
// Stop ringShard2.
Expect(ringShard2.Close()).NotTo(HaveOccurred())
2016-08-09 16:32:08 +03:00
// Ring needs 3 * heartbeat time to detect that node is down.
2015-05-25 16:22:27 +03:00
// Give it more to be sure.
2016-08-09 16:32:08 +03:00
time.Sleep(2 * 3 * heartbeat)
2015-05-25 16:22:27 +03:00
setRingKeys()
// RingShard1 should have all keys.
Expect(ringShard1.Info().Val()).To(ContainSubstring("keys=100"))
// Start ringShard2.
var err error
ringShard2, err = startRedis(ringShard2Port)
Expect(err).NotTo(HaveOccurred())
// Wait for ringShard2 to come up.
Eventually(func() error {
return ringShard2.Ping().Err()
}, "1s").ShouldNot(HaveOccurred())
// Ring needs heartbeat time to detect that node is up.
// Give it more to be sure.
time.Sleep(heartbeat + heartbeat)
setRingKeys()
// RingShard2 should have its keys.
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
})
2015-06-04 11:50:24 +03:00
It("supports hash tags", func() {
for i := 0; i < 100; i++ {
err := ring.Set(fmt.Sprintf("key%d{tag}", i), "value", 0).Err()
Expect(err).NotTo(HaveOccurred())
}
Expect(ringShard1.Info().Val()).ToNot(ContainSubstring("keys="))
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=100"))
})
2016-10-09 14:12:32 +03:00
Describe("pipeline", func() {
It("distributes keys", func() {
2015-06-24 13:41:50 +03:00
pipe := ring.Pipeline()
for i := 0; i < 100; i++ {
err := pipe.Set(fmt.Sprintf("key%d", i), "value", 0).Err()
Expect(err).NotTo(HaveOccurred())
}
cmds, err := pipe.Exec()
2015-06-04 11:50:24 +03:00
Expect(err).NotTo(HaveOccurred())
2015-06-24 13:41:50 +03:00
Expect(cmds).To(HaveLen(100))
Expect(pipe.Close()).NotTo(HaveOccurred())
2015-06-04 11:50:24 +03:00
2015-06-24 13:41:50 +03:00
for _, cmd := range cmds {
Expect(cmd.Err()).NotTo(HaveOccurred())
Expect(cmd.(*redis.StatusCmd).Val()).To(Equal("OK"))
}
2015-06-04 11:50:24 +03:00
2015-06-24 13:41:50 +03:00
// Both shards should have some keys now.
Expect(ringShard1.Info().Val()).To(ContainSubstring("keys=57"))
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=43"))
})
It("is consistent with ring", func() {
var keys []string
for i := 0; i < 100; i++ {
key := make([]byte, 64)
_, err := rand.Read(key)
Expect(err).NotTo(HaveOccurred())
keys = append(keys, string(key))
}
2017-05-02 18:00:53 +03:00
_, err := ring.Pipelined(func(pipe redis.Pipeliner) error {
for _, key := range keys {
pipe.Set(key, "value", 0).Err()
}
2015-06-24 13:41:50 +03:00
return nil
})
Expect(err).NotTo(HaveOccurred())
for _, key := range keys {
val, err := ring.Get(key).Result()
Expect(err).NotTo(HaveOccurred())
Expect(val).To(Equal("value"))
}
})
It("supports hash tags", func() {
2017-05-02 18:00:53 +03:00
_, err := ring.Pipelined(func(pipe redis.Pipeliner) error {
for i := 0; i < 100; i++ {
pipe.Set(fmt.Sprintf("key%d{tag}", i), "value", 0).Err()
}
return nil
})
2015-06-24 13:41:50 +03:00
Expect(err).NotTo(HaveOccurred())
Expect(ringShard1.Info().Val()).ToNot(ContainSubstring("keys="))
Expect(ringShard2.Info().Val()).To(ContainSubstring("keys=100"))
2015-06-24 13:41:50 +03:00
})
2015-06-04 11:50:24 +03:00
})
2015-05-25 16:22:27 +03:00
})
2016-10-09 14:12:32 +03:00
var _ = Describe("empty Redis Ring", func() {
var ring *redis.Ring
BeforeEach(func() {
ring = redis.NewRing(&redis.RingOptions{})
})
AfterEach(func() {
Expect(ring.Close()).NotTo(HaveOccurred())
})
It("returns an error", func() {
err := ring.Ping().Err()
Expect(err).To(MatchError("redis: all ring shards are down"))
})
It("pipeline returns an error", func() {
2017-05-02 18:00:53 +03:00
_, err := ring.Pipelined(func(pipe redis.Pipeliner) error {
2016-10-09 14:12:32 +03:00
pipe.Ping()
return nil
})
Expect(err).To(MatchError("redis: all ring shards are down"))
})
})