mirror of
https://github.com/TeamOctolings/Octobot.git
synced 2025-04-20 00:43:36 +03:00
Add /kick command
This commit is contained in:
parent
2ee30ddfb8
commit
6bee9ac0fc
7 changed files with 1475 additions and 835 deletions
|
@ -76,7 +76,8 @@ public class Boyfriend {
|
||||||
.AddSingleton<UtilityService>()
|
.AddSingleton<UtilityService>()
|
||||||
.AddHostedService<GuildUpdateService>()
|
.AddHostedService<GuildUpdateService>()
|
||||||
.AddCommandTree()
|
.AddCommandTree()
|
||||||
.WithCommandGroup<BanCommandGroup>();
|
.WithCommandGroup<BanCommandGroup>()
|
||||||
|
.WithCommandGroup<KickCommandGroup>();
|
||||||
var responderTypes = typeof(Boyfriend).Assembly
|
var responderTypes = typeof(Boyfriend).Assembly
|
||||||
.GetExportedTypes()
|
.GetExportedTypes()
|
||||||
.Where(t => t.IsResponder());
|
.Where(t => t.IsResponder());
|
||||||
|
|
|
@ -56,14 +56,14 @@ public class BanCommandGroup : CommandGroup {
|
||||||
/// A feedback sending result which may or may not have succeeded. A successful result does not mean that the user
|
/// A feedback sending result which may or may not have succeeded. A successful result does not mean that the user
|
||||||
/// was banned and vice-versa.
|
/// was banned and vice-versa.
|
||||||
/// </returns>
|
/// </returns>
|
||||||
/// <seealso cref="UnBanUserAsync" />
|
/// <seealso cref="UnbanUserAsync" />
|
||||||
[Command("ban", "бан")]
|
[Command("ban", "бан")]
|
||||||
[RequireContext(ChannelContext.Guild)]
|
[RequireContext(ChannelContext.Guild)]
|
||||||
[RequireDiscordPermission(DiscordPermission.BanMembers)]
|
[RequireDiscordPermission(DiscordPermission.BanMembers)]
|
||||||
[RequireBotDiscordPermissions(DiscordPermission.BanMembers)]
|
[RequireBotDiscordPermissions(DiscordPermission.BanMembers)]
|
||||||
[Description("банит пидора")]
|
[Description("банит пидора")]
|
||||||
public async Task<Result> BanUserAsync(
|
public async Task<Result> BanUserAsync(
|
||||||
[Description("Юзер, кого банить")] IUser target, string reason, TimeSpan? duration = null) {
|
[Description("юзер кого банить")] IUser target, [Description("причина зачем банить")] string reason, TimeSpan? duration = null) {
|
||||||
// Data checks
|
// Data checks
|
||||||
if (!_context.TryGetGuildID(out var guildId))
|
if (!_context.TryGetGuildID(out var guildId))
|
||||||
return Result.FromError(new ArgumentNullError(nameof(guildId)));
|
return Result.FromError(new ArgumentNullError(nameof(guildId)));
|
||||||
|
@ -122,7 +122,7 @@ public class BanCommandGroup : CommandGroup {
|
||||||
|| (cfg.PrivateFeedbackChannel is not 0 && cfg.PrivateFeedbackChannel != channelId.Value)) {
|
|| (cfg.PrivateFeedbackChannel is not 0 && cfg.PrivateFeedbackChannel != channelId.Value)) {
|
||||||
var logEmbed = new EmbedBuilder().WithSmallTitle(
|
var logEmbed = new EmbedBuilder().WithSmallTitle(
|
||||||
string.Format(Messages.UserBanned, target.GetTag()), target)
|
string.Format(Messages.UserBanned, target.GetTag()), target)
|
||||||
.WithDescription(string.Format(Messages.DescriptionUserBanned, reason))
|
.WithDescription(string.Format(Messages.DescriptionUserPunished, reason))
|
||||||
.WithActionFooter(user)
|
.WithActionFooter(user)
|
||||||
.WithCurrentTimestamp()
|
.WithCurrentTimestamp()
|
||||||
.WithColour(ColorsList.Red)
|
.WithColour(ColorsList.Red)
|
||||||
|
@ -169,7 +169,7 @@ public class BanCommandGroup : CommandGroup {
|
||||||
[RequireDiscordPermission(DiscordPermission.BanMembers)]
|
[RequireDiscordPermission(DiscordPermission.BanMembers)]
|
||||||
[RequireBotDiscordPermissions(DiscordPermission.BanMembers)]
|
[RequireBotDiscordPermissions(DiscordPermission.BanMembers)]
|
||||||
[Description("разбанит пидора")]
|
[Description("разбанит пидора")]
|
||||||
public async Task<Result> UnBanUserAsync([Description("Юзер, кого разбанить")] IUser target, string reason) {
|
public async Task<Result> UnbanUserAsync([Description("Юзер, кого разбанить")] IUser target, string reason) {
|
||||||
// Data checks
|
// Data checks
|
||||||
if (!_context.TryGetGuildID(out var guildId))
|
if (!_context.TryGetGuildID(out var guildId))
|
||||||
return Result.FromError(new ArgumentNullError(nameof(guildId)));
|
return Result.FromError(new ArgumentNullError(nameof(guildId)));
|
||||||
|
|
145
Commands/KickCommandGroup.cs
Normal file
145
Commands/KickCommandGroup.cs
Normal file
|
@ -0,0 +1,145 @@
|
||||||
|
using System.ComponentModel;
|
||||||
|
using Boyfriend.Services;
|
||||||
|
using Boyfriend.Services.Data;
|
||||||
|
using Remora.Commands.Attributes;
|
||||||
|
using Remora.Commands.Groups;
|
||||||
|
using Remora.Discord.API.Abstractions.Objects;
|
||||||
|
using Remora.Discord.API.Abstractions.Rest;
|
||||||
|
using Remora.Discord.API.Objects;
|
||||||
|
using Remora.Discord.Commands.Conditions;
|
||||||
|
using Remora.Discord.Commands.Contexts;
|
||||||
|
using Remora.Discord.Commands.Extensions;
|
||||||
|
using Remora.Discord.Commands.Feedback.Services;
|
||||||
|
using Remora.Discord.Extensions.Embeds;
|
||||||
|
using Remora.Results;
|
||||||
|
|
||||||
|
// ReSharper disable ClassNeverInstantiated.Global
|
||||||
|
// ReSharper disable UnusedMember.Global
|
||||||
|
|
||||||
|
namespace Boyfriend.Commands;
|
||||||
|
|
||||||
|
public class KickCommandGroup : CommandGroup {
|
||||||
|
private readonly IDiscordRestChannelAPI _channelApi;
|
||||||
|
private readonly ICommandContext _context;
|
||||||
|
private readonly GuildDataService _dataService;
|
||||||
|
private readonly FeedbackService _feedbackService;
|
||||||
|
private readonly IDiscordRestGuildAPI _guildApi;
|
||||||
|
private readonly IDiscordRestUserAPI _userApi;
|
||||||
|
private readonly UtilityService _utility;
|
||||||
|
|
||||||
|
public KickCommandGroup(
|
||||||
|
ICommandContext context, IDiscordRestChannelAPI channelApi, GuildDataService dataService,
|
||||||
|
FeedbackService feedbackService, IDiscordRestGuildAPI guildApi, IDiscordRestUserAPI userApi,
|
||||||
|
UtilityService utility) {
|
||||||
|
_context = context;
|
||||||
|
_channelApi = channelApi;
|
||||||
|
_dataService = dataService;
|
||||||
|
_feedbackService = feedbackService;
|
||||||
|
_guildApi = guildApi;
|
||||||
|
_userApi = userApi;
|
||||||
|
_utility = utility;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// A slash command that kicks a Discord user with the specified reason.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="target">The user to kick.</param>
|
||||||
|
/// <param name="reason">
|
||||||
|
/// The reason for this kick. Must be encoded with <see cref="Extensions.EncodeHeader" /> when passed to
|
||||||
|
/// <see cref="IDiscordRestGuildAPI.RemoveGuildMemberAsync" />.
|
||||||
|
/// </param>
|
||||||
|
/// <returns>
|
||||||
|
/// A feedback sending result which may or may not have succeeded. A successful result does not mean that the user
|
||||||
|
/// was kicked and vice-versa.
|
||||||
|
/// </returns>
|
||||||
|
[Command("kick", "кик")]
|
||||||
|
[RequireContext(ChannelContext.Guild)]
|
||||||
|
[RequireDiscordPermission(DiscordPermission.KickMembers)]
|
||||||
|
[RequireBotDiscordPermissions(DiscordPermission.KickMembers)]
|
||||||
|
[Description("кикает твоего друга <3")]
|
||||||
|
public async Task<Result> KickUserAsync(
|
||||||
|
[Description("друг которого кикать")] IUser target, [Description("причина зачем кикать")] string reason) {
|
||||||
|
// Data checks
|
||||||
|
if (!_context.TryGetGuildID(out var guildId))
|
||||||
|
return Result.FromError(new ArgumentNullError(nameof(guildId)));
|
||||||
|
if (!_context.TryGetUserID(out var userId))
|
||||||
|
return Result.FromError(new ArgumentNullError(nameof(userId)));
|
||||||
|
if (!_context.TryGetChannelID(out var channelId))
|
||||||
|
return Result.FromError(new ArgumentNullError(nameof(channelId)));
|
||||||
|
|
||||||
|
// The current user's avatar is used when sending error messages
|
||||||
|
var currentUserResult = await _userApi.GetCurrentUserAsync(CancellationToken);
|
||||||
|
if (!currentUserResult.IsDefined(out var currentUser))
|
||||||
|
return Result.FromError(currentUserResult);
|
||||||
|
|
||||||
|
var data = await _dataService.GetData(guildId.Value, CancellationToken);
|
||||||
|
var cfg = data.Configuration;
|
||||||
|
Messages.Culture = data.Culture;
|
||||||
|
|
||||||
|
var existingKickResult = await _guildApi.GetGuildMemberAsync(guildId.Value, target.ID);
|
||||||
|
if (!existingKickResult.IsSuccess) {
|
||||||
|
var embed = new EmbedBuilder().WithSmallTitle(Messages.UserNotFoundShort, currentUser)
|
||||||
|
.WithColour(ColorsList.Red).Build();
|
||||||
|
|
||||||
|
if (!embed.IsDefined(out var alreadyBuilt))
|
||||||
|
return Result.FromError(embed);
|
||||||
|
|
||||||
|
return (Result)await _feedbackService.SendContextualEmbedAsync(alreadyBuilt, ct: CancellationToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
var interactionResult
|
||||||
|
= await _utility.CheckInteractionsAsync(guildId.Value, userId.Value, target.ID, "Kick", CancellationToken);
|
||||||
|
if (!interactionResult.IsSuccess)
|
||||||
|
return Result.FromError(interactionResult);
|
||||||
|
|
||||||
|
Result<Embed> responseEmbed;
|
||||||
|
if (interactionResult.Entity is not null) {
|
||||||
|
responseEmbed = new EmbedBuilder().WithSmallTitle(interactionResult.Entity, currentUser)
|
||||||
|
.WithColour(ColorsList.Red).Build();
|
||||||
|
} else {
|
||||||
|
var userResult = await _userApi.GetUserAsync(userId.Value, CancellationToken);
|
||||||
|
if (!userResult.IsDefined(out var user))
|
||||||
|
return Result.FromError(userResult);
|
||||||
|
|
||||||
|
var kickResult = await _guildApi.RemoveGuildMemberAsync(
|
||||||
|
guildId.Value, target.ID, reason: $"({user.GetTag()}) {reason.EncodeHeader()}",
|
||||||
|
ct: CancellationToken);
|
||||||
|
if (!kickResult.IsSuccess)
|
||||||
|
return Result.FromError(kickResult.Error);
|
||||||
|
|
||||||
|
responseEmbed = new EmbedBuilder().WithSmallTitle(
|
||||||
|
string.Format(Messages.UserKicked, target.GetTag()), target)
|
||||||
|
.WithColour(ColorsList.Green).Build();
|
||||||
|
|
||||||
|
if ((cfg.PublicFeedbackChannel is not 0 && cfg.PublicFeedbackChannel != channelId.Value)
|
||||||
|
|| (cfg.PrivateFeedbackChannel is not 0 && cfg.PrivateFeedbackChannel != channelId.Value)) {
|
||||||
|
var logEmbed = new EmbedBuilder().WithSmallTitle(
|
||||||
|
string.Format(Messages.UserKicked, target.GetTag()), target)
|
||||||
|
.WithDescription(string.Format(Messages.DescriptionUserPunished, reason))
|
||||||
|
.WithActionFooter(user)
|
||||||
|
.WithCurrentTimestamp()
|
||||||
|
.WithColour(ColorsList.Red)
|
||||||
|
.Build();
|
||||||
|
|
||||||
|
if (!logEmbed.IsDefined(out var logBuilt))
|
||||||
|
return Result.FromError(logEmbed);
|
||||||
|
|
||||||
|
var builtArray = new[] { logBuilt };
|
||||||
|
// Not awaiting to reduce response time
|
||||||
|
if (cfg.PrivateFeedbackChannel != channelId.Value)
|
||||||
|
_ = _channelApi.CreateMessageAsync(
|
||||||
|
cfg.PrivateFeedbackChannel.ToDiscordSnowflake(), embeds: builtArray,
|
||||||
|
ct: CancellationToken);
|
||||||
|
if (cfg.PublicFeedbackChannel != channelId.Value)
|
||||||
|
_ = _channelApi.CreateMessageAsync(
|
||||||
|
cfg.PublicFeedbackChannel.ToDiscordSnowflake(), embeds: builtArray,
|
||||||
|
ct: CancellationToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!responseEmbed.IsDefined(out var built))
|
||||||
|
return Result.FromError(responseEmbed);
|
||||||
|
|
||||||
|
return (Result)await _feedbackService.SendContextualEmbedAsync(built, ct: CancellationToken);
|
||||||
|
}
|
||||||
|
}
|
1733
Messages.Designer.cs
generated
1733
Messages.Designer.cs
generated
File diff suppressed because it is too large
Load diff
|
@ -486,13 +486,28 @@
|
||||||
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
||||||
<value>The event is happening at {0} until {1}</value>
|
<value>The event is happening at {0} until {1}</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DescriptionUserBanned" xml:space="preserve">
|
|
||||||
<value>Reason: {0}</value>
|
|
||||||
</data>
|
|
||||||
<data name="UserAlreadyBanned" xml:space="preserve">
|
<data name="UserAlreadyBanned" xml:space="preserve">
|
||||||
<value>This user is already banned!</value>
|
<value>This user is already banned!</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="UserUnbanned" xml:space="preserve">
|
<data name="UserUnbanned" xml:space="preserve">
|
||||||
<value>{0} was unbanned</value>
|
<value>{0} was unbanned</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="UserMuted" xml:space="preserve">
|
||||||
|
<value>{0} was muted</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserUnmuted" xml:space="preserve">
|
||||||
|
<value>{0} was unmuted</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotMuted" xml:space="preserve">
|
||||||
|
<value>This member is not muted!</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotFoundShort" xml:space="preserve">
|
||||||
|
<value>I could not find this user!</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserKicked" xml:space="preserve">
|
||||||
|
<value>{0} was kicked</value>
|
||||||
|
</data>
|
||||||
|
<data name="DescriptionUserPunished" xml:space="preserve">
|
||||||
|
<value>Reason: {0}</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
|
|
|
@ -486,13 +486,28 @@
|
||||||
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
||||||
<value>Событие происходит в {0} до {1}</value>
|
<value>Событие происходит в {0} до {1}</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DescriptionUserBanned" xml:space="preserve">
|
|
||||||
<value>Причина: {0}</value>
|
|
||||||
</data>
|
|
||||||
<data name="UserAlreadyBanned" xml:space="preserve">
|
<data name="UserAlreadyBanned" xml:space="preserve">
|
||||||
<value>Этот пользователь уже забанен!</value>
|
<value>Этот пользователь уже забанен!</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="UserUnbanned" xml:space="preserve">
|
<data name="UserUnbanned" xml:space="preserve">
|
||||||
<value>{0} был(-а) разбанен(-а)</value>
|
<value>{0} был(-а) разбанен(-а)</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="UserMuted" xml:space="preserve">
|
||||||
|
<value>{0} был(-а) заглушен(-а)</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotMuted" xml:space="preserve">
|
||||||
|
<value>Этот участник не заглушен!</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserUnmuted" xml:space="preserve">
|
||||||
|
<value>{0} был(-а) разглушен(-а)</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotFoundShort" xml:space="preserve">
|
||||||
|
<value>Я не смог найти этого пользователя!</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserKicked" xml:space="preserve">
|
||||||
|
<value>{0} был(-а) выгнан(-а)</value>
|
||||||
|
</data>
|
||||||
|
<data name="DescriptionUserPunished" xml:space="preserve">
|
||||||
|
<value>Причина: {0}</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
|
|
|
@ -486,13 +486,28 @@
|
||||||
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
<data name="DescriptionExternalEventStarted" xml:space="preserve">
|
||||||
<value>движуха происходит в {0} до {1}</value>
|
<value>движуха происходит в {0} до {1}</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="DescriptionUserBanned" xml:space="preserve">
|
|
||||||
<value>причина: {0}</value>
|
|
||||||
</data>
|
|
||||||
<data name="UserAlreadyBanned" xml:space="preserve">
|
<data name="UserAlreadyBanned" xml:space="preserve">
|
||||||
<value>этот шизоид уже лежит в бане</value>
|
<value>этот шизоид уже лежит в бане</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="UserUnbanned" xml:space="preserve">
|
<data name="UserUnbanned" xml:space="preserve">
|
||||||
<value>{0} раззабанен</value>
|
<value>{0} раззабанен</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="UserMuted" xml:space="preserve">
|
||||||
|
<value>{0} в муте</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserUnmuted" xml:space="preserve">
|
||||||
|
<value>{0} в размуте</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotMuted" xml:space="preserve">
|
||||||
|
<value>этого шизоида никто не мутил.</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserNotFoundShort" xml:space="preserve">
|
||||||
|
<value>у нас такого шизоида нету...</value>
|
||||||
|
</data>
|
||||||
|
<data name="UserKicked" xml:space="preserve">
|
||||||
|
<value>{0} вышел с посторонней помощью</value>
|
||||||
|
</data>
|
||||||
|
<data name="DescriptionUserPunished" xml:space="preserve">
|
||||||
|
<value>причина: {0}</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
|
|
Loading…
Add table
Reference in a new issue