From b5fcc6e87202666d58bb3c6d295305f9a95916ee Mon Sep 17 00:00:00 2001 From: "jiahua.liu" <n@mamoe.net> Date: Thu, 13 Feb 2020 21:03:45 +0800 Subject: [PATCH] Config supported 10/11 --- mirai-console/src/main/kotlin/MiraiConsole.kt | 21 +-- .../net/mamoe/mirai/plugin/ConfigSection.kt | 145 ++++++++++++------ 2 files changed, 105 insertions(+), 61 deletions(-) diff --git a/mirai-console/src/main/kotlin/MiraiConsole.kt b/mirai-console/src/main/kotlin/MiraiConsole.kt index 20c177d7d..1cda36529 100644 --- a/mirai-console/src/main/kotlin/MiraiConsole.kt +++ b/mirai-console/src/main/kotlin/MiraiConsole.kt @@ -45,7 +45,6 @@ object MiraiConsole { logger("\"/login qqnumber qqpassword \" to login a bot") logger("\"/login qq号 qq密码 \" 来登陆一个BOT") - MiraiProperties() CommandManager.register(DefaultCommands.DefaultLoginCommand()) pluginManager.loadPlugins() CommandListener.start() @@ -117,22 +116,16 @@ object MiraiConsole { } object MiraiProperties { - val init = !File("$path/mirai.json").exists() + var config = File("$path/mirai.json").loadAsConfig() - var config = Config.load("$path/mirai.json") - - var HTTP_API_ENABLE: Boolean by config.withDefault { true } - var HTTP_API_PORT: Int by config.withDefault { 8080 } - var HTTP_API_AUTH_KEY: String by config - - operator fun invoke() { - if (init) { - HTTP_API_AUTH_KEY = "INITKEY" + generateSessionKey() - logger("Mirai HTTPAPI authkey 已随机生成, 请注意修改") - } + var HTTP_API_ENABLE: Boolean by config.withDefaultWrite { true } + var HTTP_API_PORT: Int by config.withDefaultWrite { 8080 } + var HTTP_API_AUTH_KEY: String by config.withDefaultWriteSave { + "InitKey".also { + logger("Mirai HTTPAPI auth key 已随机生成 请注意修改") + } + generateSessionKey() } - } } diff --git a/mirai-console/src/main/kotlin/net/mamoe/mirai/plugin/ConfigSection.kt b/mirai-console/src/main/kotlin/net/mamoe/mirai/plugin/ConfigSection.kt index 432f2df1e..1f1c6a488 100644 --- a/mirai-console/src/main/kotlin/net/mamoe/mirai/plugin/ConfigSection.kt +++ b/mirai-console/src/main/kotlin/net/mamoe/mirai/plugin/ConfigSection.kt @@ -14,12 +14,11 @@ import com.alibaba.fastjson.JSONObject import com.alibaba.fastjson.TypeReference import com.alibaba.fastjson.parser.Feature import kotlinx.serialization.* -import kotlinx.serialization.json.Json -import net.mamoe.mirai.utils.cryptor.contentToString import java.io.File import java.util.concurrent.ConcurrentHashMap -import java.util.function.BiConsumer +import java.util.concurrent.ConcurrentSkipListMap import kotlin.properties.ReadWriteProperty +import kotlin.reflect.KClass import kotlin.reflect.KProperty import kotlin.reflect.full.isSubclassOf @@ -47,6 +46,7 @@ interface Config { operator fun set(key: String, value: Any) operator fun get(key: String): Any? fun exist(key: String): Boolean + fun setIfAbsent(key: String, value: Any) fun asMap(): Map<String, Any> fun save() @@ -67,47 +67,109 @@ interface Config { } } + +fun File.loadAsConfig(): Config { + return Config.load(this) +} + +/* 最简单的代理 */ +inline operator fun <reified T : Any> Config.getValue(thisRef: Any?, property: KProperty<*>): T { + return smartCast(property) +} + +inline operator fun <reified T : Any> Config.setValue(thisRef: Any?, property: KProperty<*>, value: T) { + this[property.name] = value +} + +/* 带有默认值的代理 */ inline fun <reified T : Any> Config.withDefault( - autoSave: Boolean = true, - crossinline defaultValue: () -> T + noinline defaultValue: () -> T ): ReadWriteProperty<Any, T> { + val default by lazy { defaultValue.invoke() } return object : ReadWriteProperty<Any, T> { override fun getValue(thisRef: Any, property: KProperty<*>): T { - if (!this@withDefault.exist(property.name)) { - return defaultValue.invoke() + if (this@withDefault.exist(property.name)) {//unsafe + return this@withDefault.smartCast(property) } - return smartCast(property) + return default } override fun setValue(thisRef: Any, property: KProperty<*>, value: T) { this@withDefault[property.name] = value - if (autoSave) save() } } } -@Suppress("IMPLICIT_CAST_TO_ANY") -inline fun <reified T> Config.smartCast(property: KProperty<*>): T { - return when (T::class) { - String::class -> this.getString(property.name) - Int::class -> this.getInt(property.name) - Float::class -> this.getFloat(property.name) - Double::class -> this.getDouble(property.name) - Long::class -> this.getLong(property.name) - Boolean::class -> this.getBoolean(property.name) +/* 带有默认值且如果为空会写入的代理 */ +inline fun <reified T : Any> Config.withDefaultWrite( + noinline defaultValue: () -> T +): WithDefaultWriteLoader<T> { + return WithDefaultWriteLoader(T::class, this, defaultValue, false) +} + +/* 带有默认值且如果为空会写入保存的代理 */ +inline fun <reified T : Any> Config.withDefaultWriteSave( + noinline defaultValue: () -> T +): WithDefaultWriteLoader<T> { + return WithDefaultWriteLoader(T::class, this, defaultValue, true) +} + +class WithDefaultWriteLoader<T : Any>( + private val _class: KClass<T>, + private val config: Config, + private val defaultValue: () -> T, + private val save: Boolean +) { + operator fun provideDelegate( + thisRef: Any, + prop: KProperty<*> + ): ReadWriteProperty<Any, T> { + val defaultValue by lazy { defaultValue.invoke() } + config.setIfAbsent(prop.name, defaultValue) + if (save) { + config.save() + } + return object : ReadWriteProperty<Any, T> { + override fun getValue(thisRef: Any, property: KProperty<*>): T { + if (config.exist(property.name)) {//unsafe + return config._smartCast(property.name, _class) + } + return defaultValue + } + + override fun setValue(thisRef: Any, property: KProperty<*>, value: T) { + config[property.name] = value + } + } + } +} + +inline fun <reified T : Any> Config.smartCast(property: KProperty<*>): T { + return _smartCast(property.name, T::class) +} + +@Suppress("IMPLICIT_CAST_TO_ANY", "UNCHECKED_CAST") +fun <T : Any> Config._smartCast(propertyName: String, _class: KClass<T>): T { + return when (_class) { + String::class -> this.getString(propertyName) + Int::class -> this.getInt(propertyName) + Float::class -> this.getFloat(propertyName) + Double::class -> this.getDouble(propertyName) + Long::class -> this.getLong(propertyName) + Boolean::class -> this.getBoolean(propertyName) else -> when { - T::class.isSubclassOf(ConfigSection::class) -> this.getConfigSection(property.name) - T::class == List::class || T::class == MutableList::class -> { - val list = this.getList(property.name) + _class.isSubclassOf(ConfigSection::class) -> this.getConfigSection(propertyName) + _class == List::class || _class == MutableList::class -> { + val list = this.getList(propertyName) return if (list.isEmpty()) { list } else { when (list[0]!!::class) { - String::class -> getStringList(property.name) - Int::class -> getIntList(property.name) - Float::class -> getFloatList(property.name) - Double::class -> getDoubleList(property.name) - Long::class -> getLongList(property.name) + String::class -> getStringList(propertyName) + Int::class -> getIntList(propertyName) + Float::class -> getFloatList(propertyName) + Double::class -> getDoubleList(propertyName) + Long::class -> getLongList(propertyName) else -> { error("unsupported type") } @@ -121,14 +183,6 @@ inline fun <reified T> Config.smartCast(property: KProperty<*>): T { } as T } -inline operator fun <reified T> Config.getValue(thisRef: Any?, property: KProperty<*>): T { - return smartCast(property) -} - -inline operator fun <reified T> Config.setValue(thisRef: Any?, property: KProperty<*>, value: T) { - this[property.name] = value!! - this.save() -} interface ConfigSection : Config { @@ -187,12 +241,16 @@ interface ConfigSection : Config { override fun exist(key: String): Boolean { return get(key) != null } + + override fun setIfAbsent(key: String, value: Any) { + if (!exist(key)) set(key, value) + } } @Serializable open class ConfigSectionImpl() : ConcurrentHashMap<String, Any>(), ConfigSection { override fun set(key: String, value: Any) { - this.put(key, value) + super.put(key, value) } override operator fun get(key: String): Any? { @@ -210,6 +268,10 @@ open class ConfigSectionImpl() : ConcurrentHashMap<String, Any>(), ConfigSection override fun save() { } + + override fun setIfAbsent(key: String, value: Any) { + this.putIfAbsent(key, value)//atomic + } } @@ -255,26 +317,15 @@ class JsonConfig internal constructor(file: File) : FileConfigImpl(file) { if (content.isEmpty() || content.isBlank() || content == "{}") { return ConfigSectionImpl() } - val section = JSON.parseObject( + return JSON.parseObject<ConfigSectionImpl>( content, object : TypeReference<ConfigSectionImpl>() {}, Feature.OrderedField ) - return section } @UnstableDefault override fun serialize(config: ConfigSection): String { return JSONObject.toJSONString(config) } - - internal class AnySerializer(override val descriptor: SerialDescriptor) : KSerializer<Any> { - override fun deserialize(decoder: Decoder): Any { - TODO("not implemented") //To change body of created functions use File | Settings | File Templates. - } - - override fun serialize(encoder: Encoder, obj: Any) { - TODO("not implemented") //To change body of created functions use File | Settings | File Templates. - } - } } \ No newline at end of file