2025-11-03 17:03:57 +08:00
|
|
|
using StackExchange.Redis;
|
2025-11-03 19:47:36 +08:00
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
using System.Text.Json;
|
2025-11-03 17:03:57 +08:00
|
|
|
|
|
|
|
|
namespace HardwarePerformance.Infrastructure.Services;
|
|
|
|
|
|
|
|
|
|
public interface IRedisCacheService
|
|
|
|
|
{
|
|
|
|
|
Task<T?> GetAsync<T>(string key);
|
|
|
|
|
Task SetAsync<T>(string key, T value, TimeSpan? expiry = null);
|
|
|
|
|
Task RemoveAsync(string key);
|
|
|
|
|
Task<bool> ExistsAsync(string key);
|
|
|
|
|
Task RemoveByPatternAsync(string pattern);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class RedisCacheService : IRedisCacheService
|
|
|
|
|
{
|
|
|
|
|
private readonly IDatabase _database;
|
|
|
|
|
private readonly ILogger<RedisCacheService> _logger;
|
|
|
|
|
|
|
|
|
|
public RedisCacheService(IConnectionMultiplexer connectionMultiplexer, ILogger<RedisCacheService> logger)
|
|
|
|
|
{
|
|
|
|
|
_database = connectionMultiplexer.GetDatabase();
|
|
|
|
|
_logger = logger;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<T?> GetAsync<T>(string key)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var value = await _database.StringGetAsync(key);
|
|
|
|
|
if (value.HasValue)
|
|
|
|
|
{
|
|
|
|
|
return JsonSerializer.Deserialize<T>(value!);
|
|
|
|
|
}
|
|
|
|
|
return default;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "Error getting value from Redis with key: {Key}", key);
|
|
|
|
|
return default;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task SetAsync<T>(string key, T value, TimeSpan? expiry = null)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var serializedValue = JsonSerializer.Serialize(value);
|
|
|
|
|
await _database.StringSetAsync(key, serializedValue, expiry);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "Error setting value to Redis with key: {Key}", key);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task RemoveAsync(string key)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
await _database.KeyDeleteAsync(key);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "Error removing key from Redis: {Key}", key);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task<bool> ExistsAsync(string key)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
return await _database.KeyExistsAsync(key);
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "Error checking if key exists in Redis: {Key}", key);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task RemoveByPatternAsync(string pattern)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var server = _database.Multiplexer.GetServer(_database.Multiplexer.GetEndPoints().First());
|
|
|
|
|
var keys = server.Keys(database: _database.Database, pattern: pattern);
|
|
|
|
|
|
|
|
|
|
foreach (var key in keys)
|
|
|
|
|
{
|
|
|
|
|
await _database.KeyDeleteAsync(key);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogError(ex, "Error removing keys by pattern from Redis: {Pattern}", pattern);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|