redis.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. package common
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "time"
  7. "github.com/go-redis/redis/v8"
  8. )
  9. var RDB *redis.Client
  10. var RedisEnabled = true
  11. // InitRedisClient This function is called after init()
  12. func InitRedisClient() (err error) {
  13. if os.Getenv("REDIS_CONN_STRING") == "" {
  14. RedisEnabled = false
  15. SysLog("REDIS_CONN_STRING not set, Redis is not enabled")
  16. return nil
  17. }
  18. if os.Getenv("SYNC_FREQUENCY") == "" {
  19. SysLog("SYNC_FREQUENCY not set, use default value 60")
  20. SyncFrequency = 60
  21. }
  22. SysLog("Redis is enabled")
  23. opt, err := redis.ParseURL(os.Getenv("REDIS_CONN_STRING"))
  24. if err != nil {
  25. FatalLog("failed to parse Redis connection string: " + err.Error())
  26. }
  27. RDB = redis.NewClient(opt)
  28. ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  29. defer cancel()
  30. _, err = RDB.Ping(ctx).Result()
  31. if err != nil {
  32. FatalLog("Redis ping test failed: " + err.Error())
  33. }
  34. return err
  35. }
  36. func ParseRedisOption() *redis.Options {
  37. opt, err := redis.ParseURL(os.Getenv("REDIS_CONN_STRING"))
  38. if err != nil {
  39. FatalLog("failed to parse Redis connection string: " + err.Error())
  40. }
  41. return opt
  42. }
  43. func RedisSet(key string, value string, expiration time.Duration) error {
  44. ctx := context.Background()
  45. return RDB.Set(ctx, key, value, expiration).Err()
  46. }
  47. func RedisGet(key string) (string, error) {
  48. ctx := context.Background()
  49. return RDB.Get(ctx, key).Result()
  50. }
  51. func RedisExpire(key string, expiration time.Duration) error {
  52. ctx := context.Background()
  53. return RDB.Expire(ctx, key, expiration).Err()
  54. }
  55. func RedisGetEx(key string, expiration time.Duration) (string, error) {
  56. ctx := context.Background()
  57. return RDB.GetSet(ctx, key, expiration).Result()
  58. }
  59. func RedisDel(key string) error {
  60. ctx := context.Background()
  61. return RDB.Del(ctx, key).Err()
  62. }
  63. func RedisDecrease(key string, value int64) error {
  64. // 检查键的剩余生存时间
  65. ttlCmd := RDB.TTL(context.Background(), key)
  66. ttl, err := ttlCmd.Result()
  67. if err != nil {
  68. // 失败则尝试直接减少
  69. return RDB.DecrBy(context.Background(), key, value).Err()
  70. }
  71. // 如果剩余生存时间大于0,则进行减少操作
  72. if ttl > 0 {
  73. ctx := context.Background()
  74. // 开始一个Redis事务
  75. txn := RDB.TxPipeline()
  76. // 减少余额
  77. decrCmd := txn.DecrBy(ctx, key, value)
  78. if err := decrCmd.Err(); err != nil {
  79. return err // 如果减少失败,则直接返回错误
  80. }
  81. // 重新设置过期时间,使用原来的过期时间
  82. txn.Expire(ctx, key, ttl)
  83. // 执行事务
  84. _, err = txn.Exec(ctx)
  85. return err
  86. } else {
  87. _ = RedisDel(key)
  88. }
  89. return nil
  90. }
  91. // RedisIncr Add this function to handle atomic increments
  92. func RedisIncr(key string, delta int) error {
  93. ctx := context.Background()
  94. // 检查键是否存在
  95. exists, err := RDB.Exists(ctx, key).Result()
  96. if err != nil {
  97. return err
  98. }
  99. if exists == 0 {
  100. return fmt.Errorf("key does not exist") // 键不存在,返回错误
  101. }
  102. // 键存在,执行INCRBY操作
  103. result := RDB.IncrBy(ctx, key, int64(delta))
  104. return result.Err()
  105. }