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 }