redis.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  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. RedisEnabled = false
  19. SysLog("SYNC_FREQUENCY not set, Redis is disabled")
  20. return nil
  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. }