From b0e978b2d10f0e89497e5335c1b792df9ff66206 Mon Sep 17 00:00:00 2001 From: DBotThePony Date: Thu, 2 Mar 2023 15:15:54 +0700 Subject: [PATCH] =?UTF-8?q?Avatar,=20QuestInstance,=20=D1=81=D0=B5=D1=80?= =?UTF-8?q?=D1=8C=D1=91=D0=B7=D0=BD=D1=8B=D0=B5=20=D1=83=D0=BB=D1=83=D1=87?= =?UTF-8?q?=D1=88=D0=B5=D0=BD=D0=B8=D1=8F=20Lua?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- build.gradle.kts | 2 +- lua_glue.c | 4 +- .../ru/dbotthepony/kstarbound/lua/LuaJNR.java | 3 +- .../kotlin/ru/dbotthepony/kstarbound/Main.kt | 45 +- .../dbotthepony/kstarbound/ObjectRegistry.kt | 12 +- .../ru/dbotthepony/kstarbound/Starbound.kt | 149 ++- .../kstarbound/api/IStarboundFileSystem.kt | 25 +- .../kstarbound/client/render/FrameAnimator.kt | 3 +- ...toryFilterConfig.kt => BagFilterConfig.kt} | 15 +- .../kstarbound/defs/player/InventoryConfig.kt | 17 +- .../defs/player/PlayerDefinition.kt | 2 +- .../kstarbound/defs/player/TechDefinition.kt | 15 + .../kstarbound/defs/quest/QuestTemplate.kt | 17 + .../ru/dbotthepony/kstarbound/io/Ext.kt | 14 + .../kstarbound/io/json/KotlinAdapters.kt | 32 + .../ru/dbotthepony/kstarbound/lua/Errors.kt | 10 - .../ru/dbotthepony/kstarbound/lua/LuaState.kt | 870 ++++++++++++++---- .../ru/dbotthepony/kstarbound/lua/Scripts.kt | 66 ++ .../dbotthepony/kstarbound/player/Avatar.kt | 492 ++++++++++ .../kstarbound/player/AvatarBag.kt | 57 ++ .../dbotthepony/kstarbound/player/Player.kt | 12 + .../kstarbound/player/QuestDescriptor.kt | 18 + .../kstarbound/player/QuestInstance.kt | 250 +++++ .../dbotthepony/kstarbound/tools/Sbon2Json.kt | 121 +++ .../ru/dbotthepony/kstarbound/util/Ext.kt | 16 + .../kstarbound/util/ItemDescriptor.kt | 145 +++ .../dbotthepony/kstarbound/util/SBPattern.kt | 3 +- .../ru/dbotthepony/kstarbound/util/Utils.kt | 89 ++ src/main/resources/scripts/config.lua | 17 + .../resources/scripts/message_handler.lua | 27 + src/main/resources/scripts/player.lua | 21 + src/main/resources/scripts/polyfill.lua | 8 + 32 files changed, 2324 insertions(+), 253 deletions(-) rename src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/{InventoryFilterConfig.kt => BagFilterConfig.kt} (57%) create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/TechDefinition.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/defs/quest/QuestTemplate.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/io/json/KotlinAdapters.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/lua/Scripts.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/player/Avatar.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/player/AvatarBag.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/player/Player.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestDescriptor.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestInstance.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/tools/Sbon2Json.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/util/Ext.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/util/ItemDescriptor.kt create mode 100644 src/main/kotlin/ru/dbotthepony/kstarbound/util/Utils.kt create mode 100644 src/main/resources/scripts/config.lua create mode 100644 src/main/resources/scripts/message_handler.lua create mode 100644 src/main/resources/scripts/player.lua create mode 100644 src/main/resources/scripts/polyfill.lua diff --git a/build.gradle.kts b/build.gradle.kts index 10153738..480c493b 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -25,7 +25,7 @@ application { mainClass.set("ru.dbotthepony.kstarbound.MainKt") } -java.toolchain.languageVersion.set(JavaLanguageVersion.of(20)) +java.toolchain.languageVersion.set(JavaLanguageVersion.of(17)) tasks.compileKotlin { kotlinOptions { diff --git a/lua_glue.c b/lua_glue.c index 84fbd00f..b345be6b 100644 --- a/lua_glue.c +++ b/lua_glue.c @@ -12,7 +12,7 @@ static int lua_jniFunc(lua_State *state) { jint result = (*env)->CallIntMethod(env, *lua_JCClosure, callback, (long long) state); - if (result != 0) { + if (result <= -1) { const char* errMsg = lua_tostring(state, -1); if (errMsg == NULL) @@ -21,7 +21,7 @@ static int lua_jniFunc(lua_State *state) { return luaL_error(state, "%s", errMsg); } - return 0; + return result; } static int mark_closure_free(lua_State *state) { diff --git a/src/main/java/ru/dbotthepony/kstarbound/lua/LuaJNR.java b/src/main/java/ru/dbotthepony/kstarbound/lua/LuaJNR.java index 74ef04d6..a5b95e28 100644 --- a/src/main/java/ru/dbotthepony/kstarbound/lua/LuaJNR.java +++ b/src/main/java/ru/dbotthepony/kstarbound/lua/LuaJNR.java @@ -32,7 +32,7 @@ public interface LuaJNR { public void luaopen_debug(@NotNull Pointer luaState); @Nullable - public Pointer lua_tolstring(@NotNull Pointer luaState, int index, @NotNull Pointer size); + public Pointer lua_tolstring(@NotNull Pointer luaState, int index, @LongLong long size); public int lua_load(@NotNull Pointer luaState, @LongLong long reader, long userData, @NotNull String chunkName, @NotNull String mode); @@ -97,6 +97,7 @@ public interface LuaJNR { public int lua_toboolean(@NotNull Pointer luaState, int index); public int lua_tocfunction(@NotNull Pointer luaState, int index); public int lua_toclose(@NotNull Pointer luaState, int index); + public int lua_gettable(@NotNull Pointer luaState, int index); @LongLong public long lua_tointegerx(@NotNull Pointer luaState, int index, @LongLong long successCode); diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/Main.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/Main.kt index a9b128b8..f304def7 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/Main.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/Main.kt @@ -1,20 +1,20 @@ package ru.dbotthepony.kstarbound -import com.google.gson.GsonBuilder -import com.google.gson.JsonElement -import com.google.gson.JsonNull -import com.google.gson.JsonObject -import com.google.gson.JsonPrimitive -import com.sun.jna.Native + import org.apache.logging.log4j.LogManager import org.lwjgl.Version import org.lwjgl.glfw.GLFW.glfwSetWindowShouldClose import ru.dbotthepony.kstarbound.client.StarboundClient import ru.dbotthepony.kstarbound.client.render.Animator import ru.dbotthepony.kstarbound.defs.animation.AnimationDefinition -import ru.dbotthepony.kstarbound.defs.item.DynamicItemDefinition import ru.dbotthepony.kstarbound.io.BTreeDB import ru.dbotthepony.kstarbound.lua.LuaState +import ru.dbotthepony.kstarbound.lua.LuaType +import ru.dbotthepony.kstarbound.player.Avatar +import ru.dbotthepony.kstarbound.player.Player +import ru.dbotthepony.kstarbound.player.QuestDescriptor +import ru.dbotthepony.kstarbound.player.QuestInstance +import ru.dbotthepony.kstarbound.util.JVMTimeSource import ru.dbotthepony.kstarbound.world.ChunkPos import ru.dbotthepony.kstarbound.world.entities.ItemEntity import ru.dbotthepony.kstarbound.world.entities.PlayerEntity @@ -22,25 +22,12 @@ import ru.dbotthepony.kvector.vector.ndouble.Vector2d import java.io.ByteArrayInputStream import java.io.DataInputStream import java.io.File -import java.util.Random +import java.util.* import java.util.zip.Inflater private val LOGGER = LogManager.getLogger() fun main() { - if (true) { - val lua = LuaState() - - lua.load("return {lemon = 'Bouncy', [4] = 'four', [3] = 'watermelon', h = {h = 'h!'}}") - lua.call(numResults = 1) - - val a = lua.popTable()!! - - println("${a["lemon"]} $a") - - return - } - val starbound = Starbound() LOGGER.info("Running LWJGL ${Version.getVersion()}") @@ -217,8 +204,22 @@ fun main() { val animator = Animator(client.world!!, def) + //client.onPostDrawWorld { + // animator.render(client.gl.matrixStack) + //} + + val avatar = Avatar(starbound, UUID.randomUUID()) + val quest = QuestInstance(avatar, descriptor = QuestDescriptor("floran_mission1")) + quest.init() + quest.start() + + var last = JVMTimeSource.INSTANCE.millis + client.onPostDrawWorld { - animator.render(client.gl.matrixStack) + if (JVMTimeSource.INSTANCE.millis - last > 20L) { + quest.update(1) + last = JVMTimeSource.INSTANCE.millis + } } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/ObjectRegistry.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/ObjectRegistry.kt index 224afc68..378672b1 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/ObjectRegistry.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/ObjectRegistry.kt @@ -28,11 +28,19 @@ class RegistryObject(val value: T, private val json: JsonObject, val fi } override fun equals(other: Any?): Boolean { - return other is RegistryObject<*> && other.value == value && other.json == json + return other === this || other is RegistryObject<*> && other.value == value && other.json == json } + private var computedHash = false + private var hash = 0 + override fun hashCode(): Int { - return value.hashCode().rotateRight(13) xor json.hashCode() + if (!computedHash) { + hash = value.hashCode().rotateRight(13) xor json.hashCode() + computedHash = true + } + + return hash } override fun toString(): String { diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt index a547e2ea..defc3b65 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/Starbound.kt @@ -32,6 +32,8 @@ import ru.dbotthepony.kstarbound.defs.tile.LiquidDefinition import ru.dbotthepony.kstarbound.defs.particle.ParticleDefinition import ru.dbotthepony.kstarbound.defs.player.BlueprintLearnList import ru.dbotthepony.kstarbound.defs.player.PlayerDefinition +import ru.dbotthepony.kstarbound.defs.player.TechDefinition +import ru.dbotthepony.kstarbound.defs.quest.QuestTemplate import ru.dbotthepony.kstarbound.defs.tile.MaterialModifier import ru.dbotthepony.kstarbound.defs.tile.TileDefinition import ru.dbotthepony.kstarbound.io.* @@ -40,6 +42,7 @@ import ru.dbotthepony.kstarbound.io.json.AABBiTypeAdapter import ru.dbotthepony.kstarbound.io.json.EitherTypeAdapter import ru.dbotthepony.kstarbound.io.json.InternedJsonElementAdapter import ru.dbotthepony.kstarbound.io.json.InternedStringAdapter +import ru.dbotthepony.kstarbound.io.json.LongRangeAdapter import ru.dbotthepony.kstarbound.io.json.NothingAdapter import ru.dbotthepony.kstarbound.io.json.Vector2dTypeAdapter import ru.dbotthepony.kstarbound.io.json.Vector2fTypeAdapter @@ -52,10 +55,14 @@ import ru.dbotthepony.kstarbound.io.json.builder.FactoryAdapter import ru.dbotthepony.kstarbound.io.json.builder.JsonImplementationTypeFactory import ru.dbotthepony.kstarbound.io.json.factory.ArrayListAdapterFactory import ru.dbotthepony.kstarbound.io.json.factory.ImmutableCollectionAdapterFactory +import ru.dbotthepony.kstarbound.lua.LuaState +import ru.dbotthepony.kstarbound.lua.loadInternalScript import ru.dbotthepony.kstarbound.math.* +import ru.dbotthepony.kstarbound.util.ItemDescriptor import ru.dbotthepony.kstarbound.util.PathStack import ru.dbotthepony.kstarbound.util.SBPattern import ru.dbotthepony.kstarbound.util.WriteOnce +import ru.dbotthepony.kstarbound.util.traverseJsonPath import java.io.* import java.text.DateFormat import java.util.* @@ -69,8 +76,7 @@ import kotlin.collections.ArrayList class Starbound : ISBFileLocator { private val logger = LogManager.getLogger() - val stringInterner: Interner = Interners.newWeakInterner() - val pathStack = PathStack(stringInterner) + val pathStack = PathStack(Starbound.STRINGS) private val _tiles = ObjectRegistry("tiles", TileDefinition::materialName, TileDefinition::materialId) val tiles = _tiles.view @@ -96,14 +102,20 @@ class Starbound : ISBFileLocator { private val _items = ObjectRegistry("items", IItemDefinition::itemName) val items = _items.view + private val _questTemplates = ObjectRegistry("quest templates", QuestTemplate::id) + val questTemplates = _questTemplates.view + + private val _techs = ObjectRegistry("techs", TechDefinition::name) + val techs = _techs.view + val gson: Gson = with(GsonBuilder()) { serializeNulls() setDateFormat(DateFormat.LONG) setFieldNamingPolicy(FieldNamingPolicy.IDENTITY) setPrettyPrinting() - registerTypeAdapter(InternedStringAdapter(stringInterner)) - registerTypeAdapter(InternedJsonElementAdapter(stringInterner)) + registerTypeAdapter(InternedStringAdapter(STRINGS)) + registerTypeAdapter(InternedJsonElementAdapter(STRINGS)) registerTypeAdapter(Nothing::class.java, NothingAdapter) @@ -111,7 +123,7 @@ class Starbound : ISBFileLocator { registerTypeAdapterFactory(JsonImplementationTypeFactory) // ImmutableList, ImmutableSet, ImmutableMap - registerTypeAdapterFactory(ImmutableCollectionAdapterFactory(stringInterner)) + registerTypeAdapterFactory(ImmutableCollectionAdapterFactory(STRINGS)) // ArrayList registerTypeAdapterFactory(ArrayListAdapterFactory) @@ -120,10 +132,10 @@ class Starbound : ISBFileLocator { registerTypeAdapterFactory(EnumAdapter.Companion) // @JsonBuilder - registerTypeAdapterFactory(BuilderAdapter.Factory(stringInterner)) + registerTypeAdapterFactory(BuilderAdapter.Factory(STRINGS)) // @JsonFactory - registerTypeAdapterFactory(FactoryAdapter.Factory(stringInterner)) + registerTypeAdapterFactory(FactoryAdapter.Factory(STRINGS)) // Either<> registerTypeAdapterFactory(EitherTypeAdapter) @@ -150,7 +162,7 @@ class Starbound : ISBFileLocator { registerTypeAdapter(JsonFunction.Companion) // Общее - registerTypeAdapterFactory(ThingDescription.Factory(stringInterner)) + registerTypeAdapterFactory(ThingDescription.Factory(STRINGS)) registerTypeAdapter(EnumAdapter(DamageType::class, default = DamageType.NORMAL)) @@ -162,6 +174,8 @@ class Starbound : ISBFileLocator { registerTypeAdapterFactory(AssetReferenceFactory(pathStack, this@Starbound)) + registerTypeAdapter(ItemDescriptor.Adapter(this@Starbound)) + registerTypeAdapterFactory(with(RegistryReferenceFactory()) { add(tiles::get) add(tileModifiers::get) @@ -172,11 +186,49 @@ class Starbound : ISBFileLocator { add(particles::get) }) - .create() + registerTypeAdapter(LongRangeAdapter) + + create() } val atlasRegistry = AtlasConfiguration.Registry(this, pathStack, gson) + fun item(name: String): ItemDescriptor { + return ItemDescriptor(items[name] ?: return ItemDescriptor.EMPTY) + } + + fun item(name: String, count: Long): ItemDescriptor { + if (count <= 0L) + return ItemDescriptor.EMPTY + + return ItemDescriptor(items[name] ?: return ItemDescriptor.EMPTY, count = count) + } + + fun item(name: String, count: Long, parameters: JsonObject): ItemDescriptor { + if (count <= 0L) + return ItemDescriptor.EMPTY + + return ItemDescriptor(items[name] ?: return ItemDescriptor.EMPTY, count = count, parameters = parameters) + } + + fun item(descriptor: JsonObject): ItemDescriptor { + return item( + (descriptor["name"] as? JsonPrimitive)?.asString ?: return ItemDescriptor.EMPTY, + descriptor["count"]?.asLong ?: return ItemDescriptor.EMPTY, + (descriptor["parameters"] as? JsonObject)?.deepCopy() ?: JsonObject() + ) + } + + fun item(descriptor: JsonElement?): ItemDescriptor { + if (descriptor is JsonPrimitive) { + return item(descriptor.asString) + } else if (descriptor is JsonObject) { + return item(descriptor) + } else { + return ItemDescriptor.EMPTY + } + } + var initializing = false private set var initialized = false @@ -185,6 +237,70 @@ class Starbound : ISBFileLocator { @Volatile var terminateLoading = false + fun loadJsonAsset(path: String): JsonElement? { + val filename: String + val jsonPath: String? + + if (path.contains(':')) { + filename = path.substringBefore(':') + jsonPath = path.substringAfter(':') + } else { + filename = path + jsonPath = null + } + + val file = locate(filename) + + if (!file.isFile) { + return null + } + + return traverseJsonPath(jsonPath, gson.fromJson(file.reader(), JsonElement::class.java)) + } + + private fun luaRequire(it: LuaState, args: LuaState.ArgStack) { + val name = args.getString() + val file = locate(name) + + if (!file.exists) { + throw FileNotFoundException("File $name does not exist") + } + + if (!file.isFile) { + throw FileNotFoundException("File $name is a directory") + } + + val read = file.readToString() + it.load(read, chunkName = file.computeFullPath()) + it.call() + } + + fun expose(state: LuaState) { + state.pushWeak(this) { args -> + luaRequire(args.lua, args) + 0 + } + + state.storeGlobal("require") + + state.pushTable() + state.storeGlobal("root") + state.loadGlobal("root") + val root = state.stackTop + + state.push("assetJson") + state.pushWeak(this) {args -> + args.lua.push(loadJsonAsset(args.getString())) + 1 + } + + state.setTableValue(root) + state.pop() + + state.load(polyfill, "starbound.jar!/scripts/polyfill.lua") + state.call() + } + private val archivePaths = ArrayList() private val fileSystems = ArrayList() @@ -380,6 +496,8 @@ class Starbound : ISBFileLocator { loadStage(callback, _statusEffects, ext2files["statuseffect"] ?: listOf()) loadStage(callback, _species, ext2files["species"] ?: listOf()) loadStage(callback, _particles, ext2files["particle"] ?: listOf()) + loadStage(callback, _questTemplates, ext2files["questtemplate"] ?: listOf()) + loadStage(callback, _techs, ext2files["tech"] ?: listOf()) pathStack.block("/") { //playerDefinition = gson.fromJson(locate("/player.config").reader(), PlayerDefinition::class.java) @@ -452,4 +570,17 @@ class Starbound : ISBFileLocator { } } } + + companion object { + /** + * Глобальный [Interner] для [String] + * + * Так как нет смысла иметь множество [Interner]'ов для потенциального "больше одного" [Starbound], + * данный [Interner] доступен глобально + */ + @JvmField + val STRINGS: Interner = Interners.newBuilder().weak().concurrencyLevel(8).build() + + private val polyfill by lazy { loadInternalScript("polyfill") } + } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/api/IStarboundFileSystem.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/api/IStarboundFileSystem.kt index 99934c09..4fb9d48d 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/api/IStarboundFileSystem.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/api/IStarboundFileSystem.kt @@ -37,12 +37,6 @@ fun interface ISBFileLocator { * @throws FileNotFoundException if file does not exist */ fun reader(path: String) = locate(path).reader() - - /** - * @throws IllegalStateException if file is a directory - * @throws FileNotFoundException if file does not exist - */ - fun readJson(path: String) = locate(path).readJson() } interface IStarboundFile : ISBFileLocator { @@ -144,13 +138,7 @@ interface IStarboundFile : ISBFileLocator { * @throws IllegalStateException if file is a directory * @throws FileNotFoundException if file does not exist */ - fun reader(): Reader = InputStreamReader(open()) - - /** - * @throws IllegalStateException if file is a directory - * @throws FileNotFoundException if file does not exist - */ - fun readJson(): JsonElement = JsonParser.parseReader(reader()) + fun reader(): Reader = InputStreamReader(BufferedInputStream(open())) /** * @throws IllegalStateException if file is a directory @@ -163,6 +151,17 @@ interface IStarboundFile : ISBFileLocator { return ByteBuffer.wrap(read) } + /** + * @throws IllegalStateException if file is a directory + * @throws FileNotFoundException if file does not exist + */ + fun readToString(): String { + val stream = open() + val read = stream.readAllBytes() + stream.close() + return String(read, charset = Charsets.UTF_8) + } + /** * @throws IllegalStateException if file is a directory * @throws FileNotFoundException if file does not exist diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/FrameAnimator.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/FrameAnimator.kt index 6598d422..c4e30871 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/FrameAnimator.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/client/render/FrameAnimator.kt @@ -2,6 +2,7 @@ package ru.dbotthepony.kstarbound.client.render import ru.dbotthepony.kstarbound.util.ITimeSource import ru.dbotthepony.kstarbound.util.JVMTimeSource +import ru.dbotthepony.kstarbound.util.sbIntern2 /** * Таймер для анимирования набора спрайтов @@ -86,7 +87,7 @@ class FrameAnimator( init { for (i in 0 .. 500) { - framenames.add(i.toString()) + framenames.add(i.toString().sbIntern2()) } } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryFilterConfig.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/BagFilterConfig.kt similarity index 57% rename from src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryFilterConfig.kt rename to src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/BagFilterConfig.kt index 815b561b..aa4f5443 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryFilterConfig.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/BagFilterConfig.kt @@ -2,19 +2,24 @@ package ru.dbotthepony.kstarbound.defs.player import com.google.common.collect.ImmutableSet import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory +import ru.dbotthepony.kstarbound.util.ItemDescriptor +import java.util.function.Predicate @JsonFactory -data class InventoryFilterConfig( +data class BagFilterConfig( val typeBlacklist: ImmutableSet? = null, val typeWhitelist: ImmutableSet? = null, val tagBlacklist: ImmutableSet? = null, val categoryBlacklist: ImmutableSet? = null, -) { - fun acceptsType(type: String): Boolean { +) : Predicate { + override fun test(t: ItemDescriptor): Boolean { + if (t.isEmpty) + return false + if (typeBlacklist != null) { - return !typeBlacklist.contains(type) + return !typeBlacklist.contains(t.item!!.value.category) } else if (typeWhitelist != null) { - return typeWhitelist.contains(type) + return typeWhitelist.contains(t.item!!.value.category) } return true diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryConfig.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryConfig.kt index e557fa73..1a32d35b 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryConfig.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/InventoryConfig.kt @@ -2,17 +2,26 @@ package ru.dbotthepony.kstarbound.defs.player import com.google.common.collect.ImmutableMap import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory +import ru.dbotthepony.kstarbound.util.WriteOnce @JsonFactory data class InventoryConfig( val customBarGroups: Int, val customBarIndexes: Int, - val itemBags: ImmutableMap, + val itemBags: ImmutableMap, ) { @JsonFactory - data class BagConfig( + class Bag( val priority: Int, - val size: Int, - ) + val size: Int + ) { + var name: String by WriteOnce() + } + + init { + for ((k, v) in itemBags) { + v.name = k + } + } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/PlayerDefinition.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/PlayerDefinition.kt index 6f7d16e0..cd103a00 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/PlayerDefinition.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/PlayerDefinition.kt @@ -73,5 +73,5 @@ data class PlayerDefinition( val genericScriptContexts: ImmutableMap, val inventory: InventoryConfig, - val inventoryFilters: ImmutableMap, + val inventoryFilters: ImmutableMap, ) diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/TechDefinition.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/TechDefinition.kt new file mode 100644 index 00000000..bd5fe913 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/player/TechDefinition.kt @@ -0,0 +1,15 @@ +package ru.dbotthepony.kstarbound.defs.player + +import com.google.common.collect.ImmutableList +import ru.dbotthepony.kstarbound.defs.DirectAssetReference +import ru.dbotthepony.kstarbound.defs.IScriptable +import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory + +@JsonFactory +data class TechDefinition( + val name: String, + val type: String, + val chipCost: Int, + override val scriptDelta: Int = 1, + override val scripts: ImmutableList +) : IScriptable diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/defs/quest/QuestTemplate.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/quest/QuestTemplate.kt new file mode 100644 index 00000000..9c4bb113 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/defs/quest/QuestTemplate.kt @@ -0,0 +1,17 @@ +package ru.dbotthepony.kstarbound.defs.quest + +import com.google.common.collect.ImmutableSet +import com.google.gson.JsonObject +import ru.dbotthepony.kstarbound.defs.DirectAssetReference +import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory + +@JsonFactory +data class QuestTemplate( + val id: String, + val prerequisites: ImmutableSet = ImmutableSet.of(), + val requiredItems: ImmutableSet = ImmutableSet.of(), + val script: DirectAssetReference, + val updateDelta: Int = 10, + val moneyRange: LongRange, + val scriptConfig: JsonObject = JsonObject() +) diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/io/Ext.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/io/Ext.kt index 97ae52ac..a9cf87c0 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/io/Ext.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/io/Ext.kt @@ -190,3 +190,17 @@ fun InputStream.readCString(): String { } }.toString(Charsets.UTF_8) } + +fun InputStream.readByteChar(): Char { + return read().toChar() +} + +fun InputStream.readHeader(header: String) { + for ((i, char) in header.withIndex()) { + val read = readByteChar() + + if (read != char) { + throw IllegalArgumentException("Malformed header at byte $i: expected $char (${char.code}) got $read (${char.code})") + } + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/io/json/KotlinAdapters.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/io/json/KotlinAdapters.kt new file mode 100644 index 00000000..171c513b --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/io/json/KotlinAdapters.kt @@ -0,0 +1,32 @@ +package ru.dbotthepony.kstarbound.io.json + +import com.google.gson.TypeAdapter +import com.google.gson.stream.JsonReader +import com.google.gson.stream.JsonToken +import com.google.gson.stream.JsonWriter + +object LongRangeAdapter : TypeAdapter() { + override fun write(out: JsonWriter, value: LongRange?) { + if (value == null) + out.nullValue() + else { + out.beginArray() + out.value(value.first) + out.value(value.last) + out.endArray() + } + } + + override fun read(`in`: JsonReader): LongRange? { + if (`in`.peek() == JsonToken.NULL) { + return null + } else { + `in`.beginArray() + val first = `in`.nextLong() + val last = `in`.nextLong() + `in`.endArray() + + return LongRange(first, last) + } + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Errors.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Errors.kt index 466139b5..ca5638b9 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Errors.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Errors.kt @@ -21,16 +21,6 @@ class LuaGCException(message: String? = null, cause: Throwable? = null) : Throwa class LuaException(message: String? = null, cause: Throwable? = null) : Throwable(message, cause) class LuaRuntimeException(message: String? = null, cause: Throwable? = null) : Throwable(message, cause) -fun throwLoadError(code: Int) { - when (code) { - LUA_OK -> {} - LUA_ERRSYNTAX -> throw InvalidLuaSyntaxException() - LUA_ERRMEM -> throw LuaMemoryAllocException() - // LUA_ERRGCMM -> throw LuaGCException() - else -> throw LuaException("Unknown Lua Loading error: $code") - } -} - fun throwPcallError(code: Int) { when (code) { LUA_OK -> {} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/lua/LuaState.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/LuaState.kt index 8dbd97da..472c0aa2 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/lua/LuaState.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/LuaState.kt @@ -1,5 +1,6 @@ package ru.dbotthepony.kstarbound.lua +import com.google.common.collect.Interner import com.google.gson.JsonArray import com.google.gson.JsonElement import com.google.gson.JsonNull @@ -11,87 +12,68 @@ import com.kenai.jffi.Closure import com.kenai.jffi.ClosureManager import com.kenai.jffi.MemoryIO import com.kenai.jffi.Type +import it.unimi.dsi.fastutil.ints.Int2ObjectAVLTreeMap import jnr.ffi.Memory import jnr.ffi.NativeType +import jnr.ffi.Pointer import org.apache.logging.log4j.LogManager import org.lwjgl.system.MemoryStack import org.lwjgl.system.MemoryUtil +import ru.dbotthepony.kstarbound.Starbound import ru.dbotthepony.kstarbound.io.json.InternedJsonElementAdapter import java.io.Closeable -import java.io.PrintWriter -import java.io.StringWriter import java.lang.ref.Cleaner +import java.lang.ref.WeakReference import java.nio.ByteBuffer import java.nio.ByteOrder import kotlin.system.exitProcess -private fun stringToBuffer(str: String): ByteBuffer { - val bytes = str.toByteArray(charset = Charsets.UTF_8) - val buf = ByteBuffer.allocateDirect(bytes.size) - buf.order(ByteOrder.nativeOrder()) - bytes.forEach(buf::put) - buf.position(0) - - return buf -} - @Suppress("unused") -class LuaState : Closeable { - private val pointer = LuaJNR.INSTANCE.luaL_newstate() ?: throw OutOfMemoryError("Unable to allocate new LuaState") - private val cleanable: Cleaner.Cleanable - private val sharedStringBufferPtr = MemoryIO.getInstance().allocateMemory(2L shl 16, false) +class LuaState private constructor(private val pointer: Pointer, val stringInterner: Interner = Starbound.STRINGS) : Closeable { + constructor(stringInterner: Interner = Starbound.STRINGS) : this(LuaJNR.INSTANCE.luaL_newstate() ?: throw OutOfMemoryError("Unable to allocate new LuaState"), stringInterner) { + val pointer = this.pointer + val panic = ClosureManager.getInstance().newClosure( + { + LOGGER.fatal("Engine Error: LuaState at $pointer has panicked!") + exitProcess(1) + }, - init { - if (sharedStringBufferPtr == 0L) { + CallContext.getCallContext(Type.SINT, arrayOf(Type.POINTER), CallingConvention.DEFAULT, false) + ) + + this.cleanable = CLEANER.register(this) { LuaJNR.INSTANCE.lua_close(pointer) - throw OutOfMemoryError("Unable to allocate new string shared buffer") + panic.dispose() } - val pointer = pointer - val sharedStringBufferPtr = sharedStringBufferPtr + panic.setAutoRelease(false) + LuaJNR.INSTANCE.lua_atpanic(pointer, panic.address) - cleanable = CLEANER.register(this) { - LuaJNR.INSTANCE.lua_close(pointer) - MemoryIO.getInstance().freeMemory(sharedStringBufferPtr) - } + LuaJNR.INSTANCE.luaopen_base(this.pointer) + this.storeGlobal("_G") + LuaJNR.INSTANCE.luaopen_table(this.pointer) + this.storeGlobal("table") + LuaJNR.INSTANCE.luaopen_coroutine(this.pointer) + this.storeGlobal("coroutine") + LuaJNR.INSTANCE.luaopen_string(this.pointer) + this.storeGlobal("string") + LuaJNR.INSTANCE.luaopen_math(this.pointer) + this.storeGlobal("math") + LuaJNR.INSTANCE.luaopen_utf8(this.pointer) + this.storeGlobal("utf8") + LuaJNR.INSTANCE.luaopen_debug(this.pointer) + this.storeGlobal("debug") } - private val panicHandler = ClosureManager.getInstance().newClosure( - { - LOGGER.fatal("${this@LuaState} at $pointer has panicked! This should be impossible!") - exitProcess(1) - }, - - CallContext.getCallContext(Type.SINT, arrayOf(Type.POINTER), CallingConvention.DEFAULT, false) - ) + private val thread = Thread.currentThread() + private var cleanable: Cleaner.Cleanable? = null override fun close() { - cleanable.clean() - } - - init { - LuaJNR.INSTANCE.lua_atpanic(pointer, panicHandler.address) - - LuaJNR.INSTANCE.luaopen_base(pointer) - storeGlobal("_G") - LuaJNR.INSTANCE.luaopen_package(pointer) - storeGlobal("package") - LuaJNR.INSTANCE.luaopen_table(pointer) - storeGlobal("table") - LuaJNR.INSTANCE.luaopen_coroutine(pointer) - storeGlobal("coroutine") - LuaJNR.INSTANCE.luaopen_string(pointer) - storeGlobal("string") - LuaJNR.INSTANCE.luaopen_math(pointer) - storeGlobal("math") - LuaJNR.INSTANCE.luaopen_utf8(pointer) - storeGlobal("utf8") - LuaJNR.INSTANCE.luaopen_debug(pointer) - storeGlobal("debug") + this.cleanable?.clean() } val stackTop: Int get() { - val value = LuaJNR.INSTANCE.lua_gettop(pointer) + val value = LuaJNR.INSTANCE.lua_gettop(this.pointer) check(value >= 0) { "Invalid stack top $value" } return value } @@ -103,24 +85,38 @@ class LuaState : Closeable { if (index >= 0) return index - return LuaJNR.INSTANCE.lua_absindex(pointer, index) + return LuaJNR.INSTANCE.lua_absindex(this.pointer, index) + } + + private fun throwLoadError(code: Int) { + when (code) { + LUA_OK -> {} + LUA_ERRSYNTAX -> throw InvalidLuaSyntaxException(this.popString()) + LUA_ERRMEM -> throw LuaMemoryAllocException() + // LUA_ERRGCMM -> throw LuaGCException() + else -> throw LuaException("Unknown Lua Loading error: $code") + } } fun load(code: String, chunkName: String = "main chunk") { - val buf = stringToBuffer(code) + val bytes = code.toByteArray(charset = Charsets.UTF_8) + val buf = ByteBuffer.allocateDirect(bytes.size) + buf.order(ByteOrder.nativeOrder()) + bytes.forEach(buf::put) + buf.position(0) val closure = ClosureManager.getInstance().newClosure( object : Closure { override fun invoke(buffer: Closure.Buffer) { - val remainingSize = LuaJNR.RUNTIME.memoryManager.newPointer(buffer.getAddress(2)) + val amountToRead = LuaJNR.RUNTIME.memoryManager.newPointer(buffer.getAddress(2)) if (buf.remaining() == 0) { - remainingSize.putLong(0L, 0L) + amountToRead.putLong(0L, 0L) buffer.setAddressReturn(0L) return } - remainingSize.putLongLong(0L, buf.remaining().toLong()) + amountToRead.putLongLong(0L, buf.remaining().toLong()) val p = MemoryUtil.memAddress(buf) buf.position(buf.remaining()) buffer.setAddressReturn(p) @@ -130,65 +126,77 @@ class LuaState : Closeable { CallContext.getCallContext(Type.POINTER, arrayOf(Type.POINTER, Type.ULONG_LONG, Type.POINTER), CallingConvention.DEFAULT, false) ) - throwLoadError(LuaJNR.INSTANCE.lua_load(pointer, closure.address, 0L, chunkName, "t")) + this.throwLoadError(LuaJNR.INSTANCE.lua_load(this.pointer, closure.address, 0L, chunkName, "t")) closure.dispose() } fun call(numArgs: Int = 0, numResults: Int = 0): Int { - val status = LuaJNR.INSTANCE.lua_pcallk(pointer, numArgs, numResults, 0, 0L, 0L) + val status = LuaJNR.INSTANCE.lua_pcallk(this.pointer, numArgs, numResults, 0, 0L, 0L) if (status == LUA_ERRRUN) { - throw LuaRuntimeException(getString()) + throw LuaRuntimeException(this.getString()) } return status } - fun getString(stackIndex: Int = -1, limit: Long = 2 shl 16): String? { - val len = Memory.allocateDirect(LuaJNR.RUNTIME, NativeType.SLONGLONG) - val p = LuaJNR.INSTANCE.lua_tolstring(pointer, absStackIndex(stackIndex), len) ?: return null - - if (len.getLong(0L) == 0L) { - return "" - } - - if (len.getLong(0L) >= limit) { - throw IllegalStateException("Unreasonably long Lua string: ${len.getLong(0L)}") - } - - val readBytes = ByteArray(len.getLong(0L).toInt()) - p.get(0L, readBytes, 0, readBytes.size) - return readBytes.toString(charset = Charsets.UTF_8) - } - - fun isCFunction(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_iscfunction(pointer, absStackIndex(stackIndex)) > 0 - fun isFunction(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.FUNCTION - fun isInteger(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isinteger(pointer, absStackIndex(stackIndex)) > 0 - fun isLightUserdata(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.LIGHTUSERDATA - fun isNil(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.NIL - fun isNone(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.NONE - fun isNoneOrNil(stackIndex: Int = -1): Boolean = typeAt(stackIndex).let { it == LuaType.NIL || it == LuaType.NONE } - fun isNumber(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isnumber(pointer, absStackIndex(stackIndex)) > 0 - fun isString(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isstring(pointer, absStackIndex(stackIndex)) > 0 - fun isTable(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.TABLE - fun isThread(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.THREAD - fun isUserdata(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isuserdata(pointer, absStackIndex(stackIndex)) > 0 - fun isBoolean(stackIndex: Int = -1): Boolean = typeAt(stackIndex) == LuaType.BOOLEAN - - fun getBoolean(stackIndex: Int = -1): Boolean? { - if (!isBoolean(stackIndex)) + fun getString(stackIndex: Int = -1, limit: Long = DEFAULT_STRING_LIMIT): String? { + if (!this.isString(stackIndex)) return null - return LuaJNR.INSTANCE.lua_toboolean(pointer, stackIndex) > 0 + return getStringRaw(stackIndex, limit) + } + + private fun getStringRaw(stackIndex: Int = -1, limit: Long = DEFAULT_STRING_LIMIT): String? { + check(limit <= Int.MAX_VALUE) { "Can't allocate string bigger than ${Int.MAX_VALUE} characters" } + val stack = MemoryStack.stackPush() + val status = stack.mallocLong(1) + val p = LuaJNR.INSTANCE.lua_tolstring(this.pointer, this.absStackIndex(stackIndex), MemoryUtil.memAddress(status)) ?: return null + val len = status[0] + stack.close() + + if (len == 0L) + return "" + else if (len >= limit) + throw IllegalStateException("Unreasonably long Lua string: $len") + + val readBytes = ByteArray(len.toInt()) + p.get(0L, readBytes, 0, readBytes.size) + return this.stringInterner.intern(readBytes.toString(charset = Charsets.UTF_8)) + } + + fun isCFunction(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_iscfunction(this.pointer, this.absStackIndex(stackIndex)) > 0 + fun isFunction(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.FUNCTION + fun isInteger(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isinteger(this.pointer, this.absStackIndex(stackIndex)) > 0 + fun isLightUserdata(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.LIGHTUSERDATA + fun isNil(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.NIL + fun isNone(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.NONE + fun isNoneOrNil(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex).let { it == LuaType.NIL || it == LuaType.NONE } + fun isNumber(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isnumber(this.pointer, this.absStackIndex(stackIndex)) > 0 + fun isString(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isstring(this.pointer, this.absStackIndex(stackIndex)) > 0 + fun isTable(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.TABLE + fun isThread(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.THREAD + fun isUserdata(stackIndex: Int = -1): Boolean = LuaJNR.INSTANCE.lua_isuserdata(this.pointer, this.absStackIndex(stackIndex)) > 0 + fun isBoolean(stackIndex: Int = -1): Boolean = this.typeAt(stackIndex) == LuaType.BOOLEAN + + fun getBoolean(stackIndex: Int = -1): Boolean? { + if (!this.isBoolean(stackIndex)) + return null + + return LuaJNR.INSTANCE.lua_toboolean(this.pointer, stackIndex) > 0 + } + + private fun getBooleanRaw(stackIndex: Int = -1): Boolean { + return LuaJNR.INSTANCE.lua_toboolean(this.pointer, stackIndex) > 0 } fun getLong(stackIndex: Int = -1): Long? { - if (!isInteger(stackIndex)) + if (!this.isNumber(stackIndex)) return null val stack = MemoryStack.stackPush() val status = stack.mallocInt(1) - val value = LuaJNR.INSTANCE.lua_tointegerx(pointer, stackIndex, MemoryUtil.memAddress(status)) + val value = LuaJNR.INSTANCE.lua_tointegerx(this.pointer, stackIndex, MemoryUtil.memAddress(status)) val b = status[0] > 0 stack.close() @@ -198,13 +206,23 @@ class LuaState : Closeable { return value } + private fun getLongRaw(stackIndex: Int = -1): Long { + val stack = MemoryStack.stackPush() + val status = stack.mallocInt(1) + val value = LuaJNR.INSTANCE.lua_tointegerx(this.pointer, stackIndex, MemoryUtil.memAddress(status)) + val b = status[0] > 0 + stack.close() + if (!b) throw NumberFormatException("Lua was unable to parse Long present on stack at ${this.absStackIndex(stackIndex)} ($stackIndex)") + return value + } + fun getDouble(stackIndex: Int = -1): Double? { - if (!isNumber(stackIndex)) + if (!this.isNumber(stackIndex)) return null val stack = MemoryStack.stackPush() val status = stack.mallocInt(1) - val value = LuaJNR.INSTANCE.lua_tonumberx(pointer, stackIndex, MemoryUtil.memAddress(status)) + val value = LuaJNR.INSTANCE.lua_tonumberx(this.pointer, stackIndex, MemoryUtil.memAddress(status)) val b = status[0] > 0 stack.close() @@ -214,8 +232,18 @@ class LuaState : Closeable { return value } + private fun getDoubleRaw(stackIndex: Int = -1): Double { + val stack = MemoryStack.stackPush() + val status = stack.mallocInt(1) + val value = LuaJNR.INSTANCE.lua_tonumberx(this.pointer, stackIndex, MemoryUtil.memAddress(status)) + val b = status[0] > 0 + stack.close() + if (!b) throw NumberFormatException("Lua was unable to parse Double present on stack at ${this.absStackIndex(stackIndex)} ($stackIndex)") + return value + } + fun typeAt(stackIndex: Int = -1): LuaType { - return when (val value = LuaJNR.INSTANCE.lua_type(pointer, stackIndex)) { + return when (val value = LuaJNR.INSTANCE.lua_type(this.pointer, stackIndex)) { LUA_TNONE -> LuaType.NONE LUA_TNIL -> LuaType.NIL LUA_TBOOLEAN -> LuaType.BOOLEAN @@ -231,153 +259,464 @@ class LuaState : Closeable { } } - fun getValue(stackIndex: Int = -1): JsonElement? { - val abs = absStackIndex(stackIndex) + fun getValue(stackIndex: Int = -1, limit: Long = DEFAULT_STRING_LIMIT): JsonElement? { + val abs = this.absStackIndex(stackIndex) if (abs == 0) return null - return when (typeAt(abs)) { + return when (this.typeAt(abs)) { LuaType.NONE -> null - LuaType.NIL -> JsonNull.INSTANCE - LuaType.BOOLEAN -> InternedJsonElementAdapter.of(getBoolean(abs)!!) - LuaType.LIGHTUSERDATA -> null - LuaType.NUMBER -> JsonPrimitive(if (isInteger(abs)) getLong(abs)!! else getDouble(abs)!!) - LuaType.STRING -> JsonPrimitive(getString(abs)) - LuaType.TABLE -> getTable(abs)!! - LuaType.FUNCTION -> null - LuaType.USERDATA -> null - LuaType.THREAD -> null - LuaType.UMTYPES -> null + LuaType.NIL -> null // JsonNull.INSTANCE + LuaType.BOOLEAN -> InternedJsonElementAdapter.of(this.getBooleanRaw(abs)) + LuaType.LIGHTUSERDATA -> throw IllegalArgumentException("Can not get light userdata from Lua stack at $abs") + LuaType.NUMBER -> JsonPrimitive(if (this.isInteger(abs)) this.getLongRaw(abs) else this.getDoubleRaw(abs)) + LuaType.STRING -> JsonPrimitive(this.getStringRaw(abs, limit = limit)) + LuaType.TABLE -> this.getTableRaw(abs) + LuaType.FUNCTION -> throw IllegalArgumentException("Can not get function from Lua stack at $abs") + LuaType.USERDATA -> throw IllegalArgumentException("Can not get userdata from Lua stack at $abs") + LuaType.THREAD -> throw IllegalArgumentException("Can not get thread from Lua stack at $abs") + LuaType.UMTYPES -> throw IllegalArgumentException("Can not get umtypes from Lua stack at $abs") } } - fun getTable(stackIndex: Int = -1): JsonObject? { - val abs = absStackIndex(stackIndex) + fun getTable(stackIndex: Int = -1, limit: Long = DEFAULT_STRING_LIMIT): JsonObject? { + val abs = this.absStackIndex(stackIndex) - if (!isTable(abs)) + if (!this.isTable(abs)) return null - val pairs = JsonObject() - push() + return getTableRaw(abs, limit) + } - while (LuaJNR.INSTANCE.lua_next(pointer, abs) != 0) { - val key = getValue(abs + 1) - val value = getValue(abs + 2) + private fun getTableRaw(abs: Int, limit: Long = DEFAULT_STRING_LIMIT): JsonObject { + val pairs = JsonObject() + this.push() + val top = this.stackTop + + while (LuaJNR.INSTANCE.lua_next(this.pointer, abs) != 0) { + val key = this.getValue(abs + 1, limit = limit) + val value = this.getValue(abs + 2, limit = limit) if (key is JsonPrimitive && value != null) { - pairs.add(key.asString, value) + pairs.add(this.stringInterner.intern(key.asString), value) } - pop() + LuaJNR.INSTANCE.lua_settop(this.pointer, top) } return pairs } + /** + * Пропуски заполняются [JsonNull.INSTANCE] + * + * Не числовые индексы игнорируются + */ + fun getArray(stackIndex: Int = -1, limit: Long = DEFAULT_STRING_LIMIT): JsonArray? { + val abs = this.absStackIndex(stackIndex) + + if (!this.isTable(abs)) + return null + + val pairs = Int2ObjectAVLTreeMap() + this.push() + + while (LuaJNR.INSTANCE.lua_next(this.pointer, abs) != 0) { + val key = this.getValue(abs + 1, limit = limit) + val value = this.getValue(abs + 2, limit = limit) + + if (key is JsonPrimitive && key.isNumber && value != null) { + pairs.put(key.asInt, value) + } + + this.pop() + } + + val result = JsonArray() + + for ((index, value) in pairs) { + while (index > result.size()) { + result.add(JsonNull.INSTANCE) + } + + result.add(value) + } + + return result + } + + fun getTableValue(stackIndex: Int = -2, limit: Long = DEFAULT_STRING_LIMIT): JsonElement? { + this.loadTableValue(stackIndex) + return this.getValue(limit = limit) + } + + fun loadTableValue(stackIndex: Int = -2) { + val abs = this.absStackIndex(stackIndex) + + if (!this.isTable(abs)) + throw IllegalArgumentException("Attempt to index an ${this.typeAt(abs)} value") + + if (LuaJNR.INSTANCE.lua_gettable(this.pointer, abs) == LUA_TNONE) + throw IllegalStateException("loaded TNONE from Lua table") + } + + fun loadTableValue(name: String, stackIndex: Int = -2) { + this.push(name) + this.loadTableValue(stackIndex) + } + fun popBoolean(): Boolean? { try { - return getBoolean() + return this.getBoolean() } finally { - pop() + this.pop() } } fun popLong(): Long? { try { - return getLong() + return this.getLong() } finally { - pop() + this.pop() } } fun popDouble(): Double? { try { - return getDouble() + return this.getDouble() } finally { - pop() + this.pop() } } fun popValue(): JsonElement? { try { - return getValue() + return this.getValue() } finally { - pop() + this.pop() } } fun popTable(): JsonObject? { try { - return getTable() + return this.getTable() } finally { - pop() + this.pop() + } + } + + fun popString(limit: Long = DEFAULT_STRING_LIMIT): String? { + try { + return this.getString(limit = limit) + } finally { + this.pop() } } fun pop(amount: Int = 1): Int { if (amount == 0) return 0 check(amount > 0) { "Invalid amount to pop: $amount" } - val old = stackTop + val old = this.stackTop val new = (old - amount).coerceAtLeast(0) - LuaJNR.INSTANCE.lua_settop(pointer, new) + LuaJNR.INSTANCE.lua_settop(this.pointer, new) return old - new } fun storeGlobal(name: String) { - LuaJNR.INSTANCE.lua_setglobal(pointer, name) + LuaJNR.INSTANCE.lua_setglobal(this.pointer, name) } fun loadGlobal(name: String) { - LuaJNR.INSTANCE.lua_getglobal(pointer, name) + LuaJNR.INSTANCE.lua_getglobal(this.pointer, name) } - fun push(closure: (state: LuaState) -> Unit) { - LuaJNI.lua_pushcclosure(pointer.address()) lazy@{ - try { - closure.invoke(this@LuaState) - } catch (err: Throwable) { - val builder = StringWriter() - val printWriter = PrintWriter(builder) - err.printStackTrace(printWriter) - push(builder.toString()) - return@lazy 1 - } + inner class ArgStack(val top: Int) { + val lua get() = this@LuaState + var position = 1 - return@lazy 0 + fun getString(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): String { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + return this@LuaState.getString(position, limit = limit) + ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: string expected, got ${this@LuaState.typeAt(position)}") + } + + fun getLong(position: Int = this.position++): Long { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + return this@LuaState.getLong(position) + ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: long expected, got ${this@LuaState.typeAt(position)}") + } + + fun getStringOrNil(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): String? { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + + val type = this@LuaState.typeAt(position) + + if (type != LuaType.STRING && type != LuaType.NIL) + throw IllegalArgumentException("Lua code error: Bad argument #$position: string expected, got $type") + + return this@LuaState.getString(position, limit = limit) + } + + fun getStringOrNull(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): String? { + if (position > this.top) + return null + + return this.getStringOrNil(position, limit = limit) + } + + fun getValue(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): JsonElement { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + val value = this@LuaState.getValue(position, limit = limit) + return value ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: anything expected, got ${this@LuaState.typeAt(position)}") + } + + fun getTable(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): JsonObject { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + val value = this@LuaState.getTable(position, limit = limit) + return value ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: table expected, got ${this@LuaState.typeAt(position)}") + } + + fun getArray(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): JsonArray { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + val value = this@LuaState.getArray(position, limit = limit) + return value ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: table expected, got ${this@LuaState.typeAt(position)}") + } + + fun getAnything(position: Int = this.position++, limit: Long = DEFAULT_STRING_LIMIT): JsonElement? { + check(position in 1..this.top) { "JVM code error: Invalid argument position: $position" } + return this@LuaState.getValue(position, limit = limit) + } + + fun getDoubleOrNil(position: Int = this.position++): Double? { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + + val type = this@LuaState.typeAt(position) + + if (type != LuaType.NUMBER && type != LuaType.NIL) + throw IllegalArgumentException("Lua code error: Bad argument #$position: double expected, got $type") + + return this@LuaState.getDouble(position) + } + + fun getDouble(position: Int = this.position++): Double { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + return this@LuaState.getDouble(position) + ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: double expected, got ${this@LuaState.typeAt(position)}") + } + + fun getBoolOrNil(position: Int = this.position++): Boolean? { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + + val type = this@LuaState.typeAt(position) + + if (type != LuaType.BOOLEAN && type != LuaType.NIL) + throw IllegalArgumentException("Lua code error: Bad argument #$position: boolean expected, got $type") + + return this@LuaState.getBoolean(position) + } + + fun getBool(position: Int = this.position++): Boolean { + check(position in 1 ..this.top) { "JVM code error: Invalid argument position: $position" } + return this@LuaState.getBoolean(position) + ?: throw IllegalArgumentException("Lua code error: Bad argument #$position: boolean expected, got ${this@LuaState.typeAt(position)}") + } + + fun getBoolOrNull(position: Int = this.position++): Boolean? { + if (position > this.top) return null + return this.getBoolOrNil(position) } } + /** + * Создаёт новое замыкание на стороне Lua. [function], будучи переданным в Lua, + * создаст новый **GC Root**. + * + * Вышестоящий код ОБЯЗАН использовать [ArgStack] и его [ArgStack.lua] для доступа к [LuaState], так как + * при вызове замыкания из Lua текущий [LuaState] может отличаться от того, которому был передан [function]. + */ + fun push(function: ArgStack.() -> Int, performanceCritical: Boolean) { + val weak = WeakReference(this) + val pointer = this.pointer + + LuaJNI.lua_pushcclosure(pointer.address()) lazy@{ + var realLuaState = weak.get() + + if (realLuaState == null || realLuaState.pointer.address() != it) { + if (realLuaState == null) + realLuaState = LuaState(LuaJNR.RUNTIME.memoryManager.newPointer(it)) + else + realLuaState = LuaState(LuaJNR.RUNTIME.memoryManager.newPointer(it), stringInterner = realLuaState.stringInterner) + } + + val args = realLuaState.ArgStack(realLuaState.stackTop) + val rememberStack: ArrayList? + + if (performanceCritical) { + rememberStack = null + } else { + rememberStack = ArrayList(Exception().stackTraceToString().split('\n')) + + rememberStack.removeAt(0) // java.lang. ... + // rememberStack.removeAt(0) // at ... push( ... ) + } + + try { + val value = function.invoke(args) + check(value >= 0) { "Internal JVM error: ${function::class.qualifiedName} returned incorrect number of arguments to be popped from stack by Lua" } + return@lazy value + } catch (err: Throwable) { + try { + if (performanceCritical) { + realLuaState.push(err.stackTraceToString()) + return@lazy -1 + } else { + rememberStack!! + val newStack = err.stackTraceToString().split('\n').toMutableList() + + val rememberIterator = rememberStack.listIterator(rememberStack.size) + val iterator = newStack.listIterator(newStack.size) + var hit = false + + while (rememberIterator.hasPrevious() && iterator.hasPrevious()) { + val a = rememberIterator.previous() + val b = iterator.previous() + + if (a == b) { + hit = true + iterator.remove() + } else { + break + } + } + + if (hit) { + newStack[newStack.size - 1] = "\t<...>" + } + + realLuaState.push(newStack.joinToString("\n")) + return@lazy -1 + } + } catch(err2: Throwable) { + realLuaState.push("JVM suffered an exception while handling earlier exception: ${err2.stackTraceToString()}; earlier: ${err.stackTraceToString()}") + return@lazy -1 + } + } + } + } + + /** + * Создаёт новое замыкание на стороне Lua. [function], будучи переданным в Lua, + * создаст новый **GC Root**, что может создать утечку память если будет создана циклическая зависимость, + * если быть неаккуратным. + * + * Пример: Parent -> Lua -> Closure -> Parent + * + * В примере выше Lua никогда не будет собран сборщиком мусора, тем самым никогда не будет (автоматически) вызван [close] + * + * Во избежание данной ситуации предоставлена функция [pushWeak] + * + * @see CClosure.invoke + */ + fun push(function: ArgStack.() -> Int) = this.push(function, !RECORD_STACK_TRACES) + + /** + * Создаёт новое замыкание на стороне Lua. [function], будучи переданным в Lua, + * создаст новый **GC Root**, что может создать утечку память если будет создана циклическая зависимость, + * если быть неаккуратным. + * + * В отличие от обычного [push] для замыканий, данный вариант создаёт [WeakReference] на [self], + * который, в свою очередь, может ссылаться обратно на данный [LuaState] через свои структуры. + * + * В силу того, что замыкание более не может создать циклическую ссылку на данный [LuaState] через [self], сборщик + * мусора сможет удалить [self], а затем удалить [LuaState]. + * + */ + fun pushWeak(self: T, function: T.(args: ArgStack) -> Int, performanceCritical: Boolean) { + val weakSelf = WeakReference(self) + + return push(performanceCritical = performanceCritical, function = lazy@{ + @Suppress("name_shadowing") + val self = weakSelf.get() + + if (self == null) { + this.lua.push("Referenced 'this' got reclaimed by JVM GC") + return@lazy -1 + } + + function.invoke(self, this) + }) + } + + /** + * Создаёт новое замыкание на стороне Lua. [function], будучи переданным в Lua, + * создаст новый **GC Root**, что может создать утечку память если будет создана циклическая зависимость, + * если быть неаккуратным. + * + * В отличие от обычного [push] для замыканий, данный вариант создаёт [WeakReference] на [self], + * который, в свою очередь, может ссылаться обратно на данный [LuaState] через свои структуры. + * + * В силу того, что замыкание более не может создать циклическую ссылку на данный [LuaState] через [self], сборщик + * мусора сможет удалить [self], а затем удалить [LuaState]. + * + */ + fun pushWeak(self: T, function: T.(args: ArgStack) -> Int) = this.pushWeak(self, function, !RECORD_STACK_TRACES) + fun push() { - LuaJNR.INSTANCE.lua_pushnil(pointer) + LuaJNR.INSTANCE.lua_pushnil(this.pointer) } fun push(value: Int) { - LuaJNR.INSTANCE.lua_pushinteger(pointer, value.toLong()) + LuaJNR.INSTANCE.lua_pushinteger(this.pointer, value.toLong()) } fun push(value: Long) { - LuaJNR.INSTANCE.lua_pushinteger(pointer, value) + LuaJNR.INSTANCE.lua_pushinteger(this.pointer, value) } fun push(value: Double) { - LuaJNR.INSTANCE.lua_pushnumber(pointer, value) + LuaJNR.INSTANCE.lua_pushnumber(this.pointer, value) } fun push(value: Float) { - LuaJNR.INSTANCE.lua_pushnumber(pointer, value.toDouble()) + LuaJNR.INSTANCE.lua_pushnumber(this.pointer, value.toDouble()) } fun push(value: Boolean) { - LuaJNR.INSTANCE.lua_pushboolean(pointer, if (value) 1 else 0) + LuaJNR.INSTANCE.lua_pushboolean(this.pointer, if (value) 1 else 0) } - fun push(value: String) { + fun pushStrings(strings: Iterable) { + val index = this.pushTable(arraySize = if (strings is Collection) strings.size else 0) + + for ((i, v) in strings.withIndex()) { + this.push(i + 1L) + this.push(v) + this.setTableValue(index) + } + } + + fun pushStrings(strings: Iterator) { + val index = this.pushTable() + + for ((i, v) in strings.withIndex()) { + this.push(i + 1L) + this.push(v) + this.setTableValue(index) + } + } + + fun push(value: String?) { + if (value == null) { + this.push() + return + } + val bytes = value.toByteArray(Charsets.UTF_8) if (bytes.size < 2 shl 16) { MemoryIO.getInstance().putByteArray(sharedStringBufferPtr, bytes, 0, bytes.size) - LuaJNR.INSTANCE.lua_pushlstring(pointer, sharedStringBufferPtr, bytes.size.toLong()) + LuaJNR.INSTANCE.lua_pushlstring(this.pointer, sharedStringBufferPtr, bytes.size.toLong()) } else { val mem = MemoryIO.getInstance() val block = mem.allocateMemory(bytes.size.toLong(), false) @@ -387,7 +726,7 @@ class LuaState : Closeable { try { mem.putByteArray(block, bytes, 0, bytes.size) - LuaJNR.INSTANCE.lua_pushlstring(pointer, block, bytes.size.toLong()) + LuaJNR.INSTANCE.lua_pushlstring(this.pointer, block, bytes.size.toLong()) } finally { mem.freeMemory(block) } @@ -395,18 +734,158 @@ class LuaState : Closeable { } fun pushTable(arraySize: Int = 0, hashSize: Int = 0): Int { - LuaJNR.INSTANCE.lua_createtable(pointer, arraySize, hashSize) - return stackTop + LuaJNR.INSTANCE.lua_createtable(this.pointer, arraySize, hashSize) + return this.stackTop } fun setTableValue(stackIndex: Int) { - LuaJNR.INSTANCE.lua_settable(pointer, stackIndex) + LuaJNR.INSTANCE.lua_settable(this.pointer, stackIndex) } - fun push(value: JsonElement) { + fun setTableFunction(key: String, self: T, value: T.(args: ArgStack) -> Int) { + val table = this.stackTop + this.push(key) + this.pushWeak(self, value) + this.setTableValue(table) + } + + fun setTableClosure(key: String, self: T, value: T.(args: ArgStack) -> Unit) { + val table = this.stackTop + this.push(key) + this.pushWeak(self) { value.invoke(this, it); 0 } + this.setTableValue(table) + } + + fun setTableValue(key: String, value: JsonElement?) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: String, value: Int) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: String, value: Long) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: String, value: String) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: String, value: Float) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: String, value: Double) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: JsonElement?) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: Int) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: Long) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: String) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: Float) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Int, value: Double) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: JsonElement?) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: Int) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: Long) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: String) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: Float) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun setTableValue(key: Long, value: Double) { + val table = this.stackTop + this.push(key) + this.push(value) + this.setTableValue(table) + } + + fun push(value: JsonElement?) { when (value) { - JsonNull.INSTANCE -> { - push() + null, JsonNull.INSTANCE -> { + this.push() } is JsonPrimitive -> { @@ -414,37 +893,37 @@ class LuaState : Closeable { val num = value.asNumber when (num) { - is Int, is Long -> push(num.toLong()) - else -> push(num.toDouble()) + is Int, is Long -> this.push(num.toLong()) + else -> this.push(num.toDouble()) } } else if (value.isString) { - push(value.asString) + this.push(value.asString) } else if (value.isBoolean) { - push(value.asBoolean) + this.push(value.asBoolean) } else { throw IllegalArgumentException(value.toString()) } } is JsonArray -> { - val index = pushTable(arraySize = value.size()) + val index = this.pushTable(arraySize = value.size()) for ((i, v) in value.withIndex()) { - push(i + 1L) - push(v) + this.push(i + 1L) + this.push(v) - setTableValue(index) + this.setTableValue(index) } } is JsonObject -> { - val index = pushTable(hashSize = value.size()) + val index = this.pushTable(hashSize = value.size()) for ((k, v) in value.entrySet()) { - push(k) - push(v) + this.push(k) + this.push(v) - setTableValue(index) + this.setTableValue(index) } } @@ -456,12 +935,35 @@ class LuaState : Closeable { companion object { private val LOGGER = LogManager.getLogger() - private val CLEANER = Cleaner.create { - val thread = Thread(it, "LuaState cleaner") + + private val CLEANER: Cleaner = Cleaner.create { + val thread = Thread(it, "Lua cleaner") thread.priority = 1 thread } + private val sharedBuffers = ThreadLocal() + + private val sharedStringBufferPtr: Long get() { + var p: Long? = sharedBuffers.get() + + if (p == null) { + p = MemoryIO.getInstance().allocateMemory(DEFAULT_STRING_LIMIT, false) + + if (p == 0L) { + throw OutOfMemoryError("Unable to allocate new string shared buffer") + } + + sharedBuffers.set(p) + val p2 = p + + CLEANER.register(Thread.currentThread()) { + MemoryIO.getInstance().freeMemory(p2) + } + } + + return p + } const val LUA_TNONE = -1 @@ -476,5 +978,11 @@ class LuaState : Closeable { const val LUA_TTHREAD = 8 const val LUA_NUMTYPES = 9 + + const val CHUNK_READ_SIZE = 2L shl 10 + + const val DEFAULT_STRING_LIMIT = 2L shl 16 + + const val RECORD_STACK_TRACES = false } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Scripts.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Scripts.kt new file mode 100644 index 00000000..f2389141 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/lua/Scripts.kt @@ -0,0 +1,66 @@ +package ru.dbotthepony.kstarbound.lua + +import com.google.gson.JsonElement +import com.google.gson.JsonNull +import it.unimi.dsi.fastutil.objects.ObjectOpenHashSet +import ru.dbotthepony.kstarbound.util.traverseJsonPath + +fun loadInternalScript(name: String): String { + return LuaState::class.java.getResourceAsStream("/scripts/$name.lua")?.readAllBytes()?.toString(Charsets.UTF_8) ?: throw RuntimeException("/scripts/$name.lua is missing!") +} + +private val messageHandlerLua by lazy { loadInternalScript("message_handler") } +private val configLua by lazy { loadInternalScript("config") } + +fun LuaState.exposeConfig(config: JsonElement) { + load(configLua, "starbound.jar!/scripts/config.lua") + call() + + loadGlobal("config") + val table = stackTop + push("_get") + push { + push(traverseJsonPath(getString(), config)) + return@push 1 + } + + setTableValue(table) + pop() +} + +class MessageHandler(val lua: LuaState) { + init { + lua.load(messageHandlerLua, "starbound.jar!/scripts/message_handler.lua") + lua.call() + + lua.loadGlobal("message") + + lua.setTableClosure("subscribe", this) { subscriptions.add(it.getString()) } + + lua.pop() + } + + private val subscriptions = ObjectOpenHashSet() + + fun isSubscribed(name: String): Boolean { + return name in subscriptions + } + + fun call(name: String, vararg values: JsonElement): JsonElement? { + if (!isSubscribed(name)) + return null + + lua.loadGlobal("message") + lua.loadTableValue("call") + + lua.push(name) + + for (value in values) + lua.push(value) + + lua.call(numArgs = 1 + values.size, numResults = 1) + val value = lua.getValue() + lua.pop() // message + return value + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/player/Avatar.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/player/Avatar.kt new file mode 100644 index 00000000..2cb933c2 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/player/Avatar.kt @@ -0,0 +1,492 @@ +package ru.dbotthepony.kstarbound.player + +import com.google.gson.JsonElement +import com.google.gson.JsonObject +import com.google.gson.JsonPrimitive +import it.unimi.dsi.fastutil.objects.Object2LongOpenHashMap +import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap +import it.unimi.dsi.fastutil.objects.ObjectOpenHashSet +import ru.dbotthepony.kstarbound.RegistryObject +import ru.dbotthepony.kstarbound.Starbound +import ru.dbotthepony.kstarbound.defs.player.TechDefinition +import ru.dbotthepony.kstarbound.lua.LuaState +import ru.dbotthepony.kstarbound.lua.loadInternalScript +import ru.dbotthepony.kstarbound.util.ItemDescriptor +import ru.dbotthepony.kstarbound.util.immutableMap +import java.util.* +import kotlin.collections.ArrayList + +/** + * Персонаж - как он есть. + * + * [Avatar] реализует Lua интерфейс `player`. + */ +class Avatar(val starbound: Starbound, val uniqueId: UUID) { + enum class EssentialSlot { + BEAM_AXE, + WIRE_TOOL, + PAINT_TOOL, + INSPECTION_TOOL; + } + + enum class EquipmentSlot { + HEAD, + CHEST, + LEGS, + BACK, + HEAD_COSMETIC, + CHEST_COSMETIC, + LEGS_COSMETIC, + BACK_COSMETIC; + } + + private val essentialSlots = EnumMap(EssentialSlot::class.java) + private val equipmentSlots = EnumMap(EquipmentSlot::class.java) + private val bags = ArrayList() + private val quests = Object2ObjectOpenHashMap() + + var cursorItem = ItemDescriptor.EMPTY + + private val availableTechs = ObjectOpenHashSet>() + private val enabledTechs = ObjectOpenHashSet>() + private val equippedTechs = Object2ObjectOpenHashMap>() + + private val knownBlueprints = ObjectOpenHashSet() + // С подписью NEW + private val newBlueprints = ObjectOpenHashSet() + + private val currencies = Object2LongOpenHashMap() + + /** + * Teaches the player any recipes which can be used to craft the specified item. + */ + fun giveBlueprint(name: String): Boolean { + val item = starbound.item(name).conciseToNull() ?: return false + + if (knownBlueprints.add(item)) { + newBlueprints.add(item) + return true + } + + return false + } + + /** + * Returns `true` if the player knows one or more recipes to create the specified item and `false` otherwise. + */ + fun blueprintKnown(name: String): Boolean { + return (starbound.item(name).conciseToNull() ?: return false) in knownBlueprints + } + + /** + * Returns `true` if the player knows one or more recipes to create the specified item and `false` otherwise. + */ + private fun blueprintKnown(name: JsonElement): Boolean { + if (name is JsonPrimitive) { + return (starbound.item(name.asString).conciseToNull() ?: return false) in knownBlueprints + } else if (name is JsonObject) { + return (starbound.item(name).conciseToNull() ?: return false) in knownBlueprints + } else { + return false + } + } + + /** + * Teaches the player any recipes which can be used to craft the specified item. + */ + private fun giveBlueprint(name: JsonElement): Boolean { + val item: ItemDescriptor + + if (name is JsonPrimitive) { + item = starbound.item(name.asString).conciseToNull() ?: return false + } else if (name is JsonObject) { + item = starbound.item(name).conciseToNull() ?: return false + } else { + return false + } + + if (knownBlueprints.add(item)) { + newBlueprints.add(item) + return true + } + + return false + } + + /** + * Adds the specified tech to the player's list of available (unlockable) techs. + */ + fun makeTechAvailable(name: String): Boolean { + return availableTechs.add(starbound.techs[name] ?: return false) + } + + /** + * Removes the specified tech from player's list of available (unlockable) techs. + */ + fun makeTechUnavailable(name: String): Boolean { + val tech = starbound.techs[name] ?: return false + + if (availableTechs.remove(tech)) { + enabledTechs.remove(tech) + equippedTechs.remove(tech.value.type) + return true + } + + return false + } + + /** + * Unlocks the specified tech, allowing it to be equipped through the tech GUI. + */ + fun enableTech(name: String): Boolean { + val tech = starbound.techs[name] ?: return false + availableTechs.add(tech) + return enabledTechs.add(tech) + } + + /** + * Equips the specified tech. + */ + fun equipTech(name: String): Boolean { + val tech = starbound.techs[name] ?: return false + availableTechs.add(tech) + enabledTechs.add(tech) + return equippedTechs.put(tech.value.type, tech) != tech + } + + /** + * Unequips the specified tech. + */ + fun unequipTech(name: String): Boolean { + val tech = starbound.techs[name] ?: return false + return equippedTechs.remove(tech.value.type) == tech + } + + /** + * Returns the player's current total reserves of the specified currency. + */ + fun currency(name: String): Long { + return currencies.getLong(name) + } + + /** + * Increases the player's reserve of the specified currency by the specified amount. + */ + fun addCurrency(name: String, amount: Long) { + check(amount >= 0L) { "Negative amount of currency: $amount (currency: $name)" } + currencies.computeLong(name) { key, old -> (old ?: 0L) + amount } + } + + /** + * Attempts to consume the specified amount of the specified currency and returns `true` if successful and `false` otherwise. + */ + fun consumeCurrency(name: String, amount: Long): Boolean { + check(amount >= 0L) { "Negative amount of currency: $amount (currency: $name)" } + val current = currencies.getLong(name) + + if (current - amount >= 0L) { + currencies[name] = current - amount + return true + } + + return false + } + + /** + * Triggers an immediate cleanup of the player's inventory, removing item stacks with 0 quantity. May rarely be required in special cases of making several sequential modifications to the player's inventory within a single tick. + */ + fun cleanupItems() { + + } + + /** + * Adds the specified item to the player's inventory. + */ + fun giveItem(descriptor: ItemDescriptor) { + + } + + /** + * Returns `true` if the player's inventory contains an item matching the specified descriptor and `false` otherwise. If exactMatch is `true` then parameters as well as item name must match. + */ + fun hasItem(descriptor: ItemDescriptor, exactMatch: Boolean = false): Boolean { + return false + } + + /** + * Returns the total number of items in the player's inventory matching the specified descriptor. If exactMatch is `true` then parameters as well as item name must match. + */ + fun hasCountOfItem(descriptor: ItemDescriptor, exactMatch: Boolean = false): Long { + return 0L + } + + /** + * Attempts to consume the specified item from the player's inventory and returns the item consumed if successful. If consumePartial is `true`, matching stacks totalling fewer items than the requested count may be consumed, otherwise the operation will only be performed if the full count can be consumed. If exactMatch is `true` then parameters as well as item name must match. + */ + fun consumeItem(descriptor: ItemDescriptor, allowPartial: Boolean = false, exactMatch: Boolean = false): ItemDescriptor { + return ItemDescriptor.EMPTY + } + + fun inventoryTags(): Map { + return mapOf() + } + + fun itemsWithTag(): List { + return listOf() + } + + fun consumeTaggedItem(tag: String): Long { + return 0L + } + + fun hasItemWithParameter(name: String, value: JsonElement): Boolean { + return false + } + + fun consumeItemWithParameter(name: String, value: JsonElement, count: Long): Long { + return 0L + } + + fun getItemWithParameter(name: String, value: JsonElement): ItemDescriptor { + return ItemDescriptor.EMPTY + } + + var primaryHandItem: ItemDescriptor? = null + var altHandItem: ItemDescriptor? = null + + fun essentialItem(slotName: EssentialSlot): ItemDescriptor? { + return essentialSlots[slotName]?.conciseToNull() + } + + fun giveEssentialItem(slotName: EssentialSlot, item: ItemDescriptor) { + + } + + fun removeEssentialItem(slotName: EssentialSlot) { + + } + + fun equippedItem(slotName: EquipmentSlot): ItemDescriptor { + return equipmentSlots[slotName] ?: ItemDescriptor.EMPTY + } + + fun setEquippedItem(slotName: EquipmentSlot, item: ItemDescriptor) { + + } + + fun addQuest(quest: QuestInstance): QuestInstance? { + check(quest.avatar === this) { "$quest does not belong to $this" } + return quests.put(quest.id, quest) + } + + private fun startQuest(value: JsonElement, serverID: String?, worldID: String?): String { + if (value is JsonPrimitive) { + val quest = QuestInstance(this, descriptor = QuestDescriptor(value.asString), serverID = serverID?.let(UUID::fromString), worldID = worldID) + addQuest(quest) + return quest.id + } else if (value is JsonObject) { + val seed = value["seed"]?.asLong ?: QuestDescriptor.makeSeed() + val questId = value["questId"]?.asString ?: throw IllegalArgumentException("Invalid 'questId' in quest descriptor") + val templateId = value["templateId"]?.asString ?: questId + val params = value["parameters"] as? JsonObject ?: JsonObject() + val quest = QuestInstance(this, descriptor = QuestDescriptor(questId, templateId, seed, params), serverID = serverID?.let(UUID::fromString), worldID = worldID) + addQuest(quest) + return quest.id + } else { + throw IllegalArgumentException("Invalid quest descriptor: $value") + } + } + + fun expose(lua: LuaState) { + lua.load(playerLua) + + lua.pushTable() + + lua.setTableValue("uniqueId", uniqueId.toString()) + lua.setTableValue("species", "human") + lua.setTableValue("gender", "male") + + lua.call(numArgs = 1) + + lua.loadGlobal("player") + lua.setTableClosure("giveBlueprint", this) { giveBlueprint(it.getValue()) } + lua.setTableFunction("blueprintKnown", this) { it.lua.push(blueprintKnown(it.getValue())); 1 } + lua.setTableClosure("makeTechAvailable", this) { makeTechAvailable(it.getString()) } + lua.setTableClosure("makeTechUnavailable", this) { makeTechUnavailable(it.getString()) } + lua.setTableClosure("enableTech", this) { enableTech(it.getString()) } + lua.setTableClosure("equipTech", this) { equipTech(it.getString()) } + lua.setTableClosure("unequipTech", this) { unequipTech(it.getString()) } + lua.setTableFunction("availableTechs", this) { it.lua.pushStrings(availableTechs.stream().map { it.value.name }.iterator()); 1 } + lua.setTableFunction("enabledTechs", this) { it.lua.pushStrings(enabledTechs.stream().map { it.value.name }.iterator()); 1 } + lua.setTableFunction("equippedTech", this) { it.lua.push(equippedTechs[it.getString()]?.value?.name); 1 } + lua.setTableFunction("currency", this) { it.lua.push(currency(it.getString())); 1 } + lua.setTableClosure("addCurrency", this) { addCurrency(it.getString(), it.getLong()) } + lua.setTableFunction("consumeCurrency", this) { it.lua.push(consumeCurrency(it.getString(), it.getLong())); 1 } + lua.setTableClosure("cleanupItems", this) { cleanupItems() } + lua.setTableClosure("giveItem", this) { giveItem(starbound.item(it.getValue())) } + + lua.setTableFunction("hasItem", this) { + it.lua.push(hasItem(starbound.item(it.getValue()), it.getBoolOrNull() ?: false)) + 1 + } + + lua.setTableFunction("hasCountOfItem", this) { + it.lua.push(hasCountOfItem(starbound.item(it.getValue()), it.getBoolOrNull() ?: false)) + 1 + } + + lua.setTableFunction("consumeItem", this) { + it.lua.push(consumeItem(starbound.item(it.getValue()), it.getBoolOrNull() ?: false, it.getBoolOrNull() ?: false).toJson()) + 1 + } + + lua.setTableFunction("inventoryTags", this) { + val mapping = inventoryTags() + + it.lua.pushTable(hashSize = mapping.size) + + for ((k, v) in mapping) { + it.lua.setTableValue(k, v) + } + + 1 + } + + lua.setTableFunction("itemsWithTag", this) { + val mapping = itemsWithTag() + + it.lua.pushTable(arraySize = mapping.size) + + for ((k, v) in mapping.withIndex()) { + it.lua.setTableValue(k, v.toJson()) + } + + 1 + } + + lua.setTableFunction("consumeTaggedItem", this) { + it.lua.push(consumeTaggedItem(it.getString())) + 1 + } + + lua.setTableFunction("hasItemWithParameter", this) { + it.lua.push(hasItemWithParameter(it.getString(), it.getValue())) + 1 + } + + lua.setTableFunction("consumeItemWithParameter", this) { + it.lua.push(consumeItemWithParameter(it.getString(), it.getValue(), it.getLong())) + 1 + } + + lua.setTableFunction("getItemWithParameter", this) { + it.lua.push(getItemWithParameter(it.getString(), it.getValue()).toJson()) + 1 + } + + lua.setTableFunction("primaryHandItem", this) { + it.lua.push(primaryHandItem?.toJson()) + 1 + } + + lua.setTableFunction("altHandItem", this) { + it.lua.push(altHandItem?.toJson()) + 1 + } + + lua.setTableFunction("primaryHandItemTags", this) { + val tags = primaryHandItem?.item?.value?.itemTags + if (tags != null) it.lua.pushStrings(tags) else it.lua.push() + 1 + } + + lua.setTableFunction("altHandItemTags", this) { + val tags = altHandItem?.item?.value?.itemTags + if (tags != null) it.lua.pushStrings(tags) else it.lua.push() + 1 + } + + lua.setTableFunction("essentialItem", this) { + val name = it.getString() + val slot = essentialSlotsMap[name] ?: throw IllegalArgumentException("Invalid slot '$name'") + it.lua.push(essentialItem(slot)?.toJson()) + 1 + } + + lua.setTableClosure("giveEssentialItem", this) { + val name = it.getString() + val item = starbound.item(it.getValue()) + val slot = essentialSlotsMap[name] ?: throw IllegalArgumentException("Invalid slot '$name'") + giveEssentialItem(slot, item) + } + + lua.setTableClosure("removeEssentialItem", this) { + val name = it.getString() + val slot = essentialSlotsMap[name] ?: throw IllegalArgumentException("Invalid slot '$name'") + removeEssentialItem(slot) + } + + lua.setTableFunction("equippedItem", this) { + val name = it.getString() + val slot = equipmentSlotsMap[name] ?: throw IllegalArgumentException("Invalid slot '$name'") + it.lua.push(equippedItem(slot).toJson()) + 1 + } + + lua.setTableClosure("setEquippedItem", this) { + val name = it.getString() + val item = starbound.item(it.getValue()) + val slot = equipmentSlotsMap[name] ?: throw IllegalArgumentException("Invalid slot '$name'") + setEquippedItem(slot, item) + } + + lua.setTableFunction("swapSlotItem", this) { + it.lua.push(cursorItem.toJson()) + 1 + } + + lua.setTableClosure("setSwapSlotItem", this) { + cursorItem = starbound.item(it.getValue()) + } + + lua.setTableFunction("startQuest", this) { + it.lua.push(startQuest(it.getValue(), it.getStringOrNull(), it.getStringOrNull())) + 1 + } + + lua.setTableFunction("hasQuest", this) { + it.lua.push(quests[it.getString()] != null) + 1 + } + + lua.setTableFunction("hasCompletedQuest", this) { + val quest = quests[it.getString()] + it.lua.push(quest != null && quest.state == QuestInstance.State.COMPLETE) + 1 + } + + lua.pop() + } + + companion object { + private val playerLua by lazy { loadInternalScript("player") } + + private val essentialSlotsMap = immutableMap { + put("beamaxe", EssentialSlot.BEAM_AXE) + put("inspectiontool", EssentialSlot.INSPECTION_TOOL) + put("wiretool", EssentialSlot.WIRE_TOOL) + put("painttool", EssentialSlot.PAINT_TOOL) + } + + private val equipmentSlotsMap = immutableMap { + put("head", EquipmentSlot.HEAD) + put("chest", EquipmentSlot.CHEST) + put("legs", EquipmentSlot.LEGS) + put("back", EquipmentSlot.BACK) + put("headCosmetic", EquipmentSlot.HEAD_COSMETIC) + put("chestCosmetic", EquipmentSlot.CHEST_COSMETIC) + put("legsCosmetic", EquipmentSlot.LEGS_COSMETIC) + put("backCosmetic", EquipmentSlot.BACK_COSMETIC) + } + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/player/AvatarBag.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/player/AvatarBag.kt new file mode 100644 index 00000000..2bfd6389 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/player/AvatarBag.kt @@ -0,0 +1,57 @@ +package ru.dbotthepony.kstarbound.player + +import com.google.common.collect.ImmutableList +import ru.dbotthepony.kstarbound.defs.player.InventoryConfig +import ru.dbotthepony.kstarbound.util.ItemDescriptor +import java.util.function.Predicate + +class AvatarBag(val avatar: Avatar, val config: InventoryConfig.Bag, val filter: Predicate) { + private val slots = ImmutableList.builder().let { + for (i in 0 until config.size) { + it.add(Slot()) + } + + it.build() + } + + private class Slot { + var item: ItemDescriptor? = null + + fun mergeFrom(value: ItemDescriptor, simulate: Boolean) { + if (item == null) { + if (!simulate) { + item = value.copy().also { it.count = value.count.coerceAtMost(value.item!!.value.maxStack) } + } + + value.count -= value.item!!.value.maxStack + } else { + item!!.mergeFrom(value, simulate) + } + } + } + + operator fun set(index: Int, value: ItemDescriptor?) { + slots[index].item = value + } + + operator fun get(index: Int): ItemDescriptor? { + return slots[index].item + } + + fun put(item: ItemDescriptor, simulate: Boolean): ItemDescriptor { + if (!filter.test(item)) + return item + + val copy = item.copy() + + for (slot in slots.indices) { + slots[slot].mergeFrom(copy, simulate) + + if (copy.isEmpty) { + return copy + } + } + + return copy + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/player/Player.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/player/Player.kt new file mode 100644 index 00000000..028cc050 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/player/Player.kt @@ -0,0 +1,12 @@ +package ru.dbotthepony.kstarbound.player + +import ru.dbotthepony.kstarbound.Starbound + +/** + * Игрок - как он есть. + * + * [Player] - источник команд для [Avatar] + */ +class Player(val starbound: Starbound) { + var avatar: Avatar? = null +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestDescriptor.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestDescriptor.kt new file mode 100644 index 00000000..d1376dd7 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestDescriptor.kt @@ -0,0 +1,18 @@ +package ru.dbotthepony.kstarbound.player + +import com.google.gson.JsonObject +import ru.dbotthepony.kstarbound.io.json.builder.JsonFactory + +@JsonFactory +data class QuestDescriptor( + val questId: String, + val templateId: String = questId, + val seed: Long = makeSeed(), + val parameters: JsonObject = JsonObject() +) { + companion object { + fun makeSeed(): Long { + return System.nanoTime().rotateLeft(27).xor(System.currentTimeMillis()) + } + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestInstance.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestInstance.kt new file mode 100644 index 00000000..432a2a52 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/player/QuestInstance.kt @@ -0,0 +1,250 @@ +package ru.dbotthepony.kstarbound.player + +import com.google.gson.JsonArray +import com.google.gson.JsonElement +import com.google.gson.JsonObject +import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap +import org.apache.logging.log4j.LogManager +import ru.dbotthepony.kstarbound.defs.quest.QuestTemplate +import ru.dbotthepony.kstarbound.lua.LuaState +import ru.dbotthepony.kstarbound.lua.MessageHandler +import ru.dbotthepony.kstarbound.lua.exposeConfig +import ru.dbotthepony.kstarbound.util.ItemDescriptor +import ru.dbotthepony.kstarbound.util.set +import java.util.UUID + +class QuestInstance( + val avatar: Avatar, + val descriptor: QuestDescriptor, + val seed: Long = System.nanoTime().rotateLeft(27).xor(System.currentTimeMillis()), + val serverID: UUID? = null, + val worldID: String? = null +) { + val template: QuestTemplate = avatar.starbound.questTemplates[descriptor.templateId]?.value ?: throw IllegalArgumentException("No such quest template ${descriptor.templateId}") + val id get() = descriptor.questId + + val lua = LuaState() + val messages = MessageHandler(lua) + + enum class State(val serializedName: String) { + NEW("New"), + OFFER("Offer"), + ACTIVE("Active"), + COMPLETE("Complete"), + FAILED("Failed") + } + + private val objectiveList = ArrayList>() + + var canTurnIn = false + var failureText = "" + var completionText = "" + var text = "" + var title = "" + var state = State.NEW + var progress: Double? = null + var compassDirection: Double? = null + + private val portraits = JsonObject() + private val params = descriptor.parameters.deepCopy() + + private val portraitTitles = Object2ObjectOpenHashMap() + + private var isInitialized = false + private var successfulInit = false + private var calledStart = false + private var successfulStart = false + + private val rewards = ArrayList() + + fun complete() { + + } + + fun fail() { + + } + + private fun setObjectiveList(value: JsonArray) { + + } + + private fun addReward(value: JsonElement) { + val item = avatar.starbound.item(value) + + if (!item.isEmpty) { + rewards.add(item) + } + } + + init { + lua.pushTable() + lua.storeGlobal("self") + lua.pushTable() + lua.storeGlobal("storage") + + lua.pushTable() + lua.storeGlobal("quest") + lua.loadGlobal("quest") + + lua.setTableClosure("setParameter", this) { setParameter(it) } + + lua.setTableFunction("parameters", this) { + it.lua.push(params) + 1 + } + + lua.setTableClosure("setPortrait", this) { setPortrait(it) } + lua.setTableClosure("setPortraitTitle", this) { setPortraitTitle(it) } + + lua.setTableFunction("state", this) { it.lua.push(state.serializedName); 1 } + lua.setTableClosure("complete", this) { complete() } + lua.setTableClosure("fail", this) { fail() } + lua.setTableClosure("setCanTurnIn", this) { canTurnIn = it.getBool() } + lua.setTableFunction("questDescriptor", this) { it.lua.push(avatar.starbound.gson.toJsonTree(descriptor)); 1 } + lua.setTableFunction("questId", this) { it.lua.push(id); 1 } + lua.setTableFunction("templateId", this) { it.lua.push(template.id); 1 } + lua.setTableFunction("seed", this) { it.lua.push(seed); 1 } + lua.setTableFunction("questArcDescriptor", this) { TODO(); 1 } + lua.setTableFunction("questArcPosition", this) { TODO(); 1 } + lua.setTableFunction("worldId", this) { it.lua.push(worldID); 1 } + lua.setTableFunction("serverUuid", this) { it.lua.push(serverID?.toString()); 1 } + lua.setTableClosure("setObjectiveList", this) { setObjectiveList(it.getArray()) } + lua.setTableClosure("setProgress", this) { progress = it.getDoubleOrNil() } + lua.setTableClosure("setCompassDirection", this) { compassDirection = it.getDoubleOrNil() } + lua.setTableClosure("setTitle", this) { title = it.getString() } + lua.setTableClosure("setText", this) { text = it.getString() } + lua.setTableClosure("setCompletionText", this) { completionText = it.getString() } + lua.setTableClosure("setFailureText", this) { failureText = it.getString() } + lua.setTableClosure("addReward", this) { addReward(it.getValue()) } + + lua.pop() + + avatar.starbound.expose(lua) + avatar.expose(lua) + lua.exposeConfig(template.scriptConfig) + } + + private fun setParameter(argStack: LuaState.ArgStack) { + params.add(argStack.getString(), argStack.getValue()) + } + + private fun setPortrait(argStack: LuaState.ArgStack) { + val name = argStack.getString() + val value = argStack.getAnything() + + if (value == null) + portraits.remove(name) + else + portraits.add(name, value) + } + + private fun setPortraitTitle(argStack: LuaState.ArgStack) { + val name = argStack.getString() + val title = argStack.getStringOrNil() + + if (title == null) + portraitTitles.remove(name) + else + portraitTitles[name] = title + } + + init { + for ((k, v) in descriptor.parameters.entrySet()) { + params[k] = v.deepCopy() + } + } + + /** + * Читает главный скриптовый файл квеста и вызывает его (глобальную) функцию init() + */ + fun init(): Boolean { + if (!isInitialized) { + isInitialized = true + + val script = avatar.starbound.locate(template.script.fullPath) + + if (!script.isFile) { + LOGGER.error("Quest ${template.id} specifies ${template.script.fullPath} as its script, but it is not a file or does not exist!") + return false + } + + try { + lua.load(script.readToString(), template.script.fullPath) + } catch(err: Throwable) { + LOGGER.error("Error loading Lua code for quest ${descriptor.questId}", err) + return false + } + + try { + lua.call() + } catch(err: Throwable) { + LOGGER.error("Error running Lua code for quest ${descriptor.questId}", err) + return false + } + + try { + lua.loadGlobal("init") + lua.call() + } catch(err: Throwable) { + LOGGER.error("Error running init() function for quest ${descriptor.questId}", err) + return false + } + + successfulInit = true + } + + return successfulInit + } + + fun start(): Boolean { + if (!init()) { + return false + } + + if (calledStart) { + return successfulStart + } + + lua.loadGlobal("questStart") + + try { + if (lua.isFunction()) { + lua.call() + } else { + lua.pop() + } + } catch(err: Throwable) { + LOGGER.error("Error running questStart() function for quest ${descriptor.questId}", err) + return false + } + + return successfulStart + } + + fun update(delta: Int): Boolean { + if (!init()) { + return false + } + + lua.loadGlobal("update") + + try { + if (lua.isFunction()) { + lua.push(delta) + lua.call(numArgs = 1) + } else { + lua.pop() + } + } catch(err: Throwable) { + LOGGER.error("Error running update() function for quest ${descriptor.questId}", err) + return false + } + + return true + } + + companion object { + private val LOGGER = LogManager.getLogger() + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/tools/Sbon2Json.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/tools/Sbon2Json.kt new file mode 100644 index 00000000..e6383ac0 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/tools/Sbon2Json.kt @@ -0,0 +1,121 @@ +package ru.dbotthepony.kstarbound.tools + +import com.google.gson.GsonBuilder +import ru.dbotthepony.kstarbound.io.json.BinaryJsonReader +import ru.dbotthepony.kstarbound.io.readHeader +import ru.dbotthepony.kstarbound.io.readString +import ru.dbotthepony.kstarbound.io.readVarInt +import java.io.BufferedInputStream +import java.io.DataInputStream +import java.io.File +import java.io.OutputStreamWriter +import java.util.Scanner +import kotlin.system.exitProcess + +fun main(vararg args: String) { + var originalFile = args.getOrNull(0) + val scanner = System.`in`?.let(::Scanner) + var interactive = false + + if (originalFile == null) { + if (scanner == null) { + System.err.println("Usage: [output file path]") + System.err.println("By default, new file is put in the same folder as original file") + System.err.println("with '.json' extension added to it") + exitProcess(1) + } else { + println("Input filename:") + originalFile = scanner.nextLine() + interactive = true + } + } + + val f = File(originalFile!!).absoluteFile + + if (!f.exists()) { + System.err.println("File $f does not exist") + exitProcess(1) + } + + if (!f.isFile) { + System.err.println("$f is not a file") + exitProcess(1) + } + + var newFile = args.getOrNull(1) + + if (newFile == null) { + if (interactive) { + val def = f.parent + "/" + f.name + ".json" + println("Output filename [$def]:") + val read = scanner!!.nextLine() + + if (read == "") { + newFile = def + } else { + newFile = read!! + } + } else { + newFile = f.parent + "/" + f.name + ".json" + } + } + + val new = File(newFile) + + if (!new.parentFile.exists()) { + System.err.println("Output directory ${new.parent} does not exist") + exitProcess(1) + } + + if (!new.parentFile.isDirectory) { + System.err.println("Output directory ${new.parent} is not a directory") + exitProcess(1) + } + + if (new.exists() && !new.isFile) { + System.err.println("Output path $new already exists and it is not a file") + exitProcess(1) + } + + if (interactive && new.exists()) { + println("$new already exists. Overwrite? [Y/n]") + var next = scanner!!.nextLine() + if (next == "") next = "y" + + if (next.lowercase()[0] != 'y') { + System.err.println("Halt") + exitProcess(1) + } + } + + try { + val t = System.nanoTime() + val dataStream = DataInputStream(BufferedInputStream(f.inputStream())) + + dataStream.readHeader("SBVJ01") + val name = dataStream.readString(dataStream.readVarInt()) + val magic = dataStream.read() + val version = dataStream.readInt() + val data = BinaryJsonReader.readElement(dataStream) + val gson = GsonBuilder().setPrettyPrinting().create() + + dataStream.close() + + println("$f:") + println("JSON Name: $name") + println("Version: $version") + println("Magic: $magic") + println("JSON Type: ${data::class.simpleName}") + + new.delete() + val output = OutputStreamWriter(new.outputStream()) + gson.toJson(data, output) + output.close() + + println("Successfully written data to $new ${"in %.2f ms".format(((System.nanoTime() - t) / 1_000L) / 1_000.0)}") + exitProcess(0) + } catch(err: Throwable) { + System.err.println(err) + exitProcess(1) + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/Ext.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/Ext.kt new file mode 100644 index 00000000..188f967f --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/Ext.kt @@ -0,0 +1,16 @@ +package ru.dbotthepony.kstarbound.util + +import com.google.gson.JsonElement +import com.google.gson.JsonNull +import com.google.gson.JsonObject +import com.google.gson.JsonPrimitive +import ru.dbotthepony.kstarbound.io.json.InternedJsonElementAdapter + +operator fun JsonObject.set(key: String, value: JsonElement) { add(key, value) } +operator fun JsonObject.set(key: String, value: String) { add(key, JsonPrimitive(value)) } +operator fun JsonObject.set(key: String, value: Int) { add(key, JsonPrimitive(value)) } +operator fun JsonObject.set(key: String, value: Long) { add(key, JsonPrimitive(value)) } +operator fun JsonObject.set(key: String, value: Float) { add(key, JsonPrimitive(value)) } +operator fun JsonObject.set(key: String, value: Double) { add(key, JsonPrimitive(value)) } +operator fun JsonObject.set(key: String, value: Boolean) { add(key, InternedJsonElementAdapter.of(value)) } +operator fun JsonObject.set(key: String, value: Nothing?) { add(key, JsonNull.INSTANCE) } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/ItemDescriptor.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/ItemDescriptor.kt new file mode 100644 index 00000000..15a9fa84 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/ItemDescriptor.kt @@ -0,0 +1,145 @@ +package ru.dbotthepony.kstarbound.util + +import com.google.gson.JsonObject +import com.google.gson.JsonPrimitive +import com.google.gson.TypeAdapter +import com.google.gson.internal.bind.TypeAdapters +import com.google.gson.stream.JsonReader +import com.google.gson.stream.JsonToken +import com.google.gson.stream.JsonWriter +import ru.dbotthepony.kstarbound.RegistryObject +import ru.dbotthepony.kstarbound.Starbound +import ru.dbotthepony.kstarbound.defs.item.IItemDefinition + +class ItemDescriptor private constructor(item: RegistryObject?, count: Long, val parameters: JsonObject, marker: Unit) { + constructor(item: RegistryObject, count: Long = 1L, parameters: JsonObject = JsonObject()) : this(item, count, parameters, Unit) + + var item: RegistryObject? = item + private set + + var count = count + set(value) { + if (field == 0L) + return + + field = value.coerceAtLeast(0L) + + if (field == 0L) { + item = null + } + } + + val isEmpty: Boolean + get() = count <= 0 || item == null + + val isNotEmpty: Boolean + get() = count > 0 && item != null + + fun grow(amount: Long) { + count += amount + } + + fun shrink(amount: Long) { + count -= amount + } + + /** + * Возвращает null если этот предмет пуст + */ + fun conciseToNull(): ItemDescriptor? { + if (isEmpty) { + return null + } else { + return this + } + } + + fun mergeFrom(other: ItemDescriptor, simulate: Boolean) { + if (isStackable(other)) { + val newCount = (count + other.count).coerceAtMost(item!!.value.maxStack) + val diff = newCount - count + other.count -= diff + + if (!simulate) + count = newCount + } + } + + fun lenientEquals(other: Any?): Boolean { + if (other !is ItemDescriptor) + return false + + if (isEmpty) + return other.isEmpty + + return other.count == count && other.item == item + } + + fun isStackable(other: ItemDescriptor): Boolean { + return count != 0L && other.count != 0L && item!!.value.maxStack < count && other.item == item && other.parameters == parameters + } + + override fun equals(other: Any?): Boolean { + if (other !is ItemDescriptor) + return false + + if (isEmpty) + return other.isEmpty + + return other.count == count && other.item == item && other.parameters == parameters + } + + // мы не можем делать хеш из count и parameters так как они изменяемы + override fun hashCode(): Int { + return item.hashCode() + } + + override fun toString(): String { + if (isEmpty) + return "ItemDescriptor.EMPTY" + + return "ItemDescriptor[${item!!.value.itemName}, count = $count, params = $parameters]" + } + + fun copy(): ItemDescriptor { + if (isEmpty) + return this + + return ItemDescriptor(item, count, parameters.deepCopy(), Unit) + } + + fun toJson(): JsonObject? { + if (isEmpty) { + return null + } + + return JsonObject().also { + it.add("name", JsonPrimitive(item!!.value.itemName)) + it.add("count", JsonPrimitive(count)) + it.add("parameters", parameters.deepCopy()) + } + } + + class Adapter(val starbound: Starbound) : TypeAdapter() { + override fun write(out: JsonWriter, value: ItemDescriptor?) { + val json = value?.toJson() + + if (json == null) + out.nullValue() + else + TypeAdapters.JSON_ELEMENT.write(out, json) + } + + override fun read(`in`: JsonReader): ItemDescriptor { + if (`in`.peek() == JsonToken.NULL) + return EMPTY + + return starbound.item(TypeAdapters.JSON_ELEMENT.read(`in`)) + } + } + + companion object { + @JvmField + val EMPTY = ItemDescriptor(null, 0L, JsonObject(), Unit) + } +} diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt index ba44fb2a..2e0408c3 100644 --- a/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/SBPattern.kt @@ -11,6 +11,7 @@ import com.google.gson.reflect.TypeToken import com.google.gson.stream.JsonReader import com.google.gson.stream.JsonWriter import it.unimi.dsi.fastutil.objects.Object2ObjectArrayMap +import ru.dbotthepony.kstarbound.Starbound /** * Шаблонизировання строка в стиле Starbound'а @@ -165,7 +166,7 @@ class SBPattern private constructor( throw IllegalArgumentException("Malformed pattern string: $raw") } - pieces.add(Piece(name = raw.substring(open + 1, closing))) + pieces.add(Piece(name = Starbound.STRINGS.intern(raw.substring(open + 1, closing)))) i = closing + 1 } } diff --git a/src/main/kotlin/ru/dbotthepony/kstarbound/util/Utils.kt b/src/main/kotlin/ru/dbotthepony/kstarbound/util/Utils.kt new file mode 100644 index 00000000..49807ef7 --- /dev/null +++ b/src/main/kotlin/ru/dbotthepony/kstarbound/util/Utils.kt @@ -0,0 +1,89 @@ +package ru.dbotthepony.kstarbound.util + +import com.google.common.collect.ImmutableList +import com.google.common.collect.ImmutableMap +import com.google.common.collect.ImmutableSet +import com.google.gson.JsonArray +import com.google.gson.JsonElement +import com.google.gson.JsonObject +import ru.dbotthepony.kstarbound.Starbound +import java.util.* +import java.util.function.Consumer +import kotlin.collections.ArrayList + +fun String.sbIntern(): String { + return Starbound.STRINGS.intern(this) +} + +fun String.sbIntern2(): String { + return Starbound.STRINGS.intern(this.intern()) +} + +fun traverseJsonPath(path: String?, element: JsonElement?): JsonElement? { + element ?: return null + path ?: return element + + if (path.contains('.')) { + var current: JsonElement? = element + + for (name in path.split('.')) { + if (current is JsonObject) { + current = current[name] + } else if (current is JsonArray) { + val toInt = name.toIntOrNull() ?: return null + if (toInt !in 0 until current.size()) return null + current = current.get(toInt) + } else { + return null + } + } + + return current + } else { + if (element is JsonObject) { + return element[path] + } else if (element is JsonArray) { + val toInt = path.toIntOrNull() ?: return null + if (toInt !in 0 until element.size()) return null + return element[toInt] + } else { + return null + } + } +} + +inline fun immutableMap(initializer: ImmutableMap.Builder.() -> Unit): ImmutableMap { + val builder = ImmutableMap.Builder() + initializer.invoke(builder) + return builder.build() +} + +inline fun immutableSet(initializer: Consumer.() -> Unit): ImmutableSet { + val builder = ImmutableSet.Builder() + initializer.invoke(builder::add) + return builder.build() +} + +inline fun immutableList(initializer: Consumer.() -> Unit): ImmutableList { + val builder = ImmutableList.Builder() + initializer.invoke(builder::add) + return builder.build() +} + +fun UUID.toStarboundString(): String { + val builder = StringBuilder(32) + val a = mostSignificantBits.toString(16) + val b = mostSignificantBits.toString(16) + + for (i in a.length until 16) + builder.append("0") + + builder.append(a) + + for (i in b.length until 16) + builder.append("0") + + builder.append(b) + + return builder.toString() +} diff --git a/src/main/resources/scripts/config.lua b/src/main/resources/scripts/config.lua new file mode 100644 index 00000000..7b5f2a7e --- /dev/null +++ b/src/main/resources/scripts/config.lua @@ -0,0 +1,17 @@ + +config = {} +local config = config + +function config.getParameter(name, default) + if type(name) ~= 'string' then + error('config.getParameter: name must be a string, got ' .. type(name), 2) + end + + local get = config._get(name) + + if get == nil then + return default + else + return get + end +end diff --git a/src/main/resources/scripts/message_handler.lua b/src/main/resources/scripts/message_handler.lua new file mode 100644 index 00000000..5b0a6d5a --- /dev/null +++ b/src/main/resources/scripts/message_handler.lua @@ -0,0 +1,27 @@ + +message = { + handlers = {} +} + +local message = message + +function message.setHandler(name, handler) + if type(name) ~= 'string' then + error('message.setHandler: Handler name must be a string, got ' .. type(name), 2) + end + + if type(handler) ~= 'function' then + error('message.setHandler: Handler itself must be a function, got ' .. type(handler), 2) + end + + message.subscribe(name) + message.handlers[name] = handler +end + +function message.call(name, ...) + local handler = message.handlers[name] + + if handler ~= nil then + return handler(...) + end +end diff --git a/src/main/resources/scripts/player.lua b/src/main/resources/scripts/player.lua new file mode 100644 index 00000000..4a042e16 --- /dev/null +++ b/src/main/resources/scripts/player.lua @@ -0,0 +1,21 @@ + +local playerData = select(1, ...) + +player = {} +local player = player + +local uniqueId = playerData.uniqueId +local species = playerData.species +local gender = playerData.gender + +function player.uniqueId() + return uniqueId +end + +function player.species() + return species +end + +function player.gender() + return gender +end diff --git a/src/main/resources/scripts/polyfill.lua b/src/main/resources/scripts/polyfill.lua new file mode 100644 index 00000000..ca875572 --- /dev/null +++ b/src/main/resources/scripts/polyfill.lua @@ -0,0 +1,8 @@ + +function jobject() + return {} +end + +function jarray() + return {} +end