From 015e76eb5094bc9c558fd8c78b399246f45e6460 Mon Sep 17 00:00:00 2001 From: Gabriel Tofvesson Date: Tue, 14 Sep 2021 20:45:32 +0200 Subject: [PATCH] Initial Commit --- .idea/.gitignore | 8 + .idea/compiler.xml | 6 + .idea/gradle.xml | 17 ++ .idea/jarRepositories.xml | 75 +++++++ .idea/misc.xml | 15 ++ .idea/modules/SpigotLandmines2.main.iml | 12 ++ .idea/runConfigurations.xml | 10 + .idea/vcs.xml | 6 + build.gradle | 96 +++++++++ gradle.properties | 3 + gradlew | 185 ++++++++++++++++++ gradlew.bat | 89 +++++++++ settings.gradle | 1 + .../dev/w1zzrd/spigot/landmines2/Buffers.kt | 19 ++ .../w1zzrd/spigot/landmines2/LandmineData.kt | 35 ++++ .../spigot/landmines2/LandmineManager.kt | 185 ++++++++++++++++++ .../spigot/landmines2/LandminePlugin.kt | 31 +++ .../dev/w1zzrd/spigot/landmines2/Numbers.kt | 85 ++++++++ .../dev/w1zzrd/spigot/landmines2/Ordering.kt | 15 ++ .../spigot/landmines2/SerializableLocation.kt | 74 +++++++ .../w1zzrd/spigot/landmines2/SortedList.kt | 89 +++++++++ .../dev/w1zzrd/spigot/landmines2/YamlFile.kt | 19 ++ src/main/resources/config.yml | 3 + 23 files changed, 1078 insertions(+) create mode 100644 .idea/.gitignore create mode 100644 .idea/compiler.xml create mode 100644 .idea/gradle.xml create mode 100644 .idea/jarRepositories.xml create mode 100644 .idea/misc.xml create mode 100644 .idea/modules/SpigotLandmines2.main.iml create mode 100644 .idea/runConfigurations.xml create mode 100644 .idea/vcs.xml create mode 100644 build.gradle create mode 100644 gradle.properties create mode 100644 gradlew create mode 100644 gradlew.bat create mode 100644 settings.gradle create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/Buffers.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineData.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineManager.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandminePlugin.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/Numbers.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/Ordering.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/SerializableLocation.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/SortedList.kt create mode 100644 src/main/kotlin/dev/w1zzrd/spigot/landmines2/YamlFile.kt create mode 100644 src/main/resources/config.yml diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..73f69e0 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,8 @@ +# Default ignored files +/shelf/ +/workspace.xml +# Datasource local storage ignored files +/dataSources/ +/dataSources.local.xml +# Editor-based HTTP Client requests +/httpRequests/ diff --git a/.idea/compiler.xml b/.idea/compiler.xml new file mode 100644 index 0000000..659bf43 --- /dev/null +++ b/.idea/compiler.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/gradle.xml b/.idea/gradle.xml new file mode 100644 index 0000000..611e7c8 --- /dev/null +++ b/.idea/gradle.xml @@ -0,0 +1,17 @@ + + + + + + + \ No newline at end of file diff --git a/.idea/jarRepositories.xml b/.idea/jarRepositories.xml new file mode 100644 index 0000000..00fddde --- /dev/null +++ b/.idea/jarRepositories.xml @@ -0,0 +1,75 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..cc57d5f --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/modules/SpigotLandmines2.main.iml b/.idea/modules/SpigotLandmines2.main.iml new file mode 100644 index 0000000..fa63d4b --- /dev/null +++ b/.idea/modules/SpigotLandmines2.main.iml @@ -0,0 +1,12 @@ + + + + + + + SPIGOT + + + + + \ No newline at end of file diff --git a/.idea/runConfigurations.xml b/.idea/runConfigurations.xml new file mode 100644 index 0000000..797acea --- /dev/null +++ b/.idea/runConfigurations.xml @@ -0,0 +1,10 @@ + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/build.gradle b/build.gradle new file mode 100644 index 0000000..7147ae1 --- /dev/null +++ b/build.gradle @@ -0,0 +1,96 @@ +plugins { + id 'org.jetbrains.kotlin.jvm' version '1.5.30' + id 'java' + id 'kr.entree.spigradle' version '2.2.4' + id("com.github.sgtsilvio.gradle.proguard") version "0.1.1" +} + +group 'dev.w1zzrd.spigot.landmines2' +version '1.0-SNAPSHOT' + +repositories { + mavenCentral() + + maven { + url = 'https://hub.spigotmc.org/nexus/content/repositories/snapshots/' + + // As of Gradle 5.1, you can limit this to only those + // dependencies you expect from it + content { + includeGroup 'org.bukkit' + includeGroup 'org.spigotmc' + } + } + maven { + url = "https://papermc.io/repo/repository/maven-public" + + content { + includeGroup 'io.papermc.paper' + } + } + maven { url = 'https://oss.sonatype.org/content/repositories/snapshots' } + maven { url = 'https://oss.sonatype.org/content/repositories/central' } + mavenLocal() + maven { url "https://jitpack.io" } +} + +dependencies { + implementation 'org.jetbrains.kotlin:kotlin-stdlib:1.5.30' + //implementation 'com.github.GabrielTofvesson:SpigotWizCompat:latest' + //implementation files('lib/SpigotWizCompat-072da001f26b738b510ac9a6edc52d338ca73070.jar') + testImplementation 'org.junit.jupiter:junit-jupiter-api:5.7.2' + testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.7.2' + + compileOnly spigot('1.17.1') +} + +test { + useJUnitPlatform() +} + + +spigot { + authors = ['IKEAJesus'] + depends = ['Kotlin'] + apiVersion = '1.17' + load = STARTUP + + /* + commands { + give { + aliases = ['i'] + description = 'Give command.' + permission = 'test.foo' + permissionMessage = 'You do not have permission!' + usage = '/ [test|stop]' + } + } + permissions { + 'test.foo' { + description = 'Allows foo command' + defaults = 'true' + } + 'test.*' { + description = 'Wildcard permission' + defaults = 'op' + children = ['test.foo': true] + } + } + */ + + debug { + jvmArgs '-Xmx4G' + buildVersion = "1.17.1" + } +} + +compileKotlin { + kotlinOptions { + jvmTarget = "16" + } +} +compileTestKotlin { + kotlinOptions { + jvmTarget = "16" + } +} \ No newline at end of file diff --git a/gradle.properties b/gradle.properties new file mode 100644 index 0000000..faa5964 --- /dev/null +++ b/gradle.properties @@ -0,0 +1,3 @@ +kotlin.code.style=official +#org.gradle.jvmargs=--add-opens jdk.compiler/com.sun.tools.javac.util=ALL-UNNAMED +org.gradle.jvmargs=--illegal-access=permit \ No newline at end of file diff --git a/gradlew b/gradlew new file mode 100644 index 0000000..744e882 --- /dev/null +++ b/gradlew @@ -0,0 +1,185 @@ +#!/usr/bin/env sh + +# +# Copyright 2015 the original author or authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn () { + echo "$*" +} + +die () { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MSYS* | MINGW* ) + msys=true + ;; + NONSTOP* ) + nonstop=true + ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin or MSYS, switch paths to Windows format before running java +if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=`expr $i + 1` + done + case $i in + 0) set -- ;; + 1) set -- "$args0" ;; + 2) set -- "$args0" "$args1" ;; + 3) set -- "$args0" "$args1" "$args2" ;; + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Escape application args +save () { + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done + echo " " +} +APP_ARGS=`save "$@"` + +# Collect all arguments for the java command, following the shell quoting and substitution rules +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" + +exec "$JAVACMD" "$@" diff --git a/gradlew.bat b/gradlew.bat new file mode 100644 index 0000000..107acd3 --- /dev/null +++ b/gradlew.bat @@ -0,0 +1,89 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto execute + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/settings.gradle b/settings.gradle new file mode 100644 index 0000000..6f00585 --- /dev/null +++ b/settings.gradle @@ -0,0 +1 @@ +rootProject.name = 'SpigotLandmines2' \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Buffers.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Buffers.kt new file mode 100644 index 0000000..7e5029b --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Buffers.kt @@ -0,0 +1,19 @@ +package dev.w1zzrd.spigot.landmines2 + +import java.nio.ByteBuffer + +fun ByteBuffer.writePacked(value: ULong) = value.toPacked(this) +fun ByteBuffer.writePacked(value: UInt) = writePacked(value.toULong()) +fun ByteBuffer.writePacked(value: UShort) = writePacked(value.toULong()) + +fun ByteBuffer.writePacked(value: Long) = value.interlace().toPacked(this) +fun ByteBuffer.writePacked(value: Int) = writePacked(value.toLong()) +fun ByteBuffer.writePacked(value: Short) = writePacked(value.toLong()) + +val ByteBuffer.packedULong: ULong get() = readPacked().first +val ByteBuffer.packedUInt: UInt get() = readPacked().first.toUInt() +val ByteBuffer.packedUShort: UShort get() = readPacked().first.toUShort() + +val ByteBuffer.packedLong: Long get() = readPacked().first.deInterlace() +val ByteBuffer.packedInt: Int get() = readPacked().first.deInterlace().toInt() +val ByteBuffer.packedShort: Short get() = readPacked().first.deInterlace().toShort() \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineData.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineData.kt new file mode 100644 index 0000000..73ba48c --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineData.kt @@ -0,0 +1,35 @@ +package dev.w1zzrd.spigot.landmines2 + +import java.nio.ByteBuffer +import java.util.* + +typealias LocationPredicate = (LandmineData) -> Int + +val SerializableLocation.locationPredicate: LocationPredicate + get() = { data -> data.location.compareTo(this) } + +private val threadLocalBuffer = ThreadLocal.withInitial { ByteBuffer.allocate(45) } + +private fun unpackData(data: String, worlds: (UInt) -> UUID): Pair { + val buffer = threadLocalBuffer.get() + buffer.position(0) + Base64.getDecoder().decode(data.toByteArray(Charsets.ISO_8859_1), buffer.array()) + + val placer = buffer.packedUInt + return unpackLocation(buffer, worlds) to placer +} + +data class LandmineData(val location: SerializableLocation, val placer: UInt) { + private constructor(pair: Pair): this(pair.first, pair.second) + constructor(landmineData: String, worlds: (UInt) -> UUID): this(unpackData(landmineData, worlds)) + + override fun toString() = "${placer.toULong().toPackedString()}$location" + fun toPackedString(worlds: (UUID) -> UInt): String { + val buffer = threadLocalBuffer.get() + buffer.position(0) + + buffer.writePacked(placer) + location.writePacked(buffer, worlds) + return Base64.getEncoder().withoutPadding().encodeToString(Arrays.copyOf(buffer.array(), buffer.position())) + } +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineManager.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineManager.kt new file mode 100644 index 0000000..48f0144 --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandmineManager.kt @@ -0,0 +1,185 @@ +package dev.w1zzrd.spigot.landmines2 + +import org.bukkit.Location +import org.bukkit.Material +import org.bukkit.OfflinePlayer +import org.bukkit.configuration.file.FileConfiguration +import org.bukkit.event.EventHandler +import org.bukkit.event.EventPriority +import org.bukkit.event.HandlerList +import org.bukkit.event.Listener +import org.bukkit.event.block.BlockPlaceEvent +import org.bukkit.event.entity.PlayerDeathEvent +import org.bukkit.event.player.PlayerMoveEvent +import org.bukkit.plugin.Plugin +import java.nio.ByteBuffer +import java.util.* +import kotlin.collections.ArrayList + +private val LANDMINE_COMPARATOR = Comparator { a, b -> a.location.compareTo(b.location) } +private const val GC_TRIGGER = 10000 + +private const val DEFAULT_EXPLOSION_STRENGTH = 2.0 + +private const val PATH_LANDMINES = "landmines" +private const val PATH_EXPLOSION_STRENGTH = "explosionStrength" +private const val PATH_PLAYER_REGISTRY = "players" +private const val PATH_WORLD_REGISTRY = "worlds" +private const val PATH_GC_TRIGGER = "gcTrigger" + +private val threadLocalBuffer = ThreadLocal.withInitial { ByteBuffer.allocate(16) } + +class LandmineManager( + private val plugin: Plugin, + private val landmineData: YamlFile +): Listener { + private var landmines: SortedList = SortedList.create(comparator = LANDMINE_COMPARATOR) + private lateinit var players: MutableList + private lateinit var worlds: MutableList + private var explosionStrength = DEFAULT_EXPLOSION_STRENGTH + private var damageTracker: LandmineData? = null + + private var isEnabled = false + + private val conf: FileConfiguration + get() = plugin.config + + + + fun onEnable() { + // Ensure manager cannot be double-enabled + if (isEnabled) throw IllegalStateException("Manager already enabled!") + isEnabled = true + + reload() + + plugin.server.pluginManager.registerEvents(this, plugin) + } + + fun onDisable() { + if (!isEnabled) + throw IllegalStateException("Manager not enabled!") + + HandlerList.unregisterAll(this) + + save() + + // Mark manager as disabled *after* disable routine completes + isEnabled = false + } + + fun reload() { + val shouldGC = landmines.size >= conf.getInt(PATH_GC_TRIGGER, GC_TRIGGER) + + landmineData.reload() + players = landmineData.getStringList(PATH_PLAYER_REGISTRY) + worlds = landmineData.getStringList(PATH_WORLD_REGISTRY) + + + landmines = if (landmineData.contains(PATH_LANDMINES)) { + val stringList = landmineData.getStringList(PATH_LANDMINES) + val decodeBuffer = ByteBuffer.allocate(16) + SortedList.create( + LANDMINE_COMPARATOR, + stringList.mapTo(ArrayList(stringList.size)) { entry -> + LandmineData(entry) { index -> + decodeBuffer.position(0) + Base64.getDecoder().decode(worlds[index.toInt()].toByteArray(Charsets.ISO_8859_1), decodeBuffer.array()) + UUID(decodeBuffer.long, decodeBuffer.long) + } + } + ) + } + else SortedList.create(comparator = LANDMINE_COMPARATOR) + + explosionStrength = conf.getDouble(PATH_EXPLOSION_STRENGTH, DEFAULT_EXPLOSION_STRENGTH) + + // If we anticipate a very large amount of garbage, trigger gc manually + if (shouldGC) + Runtime.getRuntime().gc() + } + + private fun save() { + landmineData.set(PATH_LANDMINES, landmines.map { it.toPackedString(this::getWorldIndex) }) + landmineData.set(PATH_PLAYER_REGISTRY, players) + landmineData.set(PATH_WORLD_REGISTRY, worlds) + landmineData.save() + + conf.set(PATH_EXPLOSION_STRENGTH, explosionStrength) + } + + private fun placeMine(player: OfflinePlayer, location: Location): Boolean { + val serializable = location.serializable + val index = landmines.binarySearch(comparison = serializable.locationPredicate) + if (index >= 0) return false + + landmines.add(LandmineData(serializable, getPlayerNameIndex(player))) + return true + } + + private fun findPlayerName(index: UInt): String? { + if (index.toInt() !in 0 until players.size) return null + val buffer = threadLocalBuffer.get() + buffer.position(0) + Base64.getDecoder().decode(players[index.toInt()].toByteArray(Charsets.ISO_8859_1), buffer.array()) + + return plugin.server.getPlayer(UUID(buffer.long, buffer.long))?.name + } + + private fun getWorldIndex(world: UUID) = worlds.getIncrementalIndex(world) + private fun getPlayerNameIndex(player: OfflinePlayer) = players.getIncrementalIndex(player.uniqueId) + private fun MutableList.getIncrementalIndex(uuid: UUID): UInt { + val buffer = threadLocalBuffer.get() + buffer.position(0) + buffer.putLong(uuid.mostSignificantBits) + buffer.putLong(uuid.leastSignificantBits) + + val b64String = Base64.getEncoder().withoutPadding().encodeToString(buffer.array()) + + val index = indexOf(b64String) + if (index >= 0) return index.toUInt() + + add(b64String) + return (size - 1).toUInt() + } + + + @EventHandler + fun onPlayerMove(moveEvent: PlayerMoveEvent) { + val index = landmines.binarySearch(comparison = (moveEvent.to ?: return).serializable.locationPredicate) + if (index >= 0) { + val landmine = landmines.removeAt(index) + + damageTracker = landmine + moveEvent.to!!.world!!.createExplosion( + landmine.location.getBukkitLocation(moveEvent.player.server), + explosionStrength.toFloat(), + false, + false, + null + ) + damageTracker = null + } + } + + @EventHandler + fun onPlayerDeath(deathEvent: PlayerDeathEvent) { + if (damageTracker != null) { + val placer = findPlayerName(damageTracker!!.placer) + deathEvent.deathMessage = + if (placer == null) "${deathEvent.entity.name} stepped on a landmine" + else "${deathEvent.entity.name} stepped on a landmine placed by $placer" + } + } + + @EventHandler(priority = EventPriority.HIGHEST) + fun onPlayerPlaceMine(placeEvent: BlockPlaceEvent) { + if (placeEvent.blockPlaced.type == Material.STONE_PRESSURE_PLATE && !placeEvent.isCancelled) { + if (placeMine(placeEvent.player, placeEvent.blockPlaced.location)) { + --placeEvent.player.inventory.getItem(placeEvent.hand).amount + } + + placeEvent.isCancelled = true + } + } +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandminePlugin.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandminePlugin.kt new file mode 100644 index 0000000..795a00c --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/LandminePlugin.kt @@ -0,0 +1,31 @@ +package dev.w1zzrd.spigot.landmines2 + +import kr.entree.spigradle.annotations.SpigotPlugin +import org.bukkit.plugin.java.JavaPlugin +import java.io.File + +@SpigotPlugin +class LandminePlugin: JavaPlugin() { + private var landmineManager: LandmineManager? = null + + override fun onEnable() { + super.onEnable() + + saveDefaultConfig() + landmineManager = LandmineManager(this, YamlFile(File(dataFolder, "data.yml"))) + landmineManager!!.onEnable() + } + + override fun reloadConfig() { + super.reloadConfig() + saveDefaultConfig() + landmineManager?.reload() + } + + override fun onDisable() { + landmineManager!!.onDisable() + saveConfig() + + super.onDisable() + } +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Numbers.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Numbers.kt new file mode 100644 index 0000000..d7572f3 --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Numbers.kt @@ -0,0 +1,85 @@ +package dev.w1zzrd.spigot.landmines2 + +import java.nio.ByteBuffer + +private val threadLocalBuffer = ThreadLocal.withInitial { ByteBuffer.allocateDirect(9) } + +private fun ByteBuffer.putUByte(value: ULong) = put((value and 0xFFUL).toByte()) +private fun String.parseHex(index: Int) = ((this[index * 2].digitToInt(16) shl 4) or (this[1 + index * 2].digitToInt(16))).toULong() and 0xFFUL + +fun Long.interlace() = (this shr 63).toULong() xor (this shl 1).toULong() +fun ULong.deInterlace() = (this shr 1).toLong() xor ((this shl 63).toLong() shr 63) + +fun ULong.toPacked(target: ByteBuffer): Int { + if (this <= 240UL) { + target.putUByte(this) + return 1 + } + else if (this <= 2287UL) { + target.putUByte(((this - 240UL) shr 8) + 241UL) + target.putUByte(this - 240UL) + return 2 + } + else if (this <= 67823UL) { + target.putUByte(249UL) + target.putUByte((this - 2288UL) shr 8) + target.putUByte(this - 2288UL) + return 3 + } + else { + var header = 255UL + var match = 0x00FF_FFFF_FFFF_FFFFUL + + while (this <= match) { + --header + match = match shr 8 + } + + target.putUByte(header) + for (i in 0 until (header - 247UL).toInt()) + target.putUByte(this shr (i shl 3)) + + return (header - 247UL).toInt() + } +} + +fun ULong.toPackedString(): String { + val buffer = threadLocalBuffer.get().position(0) + val len = toPacked(buffer) + + val builder = StringBuilder(len * 2) + for (i in 0 until len) { + builder.append(((buffer[i].toInt() ushr 4) and 0xF).toString(16)) + builder.append((buffer[i].toInt() and 0xF).toString(16)) + } + + return builder.toString() +} + +fun ByteBuffer.readPacked(): Pair { + val header = get().toULong() and 0xFFUL + if (header <= 240UL) return header to 1 + else if (header <= 248UL) return (240UL + ((header - 241UL) shl 8) + (get().toULong() and 0xFFUL)) to 2 + else if (header == 249UL) return (2288UL + ((get().toULong() and 0xFFUL) shl 8) + (get().toULong() and 0xFFUL)) to 3 + + var res = (get().toULong() and 0xFFUL) or ((get().toULong() and 0xFFUL) shl 8) or ((get().toULong() and 0xFFUL) shl 16) + + for (cmp in 3 until (header - 247UL).toInt()) + res = res or ((get().toULong() and 0xFFUL) shl (cmp shl 3)) + + return res to (header - 247UL).toInt() +} + +fun String.readPacked(): Pair { + val header = parseHex(0) + if (header <= 240UL) return header to 1 + else if (header <= 248UL) return (240UL + ((header - 241UL) shl 8) + parseHex(1)) to 2 + else if (header == 249UL) return (2288UL + (parseHex(1) shl 8) + parseHex(2)) to 3 + + var res = parseHex(1) or (parseHex(2) shl 8) or (parseHex(3) shl 16) + + for (cmp in 3 until (header - 247UL).toInt()) + res = res or (parseHex(cmp + 1) shl (cmp shl 3)) + + return res to (header - 247UL).toInt() +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Ordering.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Ordering.kt new file mode 100644 index 0000000..e2b6f1d --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/Ordering.kt @@ -0,0 +1,15 @@ +package dev.w1zzrd.spigot.landmines2 + +import kotlin.reflect.KProperty1 + +fun T.compareByOrder(other: T, vararg comparables: KProperty1>): Int where T: Comparable { + for (comparable in comparables) { + @Suppress("UNCHECKED_CAST") + val result = (comparable(this) as Comparable).compareTo(comparable(other)) + + if (result != 0) + return result + } + + return 0 +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SerializableLocation.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SerializableLocation.kt new file mode 100644 index 0000000..033e09b --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SerializableLocation.kt @@ -0,0 +1,74 @@ +package dev.w1zzrd.spigot.landmines2 + +import org.bukkit.Location +import org.bukkit.Server +import java.nio.ByteBuffer +import java.util.* + +// Low-overhead, thread-safe serialization +private val threadLocalBuffer = ThreadLocal.withInitial { ByteBuffer.allocate(31) } + + +fun parseLocation(encoded: String): SerializableLocation { + val buffer = threadLocalBuffer.get() + buffer.position(0) + + Base64.getDecoder().decode(encoded.toByteArray(Charsets.ISO_8859_1), buffer.array()) + + return SerializableLocation(UUID(buffer.long, buffer.long), buffer.packedInt, buffer.packedInt, buffer.packedInt) +} + +fun unpackLocation(buffer: ByteBuffer, worlds: (UInt) -> UUID): SerializableLocation { + return SerializableLocation(worlds(buffer.packedUInt), buffer.packedInt, buffer.packedInt, buffer.packedInt) +} + +val Location.serializable: SerializableLocation + get() = SerializableLocation(world!!.uid, blockX, blockY, blockZ) + +class SerializableLocation(val world: UUID, val x: Int, val y: Int, val z: Int): Comparable { + fun getBukkitLocation(server: Server) = Location(server.getWorld(world), x.toDouble(), y.toDouble(), z.toDouble()) + + override fun compareTo(other: SerializableLocation) = + compareByOrder( + other, + SerializableLocation::world, + SerializableLocation::x, + SerializableLocation::y, + SerializableLocation::z + ) + + override fun equals(other: Any?) = + other is SerializableLocation && + world == other.world && + x == other.x && + y == other.y && + z == other.z + + override fun toString(): String { + val buffer = threadLocalBuffer.get() + buffer.position(0) + + buffer.putLong(world.mostSignificantBits) + buffer.putLong(world.leastSignificantBits) + buffer.writePacked(x) + buffer.writePacked(y) + buffer.writePacked(z) + + return Base64.getEncoder().withoutPadding().encodeToString(Arrays.copyOf(buffer.array(), buffer.position())) + } + + fun writePacked(buffer: ByteBuffer, worlds: (UUID) -> UInt) { + buffer.writePacked(worlds(world)) + buffer.writePacked(x) + buffer.writePacked(y) + buffer.writePacked(z) + } + + override fun hashCode(): Int { + var result = world.hashCode() + result = 31 * result + x + result = 31 * result + y + result = 31 * result + z + return result + } +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SortedList.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SortedList.kt new file mode 100644 index 0000000..071027e --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/SortedList.kt @@ -0,0 +1,89 @@ +package dev.w1zzrd.spigot.landmines2 + +import java.util.* + +class SortedList private constructor( + private val underlying: MutableList, + private val comparator: Comparator +): MutableList by underlying { + + companion object { + fun create( + type: Class, + underlying: MutableList, + comparator: Comparator + ) = SortedList(Collections.checkedList(underlying, type), comparator) + + inline fun > create( + underlying: MutableList = ArrayList(), + comparator: Comparator = Comparator { a, b -> a.compareTo(b) } + ) = create(T::class.java, underlying, comparator) + + inline fun create(comparator: Comparator, underlying: MutableList = ArrayList()) = + create(T::class.java, underlying, comparator) + } + + init { + if (underlying.size > 0) + underlying.sortWith(comparator) + } + + override fun add(element: E): Boolean { + val index = underlying.binarySearch(element, comparator) + + if (index < 0) + underlying.add(-(index + 1), element) + else + underlying[index] = element + + return index < 0 + } + + override fun add(index: Int, element: E) = + throw UnsupportedOperationException("Cannot insert at index for sorted list") + + override fun addAll(elements: Collection): Boolean { + var result = false + + for (element in elements) + result = add(element) || result + + return result + } + + override fun addAll(index: Int, elements: Collection) = + throw UnsupportedOperationException("Cannot insert at index for sorted list") + + override fun contains(element: E) = underlying.binarySearch(element, comparator) >= 0 + + override fun containsAll(elements: Collection): Boolean { + for (element in elements) + if (!contains(element)) + return false + + return true + } + + override fun remove(element: E): Boolean { + val index = underlying.binarySearch(element, comparator) + + if (index >= 0) { + underlying.removeAt(index) + return true + } + + return false + } + + override fun removeAt(index: Int) = + underlying.removeAt(index) + + override fun removeAll(elements: Collection): Boolean { + var result = false + + for (element in elements) + result = remove(element) || result + + return result + } +} \ No newline at end of file diff --git a/src/main/kotlin/dev/w1zzrd/spigot/landmines2/YamlFile.kt b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/YamlFile.kt new file mode 100644 index 0000000..c21a9cf --- /dev/null +++ b/src/main/kotlin/dev/w1zzrd/spigot/landmines2/YamlFile.kt @@ -0,0 +1,19 @@ +package dev.w1zzrd.spigot.landmines2 + +import org.bukkit.configuration.Configuration +import org.bukkit.configuration.ConfigurationSection +import org.bukkit.configuration.file.YamlConfiguration +import java.io.File + +class YamlFile( + private val file: File, + private val conf: YamlConfiguration = YamlConfiguration.loadConfiguration(file) +): ConfigurationSection, Configuration by conf { + private var firstLoad = false + + fun save() = conf.save(file) + fun reload() { + if (firstLoad) firstLoad = false + else if (file.isFile) conf.load(file) + } +} \ No newline at end of file diff --git a/src/main/resources/config.yml b/src/main/resources/config.yml new file mode 100644 index 0000000..56cce98 --- /dev/null +++ b/src/main/resources/config.yml @@ -0,0 +1,3 @@ +# Leave gcTrigger alone unless you know what you are doing +gcTrigger: 10000 +explosionStrength: 2.0 \ No newline at end of file