using System.Net.Http;
using System.Text.Json.Nodes;
using Dpz.Core.Public.Entity.Steam;
using Dpz.Core.Public.ViewModel.Response.Steam;
using Microsoft.AspNetCore.WebUtilities;

namespace Dpz.Core.Service.RepositoryServiceImpl;

public class SteamGameService(
    IRepository<SteamGame> repository,
    IMapper mapper,
    HttpClient httpClient,
    IConfiguration configuration,
    ILogger<SteamGameService> logger,
    IFusionCache fusionCache
) : AbstractCacheService(fusionCache), ISteamGameService
{
    protected override TimeSpan CacheDefaultExpiration => TimeSpan.FromDays(1);

    private record SteamConfig(string AppKey = "", string SteamId = "");

    private record CdnConfig
    {
        public string BaseUrl { get; init; } = "https://cdn.dpangzi.com";
        public string LogoUrlTemplate { get; init; } =
            "https://shared.akamai.steamstatic.com/store_item_assets/steam/apps/{0}/header.jpg";
    }

    private readonly SteamConfig _steamConfig =
        configuration.GetSection("Steam").Get<SteamConfig>()
        ?? throw new InvalidConfigurationException(
            "Steam 配置缺失:请检查 AppKey 和 SteamId 是否配置。"
        );

    private readonly CdnConfig _cdnConfig =
        configuration.GetSection("Cdn").Get<CdnConfig>() ?? new CdnConfig();

    private readonly Lazy<List<int>> _filterIgnoreGameIds = new(() =>
        configuration.GetSection("Steam:IgnoreFilterIds").Get<List<int>>() ?? []
    );

    // API请求延迟时间,防止请求过于频繁
    private const int ApiDelayMs = 500;

    /// <summary>
    /// 当前实例ID,用于区分不同的实例
    /// </summary>
    private readonly Lazy<string> _instanceId = new(() => Guid.NewGuid().ToString("D"));

    private readonly IFusionCache _fusionCache = fusionCache;

    /// <summary>
    /// 当前实例ID的缓存键
    /// </summary>
    private const string RunningInstanceIdKey = "SteamGameService.Running.InstanceId";

    public event LogoDownload? OnLogoDownloadComplete;
    public event AchievementIconGrayDownload? OnAchievementIconGrayDownloadComplete;
    public event AchievementIconDownload? OnAchievementIconDownloadComplete;

    public async Task UpdateGamesAsync(CancellationToken cancellationToken = default)
    {
        var runningInstanceId = await _fusionCache.TryGetAsync<string>(
            RunningInstanceIdKey,
            token: cancellationToken
        );
        if (runningInstanceId.HasValue && runningInstanceId.Value != _instanceId.Value)
        {
            logger.LogInformation(
                "当前已有实例正在运行,跳过此次更新,实例ID:{RunningInstanceId},当前实例ID:{CurrentInstanceId}",
                runningInstanceId.Value,
                _instanceId.Value
            );
            return;
        }

        await _fusionCache.SetAsync(
            RunningInstanceIdKey,
            _instanceId.Value,
            TimeSpan.FromHours(3),
            token: cancellationToken
        );

        try
        {
            logger.LogInformation("开始更新Steam游戏数据,实例ID:{InstanceId}", _instanceId.Value);
            var dbGames = await repository.SearchFor(x => true).ToListAsync(cancellationToken);
            var games = await FetchGamesAsync(dbGames, cancellationToken);

            // 安全检查:如果获取到空列表,直接返回,避免误删数据
            if (games.Count == 0)
            {
                logger.LogWarning("获取到的Steam游戏数量为0,跳过此次更新以防止误删数据");
                return;
            }

            // 新增游戏
            var newGames = games.ExceptBy(dbGames.Select(x => x.Id), x => x.Id).ToList();
            if (newGames.Count > 0)
            {
                logger.LogInformation("发现{Count}个新游戏,准备添加", newGames.Count);
                await repository.InsertAsync(newGames, cancellationToken);
                // var batchSize =
                //     newGames.Count % 10 > 0 ? newGames.Count / 10 + 1 : newGames.Count / 10;
                //
                // for (var i = 0; i < batchSize; i++)
                // {
                //     var startIndex = i * 10;
                //     var count = Math.Min(10, newGames.Count - startIndex);
                //     var batch = newGames.GetRange(startIndex, count);
                //     logger.LogInformation(
                //         "批量添加游戏,开始索引:{StartIndex},数量:{Count}",
                //         startIndex,
                //         count
                //     );
                //     await repository.InsertAsync(batch);
                //     await Task.Delay(1000);
                // }
            }

            // DB已有游戏,待更新
            var updateGames = games.IntersectBy(dbGames.Select(x => x.Id), x => x.Id).ToList();
            if (updateGames.Count > 0)
            {
                logger.LogInformation("开始更新{Count}个已有游戏", updateGames.Count);
                await Parallel.ForEachAsync(
                    updateGames,
                    cancellationToken,
                    async (game, _) => await UpdateGameAsync(dbGames, game)
                );
            }

            // 删除已删除的游戏
            var deletedGames = dbGames.ExceptBy(games.Select(x => x.Id), x => x.Id).ToList();
            if (deletedGames.Count > 0)
            {
                logger.LogInformation("发现{Count}个已删除游戏,准备删除", deletedGames.Count);

                var filter = Builders<SteamGame>.Filter.In(
                    x => x.Id,
                    deletedGames.Select(x => x.Id)
                );
                await repository.DeleteAsync(filter, cancellationToken);
            }

            await OnRemoveByPrefixAfterAsync(cancellationToken);
            logger.LogInformation(
                "Steam游戏数据更新完成,新增:{NewCount},更新:{UpdateCount},删除:{DeletedCount},实例ID:{InstanceId}",
                newGames.Count,
                updateGames.Count,
                deletedGames.Count,
                _instanceId.Value
            );
        }
        catch (Exception ex)
        {
            logger.LogError(
                ex,
                "更新Steam游戏数据时发生异常,实例ID:{InstanceId}",
                _instanceId.Value
            );
            throw;
        }
        finally
        {
            // 确保在任何情况下都释放锁
            try
            {
                await _fusionCache.RemoveAsync(RunningInstanceIdKey, token: cancellationToken);
                logger.LogDebug("已释放Steam游戏更新锁,实例ID:{InstanceId}", _instanceId.Value);
            }
            catch (Exception ex)
            {
                logger.LogError(
                    ex,
                    "释放Steam游戏更新锁时发生异常,实例ID:{InstanceId}",
                    _instanceId.Value
                );
            }
        }
    }

    private async Task UpdateGameAsync(List<SteamGame> dbGames, SteamGame updateGame)
    {
        var dbGame = dbGames.Find(x => x.Id == updateGame.Id);
        if (dbGame != null)
        {
            var dic = dbGame.UpdateContent(updateGame);
            if (dic.Count > 0)
            {
                logger.LogInformation(
                    "更新游戏 {GameId}/{GameName} 的数据,更新细节:{@UpdateContent}",
                    updateGame.Id,
                    updateGame.Name,
                    dic
                );
            }
        }
        await repository.UpdateAsync(updateGame);
    }

    private async Task<List<AchievementDetail>> FetchAchievementsAsync(
        int id,
        List<SteamGame> dbGames,
        CancellationToken cancellationToken = default
    )
    {
        var queryString = new Dictionary<string, string?>
        {
            { "steamid", _steamConfig.SteamId },
            { "key", _steamConfig.AppKey },
            { "format", "json" },
            { "appid", id.ToString() },
            { "l", "schinese" },
        };

        // 添加API请求延迟
        await Task.Delay(ApiDelayMs, cancellationToken);

        logger.LogDebug("获取游戏 {GameId} 的成就数据", id);
        var achievements =
            await ExecuteHttpRequestAsync<List<AchievementDetail>>(
                "/ISteamUserStats/GetSchemaForGame/v2/",
                queryString,
                root => root?["game"]?["availableGameStats"]?["achievements"]
            ) ?? [];

        var unlockAchievements = await FetchUnlockAchievementAsync(queryString);
#if DEBUG
        var options = new ParallelOptions { MaxDegreeOfParallelism = 1 };
#endif
        await Parallel.ForEachAsync(
            achievements,
#if DEBUG
            options,
#endif
            async (achievement, _) =>
            {
                if (
                    achievement.Name != null
                    && unlockAchievements.TryGetValue(achievement.Name, out var unlockTime)
                    && unlockTime > 0
                )
                {
                    achievement.UnlockTime = unlockTime.ToDateTime();
                }

                await DownloadAchievementAsync(dbGames, id, achievement);
                await DownloadAchievementGrayAsync(dbGames, id, achievement);
            }
        );

        logger.LogDebug("游戏 {GameId} 共获取到 {Count} 个成就", id, achievements.Count);
        return achievements;
    }

    private async Task<Dictionary<string, long>> FetchUnlockAchievementAsync(
        Dictionary<string, string?> queryString
    )
    {
        logger.LogDebug("获取已解锁的成就数据");

        // 添加API请求延迟
        await Task.Delay(ApiDelayMs);

        return await ExecuteHttpRequestAsync<Dictionary<string, long>>(
                "/ISteamUserStats/GetPlayerAchievements/v0001/",
                queryString,
                root =>
                {
                    var achievements = root
                        ?["playerstats"]?["achievements"]?.AsArray()
                        .Select(x =>
                        {
                            var name = x?["apiname"]?.GetValue<string>();
                            if (name == null)
                            {
                                logger.LogError(
                                    "Achievement name is null for {SteamId}",
                                    _steamConfig.SteamId
                                );
                                return (KeyValuePair<string, long>?)null;
                            }
                            var unlockTime = x?["unlocktime"]?.GetValue<long>() ?? 0;
                            return x?["achieved"]?.GetValue<int>() == 1
                                ? KeyValuePair.Create(name, unlockTime)
                                : KeyValuePair.Create(name, 0L);
                        })
                        .Where(x => x != null)
                        .Select(x => x!.Value)
                        .ToDictionary(x => x.Key, x => x.Value);
                    return JsonValue.Create(achievements);
                }
            ) ?? [];
    }

    private async Task<List<SteamGame>> FetchGamesAsync(
        List<SteamGame> dbGames,
        CancellationToken cancellationToken = default
    )
    {
        logger.LogInformation("开始获取Steam游戏列表");

        var games = await ApplicationTools.RetryAsync(
            async () =>
            {
                var queryString = new Dictionary<string, string?>
                {
                    { "steamid", _steamConfig.SteamId },
                    { "key", _steamConfig.AppKey },
                    { "include_appinfo", "1" },
                    { "format", "json" },
                };

                // 添加API请求延迟
                await Task.Delay(ApiDelayMs, cancellationToken);

                var result = await ExecuteHttpRequestAsync<List<SteamGame>>(
                    "/IPlayerService/GetOwnedGames/v0001/",
                    queryString,
                    root => root?["response"]?["games"]?.AsArray()
                );

                // 如果返回null,抛出异常以触发重试
                if (result == null)
                {
                    throw new InvalidOperationException("Steam API返回null,触发重试");
                }

                return result
                    // 已失效的游戏,导致后续请求图标、成就失败
                    // 不知道为什么steam不移除它
                    // 屏蔽
                    //.Where(x => x.Id != 692850)
                    .Where(x => !_filterIgnoreGameIds.Value.Contains(x.Id))
                    .ToList();
            },
            retryInterval: TimeSpan.FromMilliseconds(ApiDelayMs),
            maxAttemptCount: 3,
            specificExceptionTypes: [typeof(JsonException)] // JSON错误不重试
        );

        logger.LogInformation("获取到 {Count} 个Steam游戏", games.Count);

#if DEBUG
        var options = new ParallelOptions { MaxDegreeOfParallelism = 1 };
#endif
        await Parallel.ForEachAsync(
            games,
#if DEBUG
            options,
#endif
            async (game, ct) => await UpdateGame(dbGames, game, ct)
        );

        return games;
    }

    private async Task UpdateGame(
        List<SteamGame> dbGames,
        SteamGame game,
        CancellationToken cancellationToken = default
    )
    {
        logger.LogDebug("更新游戏 {GameId}/{GameName} 的基本信息", game.Id, game.Name);
        var achievements = await FetchAchievementsAsync(game.Id, dbGames, cancellationToken);

        var originGame = dbGames.Find(x => x.Id == game.Id);

        if (achievements is { Count: > 0 })
        {
            game.Achievements = achievements;
            game.LastUpdateTime = DateTime.Now;
        }
        else if (originGame != null)
        {
            game.Achievements = originGame.Achievements;
        }
        else
        {
            game.Achievements = [];
            game.LastUpdateTime = DateTime.Now;
        }

        if (!AreAssetsDownloaded(originGame))
        {
            logger.LogDebug("下载游戏 {GameId} 的Logo", game.Id);

            // 添加API请求延迟
            await Task.Delay(ApiDelayMs, cancellationToken);

            await using var stream = await HttpDownloadAsync(
                string.Format(_cdnConfig.LogoUrlTemplate, game.Id)
            );
            if (stream != null && OnLogoDownloadComplete != null)
            {
                var url = await OnLogoDownloadComplete(stream, game.Id);
                if (!string.IsNullOrWhiteSpace(url))
                {
                    game.ImageLogo = url;
                    game.ImageIcon = url;
                    game.LastUpdateTime = DateTime.Now;
                }
            }
        }
        else
        {
            game.ImageLogo = originGame?.ImageLogo;
            game.ImageIcon = originGame?.ImageIcon;
            logger.LogDebug("游戏 {GameId} Logo 无需更新", game.Id);
        }
    }

    private async Task<Stream?> HttpDownloadAsync(string uri)
    {
        try
        {
            var response = await httpClient.GetAsync(uri);
            response.EnsureSuccessStatusCode();
            var memoryStream = new MemoryStream();
            await response.Content.CopyToAsync(memoryStream);
            memoryStream.Position = 0;
            return memoryStream;
        }
        catch (Exception e)
        {
            logger.LogError(e, "Download failed for {Uri}", uri);
            return null;
        }
    }

    private bool AreAssetsDownloaded(SteamGame? originGame)
    {
        return originGame != null
            && originGame.ImageIcon?.StartsWith(
                _cdnConfig.BaseUrl,
                StringComparison.OrdinalIgnoreCase
            ) == true
            && originGame.ImageLogo?.StartsWith(
                _cdnConfig.BaseUrl,
                StringComparison.OrdinalIgnoreCase
            ) == true;
    }

    private async Task DownloadAchievementAsync(
        List<SteamGame> dbGames,
        int id,
        AchievementDetail achievement
    )
    {
        var dbAchievement = dbGames
            .FirstOrDefault(x => x.Id == id)
            ?.Achievements.Find(x => x.Name == achievement.Name);

        if (
            dbAchievement?.Icon?.StartsWith(_cdnConfig.BaseUrl, StringComparison.OrdinalIgnoreCase)
            == true
        )
        {
            achievement.Icon = dbAchievement.Icon;
            return;
        }

        if (
            string.IsNullOrWhiteSpace(achievement.Icon)
            || string.IsNullOrWhiteSpace(achievement.Name)
        )
        {
            return;
        }

        logger.LogDebug("下载游戏 {GameId} 成就 {AchievementName} 的图标", id, achievement.Name);

        // 添加API请求延迟
        await Task.Delay(ApiDelayMs);

        await using var icon = await HttpDownloadAsync(achievement.Icon);
        var handler = OnAchievementIconDownloadComplete;
        if (icon != null && handler != null)
        {
            var url = await handler.Invoke(icon, id, achievement.Name);
            if (!string.IsNullOrWhiteSpace(url))
            {
                achievement.Icon = url;
            }
        }
    }

    private async Task DownloadAchievementGrayAsync(
        IEnumerable<SteamGame> dbGames,
        int id,
        AchievementDetail achievement
    )
    {
        var dbAchievement = dbGames
            .FirstOrDefault(x => x.Id == id)
            ?.Achievements.Find(x => x.Name == achievement.Name);

        if (
            dbAchievement?.IconGray?.StartsWith(
                _cdnConfig.BaseUrl,
                StringComparison.OrdinalIgnoreCase
            ) == true
        )
        {
            achievement.IconGray = dbAchievement.IconGray;
            return;
        }
        if (
            string.IsNullOrWhiteSpace(achievement.IconGray)
            || string.IsNullOrWhiteSpace(achievement.Name)
        )
        {
            return;
        }

        logger.LogDebug(
            "下载游戏 {GameId} 成就 {AchievementName} 的灰色图标",
            id,
            achievement.Name
        );

        // 添加API请求延迟
        await Task.Delay(ApiDelayMs);

        await using var icon = await HttpDownloadAsync(achievement.IconGray);
        var handler = OnAchievementIconGrayDownloadComplete;
        if (icon != null && handler != null)
        {
            var url = await handler.Invoke(icon, id, achievement.Name);
            if (!string.IsNullOrWhiteSpace(url))
            {
                achievement.IconGray = url;
            }
        }
    }

    public async Task<List<SteamGameResponse>> GetGamesAsync(
        CancellationToken cancellationToken = default
    )
    {
        return await GetOrSetCacheAsync<List<SteamGameResponse>>(
            nameof(GetGamesAsync),
            async (_, ct) =>
            {
                var list = await repository.SearchFor(x => true).ToListAsync(ct);
                var result = mapper.Map<List<SteamGameResponse>>(list);
                return result;
            },
            cancellationToken: cancellationToken
        );
    }

    public async Task<SteamGameResponse?> GetGameAsync(
        int id,
        CancellationToken cancellationToken = default
    )
    {
        return await GetOrSetCacheAsync<SteamGameResponse?>(
            nameof(GetGameAsync),
            async (_, ct) =>
            {
                var game = await repository.SearchFor(x => x.Id == id).FirstOrDefaultAsync(ct);
                if (game == null)
                {
                    logger.LogWarning("未找到游戏 {GameId} 的信息", id);
                    return null;
                }
                var result = mapper.Map<SteamGameResponse>(game);
                return result;
            },
            new { id },
            cancellationToken: cancellationToken
        );
    }

    private async Task<T?> ExecuteHttpRequestAsync<T>(
        string endpoint,
        Dictionary<string, string?> queryParams,
        Func<JsonNode?, JsonNode?>? dataSelector = null
    )
    {
        var requestUri = QueryHelpers.AddQueryString(endpoint, queryParams);
        var jsonContent = "";
        try
        {
            var response = await httpClient.GetAsync(requestUri);
            response.EnsureSuccessStatusCode();
            jsonContent = await response.Content.ReadAsStringAsync();

            if (string.IsNullOrWhiteSpace(jsonContent))
            {
                logger.LogWarning("Steam API返回空内容,请求地址:{RequestUri}", requestUri);
                return default;
            }

            var jsonNode = JsonNode.Parse(jsonContent);
            if (jsonNode == null)
            {
                logger.LogWarning(
                    "Steam API返回的JSON无法解析,请求地址:{RequestUri},内容:{Content}",
                    requestUri,
                    jsonContent
                );
                return default;
            }

            var data = dataSelector?.Invoke(jsonNode) ?? jsonNode;

            var result = data.Deserialize<T>();
            if (result == null)
            {
                logger.LogWarning(
                    "Steam API数据反序列化失败,请求地址:{RequestUri},数据:{Data}",
                    requestUri,
                    data.ToString()
                );
            }

            return result;
        }
        catch (JsonException e)
        {
            logger.LogError(
                e,
                "Steam API数据解析失败,请求地址:{RequestUri},内容:{Content}",
                requestUri,
                jsonContent
            );
            return default;
        }
        catch (Exception e)
        {
            logger.LogError(e, "Steam API请求失败,请求地址:{RequestUri}", requestUri);
            return default;
        }
    }
}
⚠⚠    以下内容为AI分析的结果,请根据实际情况进行判断。

