2023-05-06 14:56:16 +00:00
|
|
|
using System.Globalization;
|
2022-11-30 18:03:00 +00:00
|
|
|
using System.Reflection;
|
2022-11-30 20:54:22 +00:00
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
2022-11-27 20:43:21 +00:00
|
|
|
using VpSharp.ClientExtensions;
|
|
|
|
using VpSharp.Commands.Attributes;
|
2022-12-05 01:43:40 +00:00
|
|
|
using VpSharp.Commands.Attributes.ExecutionChecks;
|
2022-11-30 18:52:49 +00:00
|
|
|
using VpSharp.Entities;
|
2022-11-27 20:43:21 +00:00
|
|
|
using VpSharp.EventData;
|
2022-12-10 13:40:47 +00:00
|
|
|
using VpSharp.Internal;
|
2022-11-27 20:43:21 +00:00
|
|
|
|
|
|
|
namespace VpSharp.Commands;
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Implements the Commands extension for <see cref="VirtualParadiseClient" />.
|
|
|
|
/// </summary>
|
|
|
|
public sealed class CommandsExtension : VirtualParadiseClientExtension
|
|
|
|
{
|
|
|
|
private const BindingFlags BindingFlags = System.Reflection.BindingFlags.Public |
|
|
|
|
System.Reflection.BindingFlags.NonPublic |
|
|
|
|
System.Reflection.BindingFlags.Instance;
|
|
|
|
|
|
|
|
private readonly Dictionary<string, Command> _commandMap = new(StringComparer.OrdinalIgnoreCase);
|
2022-11-30 18:53:19 +00:00
|
|
|
private readonly CommandsExtensionConfiguration _configuration;
|
2022-11-27 20:43:21 +00:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Initializes a new instance of the <see cref="CommandsExtension" /> class.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="client">The owning client.</param>
|
|
|
|
/// <param name="configuration">The configuration to use.</param>
|
2022-11-29 15:13:59 +00:00
|
|
|
/// <exception cref="ArgumentNullException">
|
|
|
|
/// <paramref name="client" /> or <paramref name="configuration" /> is <see langword="null" />.
|
|
|
|
/// </exception>
|
2022-11-27 20:43:21 +00:00
|
|
|
public CommandsExtension(VirtualParadiseClient client, CommandsExtensionConfiguration configuration)
|
|
|
|
: base(client)
|
|
|
|
{
|
2024-02-15 22:38:48 +00:00
|
|
|
if (client is null)
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(client));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (configuration is null)
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(configuration));
|
|
|
|
}
|
|
|
|
|
2022-11-30 20:57:39 +00:00
|
|
|
|
|
|
|
_configuration = configuration;
|
2022-11-30 20:54:22 +00:00
|
|
|
_configuration.Services ??= client.Services;
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Registers all commands from all modules in the specified assembly.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="assembly">The assembly whose command modules to register.</param>
|
|
|
|
/// <exception cref="ArgumentException">
|
2022-11-30 20:54:22 +00:00
|
|
|
/// A module in the specified assembly does not have a public constructor, or has more than one public constructor.
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
/// <exception cref="TypeInitializationException">A command module could not be instantiated.</exception>
|
|
|
|
/// <exception cref="InvalidOperationException">
|
|
|
|
/// <para>A command in the specified assembly does not have a <see cref="CommandContext" /> as its first parameter.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>A command in the specified assembly contains a duplicate name or alias for a command.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// A command in the specified assembly has <see cref="RemainderAttribute" /> on a parameter that is not the last in
|
|
|
|
/// the parameter list.
|
|
|
|
/// </para>
|
2022-11-30 20:54:22 +00:00
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// A module is expecting services but no <see cref="IServiceProvider" /> was registered.
|
|
|
|
/// </para>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
public void RegisterCommands(Assembly assembly)
|
|
|
|
{
|
2024-02-15 22:38:48 +00:00
|
|
|
if (assembly is null)
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(assembly));
|
|
|
|
}
|
2022-11-27 20:43:21 +00:00
|
|
|
|
|
|
|
foreach (Type type in assembly.GetTypes())
|
|
|
|
{
|
|
|
|
if (!type.IsAbstract && type.IsSubclassOf(typeof(CommandModule)))
|
|
|
|
{
|
|
|
|
RegisterCommands(type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
2022-11-30 18:46:02 +00:00
|
|
|
/// Registers the commands defined in the specified type.
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </summary>
|
|
|
|
/// <exception cref="ArgumentException">
|
2022-11-30 20:54:22 +00:00
|
|
|
/// <para><typeparamref name="T" /> refers to a type that does not inherit <see cref="CommandModule" />.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// <typeparamref name="T" /> does not have a public constructor, or has more than one public constructor.
|
|
|
|
/// </para>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
/// <exception cref="TypeInitializationException"><typeparamref name="T" /> could not be instantiated.</exception>
|
|
|
|
/// <exception cref="InvalidOperationException">
|
|
|
|
/// <para>A command in the specified module does not have a <see cref="CommandContext" /> as its first parameter.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>A command in the specified module contains a duplicate name or alias for a command.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// A command in the specified module has <see cref="RemainderAttribute" /> on a parameter that is not the last in the
|
|
|
|
/// parameter list.
|
|
|
|
/// </para>
|
2022-11-30 20:54:22 +00:00
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// The module is expecting services but no <see cref="IServiceProvider" /> was registered.
|
|
|
|
/// </para>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
public void RegisterCommands<T>() where T : CommandModule
|
|
|
|
{
|
|
|
|
RegisterCommands(typeof(T));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
2022-11-30 18:46:02 +00:00
|
|
|
/// Registers the commands defined in the specified type.
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </summary>
|
2022-11-30 18:46:02 +00:00
|
|
|
/// <param name="moduleType">The type whose command methods to register.</param>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// <exception cref="ArgumentNullException"><paramref name="moduleType" /> is <see langword="null" />.</exception>
|
|
|
|
/// <exception cref="ArgumentException">
|
|
|
|
/// <para><paramref name="moduleType" /> refers to an <c>abstract</c> type.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para><paramref name="moduleType" /> refers to a type that does not inherit <see cref="CommandModule" />.</para>
|
2022-11-30 20:54:22 +00:00
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// <paramref name="moduleType" /> does not have a public constructor, or has more than one public constructor.
|
|
|
|
/// </para>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
/// <exception cref="TypeInitializationException"><paramref name="moduleType" /> could not be instantiated.</exception>
|
|
|
|
/// <exception cref="InvalidOperationException">
|
|
|
|
/// <para>A command in the specified module does not have a <see cref="CommandContext" /> as its first parameter.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>A command in the specified module contains a duplicate name or alias for a command.</para>
|
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// A command in the specified module has <see cref="RemainderAttribute" /> on a parameter that is not the last in the
|
|
|
|
/// parameter list.
|
|
|
|
/// </para>
|
2022-11-30 20:54:22 +00:00
|
|
|
/// -or-
|
|
|
|
/// <para>
|
|
|
|
/// The module is expecting services but no <see cref="IServiceProvider" /> was registered.
|
|
|
|
/// </para>
|
2022-11-27 20:43:21 +00:00
|
|
|
/// </exception>
|
|
|
|
public void RegisterCommands(Type moduleType)
|
|
|
|
{
|
2024-02-15 22:38:48 +00:00
|
|
|
if (moduleType is null)
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(moduleType));
|
|
|
|
}
|
2022-11-27 20:43:21 +00:00
|
|
|
|
|
|
|
if (moduleType.IsAbstract)
|
|
|
|
{
|
|
|
|
throw new ArgumentException("Module type cannot be abstract");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!moduleType.IsSubclassOf(typeof(CommandModule)))
|
|
|
|
{
|
|
|
|
throw new ArgumentException($"Module type is not a subclass of {typeof(CommandModule)}");
|
|
|
|
}
|
|
|
|
|
2022-11-30 20:54:22 +00:00
|
|
|
IServiceProvider? serviceProvider = _configuration.Services;
|
2022-12-10 13:40:47 +00:00
|
|
|
object instance = DependencyInjectionUtility.CreateInstance(moduleType, serviceProvider);
|
|
|
|
if (instance is not CommandModule module)
|
2022-11-27 20:43:21 +00:00
|
|
|
{
|
2022-11-30 18:47:57 +00:00
|
|
|
throw new TypeInitializationException(moduleType.FullName, null);
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach (MethodInfo method in moduleType.GetMethods(BindingFlags))
|
|
|
|
{
|
|
|
|
RegisterCommandMethod(module, method);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <inheritdoc />
|
2023-05-08 14:44:06 +00:00
|
|
|
protected internal override Task OnMessageReceived(VirtualParadiseMessage message)
|
2022-11-27 20:43:21 +00:00
|
|
|
{
|
2024-02-15 22:38:48 +00:00
|
|
|
if (message is null)
|
|
|
|
{
|
|
|
|
throw new ArgumentNullException(nameof(message));
|
|
|
|
}
|
2022-11-27 20:43:21 +00:00
|
|
|
|
|
|
|
if (message.Type != MessageType.ChatMessage)
|
|
|
|
{
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach (ReadOnlySpan<char> prefix in _configuration.Prefixes)
|
|
|
|
{
|
|
|
|
ReadOnlySpan<char> content = message.Content;
|
|
|
|
if (!content.StartsWith(prefix))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ReadOnlySpan<char> commandName, rawArguments;
|
|
|
|
int spaceIndex = content.IndexOf(' ');
|
|
|
|
|
|
|
|
if (spaceIndex == -1)
|
|
|
|
{
|
|
|
|
commandName = content[prefix.Length..];
|
|
|
|
rawArguments = ReadOnlySpan<char>.Empty;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
commandName = content[prefix.Length..spaceIndex];
|
|
|
|
rawArguments = content[(spaceIndex + 1)..];
|
|
|
|
}
|
|
|
|
|
|
|
|
var commandNameString = commandName.ToString();
|
|
|
|
if (!_commandMap.TryGetValue(commandNameString, out Command? command))
|
|
|
|
{
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var context = new CommandContext(Client, message.Author, command.Name, commandNameString, rawArguments.ToString());
|
2023-05-06 14:56:16 +00:00
|
|
|
MethodInfo commandMethod = command.Method;
|
2022-11-27 20:43:21 +00:00
|
|
|
|
2023-05-06 14:56:16 +00:00
|
|
|
foreach (var attribute in commandMethod.GetCustomAttributes<PreExecutionCheckAttribute>())
|
2022-12-05 01:43:40 +00:00
|
|
|
{
|
|
|
|
if (!attribute.PerformAsync(context).ConfigureAwait(false).GetAwaiter().GetResult())
|
|
|
|
{
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-12-05 01:43:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object?[] arguments = {context};
|
2023-05-08 14:59:39 +00:00
|
|
|
arguments = ParseArguments(rawArguments, arguments, command);
|
2022-11-27 20:43:21 +00:00
|
|
|
|
2023-05-06 14:56:16 +00:00
|
|
|
ParameterInfo[] parameters = commandMethod.GetParameters();
|
2023-05-06 15:00:41 +00:00
|
|
|
if (parameters.Length != arguments.Length || parameters[arguments.Length..].Any(p => !p.IsOptional))
|
2022-12-08 18:31:06 +00:00
|
|
|
{
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-12-08 18:31:06 +00:00
|
|
|
}
|
|
|
|
|
2023-05-06 14:56:16 +00:00
|
|
|
for (var index = 0; index < arguments.Length; index++)
|
|
|
|
{
|
|
|
|
Type parameterType = parameters[index].ParameterType;
|
|
|
|
|
|
|
|
if (parameterType.IsEnum)
|
|
|
|
{
|
|
|
|
var argumentString = arguments[index]?.ToString();
|
|
|
|
|
|
|
|
Type enumUnderlyingType = parameterType.GetEnumUnderlyingType();
|
|
|
|
if (enumUnderlyingType == typeof(int) && int.TryParse(argumentString, out int enumInt))
|
|
|
|
{
|
|
|
|
arguments[index] = Enum.ToObject(parameterType, enumInt);
|
|
|
|
}
|
|
|
|
else if (enumUnderlyingType == typeof(long) && long.TryParse(argumentString, out long enumLong))
|
|
|
|
{
|
|
|
|
arguments[index] = Enum.ToObject(parameterType, enumLong);
|
|
|
|
}
|
|
|
|
else if (Enum.TryParse(parameterType, argumentString, true, out object? enumValue))
|
|
|
|
{
|
|
|
|
arguments[index] = enumValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
arguments[index] = Convert.ChangeType(arguments[index], parameterType, CultureInfo.InvariantCulture);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-06 15:00:41 +00:00
|
|
|
for (int index = arguments.Length; index < parameters.Length; index++)
|
|
|
|
{
|
|
|
|
arguments[index] = parameters[index].DefaultValue;
|
|
|
|
}
|
|
|
|
|
2023-05-06 14:56:16 +00:00
|
|
|
object? returnValue = commandMethod.Invoke(command.Module, arguments);
|
2022-11-27 20:43:21 +00:00
|
|
|
if (returnValue is Task task)
|
|
|
|
{
|
|
|
|
return task;
|
|
|
|
}
|
|
|
|
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
2023-05-08 14:44:06 +00:00
|
|
|
return base.OnMessageReceived(message);
|
2022-11-27 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
2023-05-08 14:59:39 +00:00
|
|
|
private static object?[] ParseArguments(ReadOnlySpan<char> rawArguments, object?[] arguments, Command command)
|
|
|
|
{
|
|
|
|
if (rawArguments.Length <= 0)
|
|
|
|
{
|
|
|
|
return arguments;
|
|
|
|
}
|
|
|
|
|
|
|
|
int spaceIndex = rawArguments.IndexOf(' ');
|
|
|
|
if (spaceIndex == -1)
|
|
|
|
{
|
|
|
|
Array.Resize(ref arguments, 2);
|
|
|
|
arguments[1] = rawArguments.ToString();
|
|
|
|
return arguments;
|
|
|
|
}
|
|
|
|
|
|
|
|
var appendLast = true;
|
|
|
|
for (var argumentIndex = 1; spaceIndex > -1; argumentIndex++)
|
|
|
|
{
|
|
|
|
Array.Resize(ref arguments, argumentIndex + 1);
|
|
|
|
|
|
|
|
if (argumentIndex == command.Parameters.Length)
|
|
|
|
{
|
|
|
|
if (command.Parameters[argumentIndex - 1].GetCustomAttribute<RemainderAttribute>() is not null)
|
|
|
|
{
|
|
|
|
appendLast = false;
|
|
|
|
arguments[argumentIndex] = rawArguments.ToString();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
arguments[argumentIndex] = rawArguments[..spaceIndex].ToString();
|
|
|
|
rawArguments = rawArguments[(spaceIndex + 1)..];
|
|
|
|
spaceIndex = rawArguments.IndexOf(' ');
|
|
|
|
argumentIndex++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (appendLast)
|
|
|
|
{
|
|
|
|
Array.Resize(ref arguments, arguments.Length + 1);
|
|
|
|
arguments[^1] = rawArguments.ToString();
|
|
|
|
}
|
|
|
|
|
|
|
|
return arguments;
|
|
|
|
}
|
|
|
|
|
2022-11-27 20:43:21 +00:00
|
|
|
private void RegisterCommandMethod(CommandModule module, MethodInfo methodInfo)
|
|
|
|
{
|
|
|
|
var commandAttribute = methodInfo.GetCustomAttribute<CommandAttribute>();
|
|
|
|
if (commandAttribute is null)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var aliasesAttribute = methodInfo.GetCustomAttribute<AliasesAttribute>();
|
|
|
|
ParameterInfo[] parameters = methodInfo.GetParameters();
|
|
|
|
if (parameters.Length == 0 || parameters[0].ParameterType != typeof(CommandContext))
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException($"Command method must have a {typeof(CommandContext)} parameter at index 0.");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var index = 0; index < parameters.Length - 1; index++)
|
|
|
|
{
|
|
|
|
if (parameters[index].GetCustomAttribute<RemainderAttribute>() is not null)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException($"{typeof(RemainderAttribute)} can only be placed on the last parameter.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var types = new Type[parameters.Length - 1];
|
|
|
|
for (var index = 1; index < parameters.Length; index++)
|
|
|
|
{
|
|
|
|
types[index - 1] = parameters[index].ParameterType;
|
|
|
|
}
|
|
|
|
|
|
|
|
var command = new Command(
|
|
|
|
commandAttribute.Name,
|
2022-11-30 18:03:00 +00:00
|
|
|
aliasesAttribute?.Aliases ?? ArraySegment<string>.Empty,
|
2022-11-27 20:43:21 +00:00
|
|
|
methodInfo,
|
|
|
|
module
|
|
|
|
);
|
|
|
|
|
|
|
|
if (_commandMap.ContainsKey(command.Name))
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException($"Duplicate command name registered ({command.Name})");
|
|
|
|
}
|
|
|
|
|
|
|
|
_commandMap[command.Name] = command;
|
|
|
|
|
|
|
|
foreach (string alias in command.Aliases)
|
|
|
|
{
|
|
|
|
if (_commandMap.ContainsKey(alias))
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException($"Duplicate command name registered ({alias})");
|
|
|
|
}
|
|
|
|
|
|
|
|
_commandMap[alias] = command;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|