| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051 |
- package redis_test
- import (
- "context"
- "fmt"
- "net"
- "strconv"
- "strings"
- "sync"
- "time"
- "github.com/go-redis/redis/v7"
- "github.com/go-redis/redis/v7/internal/hashtag"
- . "github.com/onsi/ginkgo"
- . "github.com/onsi/gomega"
- )
- type clusterScenario struct {
- ports []string
- nodeIDs []string
- processes map[string]*redisProcess
- clients map[string]*redis.Client
- }
- func (s *clusterScenario) masters() []*redis.Client {
- result := make([]*redis.Client, 3)
- for pos, port := range s.ports[:3] {
- result[pos] = s.clients[port]
- }
- return result
- }
- func (s *clusterScenario) slaves() []*redis.Client {
- result := make([]*redis.Client, 3)
- for pos, port := range s.ports[3:] {
- result[pos] = s.clients[port]
- }
- return result
- }
- func (s *clusterScenario) addrs() []string {
- addrs := make([]string, len(s.ports))
- for i, port := range s.ports {
- addrs[i] = net.JoinHostPort("127.0.0.1", port)
- }
- return addrs
- }
- func (s *clusterScenario) clusterClientUnsafe(opt *redis.ClusterOptions) *redis.ClusterClient {
- opt.Addrs = s.addrs()
- return redis.NewClusterClient(opt)
- }
- func (s *clusterScenario) clusterClient(opt *redis.ClusterOptions) *redis.ClusterClient {
- client := s.clusterClientUnsafe(opt)
- err := eventually(func() error {
- if opt.ClusterSlots != nil {
- return nil
- }
- state, err := client.LoadState()
- if err != nil {
- return err
- }
- if !state.IsConsistent() {
- return fmt.Errorf("cluster state is not consistent")
- }
- return nil
- }, 30*time.Second)
- if err != nil {
- panic(err)
- }
- return client
- }
- func startCluster(scenario *clusterScenario) error {
- // Start processes and collect node ids
- for pos, port := range scenario.ports {
- process, err := startRedis(port, "--cluster-enabled", "yes")
- if err != nil {
- return err
- }
- client := redis.NewClient(&redis.Options{
- Addr: ":" + port,
- })
- info, err := client.ClusterNodes().Result()
- if err != nil {
- return err
- }
- scenario.processes[port] = process
- scenario.clients[port] = client
- scenario.nodeIDs[pos] = info[:40]
- }
- // Meet cluster nodes.
- for _, client := range scenario.clients {
- err := client.ClusterMeet("127.0.0.1", scenario.ports[0]).Err()
- if err != nil {
- return err
- }
- }
- // Bootstrap masters.
- slots := []int{0, 5000, 10000, 16384}
- for pos, master := range scenario.masters() {
- err := master.ClusterAddSlotsRange(slots[pos], slots[pos+1]-1).Err()
- if err != nil {
- return err
- }
- }
- // Bootstrap slaves.
- for idx, slave := range scenario.slaves() {
- masterID := scenario.nodeIDs[idx]
- // Wait until master is available
- err := eventually(func() error {
- s := slave.ClusterNodes().Val()
- wanted := masterID
- if !strings.Contains(s, wanted) {
- return fmt.Errorf("%q does not contain %q", s, wanted)
- }
- return nil
- }, 10*time.Second)
- if err != nil {
- return err
- }
- err = slave.ClusterReplicate(masterID).Err()
- if err != nil {
- return err
- }
- }
- // Wait until all nodes have consistent info.
- wanted := []redis.ClusterSlot{{
- Start: 0,
- End: 4999,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8220",
- }, {
- ID: "",
- Addr: "127.0.0.1:8223",
- }},
- }, {
- Start: 5000,
- End: 9999,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8221",
- }, {
- ID: "",
- Addr: "127.0.0.1:8224",
- }},
- }, {
- Start: 10000,
- End: 16383,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8222",
- }, {
- ID: "",
- Addr: "127.0.0.1:8225",
- }},
- }}
- for _, client := range scenario.clients {
- err := eventually(func() error {
- res, err := client.ClusterSlots().Result()
- if err != nil {
- return err
- }
- return assertSlotsEqual(res, wanted)
- }, 30*time.Second)
- if err != nil {
- return err
- }
- }
- return nil
- }
- func assertSlotsEqual(slots, wanted []redis.ClusterSlot) error {
- outerLoop:
- for _, s2 := range wanted {
- for _, s1 := range slots {
- if slotEqual(s1, s2) {
- continue outerLoop
- }
- }
- return fmt.Errorf("%v not found in %v", s2, slots)
- }
- return nil
- }
- func slotEqual(s1, s2 redis.ClusterSlot) bool {
- if s1.Start != s2.Start {
- return false
- }
- if s1.End != s2.End {
- return false
- }
- if len(s1.Nodes) != len(s2.Nodes) {
- return false
- }
- for i, n1 := range s1.Nodes {
- if n1.Addr != s2.Nodes[i].Addr {
- return false
- }
- }
- return true
- }
- func stopCluster(scenario *clusterScenario) error {
- for _, client := range scenario.clients {
- if err := client.Close(); err != nil {
- return err
- }
- }
- for _, process := range scenario.processes {
- if err := process.Close(); err != nil {
- return err
- }
- }
- return nil
- }
- //------------------------------------------------------------------------------
- var _ = Describe("ClusterClient", func() {
- var failover bool
- var opt *redis.ClusterOptions
- var client *redis.ClusterClient
- assertClusterClient := func() {
- It("supports WithContext", func() {
- c, cancel := context.WithCancel(context.Background())
- cancel()
- err := client.WithContext(c).Ping().Err()
- Expect(err).To(MatchError("context canceled"))
- })
- It("should GET/SET/DEL", func() {
- err := client.Get("A").Err()
- Expect(err).To(Equal(redis.Nil))
- err = client.Set("A", "VALUE", 0).Err()
- Expect(err).NotTo(HaveOccurred())
- Eventually(func() string {
- return client.Get("A").Val()
- }, 30*time.Second).Should(Equal("VALUE"))
- cnt, err := client.Del("A").Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(cnt).To(Equal(int64(1)))
- })
- It("GET follows redirects", func() {
- err := client.Set("A", "VALUE", 0).Err()
- Expect(err).NotTo(HaveOccurred())
- if !failover {
- Eventually(func() int64 {
- nodes, err := client.Nodes("A")
- if err != nil {
- return 0
- }
- return nodes[1].Client.DBSize().Val()
- }, 30*time.Second).Should(Equal(int64(1)))
- Eventually(func() error {
- return client.SwapNodes("A")
- }, 30*time.Second).ShouldNot(HaveOccurred())
- }
- v, err := client.Get("A").Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(v).To(Equal("VALUE"))
- })
- It("SET follows redirects", func() {
- if !failover {
- Eventually(func() error {
- return client.SwapNodes("A")
- }, 30*time.Second).ShouldNot(HaveOccurred())
- }
- err := client.Set("A", "VALUE", 0).Err()
- Expect(err).NotTo(HaveOccurred())
- v, err := client.Get("A").Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(v).To(Equal("VALUE"))
- })
- It("distributes keys", func() {
- for i := 0; i < 100; i++ {
- err := client.Set(fmt.Sprintf("key%d", i), "value", 0).Err()
- Expect(err).NotTo(HaveOccurred())
- }
- client.ForEachMaster(func(master *redis.Client) error {
- defer GinkgoRecover()
- Eventually(func() string {
- return master.Info("keyspace").Val()
- }, 30*time.Second).Should(Or(
- ContainSubstring("keys=31"),
- ContainSubstring("keys=29"),
- ContainSubstring("keys=40"),
- ))
- return nil
- })
- })
- 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(client, []string{key}, "value").Err()
- Expect(err).NotTo(HaveOccurred())
- }
- client.ForEachMaster(func(master *redis.Client) error {
- defer GinkgoRecover()
- Eventually(func() string {
- return master.Info("keyspace").Val()
- }, 30*time.Second).Should(Or(
- ContainSubstring("keys=31"),
- ContainSubstring("keys=29"),
- ContainSubstring("keys=40"),
- ))
- return nil
- })
- })
- It("supports Watch", func() {
- var incr func(string) error
- // Transactionally increments key using GET and SET commands.
- incr = func(key string) error {
- err := client.Watch(func(tx *redis.Tx) error {
- n, err := tx.Get(key).Int64()
- if err != nil && err != redis.Nil {
- return err
- }
- _, err = tx.TxPipelined(func(pipe redis.Pipeliner) error {
- pipe.Set(key, strconv.FormatInt(n+1, 10), 0)
- return nil
- })
- return err
- }, key)
- if err == redis.TxFailedErr {
- return incr(key)
- }
- return err
- }
- var wg sync.WaitGroup
- for i := 0; i < 100; i++ {
- wg.Add(1)
- go func() {
- defer GinkgoRecover()
- defer wg.Done()
- err := incr("key")
- Expect(err).NotTo(HaveOccurred())
- }()
- }
- wg.Wait()
- Eventually(func() string {
- return client.Get("key").Val()
- }, 30*time.Second).Should(Equal("100"))
- })
- Describe("pipelining", func() {
- var pipe *redis.Pipeline
- assertPipeline := func() {
- keys := []string{"A", "B", "C", "D", "E", "F", "G"}
- It("follows redirects", func() {
- if !failover {
- for _, key := range keys {
- Eventually(func() error {
- return client.SwapNodes(key)
- }, 30*time.Second).ShouldNot(HaveOccurred())
- }
- }
- for i, key := range keys {
- pipe.Set(key, key+"_value", 0)
- pipe.Expire(key, time.Duration(i+1)*time.Hour)
- }
- cmds, err := pipe.Exec()
- Expect(err).NotTo(HaveOccurred())
- Expect(cmds).To(HaveLen(14))
- _ = client.ForEachNode(func(node *redis.Client) error {
- defer GinkgoRecover()
- Eventually(func() int64 {
- return node.DBSize().Val()
- }, 30*time.Second).ShouldNot(BeZero())
- return nil
- })
- if !failover {
- for _, key := range keys {
- Eventually(func() error {
- return client.SwapNodes(key)
- }, 30*time.Second).ShouldNot(HaveOccurred())
- }
- }
- for _, key := range keys {
- pipe.Get(key)
- pipe.TTL(key)
- }
- cmds, err = pipe.Exec()
- Expect(err).NotTo(HaveOccurred())
- Expect(cmds).To(HaveLen(14))
- for i, key := range keys {
- get := cmds[i*2].(*redis.StringCmd)
- Expect(get.Val()).To(Equal(key + "_value"))
- ttl := cmds[(i*2)+1].(*redis.DurationCmd)
- dur := time.Duration(i+1) * time.Hour
- Expect(ttl.Val()).To(BeNumerically("~", dur, 30*time.Second))
- }
- })
- It("works with missing keys", func() {
- pipe.Set("A", "A_value", 0)
- pipe.Set("C", "C_value", 0)
- _, err := pipe.Exec()
- Expect(err).NotTo(HaveOccurred())
- a := pipe.Get("A")
- b := pipe.Get("B")
- c := pipe.Get("C")
- cmds, err := pipe.Exec()
- Expect(err).To(Equal(redis.Nil))
- Expect(cmds).To(HaveLen(3))
- Expect(a.Err()).NotTo(HaveOccurred())
- Expect(a.Val()).To(Equal("A_value"))
- Expect(b.Err()).To(Equal(redis.Nil))
- Expect(b.Val()).To(Equal(""))
- Expect(c.Err()).NotTo(HaveOccurred())
- Expect(c.Val()).To(Equal("C_value"))
- })
- }
- Describe("with Pipeline", func() {
- BeforeEach(func() {
- pipe = client.Pipeline().(*redis.Pipeline)
- })
- AfterEach(func() {
- Expect(pipe.Close()).NotTo(HaveOccurred())
- })
- assertPipeline()
- })
- Describe("with TxPipeline", func() {
- BeforeEach(func() {
- pipe = client.TxPipeline().(*redis.Pipeline)
- })
- AfterEach(func() {
- Expect(pipe.Close()).NotTo(HaveOccurred())
- })
- assertPipeline()
- })
- })
- It("supports PubSub", func() {
- pubsub := client.Subscribe("mychannel")
- defer pubsub.Close()
- Eventually(func() error {
- _, err := client.Publish("mychannel", "hello").Result()
- if err != nil {
- return err
- }
- msg, err := pubsub.ReceiveTimeout(time.Second)
- if err != nil {
- return err
- }
- _, ok := msg.(*redis.Message)
- if !ok {
- return fmt.Errorf("got %T, wanted *redis.Message", msg)
- }
- return nil
- }, 30*time.Second).ShouldNot(HaveOccurred())
- })
- It("supports PubSub.Ping without channels", func() {
- pubsub := client.Subscribe()
- defer pubsub.Close()
- err := pubsub.Ping()
- Expect(err).NotTo(HaveOccurred())
- })
- }
- Describe("ClusterClient", func() {
- BeforeEach(func() {
- opt = redisClusterOptions()
- client = cluster.clusterClient(opt)
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- })
- AfterEach(func() {
- _ = client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- It("returns pool stats", func() {
- stats := client.PoolStats()
- Expect(stats).To(BeAssignableToTypeOf(&redis.PoolStats{}))
- })
- It("returns an error when there are no attempts left", func() {
- opt := redisClusterOptions()
- opt.MaxRedirects = -1
- client := cluster.clusterClient(opt)
- Eventually(func() error {
- return client.SwapNodes("A")
- }, 30*time.Second).ShouldNot(HaveOccurred())
- err := client.Get("A").Err()
- Expect(err).To(HaveOccurred())
- Expect(err.Error()).To(ContainSubstring("MOVED"))
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- It("calls fn for every master node", func() {
- for i := 0; i < 10; i++ {
- Expect(client.Set(strconv.Itoa(i), "", 0).Err()).NotTo(HaveOccurred())
- }
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- size, err := client.DBSize().Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(size).To(Equal(int64(0)))
- })
- It("should CLUSTER SLOTS", func() {
- res, err := client.ClusterSlots().Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(res).To(HaveLen(3))
- wanted := []redis.ClusterSlot{{
- Start: 0,
- End: 4999,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8220",
- }, {
- ID: "",
- Addr: "127.0.0.1:8223",
- }},
- }, {
- Start: 5000,
- End: 9999,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8221",
- }, {
- ID: "",
- Addr: "127.0.0.1:8224",
- }},
- }, {
- Start: 10000,
- End: 16383,
- Nodes: []redis.ClusterNode{{
- ID: "",
- Addr: "127.0.0.1:8222",
- }, {
- ID: "",
- Addr: "127.0.0.1:8225",
- }},
- }}
- Expect(assertSlotsEqual(res, wanted)).NotTo(HaveOccurred())
- })
- It("should CLUSTER NODES", func() {
- res, err := client.ClusterNodes().Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(len(res)).To(BeNumerically(">", 400))
- })
- It("should CLUSTER INFO", func() {
- res, err := client.ClusterInfo().Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(res).To(ContainSubstring("cluster_known_nodes:6"))
- })
- It("should CLUSTER KEYSLOT", func() {
- hashSlot, err := client.ClusterKeySlot("somekey").Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(hashSlot).To(Equal(int64(hashtag.Slot("somekey"))))
- })
- It("should CLUSTER GETKEYSINSLOT", func() {
- keys, err := client.ClusterGetKeysInSlot(hashtag.Slot("somekey"), 1).Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(len(keys)).To(Equal(0))
- })
- It("should CLUSTER COUNT-FAILURE-REPORTS", func() {
- n, err := client.ClusterCountFailureReports(cluster.nodeIDs[0]).Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(n).To(Equal(int64(0)))
- })
- It("should CLUSTER COUNTKEYSINSLOT", func() {
- n, err := client.ClusterCountKeysInSlot(10).Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(n).To(Equal(int64(0)))
- })
- It("should CLUSTER SAVECONFIG", func() {
- res, err := client.ClusterSaveConfig().Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(res).To(Equal("OK"))
- })
- It("should CLUSTER SLAVES", func() {
- nodesList, err := client.ClusterSlaves(cluster.nodeIDs[0]).Result()
- Expect(err).NotTo(HaveOccurred())
- Expect(nodesList).Should(ContainElement(ContainSubstring("slave")))
- Expect(nodesList).Should(HaveLen(1))
- })
- It("should RANDOMKEY", func() {
- const nkeys = 100
- for i := 0; i < nkeys; i++ {
- err := client.Set(fmt.Sprintf("key%d", i), "value", 0).Err()
- Expect(err).NotTo(HaveOccurred())
- }
- var keys []string
- addKey := func(key string) {
- for _, k := range keys {
- if k == key {
- return
- }
- }
- keys = append(keys, key)
- }
- for i := 0; i < nkeys*10; i++ {
- key := client.RandomKey().Val()
- addKey(key)
- }
- Expect(len(keys)).To(BeNumerically("~", nkeys, nkeys/10))
- })
- assertClusterClient()
- })
- Describe("ClusterClient failover", func() {
- BeforeEach(func() {
- failover = true
- opt = redisClusterOptions()
- opt.MinRetryBackoff = 250 * time.Millisecond
- opt.MaxRetryBackoff = time.Second
- client = cluster.clusterClient(opt)
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- err = client.ForEachSlave(func(slave *redis.Client) error {
- defer GinkgoRecover()
- Eventually(func() int64 {
- return slave.DBSize().Val()
- }, "30s").Should(Equal(int64(0)))
- return nil
- })
- Expect(err).NotTo(HaveOccurred())
- state, err := client.LoadState()
- Eventually(func() bool {
- state, err = client.LoadState()
- if err != nil {
- return false
- }
- return state.IsConsistent()
- }, "30s").Should(BeTrue())
- for _, slave := range state.Slaves {
- err = slave.Client.ClusterFailover().Err()
- Expect(err).NotTo(HaveOccurred())
- Eventually(func() bool {
- state, _ := client.LoadState()
- return state.IsConsistent()
- }, "30s").Should(BeTrue())
- }
- })
- AfterEach(func() {
- failover = false
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- assertClusterClient()
- })
- Describe("ClusterClient with RouteByLatency", func() {
- BeforeEach(func() {
- opt = redisClusterOptions()
- opt.RouteByLatency = true
- client = cluster.clusterClient(opt)
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- err = client.ForEachSlave(func(slave *redis.Client) error {
- Eventually(func() int64 {
- return client.DBSize().Val()
- }, 30*time.Second).Should(Equal(int64(0)))
- return nil
- })
- Expect(err).NotTo(HaveOccurred())
- })
- AfterEach(func() {
- err := client.ForEachSlave(func(slave *redis.Client) error {
- return slave.ReadWrite().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- err = client.Close()
- Expect(err).NotTo(HaveOccurred())
- })
- assertClusterClient()
- })
- Describe("ClusterClient with ClusterSlots", func() {
- BeforeEach(func() {
- failover = true
- opt = redisClusterOptions()
- opt.ClusterSlots = func() ([]redis.ClusterSlot, error) {
- slots := []redis.ClusterSlot{{
- Start: 0,
- End: 4999,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard1Port,
- }},
- }, {
- Start: 5000,
- End: 9999,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard2Port,
- }},
- }, {
- Start: 10000,
- End: 16383,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard3Port,
- }},
- }}
- return slots, nil
- }
- client = cluster.clusterClient(opt)
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- err = client.ForEachSlave(func(slave *redis.Client) error {
- Eventually(func() int64 {
- return client.DBSize().Val()
- }, 30*time.Second).Should(Equal(int64(0)))
- return nil
- })
- Expect(err).NotTo(HaveOccurred())
- })
- AfterEach(func() {
- failover = false
- err := client.Close()
- Expect(err).NotTo(HaveOccurred())
- })
- assertClusterClient()
- })
- Describe("ClusterClient with RouteRandomly and ClusterSlots", func() {
- BeforeEach(func() {
- failover = true
- opt = redisClusterOptions()
- opt.RouteRandomly = true
- opt.ClusterSlots = func() ([]redis.ClusterSlot, error) {
- slots := []redis.ClusterSlot{{
- Start: 0,
- End: 4999,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard1Port,
- }},
- }, {
- Start: 5000,
- End: 9999,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard2Port,
- }},
- }, {
- Start: 10000,
- End: 16383,
- Nodes: []redis.ClusterNode{{
- Addr: ":" + ringShard3Port,
- }},
- }}
- return slots, nil
- }
- client = cluster.clusterClient(opt)
- err := client.ForEachMaster(func(master *redis.Client) error {
- return master.FlushDB().Err()
- })
- Expect(err).NotTo(HaveOccurred())
- err = client.ForEachSlave(func(slave *redis.Client) error {
- Eventually(func() int64 {
- return client.DBSize().Val()
- }, 30*time.Second).Should(Equal(int64(0)))
- return nil
- })
- Expect(err).NotTo(HaveOccurred())
- })
- AfterEach(func() {
- failover = false
- err := client.Close()
- Expect(err).NotTo(HaveOccurred())
- })
- assertClusterClient()
- })
- })
- var _ = Describe("ClusterClient without nodes", func() {
- var client *redis.ClusterClient
- BeforeEach(func() {
- client = redis.NewClusterClient(&redis.ClusterOptions{})
- })
- AfterEach(func() {
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- It("Ping returns an error", func() {
- err := client.Ping().Err()
- Expect(err).To(MatchError("redis: cluster has no nodes"))
- })
- It("pipeline returns an error", func() {
- _, err := client.Pipelined(func(pipe redis.Pipeliner) error {
- pipe.Ping()
- return nil
- })
- Expect(err).To(MatchError("redis: cluster has no nodes"))
- })
- })
- var _ = Describe("ClusterClient without valid nodes", func() {
- var client *redis.ClusterClient
- BeforeEach(func() {
- client = redis.NewClusterClient(&redis.ClusterOptions{
- Addrs: []string{redisAddr},
- })
- })
- AfterEach(func() {
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- It("returns an error", func() {
- err := client.Ping().Err()
- Expect(err).To(MatchError("ERR This instance has cluster support disabled"))
- })
- It("pipeline returns an error", func() {
- _, err := client.Pipelined(func(pipe redis.Pipeliner) error {
- pipe.Ping()
- return nil
- })
- Expect(err).To(MatchError("ERR This instance has cluster support disabled"))
- })
- })
- var _ = Describe("ClusterClient with unavailable Cluster", func() {
- var client *redis.ClusterClient
- BeforeEach(func() {
- for _, node := range cluster.clients {
- err := node.ClientPause(5 * time.Second).Err()
- Expect(err).NotTo(HaveOccurred())
- }
- opt := redisClusterOptions()
- opt.ReadTimeout = 250 * time.Millisecond
- opt.WriteTimeout = 250 * time.Millisecond
- opt.MaxRedirects = 1
- client = cluster.clusterClientUnsafe(opt)
- })
- AfterEach(func() {
- Expect(client.Close()).NotTo(HaveOccurred())
- })
- It("recovers when Cluster recovers", func() {
- err := client.Ping().Err()
- Expect(err).To(HaveOccurred())
- Eventually(func() error {
- return client.Ping().Err()
- }, "30s").ShouldNot(HaveOccurred())
- })
- })
- var _ = Describe("ClusterClient timeout", func() {
- var client *redis.ClusterClient
- AfterEach(func() {
- _ = client.Close()
- })
- testTimeout := func() {
- It("Ping timeouts", func() {
- err := client.Ping().Err()
- Expect(err).To(HaveOccurred())
- Expect(err.(net.Error).Timeout()).To(BeTrue())
- })
- It("Pipeline timeouts", func() {
- _, err := client.Pipelined(func(pipe redis.Pipeliner) error {
- pipe.Ping()
- return nil
- })
- Expect(err).To(HaveOccurred())
- Expect(err.(net.Error).Timeout()).To(BeTrue())
- })
- It("Tx timeouts", func() {
- err := client.Watch(func(tx *redis.Tx) error {
- return tx.Ping().Err()
- }, "foo")
- Expect(err).To(HaveOccurred())
- Expect(err.(net.Error).Timeout()).To(BeTrue())
- })
- It("Tx Pipeline timeouts", func() {
- err := client.Watch(func(tx *redis.Tx) error {
- _, err := tx.TxPipelined(func(pipe redis.Pipeliner) error {
- pipe.Ping()
- return nil
- })
- return err
- }, "foo")
- Expect(err).To(HaveOccurred())
- Expect(err.(net.Error).Timeout()).To(BeTrue())
- })
- }
- const pause = 5 * time.Second
- Context("read/write timeout", func() {
- BeforeEach(func() {
- opt := redisClusterOptions()
- opt.ReadTimeout = 250 * time.Millisecond
- opt.WriteTimeout = 250 * time.Millisecond
- opt.MaxRedirects = 1
- client = cluster.clusterClient(opt)
- err := client.ForEachNode(func(client *redis.Client) error {
- return client.ClientPause(pause).Err()
- })
- Expect(err).NotTo(HaveOccurred())
- })
- AfterEach(func() {
- _ = client.ForEachNode(func(client *redis.Client) error {
- defer GinkgoRecover()
- Eventually(func() error {
- return client.Ping().Err()
- }, 2*pause).ShouldNot(HaveOccurred())
- return nil
- })
- })
- testTimeout()
- })
- })
|