mirror of
https://github.com/barelyprofessional/KfChatDotNet.git
synced 2026-04-30 03:22:04 -04:00
224 lines
10 KiB
C#
224 lines
10 KiB
C#
using System.Net.Http.Headers;
|
|
using System.Text.RegularExpressions;
|
|
using Humanizer;
|
|
using KfChatDotNetBot.Extensions;
|
|
using KfChatDotNetBot.Models;
|
|
using KfChatDotNetBot.Models.DbModels;
|
|
using KfChatDotNetBot.Services;
|
|
using KfChatDotNetBot.Settings;
|
|
using KfChatDotNetWsClient.Models.Events;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using NLog;
|
|
|
|
namespace KfChatDotNetBot.Commands;
|
|
|
|
public class AddImageCommand : ICommand
|
|
{
|
|
public List<Regex> Patterns => [
|
|
new Regex(@"^admin image (?<key>\w+) add (?<url>.+)$"),
|
|
new Regex(@"^admin images (?<key>\w+) add (?<url>.+)$"),
|
|
new Regex(@"^admin image (?<key>\w+) add_nigger (?<url>.+)$"),
|
|
new Regex(@"^admin images (?<key>\w+) add_nigger (?<url>.+)$")
|
|
];
|
|
public string? HelpText => "Add an image to the image rotation specified";
|
|
public UserRight RequiredRight => UserRight.TrueAndHonest;
|
|
public TimeSpan Timeout => TimeSpan.FromSeconds(10);
|
|
public RateLimitOptionsModel? RateLimitOptions => null;
|
|
public bool WhisperCanInvoke => false;
|
|
public async Task RunCommand(ChatBot botInstance, BotCommandMessageModel message, UserDbModel user, GroupCollection arguments,
|
|
CancellationToken ctx)
|
|
{
|
|
await using var db = new ApplicationDbContext();
|
|
var imageKeys = (await SettingsProvider.GetValueAsync(BuiltIn.Keys.BotImageAcceptableKeys)).JsonDeserialize<List<string>>();
|
|
if (imageKeys == null) throw new InvalidOperationException($"{BuiltIn.Keys.BotImageAcceptableKeys} was null");
|
|
var key = arguments["key"].Value;
|
|
var url = arguments["url"].Value;
|
|
var niggerMode = message.Message.Contains("add_nigger");
|
|
if (!imageKeys.Contains(key))
|
|
{
|
|
await botInstance.SendChatMessageAsync(
|
|
$"Key you specified is not supported. Available keys are: {string.Join(' ', imageKeys)}", true);
|
|
return;
|
|
}
|
|
|
|
if (!niggerMode && await db.Images.AnyAsync(i => i.Key == key && i.Url == url, ctx))
|
|
{
|
|
await botInstance.SendChatMessageAsync("This image already exists in the database with this key", true);
|
|
return;
|
|
}
|
|
|
|
await db.Images.AddAsync(new ImageDbModel { Key = key, Url = url, LastSeen = DateTimeOffset.MinValue }, ctx);
|
|
await db.SaveChangesAsync(ctx);
|
|
//await botInstance.SendChatMessageAsync("Added image to database", true);
|
|
await botInstance.SendChatMessageAsync(
|
|
$"{user.FormatUsername()}, you added the following media to the {key} carousel\n[img]{url}[/img]", true);
|
|
}
|
|
}
|
|
|
|
public class RemoveImageCommand : ICommand
|
|
{
|
|
public List<Regex> Patterns => [
|
|
new Regex(@"^admin image (?<key>\w+) remove (?<url>.+)$"),
|
|
new Regex(@"^admin images (?<key>\w+) remove (?<url>.+)$"),
|
|
new Regex(@"^admin image (?<key>\w+) delete (?<url>.+)$"),
|
|
new Regex(@"^admin images (?<key>\w+) delete (?<url>.+)$")
|
|
];
|
|
public string? HelpText => "Remove an image from the image rotation specified";
|
|
public UserRight RequiredRight => UserRight.TrueAndHonest;
|
|
public TimeSpan Timeout => TimeSpan.FromSeconds(10);
|
|
public RateLimitOptionsModel? RateLimitOptions => null;
|
|
public bool WhisperCanInvoke => false;
|
|
public async Task RunCommand(ChatBot botInstance, BotCommandMessageModel message, UserDbModel user, GroupCollection arguments,
|
|
CancellationToken ctx)
|
|
{
|
|
await using var db = new ApplicationDbContext();
|
|
var imageKeys = (await SettingsProvider.GetValueAsync(BuiltIn.Keys.BotImageAcceptableKeys)).JsonDeserialize<List<string>>();
|
|
if (imageKeys == null) throw new InvalidOperationException($"{BuiltIn.Keys.BotImageAcceptableKeys} was null");
|
|
var key = arguments["key"].Value;
|
|
var url = arguments["url"].Value;
|
|
if (!imageKeys.Contains(key))
|
|
{
|
|
await botInstance.SendChatMessageAsync(
|
|
$"Key you specified is not supported. Available keys are: {string.Join(' ', imageKeys)}", true);
|
|
return;
|
|
}
|
|
|
|
var image = await db.Images.FirstOrDefaultAsync(i => i.Key == key && i.Url == url, ctx);
|
|
if (image == null)
|
|
{
|
|
await botInstance.SendChatMessageAsync("This image isn't in the database with this key", true);
|
|
return;
|
|
}
|
|
|
|
db.Images.Remove(image);
|
|
await db.SaveChangesAsync(ctx);
|
|
// await botInstance.SendChatMessageAsync("Removed image from database", true);
|
|
await botInstance.SendChatMessageAsync(
|
|
$"{user.FormatUsername()}, you removed the following media from the {key} carousel\n[img]{url}[/img]", true);
|
|
}
|
|
}
|
|
|
|
public class ListImageCommand : ICommand
|
|
{
|
|
public List<Regex> Patterns => [
|
|
new Regex(@"^admin image (?<key>\w+) list$"),
|
|
new Regex(@"^admin images (?<key>\w+) list$")
|
|
];
|
|
public string? HelpText => "Remove an image from the image rotation specified";
|
|
public UserRight RequiredRight => UserRight.TrueAndHonest;
|
|
public TimeSpan Timeout => TimeSpan.FromSeconds(10);
|
|
public RateLimitOptionsModel? RateLimitOptions => null;
|
|
public bool WhisperCanInvoke => false;
|
|
public async Task RunCommand(ChatBot botInstance, BotCommandMessageModel message, UserDbModel user, GroupCollection arguments,
|
|
CancellationToken ctx)
|
|
{
|
|
await using var db = new ApplicationDbContext();
|
|
var imageKeys = (await SettingsProvider.GetValueAsync(BuiltIn.Keys.BotImageAcceptableKeys))
|
|
.JsonDeserialize<List<string>>();
|
|
if (imageKeys == null) throw new InvalidOperationException($"{BuiltIn.Keys.BotImageAcceptableKeys} was null");
|
|
var key = arguments["key"].Value;
|
|
if (!imageKeys.Contains(key))
|
|
{
|
|
await botInstance.SendChatMessageAsync(
|
|
$"Key you specified is not supported. Available keys are: {string.Join(' ', imageKeys)}", true);
|
|
return;
|
|
}
|
|
|
|
var images = db.Images.Where(i => i.Key == key);
|
|
if (await images.CountAsync(cancellationToken: ctx) > 20 && await Zipline.IsZiplineEnabled())
|
|
{
|
|
var content = string.Empty;
|
|
foreach (var image in images)
|
|
{
|
|
content += image.Url + Environment.NewLine;
|
|
}
|
|
|
|
var paste = await Zipline.Upload(content, new MediaTypeHeaderValue("text/plain"), "1d", ctx);
|
|
await botInstance.SendChatMessageAsync($"List of images for {key}: {paste}", true);
|
|
return;
|
|
}
|
|
|
|
var i = 0;
|
|
var result = $"List of images for {key}:";
|
|
foreach (var image in images)
|
|
{
|
|
i++;
|
|
var ts = DateTimeOffset.UtcNow - image.LastSeen;
|
|
result += $"[br]{i}: {image.Url} (Last seen {ts.TotalDays:N0}d{ts.Hours:N0}h{ts.Minutes:N0}m{ts.Seconds:N0}s ago)";
|
|
}
|
|
|
|
await botInstance.SendChatMessagesAsync(result.FancySplitMessage(partSeparator: "[br]"),
|
|
bypassSeshDetect: true);
|
|
}
|
|
}
|
|
|
|
[AllowAdditionalMatches]
|
|
public class GetRandomImage : ICommand
|
|
{
|
|
public List<Regex> Patterns => [
|
|
new Regex(@"^(?<key>\w+)")
|
|
];
|
|
public string? HelpText => "Get a random image";
|
|
public UserRight RequiredRight => UserRight.Loser;
|
|
public TimeSpan Timeout => TimeSpan.FromMinutes(10);
|
|
public RateLimitOptionsModel? RateLimitOptions => new()
|
|
{
|
|
Window = TimeSpan.FromSeconds(30),
|
|
MaxInvocations = 7,
|
|
Flags = RateLimitFlags.UseEntireMessage | RateLimitFlags.NoResponse
|
|
};
|
|
public bool WhisperCanInvoke => false;
|
|
public async Task RunCommand(ChatBot botInstance, BotCommandMessageModel message, UserDbModel user, GroupCollection arguments,
|
|
CancellationToken ctx)
|
|
{
|
|
await using var db = new ApplicationDbContext();
|
|
var key = arguments["key"].Value.ToLower();
|
|
var images = db.Images.Where(i => i.Key == key);
|
|
if (!await images.AnyAsync(ctx))
|
|
{
|
|
RateLimitService.RemoveMostRecentEntry(user, this);
|
|
return;
|
|
}
|
|
|
|
if (key == "sloppa" && user.UserRight < UserRight.TrueAndHonest)
|
|
{
|
|
await botInstance.SendWhisperAsync(user.KfId, $"{user.FormatUsername()}, sloppa requires at least {UserRight.TrueAndHonest.Humanize()}");
|
|
return;
|
|
}
|
|
var settings = await SettingsProvider.GetMultipleValuesAsync([
|
|
BuiltIn.Keys.BotImageRandomSliceDivideBy, BuiltIn.Keys.BotImagePigCubeSelfDestruct,
|
|
BuiltIn.Keys.BotImageInvertedCubeUrl, BuiltIn.Keys.BotImagePigCubeSelfDestructMin,
|
|
BuiltIn.Keys.BotImagePigCubeSelfDestructMax, BuiltIn.Keys.BotImageInvertedPigCubeSelfDestructDelay,
|
|
BuiltIn.Keys.BotImageChinkSelfDestruct, BuiltIn.Keys.BotImageChinkSelfDestructDelay
|
|
]);
|
|
var divideBy = settings[BuiltIn.Keys.BotImageRandomSliceDivideBy].ToType<int>();
|
|
var limit = 1;
|
|
var count = await images.CountAsync(ctx);
|
|
if (count > divideBy)
|
|
{
|
|
limit = count / divideBy;
|
|
}
|
|
|
|
// EF with SQLite can't sort on dates as it's just TEXT
|
|
var selection = (await images.ToListAsync(ctx)).OrderBy(i => i.LastSeen).Take(limit).ToList();
|
|
// MaxValue is never returned by Next so you don't need to -1 for indexing
|
|
var image = selection[new Random().Next(0, selection.Count)];
|
|
image.LastSeen = DateTimeOffset.UtcNow;
|
|
db.Images.Update(image);
|
|
await db.SaveChangesAsync(ctx);
|
|
TimeSpan? timeToDeletion = null;
|
|
if (key == "pigcube" && settings[BuiltIn.Keys.BotImagePigCubeSelfDestruct].ToBoolean())
|
|
{
|
|
timeToDeletion = TimeSpan.FromMilliseconds(image.Url == settings[BuiltIn.Keys.BotImageInvertedCubeUrl].Value
|
|
? settings[BuiltIn.Keys.BotImageInvertedPigCubeSelfDestructDelay].ToType<int>()
|
|
: new Random().Next(settings[BuiltIn.Keys.BotImagePigCubeSelfDestructMin].ToType<int>(),
|
|
settings[BuiltIn.Keys.BotImagePigCubeSelfDestructMax].ToType<int>()));
|
|
}
|
|
else if (key is "chink" or "sloppa" && settings[BuiltIn.Keys.BotImageChinkSelfDestruct].ToBoolean())
|
|
{
|
|
RateLimitService.AddEntry(user, this, message.MessageRawHtmlDecoded);
|
|
timeToDeletion = TimeSpan.FromMilliseconds(settings[BuiltIn.Keys.BotImageChinkSelfDestructDelay].ToType<int>());
|
|
}
|
|
await botInstance.SendChatMessageAsync($"[img]{image.Url}[/img]", true, autoDeleteAfter: timeToDeletion);
|
|
}
|
|
} |