database_storage.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. package distcache
  2. import (
  3. "time"
  4. "github.com/grafana/grafana/pkg/log"
  5. "github.com/grafana/grafana/pkg/services/sqlstore"
  6. )
  7. type databaseCache struct {
  8. SQLStore *sqlstore.SqlStore
  9. log log.Logger
  10. }
  11. func newDatabaseCache(sqlstore *sqlstore.SqlStore) *databaseCache {
  12. dc := &databaseCache{
  13. SQLStore: sqlstore,
  14. log: log.New("distcache.database"),
  15. }
  16. //go dc.StartGC() //TODO: start the GC somehow
  17. return dc
  18. }
  19. var getTime = time.Now
  20. func (dc *databaseCache) internalRunGC() {
  21. now := getTime().Unix()
  22. sql := `DELETE FROM cache_data WHERE (? - created) >= expire`
  23. //EXTRACT(EPOCH FROM NOW()) - created >= expire
  24. //UNIX_TIMESTAMP(NOW()) - created >= expire
  25. _, err := dc.SQLStore.NewSession().Exec(sql, now)
  26. if err != nil {
  27. dc.log.Error("failed to run garbage collect", "error", err)
  28. }
  29. }
  30. func (dc *databaseCache) StartGC() {
  31. dc.internalRunGC()
  32. time.AfterFunc(time.Second*10, func() {
  33. dc.StartGC()
  34. })
  35. }
  36. func (dc *databaseCache) Get(key string) (interface{}, error) {
  37. cacheHits := []CacheData{}
  38. err := dc.SQLStore.NewSession().Where(`key = ?`, key).Find(&cacheHits)
  39. if err != nil {
  40. return nil, err
  41. }
  42. var cacheHit CacheData
  43. if len(cacheHits) == 0 {
  44. return nil, ErrCacheItemNotFound
  45. }
  46. cacheHit = cacheHits[0]
  47. if cacheHit.Expires > 0 {
  48. if getTime().Unix()-cacheHit.CreatedAt >= cacheHit.Expires {
  49. dc.Delete(key)
  50. return nil, ErrCacheItemNotFound
  51. }
  52. }
  53. item := &Item{}
  54. if err = DecodeGob(cacheHit.Data, item); err != nil {
  55. return nil, err
  56. }
  57. return item.Val, nil
  58. }
  59. type CacheData struct {
  60. Key string
  61. Data []byte
  62. Expires int64
  63. CreatedAt int64
  64. }
  65. func (dc *databaseCache) Put(key string, value interface{}, expire time.Duration) error {
  66. item := &Item{Val: value}
  67. data, err := EncodeGob(item)
  68. if err != nil {
  69. return err
  70. }
  71. now := getTime().Unix()
  72. cacheHits := []CacheData{}
  73. err = dc.SQLStore.NewSession().Where(`key = ?`, key).Find(&cacheHits)
  74. if err != nil {
  75. return err
  76. }
  77. var expiresInEpoch int64
  78. if expire != 0 {
  79. expiresInEpoch = int64(expire) / int64(time.Second)
  80. }
  81. if len(cacheHits) > 0 {
  82. _, err = dc.SQLStore.NewSession().Exec("UPDATE cache_data SET data=?, created=?, expire=? WHERE key=?", data, now, expiresInEpoch, key)
  83. } else {
  84. _, err = dc.SQLStore.NewSession().Exec("INSERT INTO cache_data(key,data,created_at,expires) VALUES(?,?,?,?)", key, data, now, expiresInEpoch)
  85. }
  86. return err
  87. }
  88. func (dc *databaseCache) Delete(key string) error {
  89. sql := `DELETE FROM cache_data WHERE key = ?`
  90. _, err := dc.SQLStore.NewSession().Exec(sql, key)
  91. return err
  92. }