gitea.com/xorm/xorm-redis-cache@v0.2.0/redis_cacher.go (about)

     1  package xormrediscache
     2  
     3  import (
     4  	"bytes"
     5  	"encoding/gob"
     6  	"fmt"
     7  	"hash/crc32"
     8  	"reflect"
     9  	"time"
    10  	"unsafe"
    11  
    12  	"github.com/garyburd/redigo/redis"
    13  	"xorm.io/xorm/caches"
    14  	"xorm.io/xorm/log"
    15  )
    16  
    17  const (
    18  	DEFAULT_EXPIRATION = time.Duration(0)
    19  	FOREVER_EXPIRATION = time.Duration(-1)
    20  
    21  	LOGGING_PREFIX = "[redis_cacher]"
    22  )
    23  
    24  // RedisCacher wraps the Redis client to meet the Cache interface.
    25  type RedisCacher struct {
    26  	pool              *redis.Pool
    27  	defaultExpiration time.Duration
    28  
    29  	Logger log.ContextLogger
    30  }
    31  
    32  // NewRedisCacher creates a Redis Cacher, host as IP endpoint, i.e., localhost:6379, provide empty string or nil if Redis server doesn't
    33  // require AUTH command, defaultExpiration sets the expire duration for a key to live. Until redigo supports
    34  // sharding/clustering, only one host will be in hostList
    35  //
    36  //     engine.SetDefaultCacher(xormrediscache.NewRedisCacher("localhost:6379", "", xormrediscache.DEFAULT_EXPIRATION, engine.Logger))
    37  //
    38  // or set MapCacher
    39  //
    40  //     engine.MapCacher(&user, xormrediscache.NewRedisCacher("localhost:6379", "", xormrediscache.DEFAULT_EXPIRATION, engine.Logger))
    41  //
    42  func NewRedisCacher(host string, password string, defaultExpiration time.Duration, logger log.ContextLogger) *RedisCacher {
    43  	var pool = &redis.Pool{
    44  		MaxIdle:     5,
    45  		IdleTimeout: 240 * time.Second,
    46  		Dial: func() (redis.Conn, error) {
    47  			// the redis protocol should probably be made sett-able
    48  			c, err := redis.Dial("tcp", host)
    49  			if err != nil {
    50  				return nil, err
    51  			}
    52  			if len(password) > 0 {
    53  				if _, err := c.Do("AUTH", password); err != nil {
    54  					c.Close()
    55  					return nil, err
    56  				}
    57  			} else {
    58  				// check with PING
    59  				if _, err := c.Do("PING"); err != nil {
    60  					c.Close()
    61  					return nil, err
    62  				}
    63  			}
    64  			return c, err
    65  		},
    66  		// custom connection test method
    67  		TestOnBorrow: func(c redis.Conn, t time.Time) error {
    68  			if _, err := c.Do("PING"); err != nil {
    69  				return err
    70  			}
    71  			return nil
    72  		},
    73  	}
    74  	return MakeRedisCacher(pool, defaultExpiration, logger)
    75  }
    76  
    77  // MakeRedisCacher build a cacher based on redis.Pool
    78  func MakeRedisCacher(pool *redis.Pool, defaultExpiration time.Duration, logger log.ContextLogger) *RedisCacher {
    79  	return &RedisCacher{pool: pool, defaultExpiration: defaultExpiration, Logger: logger}
    80  }
    81  
    82  func exists(conn redis.Conn, key string) bool {
    83  	existed, _ := redis.Bool(conn.Do("EXISTS", key))
    84  	return existed
    85  }
    86  
    87  func (c *RedisCacher) logErrf(format string, contents ...interface{}) {
    88  	if c.Logger != nil {
    89  		c.Logger.Errorf(fmt.Sprintf("%s %s", LOGGING_PREFIX, format), contents...)
    90  	}
    91  }
    92  
    93  func (c *RedisCacher) logDebugf(format string, contents ...interface{}) {
    94  	if c.Logger != nil {
    95  		c.Logger.Debugf(fmt.Sprintf("%s %s", LOGGING_PREFIX, format), contents...)
    96  	}
    97  }
    98  
    99  func (c *RedisCacher) getBeanKey(tableName string, id string) string {
   100  	return fmt.Sprintf("xorm:bean:%s:%s", tableName, id)
   101  }
   102  
   103  func (c *RedisCacher) getSqlKey(tableName string, sql string) string {
   104  	// hash sql to minimize key length
   105  	crc := crc32.ChecksumIEEE([]byte(sql))
   106  	return fmt.Sprintf("xorm:sql:%s:%d", tableName, crc)
   107  }
   108  
   109  // Flush deletes all xorm cached objects
   110  func (c *RedisCacher) Flush() error {
   111  	// conn := c.pool.Get()
   112  	// defer conn.Close()
   113  	// _, err := conn.Do("FLUSHALL")
   114  	// return err
   115  	return c.delObject("xorm:*")
   116  }
   117  
   118  func (c *RedisCacher) getObject(key string) interface{} {
   119  	conn := c.pool.Get()
   120  	defer conn.Close()
   121  	raw, err := conn.Do("GET", key)
   122  	if raw == nil {
   123  		return nil
   124  	}
   125  	item, err := redis.Bytes(raw, err)
   126  	if err != nil {
   127  		c.logErrf("redis.Bytes failed: %s", err)
   128  		return nil
   129  	}
   130  
   131  	value, err := c.deserialize(item)
   132  
   133  	return value
   134  }
   135  
   136  func (c *RedisCacher) GetIds(tableName, sql string) interface{} {
   137  	sqlKey := c.getSqlKey(tableName, sql)
   138  	c.logDebugf(" GetIds|tableName:%s|sql:%s|key:%s", tableName, sql, sqlKey)
   139  	return c.getObject(sqlKey)
   140  }
   141  
   142  func (c *RedisCacher) GetBean(tableName string, id string) interface{} {
   143  	beanKey := c.getBeanKey(tableName, id)
   144  	c.logDebugf("[xorm/redis_cacher] GetBean|tableName:%s|id:%s|key:%s", tableName, id, beanKey)
   145  	return c.getObject(beanKey)
   146  }
   147  
   148  func (c *RedisCacher) putObject(key string, value interface{}) {
   149  	c.invoke(c.pool.Get().Do, key, value, c.defaultExpiration)
   150  }
   151  
   152  func (c *RedisCacher) PutIds(tableName, sql string, ids interface{}) {
   153  	sqlKey := c.getSqlKey(tableName, sql)
   154  	c.logDebugf("PutIds|tableName:%s|sql:%s|key:%s|obj:%s|type:%v", tableName, sql, sqlKey, ids, reflect.TypeOf(ids))
   155  	c.putObject(sqlKey, ids)
   156  }
   157  
   158  func (c *RedisCacher) PutBean(tableName string, id string, obj interface{}) {
   159  	beanKey := c.getBeanKey(tableName, id)
   160  	c.logDebugf("PutBean|tableName:%s|id:%s|key:%s|type:%v", tableName, id, beanKey, reflect.TypeOf(obj))
   161  	c.putObject(beanKey, obj)
   162  }
   163  
   164  func (c *RedisCacher) delObject(key string) error {
   165  	c.logDebugf("delObject key:[%s]", key)
   166  
   167  	conn := c.pool.Get()
   168  	defer conn.Close()
   169  	if !exists(conn, key) {
   170  		c.logErrf("delObject key:[%s] err: %v", key, caches.ErrCacheMiss)
   171  		return caches.ErrCacheMiss
   172  	}
   173  	_, err := conn.Do("DEL", key)
   174  	return err
   175  }
   176  
   177  func (c *RedisCacher) delObjects(key string) error {
   178  
   179  	c.logDebugf("delObjects key:[%s]", key)
   180  
   181  	conn := c.pool.Get()
   182  	defer conn.Close()
   183  
   184  	keys, err := conn.Do("KEYS", key)
   185  	c.logDebugf("delObjects keys: %v", keys)
   186  
   187  	if err == nil {
   188  		for _, key := range keys.([]interface{}) {
   189  			conn.Do("DEL", key)
   190  		}
   191  	}
   192  	return err
   193  }
   194  
   195  func (c *RedisCacher) DelIds(tableName, sql string) {
   196  	c.delObject(c.getSqlKey(tableName, sql))
   197  }
   198  
   199  func (c *RedisCacher) DelBean(tableName string, id string) {
   200  	c.delObject(c.getBeanKey(tableName, id))
   201  }
   202  
   203  func (c *RedisCacher) ClearIds(tableName string) {
   204  	c.delObjects(fmt.Sprintf("xorm:sql:%s:*", tableName))
   205  }
   206  
   207  func (c *RedisCacher) ClearBeans(tableName string) {
   208  	c.delObjects(c.getBeanKey(tableName, "*"))
   209  }
   210  
   211  func (c *RedisCacher) invoke(f func(string, ...interface{}) (interface{}, error),
   212  	key string, value interface{}, expires time.Duration) error {
   213  
   214  	switch expires {
   215  	case DEFAULT_EXPIRATION:
   216  		expires = c.defaultExpiration
   217  	case FOREVER_EXPIRATION:
   218  		expires = time.Duration(0)
   219  	}
   220  
   221  	b, err := c.serialize(value)
   222  	if err != nil {
   223  		return err
   224  	}
   225  	conn := c.pool.Get()
   226  	defer conn.Close()
   227  	if expires > 0 {
   228  		_, err := f("SETEX", key, int32(expires/time.Second), b)
   229  		return err
   230  	} else {
   231  		_, err := f("SET", key, b)
   232  		return err
   233  	}
   234  }
   235  
   236  func (c *RedisCacher) serialize(value interface{}) ([]byte, error) {
   237  
   238  	err := c.registerGobConcreteType(value)
   239  	if err != nil {
   240  		return nil, err
   241  	}
   242  
   243  	if reflect.TypeOf(value).Kind() == reflect.Struct {
   244  		return nil, fmt.Errorf("serialize func only take pointer of a struct")
   245  	}
   246  
   247  	var b bytes.Buffer
   248  	encoder := gob.NewEncoder(&b)
   249  
   250  	c.logDebugf("serialize type:%v", reflect.TypeOf(value))
   251  	err = encoder.Encode(&value)
   252  	if err != nil {
   253  		c.logErrf("gob encoding '%s' failed: %s|value:%v", value, err, value)
   254  		return nil, err
   255  	}
   256  	return b.Bytes(), nil
   257  }
   258  
   259  func (c *RedisCacher) deserialize(byt []byte) (ptr interface{}, err error) {
   260  	b := bytes.NewBuffer(byt)
   261  	decoder := gob.NewDecoder(b)
   262  
   263  	var p interface{}
   264  	err = decoder.Decode(&p)
   265  	if err != nil {
   266  		c.logErrf("decode failed: %v", err)
   267  		return
   268  	}
   269  
   270  	v := reflect.ValueOf(p)
   271  	c.logDebugf("deserialize type:%v", v.Type())
   272  	if v.Kind() == reflect.Struct {
   273  
   274  		var pp interface{} = &p
   275  		datas := reflect.ValueOf(pp).Elem().InterfaceData()
   276  
   277  		sp := reflect.NewAt(v.Type(),
   278  			unsafe.Pointer(datas[1])).Interface()
   279  		ptr = sp
   280  		vv := reflect.ValueOf(ptr)
   281  		c.logDebugf("deserialize convert ptr type:%v | CanAddr:%t", vv.Type(), vv.CanAddr())
   282  	} else {
   283  		ptr = p
   284  	}
   285  	return
   286  }
   287  
   288  func (c *RedisCacher) registerGobConcreteType(value interface{}) error {
   289  
   290  	t := reflect.TypeOf(value)
   291  
   292  	c.logDebugf("registerGobConcreteType:%v", t)
   293  
   294  	switch t.Kind() {
   295  	case reflect.Ptr:
   296  		v := reflect.ValueOf(value)
   297  		i := v.Elem().Interface()
   298  		gob.Register(&i)
   299  	case reflect.Struct, reflect.Map, reflect.Slice:
   300  		gob.Register(value)
   301  	case reflect.String, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Bool, reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128:
   302  		// do nothing since already registered known type
   303  	default:
   304  		return fmt.Errorf("unhandled type: %v", t)
   305  	}
   306  	return nil
   307  }
   308  
   309  func (c *RedisCacher) GetPool() (*redis.Pool, error) {
   310  	return c.pool, nil
   311  }
   312  
   313  func (c *RedisCacher) SetPool(pool *redis.Pool) {
   314  	c.pool = pool
   315  }