2020-11-24 17:19:26 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Discord;
|
|
|
|
|
using Discord.WebSocket;
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
using OMODFramework;
|
|
|
|
|
using Wabbajack.BuildServer;
|
2020-12-07 04:26:02 +00:00
|
|
|
|
using Wabbajack.Common;
|
2020-11-24 17:19:26 +00:00
|
|
|
|
using Wabbajack.Server.DataLayer;
|
|
|
|
|
using Utils = Wabbajack.Common.Utils;
|
|
|
|
|
|
|
|
|
|
namespace Wabbajack.Server.Services
|
|
|
|
|
{
|
|
|
|
|
public class DiscordFrontend : IStartable
|
|
|
|
|
{
|
|
|
|
|
private ILogger<DiscordFrontend> _logger;
|
|
|
|
|
private AppSettings _settings;
|
|
|
|
|
private QuickSync _quickSync;
|
|
|
|
|
private DiscordSocketClient _client;
|
|
|
|
|
private SqlService _sql;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
private MetricsKeyCache _keyCache;
|
2021-03-11 02:28:28 +00:00
|
|
|
|
private ListValidator _listValidator;
|
2020-11-24 17:19:26 +00:00
|
|
|
|
|
2021-03-11 02:28:28 +00:00
|
|
|
|
public DiscordFrontend(ILogger<DiscordFrontend> logger, AppSettings settings, QuickSync quickSync, ListValidator listValidator, SqlService sql, MetricsKeyCache keyCache)
|
2020-11-24 17:19:26 +00:00
|
|
|
|
{
|
|
|
|
|
_logger = logger;
|
|
|
|
|
_settings = settings;
|
|
|
|
|
_quickSync = quickSync;
|
|
|
|
|
|
|
|
|
|
_client = new DiscordSocketClient();
|
|
|
|
|
|
|
|
|
|
_client.Log += LogAsync;
|
|
|
|
|
_client.Ready += ReadyAsync;
|
|
|
|
|
_client.MessageReceived += MessageReceivedAsync;
|
|
|
|
|
|
|
|
|
|
_sql = sql;
|
2021-02-18 05:44:54 +00:00
|
|
|
|
_keyCache = keyCache;
|
2021-03-11 02:28:28 +00:00
|
|
|
|
_listValidator = listValidator;
|
2020-11-24 17:19:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task MessageReceivedAsync(SocketMessage arg)
|
|
|
|
|
{
|
|
|
|
|
_logger.LogInformation(arg.Content);
|
|
|
|
|
if (arg.Content.StartsWith("!dervenin"))
|
|
|
|
|
{
|
|
|
|
|
var parts = arg.Content.Split(" ", StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
if (parts[0] != "!dervenin")
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (parts.Length == 1)
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(arg, "Wat?");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (parts[1] == "purge-nexus-cache")
|
|
|
|
|
{
|
|
|
|
|
if (parts.Length != 3)
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(arg, "Welp you did that wrong, gotta give me a mod-id or url");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
await PurgeNexusCache(arg, parts[2]);
|
|
|
|
|
}
|
2021-02-04 03:48:30 +00:00
|
|
|
|
else if (parts[1] == "quick-sync")
|
2020-12-07 04:26:02 +00:00
|
|
|
|
{
|
2021-02-04 03:48:30 +00:00
|
|
|
|
var options = await _quickSync.Report();
|
|
|
|
|
if (parts.Length != 3)
|
|
|
|
|
{
|
|
|
|
|
var optionsStr = string.Join(", ", options.Select(o => o.Key.Name));
|
|
|
|
|
await ReplyTo(arg, $"Can't expect me to quicksync the whole damn world! Try: {optionsStr}");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
foreach (var pair in options.Where(o => o.Key.Name == parts[2]))
|
|
|
|
|
{
|
|
|
|
|
await _quickSync.Notify(pair.Key);
|
|
|
|
|
await ReplyTo(arg, $"Notified {pair.Key}");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (parts[1] == "purge-list")
|
|
|
|
|
{
|
|
|
|
|
if (parts.Length != 3)
|
2020-12-07 04:26:02 +00:00
|
|
|
|
{
|
2021-02-04 03:48:30 +00:00
|
|
|
|
await ReplyTo(arg, $"Yeah, I'm not gonna purge the whole server...");
|
2020-12-07 04:26:02 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-02-04 03:48:30 +00:00
|
|
|
|
var deleted = await _sql.PurgeList(parts[2]);
|
2021-03-11 02:28:28 +00:00
|
|
|
|
_listValidator.ValidationInfo.TryRemove(parts[2], out var _);
|
2021-02-04 03:48:30 +00:00
|
|
|
|
await _quickSync.Notify<ModListDownloader>();
|
|
|
|
|
await ReplyTo(arg, $"Purged all traces of #{parts[2]} from the server, triggered list downloading. {deleted} records removed");
|
2020-12-07 04:26:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-03-11 02:28:28 +00:00
|
|
|
|
else if (parts[1] == "mirror-mod")
|
|
|
|
|
{
|
|
|
|
|
await MirrorModCommand(arg, parts);
|
|
|
|
|
}
|
2021-02-18 05:44:54 +00:00
|
|
|
|
else if (parts[1] == "users")
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(arg, $"Wabbajack has {await _keyCache.KeyCount()} known unique users");
|
|
|
|
|
}
|
2020-11-24 17:19:26 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-11 02:28:28 +00:00
|
|
|
|
private async Task MirrorModCommand(SocketMessage msg, string[] parts)
|
|
|
|
|
{
|
|
|
|
|
if (parts.Length != 2)
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(msg, "Command is: mirror-mod <game-name> <mod-id>");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (long.TryParse(parts[2], out var modId))
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(msg, $"Got {modId} for a mod-id, expected a integer");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (GameRegistry.TryGetByFuzzyName(parts[1], out var game))
|
|
|
|
|
{
|
|
|
|
|
var gameNames = GameRegistry.Games.Select(g => g.Value.NexusName)
|
|
|
|
|
.Where(g => !string.IsNullOrWhiteSpace(g))
|
|
|
|
|
.Select(g => (string)g)
|
|
|
|
|
.ToHashSet();
|
|
|
|
|
var joined = string.Join(", ", gameNames.OrderBy(g => g));
|
|
|
|
|
await ReplyTo(msg, $"Got {parts[1]} for a game name, expected something like: {joined}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (game!.NexusGameId == default)
|
|
|
|
|
{
|
|
|
|
|
await ReplyTo(msg, $"No NexusGameID found for {game}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
await _sql.AddNexusModWithOpenPerms(game.Game, modId);
|
|
|
|
|
await _quickSync.Notify<MirrorUploader>();
|
|
|
|
|
await ReplyTo(msg, "Done, and I notified the uploader");
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-24 17:19:26 +00:00
|
|
|
|
private async Task PurgeNexusCache(SocketMessage arg, string mod)
|
|
|
|
|
{
|
|
|
|
|
if (Uri.TryCreate(mod, UriKind.Absolute, out var url))
|
|
|
|
|
{
|
2020-12-07 04:26:02 +00:00
|
|
|
|
mod = Enumerable.Last(url.AbsolutePath.Split("/", StringSplitOptions.RemoveEmptyEntries));
|
2020-11-24 17:19:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (int.TryParse(mod, out var mod_id))
|
|
|
|
|
{
|
|
|
|
|
await _sql.PurgeNexusCache(mod_id);
|
|
|
|
|
await _quickSync.Notify<ListValidator>();
|
|
|
|
|
await ReplyTo(arg, $"It is done, {mod_id} has been purged, list validation has been triggered");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task ReplyTo(SocketMessage socketMessage, string message)
|
|
|
|
|
{
|
|
|
|
|
await socketMessage.Channel.SendMessageAsync(message);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task ReadyAsync()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task LogAsync(LogMessage arg)
|
|
|
|
|
{
|
|
|
|
|
switch (arg.Severity)
|
|
|
|
|
{
|
|
|
|
|
case LogSeverity.Info:
|
|
|
|
|
_logger.LogInformation(arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
case LogSeverity.Warning:
|
|
|
|
|
_logger.LogWarning(arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
case LogSeverity.Critical:
|
|
|
|
|
_logger.LogCritical(arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
case LogSeverity.Error:
|
|
|
|
|
_logger.LogError(arg.Exception, arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
case LogSeverity.Verbose:
|
|
|
|
|
_logger.LogTrace(arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
case LogSeverity.Debug:
|
|
|
|
|
_logger.LogDebug(arg.Message);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
throw new ArgumentOutOfRangeException();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Start()
|
|
|
|
|
{
|
|
|
|
|
_client.LoginAsync(TokenType.Bot, Utils.FromEncryptedJson<string>("discord-key").Result).Wait();
|
|
|
|
|
_client.StartAsync().Wait();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|