这是一个Steam游戏服务类的完整实现,主要负责从Steam API获取游戏数据并进行管理。让我详细解释一下这个代码的结构和功能:

类概述

SteamGameService 是一个依赖注入的服务类,继承自 AbstractCacheService 并实现 ISteamGameService 接口,用于管理Steam游戏数据的获取、存储和更新。

主要功能

1. 配置管理

private record SteamConfig(string AppKey = "", string SteamId = "");
private record CdnConfig { ... }
  • SteamConfig: 存储Steam API的密钥和用户ID
  • CdnConfig: 配置CDN基础URL和图片模板路径

2. 核心方法

UpdateGamesAsync() - 游戏数据更新

这是核心方法,负责:

  • 分布式锁机制: 使用实例ID和缓存确保同时只有一个实例在执行更新
  • 数据同步:
    • 获取Steam API的游戏列表
    • 与数据库现有数据对比
    • 新增新游戏、更新已有游戏、删除已删除的游戏
  • 安全检查: 如果API返回空列表,跳过更新防止误删数据

FetchGamesAsync() - 获取游戏列表

  • 调用Steam API获取用户拥有的游戏
  • 过滤掉配置中指定忽略的游戏ID
  • 使用重试机制处理API请求失败

FetchAchievementsAsync() - 获取成就数据

  • 获取游戏的成就信息
  • 获取用户已解锁的成就状态
  • 并行下载成就图标(普通和灰度版本)

