123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262 |
- package cache
- import (
- "encoding/json"
- "fmt"
- "math"
- "strconv"
- "testing"
- "time"
- "github.com/stretchr/testify/assert"
- "github.com/tal-tech/go-zero/core/errorx"
- "github.com/tal-tech/go-zero/core/hash"
- "github.com/tal-tech/go-zero/core/stores/redis"
- "github.com/tal-tech/go-zero/core/stores/redis/redistest"
- "github.com/tal-tech/go-zero/core/syncx"
- )
- type mockedNode struct {
- vals map[string][]byte
- errNotFound error
- }
- func (mc *mockedNode) Del(keys ...string) error {
- var be errorx.BatchError
- for _, key := range keys {
- if _, ok := mc.vals[key]; !ok {
- be.Add(mc.errNotFound)
- } else {
- delete(mc.vals, key)
- }
- }
- return be.Err()
- }
- func (mc *mockedNode) Get(key string, v interface{}) error {
- bs, ok := mc.vals[key]
- if ok {
- return json.Unmarshal(bs, v)
- }
- return mc.errNotFound
- }
- func (mc *mockedNode) IsNotFound(err error) bool {
- return err == mc.errNotFound
- }
- func (mc *mockedNode) Set(key string, v interface{}) error {
- data, err := json.Marshal(v)
- if err != nil {
- return err
- }
- mc.vals[key] = data
- return nil
- }
- func (mc *mockedNode) SetWithExpire(key string, v interface{}, expire time.Duration) error {
- return mc.Set(key, v)
- }
- func (mc *mockedNode) Take(v interface{}, key string, query func(v interface{}) error) error {
- if _, ok := mc.vals[key]; ok {
- return mc.Get(key, v)
- }
- if err := query(v); err != nil {
- return err
- }
- return mc.Set(key, v)
- }
- func (mc *mockedNode) TakeWithExpire(v interface{}, key string, query func(v interface{}, expire time.Duration) error) error {
- return mc.Take(v, key, func(v interface{}) error {
- return query(v, 0)
- })
- }
- func TestCache_SetDel(t *testing.T) {
- const total = 1000
- r1, clean1, err := redistest.CreateRedis()
- assert.Nil(t, err)
- defer clean1()
- r2, clean2, err := redistest.CreateRedis()
- assert.Nil(t, err)
- defer clean2()
- conf := ClusterConf{
- {
- RedisConf: redis.RedisConf{
- Host: r1.Addr,
- Type: redis.NodeType,
- },
- Weight: 100,
- },
- {
- RedisConf: redis.RedisConf{
- Host: r2.Addr,
- Type: redis.NodeType,
- },
- Weight: 100,
- },
- }
- c := New(conf, syncx.NewSharedCalls(), NewCacheStat("mock"), errPlaceholder)
- for i := 0; i < total; i++ {
- if i%2 == 0 {
- assert.Nil(t, c.Set(fmt.Sprintf("key/%d", i), i))
- } else {
- assert.Nil(t, c.SetWithExpire(fmt.Sprintf("key/%d", i), i, 0))
- }
- }
- for i := 0; i < total; i++ {
- var v int
- assert.Nil(t, c.Get(fmt.Sprintf("key/%d", i), &v))
- assert.Equal(t, i, v)
- }
- assert.Nil(t, c.Del())
- for i := 0; i < total; i++ {
- assert.Nil(t, c.Del(fmt.Sprintf("key/%d", i)))
- }
- for i := 0; i < total; i++ {
- var v int
- assert.True(t, c.IsNotFound(c.Get(fmt.Sprintf("key/%d", i), &v)))
- assert.Equal(t, 0, v)
- }
- }
- func TestCache_OneNode(t *testing.T) {
- const total = 1000
- r, clean, err := redistest.CreateRedis()
- assert.Nil(t, err)
- defer clean()
- conf := ClusterConf{
- {
- RedisConf: redis.RedisConf{
- Host: r.Addr,
- Type: redis.NodeType,
- },
- Weight: 100,
- },
- }
- c := New(conf, syncx.NewSharedCalls(), NewCacheStat("mock"), errPlaceholder)
- for i := 0; i < total; i++ {
- if i%2 == 0 {
- assert.Nil(t, c.Set(fmt.Sprintf("key/%d", i), i))
- } else {
- assert.Nil(t, c.SetWithExpire(fmt.Sprintf("key/%d", i), i, 0))
- }
- }
- for i := 0; i < total; i++ {
- var v int
- assert.Nil(t, c.Get(fmt.Sprintf("key/%d", i), &v))
- assert.Equal(t, i, v)
- }
- assert.Nil(t, c.Del())
- for i := 0; i < total; i++ {
- assert.Nil(t, c.Del(fmt.Sprintf("key/%d", i)))
- }
- for i := 0; i < total; i++ {
- var v int
- assert.True(t, c.IsNotFound(c.Get(fmt.Sprintf("key/%d", i), &v)))
- assert.Equal(t, 0, v)
- }
- }
- func TestCache_Balance(t *testing.T) {
- const (
- numNodes = 100
- total = 10000
- )
- dispatcher := hash.NewConsistentHash()
- maps := make([]map[string][]byte, numNodes)
- for i := 0; i < numNodes; i++ {
- maps[i] = map[string][]byte{
- strconv.Itoa(i): []byte(strconv.Itoa(i)),
- }
- }
- for i := 0; i < numNodes; i++ {
- dispatcher.AddWithWeight(&mockedNode{
- vals: maps[i],
- errNotFound: errPlaceholder,
- }, 100)
- }
- c := cacheCluster{
- dispatcher: dispatcher,
- errNotFound: errPlaceholder,
- }
- for i := 0; i < total; i++ {
- assert.Nil(t, c.Set(strconv.Itoa(i), i))
- }
- counts := make(map[int]int)
- for i, m := range maps {
- counts[i] = len(m)
- }
- entropy := calcEntropy(counts, total)
- assert.True(t, len(counts) > 1)
- assert.True(t, entropy > .95, fmt.Sprintf("entropy should be greater than 0.95, but got %.2f", entropy))
- for i := 0; i < total; i++ {
- var v int
- assert.Nil(t, c.Get(strconv.Itoa(i), &v))
- assert.Equal(t, i, v)
- }
- for i := 0; i < total/10; i++ {
- assert.Nil(t, c.Del(strconv.Itoa(i*10), strconv.Itoa(i*10+1), strconv.Itoa(i*10+2)))
- assert.Nil(t, c.Del(strconv.Itoa(i*10+9)))
- }
- var count int
- for i := 0; i < total/10; i++ {
- var val int
- if i%2 == 0 {
- assert.Nil(t, c.Take(&val, strconv.Itoa(i*10), func(v interface{}) error {
- *v.(*int) = i
- count++
- return nil
- }))
- } else {
- assert.Nil(t, c.TakeWithExpire(&val, strconv.Itoa(i*10), func(v interface{}, expire time.Duration) error {
- *v.(*int) = i
- count++
- return nil
- }))
- }
- assert.Equal(t, i, val)
- }
- assert.Equal(t, total/10, count)
- }
- func TestCacheNoNode(t *testing.T) {
- dispatcher := hash.NewConsistentHash()
- c := cacheCluster{
- dispatcher: dispatcher,
- errNotFound: errPlaceholder,
- }
- assert.NotNil(t, c.Del("foo"))
- assert.NotNil(t, c.Del("foo", "bar", "any"))
- assert.NotNil(t, c.Get("foo", nil))
- assert.NotNil(t, c.Set("foo", nil))
- assert.NotNil(t, c.SetWithExpire("foo", nil, time.Second))
- assert.NotNil(t, c.Take(nil, "foo", func(v interface{}) error {
- return nil
- }))
- assert.NotNil(t, c.TakeWithExpire(nil, "foo", func(v interface{}, duration time.Duration) error {
- return nil
- }))
- }
- func calcEntropy(m map[int]int, total int) float64 {
- var entropy float64
- for _, v := range m {
- proba := float64(v) / float64(total)
- entropy -= proba * math.Log2(proba)
- }
- return entropy / math.Log2(float64(len(m)))
- }
|