redis.go 2.5 KB

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