2020-07-24 20:11:10 +08:00
|
|
|
/*
|
2020-08-16 23:36:24 +08:00
|
|
|
* Copyright 2019-2020 Mamoe Technologies and contributors.
|
2020-07-24 20:11:10 +08:00
|
|
|
*
|
2020-08-23 17:46:51 +08:00
|
|
|
* 此源代码的使用受 GNU AFFERO GENERAL PUBLIC LICENSE version 3 许可证的约束, 可以在以下链接找到该许可证.
|
2020-10-28 13:35:15 +08:00
|
|
|
* Use of this source code is governed by the GNU AFFERO GENERAL PUBLIC LICENSE version 3 license that can be found through the following link.
|
2020-07-24 20:11:10 +08:00
|
|
|
*
|
|
|
|
* https://github.com/mamoe/mirai/blob/master/LICENSE
|
|
|
|
*/
|
|
|
|
|
|
|
|
@file:Suppress("unused")
|
|
|
|
|
|
|
|
package net.mamoe.mirai.console
|
|
|
|
|
|
|
|
import kotlinx.atomicfu.locks.withLock
|
2020-08-30 17:31:52 +08:00
|
|
|
import kotlinx.coroutines.*
|
2020-08-22 19:43:07 +08:00
|
|
|
import net.mamoe.mirai.Bot
|
2020-08-16 23:21:11 +08:00
|
|
|
import net.mamoe.mirai.console.MiraiConsoleImplementation.Companion.start
|
2020-07-24 20:11:10 +08:00
|
|
|
import net.mamoe.mirai.console.command.ConsoleCommandSender
|
2020-08-22 01:36:22 +08:00
|
|
|
import net.mamoe.mirai.console.data.PluginDataStorage
|
2020-08-16 23:21:11 +08:00
|
|
|
import net.mamoe.mirai.console.internal.MiraiConsoleImplementationBridge
|
2020-09-12 19:43:51 +08:00
|
|
|
import net.mamoe.mirai.console.plugin.jvm.JvmPluginLoader
|
2020-09-12 20:20:41 +08:00
|
|
|
import net.mamoe.mirai.console.plugin.loader.PluginLoader
|
2020-08-22 19:43:07 +08:00
|
|
|
import net.mamoe.mirai.console.util.ConsoleInput
|
2020-08-30 17:31:52 +08:00
|
|
|
import net.mamoe.mirai.message.data.Message
|
2020-08-22 19:43:07 +08:00
|
|
|
import net.mamoe.mirai.utils.BotConfiguration
|
|
|
|
import net.mamoe.mirai.utils.LoginSolver
|
2020-07-24 20:11:10 +08:00
|
|
|
import net.mamoe.mirai.utils.MiraiLogger
|
2020-08-22 19:43:07 +08:00
|
|
|
import java.nio.file.Path
|
2020-08-30 17:31:52 +08:00
|
|
|
import java.util.*
|
2020-07-24 20:11:10 +08:00
|
|
|
import java.util.concurrent.locks.ReentrantLock
|
|
|
|
import kotlin.annotation.AnnotationTarget.*
|
2020-08-20 23:46:51 +08:00
|
|
|
import kotlin.coroutines.CoroutineContext
|
2020-10-25 21:10:45 +08:00
|
|
|
import kotlin.system.exitProcess
|
2020-07-24 20:11:10 +08:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* 标记一个仅用于 [MiraiConsole] 前端实现的 API.
|
|
|
|
*
|
|
|
|
* 这些 API 只应由前端实现者使用, 而不应该被插件或其他调用者使用.
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
2020-09-12 19:26:36 +08:00
|
|
|
@Retention(AnnotationRetention.BINARY)
|
|
|
|
@RequiresOptIn(level = RequiresOptIn.Level.ERROR)
|
2020-07-24 20:11:10 +08:00
|
|
|
@Target(CLASS, TYPEALIAS, FUNCTION, PROPERTY, FIELD, CONSTRUCTOR)
|
|
|
|
@MustBeDocumented
|
|
|
|
public annotation class ConsoleFrontEndImplementation
|
|
|
|
|
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* 实现 [MiraiConsole] 的接口
|
|
|
|
*
|
2020-10-27 20:39:38 +08:00
|
|
|
* **注意**: 随着 Console 的更新, 在版本号 `x.y.z` 的 `y` 修改时此接口可能就会发生 ABI 变动. 意味着前端实现着需要跟随 Console 更新.
|
2020-08-16 23:21:11 +08:00
|
|
|
*
|
2020-08-30 17:31:52 +08:00
|
|
|
* @see MiraiConsoleImplementation.start 启动
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
|
|
|
@ConsoleFrontEndImplementation
|
|
|
|
public interface MiraiConsoleImplementation : CoroutineScope {
|
2020-08-20 23:46:51 +08:00
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* [MiraiConsole] 的 [CoroutineScope.coroutineContext], 必须拥有如下元素
|
|
|
|
*
|
|
|
|
* - [Job]: 用于管理整个 [MiraiConsole] 的生命周期. 当此 [Job] 被 [Job.cancel] 后, [MiraiConsole] 就会结束.
|
|
|
|
* - [CoroutineExceptionHandler]: 用于处理 [MiraiConsole] 所有协程抛出的 **未被捕捉** 的异常. 不是所有异常都会被传递到这里.
|
2020-08-20 23:46:51 +08:00
|
|
|
*/
|
|
|
|
public override val coroutineContext: CoroutineContext
|
|
|
|
|
2020-07-24 20:11:10 +08:00
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* Console 运行根目录绝对路径 (否则可能会被一些 native 插件覆盖相对路径)
|
|
|
|
* @see MiraiConsole.rootPath 获取更多信息
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
2020-08-22 19:43:07 +08:00
|
|
|
public val rootPath: Path
|
2020-07-24 20:11:10 +08:00
|
|
|
|
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* 本前端实现的描述信息
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
2020-08-22 19:43:07 +08:00
|
|
|
public val frontEndDescription: MiraiConsoleFrontEndDescription
|
2020-07-24 20:11:10 +08:00
|
|
|
|
|
|
|
/**
|
2020-09-12 19:43:51 +08:00
|
|
|
* 内建加载器列表, 一般需要包含 [JvmPluginLoader].
|
2020-08-30 17:31:52 +08:00
|
|
|
*
|
|
|
|
* @return 不可变的 [List], [Collections.unmodifiableList]
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
2020-09-12 20:20:41 +08:00
|
|
|
public val builtInPluginLoaders: List<Lazy<PluginLoader<*, *>>>
|
2020-07-24 20:11:10 +08:00
|
|
|
|
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* 由 Kotlin 用户实现
|
2020-07-31 16:35:41 +08:00
|
|
|
*
|
2020-08-30 17:31:52 +08:00
|
|
|
* @see [ConsoleCommandSender]
|
2020-07-24 20:11:10 +08:00
|
|
|
*/
|
2020-08-30 17:31:52 +08:00
|
|
|
@ConsoleFrontEndImplementation
|
|
|
|
public interface ConsoleCommandSenderImpl {
|
|
|
|
@JvmSynthetic
|
|
|
|
public suspend fun sendMessage(message: Message)
|
|
|
|
|
|
|
|
@JvmSynthetic
|
|
|
|
public suspend fun sendMessage(message: String)
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 由 Java 用户实现
|
|
|
|
*
|
|
|
|
* @see [ConsoleCommandSender]
|
|
|
|
*/
|
|
|
|
@Suppress("INAPPLICABLE_JVM_NAME")
|
|
|
|
@ConsoleFrontEndImplementation
|
|
|
|
public interface JConsoleCommandSenderImpl : ConsoleCommandSenderImpl {
|
|
|
|
@JvmName("sendMessage")
|
|
|
|
public fun sendMessageJ(message: Message)
|
|
|
|
|
|
|
|
@JvmName("sendMessage")
|
|
|
|
public fun sendMessageJ(message: String)
|
2020-07-24 20:11:10 +08:00
|
|
|
|
2020-08-30 17:31:52 +08:00
|
|
|
|
|
|
|
@JvmSynthetic
|
|
|
|
public override suspend fun sendMessage(message: Message): Unit =
|
|
|
|
withContext(Dispatchers.IO) { sendMessageJ(message) }
|
|
|
|
|
|
|
|
@JvmSynthetic
|
|
|
|
public override suspend fun sendMessage(message: String): Unit =
|
|
|
|
withContext(Dispatchers.IO) { sendMessageJ(message) }
|
|
|
|
}
|
|
|
|
|
2020-09-13 00:26:50 +08:00
|
|
|
/**
|
|
|
|
* [ConsoleCommandSender]
|
|
|
|
*/
|
2020-08-30 17:31:52 +08:00
|
|
|
public val consoleCommandSender: ConsoleCommandSenderImpl
|
2020-07-24 20:11:10 +08:00
|
|
|
|
2020-09-12 20:20:41 +08:00
|
|
|
public val dataStorageForJvmPluginLoader: PluginDataStorage
|
|
|
|
public val configStorageForJvmPluginLoader: PluginDataStorage
|
2020-08-22 01:36:22 +08:00
|
|
|
public val dataStorageForBuiltIns: PluginDataStorage
|
2020-08-26 21:27:28 +08:00
|
|
|
public val configStorageForBuiltIns: PluginDataStorage
|
2020-07-24 20:11:10 +08:00
|
|
|
|
2020-08-22 19:43:07 +08:00
|
|
|
/**
|
|
|
|
* @see ConsoleInput 的实现
|
2020-09-13 00:26:50 +08:00
|
|
|
* @see JConsoleInput
|
2020-08-22 19:43:07 +08:00
|
|
|
*/
|
|
|
|
public val consoleInput: ConsoleInput
|
|
|
|
|
2020-09-13 00:26:50 +08:00
|
|
|
/**
|
|
|
|
* 供 Java 用户实现 [ConsoleInput]
|
|
|
|
*/
|
|
|
|
@Suppress("INAPPLICABLE_JVM_NAME")
|
|
|
|
@ConsoleFrontEndImplementation
|
|
|
|
public interface JConsoleInput : ConsoleInput {
|
|
|
|
/**
|
|
|
|
* @see ConsoleInput.requestInput
|
|
|
|
*/
|
|
|
|
@JvmName("requestInput")
|
|
|
|
public fun requestInputJ(hint: String): String
|
|
|
|
|
|
|
|
override suspend fun requestInput(hint: String): String {
|
|
|
|
return withContext(Dispatchers.IO) { requestInputJ(hint) }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 19:43:07 +08:00
|
|
|
/**
|
|
|
|
* 创建一个 [LoginSolver]
|
|
|
|
*
|
2020-08-30 17:31:52 +08:00
|
|
|
* **备注**: 此函数通常在构造 [Bot] 实例, 即 [MiraiConsole.addBot] 时调用.
|
2020-08-22 19:43:07 +08:00
|
|
|
*
|
|
|
|
* @param requesterBot 请求者 [Bot.id]
|
|
|
|
* @param configuration 请求者 [Bot.configuration]
|
|
|
|
*
|
|
|
|
* @see LoginSolver.Default
|
|
|
|
*/
|
|
|
|
public fun createLoginSolver(requesterBot: Long, configuration: BotConfiguration): LoginSolver
|
|
|
|
|
|
|
|
/**
|
2020-08-30 17:31:52 +08:00
|
|
|
* 创建一个 [MiraiLogger].
|
|
|
|
*
|
|
|
|
* **注意**: [MiraiConsole] 会将 [net.mamoe.mirai.utils.DefaultLogger] 设置为 `MiraiConsole::createLogger`.
|
|
|
|
* 因此不要在 [createLogger] 中调用 [net.mamoe.mirai.utils.DefaultLogger]
|
2020-08-22 19:43:07 +08:00
|
|
|
*/
|
2020-08-30 17:31:52 +08:00
|
|
|
public fun createLogger(identity: String?): MiraiLogger
|
2020-08-22 19:43:07 +08:00
|
|
|
|
2020-11-06 13:41:46 +08:00
|
|
|
/**
|
|
|
|
* 该前端是否支持使用 Ansi 输出彩色信息
|
|
|
|
*
|
|
|
|
* 注: 若为 `true`, 建议携带 `org.fusesource.jansi:jansi`
|
|
|
|
*/
|
|
|
|
public val isAnsiSupport: Boolean get() = false
|
|
|
|
|
2020-07-24 20:11:10 +08:00
|
|
|
public companion object {
|
|
|
|
internal lateinit var instance: MiraiConsoleImplementation
|
|
|
|
private val initLock = ReentrantLock()
|
|
|
|
|
2020-09-17 10:23:29 +08:00
|
|
|
/**
|
|
|
|
* 可由前端调用, 获取当前的 [MiraiConsoleImplementation] 实例
|
|
|
|
*
|
2020-10-25 21:10:45 +08:00
|
|
|
* 必须在 [start] 之后才能使用, 否则抛出 [UninitializedPropertyAccessException]
|
2020-09-17 10:23:29 +08:00
|
|
|
*/
|
|
|
|
@JvmStatic
|
|
|
|
@ConsoleFrontEndImplementation
|
|
|
|
public fun getInstance(): MiraiConsoleImplementation = instance
|
|
|
|
|
2020-08-30 17:31:52 +08:00
|
|
|
/** 由前端调用, 初始化 [MiraiConsole] 实例并启动 */
|
2020-07-24 20:11:10 +08:00
|
|
|
@JvmStatic
|
2020-08-16 23:21:11 +08:00
|
|
|
@ConsoleFrontEndImplementation
|
2020-08-26 21:27:28 +08:00
|
|
|
@Throws(MalformedMiraiConsoleImplementationError::class)
|
2020-07-24 20:11:10 +08:00
|
|
|
public fun MiraiConsoleImplementation.start(): Unit = initLock.withLock {
|
2020-09-17 10:24:48 +08:00
|
|
|
if (::instance.isInitialized) error("Mirai Console is already initialized.")
|
2020-07-24 20:11:10 +08:00
|
|
|
this@Companion.instance = this
|
2020-10-25 21:10:45 +08:00
|
|
|
kotlin.runCatching {
|
|
|
|
MiraiConsoleImplementationBridge.doStart()
|
|
|
|
}.onFailure { e ->
|
|
|
|
kotlin.runCatching {
|
|
|
|
MiraiConsole.mainLogger.error("Failed to init MiraiConsole.", e)
|
|
|
|
}.onFailure {
|
|
|
|
e.printStackTrace()
|
|
|
|
}
|
|
|
|
|
|
|
|
kotlin.runCatching {
|
|
|
|
MiraiConsole.cancel()
|
|
|
|
}.onFailure {
|
|
|
|
it.printStackTrace()
|
|
|
|
}
|
|
|
|
exitProcess(1)
|
|
|
|
}
|
2020-07-24 20:11:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|