Implement new features in PortalManager

This commit is contained in:
Gabriel Tofvesson 2021-09-24 02:14:38 +02:00
parent 4b91481e96
commit b25f91c3e3

View File

@ -1,22 +1,37 @@
import org.bukkit.Location import org.bukkit.Location
import org.bukkit.OfflinePlayer
import org.bukkit.configuration.ConfigurationSection import org.bukkit.configuration.ConfigurationSection
import org.bukkit.event.EventHandler import org.bukkit.event.EventHandler
import org.bukkit.event.HandlerList import org.bukkit.event.HandlerList
import org.bukkit.event.Listener import org.bukkit.event.Listener
import org.bukkit.event.player.PlayerMoveEvent import org.bukkit.event.player.PlayerMoveEvent
import org.bukkit.plugin.Plugin import org.bukkit.plugin.Plugin
import java.lang.Long.max
import java.util.* import java.util.*
import kotlin.collections.ArrayList
private const val PATH_PLAYERS = "players" private const val PATH_DATA_PLAYERS = "players"
private const val PATH_WORLDS = "worlds" private const val PATH_DATA_WORLDS = "worlds"
private const val PATH_PORTALS = "portals" private const val PATH_DATA_PORTALS = "portals"
private const val PATH_DATA_INVITES = "invites"
private const val PATH_CONFIG_COOLDOWN = "playerTeleportCooldownTicks"
private const val DEFAULT_COOLDOWN = 100L
private const val DEFAULT_COOLDOWN_MIN = 5L
// TODO: Adapt to proper DBMS because this is depressing garbage
class PortalManager(private val data: ConfigurationSection, private val config: () -> ConfigurationSection): Listener { class PortalManager(private val data: ConfigurationSection, private val config: () -> ConfigurationSection): Listener {
private val players = PlayerMapper(data, PATH_PLAYERS) private val players = PlayerMapper(data, PATH_DATA_PLAYERS)
private val worlds = WorldMapper(data, PATH_WORLDS) private val worlds = WorldMapper(data, PATH_DATA_WORLDS)
private var portals = MultiSortedList(ArrayList(), PORTAL_LOCATION_COMPARATOR, PORTAL_UID_COMPARATOR) private var portals = MultiSortedList(ArrayList(), COMPARATOR_PORTAL_LOCATION_OWNER, COMPARATOR_PORTAL_UID, COMPARATOR_PORTAL_OWNER_NAME, COMPARATOR_PORTAL_LINKS)
private var invitations = MultiSortedList(ArrayList(), COMPARATOR_INVITE_RECIPIENT, COMPARATOR_INVITE_PORTAL)
private val cooldowns = LinkedList<Pair<OfflinePlayer, Long>>()
private val cooldownsLookup = SortedList<OfflinePlayer>(ArrayList(), COMPARATOR_PLAYER)
private var cooldownTime = DEFAULT_COOLDOWN
// Make UUIDs as "sequential" as possible // Make UUIDs as "sequential" as possible
private var nextUUIDUsed = false private var nextUUIDUsed = false
@ -33,7 +48,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
var msb = field.mostSignificantBits.toULong() var msb = field.mostSignificantBits.toULong()
// Start sequential search at the resulting index if it is populated // Start sequential search at the resulting index if it is populated
val index = portals.binSearch(PORTAL_UID_COMPARATOR) { val index = portals.binSearch(COMPARATOR_PORTAL_UID) {
compareValues( compareValues(
{ msb } to { it.id.mostSignificantBits.toULong() }, { msb } to { it.id.mostSignificantBits.toULong() },
{ lsb } to { it.id.leastSignificantBits.toULong() } { lsb } to { it.id.leastSignificantBits.toULong() }
@ -46,7 +61,7 @@ class PortalManager(private val data: ConfigurationSection, private val config:
++msb ++msb
for (i in index until portals.size) { for (i in index until portals.size) {
val find = portals.get(index, PORTAL_UID_COMPARATOR).id val find = portals.get(index, COMPARATOR_PORTAL_UID).id
// Found a gap in the UUIDs // Found a gap in the UUIDs
if (find.mostSignificantBits.toULong() != msb || find.leastSignificantBits.toULong() != lsb) if (find.mostSignificantBits.toULong() != msb || find.leastSignificantBits.toULong() != lsb)
@ -64,33 +79,41 @@ class PortalManager(private val data: ConfigurationSection, private val config:
} }
fun reload() { fun reload() {
cooldownTime = max(config().getLong(PATH_CONFIG_COOLDOWN), DEFAULT_COOLDOWN_MIN)
players.reload() players.reload()
worlds.reload() worlds.reload()
val portalList = ArrayList<Portal>() val portalList = ArrayList<Portal>()
data.getStringList(PATH_PORTALS).forEach { data.getStringList(PATH_DATA_PORTALS).forEach {
val portal = readCompressedPortal(it, worlds::getValue, players::getValue) ?: return@forEach val portal = readCompressedPortal(it, worlds::getValue, players::getValue) ?: return@forEach
portalList += portal portalList += portal
if (portal.id >= nextUUID) if (portal.id >= nextUUID)
nextUUID = portal.id + 1UL nextUUID = portal.id + 1UL
} }
portals = MultiSortedList(portalList, PORTAL_LOCATION_COMPARATOR, PORTAL_UID_COMPARATOR) portals = MultiSortedList(portalList, COMPARATOR_PORTAL_LOCATION_OWNER, COMPARATOR_PORTAL_UID)
if(portals.isEmpty()) nextUUID = UUID(0, 0) if(portals.isEmpty()) nextUUID = UUID(0, 0)
else { else {
nextUUID = portals.get(0, PORTAL_UID_COMPARATOR).id + 1UL nextUUID = portals.get(0, COMPARATOR_PORTAL_UID).id + 1UL
// Compute next UUID // Compute next UUID
nextUUID nextUUID
nextUUIDUsed = false nextUUIDUsed = false
} }
invitations = MultiSortedList(
data.getStringList(PATH_DATA_INVITES).mapTo(ArrayList(), ::Invite),
COMPARATOR_INVITE_RECIPIENT,
COMPARATOR_INVITE_PORTAL
)
} }
fun save() { fun save() {
players.save() players.save()
worlds.save() worlds.save()
data.set(PATH_PORTALS, portals.map { it.toCompressedString(worlds::getIndex, players::getIndex) }) data.set(PATH_DATA_PORTALS, portals.map { it.toCompressedString(worlds::getIndex, players::getIndex) })
} }
fun onEnable(plugin: Plugin) { fun onEnable(plugin: Plugin) {
@ -101,40 +124,144 @@ class PortalManager(private val data: ConfigurationSection, private val config:
HandlerList.unregisterAll(this) HandlerList.unregisterAll(this)
} }
fun getInvitationsForPlayer(player: OfflinePlayer) =
invitations.getAll(COMPARATOR_INVITE_RECIPIENT) { player.uniqueId.compareTo(it.recipient.uniqueId) }
fun getInvitationsForPortal(portalID: UUID) =
invitations.getAll(COMPARATOR_INVITE_PORTAL) { portalID.compareTo(it.portalID) }
fun getInvitationsForPortal(portal: Portal) = getInvitationsForPortal(portal.id)
// This is a perfect example of why mysql is better
fun getInvitationsForPlayerFromPlayer(recipient: OfflinePlayer, sender: OfflinePlayer) =
invitations.getAll(COMPARATOR_INVITE_RECIPIENT) {
compareValues(
recipient::getUniqueId to it.recipient::getUniqueId,
portals.get(
portals.binSearch(COMPARATOR_PORTAL_UID, it.portalID.COMPARISON_PORTAL_ID), COMPARATOR_PORTAL_UID
).owner::getUniqueId to sender::getUniqueId
)
}
fun invitePlayer(player: OfflinePlayer, portal: Portal): Boolean {
// Player is already invited or already has a pending invitation
if (player in portal.accessExclusions || invitations.binSearch(COMPARATOR_INVITE_RECIPIENT) {
compareValues(player::getUniqueId to it.recipient::getUniqueId, portal::id to it::portalID)
} >= 0)
return false
invitations += Invite(player, portal)
return true
}
fun cancelInvite(player: OfflinePlayer, portal: Portal): Boolean {
val index = invitations.binSearch(COMPARATOR_INVITE_RECIPIENT) {
compareValues(
player::getUniqueId to it.recipient::getUniqueId,
portal::id to it::portalID
)
}
if (index < 0) return false
invitations.removeAt(index, COMPARATOR_INVITE_RECIPIENT)
return true
}
fun acceptInvite(player: OfflinePlayer, portal: Portal): Boolean {
if (!cancelInvite(player, portal)) return false
portal.accessExclusions += player
return true
}
fun declineInvite(player: OfflinePlayer, portal: Portal) = cancelInvite(player, portal)
// This makes me cry
fun makePortal(portal: Portal) =
!portals.contains(portal, COMPARATOR_PORTAL_OWNER_NAME) &&
!portals.contains(portal, COMPARATOR_PORTAL_LOCATION_OWNER) &&
portals.add(portal)
fun clearInvites(portal: Portal) = clearInvites(COMPARATOR_INVITE_PORTAL, portal.COMPARISON_INVITE)
fun clearInvites(recipient: OfflinePlayer) = clearInvites(COMPARATOR_INVITE_RECIPIENT, recipient.COMPARISON_INVITE)
private fun clearInvites(comparator: Comparator<Invite>, comparison: Comparison<Invite>) =
invitations.getAll(comparator, comparison)?.forEach(invitations::remove)
fun removePortal(owner: OfflinePlayer, name: String): Boolean {
val index = portals.binSearch(COMPARATOR_PORTAL_OWNER_NAME) {
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
}
if (index < 0) return false
// Remove invites linked to this portal
clearInvites(portals.get(index, COMPARATOR_PORTAL_OWNER_NAME))
val removed = portals.removeAt(index, COMPARATOR_PORTAL_OWNER_NAME)
// Unlink portals
portals.getAll(COMPARATOR_PORTAL_LINKS, removed.COMPARISON_PORTAL_LINKEDTO)?.forEach(Portal::unlink)
return true
}
fun getPortal(owner: OfflinePlayer, name: String): Portal? {
val index = portals.binSearch(COMPARATOR_PORTAL_OWNER_NAME) {
compareValues(owner::getUniqueId to it.owner::getUniqueId, { name } to it::name)
}
if (index < 0) return null
return portals.get(index, COMPARATOR_PORTAL_OWNER_NAME)
}
fun getPortalsAt(location: Location) =
portals.getAll(COMPARATOR_PORTAL_LOCATION_OWNER, location.COMPARISON_PORTAL)
private fun popCooldowns() {
val time = System.currentTimeMillis()
while (true) {
val front = cooldowns.first
if (front.second < time) {
cooldowns.removeFirst()
cooldownsLookup.remove(front.first)
}
else break
}
}
private fun isOnCooldown(player: OfflinePlayer): Boolean {
popCooldowns()
return cooldownsLookup.contains(player)
}
private fun triggerCooldown(player: OfflinePlayer) {
cooldowns.addLast(Pair(player, System.currentTimeMillis() + cooldownTime))
}
@EventHandler @EventHandler
fun onPlayerMove(moveEvent: PlayerMoveEvent) { fun onPlayerMove(moveEvent: PlayerMoveEvent) {
// If we're ignoring player movements for this player, just return immediately
if (isOnCooldown(moveEvent.player)) return
fun UUID.portalMapper() = portals.firstOrNull { it.id == this } fun UUID.portalMapper() = portals.firstOrNull { it.id == this }
val to = moveEvent.to val to = moveEvent.to
if (!moveEvent.isCancelled && to != null) { if (!moveEvent.isCancelled && to != null) {
val found = getPortalsAt(to) val found = getPortalsAt(to)
found?.firstOrNull { it.owner.uniqueId == moveEvent.player.uniqueId } if ((found?.firstOrNull { it.owner.uniqueId == moveEvent.player.uniqueId }
?.enterPortal(moveEvent.player, UUID::portalMapper) ?.enterPortal(moveEvent.player, UUID::portalMapper)
?: found?.firstOrNull { it.enterPortal(moveEvent.player, UUID::portalMapper) == PortalResult.SUCCESS } ?: found?.firstOrNull {
it.enterPortal(
moveEvent.player,
UUID::portalMapper
) == PortalResult.SUCCESS
}) != null)
triggerCooldown(moveEvent.player)
} }
} }
// This is a very hot function: allocate with extreme care!
fun getPortalsAt(location: Location): LinkedList<Portal>? {
fun portalFinder(portal: Portal) =
compareValues(
location.world!!::getUID to portal.world::getUID,
location::getBlockX to portal::x,
location::getBlockY to portal::y,
location::getBlockZ to portal::z
)
// Don't allocate list unless there is data
var index = portals.binarySearch(comparison = ::portalFinder)
if (index < 0) return null
val result = LinkedList<Portal>()
do result += portals[index]
while (++index < portals.size && portalFinder(portals[index]) == 0)
return result
}
} }