mirror of
https://github.com/RPCS3/discord-bot.git
synced 2026-01-31 01:25:22 +01:00
190 lines
8.5 KiB
C#
190 lines
8.5 KiB
C#
using System.Text.RegularExpressions;
|
|
using CompatApiClient;
|
|
using CompatApiClient.POCOs;
|
|
using CompatBot.Commands;
|
|
using CompatBot.Database.Providers;
|
|
using CompatBot.Utils.ResultFormatters;
|
|
|
|
namespace CompatBot.EventHandlers;
|
|
|
|
internal static partial class ProductCodeLookup
|
|
{
|
|
// see http://www.psdevwiki.com/ps3/Productcode
|
|
[GeneratedRegex(@"(?<letters>(?:[BPSUVX][CL]|P[ETU]|NP)[AEHJKPUIX][ABDJKLMOPQRSTX]|MRTC)[ \-]?(?<numbers>\d{5})", RegexOptions.IgnoreCase | RegexOptions.Compiled, "en-GB")]
|
|
public static partial Regex Pattern();
|
|
private static readonly Client CompatClient = new();
|
|
|
|
public static async Task OnMessageCreated(DiscordClient c, MessageCreatedEventArgs args)
|
|
{
|
|
if (DefaultHandlerFilter.IsFluff(args.Message))
|
|
return;
|
|
|
|
var lastBotMessages = await args.Channel.GetMessagesBeforeAsync(args.Message.Id, 20, DateTime.UtcNow.AddSeconds(-30)).ConfigureAwait(false);
|
|
if (lastBotMessages.Any(msg => BotReactionsHandler.NeedToSilence(msg).needToChill))
|
|
return;
|
|
|
|
lastBotMessages = await args.Channel.GetMessagesBeforeCachedAsync(args.Message.Id, Config.ProductCodeLookupHistoryThrottle).ConfigureAwait(false);
|
|
StringBuilder? previousRepliesBuilder = null;
|
|
foreach (var msg in lastBotMessages.Where(m => m.Author.IsCurrent))
|
|
{
|
|
previousRepliesBuilder ??= new StringBuilder();
|
|
previousRepliesBuilder.AppendLine(msg.Content);
|
|
var embeds = msg.Embeds;
|
|
if (embeds?.Count > 0)
|
|
foreach (var embed in embeds)
|
|
previousRepliesBuilder.AppendLine(embed.Title).AppendLine(embed.Description);
|
|
}
|
|
var previousReplies = previousRepliesBuilder?.ToString() ?? "";
|
|
|
|
var codesToLookup = GetProductIds(args.Message.Content)
|
|
.Where(pc => !previousReplies.Contains(pc, StringComparison.InvariantCultureIgnoreCase))
|
|
.Take(args.Channel.IsPrivate ? 50 : 5)
|
|
.ToList();
|
|
if (codesToLookup.Count == 0)
|
|
return;
|
|
|
|
await LookupAndPostProductCodeEmbedAsync(c, args.Message, args.Channel, codesToLookup).ConfigureAwait(false);
|
|
}
|
|
|
|
public static async ValueTask LookupAndPostProductCodeEmbedAsync(DiscordClient client, DiscordMessage message, DiscordChannel channel, List<string> codesToLookup)
|
|
{
|
|
await message.ReactWithAsync(Config.Reactions.PleaseWait).ConfigureAwait(false);
|
|
try
|
|
{
|
|
var formattedResults = await LookupProductCodeAndFormatAsync(client, codesToLookup).ConfigureAwait(false);
|
|
foreach (var result in formattedResults)
|
|
try
|
|
{
|
|
var messageBuilder = new DiscordMessageBuilder().AddEmbed(result.builder);
|
|
//todo: pass author from context and update OnCheckUpdatesButtonClick in psn check updates
|
|
if (message is {Author: not null} && channel.IsSpamChannel())
|
|
messageBuilder.AddActionRowComponent(
|
|
new DiscordButtonComponent(
|
|
DiscordButtonStyle.Secondary,
|
|
$"{GlobalButtonHandler.ReplaceWithUpdatesPrefix}{result.code}",
|
|
"How to check for updates",
|
|
emoji: new(DiscordEmoji.FromUnicode("❓"))
|
|
)
|
|
);
|
|
await DiscordMessageExtensions.UpdateOrCreateMessageAsync(null, channel, messageBuilder).ConfigureAwait(false);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Config.Log.Warn(e, $"Couldn't post result for {result.code} ({result.builder.Title})");
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
await message.RemoveReactionAsync(Config.Reactions.PleaseWait).ConfigureAwait(false);
|
|
}
|
|
}
|
|
|
|
internal static async ValueTask<List<(string code, DiscordEmbedBuilder builder)>> LookupProductCodeAndFormatAsync(DiscordClient client, List<string> codesToLookup)
|
|
{
|
|
var results = codesToLookup.Select(code => (code, client.LookupGameInfoAsync(code))).ToList();
|
|
var formattedResults = new List<(string code, DiscordEmbedBuilder builder)>(results.Count);
|
|
foreach (var (code, task) in results)
|
|
try
|
|
{
|
|
formattedResults.Add((code, await task.ConfigureAwait(false)));
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Config.Log.Warn(e, $"Couldn't get product code info for {code}");
|
|
}
|
|
// get only results with unique titles
|
|
return formattedResults.DistinctBy(e => e.builder.Title).ToList();
|
|
}
|
|
|
|
public static List<string> GetProductIds(string? input)
|
|
{
|
|
if (string.IsNullOrEmpty(input))
|
|
return [];
|
|
|
|
return Pattern().Matches(input)
|
|
.Select(match => (match.Groups["letters"].Value + match.Groups["numbers"]).ToUpper())
|
|
.Distinct()
|
|
.ToList();
|
|
}
|
|
|
|
public static async Task<DiscordEmbedBuilder> LookupGameInfoAsync(this DiscordClient client, string? code, string? gameTitle = null, bool forLog = false, string? category = null)
|
|
=> (await LookupGameInfoWithEmbedAsync(client, code, gameTitle, forLog, category).ConfigureAwait(false)).embedBuilder;
|
|
|
|
public static async ValueTask<(DiscordEmbedBuilder embedBuilder, CompatResult? compatResult)> LookupGameInfoWithEmbedAsync(this DiscordClient client, string? code, string? gameTitle = null, bool forLog = false, string? category = null)
|
|
{
|
|
if (string.IsNullOrEmpty(code))
|
|
return (
|
|
await TitleInfo.Unknown.AsEmbedAsync(code, gameTitle, forLog).ConfigureAwait(false),
|
|
null
|
|
);
|
|
|
|
string? thumbnailUrl = null;
|
|
CompatResult? result = null;
|
|
try
|
|
{
|
|
var requestBuilder = RequestBuilder.Start().SetSearch(code);
|
|
var searchCompatListTask = CompatClient.GetCompatResultAsync(requestBuilder, Config.Cts.Token);
|
|
var localList = await CompatList.GetLocalCompatResultAsync(requestBuilder).ConfigureAwait(false);
|
|
var status = await searchCompatListTask.ConfigureAwait(false);
|
|
result = status?.Append(localList);
|
|
if (result?.ReturnCode is CompatApiStatus.Maintenance)
|
|
return (
|
|
await TitleInfo.Maintenance.AsEmbedAsync(code).ConfigureAwait(false),
|
|
result
|
|
);
|
|
|
|
if (result?.ReturnCode is CompatApiStatus.InternalError)
|
|
return (
|
|
await TitleInfo.CommunicationError.AsEmbedAsync(code).ConfigureAwait(false),
|
|
result
|
|
);
|
|
|
|
thumbnailUrl = await client.GetThumbnailUrlAsync(code).ConfigureAwait(false);
|
|
if (result?.Results != null && result.Results.TryGetValue(code, out var info))
|
|
return (
|
|
await info.AsEmbedAsync(code, gameTitle, forLog, thumbnailUrl).ConfigureAwait(false),
|
|
result
|
|
);
|
|
|
|
gameTitle ??= await ThumbnailProvider.GetTitleNameAsync(code, Config.Cts.Token).ConfigureAwait(false);
|
|
if (category == "1P")
|
|
{
|
|
var ti = new TitleInfo
|
|
{
|
|
Commit = "8b449ce76c91d5ff7a2829b233befe7d6df4b24f",
|
|
Date = "2018-06-23",
|
|
Pr = 4802,
|
|
Status = "Playable",
|
|
};
|
|
return (
|
|
await ti.AsEmbedAsync(code, gameTitle, forLog, thumbnailUrl).ConfigureAwait(false),
|
|
result
|
|
);
|
|
}
|
|
if (category is "2P" or "2G" or "2D" or "PP" or "PE" or "MN")
|
|
{
|
|
var ti = new TitleInfo
|
|
{
|
|
Status = "Nothing"
|
|
};
|
|
return (
|
|
await ti.AsEmbedAsync(code, gameTitle, forLog, thumbnailUrl).ConfigureAwait(false),
|
|
result
|
|
);
|
|
}
|
|
return (
|
|
await TitleInfo.Unknown.AsEmbedAsync(code, gameTitle, forLog, thumbnailUrl).ConfigureAwait(false),
|
|
result
|
|
);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Config.Log.Warn(e, $"Couldn't get compat result for {code}");
|
|
return (
|
|
await TitleInfo.CommunicationError.AsEmbedAsync(code, gameTitle, forLog, thumbnailUrl).ConfigureAwait(false),
|
|
result
|
|
);
|
|
}
|
|
}
|
|
}
|