2020-05-09 03:56:06 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Microsoft.AspNetCore.Authorization;
|
|
|
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
|
|
|
|
|
using Wabbajack.Common;
|
2020-08-12 04:25:12 +00:00
|
|
|
|
using Wabbajack.Common.Exceptions;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
using Wabbajack.Lib.NexusApi;
|
|
|
|
|
using Wabbajack.Server.DataLayer;
|
2020-08-12 04:25:12 +00:00
|
|
|
|
using Wabbajack.Server.Services;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
|
|
|
|
|
namespace Wabbajack.BuildServer.Controllers
|
|
|
|
|
{
|
|
|
|
|
//[Authorize]
|
|
|
|
|
[ApiController]
|
2020-06-16 22:21:01 +00:00
|
|
|
|
[Authorize(Roles = "User")]
|
2020-05-09 03:56:06 +00:00
|
|
|
|
[Route("/v1/games/")]
|
|
|
|
|
public class NexusCache : ControllerBase
|
|
|
|
|
{
|
|
|
|
|
private AppSettings _settings;
|
|
|
|
|
private static long CachedCount = 0;
|
|
|
|
|
private static long ForwardCount = 0;
|
|
|
|
|
private SqlService _sql;
|
|
|
|
|
private ILogger<NexusCache> _logger;
|
2020-08-12 04:25:12 +00:00
|
|
|
|
private NexusKeyMaintainance _keys;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
|
2020-08-12 04:25:12 +00:00
|
|
|
|
public NexusCache(ILogger<NexusCache> logger, SqlService sql, AppSettings settings, NexusKeyMaintainance keys)
|
2020-05-09 03:56:06 +00:00
|
|
|
|
{
|
|
|
|
|
_settings = settings;
|
|
|
|
|
_sql = sql;
|
|
|
|
|
_logger = logger;
|
2020-08-12 04:25:12 +00:00
|
|
|
|
_keys = keys;
|
2020-05-09 03:56:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Looks up the mod details for a given Gamename/ModId pair. If the entry is not found in the cache it will
|
|
|
|
|
/// be requested from the server (using the caller's Nexus API key if provided).
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="db"></param>
|
|
|
|
|
/// <param name="GameName">The Nexus game name</param>
|
|
|
|
|
/// <param name="ModId">The Nexus mod id</param>
|
|
|
|
|
/// <returns>A Mod Info result</returns>
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{GameName}/mods/{ModId}.json")]
|
|
|
|
|
public async Task<ModInfo> GetModInfo(string GameName, long ModId)
|
|
|
|
|
{
|
|
|
|
|
var game = GameRegistry.GetByFuzzyName(GameName).Game;
|
|
|
|
|
var result = await _sql.GetNexusModInfoString(game, ModId);
|
|
|
|
|
|
|
|
|
|
string method = "CACHED";
|
|
|
|
|
if (result == null)
|
|
|
|
|
{
|
2020-05-16 15:08:40 +00:00
|
|
|
|
var api = await GetClient();
|
2020-05-09 03:56:06 +00:00
|
|
|
|
result = await api.GetModInfo(game, ModId, false);
|
2020-05-12 23:19:54 +00:00
|
|
|
|
await _sql.AddNexusModInfo(game, ModId, result.updated_time, result);
|
2020-05-09 03:56:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
method = "NOT_CACHED";
|
|
|
|
|
Interlocked.Increment(ref ForwardCount);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Interlocked.Increment(ref CachedCount);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Response.Headers.Add("x-cache-result", method);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-16 15:08:40 +00:00
|
|
|
|
private async Task<NexusApiClient> GetClient()
|
|
|
|
|
{
|
|
|
|
|
var key = Request.Headers["apikey"].FirstOrDefault();
|
|
|
|
|
if (key == null)
|
2020-08-12 04:25:12 +00:00
|
|
|
|
return await _keys.GetClient();
|
2020-05-16 15:08:40 +00:00
|
|
|
|
|
|
|
|
|
if (await _sql.HaveKey(key))
|
|
|
|
|
return await NexusApiClient.Get(key);
|
|
|
|
|
|
|
|
|
|
var client = await NexusApiClient.Get(key);
|
|
|
|
|
var (daily, hourly) = await client.GetRemainingApiCalls();
|
|
|
|
|
await _sql.SetNexusAPIKey(key, daily, hourly);
|
|
|
|
|
return client;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 03:56:06 +00:00
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{GameName}/mods/{ModId}/files.json")]
|
|
|
|
|
public async Task<NexusApiClient.GetModFilesResponse> GetModFiles(string GameName, long ModId)
|
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
//_logger.Log(LogLevel.Information, $"{GameName} {ModId}");
|
2020-05-09 03:56:06 +00:00
|
|
|
|
var game = GameRegistry.GetByFuzzyName(GameName).Game;
|
|
|
|
|
var result = await _sql.GetModFiles(game, ModId);
|
|
|
|
|
|
|
|
|
|
string method = "CACHED";
|
|
|
|
|
if (result == null)
|
|
|
|
|
{
|
2020-08-12 04:25:12 +00:00
|
|
|
|
var api = await GetClient();
|
|
|
|
|
var permission = HTMLInterface.GetUploadPermissions(game, ModId);
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
result = await api.GetModFiles(game, ModId, false);
|
|
|
|
|
}
|
|
|
|
|
catch (HttpException ex)
|
|
|
|
|
{
|
|
|
|
|
if (ex.Code == 403)
|
|
|
|
|
result = new NexusApiClient.GetModFilesResponse {files = new List<NexusFileInfo>()};
|
|
|
|
|
else
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 23:19:54 +00:00
|
|
|
|
var date = result.files.Select(f => f.uploaded_time).OrderByDescending(o => o).FirstOrDefault();
|
|
|
|
|
date = date == default ? DateTime.UtcNow : date;
|
|
|
|
|
await _sql.AddNexusModFiles(game, ModId, date, result);
|
2020-08-12 04:25:12 +00:00
|
|
|
|
await _sql.SetNexusPermission(game, ModId, await permission);
|
2020-05-09 03:56:06 +00:00
|
|
|
|
|
|
|
|
|
method = "NOT_CACHED";
|
|
|
|
|
Interlocked.Increment(ref ForwardCount);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Interlocked.Increment(ref CachedCount);
|
|
|
|
|
}
|
|
|
|
|
Response.Headers.Add("x-cache-result", method);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
2021-06-24 23:01:03 +00:00
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Route("{GameName}/mods/{ModId}/files/{FileId}.json")]
|
|
|
|
|
public async Task<ActionResult<NexusFileInfo>> GetModFile(string GameName, long ModId, long FileId)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var game = GameRegistry.GetByFuzzyName(GameName).Game;
|
|
|
|
|
var result = await _sql.GetModFile(game, ModId, FileId);
|
|
|
|
|
|
|
|
|
|
string method = "CACHED";
|
|
|
|
|
if (result == null)
|
|
|
|
|
{
|
|
|
|
|
var api = await GetClient();
|
|
|
|
|
result = await api.GetModFile(game, ModId, FileId, false);
|
|
|
|
|
|
|
|
|
|
var date = result.uploaded_time;
|
|
|
|
|
date = date == default ? DateTime.UtcNow : date;
|
|
|
|
|
await _sql.AddNexusModFile(game, ModId, FileId, date, result);
|
|
|
|
|
|
|
|
|
|
method = "NOT_CACHED";
|
|
|
|
|
Interlocked.Increment(ref ForwardCount);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Interlocked.Increment(ref CachedCount);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Response.Headers.Add("x-cache-result", method);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogInformation("Unable to find mod file {GameName} {ModId}, {FileId}", GameName, ModId, FileId);
|
|
|
|
|
return NotFound();
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-07-08 20:48:48 +00:00
|
|
|
|
|
|
|
|
|
[HttpGet]
|
|
|
|
|
[Authorize(Roles ="Author")]
|
|
|
|
|
[Route("/purge_nexus_cache/{ModId}")]
|
|
|
|
|
public async Task<IActionResult> PurgeNexusCache(long ModId)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogInformation($"Purging nexus cache for {ModId}");
|
|
|
|
|
await _sql.PurgeNexusCache(ModId);
|
|
|
|
|
return Ok("Purged");
|
|
|
|
|
}
|
2020-05-09 03:56:06 +00:00
|
|
|
|
}
|
|
|
|
|
}
|