3. 资源下载功能

  • 游戏Logo下载: UpdateGame() 方法中处理
  • 成就图标下载: 包括彩色和灰度两种版本
  • 缓存机制: 避免重复下载已存在的资源

4. HTTP请求处理

ExecuteHttpRequestAsync<T>() - 通用HTTP请求方法

  • 构建查询参数
  • 发送HTTP请求
  • JSON解析和反序列化
  • 错误处理和日志记录

5. 数据访问方法

  • GetGamesAsync(): 获取所有游戏列表(带缓存)
  • GetGameAsync(): 获取单个游戏信息(带缓存)

设计特点

1. 防止API过频繁调用

private const int ApiDelayMs = 500;
await Task.Delay(ApiDelayMs, cancellationToken);

在每个API调用间添加500ms延迟

2. 并发控制

  • 使用实例ID和分布式缓存实现分布式锁
  • 防止多个实例同时执行数据更新

3. 错误处理

  • 全面的异常处理和日志记录
  • API请求失败时的重试机制
  • 资源下载失败的优雅降级

4. 性能优化

  • 并行处理游戏和成就数据
  • 缓存机制减少重复查询
  • 条件性资源下载(仅在需要时下载)

5. 事件机制

public event LogoDownload? OnLogoDownloadComplete;
public event AchievementIconGrayDownload? OnAchievementIconGrayDownloadComplete;
public event AchievementIconDownload? OnAchievementIconDownloadComplete;

通过事件通知外部系统处理下载完成的资源

使用场景

这个服务适用于需要:

  • 同步Steam游戏库数据
  • 管理游戏成就信息
  • 下载和缓存游戏相关图片资源
  • 提供Steam游戏数据的API接口

整体来说,这是一个设计良好的企业级服务,具有完善的错误处理、缓存机制和并发控制。

评论加载中...