mirai/backend/mirai-console/test/command/TestCommand.kt

502 lines
17 KiB
Kotlin
Raw Normal View History

/*
2021-07-02 00:56:18 +08:00
* Copyright 2019-2021 Mamoe Technologies and contributors.
*
2020-08-23 17:46:51 +08:00
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
2021-07-02 00:56:18 +08:00
* Use of this source code is governed by the GNU AGPLv3 license that can be found through the following link.
*
2021-07-02 00:56:18 +08:00
* https://github.com/mamoe/mirai/blob/dev/LICENSE
*/
2021-04-08 12:43:19 +08:00
@file:Suppress("unused", "UNUSED_PARAMETER")
package net.mamoe.mirai.console.command
2020-08-01 23:13:49 +08:00
import kotlinx.coroutines.cancel
import kotlinx.coroutines.runBlocking
import net.mamoe.mirai.console.MiraiConsole
import net.mamoe.mirai.console.Testing
2020-08-01 23:13:49 +08:00
import net.mamoe.mirai.console.Testing.withTesting
2020-10-24 21:19:50 +08:00
import net.mamoe.mirai.console.command.CommandManager.INSTANCE.getRegisteredCommands
2020-08-16 23:21:11 +08:00
import net.mamoe.mirai.console.command.CommandManager.INSTANCE.register
2020-10-24 21:19:50 +08:00
import net.mamoe.mirai.console.command.CommandManager.INSTANCE.registerCommand
2020-08-16 23:21:11 +08:00
import net.mamoe.mirai.console.command.CommandManager.INSTANCE.unregisterAllCommands
2020-10-24 21:19:50 +08:00
import net.mamoe.mirai.console.command.CommandManager.INSTANCE.unregisterCommand
2020-10-09 12:35:16 +08:00
import net.mamoe.mirai.console.command.descriptor.CommandValueArgumentParser
2020-10-20 13:27:13 +08:00
import net.mamoe.mirai.console.command.descriptor.ExperimentalCommandDescriptors
import net.mamoe.mirai.console.command.descriptor.buildCommandArgumentContext
2020-08-01 23:13:49 +08:00
import net.mamoe.mirai.console.initTestEnvironment
import net.mamoe.mirai.console.internal.command.CommandManagerImpl
2020-08-16 23:21:11 +08:00
import net.mamoe.mirai.console.internal.command.flattenCommandComponents
2020-09-04 10:57:32 +08:00
import net.mamoe.mirai.message.data.*
2020-08-01 23:13:49 +08:00
import org.junit.jupiter.api.AfterAll
import org.junit.jupiter.api.BeforeAll
import org.junit.jupiter.api.Test
2020-10-24 13:14:25 +08:00
import kotlin.test.*
object TestCompositeCommand : CompositeCommand(
2020-11-29 18:07:02 +08:00
owner,
"testComposite", "tsC"
) {
@SubCommand
2020-10-24 11:35:10 +08:00
fun mute(seconds: Int = 60) {
Testing.ok(seconds)
}
@SubCommand
fun mute(target: Long, seconds: Int) {
Testing.ok(seconds)
}
}
2020-11-10 08:55:54 +08:00
object TestRawCommand : RawCommand(
2020-11-29 18:07:02 +08:00
owner,
2020-11-10 08:55:54 +08:00
"testRaw"
) {
override suspend fun CommandSender.onCommand(args: MessageChain) {
Testing.ok(args)
}
}
object TestSimpleCommand : RawCommand(owner, "testSimple", "tsS") {
2020-09-04 10:57:32 +08:00
override suspend fun CommandSender.onCommand(args: MessageChain) {
Testing.ok(args)
}
}
2021-01-24 02:34:13 +08:00
@Suppress("EnumEntryName")
object TestEnumArgCommand : CompositeCommand(owner, "testenum") {
enum class TestEnum {
V1, V2, V3
}
2021-01-24 02:34:13 +08:00
enum class TestCase {
A, a
}
2021-01-24 02:34:13 +08:00
enum class TestCamelCase {
A, B, A_B
}
@SubCommand("tcc")
fun CommandSender.testCamelCase(enum: TestCamelCase) {
Testing.ok(enum)
}
@SubCommand("tc")
fun CommandSender.testCase(enum: TestCase) {
Testing.ok(enum)
}
@SubCommand
fun CommandSender.e1(enum: TestEnum) {
Testing.ok(enum)
}
}
internal val sender by lazy { ConsoleCommandSender }
2020-11-29 18:07:02 +08:00
internal object TestUnitCommandOwner : CommandOwner by ConsoleCommandOwner
2021-01-24 02:34:13 +08:00
2020-11-29 18:07:02 +08:00
internal val owner by lazy { TestUnitCommandOwner }
2020-08-01 23:13:49 +08:00
2020-10-24 11:35:10 +08:00
@OptIn(ExperimentalCommandDescriptors::class)
internal class TestCommand {
companion object {
@JvmStatic
@BeforeAll
fun init() {
initTestEnvironment()
}
2020-08-01 23:05:03 +08:00
@AfterAll
@JvmStatic
fun destroy() {
2020-08-01 23:13:49 +08:00
MiraiConsole.cancel()
2020-08-01 23:05:03 +08:00
}
}
@Test
fun testRegister() {
try {
2020-10-24 21:19:50 +08:00
unregisterAllCommands(ConsoleCommandOwner) // builtins
2020-11-29 18:07:02 +08:00
unregisterAllCommands(owner) // testing unit
2020-10-24 21:19:50 +08:00
unregisterCommand(TestSimpleCommand)
2020-08-01 23:05:03 +08:00
assertTrue(TestCompositeCommand.register())
assertFalse(TestCompositeCommand.register())
2020-11-29 18:07:02 +08:00
assertEquals(1, getRegisteredCommands(owner).size)
2020-09-18 23:26:16 +08:00
assertEquals(1, CommandManagerImpl._registeredCommands.size)
2020-10-23 13:40:16 +08:00
assertEquals(2,
CommandManagerImpl.requiredPrefixCommandMap.size,
CommandManagerImpl.requiredPrefixCommandMap.entries.joinToString { it.toString() })
} finally {
2020-10-24 21:19:50 +08:00
unregisterCommand(TestCompositeCommand)
}
}
@Test
fun testSimpleExecute() = runBlocking {
2020-10-20 13:27:13 +08:00
TestSimpleCommand.withRegistration {
assertEquals("test", withTesting<MessageChain> {
assertSuccess(TestSimpleCommand.execute(sender, "test"))
}.contentToString())
}
}
2020-11-10 08:55:54 +08:00
@Test
fun `test raw command`() = runBlocking {
TestRawCommand.withRegistration {
val result = withTesting<MessageChain> {
assertSuccess(TestRawCommand.execute(sender, PlainText("a1"), PlainText("a2"), PlainText("a3")))
}
assertEquals(3, result.size)
assertEquals("a1, a2, a3", result.joinToString())
}
}
@Test
fun `test flattenCommandArgs`() {
val result = arrayOf("test", image).flattenCommandComponents().toTypedArray()
2020-09-04 10:57:32 +08:00
assertEquals("test", result[0].content)
assertSame(image, result[1])
assertEquals(2, result.size)
}
2021-01-24 02:34:13 +08:00
@Test
fun `test enum argument`() = runBlocking {
TestEnumArgCommand.withRegistration {
assertEquals(TestEnumArgCommand.TestEnum.V1, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("V1")))
})
assertEquals(TestEnumArgCommand.TestEnum.V2, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("V2")))
})
assertEquals(TestEnumArgCommand.TestEnum.V3, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("V3")))
})
withTesting<Unit> {
assertFailure(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("ENUM_NOT_FOUND")))
Testing.ok(Unit)
}
assertEquals(TestEnumArgCommand.TestEnum.V1, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("v1")))
})
assertEquals(TestEnumArgCommand.TestEnum.V2, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("v2")))
})
assertEquals(TestEnumArgCommand.TestEnum.V3, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("e1"), PlainText("v3")))
})
assertEquals(TestEnumArgCommand.TestCase.A, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tc"), PlainText("A")))
})
assertEquals(TestEnumArgCommand.TestCase.a, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tc"), PlainText("a")))
})
withTesting<Unit> {
assertFailure(TestEnumArgCommand.execute(sender, PlainText("tc"), PlainText("ENUM_NOT_FOUND")))
Testing.ok(Unit)
}
assertEquals(TestEnumArgCommand.TestCamelCase.A, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("A")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.A, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("a")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.B, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("B")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.B, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("b")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.A_B, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("A_B")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.A_B, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("a_b")))
})
assertEquals(TestEnumArgCommand.TestCamelCase.A_B, withTesting {
assertSuccess(TestEnumArgCommand.execute(sender, PlainText("tcc"), PlainText("aB")))
})
withTesting<Unit> {
assertFailure(TestEnumArgCommand.execute(sender, PlainText("tc"), PlainText("ENUM_NOT_FOUND")))
Testing.ok(Unit)
}
}
}
@Test
fun testSimpleArgsSplitting() = runBlocking {
2020-10-23 13:40:16 +08:00
TestSimpleCommand.withRegistration {
assertEquals(arrayOf("test", "ttt", "tt").joinToString(), withTesting<MessageChain> {
assertSuccess(TestSimpleCommand.execute(sender, PlainText("test ttt tt")))
}.joinToString())
}
}
val image = Image("/f8f1ab55-bf8e-4236-b55e-955848d7069f")
@Test
fun `PlainText and Image args splitting`() = runBlocking {
2020-10-23 13:40:16 +08:00
TestSimpleCommand.withRegistration {
val result = withTesting<MessageChain> {
assertSuccess(TestSimpleCommand.execute(sender, buildMessageChain {
+"test"
+image
+"tt"
}))
}
assertEquals<Any>(arrayOf("test", image, "tt").joinToString(), result.toTypedArray().joinToString())
assertSame(image, result[1])
}
}
@Test
fun `test throw Exception`() {
runBlocking {
assertTrue(sender.executeCommand("").isFailure())
}
}
@Test
fun `executing command by string command`() = runBlocking {
TestCompositeCommand.withRegistration {
val result = withTesting<Int> {
assertSuccess(sender.executeCommand("/testComposite mute 1"))
}
assertEquals(1, result)
}
}
2020-10-24 11:35:10 +08:00
@Test
fun `composite command descriptors`() {
val overloads = TestCompositeCommand.overloads
assertEquals("CommandSignature(<mute>, seconds: Int = ...)", overloads[0].toString())
assertEquals("CommandSignature(<mute>, target: Long, seconds: Int)", overloads[1].toString())
2020-10-24 11:35:10 +08:00
}
@Test
fun `composite command executing`() = runBlocking {
2020-10-23 13:40:16 +08:00
TestCompositeCommand.withRegistration {
assertEquals(1, withTesting {
assertSuccess(TestCompositeCommand.execute(sender, "mute 1"))
})
}
}
@Test
fun `test first param command sender`() = runBlocking<Unit> {
object : CompositeCommand(owner, "cmd") {
@SubCommand
fun handle(sender: CommandSender, arg: String) {
Testing.ok(arg)
}
}.withRegistration {
assertEquals("test", withTesting { assertSuccess(execute(sender, "handle test")) })
}
object : SimpleCommand(owner, "cmd") {
@Handler
fun handle(sender: CommandSender, arg: String) {
Testing.ok(arg)
}
}.withRegistration {
assertEquals("hello", withTesting { assertSuccess(execute(sender, "hello")) })
}
object : SimpleCommand(owner, "cmd") {
@Handler
fun handle(arg: String, sender: CommandSender) {
Testing.ok(arg)
}
}.withRegistration {
assertFailure(execute(sender, "hello"))
}
}
@Test
fun `composite sub command resolution conflict`() {
runBlocking {
val composite = object : CompositeCommand(
2020-11-29 18:07:02 +08:00
owner,
"tr"
) {
@Suppress("UNUSED_PARAMETER")
@SubCommand
fun mute(seconds: Int) {
Testing.ok(1)
}
@Suppress("UNUSED_PARAMETER")
@SubCommand
2020-10-24 11:35:10 +08:00
fun mute(seconds: Int, arg2: Int = 1) {
Testing.ok(2)
}
}
2020-10-24 21:19:50 +08:00
registerCommand(composite)
2020-10-24 11:35:10 +08:00
println(composite.overloads.joinToString())
composite.withRegistration {
2020-10-24 11:35:10 +08:00
assertEquals(1, withTesting { assertSuccess(composite.execute(sender, "mute 123")) }) // one arg, resolves to mute(Int)
assertEquals(2, withTesting { assertSuccess(composite.execute(sender, "mute 123 1")) }) // two arg, resolved to mute(Int, Int)
}
}
}
@Test
fun `composite sub command parsing`() {
runBlocking {
class MyClass(
2020-10-23 13:40:16 +08:00
val value: Int,
)
val composite = object : CompositeCommand(
2020-11-29 18:07:02 +08:00
owner,
2020-09-09 22:49:48 +08:00
"test22",
overrideContext = buildCommandArgumentContext {
2020-10-09 12:35:16 +08:00
add(object : CommandValueArgumentParser<MyClass> {
override fun parse(raw: String, sender: CommandSender): MyClass {
return MyClass(raw.toInt())
}
override fun parse(raw: MessageContent, sender: CommandSender): MyClass {
2020-10-24 12:00:56 +08:00
if (raw is PlainText) return parse(raw.content, sender)
assertSame(image, raw)
return MyClass(2)
}
})
}
) {
@SubCommand
fun mute(seconds: MyClass) {
Testing.ok(seconds)
}
}
composite.withRegistration {
2020-10-24 12:00:56 +08:00
assertEquals(333, withTesting<MyClass> { assertSuccess(execute(sender, "mute 333")) }.value)
assertEquals(2, withTesting<MyClass> {
2020-10-23 13:40:16 +08:00
assertSuccess(
execute(sender, buildMessageChain {
+"mute"
+image
})
)
}.value)
}
}
}
2020-06-28 11:46:06 +08:00
@Test
fun `test simple command`() {
runBlocking {
val simple = object : SimpleCommand(owner, "test") {
@Handler
fun onCommand(string: String) {
Testing.ok(string)
}
}
simple.withRegistration {
2020-09-04 18:56:42 +08:00
// assertEquals("xxx", withTesting { simple.execute(sender, "xxx") })
2020-09-09 22:49:48 +08:00
assertEquals("xxx", withTesting { assertSuccess(sender.executeCommand("/test xxx")) })
2020-06-28 11:46:06 +08:00
}
}
}
@Test
fun `test optional argument command`() {
runBlocking {
val optionCommand = object : CompositeCommand(
2020-11-29 18:07:02 +08:00
owner,
"testOptional"
) {
@SubCommand
fun optional(arg1: String, arg2: String = "Here is optional", arg3: String? = null) {
println(arg1)
println(arg2)
println(arg3)
// println(arg3)
Testing.ok(Unit)
}
}
optionCommand.withRegistration {
withTesting<Unit> {
assertSuccess(sender.executeCommand("/testOptional optional 1"))
}
}
}
}
2020-10-24 13:14:25 +08:00
@Test
fun `test vararg`() {
runBlocking {
val optionCommand = object : CompositeCommand(
2020-11-29 18:07:02 +08:00
owner,
2020-10-24 13:14:25 +08:00
"test"
) {
@SubCommand
fun vararg(arg1: Int, vararg x: String) {
assertEquals(1, arg1)
Testing.ok(x)
}
}
optionCommand.withRegistration {
assertArrayEquals(
emptyArray<String>(),
withTesting {
assertSuccess(sender.executeCommand("/test vararg 1"))
}
)
assertArrayEquals(
arrayOf("s"),
withTesting<Array<String>> {
assertSuccess(sender.executeCommand("/test vararg 1 s"))
}
)
assertArrayEquals(
arrayOf("s", "s", "s"),
withTesting {
assertSuccess(sender.executeCommand("/test vararg 1 s s s"))
}
)
}
}
}
}
fun <T> assertArrayEquals(expected: Array<out T>, actual: Array<out T>, message: String? = null) {
asserter.assertEquals(message, expected.contentToString(), actual.contentToString())
2020-08-01 23:05:03 +08:00
}
2020-10-20 13:27:13 +08:00
@OptIn(ExperimentalCommandDescriptors::class)
internal fun assertSuccess(result: CommandExecuteResult) {
if (result.isFailure()) {
throw result.exception ?: AssertionError(result.toString())
}
2020-10-27 21:58:57 +08:00
}
2021-01-24 02:34:13 +08:00
@OptIn(ExperimentalCommandDescriptors::class)
internal fun assertFailure(result: CommandExecuteResult) {
if (!result.isFailure()) {
throw AssertionError("$result not a failure")
}
}