2020-09-27 22:09:46 +01:00
// Copyright 2020 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2016-11-03 23:16:01 +01:00
package cache
import (
"fmt"
"time"
2021-02-10 22:28:32 +01:00
"code.gitea.io/gitea/modules/graceful"
2020-09-27 22:09:46 +01:00
"code.gitea.io/gitea/modules/nosql"
2016-11-03 23:16:01 +01:00
2021-01-26 23:36:53 +08:00
"gitea.com/go-chi/cache"
2021-02-10 22:28:32 +01:00
"github.com/go-redis/redis/v8"
2020-09-27 22:09:46 +01:00
"github.com/unknwon/com"
2016-11-03 23:16:01 +01:00
)
// RedisCacher represents a redis cache adapter implementation.
type RedisCacher struct {
2020-09-27 22:09:46 +01:00
c redis . UniversalClient
2016-11-03 23:16:01 +01:00
prefix string
hsetName string
occupyMode bool
}
// Put puts value into cache with key and expire time.
// If expired is 0, it lives forever.
func ( c * RedisCacher ) Put ( key string , val interface { } , expire int64 ) error {
key = c . prefix + key
if expire == 0 {
2021-02-10 22:28:32 +01:00
if err := c . c . Set ( graceful . GetManager ( ) . HammerContext ( ) , key , com . ToStr ( val ) , 0 ) . Err ( ) ; err != nil {
2016-11-03 23:16:01 +01:00
return err
}
} else {
dur , err := time . ParseDuration ( com . ToStr ( expire ) + "s" )
if err != nil {
return err
}
2021-02-10 22:28:32 +01:00
if err = c . c . Set ( graceful . GetManager ( ) . HammerContext ( ) , key , com . ToStr ( val ) , dur ) . Err ( ) ; err != nil {
2016-11-03 23:16:01 +01:00
return err
}
}
if c . occupyMode {
return nil
}
2021-02-10 22:28:32 +01:00
return c . c . HSet ( graceful . GetManager ( ) . HammerContext ( ) , c . hsetName , key , "0" ) . Err ( )
2016-11-03 23:16:01 +01:00
}
// Get gets cached value by given key.
func ( c * RedisCacher ) Get ( key string ) interface { } {
2021-02-10 22:28:32 +01:00
val , err := c . c . Get ( graceful . GetManager ( ) . HammerContext ( ) , c . prefix + key ) . Result ( )
2016-11-03 23:16:01 +01:00
if err != nil {
return nil
}
return val
}
// Delete deletes cached value by given key.
func ( c * RedisCacher ) Delete ( key string ) error {
key = c . prefix + key
2021-02-10 22:28:32 +01:00
if err := c . c . Del ( graceful . GetManager ( ) . HammerContext ( ) , key ) . Err ( ) ; err != nil {
2016-11-03 23:16:01 +01:00
return err
}
if c . occupyMode {
return nil
}
2021-02-10 22:28:32 +01:00
return c . c . HDel ( graceful . GetManager ( ) . HammerContext ( ) , c . hsetName , key ) . Err ( )
2016-11-03 23:16:01 +01:00
}
// Incr increases cached int-type value by given key as a counter.
func ( c * RedisCacher ) Incr ( key string ) error {
if ! c . IsExist ( key ) {
return fmt . Errorf ( "key '%s' not exist" , key )
}
2021-02-10 22:28:32 +01:00
return c . c . Incr ( graceful . GetManager ( ) . HammerContext ( ) , c . prefix + key ) . Err ( )
2016-11-03 23:16:01 +01:00
}
// Decr decreases cached int-type value by given key as a counter.
func ( c * RedisCacher ) Decr ( key string ) error {
if ! c . IsExist ( key ) {
return fmt . Errorf ( "key '%s' not exist" , key )
}
2021-02-10 22:28:32 +01:00
return c . c . Decr ( graceful . GetManager ( ) . HammerContext ( ) , c . prefix + key ) . Err ( )
2016-11-03 23:16:01 +01:00
}
// IsExist returns true if cached value exists.
func ( c * RedisCacher ) IsExist ( key string ) bool {
2021-02-10 22:28:32 +01:00
if c . c . Exists ( graceful . GetManager ( ) . HammerContext ( ) , c . prefix + key ) . Val ( ) == 1 {
2016-11-03 23:16:01 +01:00
return true
}
if ! c . occupyMode {
2021-02-10 22:28:32 +01:00
c . c . HDel ( graceful . GetManager ( ) . HammerContext ( ) , c . hsetName , c . prefix + key )
2016-11-03 23:16:01 +01:00
}
return false
}
// Flush deletes all cached data.
func ( c * RedisCacher ) Flush ( ) error {
if c . occupyMode {
2021-02-10 22:28:32 +01:00
return c . c . FlushDB ( graceful . GetManager ( ) . HammerContext ( ) ) . Err ( )
2016-11-03 23:16:01 +01:00
}
2021-02-10 22:28:32 +01:00
keys , err := c . c . HKeys ( graceful . GetManager ( ) . HammerContext ( ) , c . hsetName ) . Result ( )
2016-11-03 23:16:01 +01:00
if err != nil {
return err
}
2021-02-10 22:28:32 +01:00
if err = c . c . Del ( graceful . GetManager ( ) . HammerContext ( ) , keys ... ) . Err ( ) ; err != nil {
2016-11-03 23:16:01 +01:00
return err
}
2021-02-10 22:28:32 +01:00
return c . c . Del ( graceful . GetManager ( ) . HammerContext ( ) , c . hsetName ) . Err ( )
2016-11-03 23:16:01 +01:00
}
// StartAndGC starts GC routine based on config string settings.
// AdapterConfig: network=tcp,addr=:6379,password=macaron,db=0,pool_size=100,idle_timeout=180,hset_name=MacaronCache,prefix=cache:
func ( c * RedisCacher ) StartAndGC ( opts cache . Options ) error {
c . hsetName = "MacaronCache"
c . occupyMode = opts . OccupyMode
2020-09-27 22:09:46 +01:00
uri := nosql . ToRedisURI ( opts . AdapterConfig )
2016-11-03 23:16:01 +01:00
2020-09-27 22:09:46 +01:00
c . c = nosql . GetManager ( ) . GetRedisClient ( uri . String ( ) )
for k , v := range uri . Query ( ) {
2016-11-03 23:16:01 +01:00
switch k {
case "hset_name" :
2020-09-27 22:09:46 +01:00
c . hsetName = v [ 0 ]
2016-11-03 23:16:01 +01:00
case "prefix" :
2020-09-27 22:09:46 +01:00
c . prefix = v [ 0 ]
2016-11-03 23:16:01 +01:00
}
}
2021-02-10 22:28:32 +01:00
return c . c . Ping ( graceful . GetManager ( ) . HammerContext ( ) ) . Err ( )
2016-11-03 23:16:01 +01:00
}
func init ( ) {
cache . Register ( "redis" , & RedisCacher { } )
}