| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394 |
- package remotecache
- import (
- "fmt"
- "strconv"
- "strings"
- "time"
- "github.com/grafana/grafana/pkg/setting"
- "github.com/grafana/grafana/pkg/util/errutil"
- redis "gopkg.in/redis.v2"
- )
- const redisCacheType = "redis"
- type redisStorage struct {
- c *redis.Client
- }
- // parseRedisConnStr parses k=v pairs in csv and builds a redis Options object
- func parseRedisConnStr(connStr string) (*redis.Options, error) {
- keyValueCSV := strings.Split(connStr, ",")
- options := &redis.Options{Network: "tcp"}
- for _, rawKeyValue := range keyValueCSV {
- keyValueTuple := strings.Split(rawKeyValue, "=")
- if len(keyValueTuple) != 2 {
- return nil, fmt.Errorf("incorrect redis connection string format detected for '%v', format is key=value,key=value", rawKeyValue)
- }
- connKey := keyValueTuple[0]
- connVal := keyValueTuple[1]
- switch connKey {
- case "addr":
- options.Addr = connVal
- case "password":
- options.Password = connVal
- case "db":
- i, err := strconv.ParseInt(connVal, 10, 64)
- if err != nil {
- return nil, errutil.Wrap("value for db in redis connection string must be a number", err)
- }
- options.DB = i
- case "pool_size":
- i, err := strconv.Atoi(connVal)
- if err != nil {
- return nil, errutil.Wrap("value for pool_size in redis connection string must be a number", err)
- }
- options.PoolSize = i
- default:
- return nil, fmt.Errorf("unrecorgnized option '%v' in redis connection string", connVal)
- }
- }
- return options, nil
- }
- func newRedisStorage(opts *setting.RemoteCacheOptions) (*redisStorage, error) {
- opt, err := parseRedisConnStr(opts.ConnStr)
- if err != nil {
- return nil, err
- }
- return &redisStorage{c: redis.NewClient(opt)}, nil
- }
- // Set sets value to given key in session.
- func (s *redisStorage) Set(key string, val interface{}, expires time.Duration) error {
- item := &cachedItem{Val: val}
- value, err := encodeGob(item)
- if err != nil {
- return err
- }
- status := s.c.SetEx(key, expires, string(value))
- return status.Err()
- }
- // Get gets value by given key in session.
- func (s *redisStorage) Get(key string) (interface{}, error) {
- v := s.c.Get(key)
- item := &cachedItem{}
- err := decodeGob([]byte(v.Val()), item)
- if err == nil {
- return item.Val, nil
- }
- if err.Error() == "EOF" {
- return nil, ErrCacheItemNotFound
- }
- return nil, err
- }
- // Delete delete a key from session.
- func (s *redisStorage) Delete(key string) error {
- cmd := s.c.Del(key)
- return cmd.Err()
- }
|