mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-21 15:51:28 +01:00
thank you for your service, old spigot mapped patches, you will not be missed.
This commit is contained in:
parent
ac95f25f35
commit
512e7c99ae
@ -1,291 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Mon, 29 Feb 2016 20:40:33 -0600
|
|
||||||
Subject: [PATCH] POM Changes
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/pom.xml b/pom.xml
|
|
||||||
index 3fc047371e8f8a626e69697fad549d689c5dce89..a5d87d22cb1588d15e08da3b37e51c5e261c7799 100644
|
|
||||||
--- a/pom.xml
|
|
||||||
+++ b/pom.xml
|
|
||||||
@@ -1,15 +1,14 @@
|
|
||||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
|
||||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
|
|
||||||
<modelVersion>4.0.0</modelVersion>
|
|
||||||
- <groupId>org.spigotmc</groupId>
|
|
||||||
- <artifactId>spigot</artifactId>
|
|
||||||
+ <artifactId>paper</artifactId>
|
|
||||||
<packaging>jar</packaging>
|
|
||||||
<version>1.16.5-R0.1-SNAPSHOT</version>
|
|
||||||
- <name>Spigot</name>
|
|
||||||
- <url>https://www.spigotmc.org/</url>
|
|
||||||
+ <name>Paper</name>
|
|
||||||
+ <url>https://papermc.io</url>
|
|
||||||
|
|
||||||
<properties>
|
|
||||||
- <skipTests>true</skipTests>
|
|
||||||
+ <!-- <skipTests>true</skipTests> Paper - This [was] not going to end well -->
|
|
||||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
|
||||||
<api.version>unknown</api.version>
|
|
||||||
<bt.name>git</bt.name>
|
|
||||||
@@ -20,21 +19,39 @@
|
|
||||||
</properties>
|
|
||||||
|
|
||||||
<parent>
|
|
||||||
- <groupId>org.spigotmc</groupId>
|
|
||||||
- <artifactId>spigot-parent</artifactId>
|
|
||||||
+ <groupId>com.destroystokyo.paper</groupId>
|
|
||||||
+ <artifactId>paper-parent</artifactId>
|
|
||||||
<version>dev-SNAPSHOT</version>
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
|
||||||
</parent>
|
|
||||||
|
|
||||||
+ <dependencyManagement>
|
|
||||||
+ <dependencies>
|
|
||||||
+ <dependency>
|
|
||||||
+ <groupId>org.apache.logging.log4j</groupId>
|
|
||||||
+ <artifactId>log4j-bom</artifactId>
|
|
||||||
+ <version>2.11.2</version>
|
|
||||||
+ <type>pom</type>
|
|
||||||
+ <scope>import</scope>
|
|
||||||
+ </dependency>
|
|
||||||
+ </dependencies>
|
|
||||||
+ </dependencyManagement>
|
|
||||||
+
|
|
||||||
<dependencies>
|
|
||||||
<dependency>
|
|
||||||
- <groupId>org.spigotmc</groupId>
|
|
||||||
- <artifactId>spigot-api</artifactId>
|
|
||||||
+ <groupId>com.destroystokyo.paper</groupId>
|
|
||||||
+ <artifactId>paper-api</artifactId>
|
|
||||||
+ <version>${project.version}</version>
|
|
||||||
+ <scope>compile</scope>
|
|
||||||
+ </dependency>
|
|
||||||
+ <dependency>
|
|
||||||
+ <groupId>com.destroystokyo.paper</groupId>
|
|
||||||
+ <artifactId>paper-mojangapi</artifactId>
|
|
||||||
<version>${project.version}</version>
|
|
||||||
<scope>compile</scope>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
- <groupId>org.spigotmc</groupId>
|
|
||||||
+ <groupId>io.papermc</groupId>
|
|
||||||
<artifactId>minecraft-server</artifactId>
|
|
||||||
<version>${minecraft.version}-SNAPSHOT</version>
|
|
||||||
<scope>compile</scope>
|
|
||||||
@@ -45,18 +62,15 @@
|
|
||||||
<version>2.12.1</version>
|
|
||||||
<scope>compile</scope>
|
|
||||||
</dependency>
|
|
||||||
+ <dependency>
|
|
||||||
+ <groupId>org.apache.logging.log4j</groupId>
|
|
||||||
+ <artifactId>log4j-api</artifactId>
|
|
||||||
+ <scope>compile</scope>
|
|
||||||
+ </dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.apache.logging.log4j</groupId>
|
|
||||||
<artifactId>log4j-iostreams</artifactId>
|
|
||||||
- <version>2.8.1</version>
|
|
||||||
<scope>compile</scope>
|
|
||||||
- <exclusions>
|
|
||||||
- <!-- included in minecraft-server -->
|
|
||||||
- <exclusion>
|
|
||||||
- <groupId>org.apache.logging.log4j</groupId>
|
|
||||||
- <artifactId>log4j-api</artifactId>
|
|
||||||
- </exclusion>
|
|
||||||
- </exclusions>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.ow2.asm</groupId>
|
|
||||||
@@ -64,12 +78,23 @@
|
|
||||||
<version>9.1</version>
|
|
||||||
<scope>compile</scope>
|
|
||||||
</dependency>
|
|
||||||
+ <dependency>
|
|
||||||
+ <!-- wrapper to use either java 8 sun cleaner or java9+ official cleaner -->
|
|
||||||
+ <groupId>co.aikar</groupId>
|
|
||||||
+ <artifactId>cleaner</artifactId>
|
|
||||||
+ <version>1.0-SNAPSHOT</version>
|
|
||||||
+ </dependency>
|
|
||||||
+ <dependency>
|
|
||||||
+ <groupId>io.netty</groupId>
|
|
||||||
+ <artifactId>netty-all</artifactId>
|
|
||||||
+ <version>4.1.50.Final</version>
|
|
||||||
+ </dependency>
|
|
||||||
<!-- deprecated API depend -->
|
|
||||||
<dependency>
|
|
||||||
<groupId>com.googlecode.json-simple</groupId>
|
|
||||||
<artifactId>json-simple</artifactId>
|
|
||||||
<version>1.1.1</version>
|
|
||||||
- <scope>runtime</scope>
|
|
||||||
+ <scope>compile</scope>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.xerial</groupId>
|
|
||||||
@@ -80,7 +105,7 @@
|
|
||||||
<dependency>
|
|
||||||
<groupId>mysql</groupId>
|
|
||||||
<artifactId>mysql-connector-java</artifactId>
|
|
||||||
- <version>5.1.49</version>
|
|
||||||
+ <version>8.0.23</version>
|
|
||||||
<scope>runtime</scope>
|
|
||||||
</dependency>
|
|
||||||
<!-- add these back in as they are not exposed by the API -->
|
|
||||||
@@ -105,7 +130,7 @@
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.apache.logging.log4j</groupId>
|
|
||||||
<artifactId>log4j-slf4j-impl</artifactId>
|
|
||||||
- <version>2.8.1</version>
|
|
||||||
+ <!--<version>2.8.1</version>--> <!--handled by bom-->
|
|
||||||
<scope>runtime</scope>
|
|
||||||
<exclusions>
|
|
||||||
<!-- included in minecraft-server -->
|
|
||||||
@@ -132,34 +157,22 @@
|
|
||||||
|
|
||||||
<!-- This builds a completely 'ready to start' jar with all dependencies inside -->
|
|
||||||
<build>
|
|
||||||
+ <finalName>paper-${minecraft.version}</finalName>
|
|
||||||
+ <defaultGoal>clean install</defaultGoal> <!-- Paper -->
|
|
||||||
<plugins>
|
|
||||||
<plugin>
|
|
||||||
- <groupId>net.md-5</groupId>
|
|
||||||
- <artifactId>scriptus</artifactId>
|
|
||||||
- <version>0.4.1</version>
|
|
||||||
+ <groupId>com.lukegb.mojo</groupId>
|
|
||||||
+ <artifactId>gitdescribe-maven-plugin</artifactId>
|
|
||||||
+ <version>1.3</version>
|
|
||||||
+ <configuration>
|
|
||||||
+ <outputPrefix>git-Paper-</outputPrefix>
|
|
||||||
+ <scmDirectory>..</scmDirectory>
|
|
||||||
+ </configuration>
|
|
||||||
<executions>
|
|
||||||
<execution>
|
|
||||||
- <id>ex-spigot</id>
|
|
||||||
- <configuration>
|
|
||||||
- <format>${bt.name}-Spigot-%s</format>
|
|
||||||
- <scmDirectory>../</scmDirectory>
|
|
||||||
- <descriptionProperty>spigot.desc</descriptionProperty>
|
|
||||||
- </configuration>
|
|
||||||
- <phase>initialize</phase>
|
|
||||||
- <goals>
|
|
||||||
- <goal>describe</goal>
|
|
||||||
- </goals>
|
|
||||||
- </execution>
|
|
||||||
- <execution>
|
|
||||||
- <id>ex-craftbukkit</id>
|
|
||||||
- <configuration>
|
|
||||||
- <format>-%s</format>
|
|
||||||
- <scmDirectory>../../CraftBukkit</scmDirectory>
|
|
||||||
- <descriptionProperty>craftbukkit.desc</descriptionProperty>
|
|
||||||
- </configuration>
|
|
||||||
- <phase>initialize</phase>
|
|
||||||
+ <phase>compile</phase>
|
|
||||||
<goals>
|
|
||||||
- <goal>describe</goal>
|
|
||||||
+ <goal>gitdescribe</goal>
|
|
||||||
</goals>
|
|
||||||
</execution>
|
|
||||||
</executions>
|
|
||||||
@@ -169,6 +182,7 @@
|
|
||||||
<artifactId>maven-jar-plugin</artifactId>
|
|
||||||
<version>3.2.0</version>
|
|
||||||
<configuration>
|
|
||||||
+ <forceCreation>true</forceCreation> <!-- Required to prevent shading the jar multiple times -->
|
|
||||||
<archive>
|
|
||||||
<manifest>
|
|
||||||
<addDefaultEntries>false</addDefaultEntries>
|
|
||||||
@@ -176,11 +190,13 @@
|
|
||||||
<manifestEntries>
|
|
||||||
<Main-Class>org.bukkit.craftbukkit.Main</Main-Class>
|
|
||||||
<Implementation-Title>CraftBukkit</Implementation-Title>
|
|
||||||
- <Implementation-Version>${spigot.desc}${craftbukkit.desc}</Implementation-Version>
|
|
||||||
- <Implementation-Vendor>${project.build.outputTimestamp}</Implementation-Vendor>
|
|
||||||
+ <!--suppress MavenModelInspection -->
|
|
||||||
+ <Implementation-Version>${describe}</Implementation-Version>
|
|
||||||
+ <Implementation-Vendor>${maven.build.timestamp}</Implementation-Vendor>
|
|
||||||
<Specification-Title>Bukkit</Specification-Title>
|
|
||||||
<Specification-Version>${api.version}</Specification-Version>
|
|
||||||
<Specification-Vendor>Bukkit Team</Specification-Vendor>
|
|
||||||
+ <Multi-Release>true</Multi-Release>
|
|
||||||
</manifestEntries>
|
|
||||||
<manifestSections>
|
|
||||||
<manifestSection>
|
|
||||||
@@ -216,14 +232,24 @@
|
|
||||||
<goal>shade</goal>
|
|
||||||
</goals>
|
|
||||||
<configuration>
|
|
||||||
+ <dependencyReducedPomLocation>${project.build.directory}/dependency-reduced-pom.xml</dependencyReducedPomLocation> <!-- Paper -->
|
|
||||||
<createSourcesJar>${shadeSourcesJar}</createSourcesJar>
|
|
||||||
<filters>
|
|
||||||
<filter>
|
|
||||||
- <artifact>org.spigotmc:minecraft-server</artifact>
|
|
||||||
+ <artifact>io.papermc:minecraft-server</artifact>
|
|
||||||
<excludes>
|
|
||||||
<exclude>com/google/common/**</exclude>
|
|
||||||
<exclude>com/google/gson/**</exclude>
|
|
||||||
<exclude>com/google/thirdparty/**</exclude>
|
|
||||||
+ <!-- paper -->
|
|
||||||
+ <exclude>io/netty/**</exclude>
|
|
||||||
+ <exclude>META-INF/native/libnetty*</exclude>
|
|
||||||
+ <exclude>com/mojang/brigadier/**</exclude>
|
|
||||||
+ <exclude>META-INF/MANIFEST.MF</exclude>
|
|
||||||
+ <exclude>com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.class</exclude>
|
|
||||||
+ <exclude>com/mojang/datafixers/util/Either*</exclude>
|
|
||||||
+ <exclude>org/apache/logging/log4j/**</exclude>
|
|
||||||
+ <exclude>META-INF/org/apache/logging/log4j/core/config/plugins/Log4j2Plugins.dat</exclude>
|
|
||||||
</excludes>
|
|
||||||
</filter>
|
|
||||||
<filter>
|
|
||||||
@@ -245,10 +271,11 @@
|
|
||||||
<pattern>jline</pattern>
|
|
||||||
<shadedPattern>org.bukkit.craftbukkit.libs.jline</shadedPattern>
|
|
||||||
</relocation>
|
|
||||||
- <relocation>
|
|
||||||
- <pattern>it.unimi</pattern>
|
|
||||||
- <shadedPattern>org.bukkit.craftbukkit.libs.it.unimi</shadedPattern>
|
|
||||||
- </relocation>
|
|
||||||
+ <!-- Paper - Don't relocate fastutil in order to prevent api breakage -->
|
|
||||||
+ <!--<relocation>-->
|
|
||||||
+ <!--<pattern>it.unimi</pattern>-->
|
|
||||||
+ <!--<shadedPattern>org.bukkit.craftbukkit.libs.it.unimi</shadedPattern>-->
|
|
||||||
+ <!--</relocation>-->
|
|
||||||
<relocation>
|
|
||||||
<pattern>org.apache.commons.codec</pattern>
|
|
||||||
<shadedPattern>org.bukkit.craftbukkit.libs.org.apache.commons.codec</shadedPattern>
|
|
||||||
@@ -316,10 +343,6 @@
|
|
||||||
<groupId>org.apache.maven.plugins</groupId>
|
|
||||||
<artifactId>maven-compiler-plugin</artifactId>
|
|
||||||
<version>3.8.1</version>
|
|
||||||
- <configuration>
|
|
||||||
- <!-- we use the Eclipse compiler as it doesn't need a JDK -->
|
|
||||||
- <compilerId>eclipse</compilerId>
|
|
||||||
- </configuration>
|
|
||||||
<dependencies>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.codehaus.plexus</groupId>
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/Main.java b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
index 4452427d0a8298d119ca29ef397b7a94f19eec28..46a16e31775b28c44f95a8ac5545ebcb656c74b6 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
@@ -186,7 +186,7 @@ public class Main {
|
|
||||||
}
|
|
||||||
|
|
||||||
if (false && Main.class.getPackage().getImplementationVendor() != null && System.getProperty("IReallyKnowWhatIAmDoingISwear") == null) {
|
|
||||||
- Date buildDate = new Date(Integer.parseInt(Main.class.getPackage().getImplementationVendor()) * 1000L);
|
|
||||||
+ Date buildDate = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'").parse(Main.class.getPackage().getImplementationVendor()); // Paper
|
|
||||||
|
|
||||||
Calendar deadline = Calendar.getInstance();
|
|
||||||
deadline.add(Calendar.DAY_OF_YEAR, -28);
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/util/Versioning.java b/src/main/java/org/bukkit/craftbukkit/util/Versioning.java
|
|
||||||
index 93046379d0cefd5d3236fc59e698809acdc18f80..674096cab190d62622f9947853b056f57d43a2a5 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/util/Versioning.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/util/Versioning.java
|
|
||||||
@@ -11,7 +11,7 @@ public final class Versioning {
|
|
||||||
public static String getBukkitVersion() {
|
|
||||||
String result = "Unknown-Version";
|
|
||||||
|
|
||||||
- InputStream stream = Bukkit.class.getClassLoader().getResourceAsStream("META-INF/maven/org.spigotmc/spigot-api/pom.properties");
|
|
||||||
+ InputStream stream = Bukkit.class.getClassLoader().getResourceAsStream("META-INF/maven/com.destroystokyo.paper/paper-api/pom.properties");
|
|
||||||
Properties properties = new Properties();
|
|
||||||
|
|
||||||
if (stream != null) {
|
|
@ -1,820 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Mon, 29 Feb 2016 21:02:09 -0600
|
|
||||||
Subject: [PATCH] Paper config files
|
|
||||||
|
|
||||||
Loads each yml file for early init too so it can be used for early options
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperCommand.java b/src/main/java/com/destroystokyo/paper/PaperCommand.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..68cd4134cb6a00c1768100462f8e9e94f3fa6279
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperCommand.java
|
|
||||||
@@ -0,0 +1,286 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import com.google.common.base.Functions;
|
|
||||||
+import com.google.common.base.Joiner;
|
|
||||||
+import com.google.common.collect.ImmutableSet;
|
|
||||||
+import com.google.common.collect.Iterables;
|
|
||||||
+import com.google.common.collect.Lists;
|
|
||||||
+import com.google.common.collect.Maps;
|
|
||||||
+import net.minecraft.resources.MinecraftKey;
|
|
||||||
+import net.minecraft.server.MinecraftServer;
|
|
||||||
+import net.minecraft.server.level.ChunkProviderServer;
|
|
||||||
+import net.minecraft.server.level.WorldServer;
|
|
||||||
+import net.minecraft.world.entity.Entity;
|
|
||||||
+import net.minecraft.world.entity.EntityTypes;
|
|
||||||
+import net.minecraft.world.level.ChunkCoordIntPair;
|
|
||||||
+import org.apache.commons.lang3.tuple.MutablePair;
|
|
||||||
+import org.apache.commons.lang3.tuple.Pair;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.ChatColor;
|
|
||||||
+import org.bukkit.Location;
|
|
||||||
+import org.bukkit.World;
|
|
||||||
+import org.bukkit.command.Command;
|
|
||||||
+import org.bukkit.command.CommandSender;
|
|
||||||
+import org.bukkit.craftbukkit.CraftServer;
|
|
||||||
+import org.bukkit.craftbukkit.CraftWorld;
|
|
||||||
+import org.bukkit.entity.Player;
|
|
||||||
+
|
|
||||||
+import java.io.File;
|
|
||||||
+import java.time.LocalDateTime;
|
|
||||||
+import java.time.format.DateTimeFormatter;
|
|
||||||
+import java.util.ArrayList;
|
|
||||||
+import java.util.Arrays;
|
|
||||||
+import java.util.Collection;
|
|
||||||
+import java.util.Collections;
|
|
||||||
+import java.util.Iterator;
|
|
||||||
+import java.util.List;
|
|
||||||
+import java.util.Locale;
|
|
||||||
+import java.util.Map;
|
|
||||||
+import java.util.Set;
|
|
||||||
+import java.util.stream.Collectors;
|
|
||||||
+
|
|
||||||
+public class PaperCommand extends Command {
|
|
||||||
+ private static final String BASE_PERM = "bukkit.command.paper.";
|
|
||||||
+ private static final ImmutableSet<String> SUBCOMMANDS = ImmutableSet.<String>builder().add("heap", "entity", "reload", "version").build();
|
|
||||||
+
|
|
||||||
+ public PaperCommand(String name) {
|
|
||||||
+ super(name);
|
|
||||||
+ this.description = "Paper related commands";
|
|
||||||
+ this.usageMessage = "/paper [" + Joiner.on(" | ").join(SUBCOMMANDS) + "]";
|
|
||||||
+ this.setPermission("bukkit.command.paper;" + Joiner.on(';').join(SUBCOMMANDS.stream().map(s -> BASE_PERM + s).collect(Collectors.toSet())));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static boolean testPermission(CommandSender commandSender, String permission) {
|
|
||||||
+ if (commandSender.hasPermission(BASE_PERM + permission) || commandSender.hasPermission("bukkit.command.paper")) return true;
|
|
||||||
+ commandSender.sendMessage(Bukkit.getPermissionMessage());
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public List<String> tabComplete(CommandSender sender, String alias, String[] args, Location location) throws IllegalArgumentException {
|
|
||||||
+ if (args.length <= 1)
|
|
||||||
+ return getListMatchingLast(sender, args, SUBCOMMANDS);
|
|
||||||
+
|
|
||||||
+ switch (args[0].toLowerCase(Locale.ENGLISH))
|
|
||||||
+ {
|
|
||||||
+ case "entity":
|
|
||||||
+ if (args.length == 2)
|
|
||||||
+ return getListMatchingLast(sender, args, "help", "list");
|
|
||||||
+ if (args.length == 3)
|
|
||||||
+ return getListMatchingLast(sender, args, EntityTypes.getEntityNameList().stream().map(MinecraftKey::toString).sorted().toArray(String[]::new));
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ return Collections.emptyList();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Code from Mojang - copyright them
|
|
||||||
+ public static List<String> getListMatchingLast(CommandSender sender, String[] args, String... matches) {
|
|
||||||
+ return getListMatchingLast(sender, args, (Collection) Arrays.asList(matches));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public static boolean matches(String s, String s1) {
|
|
||||||
+ return s1.regionMatches(true, 0, s, 0, s.length());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public static List<String> getListMatchingLast(CommandSender sender, String[] strings, Collection<?> collection) {
|
|
||||||
+ String last = strings[strings.length - 1];
|
|
||||||
+ ArrayList<String> results = Lists.newArrayList();
|
|
||||||
+
|
|
||||||
+ if (!collection.isEmpty()) {
|
|
||||||
+ Iterator iterator = Iterables.transform(collection, Functions.toStringFunction()).iterator();
|
|
||||||
+
|
|
||||||
+ while (iterator.hasNext()) {
|
|
||||||
+ String s1 = (String) iterator.next();
|
|
||||||
+
|
|
||||||
+ if (matches(last, s1) && (sender.hasPermission(BASE_PERM + s1) || sender.hasPermission("bukkit.command.paper"))) {
|
|
||||||
+ results.add(s1);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (results.isEmpty()) {
|
|
||||||
+ iterator = collection.iterator();
|
|
||||||
+
|
|
||||||
+ while (iterator.hasNext()) {
|
|
||||||
+ Object object = iterator.next();
|
|
||||||
+
|
|
||||||
+ if (object instanceof MinecraftKey && matches(last, ((MinecraftKey) object).getKey())) {
|
|
||||||
+ results.add(String.valueOf(object));
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return results;
|
|
||||||
+ }
|
|
||||||
+ // end copy stuff
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public boolean execute(CommandSender sender, String commandLabel, String[] args) {
|
|
||||||
+ if (!testPermission(sender)) return true;
|
|
||||||
+
|
|
||||||
+ if (args.length == 0) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Usage: " + usageMessage);
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+ if (SUBCOMMANDS.contains(args[0].toLowerCase(Locale.ENGLISH))) {
|
|
||||||
+ if (!testPermission(sender, args[0].toLowerCase(Locale.ENGLISH))) return true;
|
|
||||||
+ }
|
|
||||||
+ switch (args[0].toLowerCase(Locale.ENGLISH)) {
|
|
||||||
+ case "heap":
|
|
||||||
+ dumpHeap(sender);
|
|
||||||
+ break;
|
|
||||||
+ case "entity":
|
|
||||||
+ listEntities(sender, args);
|
|
||||||
+ break;
|
|
||||||
+ case "reload":
|
|
||||||
+ doReload(sender);
|
|
||||||
+ break;
|
|
||||||
+ case "ver":
|
|
||||||
+ if (!testPermission(sender, "version")) break; // "ver" needs a special check because it's an alias. All other commands are checked up before the switch statement (because they are present in the SUBCOMMANDS set)
|
|
||||||
+ case "version":
|
|
||||||
+ Command ver = org.bukkit.Bukkit.getServer().getCommandMap().getCommand("version");
|
|
||||||
+ if (ver != null) {
|
|
||||||
+ ver.execute(sender, commandLabel, new String[0]);
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ // else - fall through to default
|
|
||||||
+ default:
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Usage: " + usageMessage);
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return true;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /*
|
|
||||||
+ * Ported from MinecraftForge - author: LexManos <LexManos@gmail.com> - License: LGPLv2.1
|
|
||||||
+ */
|
|
||||||
+ private void listEntities(CommandSender sender, String[] args) {
|
|
||||||
+ if (args.length < 2 || args[1].toLowerCase(Locale.ENGLISH).equals("help")) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Use /paper entity [list] help for more information on a specific command.");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ switch (args[1].toLowerCase(Locale.ENGLISH)) {
|
|
||||||
+ case "list":
|
|
||||||
+ String filter = "*";
|
|
||||||
+ if (args.length > 2) {
|
|
||||||
+ if (args[2].toLowerCase(Locale.ENGLISH).equals("help")) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Use /paper entity list [filter] [worldName] to get entity info that matches the optional filter.");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ filter = args[2];
|
|
||||||
+ }
|
|
||||||
+ final String cleanfilter = filter.replace("?", ".?").replace("*", ".*?");
|
|
||||||
+ Set<MinecraftKey> names = EntityTypes.getEntityNameList().stream()
|
|
||||||
+ .filter(n -> n.toString().matches(cleanfilter))
|
|
||||||
+ .collect(Collectors.toSet());
|
|
||||||
+
|
|
||||||
+ if (names.isEmpty()) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Invalid filter, does not match any entities. Use /paper entity list for a proper list");
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Usage: /paper entity list [filter] [worldName]");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ String worldName;
|
|
||||||
+ if (args.length > 3) {
|
|
||||||
+ worldName = args[3];
|
|
||||||
+ } else if (sender instanceof Player) {
|
|
||||||
+ worldName = ((Player) sender).getWorld().getName();
|
|
||||||
+ } else {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Please specify the name of a world");
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "To do so without a filter, specify '*' as the filter");
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Usage: /paper entity list [filter] [worldName]");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ Map<MinecraftKey, MutablePair<Integer, Map<ChunkCoordIntPair, Integer>>> list = Maps.newHashMap();
|
|
||||||
+ World bukkitWorld = Bukkit.getWorld(worldName);
|
|
||||||
+ if (bukkitWorld == null) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Could not load world for " + worldName + ". Please select a valid world.");
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Usage: /paper entity list [filter] [worldName]");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ WorldServer world = ((CraftWorld) Bukkit.getWorld(worldName)).getHandle();
|
|
||||||
+
|
|
||||||
+ Map<MinecraftKey, Integer> nonEntityTicking = Maps.newHashMap();
|
|
||||||
+ ChunkProviderServer chunkProviderServer = world.getChunkProvider();
|
|
||||||
+
|
|
||||||
+ Collection<Entity> entities = world.entitiesById.values();
|
|
||||||
+ entities.forEach(e -> {
|
|
||||||
+ MinecraftKey key = new MinecraftKey(""); // TODO: update in next patch
|
|
||||||
+
|
|
||||||
+ MutablePair<Integer, Map<ChunkCoordIntPair, Integer>> info = list.computeIfAbsent(key, k -> MutablePair.of(0, Maps.newHashMap()));
|
|
||||||
+ ChunkCoordIntPair chunk = new ChunkCoordIntPair(e.chunkX, e.chunkZ);
|
|
||||||
+ info.left++;
|
|
||||||
+ info.right.put(chunk, info.right.getOrDefault(chunk, 0) + 1);
|
|
||||||
+ if (!chunkProviderServer.isInEntityTickingChunk(e)) {
|
|
||||||
+ nonEntityTicking.merge(key, Integer.valueOf(1), Integer::sum);
|
|
||||||
+ }
|
|
||||||
+ });
|
|
||||||
+
|
|
||||||
+ if (names.size() == 1) {
|
|
||||||
+ MinecraftKey name = names.iterator().next();
|
|
||||||
+ Pair<Integer, Map<ChunkCoordIntPair, Integer>> info = list.get(name);
|
|
||||||
+ int nonTicking = nonEntityTicking.getOrDefault(name, Integer.valueOf(0)).intValue();
|
|
||||||
+ if (info == null) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "No entities found.");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ sender.sendMessage("Entity: " + name + " Total Ticking: " + (info.getLeft() - nonTicking) + ", Total Non-Ticking: " + nonTicking);
|
|
||||||
+ info.getRight().entrySet().stream()
|
|
||||||
+ .sorted((a, b) -> !a.getValue().equals(b.getValue()) ? b.getValue() - a.getValue() : a.getKey().toString().compareTo(b.getKey().toString()))
|
|
||||||
+ .limit(10).forEach(e -> sender.sendMessage(" " + e.getValue() + ": " + e.getKey().x + ", " + e.getKey().z + (chunkProviderServer.isEntityTickingChunk(e.getKey()) ? " (Ticking)" : " (Non-Ticking)")));
|
|
||||||
+ } else {
|
|
||||||
+ List<Pair<MinecraftKey, Integer>> info = list.entrySet().stream()
|
|
||||||
+ .filter(e -> names.contains(e.getKey()))
|
|
||||||
+ .map(e -> Pair.of(e.getKey(), e.getValue().left))
|
|
||||||
+ .sorted((a, b) -> !a.getRight().equals(b.getRight()) ? b.getRight() - a.getRight() : a.getKey().toString().compareTo(b.getKey().toString()))
|
|
||||||
+ .collect(Collectors.toList());
|
|
||||||
+
|
|
||||||
+ if (info == null || info.size() == 0) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "No entities found.");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ int count = info.stream().mapToInt(Pair::getRight).sum();
|
|
||||||
+ int nonTickingCount = nonEntityTicking.values().stream().mapToInt(Integer::intValue).sum();
|
|
||||||
+ sender.sendMessage("Total Ticking: " + (count - nonTickingCount) + ", Total Non-Ticking: " + nonTickingCount);
|
|
||||||
+ info.forEach(e -> {
|
|
||||||
+ int nonTicking = nonEntityTicking.getOrDefault(e.getKey(), Integer.valueOf(0)).intValue();
|
|
||||||
+ sender.sendMessage(" " + (e.getValue() - nonTicking) + " (" + nonTicking + ") " + ": " + e.getKey());
|
|
||||||
+ });
|
|
||||||
+ sender.sendMessage("* First number is ticking entities, second number is non-ticking entities");
|
|
||||||
+ }
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void dumpHeap(CommandSender sender) {
|
|
||||||
+ java.nio.file.Path dir = java.nio.file.Paths.get("./dumps");
|
|
||||||
+ String name = "heap-dump-" + DateTimeFormatter.ofPattern("yyyy-MM-dd_HH.mm.ss").format(LocalDateTime.now());
|
|
||||||
+
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.YELLOW + "Writing JVM heap data...");
|
|
||||||
+
|
|
||||||
+ java.nio.file.Path file = CraftServer.dumpHeap(dir, name);
|
|
||||||
+ if (file != null) {
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.GREEN + "Heap dump saved to " + file);
|
|
||||||
+ } else {
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.RED + "Failed to write heap dump, see sever log for details");
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void doReload(CommandSender sender) {
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.RED + "Please note that this command is not supported and may cause issues.");
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.RED + "If you encounter any issues please use the /stop command to restart your server.");
|
|
||||||
+
|
|
||||||
+ MinecraftServer console = MinecraftServer.getServer();
|
|
||||||
+ com.destroystokyo.paper.PaperConfig.init((File) console.options.valueOf("paper-settings"));
|
|
||||||
+ for (WorldServer world : console.getWorlds()) {
|
|
||||||
+ world.paperConfig.init();
|
|
||||||
+ }
|
|
||||||
+ console.server.reloadCount++;
|
|
||||||
+
|
|
||||||
+ Command.broadcastCommandMessage(sender, ChatColor.GREEN + "Paper config reload complete.");
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..2c0514892d3993bef57ecf677cf8bb0fbe0216e4
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -0,0 +1,185 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import com.google.common.base.Throwables;
|
|
||||||
+
|
|
||||||
+import java.io.File;
|
|
||||||
+import java.io.IOException;
|
|
||||||
+import java.lang.reflect.InvocationTargetException;
|
|
||||||
+import java.lang.reflect.Method;
|
|
||||||
+import java.lang.reflect.Modifier;
|
|
||||||
+import java.util.HashMap;
|
|
||||||
+import java.util.List;
|
|
||||||
+import java.util.Map;
|
|
||||||
+import java.util.concurrent.TimeUnit;
|
|
||||||
+import java.util.logging.Level;
|
|
||||||
+import java.util.regex.Pattern;
|
|
||||||
+
|
|
||||||
+import net.minecraft.server.MinecraftServer;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.command.Command;
|
|
||||||
+import org.bukkit.configuration.ConfigurationSection;
|
|
||||||
+import org.bukkit.configuration.InvalidConfigurationException;
|
|
||||||
+import org.bukkit.configuration.file.YamlConfiguration;
|
|
||||||
+
|
|
||||||
+public class PaperConfig {
|
|
||||||
+
|
|
||||||
+ private static File CONFIG_FILE;
|
|
||||||
+ private static final String HEADER = "This is the main configuration file for Paper.\n"
|
|
||||||
+ + "As you can see, there's tons to configure. Some options may impact gameplay, so use\n"
|
|
||||||
+ + "with caution, and make sure you know what each option does before configuring.\n"
|
|
||||||
+ + "\n"
|
|
||||||
+ + "If you need help with the configuration or have any questions related to Paper,\n"
|
|
||||||
+ + "join us in our Discord or IRC channel.\n"
|
|
||||||
+ + "\n"
|
|
||||||
+ + "Discord: https://discord.gg/papermc\n"
|
|
||||||
+ + "IRC: #paper @ irc.esper.net ( https://webchat.esper.net/?channels=paper ) \n"
|
|
||||||
+ + "Website: https://papermc.io/ \n"
|
|
||||||
+ + "Docs: https://paper.readthedocs.org/ \n";
|
|
||||||
+ /*========================================================================*/
|
|
||||||
+ public static YamlConfiguration config;
|
|
||||||
+ static int version;
|
|
||||||
+ static Map<String, Command> commands;
|
|
||||||
+ private static boolean verbose;
|
|
||||||
+ private static boolean fatalError;
|
|
||||||
+ /*========================================================================*/
|
|
||||||
+
|
|
||||||
+ public static void init(File configFile) {
|
|
||||||
+ CONFIG_FILE = configFile;
|
|
||||||
+ config = new YamlConfiguration();
|
|
||||||
+ try {
|
|
||||||
+ config.load(CONFIG_FILE);
|
|
||||||
+ } catch (IOException ex) {
|
|
||||||
+ } catch (InvalidConfigurationException ex) {
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Could not load paper.yml, please correct your syntax errors", ex);
|
|
||||||
+ throw Throwables.propagate(ex);
|
|
||||||
+ }
|
|
||||||
+ config.options().header(HEADER);
|
|
||||||
+ config.options().copyDefaults(true);
|
|
||||||
+ verbose = getBoolean("verbose", false);
|
|
||||||
+
|
|
||||||
+ commands = new HashMap<String, Command>();
|
|
||||||
+ commands.put("paper", new PaperCommand("paper"));
|
|
||||||
+
|
|
||||||
+ version = getInt("config-version", 20);
|
|
||||||
+ set("config-version", 20);
|
|
||||||
+ readConfig(PaperConfig.class, null);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ protected static void logError(String s) {
|
|
||||||
+ Bukkit.getLogger().severe(s);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ protected static void fatal(String s) {
|
|
||||||
+ fatalError = true;
|
|
||||||
+ throw new RuntimeException("Fatal paper.yml config error: " + s);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ protected static void log(String s) {
|
|
||||||
+ if (verbose) {
|
|
||||||
+ Bukkit.getLogger().info(s);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public static void registerCommands() {
|
|
||||||
+ for (Map.Entry<String, Command> entry : commands.entrySet()) {
|
|
||||||
+ MinecraftServer.getServer().server.getCommandMap().register(entry.getKey(), "Paper", entry.getValue());
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ static void readConfig(Class<?> clazz, Object instance) {
|
|
||||||
+ for (Method method : clazz.getDeclaredMethods()) {
|
|
||||||
+ if (Modifier.isPrivate(method.getModifiers())) {
|
|
||||||
+ if (method.getParameterTypes().length == 0 && method.getReturnType() == Void.TYPE) {
|
|
||||||
+ try {
|
|
||||||
+ method.setAccessible(true);
|
|
||||||
+ method.invoke(instance);
|
|
||||||
+ } catch (InvocationTargetException ex) {
|
|
||||||
+ throw Throwables.propagate(ex.getCause());
|
|
||||||
+ } catch (Exception ex) {
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Error invoking " + method, ex);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ try {
|
|
||||||
+ config.save(CONFIG_FILE);
|
|
||||||
+ } catch (IOException ex) {
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Could not save " + CONFIG_FILE, ex);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static final Pattern SPACE = Pattern.compile(" ");
|
|
||||||
+ private static final Pattern NOT_NUMERIC = Pattern.compile("[^-\\d.]");
|
|
||||||
+ public static int getSeconds(String str) {
|
|
||||||
+ str = SPACE.matcher(str).replaceAll("");
|
|
||||||
+ final char unit = str.charAt(str.length() - 1);
|
|
||||||
+ str = NOT_NUMERIC.matcher(str).replaceAll("");
|
|
||||||
+ double num;
|
|
||||||
+ try {
|
|
||||||
+ num = Double.parseDouble(str);
|
|
||||||
+ } catch (Exception e) {
|
|
||||||
+ num = 0D;
|
|
||||||
+ }
|
|
||||||
+ switch (unit) {
|
|
||||||
+ case 'd': num *= (double) 60*60*24; break;
|
|
||||||
+ case 'h': num *= (double) 60*60; break;
|
|
||||||
+ case 'm': num *= (double) 60; break;
|
|
||||||
+ default: case 's': break;
|
|
||||||
+ }
|
|
||||||
+ return (int) num;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ protected static String timeSummary(int seconds) {
|
|
||||||
+ String time = "";
|
|
||||||
+
|
|
||||||
+ if (seconds > 60 * 60 * 24) {
|
|
||||||
+ time += TimeUnit.SECONDS.toDays(seconds) + "d";
|
|
||||||
+ seconds %= 60 * 60 * 24;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (seconds > 60 * 60) {
|
|
||||||
+ time += TimeUnit.SECONDS.toHours(seconds) + "h";
|
|
||||||
+ seconds %= 60 * 60;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (seconds > 0) {
|
|
||||||
+ time += TimeUnit.SECONDS.toMinutes(seconds) + "m";
|
|
||||||
+ }
|
|
||||||
+ return time;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static void set(String path, Object val) {
|
|
||||||
+ config.set(path, val);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static boolean getBoolean(String path, boolean def) {
|
|
||||||
+ config.addDefault(path, def);
|
|
||||||
+ return config.getBoolean(path, config.getBoolean(path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static double getDouble(String path, double def) {
|
|
||||||
+ config.addDefault(path, def);
|
|
||||||
+ return config.getDouble(path, config.getDouble(path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static float getFloat(String path, float def) {
|
|
||||||
+ // TODO: Figure out why getFloat() always returns the default value.
|
|
||||||
+ return (float) getDouble(path, (double) def);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static int getInt(String path, int def) {
|
|
||||||
+ config.addDefault(path, def);
|
|
||||||
+ return config.getInt(path, config.getInt(path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static <T> List getList(String path, T def) {
|
|
||||||
+ config.addDefault(path, def);
|
|
||||||
+ return (List<T>) config.getList(path, config.getList(path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static String getString(String path, String def) {
|
|
||||||
+ config.addDefault(path, def);
|
|
||||||
+ return config.getString(path, config.getString(path));
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..b31109d2dadd29e8852468c19265066b773d2be0
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -0,0 +1,68 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import java.util.List;
|
|
||||||
+
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.configuration.file.YamlConfiguration;
|
|
||||||
+import org.spigotmc.SpigotWorldConfig;
|
|
||||||
+
|
|
||||||
+import static com.destroystokyo.paper.PaperConfig.log;
|
|
||||||
+import static com.destroystokyo.paper.PaperConfig.logError;
|
|
||||||
+
|
|
||||||
+public class PaperWorldConfig {
|
|
||||||
+
|
|
||||||
+ private final String worldName;
|
|
||||||
+ private final SpigotWorldConfig spigotConfig;
|
|
||||||
+ private YamlConfiguration config;
|
|
||||||
+ private boolean verbose;
|
|
||||||
+
|
|
||||||
+ public PaperWorldConfig(String worldName, SpigotWorldConfig spigotConfig) {
|
|
||||||
+ this.worldName = worldName;
|
|
||||||
+ this.spigotConfig = spigotConfig;
|
|
||||||
+ this.config = PaperConfig.config;
|
|
||||||
+ init();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void init() {
|
|
||||||
+ this.config = PaperConfig.config; // grab updated reference
|
|
||||||
+ log("-------- World Settings For [" + worldName + "] --------");
|
|
||||||
+ PaperConfig.readConfig(PaperWorldConfig.class, this);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void set(String path, Object val) {
|
|
||||||
+ config.set("world-settings.default." + path, val);
|
|
||||||
+ if (config.get("world-settings." + worldName + "." + path) != null) {
|
|
||||||
+ config.set("world-settings." + worldName + "." + path, val);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private boolean getBoolean(String path, boolean def) {
|
|
||||||
+ config.addDefault("world-settings.default." + path, def);
|
|
||||||
+ return config.getBoolean("world-settings." + worldName + "." + path, config.getBoolean("world-settings.default." + path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private double getDouble(String path, double def) {
|
|
||||||
+ config.addDefault("world-settings.default." + path, def);
|
|
||||||
+ return config.getDouble("world-settings." + worldName + "." + path, config.getDouble("world-settings.default." + path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private int getInt(String path, int def) {
|
|
||||||
+ config.addDefault("world-settings.default." + path, def);
|
|
||||||
+ return config.getInt("world-settings." + worldName + "." + path, config.getInt("world-settings.default." + path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private float getFloat(String path, float def) {
|
|
||||||
+ // TODO: Figure out why getFloat() always returns the default value.
|
|
||||||
+ return (float) getDouble(path, (double) def);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private <T> List<T> getList(String path, List<T> def) {
|
|
||||||
+ config.addDefault("world-settings.default." + path, def);
|
|
||||||
+ return (List<T>) config.getList("world-settings." + worldName + "." + path, config.getList("world-settings.default." + path));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private String getString(String path, String def) {
|
|
||||||
+ config.addDefault("world-settings.default." + path, def);
|
|
||||||
+ return config.getString("world-settings." + worldName + "." + path, config.getString("world-settings.default." + path));
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/Main.java b/src/main/java/net/minecraft/server/Main.java
|
|
||||||
index b746e30e6fc5ea0c17465b510d737316f1ab7004..89db31061fcc3420bc8e668533a4051cdbd12253 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/Main.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/Main.java
|
|
||||||
@@ -103,6 +103,12 @@ public class Main {
|
|
||||||
DedicatedServerSettings dedicatedserversettings = new DedicatedServerSettings(iregistrycustom_dimension, optionset); // CraftBukkit - CLI argument support
|
|
||||||
|
|
||||||
dedicatedserversettings.save();
|
|
||||||
+ // Paper start - load config files for access below if needed
|
|
||||||
+ org.bukkit.configuration.file.YamlConfiguration bukkitConfiguration = loadConfigFile((File) optionset.valueOf("bukkit-settings"));
|
|
||||||
+ org.bukkit.configuration.file.YamlConfiguration spigotConfiguration = loadConfigFile((File) optionset.valueOf("spigot-settings"));
|
|
||||||
+ org.bukkit.configuration.file.YamlConfiguration paperConfiguration = loadConfigFile((File) optionset.valueOf("paper-settings"));
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
java.nio.file.Path java_nio_file_path1 = Paths.get("eula.txt");
|
|
||||||
EULA eula = new EULA(java_nio_file_path1);
|
|
||||||
|
|
||||||
@@ -244,6 +250,20 @@ public class Main {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - load config files
|
|
||||||
+ private static org.bukkit.configuration.file.YamlConfiguration loadConfigFile(File configFile) throws Exception {
|
|
||||||
+ org.bukkit.configuration.file.YamlConfiguration config = new org.bukkit.configuration.file.YamlConfiguration();
|
|
||||||
+ if (configFile.exists()) {
|
|
||||||
+ try {
|
|
||||||
+ config.load(configFile);
|
|
||||||
+ } catch (Exception ex) {
|
|
||||||
+ throw new Exception("Failed to load configuration file: " + configFile.getName(), ex);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ return config;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
public static void convertWorld(Convertable.ConversionSession convertable_conversionsession, DataFixer datafixer, boolean flag, BooleanSupplier booleansupplier, ImmutableSet<ResourceKey<DimensionManager>> immutableset) { // CraftBukkit
|
|
||||||
Main.LOGGER.info("Forcing world upgrade! {}", convertable_conversionsession.getLevelName()); // CraftBukkit
|
|
||||||
WorldUpgrader worldupgrader = new WorldUpgrader(convertable_conversionsession, datafixer, immutableset, flag);
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
index d5628a0197125506f7aaeb89bab7bd8a811b3ad5..8e2e415a022ccd486465f19d9bbf1f287b21fb95 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
@@ -187,6 +187,15 @@ public class DedicatedServer extends MinecraftServer implements IMinecraftServer
|
|
||||||
org.spigotmc.SpigotConfig.init((java.io.File) options.valueOf("spigot-settings"));
|
|
||||||
org.spigotmc.SpigotConfig.registerCommands();
|
|
||||||
// Spigot end
|
|
||||||
+ // Paper start
|
|
||||||
+ try {
|
|
||||||
+ com.destroystokyo.paper.PaperConfig.init((java.io.File) options.valueOf("paper-settings"));
|
|
||||||
+ } catch (Exception e) {
|
|
||||||
+ DedicatedServer.LOGGER.error("Unable to load server configuration", e);
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+ com.destroystokyo.paper.PaperConfig.registerCommands();
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
this.setPVP(dedicatedserverproperties.pvp);
|
|
||||||
this.setAllowFlight(dedicatedserverproperties.allowFlight);
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/ChunkProviderServer.java b/src/main/java/net/minecraft/server/level/ChunkProviderServer.java
|
|
||||||
index e95f516756b2613972f20636443b3ff753342917..7fd6893c30fbb34367181620aa159ed79b803455 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/ChunkProviderServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/ChunkProviderServer.java
|
|
||||||
@@ -306,15 +306,15 @@ public class ChunkProviderServer extends IChunkProvider {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
- public boolean a(Entity entity) {
|
|
||||||
+ public final boolean isInEntityTickingChunk(Entity entity) { return this.a(entity); } // Paper - OBFHELPER
|
|
||||||
+ @Override public boolean a(Entity entity) {
|
|
||||||
long i = ChunkCoordIntPair.pair(MathHelper.floor(entity.locX()) >> 4, MathHelper.floor(entity.locZ()) >> 4);
|
|
||||||
|
|
||||||
return this.a(i, (Function<PlayerChunk, CompletableFuture<Either<Chunk, PlayerChunk.Failure>>>) PlayerChunk::b); // CraftBukkit - decompile error
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
- public boolean a(ChunkCoordIntPair chunkcoordintpair) {
|
|
||||||
+ public final boolean isEntityTickingChunk(ChunkCoordIntPair chunkcoordintpair) { return this.a(chunkcoordintpair); } // Paper - OBFHELPER
|
|
||||||
+ @Override public boolean a(ChunkCoordIntPair chunkcoordintpair) {
|
|
||||||
return this.a(chunkcoordintpair.pair(), (Function<PlayerChunk, CompletableFuture<Either<Chunk, PlayerChunk.Failure>>>) PlayerChunk::b); // CraftBukkit - decompile error
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityTypes.java b/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
index 4b0b782ab19a6cea4412b5e9d8c0524aee1402e4..a32bc63ff1960bdb874d546ee42633063834da24 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
@@ -2,6 +2,7 @@ package net.minecraft.world.entity;
|
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableSet;
|
|
||||||
import java.util.Optional;
|
|
||||||
+import java.util.Set; // Paper
|
|
||||||
import java.util.UUID;
|
|
||||||
import java.util.function.Function;
|
|
||||||
import java.util.stream.Stream;
|
|
||||||
@@ -599,4 +600,10 @@ public class EntityTypes<T extends Entity> {
|
|
||||||
return new EntityTypes<>(this.a, this.b, this.d, this.e, this.f, this.g, this.c, this.j, this.h, this.i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ public static Set<MinecraftKey> getEntityNameList() {
|
|
||||||
+ return IRegistry.ENTITY_TYPE.keySet();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 344b7f52e85af3e543da0bb1dd14b68eb41ebb84..f794113e7cc5809d1da0c85648fb7311fb633f40 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -129,6 +129,8 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
public boolean populating;
|
|
||||||
public final org.spigotmc.SpigotWorldConfig spigotConfig; // Spigot
|
|
||||||
|
|
||||||
+ public final com.destroystokyo.paper.PaperWorldConfig paperConfig; // Paper
|
|
||||||
+
|
|
||||||
public final SpigotTimings.WorldTimingsHandler timings; // Spigot
|
|
||||||
public static BlockPosition lastPhysicsProblem; // Spigot
|
|
||||||
private org.spigotmc.TickLimiter entityLimiter;
|
|
||||||
@@ -149,6 +151,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
|
|
||||||
protected World(WorldDataMutable worlddatamutable, ResourceKey<World> resourcekey, final DimensionManager dimensionmanager, Supplier<GameProfilerFiller> supplier, boolean flag, boolean flag1, long i, org.bukkit.generator.ChunkGenerator gen, org.bukkit.World.Environment env) {
|
|
||||||
this.spigotConfig = new org.spigotmc.SpigotWorldConfig(((net.minecraft.world.level.storage.WorldDataServer) worlddatamutable).getName()); // Spigot
|
|
||||||
+ this.paperConfig = new com.destroystokyo.paper.PaperWorldConfig(((net.minecraft.world.level.storage.WorldDataServer) worlddatamutable).getName(), this.spigotConfig); // Paper
|
|
||||||
this.generator = gen;
|
|
||||||
this.world = new CraftWorld((WorldServer) this, gen, env);
|
|
||||||
this.ticksPerAnimalSpawns = this.getServer().getTicksPerAnimalSpawns(); // CraftBukkit
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index f03d83440291a329f5a1922c8710ed975025fdbb..82f5b3387f9e71ae258111b7196986623452dc02 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -807,6 +807,7 @@ public final class CraftServer implements Server {
|
|
||||||
}
|
|
||||||
|
|
||||||
org.spigotmc.SpigotConfig.init((File) console.options.valueOf("spigot-settings")); // Spigot
|
|
||||||
+ com.destroystokyo.paper.PaperConfig.init((File) console.options.valueOf("paper-settings")); // Paper
|
|
||||||
for (WorldServer world : console.getWorlds()) {
|
|
||||||
world.worldDataServer.setDifficulty(config.difficulty);
|
|
||||||
world.setSpawnFlags(config.spawnMonsters, config.spawnAnimals);
|
|
||||||
@@ -840,6 +841,7 @@ public final class CraftServer implements Server {
|
|
||||||
world.ticksPerAmbientSpawns = this.getTicksPerAmbientSpawns();
|
|
||||||
}
|
|
||||||
world.spigotConfig.init(); // Spigot
|
|
||||||
+ world.paperConfig.init(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
pluginManager.clearPlugins();
|
|
||||||
@@ -847,6 +849,7 @@ public final class CraftServer implements Server {
|
|
||||||
resetRecipes();
|
|
||||||
reloadData();
|
|
||||||
org.spigotmc.SpigotConfig.registerCommands(); // Spigot
|
|
||||||
+ com.destroystokyo.paper.PaperConfig.registerCommands(); // Paper
|
|
||||||
overrideAllCommandBlockCommands = commandsConfiguration.getStringList("command-block-overrides").contains("*");
|
|
||||||
ignoreVanillaPermissions = commandsConfiguration.getBoolean("ignore-vanilla-permissions");
|
|
||||||
|
|
||||||
@@ -2102,4 +2105,35 @@ public final class CraftServer implements Server {
|
|
||||||
return spigot;
|
|
||||||
}
|
|
||||||
// Spigot end
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ @SuppressWarnings({"rawtypes", "unchecked"})
|
|
||||||
+ public static java.nio.file.Path dumpHeap(java.nio.file.Path dir, String name) {
|
|
||||||
+ try {
|
|
||||||
+ java.nio.file.Files.createDirectories(dir);
|
|
||||||
+
|
|
||||||
+ javax.management.MBeanServer server = java.lang.management.ManagementFactory.getPlatformMBeanServer();
|
|
||||||
+ java.nio.file.Path file;
|
|
||||||
+
|
|
||||||
+ try {
|
|
||||||
+ Class clazz = Class.forName("openj9.lang.management.OpenJ9DiagnosticsMXBean");
|
|
||||||
+ Object openj9Mbean = java.lang.management.ManagementFactory.newPlatformMXBeanProxy(server, "openj9.lang.management:type=OpenJ9Diagnostics", clazz);
|
|
||||||
+ java.lang.reflect.Method m = clazz.getMethod("triggerDumpToFile", String.class, String.class);
|
|
||||||
+ file = dir.resolve(name + ".phd");
|
|
||||||
+ m.invoke(openj9Mbean, "heap", file.toString());
|
|
||||||
+ } catch (ClassNotFoundException e) {
|
|
||||||
+ Class clazz = Class.forName("com.sun.management.HotSpotDiagnosticMXBean");
|
|
||||||
+ Object hotspotMBean = java.lang.management.ManagementFactory.newPlatformMXBeanProxy(server, "com.sun.management:type=HotSpotDiagnostic", clazz);
|
|
||||||
+ java.lang.reflect.Method m = clazz.getMethod("dumpHeap", String.class, boolean.class);
|
|
||||||
+ file = dir.resolve(name + ".hprof");
|
|
||||||
+ m.invoke(hotspotMBean, file.toString(), true);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return file;
|
|
||||||
+ } catch (Throwable t) {
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Could not write heap", t);
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/Main.java b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
index 46a16e31775b28c44f95a8ac5545ebcb656c74b6..05aedca561919a12ced1925c5cc9af585bb04523 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
@@ -129,6 +129,14 @@ public class Main {
|
|
||||||
.defaultsTo(new File("spigot.yml"))
|
|
||||||
.describedAs("Yml file");
|
|
||||||
// Spigot End
|
|
||||||
+
|
|
||||||
+ // Paper Start
|
|
||||||
+ acceptsAll(asList("paper", "paper-settings"), "File for paper settings")
|
|
||||||
+ .withRequiredArg()
|
|
||||||
+ .ofType(File.class)
|
|
||||||
+ .defaultsTo(new File("paper.yml"))
|
|
||||||
+ .describedAs("Yml file");
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/spigotmc/SpigotWorldConfig.java b/src/main/java/org/spigotmc/SpigotWorldConfig.java
|
|
||||||
index 83d83ff7ceffbb77723da721b869dfd0091e496d..0efcbab8f8806aeb8dd8bd6384e5a7cee375d100 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/SpigotWorldConfig.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/SpigotWorldConfig.java
|
|
||||||
@@ -39,36 +39,36 @@ public class SpigotWorldConfig
|
|
||||||
config.set( "world-settings.default." + path, val );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private boolean getBoolean(String path, boolean def)
|
|
||||||
+ public boolean getBoolean(String path, boolean def) // Paper - private -> public
|
|
||||||
{
|
|
||||||
config.addDefault( "world-settings.default." + path, def );
|
|
||||||
return config.getBoolean( "world-settings." + worldName + "." + path, config.getBoolean( "world-settings.default." + path ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private double getDouble(String path, double def)
|
|
||||||
+ public double getDouble(String path, double def) // Paper - private -> public
|
|
||||||
{
|
|
||||||
config.addDefault( "world-settings.default." + path, def );
|
|
||||||
return config.getDouble( "world-settings." + worldName + "." + path, config.getDouble( "world-settings.default." + path ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private int getInt(String path)
|
|
||||||
+ public int getInt(String path) // Paper - private -> public
|
|
||||||
{
|
|
||||||
return config.getInt( "world-settings." + worldName + "." + path );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private int getInt(String path, int def)
|
|
||||||
+ public int getInt(String path, int def) // Paper - private -> public
|
|
||||||
{
|
|
||||||
config.addDefault( "world-settings.default." + path, def );
|
|
||||||
return config.getInt( "world-settings." + worldName + "." + path, config.getInt( "world-settings.default." + path ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private <T> List getList(String path, T def)
|
|
||||||
+ public <T> List getList(String path, T def) // Paper - private -> public
|
|
||||||
{
|
|
||||||
config.addDefault( "world-settings.default." + path, def );
|
|
||||||
return (List<T>) config.getList( "world-settings." + worldName + "." + path, config.getList( "world-settings.default." + path ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
- private String getString(String path, String def)
|
|
||||||
+ public String getString(String path, String def) // Paper - private -> public
|
|
||||||
{
|
|
||||||
config.addDefault( "world-settings.default." + path, def );
|
|
||||||
return config.getString( "world-settings." + worldName + "." + path, config.getString( "world-settings.default." + path ) );
|
|
File diff suppressed because one or more lines are too long
File diff suppressed because it is too large
Load Diff
@ -1,735 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Fri, 24 Mar 2017 23:56:01 -0500
|
|
||||||
Subject: [PATCH] Paper Metrics
|
|
||||||
|
|
||||||
Removes Spigot's mcstats metrics in favor of a system using bStats
|
|
||||||
|
|
||||||
To disable for privacy or other reasons go to the bStats folder in your plugins folder
|
|
||||||
and edit the config.yml file present there.
|
|
||||||
|
|
||||||
Please keep in mind the data collected is anonymous and collection should have no
|
|
||||||
tangible effect on server performance. The data is used to allow the authors of
|
|
||||||
PaperMC to track version and platform usage so that we can make better management
|
|
||||||
decisions on behalf of the project.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/Metrics.java b/src/main/java/com/destroystokyo/paper/Metrics.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..0b9e689d57705965721b5c55bc45d36657f360e4
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/Metrics.java
|
|
||||||
@@ -0,0 +1,670 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import net.minecraft.server.MinecraftServer;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.configuration.file.YamlConfiguration;
|
|
||||||
+import org.bukkit.craftbukkit.util.CraftMagicNumbers;
|
|
||||||
+import org.bukkit.plugin.Plugin;
|
|
||||||
+
|
|
||||||
+import org.json.simple.JSONArray;
|
|
||||||
+import org.json.simple.JSONObject;
|
|
||||||
+
|
|
||||||
+import javax.net.ssl.HttpsURLConnection;
|
|
||||||
+import java.io.ByteArrayOutputStream;
|
|
||||||
+import java.io.DataOutputStream;
|
|
||||||
+import java.io.File;
|
|
||||||
+import java.io.IOException;
|
|
||||||
+import java.net.URL;
|
|
||||||
+import java.util.*;
|
|
||||||
+import java.util.concurrent.Callable;
|
|
||||||
+import java.util.concurrent.Executors;
|
|
||||||
+import java.util.concurrent.ScheduledExecutorService;
|
|
||||||
+import java.util.concurrent.TimeUnit;
|
|
||||||
+import java.util.logging.Level;
|
|
||||||
+import java.util.logging.Logger;
|
|
||||||
+import java.util.regex.Matcher;
|
|
||||||
+import java.util.regex.Pattern;
|
|
||||||
+import java.util.zip.GZIPOutputStream;
|
|
||||||
+
|
|
||||||
+/**
|
|
||||||
+ * bStats collects some data for plugin authors.
|
|
||||||
+ *
|
|
||||||
+ * Check out https://bStats.org/ to learn more about bStats!
|
|
||||||
+ */
|
|
||||||
+public class Metrics {
|
|
||||||
+
|
|
||||||
+ // Executor service for requests
|
|
||||||
+ // We use an executor service because the Bukkit scheduler is affected by server lags
|
|
||||||
+ private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
|
|
||||||
+
|
|
||||||
+ // The version of this bStats class
|
|
||||||
+ public static final int B_STATS_VERSION = 1;
|
|
||||||
+
|
|
||||||
+ // The url to which the data is sent
|
|
||||||
+ private static final String URL = "https://bStats.org/submitData/server-implementation";
|
|
||||||
+
|
|
||||||
+ // Should failed requests be logged?
|
|
||||||
+ private static boolean logFailedRequests = false;
|
|
||||||
+
|
|
||||||
+ // The logger for the failed requests
|
|
||||||
+ private static Logger logger = Logger.getLogger("bStats");
|
|
||||||
+
|
|
||||||
+ // The name of the server software
|
|
||||||
+ private final String name;
|
|
||||||
+
|
|
||||||
+ // The uuid of the server
|
|
||||||
+ private final String serverUUID;
|
|
||||||
+
|
|
||||||
+ // A list with all custom charts
|
|
||||||
+ private final List<CustomChart> charts = new ArrayList<>();
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param name The name of the server software.
|
|
||||||
+ * @param serverUUID The uuid of the server.
|
|
||||||
+ * @param logFailedRequests Whether failed requests should be logged or not.
|
|
||||||
+ * @param logger The logger for the failed requests.
|
|
||||||
+ */
|
|
||||||
+ public Metrics(String name, String serverUUID, boolean logFailedRequests, Logger logger) {
|
|
||||||
+ this.name = name;
|
|
||||||
+ this.serverUUID = serverUUID;
|
|
||||||
+ Metrics.logFailedRequests = logFailedRequests;
|
|
||||||
+ Metrics.logger = logger;
|
|
||||||
+
|
|
||||||
+ // Start submitting the data
|
|
||||||
+ startSubmitting();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Adds a custom chart.
|
|
||||||
+ *
|
|
||||||
+ * @param chart The chart to add.
|
|
||||||
+ */
|
|
||||||
+ public void addCustomChart(CustomChart chart) {
|
|
||||||
+ if (chart == null) {
|
|
||||||
+ throw new IllegalArgumentException("Chart cannot be null!");
|
|
||||||
+ }
|
|
||||||
+ charts.add(chart);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Starts the Scheduler which submits our data every 30 minutes.
|
|
||||||
+ */
|
|
||||||
+ private void startSubmitting() {
|
|
||||||
+ final Runnable submitTask = this::submitData;
|
|
||||||
+
|
|
||||||
+ // Many servers tend to restart at a fixed time at xx:00 which causes an uneven distribution of requests on the
|
|
||||||
+ // bStats backend. To circumvent this problem, we introduce some randomness into the initial and second delay.
|
|
||||||
+ // WARNING: You must not modify any part of this Metrics class, including the submit delay or frequency!
|
|
||||||
+ // WARNING: Modifying this code will get your plugin banned on bStats. Just don't do it!
|
|
||||||
+ long initialDelay = (long) (1000 * 60 * (3 + Math.random() * 3));
|
|
||||||
+ long secondDelay = (long) (1000 * 60 * (Math.random() * 30));
|
|
||||||
+ scheduler.schedule(submitTask, initialDelay, TimeUnit.MILLISECONDS);
|
|
||||||
+ scheduler.scheduleAtFixedRate(submitTask, initialDelay + secondDelay, 1000 * 60 * 30, TimeUnit.MILLISECONDS);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Gets the plugin specific data.
|
|
||||||
+ *
|
|
||||||
+ * @return The plugin specific data.
|
|
||||||
+ */
|
|
||||||
+ private JSONObject getPluginData() {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+
|
|
||||||
+ data.put("pluginName", name); // Append the name of the server software
|
|
||||||
+ JSONArray customCharts = new JSONArray();
|
|
||||||
+ for (CustomChart customChart : charts) {
|
|
||||||
+ // Add the data of the custom charts
|
|
||||||
+ JSONObject chart = customChart.getRequestJsonObject();
|
|
||||||
+ if (chart == null) { // If the chart is null, we skip it
|
|
||||||
+ continue;
|
|
||||||
+ }
|
|
||||||
+ customCharts.add(chart);
|
|
||||||
+ }
|
|
||||||
+ data.put("customCharts", customCharts);
|
|
||||||
+
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Gets the server specific data.
|
|
||||||
+ *
|
|
||||||
+ * @return The server specific data.
|
|
||||||
+ */
|
|
||||||
+ private JSONObject getServerData() {
|
|
||||||
+ // OS specific data
|
|
||||||
+ String osName = System.getProperty("os.name");
|
|
||||||
+ String osArch = System.getProperty("os.arch");
|
|
||||||
+ String osVersion = System.getProperty("os.version");
|
|
||||||
+ int coreCount = Runtime.getRuntime().availableProcessors();
|
|
||||||
+
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+
|
|
||||||
+ data.put("serverUUID", serverUUID);
|
|
||||||
+
|
|
||||||
+ data.put("osName", osName);
|
|
||||||
+ data.put("osArch", osArch);
|
|
||||||
+ data.put("osVersion", osVersion);
|
|
||||||
+ data.put("coreCount", coreCount);
|
|
||||||
+
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Collects the data and sends it afterwards.
|
|
||||||
+ */
|
|
||||||
+ private void submitData() {
|
|
||||||
+ final JSONObject data = getServerData();
|
|
||||||
+
|
|
||||||
+ JSONArray pluginData = new JSONArray();
|
|
||||||
+ pluginData.add(getPluginData());
|
|
||||||
+ data.put("plugins", pluginData);
|
|
||||||
+
|
|
||||||
+ try {
|
|
||||||
+ // We are still in the Thread of the timer, so nothing get blocked :)
|
|
||||||
+ sendData(data);
|
|
||||||
+ } catch (Exception e) {
|
|
||||||
+ // Something went wrong! :(
|
|
||||||
+ if (logFailedRequests) {
|
|
||||||
+ logger.log(Level.WARNING, "Could not submit stats of " + name, e);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Sends the data to the bStats server.
|
|
||||||
+ *
|
|
||||||
+ * @param data The data to send.
|
|
||||||
+ * @throws Exception If the request failed.
|
|
||||||
+ */
|
|
||||||
+ private static void sendData(JSONObject data) throws Exception {
|
|
||||||
+ if (data == null) {
|
|
||||||
+ throw new IllegalArgumentException("Data cannot be null!");
|
|
||||||
+ }
|
|
||||||
+ HttpsURLConnection connection = (HttpsURLConnection) new URL(URL).openConnection();
|
|
||||||
+
|
|
||||||
+ // Compress the data to save bandwidth
|
|
||||||
+ byte[] compressedData = compress(data.toString());
|
|
||||||
+
|
|
||||||
+ // Add headers
|
|
||||||
+ connection.setRequestMethod("POST");
|
|
||||||
+ connection.addRequestProperty("Accept", "application/json");
|
|
||||||
+ connection.addRequestProperty("Connection", "close");
|
|
||||||
+ connection.addRequestProperty("Content-Encoding", "gzip"); // We gzip our request
|
|
||||||
+ connection.addRequestProperty("Content-Length", String.valueOf(compressedData.length));
|
|
||||||
+ connection.setRequestProperty("Content-Type", "application/json"); // We send our data in JSON format
|
|
||||||
+ connection.setRequestProperty("User-Agent", "MC-Server/" + B_STATS_VERSION);
|
|
||||||
+
|
|
||||||
+ // Send data
|
|
||||||
+ connection.setDoOutput(true);
|
|
||||||
+ DataOutputStream outputStream = new DataOutputStream(connection.getOutputStream());
|
|
||||||
+ outputStream.write(compressedData);
|
|
||||||
+ outputStream.flush();
|
|
||||||
+ outputStream.close();
|
|
||||||
+
|
|
||||||
+ connection.getInputStream().close(); // We don't care about the response - Just send our data :)
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Gzips the given String.
|
|
||||||
+ *
|
|
||||||
+ * @param str The string to gzip.
|
|
||||||
+ * @return The gzipped String.
|
|
||||||
+ * @throws IOException If the compression failed.
|
|
||||||
+ */
|
|
||||||
+ private static byte[] compress(final String str) throws IOException {
|
|
||||||
+ if (str == null) {
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
|
|
||||||
+ GZIPOutputStream gzip = new GZIPOutputStream(outputStream);
|
|
||||||
+ gzip.write(str.getBytes("UTF-8"));
|
|
||||||
+ gzip.close();
|
|
||||||
+ return outputStream.toByteArray();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom chart.
|
|
||||||
+ */
|
|
||||||
+ public static abstract class CustomChart {
|
|
||||||
+
|
|
||||||
+ // The id of the chart
|
|
||||||
+ final String chartId;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ */
|
|
||||||
+ CustomChart(String chartId) {
|
|
||||||
+ if (chartId == null || chartId.isEmpty()) {
|
|
||||||
+ throw new IllegalArgumentException("ChartId cannot be null or empty!");
|
|
||||||
+ }
|
|
||||||
+ this.chartId = chartId;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private JSONObject getRequestJsonObject() {
|
|
||||||
+ JSONObject chart = new JSONObject();
|
|
||||||
+ chart.put("chartId", chartId);
|
|
||||||
+ try {
|
|
||||||
+ JSONObject data = getChartData();
|
|
||||||
+ if (data == null) {
|
|
||||||
+ // If the data is null we don't send the chart.
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ chart.put("data", data);
|
|
||||||
+ } catch (Throwable t) {
|
|
||||||
+ if (logFailedRequests) {
|
|
||||||
+ logger.log(Level.WARNING, "Failed to get data for custom chart with id " + chartId, t);
|
|
||||||
+ }
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ return chart;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ protected abstract JSONObject getChartData() throws Exception;
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom simple pie.
|
|
||||||
+ */
|
|
||||||
+ public static class SimplePie extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<String> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public SimplePie(String chartId, Callable<String> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ String value = callable.call();
|
|
||||||
+ if (value == null || value.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("value", value);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom advanced pie.
|
|
||||||
+ */
|
|
||||||
+ public static class AdvancedPie extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Map<String, Integer>> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public AdvancedPie(String chartId, Callable<Map<String, Integer>> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ JSONObject values = new JSONObject();
|
|
||||||
+ Map<String, Integer> map = callable.call();
|
|
||||||
+ if (map == null || map.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ boolean allSkipped = true;
|
|
||||||
+ for (Map.Entry<String, Integer> entry : map.entrySet()) {
|
|
||||||
+ if (entry.getValue() == 0) {
|
|
||||||
+ continue; // Skip this invalid
|
|
||||||
+ }
|
|
||||||
+ allSkipped = false;
|
|
||||||
+ values.put(entry.getKey(), entry.getValue());
|
|
||||||
+ }
|
|
||||||
+ if (allSkipped) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("values", values);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom drilldown pie.
|
|
||||||
+ */
|
|
||||||
+ public static class DrilldownPie extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Map<String, Map<String, Integer>>> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public DrilldownPie(String chartId, Callable<Map<String, Map<String, Integer>>> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ JSONObject values = new JSONObject();
|
|
||||||
+ Map<String, Map<String, Integer>> map = callable.call();
|
|
||||||
+ if (map == null || map.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ boolean reallyAllSkipped = true;
|
|
||||||
+ for (Map.Entry<String, Map<String, Integer>> entryValues : map.entrySet()) {
|
|
||||||
+ JSONObject value = new JSONObject();
|
|
||||||
+ boolean allSkipped = true;
|
|
||||||
+ for (Map.Entry<String, Integer> valueEntry : map.get(entryValues.getKey()).entrySet()) {
|
|
||||||
+ value.put(valueEntry.getKey(), valueEntry.getValue());
|
|
||||||
+ allSkipped = false;
|
|
||||||
+ }
|
|
||||||
+ if (!allSkipped) {
|
|
||||||
+ reallyAllSkipped = false;
|
|
||||||
+ values.put(entryValues.getKey(), value);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (reallyAllSkipped) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("values", values);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom single line chart.
|
|
||||||
+ */
|
|
||||||
+ public static class SingleLineChart extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Integer> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public SingleLineChart(String chartId, Callable<Integer> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ int value = callable.call();
|
|
||||||
+ if (value == 0) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("value", value);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom multi line chart.
|
|
||||||
+ */
|
|
||||||
+ public static class MultiLineChart extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Map<String, Integer>> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public MultiLineChart(String chartId, Callable<Map<String, Integer>> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ JSONObject values = new JSONObject();
|
|
||||||
+ Map<String, Integer> map = callable.call();
|
|
||||||
+ if (map == null || map.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ boolean allSkipped = true;
|
|
||||||
+ for (Map.Entry<String, Integer> entry : map.entrySet()) {
|
|
||||||
+ if (entry.getValue() == 0) {
|
|
||||||
+ continue; // Skip this invalid
|
|
||||||
+ }
|
|
||||||
+ allSkipped = false;
|
|
||||||
+ values.put(entry.getKey(), entry.getValue());
|
|
||||||
+ }
|
|
||||||
+ if (allSkipped) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("values", values);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom simple bar chart.
|
|
||||||
+ */
|
|
||||||
+ public static class SimpleBarChart extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Map<String, Integer>> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public SimpleBarChart(String chartId, Callable<Map<String, Integer>> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ JSONObject values = new JSONObject();
|
|
||||||
+ Map<String, Integer> map = callable.call();
|
|
||||||
+ if (map == null || map.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ for (Map.Entry<String, Integer> entry : map.entrySet()) {
|
|
||||||
+ JSONArray categoryValues = new JSONArray();
|
|
||||||
+ categoryValues.add(entry.getValue());
|
|
||||||
+ values.put(entry.getKey(), categoryValues);
|
|
||||||
+ }
|
|
||||||
+ data.put("values", values);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Represents a custom advanced bar chart.
|
|
||||||
+ */
|
|
||||||
+ public static class AdvancedBarChart extends CustomChart {
|
|
||||||
+
|
|
||||||
+ private final Callable<Map<String, int[]>> callable;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Class constructor.
|
|
||||||
+ *
|
|
||||||
+ * @param chartId The id of the chart.
|
|
||||||
+ * @param callable The callable which is used to request the chart data.
|
|
||||||
+ */
|
|
||||||
+ public AdvancedBarChart(String chartId, Callable<Map<String, int[]>> callable) {
|
|
||||||
+ super(chartId);
|
|
||||||
+ this.callable = callable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ protected JSONObject getChartData() throws Exception {
|
|
||||||
+ JSONObject data = new JSONObject();
|
|
||||||
+ JSONObject values = new JSONObject();
|
|
||||||
+ Map<String, int[]> map = callable.call();
|
|
||||||
+ if (map == null || map.isEmpty()) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ boolean allSkipped = true;
|
|
||||||
+ for (Map.Entry<String, int[]> entry : map.entrySet()) {
|
|
||||||
+ if (entry.getValue().length == 0) {
|
|
||||||
+ continue; // Skip this invalid
|
|
||||||
+ }
|
|
||||||
+ allSkipped = false;
|
|
||||||
+ JSONArray categoryValues = new JSONArray();
|
|
||||||
+ for (int categoryValue : entry.getValue()) {
|
|
||||||
+ categoryValues.add(categoryValue);
|
|
||||||
+ }
|
|
||||||
+ values.put(entry.getKey(), categoryValues);
|
|
||||||
+ }
|
|
||||||
+ if (allSkipped) {
|
|
||||||
+ // Null = skip the chart
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ data.put("values", values);
|
|
||||||
+ return data;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ static class PaperMetrics {
|
|
||||||
+ static void startMetrics() {
|
|
||||||
+ // Get the config file
|
|
||||||
+ File configFile = new File(new File((File) MinecraftServer.getServer().options.valueOf("plugins"), "bStats"), "config.yml");
|
|
||||||
+ YamlConfiguration config = YamlConfiguration.loadConfiguration(configFile);
|
|
||||||
+
|
|
||||||
+ // Check if the config file exists
|
|
||||||
+ if (!config.isSet("serverUuid")) {
|
|
||||||
+
|
|
||||||
+ // Add default values
|
|
||||||
+ config.addDefault("enabled", true);
|
|
||||||
+ // Every server gets it's unique random id.
|
|
||||||
+ config.addDefault("serverUuid", UUID.randomUUID().toString());
|
|
||||||
+ // Should failed request be logged?
|
|
||||||
+ config.addDefault("logFailedRequests", false);
|
|
||||||
+
|
|
||||||
+ // Inform the server owners about bStats
|
|
||||||
+ config.options().header(
|
|
||||||
+ "bStats collects some data for plugin authors like how many servers are using their plugins.\n" +
|
|
||||||
+ "To honor their work, you should not disable it.\n" +
|
|
||||||
+ "This has nearly no effect on the server performance!\n" +
|
|
||||||
+ "Check out https://bStats.org/ to learn more :)"
|
|
||||||
+ ).copyDefaults(true);
|
|
||||||
+ try {
|
|
||||||
+ config.save(configFile);
|
|
||||||
+ } catch (IOException ignored) {
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Load the data
|
|
||||||
+ String serverUUID = config.getString("serverUuid");
|
|
||||||
+ boolean logFailedRequests = config.getBoolean("logFailedRequests", false);
|
|
||||||
+ // Only start Metrics, if it's enabled in the config
|
|
||||||
+ if (config.getBoolean("enabled", true)) {
|
|
||||||
+ Metrics metrics = new Metrics("Paper", serverUUID, logFailedRequests, Bukkit.getLogger());
|
|
||||||
+
|
|
||||||
+ metrics.addCustomChart(new Metrics.SimplePie("minecraft_version", () -> {
|
|
||||||
+ String minecraftVersion = Bukkit.getVersion();
|
|
||||||
+ minecraftVersion = minecraftVersion.substring(minecraftVersion.indexOf("MC: ") + 4, minecraftVersion.length() - 1);
|
|
||||||
+ return minecraftVersion;
|
|
||||||
+ }));
|
|
||||||
+
|
|
||||||
+ metrics.addCustomChart(new Metrics.SingleLineChart("players", () -> Bukkit.getOnlinePlayers().size()));
|
|
||||||
+ metrics.addCustomChart(new Metrics.SimplePie("online_mode", () -> Bukkit.getOnlineMode() || PaperConfig.isProxyOnlineMode() ? "online" : "offline"));
|
|
||||||
+ metrics.addCustomChart(new Metrics.SimplePie("paper_version", () -> (Metrics.class.getPackage().getImplementationVersion() != null) ? Metrics.class.getPackage().getImplementationVersion() : "unknown"));
|
|
||||||
+
|
|
||||||
+ metrics.addCustomChart(new Metrics.DrilldownPie("java_version", () -> {
|
|
||||||
+ Map<String, Map<String, Integer>> map = new HashMap<>();
|
|
||||||
+ String javaVersion = System.getProperty("java.version");
|
|
||||||
+ Map<String, Integer> entry = new HashMap<>();
|
|
||||||
+ entry.put(javaVersion, 1);
|
|
||||||
+
|
|
||||||
+ // http://openjdk.java.net/jeps/223
|
|
||||||
+ // Java decided to change their versioning scheme and in doing so modified the java.version system
|
|
||||||
+ // property to return $major[.$minor][.$secuity][-ea], as opposed to 1.$major.0_$identifier
|
|
||||||
+ // we can handle pre-9 by checking if the "major" is equal to "1", otherwise, 9+
|
|
||||||
+ String majorVersion = javaVersion.split("\\.")[0];
|
|
||||||
+ String release;
|
|
||||||
+
|
|
||||||
+ int indexOf = javaVersion.lastIndexOf('.');
|
|
||||||
+
|
|
||||||
+ if (majorVersion.equals("1")) {
|
|
||||||
+ release = "Java " + javaVersion.substring(0, indexOf);
|
|
||||||
+ } else {
|
|
||||||
+ // of course, it really wouldn't be all that simple if they didn't add a quirk, now would it
|
|
||||||
+ // valid strings for the major may potentially include values such as -ea to deannotate a pre release
|
|
||||||
+ Matcher versionMatcher = Pattern.compile("\\d+").matcher(majorVersion);
|
|
||||||
+ if (versionMatcher.find()) {
|
|
||||||
+ majorVersion = versionMatcher.group(0);
|
|
||||||
+ }
|
|
||||||
+ release = "Java " + majorVersion;
|
|
||||||
+ }
|
|
||||||
+ map.put(release, entry);
|
|
||||||
+
|
|
||||||
+ return map;
|
|
||||||
+ }));
|
|
||||||
+
|
|
||||||
+ metrics.addCustomChart(new Metrics.DrilldownPie("legacy_plugins", () -> {
|
|
||||||
+ Map<String, Map<String, Integer>> map = new HashMap<>();
|
|
||||||
+
|
|
||||||
+ // count legacy plugins
|
|
||||||
+ int legacy = 0;
|
|
||||||
+ for (Plugin plugin : Bukkit.getPluginManager().getPlugins()) {
|
|
||||||
+ if (CraftMagicNumbers.isLegacy(plugin.getDescription())) {
|
|
||||||
+ legacy++;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // insert real value as lower dimension
|
|
||||||
+ Map<String, Integer> entry = new HashMap<>();
|
|
||||||
+ entry.put(String.valueOf(legacy), 1);
|
|
||||||
+
|
|
||||||
+ // create buckets as higher dimension
|
|
||||||
+ if (legacy == 0) {
|
|
||||||
+ map.put("0 \uD83D\uDE0E", entry); // :sunglasses:
|
|
||||||
+ } else if (legacy <= 5) {
|
|
||||||
+ map.put("1-5", entry);
|
|
||||||
+ } else if (legacy <= 10) {
|
|
||||||
+ map.put("6-10", entry);
|
|
||||||
+ } else if (legacy <= 25) {
|
|
||||||
+ map.put("11-25", entry);
|
|
||||||
+ } else if (legacy <= 50) {
|
|
||||||
+ map.put("26-50", entry);
|
|
||||||
+ } else {
|
|
||||||
+ map.put("50+ \uD83D\uDE2D", entry); // :cry:
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return map;
|
|
||||||
+ }));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
index 2c0514892d3993bef57ecf677cf8bb0fbe0216e4..da922f395f0fff0881ead893c900c5b2623f48f0 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -42,6 +42,7 @@ public class PaperConfig {
|
|
||||||
private static boolean verbose;
|
|
||||||
private static boolean fatalError;
|
|
||||||
/*========================================================================*/
|
|
||||||
+ private static boolean metricsStarted;
|
|
||||||
|
|
||||||
public static void init(File configFile) {
|
|
||||||
CONFIG_FILE = configFile;
|
|
||||||
@@ -84,6 +85,11 @@ public class PaperConfig {
|
|
||||||
for (Map.Entry<String, Command> entry : commands.entrySet()) {
|
|
||||||
MinecraftServer.getServer().server.getCommandMap().register(entry.getKey(), "Paper", entry.getValue());
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ if (!metricsStarted) {
|
|
||||||
+ Metrics.PaperMetrics.startMetrics();
|
|
||||||
+ metricsStarted = true;
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
|
|
||||||
static void readConfig(Class<?> clazz, Object instance) {
|
|
||||||
diff --git a/src/main/java/org/spigotmc/SpigotConfig.java b/src/main/java/org/spigotmc/SpigotConfig.java
|
|
||||||
index c04d912adf0da8f7a5b75dd2f58739a11ca31601..3c93a497a790b8d800852db2ac48feca41f45cef 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/SpigotConfig.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/SpigotConfig.java
|
|
||||||
@@ -83,6 +83,7 @@ public class SpigotConfig
|
|
||||||
MinecraftServer.getServer().server.getCommandMap().register( entry.getKey(), "Spigot", entry.getValue() );
|
|
||||||
}
|
|
||||||
|
|
||||||
+ /* // Paper - Replace with our own
|
|
||||||
if ( metrics == null )
|
|
||||||
{
|
|
||||||
try
|
|
||||||
@@ -94,6 +95,7 @@ public class SpigotConfig
|
|
||||||
Bukkit.getServer().getLogger().log( Level.SEVERE, "Could not start metrics service", ex );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+ */ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
static void readConfig(Class<?> clazz, Object instance)
|
|
@ -1,144 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 4 Jul 2018 01:40:13 -0400
|
|
||||||
Subject: [PATCH] Add MinecraftKey Information to Objects
|
|
||||||
|
|
||||||
Stores the reference to the objects respective MinecraftKey
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperCommand.java b/src/main/java/com/destroystokyo/paper/PaperCommand.java
|
|
||||||
index 68cd4134cb6a00c1768100462f8e9e94f3fa6279..6943524c2dd8b12691b8ac5b08daee823ce50c3d 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperCommand.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperCommand.java
|
|
||||||
@@ -208,7 +208,7 @@ public class PaperCommand extends Command {
|
|
||||||
|
|
||||||
Collection<Entity> entities = world.entitiesById.values();
|
|
||||||
entities.forEach(e -> {
|
|
||||||
- MinecraftKey key = new MinecraftKey(""); // TODO: update in next patch
|
|
||||||
+ MinecraftKey key = e.getMinecraftKey();
|
|
||||||
|
|
||||||
MutablePair<Integer, Map<ChunkCoordIntPair, Integer>> info = list.computeIfAbsent(key, k -> MutablePair.of(0, Maps.newHashMap()));
|
|
||||||
ChunkCoordIntPair chunk = new ChunkCoordIntPair(e.chunkX, e.chunkZ);
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/KeyedObject.java b/src/main/java/net/minecraft/server/KeyedObject.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..500477f0d98a277145c2b0dbb4d1496e2a588542
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/net/minecraft/server/KeyedObject.java
|
|
||||||
@@ -0,0 +1,11 @@
|
|
||||||
+package net.minecraft.server;
|
|
||||||
+
|
|
||||||
+import net.minecraft.resources.MinecraftKey;
|
|
||||||
+
|
|
||||||
+public interface KeyedObject {
|
|
||||||
+ MinecraftKey getMinecraftKey();
|
|
||||||
+ default String getMinecraftKeyString() {
|
|
||||||
+ MinecraftKey key = getMinecraftKey();
|
|
||||||
+ return key != null ? key.toString() : null;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index 3c8767d5fab575e61ca179b517f0bff1a38405ce..ec553e7d7595ef3652bfa3325a07483bb3c32245 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -136,7 +136,7 @@ import org.bukkit.event.player.PlayerTeleportEvent;
|
|
||||||
import org.bukkit.plugin.PluginManager;
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
-public abstract class Entity implements INamableTileEntity, ICommandListener {
|
|
||||||
+public abstract class Entity implements INamableTileEntity, ICommandListener, net.minecraft.server.KeyedObject { // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
private static final int CURRENT_LEVEL = 2;
|
|
||||||
@@ -1762,12 +1762,31 @@ public abstract class Entity implements INamableTileEntity, ICommandListener {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ private MinecraftKey entityKey;
|
|
||||||
+ private String entityKeyString;
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public MinecraftKey getMinecraftKey() {
|
|
||||||
+ if (entityKey == null) {
|
|
||||||
+ this.entityKey = EntityTypes.getName(this.getEntityType());
|
|
||||||
+ this.entityKeyString = this.entityKey != null ? this.entityKey.toString() : null;
|
|
||||||
+ }
|
|
||||||
+ return entityKey;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public String getMinecraftKeyString() {
|
|
||||||
+ getMinecraftKey(); // Try to load if it doesn't exists. see: https://github.com/PaperMC/Paper/issues/1280
|
|
||||||
+ return entityKeyString;
|
|
||||||
+ }
|
|
||||||
@Nullable
|
|
||||||
public final String getSaveID() {
|
|
||||||
EntityTypes<?> entitytypes = this.getEntityType();
|
|
||||||
MinecraftKey minecraftkey = EntityTypes.getName(entitytypes);
|
|
||||||
|
|
||||||
- return entitytypes.a() && minecraftkey != null ? minecraftkey.toString() : null;
|
|
||||||
+ return entitytypes != null && entitytypes.isPersistable() ? getMinecraftKeyString() : null;
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
protected abstract void loadData(NBTTagCompound nbttagcompound);
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityTypes.java b/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
index ac57ab9992e141c91cf48f033148ad78433b364c..dc92b112770955f9fa49a408262da2e5bbc4bf98 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityTypes.java
|
|
||||||
@@ -384,6 +384,7 @@ public class EntityTypes<T extends Entity> {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public boolean isPersistable() { return a(); } // Paper - OBFHELPER
|
|
||||||
public boolean a() {
|
|
||||||
return this.bi;
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
index 72d43b38de3e797a509de6591874af12fa7b9ec0..2b58ae6d91fe0d0f36eedbb78a3c8a8a66d92405 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
@@ -23,7 +23,7 @@ import org.bukkit.inventory.InventoryHolder;
|
|
||||||
|
|
||||||
import org.spigotmc.CustomTimingsHandler; // Spigot
|
|
||||||
|
|
||||||
-public abstract class TileEntity {
|
|
||||||
+public abstract class TileEntity implements net.minecraft.server.KeyedObject { // Paper
|
|
||||||
|
|
||||||
public CustomTimingsHandler tickTimer = org.bukkit.craftbukkit.SpigotTimings.getTileEntityTimings(this); // Spigot
|
|
||||||
// CraftBukkit start - data containers
|
|
||||||
@@ -31,7 +31,7 @@ public abstract class TileEntity {
|
|
||||||
public CraftPersistentDataContainer persistentDataContainer;
|
|
||||||
// CraftBukkit end
|
|
||||||
private static final Logger LOGGER = LogManager.getLogger();
|
|
||||||
- private final TileEntityTypes<?> tileType;
|
|
||||||
+ private final TileEntityTypes<?> tileType; public TileEntityTypes getTileEntityType() { return tileType; } // Paper - OBFHELPER
|
|
||||||
@Nullable
|
|
||||||
protected World world;
|
|
||||||
protected BlockPosition position;
|
|
||||||
@@ -45,6 +45,26 @@ public abstract class TileEntity {
|
|
||||||
this.tileType = tileentitytypes;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ private String tileEntityKeyString = null;
|
|
||||||
+ private MinecraftKey tileEntityKey = null;
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public MinecraftKey getMinecraftKey() {
|
|
||||||
+ if (tileEntityKey == null) {
|
|
||||||
+ tileEntityKey = TileEntityTypes.a(this.getTileEntityType());
|
|
||||||
+ tileEntityKeyString = tileEntityKey != null ? tileEntityKey.toString() : null;
|
|
||||||
+ }
|
|
||||||
+ return tileEntityKey;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public String getMinecraftKeyString() {
|
|
||||||
+ getMinecraftKey(); // Try to load if it doesn't exists.
|
|
||||||
+ return tileEntityKeyString;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
@Nullable
|
|
||||||
public World getWorld() {
|
|
||||||
return this.world;
|
|
@ -1,171 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 4 Jul 2018 02:10:36 -0400
|
|
||||||
Subject: [PATCH] Store reference to current Chunk for Entity and Block
|
|
||||||
Entities
|
|
||||||
|
|
||||||
This enables us a fast reference to the entities current chunk instead
|
|
||||||
of having to look it up by hashmap lookups.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index ec553e7d7595ef3652bfa3325a07483bb3c32245..2bea2f4748cadf479dd4f89792ef5ffdd88e9cab 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -261,7 +261,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isChunkLoaded() {
|
|
||||||
- return world.isChunkLoaded((int) Math.floor(this.locX()) >> 4, (int) Math.floor(this.locZ()) >> 4);
|
|
||||||
+ return getCurrentChunk() != null;
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
@@ -1763,6 +1763,23 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
}
|
|
||||||
|
|
||||||
// Paper start
|
|
||||||
+ public java.lang.ref.WeakReference<net.minecraft.world.level.chunk.Chunk> currentChunk = null;
|
|
||||||
+
|
|
||||||
+ public void setCurrentChunk(net.minecraft.world.level.chunk.Chunk chunk) {
|
|
||||||
+ this.currentChunk = chunk != null ? new java.lang.ref.WeakReference<>(chunk) : null;
|
|
||||||
+ }
|
|
||||||
+ /**
|
|
||||||
+ * Returns the entities current registered chunk. If the entity is not added to a chunk yet, it will return null
|
|
||||||
+ */
|
|
||||||
+ public net.minecraft.world.level.chunk.Chunk getCurrentChunk() {
|
|
||||||
+ final net.minecraft.world.level.chunk.Chunk chunk = currentChunk != null ? currentChunk.get() : null;
|
|
||||||
+ if (chunk != null && chunk.loaded) {
|
|
||||||
+ return chunk;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return !inChunk ? null : ((WorldServer)world).getChunkProvider().getChunkAtIfLoadedMainThreadNoCache(chunkX, chunkZ);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
private MinecraftKey entityKey;
|
|
||||||
private String entityKeyString;
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
index 2b58ae6d91fe0d0f36eedbb78a3c8a8a66d92405..75110c41af3e0097aef65091a2497dd87d08b4b2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
@@ -11,6 +11,7 @@ import net.minecraft.world.level.World;
|
|
||||||
import net.minecraft.world.level.block.EnumBlockMirror;
|
|
||||||
import net.minecraft.world.level.block.EnumBlockRotation;
|
|
||||||
import net.minecraft.world.level.block.state.IBlockData;
|
|
||||||
+import net.minecraft.world.level.chunk.Chunk;
|
|
||||||
import org.apache.logging.log4j.LogManager;
|
|
||||||
import org.apache.logging.log4j.Logger;
|
|
||||||
import org.apache.logging.log4j.util.Supplier;
|
|
||||||
@@ -63,6 +64,15 @@ public abstract class TileEntity implements net.minecraft.server.KeyedObject { /
|
|
||||||
getMinecraftKey(); // Try to load if it doesn't exists.
|
|
||||||
return tileEntityKeyString;
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ private java.lang.ref.WeakReference<Chunk> currentChunk = null;
|
|
||||||
+ public Chunk getCurrentChunk() {
|
|
||||||
+ final Chunk chunk = currentChunk != null ? currentChunk.get() : null;
|
|
||||||
+ return chunk != null && chunk.loaded ? chunk : null;
|
|
||||||
+ }
|
|
||||||
+ public void setCurrentChunk(Chunk chunk) {
|
|
||||||
+ this.currentChunk = chunk != null ? new java.lang.ref.WeakReference<>(chunk) : null;
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index b15200c2a3923bd8be2ee5e73fdadfeea3e3a8dc..929f6fcd4b9f1b9a1488e170d6a77a5d64beecf3 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -90,11 +90,36 @@ public class Chunk implements IChunkAccess {
|
|
||||||
this(world, chunkcoordintpair, biomestorage, ChunkConverter.a, TickListEmpty.b(), TickListEmpty.b(), 0L, (ChunkSection[]) null, (Consumer) null);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ private class TileEntityHashMap extends java.util.HashMap<BlockPosition, TileEntity> {
|
|
||||||
+ @Override
|
|
||||||
+ public TileEntity put(BlockPosition key, TileEntity value) {
|
|
||||||
+ TileEntity replaced = super.put(key, value);
|
|
||||||
+ if (replaced != null) {
|
|
||||||
+ replaced.setCurrentChunk(null);
|
|
||||||
+ }
|
|
||||||
+ if (value != null) {
|
|
||||||
+ value.setCurrentChunk(Chunk.this);
|
|
||||||
+ }
|
|
||||||
+ return replaced;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public TileEntity remove(Object key) {
|
|
||||||
+ TileEntity removed = super.remove(key);
|
|
||||||
+ if (removed != null) {
|
|
||||||
+ removed.setCurrentChunk(null);
|
|
||||||
+ }
|
|
||||||
+ return removed;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
public Chunk(World world, ChunkCoordIntPair chunkcoordintpair, BiomeStorage biomestorage, ChunkConverter chunkconverter, TickList<Block> ticklist, TickList<FluidType> ticklist1, long i, @Nullable ChunkSection[] achunksection, @Nullable Consumer<Chunk> consumer) {
|
|
||||||
this.sections = new ChunkSection[16];
|
|
||||||
this.e = Maps.newHashMap();
|
|
||||||
this.heightMap = Maps.newEnumMap(HeightMap.Type.class);
|
|
||||||
- this.tileEntities = Maps.newHashMap();
|
|
||||||
+ this.tileEntities = new TileEntityHashMap(); // Paper
|
|
||||||
this.l = Maps.newHashMap();
|
|
||||||
this.m = Maps.newHashMap();
|
|
||||||
this.n = new ShortList[16];
|
|
||||||
@@ -505,6 +530,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
}
|
|
||||||
|
|
||||||
entity.inChunk = true;
|
|
||||||
+ entity.setCurrentChunk(this); // Paper
|
|
||||||
entity.chunkX = this.loc.x;
|
|
||||||
entity.chunkY = k;
|
|
||||||
entity.chunkZ = this.loc.z;
|
|
||||||
@@ -517,6 +543,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
((HeightMap) this.heightMap.get(heightmap_type)).a(along);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public final void removeEntity(Entity entity) { this.b(entity); } // Paper - OBFHELPER
|
|
||||||
public void b(Entity entity) {
|
|
||||||
this.a(entity, entity.chunkY);
|
|
||||||
}
|
|
||||||
@@ -531,7 +558,12 @@ public class Chunk implements IChunkAccess {
|
|
||||||
i = this.entitySlices.length - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
- this.entitySlices[i].remove(entity);
|
|
||||||
+ // Paper start
|
|
||||||
+ if (entity.currentChunk != null && entity.currentChunk.get() == this) entity.setCurrentChunk(null);
|
|
||||||
+ if (!this.entitySlices[i].remove(entity)) {
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
this.entities.remove(entity); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
index df7e5f1d17ddfeffc15df02906c3bf9f9461d82b..eea242af23825ad29ada6e997205e87edffb6bb9 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
@@ -145,6 +145,7 @@ import net.minecraft.world.entity.vehicle.EntityMinecartMobSpawner;
|
|
||||||
import net.minecraft.world.entity.vehicle.EntityMinecartRideable;
|
|
||||||
import net.minecraft.world.entity.vehicle.EntityMinecartTNT;
|
|
||||||
import net.minecraft.world.phys.AxisAlignedBB;
|
|
||||||
+import org.bukkit.Chunk; // Paper
|
|
||||||
import org.bukkit.EntityEffect;
|
|
||||||
import org.bukkit.Location;
|
|
||||||
import org.bukkit.Server;
|
|
||||||
@@ -186,6 +187,12 @@ public abstract class CraftEntity implements org.bukkit.entity.Entity {
|
|
||||||
this.entity = entity;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ @Override
|
|
||||||
+ public Chunk getChunk() {
|
|
||||||
+ net.minecraft.world.level.chunk.Chunk currentChunk = entity.getCurrentChunk();
|
|
||||||
+ return currentChunk != null ? currentChunk.bukkitChunk : getLocation().getChunk();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
public static CraftEntity getEntity(CraftServer server, Entity entity) {
|
|
||||||
/*
|
|
||||||
* Order is *EXTREMELY* important -- keep it right! =D
|
|
@ -1,55 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 4 Jul 2018 02:13:59 -0400
|
|
||||||
Subject: [PATCH] Store counts for each Entity/Block Entity Type
|
|
||||||
|
|
||||||
Opens door for future patches to optimize performance
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index 929f6fcd4b9f1b9a1488e170d6a77a5d64beecf3..acdcece38a4b30d6c89eb4342918ae8997db9f0b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -91,15 +91,19 @@ public class Chunk implements IChunkAccess {
|
|
||||||
}
|
|
||||||
|
|
||||||
// Paper start
|
|
||||||
+ public final co.aikar.util.Counter<String> entityCounts = new co.aikar.util.Counter<>();
|
|
||||||
+ public final co.aikar.util.Counter<String> tileEntityCounts = new co.aikar.util.Counter<>();
|
|
||||||
private class TileEntityHashMap extends java.util.HashMap<BlockPosition, TileEntity> {
|
|
||||||
@Override
|
|
||||||
public TileEntity put(BlockPosition key, TileEntity value) {
|
|
||||||
TileEntity replaced = super.put(key, value);
|
|
||||||
if (replaced != null) {
|
|
||||||
replaced.setCurrentChunk(null);
|
|
||||||
+ tileEntityCounts.decrement(replaced.getMinecraftKeyString());
|
|
||||||
}
|
|
||||||
if (value != null) {
|
|
||||||
value.setCurrentChunk(Chunk.this);
|
|
||||||
+ tileEntityCounts.increment(value.getMinecraftKeyString());
|
|
||||||
}
|
|
||||||
return replaced;
|
|
||||||
}
|
|
||||||
@@ -109,6 +113,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
TileEntity removed = super.remove(key);
|
|
||||||
if (removed != null) {
|
|
||||||
removed.setCurrentChunk(null);
|
|
||||||
+ tileEntityCounts.decrement(removed.getMinecraftKeyString());
|
|
||||||
}
|
|
||||||
return removed;
|
|
||||||
}
|
|
||||||
@@ -529,6 +534,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
k = this.entitySlices.length - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ if (!entity.inChunk || entity.getCurrentChunk() != this) entityCounts.increment(entity.getMinecraftKeyString()); // Paper
|
|
||||||
entity.inChunk = true;
|
|
||||||
entity.setCurrentChunk(this); // Paper
|
|
||||||
entity.chunkX = this.loc.x;
|
|
||||||
@@ -563,6 +569,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
if (!this.entitySlices[i].remove(entity)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
+ entityCounts.decrement(entity.getMinecraftKeyString());
|
|
||||||
// Paper end
|
|
||||||
this.entities.remove(entity); // Paper
|
|
||||||
}
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -1,114 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 13:02:51 -0600
|
|
||||||
Subject: [PATCH] Configurable cactus bamboo and reed growth heights
|
|
||||||
|
|
||||||
Bamboo - Both the minimum fully-grown heights and the maximum are configurable
|
|
||||||
- Machine_Maker
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index b31109d2dadd29e8852468c19265066b773d2be0..3618cc017feb60e257a28f67cbddca3f792a9833 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -65,4 +65,17 @@ public class PaperWorldConfig {
|
|
||||||
config.addDefault("world-settings.default." + path, def);
|
|
||||||
return config.getString("world-settings." + worldName + "." + path, config.getString("world-settings.default." + path));
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int cactusMaxHeight;
|
|
||||||
+ public int reedMaxHeight;
|
|
||||||
+ public int bambooMaxHeight;
|
|
||||||
+ public int bambooMinHeight;
|
|
||||||
+ private void blockGrowthHeight() {
|
|
||||||
+ cactusMaxHeight = getInt("max-growth-height.cactus", 3);
|
|
||||||
+ reedMaxHeight = getInt("max-growth-height.reeds", 3);
|
|
||||||
+ bambooMaxHeight = getInt("max-growth-height.bamboo.max", 16);
|
|
||||||
+ bambooMinHeight = getInt("max-growth-height.bamboo.min", 11);
|
|
||||||
+ log("Max height for cactus growth " + cactusMaxHeight + ". Max height for reed growth " + reedMaxHeight + ". Max height for bamboo growth " + bambooMaxHeight + ". Min height for fully-grown bamboo " + bambooMinHeight + ".");
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockBamboo.java b/src/main/java/net/minecraft/world/level/block/BlockBamboo.java
|
|
||||||
index 6d10a7cb48627f37dc91b73064edc5f18c2b5ce9..cd132eff3f82349518555d1d5a16778ca08a521b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockBamboo.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockBamboo.java
|
|
||||||
@@ -124,7 +124,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
if (random.nextInt(Math.max(1, (int) (100.0F / worldserver.spigotConfig.bambooModifier) * 3)) == 0 && worldserver.isEmpty(blockposition.up()) && worldserver.getLightLevel(blockposition.up(), 0) >= 9) { // Spigot
|
|
||||||
int i = this.b(worldserver, blockposition) + 1;
|
|
||||||
|
|
||||||
- if (i < 16) {
|
|
||||||
+ if (i < worldserver.paperConfig.bambooMaxHeight) { // Paper
|
|
||||||
this.a(iblockdata, (World) worldserver, blockposition, random, i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -155,7 +155,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
int i = this.a(iblockaccess, blockposition);
|
|
||||||
int j = this.b(iblockaccess, blockposition);
|
|
||||||
|
|
||||||
- return i + j + 1 < 16 && (Integer) iblockaccess.getType(blockposition.up(i)).get(BlockBamboo.f) != 1;
|
|
||||||
+ return i + j + 1 < ((World) iblockaccess).paperConfig.bambooMaxHeight && (Integer) iblockaccess.getType(blockposition.up(i)).get(BlockBamboo.f) != 1; // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
@@ -174,7 +174,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
BlockPosition blockposition1 = blockposition.up(i);
|
|
||||||
IBlockData iblockdata1 = worldserver.getType(blockposition1);
|
|
||||||
|
|
||||||
- if (k >= 16 || !iblockdata1.a(Blocks.BAMBOO) || (Integer) iblockdata1.get(BlockBamboo.f) == 1 || !worldserver.isEmpty(blockposition1.up())) { // CraftBukkit - If the BlockSpreadEvent was cancelled, we have no bamboo here
|
|
||||||
+ if (k >= worldserver.paperConfig.bambooMaxHeight || !iblockdata1.a(Blocks.BAMBOO) || (Integer) iblockdata1.get(BlockBamboo.f) == 1 || !worldserver.isEmpty(blockposition1.up())) { // CraftBukkit - If the BlockSpreadEvent was cancelled, we have no bamboo here // Paper - Configurable cactus bamboo and reed growth heights
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -215,7 +215,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
}
|
|
||||||
|
|
||||||
int j = (Integer) iblockdata.get(BlockBamboo.d) != 1 && !iblockdata2.a(Blocks.BAMBOO) ? 0 : 1;
|
|
||||||
- int k = (i < 11 || random.nextFloat() >= 0.25F) && i != 15 ? 0 : 1;
|
|
||||||
+ int k = (i < world.paperConfig.bambooMinHeight || random.nextFloat() >= 0.25F) && i != (world.paperConfig.bambooMaxHeight - 1) ? 0 : 1; // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
if (org.bukkit.craftbukkit.event.CraftEventFactory.handleBlockSpreadEvent(world, blockposition, blockposition.up(), (IBlockData) ((IBlockData) ((IBlockData) this.getBlockData().set(BlockBamboo.d, j)).set(BlockBamboo.e, blockpropertybamboosize)).set(BlockBamboo.f, k), 3)) {
|
|
||||||
@@ -230,7 +230,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
protected int a(IBlockAccess iblockaccess, BlockPosition blockposition) {
|
|
||||||
int i;
|
|
||||||
|
|
||||||
- for (i = 0; i < 16 && iblockaccess.getType(blockposition.up(i + 1)).a(Blocks.BAMBOO); ++i) {
|
|
||||||
+ for (i = 0; i < ((World) iblockaccess).paperConfig.bambooMaxHeight && iblockaccess.getType(blockposition.up(i + 1)).a(Blocks.BAMBOO); ++i) { // Paper
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -240,7 +240,7 @@ public class BlockBamboo extends Block implements IBlockFragilePlantElement {
|
|
||||||
protected int b(IBlockAccess iblockaccess, BlockPosition blockposition) {
|
|
||||||
int i;
|
|
||||||
|
|
||||||
- for (i = 0; i < 16 && iblockaccess.getType(blockposition.down(i + 1)).a(Blocks.BAMBOO); ++i) {
|
|
||||||
+ for (i = 0; i < ((World) iblockaccess).paperConfig.bambooMaxHeight && iblockaccess.getType(blockposition.down(i + 1)).a(Blocks.BAMBOO); ++i) { // Paper
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockCactus.java b/src/main/java/net/minecraft/world/level/block/BlockCactus.java
|
|
||||||
index c2316e346f9b647edae4c9709defa4531e04eaa6..9f1e9fc5361cd051b909e2e1b2095722064185da 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockCactus.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockCactus.java
|
|
||||||
@@ -55,7 +55,7 @@ public class BlockCactus extends Block {
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
- if (i < 3) {
|
|
||||||
+ if (i < worldserver.paperConfig.cactusMaxHeight) { // Paper - Configurable growth height
|
|
||||||
int j = (Integer) iblockdata.get(BlockCactus.AGE);
|
|
||||||
|
|
||||||
if (j >= (byte) range(3, ((100.0F / worldserver.spigotConfig.cactusModifier) * 15) + 0.5F, 15)) { // Spigot
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockReed.java b/src/main/java/net/minecraft/world/level/block/BlockReed.java
|
|
||||||
index eb95d65e1a37b91a71a9ad8968710ba9047f3980..a4ede7968ba0134f0d2cf880a6b4a6b9a81f2fcd 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockReed.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockReed.java
|
|
||||||
@@ -51,7 +51,7 @@ public class BlockReed extends Block {
|
|
||||||
;
|
|
||||||
}
|
|
||||||
|
|
||||||
- if (i < 3) {
|
|
||||||
+ if (i < worldserver.paperConfig.reedMaxHeight) { // Paper - Configurable growth height
|
|
||||||
int j = (Integer) iblockdata.get(BlockReed.AGE);
|
|
||||||
|
|
||||||
if (j >= (byte) range(3, ((100.0F / worldserver.spigotConfig.caneModifier) * 15) + 0.5F, 15)) { // Spigot
|
|
@ -1,51 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 13:09:16 -0600
|
|
||||||
Subject: [PATCH] Configurable baby zombie movement speed
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 3618cc017feb60e257a28f67cbddca3f792a9833..796c17e0941922a9716212c6eae91643d8360418 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -78,4 +78,15 @@ public class PaperWorldConfig {
|
|
||||||
log("Max height for cactus growth " + cactusMaxHeight + ". Max height for reed growth " + reedMaxHeight + ". Max height for bamboo growth " + bambooMaxHeight + ". Min height for fully-grown bamboo " + bambooMinHeight + ".");
|
|
||||||
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public double babyZombieMovementModifier;
|
|
||||||
+ private void babyZombieMovementModifier() {
|
|
||||||
+ babyZombieMovementModifier = getDouble("baby-zombie-movement-modifier", 0.5D);
|
|
||||||
+ if (PaperConfig.version < 20) {
|
|
||||||
+ babyZombieMovementModifier = getDouble("baby-zombie-movement-speed", 0.5D);
|
|
||||||
+ set("baby-zombie-movement-modifier", babyZombieMovementModifier);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ log("Baby zombies will move at the speed of " + babyZombieMovementModifier);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/monster/EntityZombie.java b/src/main/java/net/minecraft/world/entity/monster/EntityZombie.java
|
|
||||||
index daa6d1ca20f952971a2ad6a0c4cba0bef6e43bf6..219e3b1626d68ede57b08a706d24bb6bc4b13fac 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/monster/EntityZombie.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/monster/EntityZombie.java
|
|
||||||
@@ -82,7 +82,7 @@ import org.bukkit.event.entity.EntityTransformEvent;
|
|
||||||
public class EntityZombie extends EntityMonster {
|
|
||||||
|
|
||||||
private static final UUID b = UUID.fromString("B9766B59-9566-4402-BC1F-2EE2A276D836");
|
|
||||||
- private static final AttributeModifier c = new AttributeModifier(EntityZombie.b, "Baby speed boost", 0.5D, AttributeModifier.Operation.MULTIPLY_BASE);
|
|
||||||
+ private final AttributeModifier c = new AttributeModifier(EntityZombie.b, "Baby speed boost", 0.5D, AttributeModifier.Operation.MULTIPLY_BASE); private final AttributeModifier babyModifier = this.c; // Paper - remove static - Make baby speed configurable
|
|
||||||
private static final DataWatcherObject<Boolean> d = DataWatcher.a(EntityZombie.class, DataWatcherRegistry.i);
|
|
||||||
private static final DataWatcherObject<Integer> bo = DataWatcher.a(EntityZombie.class, DataWatcherRegistry.b);
|
|
||||||
public static final DataWatcherObject<Boolean> DROWN_CONVERTING = DataWatcher.a(EntityZombie.class, DataWatcherRegistry.i);
|
|
||||||
@@ -185,9 +185,9 @@ public class EntityZombie extends EntityMonster {
|
|
||||||
if (this.world != null && !this.world.isClientSide) {
|
|
||||||
AttributeModifiable attributemodifiable = this.getAttributeInstance(GenericAttributes.MOVEMENT_SPEED);
|
|
||||||
|
|
||||||
- attributemodifiable.removeModifier(EntityZombie.c);
|
|
||||||
+ attributemodifiable.removeModifier(this.babyModifier); // Paper
|
|
||||||
if (flag) {
|
|
||||||
- attributemodifiable.b(EntityZombie.c);
|
|
||||||
+ attributemodifiable.b(this.babyModifier); // Paper
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,38 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 13:14:11 -0600
|
|
||||||
Subject: [PATCH] Configurable fishing time ranges
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 796c17e0941922a9716212c6eae91643d8360418..78948c42b13194005bdbbbc69c2b7ae0732a78c5 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -89,4 +89,12 @@ public class PaperWorldConfig {
|
|
||||||
|
|
||||||
log("Baby zombies will move at the speed of " + babyZombieMovementModifier);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int fishingMinTicks;
|
|
||||||
+ public int fishingMaxTicks;
|
|
||||||
+ private void fishingTickRange() {
|
|
||||||
+ fishingMinTicks = getInt("fishing-time-range.MinimumTicks", 100);
|
|
||||||
+ fishingMaxTicks = getInt("fishing-time-range.MaximumTicks", 600);
|
|
||||||
+ log("Fishing time ranges are between " + fishingMinTicks +" and " + fishingMaxTicks + " ticks");
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/projectile/EntityFishingHook.java b/src/main/java/net/minecraft/world/entity/projectile/EntityFishingHook.java
|
|
||||||
index 382b40a2a030993e1e98f34bc91befdd45c16895..7402e3b305d9bb7b27b97ff2078c26dd578232a1 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/projectile/EntityFishingHook.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/projectile/EntityFishingHook.java
|
|
||||||
@@ -85,6 +85,10 @@ public class EntityFishingHook extends IProjectile {
|
|
||||||
entityhuman.hookedFish = this;
|
|
||||||
this.an = Math.max(0, i);
|
|
||||||
this.lureLevel = Math.max(0, j);
|
|
||||||
+ // Paper start
|
|
||||||
+ minWaitTime = world.paperConfig.fishingMinTicks;
|
|
||||||
+ maxWaitTime = world.paperConfig.fishingMaxTicks;
|
|
||||||
+ // paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
public EntityFishingHook(EntityHuman entityhuman, World world, int i, int j) {
|
|
@ -1,98 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 13:24:16 -0600
|
|
||||||
Subject: [PATCH] Allow nerfed mobs to jump and take water damage
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 78948c42b13194005bdbbbc69c2b7ae0732a78c5..b41e7922dd96c3358eb849ab39982a75736e3476 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -97,4 +97,9 @@ public class PaperWorldConfig {
|
|
||||||
fishingMaxTicks = getInt("fishing-time-range.MaximumTicks", 600);
|
|
||||||
log("Fishing time ranges are between " + fishingMinTicks +" and " + fishingMaxTicks + " ticks");
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean nerfedMobsShouldJump;
|
|
||||||
+ private void nerfedMobsShouldJump() {
|
|
||||||
+ nerfedMobsShouldJump = getBoolean("spawner-nerfed-mobs-should-jump", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index 306f6c0db2333cce5dfc4bf1c09bfef05119a28b..f823763a2f7f40d0be8d058a1bd61386bcd951e6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -1105,6 +1105,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
return this.isInWater() || this.isInRain();
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public final boolean isInWaterOrRainOrBubble() { return aG(); } // Paper - OBFHELPER
|
|
||||||
public boolean aG() {
|
|
||||||
return this.isInWater() || this.isInRain() || this.k();
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityInsentient.java b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
index dbfcdc3cc7c1dccf785f5e13634e84c5af088985..28aa0a9361e8a32763d7fe1af060f0016e8c1e50 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
@@ -98,6 +98,7 @@ public abstract class EntityInsentient extends EntityLiving {
|
|
||||||
private final EntityAIBodyControl c;
|
|
||||||
protected NavigationAbstract navigation;
|
|
||||||
public PathfinderGoalSelector goalSelector;
|
|
||||||
+ @Nullable public PathfinderGoalFloat goalFloat; // Paper
|
|
||||||
public PathfinderGoalSelector targetSelector;
|
|
||||||
private EntityLiving goalTarget;
|
|
||||||
private final EntitySenses bo;
|
|
||||||
@@ -784,7 +785,17 @@ public abstract class EntityInsentient extends EntityLiving {
|
|
||||||
@Override
|
|
||||||
protected final void doTick() {
|
|
||||||
++this.ticksFarFromPlayer;
|
|
||||||
- if (!this.aware) return; // CraftBukkit
|
|
||||||
+ if (!this.aware) { // Paper start - Allow nerfed mobs to jump, float and take water damage
|
|
||||||
+ if (goalFloat != null) {
|
|
||||||
+ if (goalFloat.validConditions()) goalFloat.update();
|
|
||||||
+ this.getControllerJump().jumpIfSet();
|
|
||||||
+ }
|
|
||||||
+ if ((this instanceof EntityBlaze || this instanceof EntityEnderman) && isInWaterOrRainOrBubble()) {
|
|
||||||
+ damageEntity(DamageSource.DROWN, 1.0F);
|
|
||||||
+ }
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
this.world.getMethodProfiler().enter("sensing");
|
|
||||||
this.bo.a();
|
|
||||||
this.world.getMethodProfiler().exit();
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/control/ControllerJump.java b/src/main/java/net/minecraft/world/entity/ai/control/ControllerJump.java
|
|
||||||
index 9767ac416fcd60a8a57b648dcb3f1e427bacd54d..1a9b3e0e0c090683e332dfa53708f8a62c8f14e0 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/control/ControllerJump.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/control/ControllerJump.java
|
|
||||||
@@ -15,6 +15,7 @@ public class ControllerJump {
|
|
||||||
this.a = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public final void jumpIfSet() { this.b(); } // Paper - OBFHELPER
|
|
||||||
public void b() {
|
|
||||||
this.b.setJumping(this.a);
|
|
||||||
this.a = false;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/goal/PathfinderGoalFloat.java b/src/main/java/net/minecraft/world/entity/ai/goal/PathfinderGoalFloat.java
|
|
||||||
index 8dfa1a6ade7f51e5d68b290f5376d999bb4c60ab..a6c8763139ed18fe73b2d6f6ec511e59666dc843 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/goal/PathfinderGoalFloat.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/goal/PathfinderGoalFloat.java
|
|
||||||
@@ -11,15 +11,18 @@ public class PathfinderGoalFloat extends PathfinderGoal {
|
|
||||||
|
|
||||||
public PathfinderGoalFloat(EntityInsentient entityinsentient) {
|
|
||||||
this.a = entityinsentient;
|
|
||||||
+ if (entityinsentient.getWorld().paperConfig.nerfedMobsShouldJump) entityinsentient.goalFloat = this; // Paper
|
|
||||||
this.a(EnumSet.of(PathfinderGoal.Type.JUMP));
|
|
||||||
entityinsentient.getNavigation().d(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public final boolean validConditions() { return this.a(); } // Paper - OBFHELPER
|
|
||||||
@Override
|
|
||||||
public boolean a() {
|
|
||||||
return this.a.isInWater() && this.a.b((Tag) TagsFluid.WATER) > this.a.cx() || this.a.aQ();
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public void update() { this.e(); } // Paper - OBFHELPER
|
|
||||||
@Override
|
|
||||||
public void e() {
|
|
||||||
if (this.a.getRandom().nextFloat() < 0.8F) {
|
|
@ -1,55 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Suddenly <suddenly@suddenly.coffee>
|
|
||||||
Date: Tue, 1 Mar 2016 13:51:54 -0600
|
|
||||||
Subject: [PATCH] Add configurable despawn distances for living entities
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index b41e7922dd96c3358eb849ab39982a75736e3476..2f0d582baf0eb2bb477944d0cb1369db6ca33956 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -102,4 +102,20 @@ public class PaperWorldConfig {
|
|
||||||
private void nerfedMobsShouldJump() {
|
|
||||||
nerfedMobsShouldJump = getBoolean("spawner-nerfed-mobs-should-jump", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int softDespawnDistance;
|
|
||||||
+ public int hardDespawnDistance;
|
|
||||||
+ private void despawnDistances() {
|
|
||||||
+ softDespawnDistance = getInt("despawn-ranges.soft", 32); // 32^2 = 1024, Minecraft Default
|
|
||||||
+ hardDespawnDistance = getInt("despawn-ranges.hard", 128); // 128^2 = 16384, Minecraft Default
|
|
||||||
+
|
|
||||||
+ if (softDespawnDistance > hardDespawnDistance) {
|
|
||||||
+ softDespawnDistance = hardDespawnDistance;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ log("Living Entity Despawn Ranges: Soft: " + softDespawnDistance + " Hard: " + hardDespawnDistance);
|
|
||||||
+
|
|
||||||
+ softDespawnDistance = softDespawnDistance*softDespawnDistance;
|
|
||||||
+ hardDespawnDistance = hardDespawnDistance*hardDespawnDistance;
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityInsentient.java b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
index 28aa0a9361e8a32763d7fe1af060f0016e8c1e50..6e30fc88fa7a3ff00c9b4b78842c3a533649bd50 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
@@ -763,16 +763,16 @@ public abstract class EntityInsentient extends EntityLiving {
|
|
||||||
int i = this.getEntityType().e().f();
|
|
||||||
int j = i * i;
|
|
||||||
|
|
||||||
- if (d0 > (double) j) { // CraftBukkit - remove isTypeNotPersistent() check
|
|
||||||
+ if (d0 > (double) world.paperConfig.hardDespawnDistance) { // CraftBukkit - remove isTypeNotPersistent() check // Paper - custom despawn distances
|
|
||||||
this.die();
|
|
||||||
}
|
|
||||||
|
|
||||||
int k = this.getEntityType().e().g();
|
|
||||||
int l = k * k;
|
|
||||||
|
|
||||||
- if (this.ticksFarFromPlayer > 600 && this.random.nextInt(800) == 0 && d0 > (double) l) { // CraftBukkit - remove isTypeNotPersistent() check
|
|
||||||
+ if (this.ticksFarFromPlayer > 600 && this.random.nextInt(800) == 0 && d0 > world.paperConfig.softDespawnDistance) { // CraftBukkit - remove isTypeNotPersistent() check // Paper - custom despawn distances
|
|
||||||
this.die();
|
|
||||||
- } else if (d0 < (double) l) {
|
|
||||||
+ } else if (d0 < world.paperConfig.softDespawnDistance) { // Paper - custom despawn distances
|
|
||||||
this.ticksFarFromPlayer = 0;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Thu, 3 Mar 2016 03:53:43 -0600
|
|
||||||
Subject: [PATCH] Allow for toggling of spawn chunks
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 2f0d582baf0eb2bb477944d0cb1369db6ca33956..89e76dd73811fd0f6f8c8e7e5af804d5a4bb5a75 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -118,4 +118,10 @@ public class PaperWorldConfig {
|
|
||||||
softDespawnDistance = softDespawnDistance*softDespawnDistance;
|
|
||||||
hardDespawnDistance = hardDespawnDistance*hardDespawnDistance;
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean keepSpawnInMemory;
|
|
||||||
+ private void keepSpawnInMemory() {
|
|
||||||
+ keepSpawnInMemory = getBoolean("keep-spawn-loaded", true);
|
|
||||||
+ log("Keep spawn chunk loaded: " + keepSpawnInMemory);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 1b03c69fa7ffeca4083470a179e18be828b0a957..62735e7908f86e31f7a89ce2071ddebe176c0385 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -217,6 +217,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
});
|
|
||||||
// CraftBukkit end
|
|
||||||
timings = new co.aikar.timings.WorldTimingsHandler(this); // Paper - code below can generate new world and access timings
|
|
||||||
+ this.keepSpawnInMemory = this.paperConfig.keepSpawnInMemory; // Paper
|
|
||||||
this.entityLimiter = new org.spigotmc.TickLimiter(spigotConfig.entityMaxTickTime);
|
|
||||||
this.tileLimiter = new org.spigotmc.TickLimiter(spigotConfig.tileMaxTickTime);
|
|
||||||
}
|
|
@ -1,94 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Byteflux <byte@byteflux.net>
|
|
||||||
Date: Tue, 1 Mar 2016 14:14:15 -0600
|
|
||||||
Subject: [PATCH] Drop falling block and tnt entities at the specified height
|
|
||||||
|
|
||||||
* Dec 2, 2020 Added tnt nerf for tnt minecarts - Machine_Maker
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 89e76dd73811fd0f6f8c8e7e5af804d5a4bb5a75..d16ae924bcbe31c964f7fb448757c748e5c4418c 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -124,4 +124,14 @@ public class PaperWorldConfig {
|
|
||||||
keepSpawnInMemory = getBoolean("keep-spawn-loaded", true);
|
|
||||||
log("Keep spawn chunk loaded: " + keepSpawnInMemory);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int fallingBlockHeightNerf;
|
|
||||||
+ public int entityTNTHeightNerf;
|
|
||||||
+ private void heightNerfs() {
|
|
||||||
+ fallingBlockHeightNerf = getInt("falling-block-height-nerf", 0);
|
|
||||||
+ entityTNTHeightNerf = getInt("tnt-entity-height-nerf", 0);
|
|
||||||
+
|
|
||||||
+ if (fallingBlockHeightNerf != 0) log("Falling Block Height Limit set to Y: " + fallingBlockHeightNerf);
|
|
||||||
+ if (entityTNTHeightNerf != 0) log("TNT Entity Height Limit set to Y: " + entityTNTHeightNerf);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index f823763a2f7f40d0be8d058a1bd61386bcd951e6..20e4ff812960a54872f2fea8fe6baf7bb1ef077d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -1850,6 +1850,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
return this.a(itemstack, 0.0F);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ @Nullable public final EntityItem dropItem(ItemStack itemstack, float offset) { return this.a(itemstack, offset); } // Paper - OBFHELPER
|
|
||||||
@Nullable
|
|
||||||
public EntityItem a(ItemStack itemstack, float f) {
|
|
||||||
if (itemstack.isEmpty()) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java b/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
index b3dea8d65c3d1e27e59be142ec9e2ebb4164aed0..901522f24b8bc58861e46eda400dbab92bb6401d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
@@ -124,6 +124,17 @@ public class EntityFallingBlock extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
this.move(EnumMoveType.SELF, this.getMot());
|
|
||||||
+
|
|
||||||
+ // Paper start - Configurable EntityFallingBlock height nerf
|
|
||||||
+ if (this.world.paperConfig.fallingBlockHeightNerf != 0 && this.locY() > this.world.paperConfig.fallingBlockHeightNerf) {
|
|
||||||
+ if (this.dropItem && this.world.getGameRules().getBoolean(GameRules.DO_ENTITY_DROPS)) {
|
|
||||||
+ this.a(block);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ this.die();
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
if (!this.world.isClientSide) {
|
|
||||||
blockposition = this.getChunkCoordinates();
|
|
||||||
boolean flag = this.block.getBlock() instanceof BlockConcretePowder;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java b/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
index c2e7161dc103c971908ff217eaf972e9f175d044..4f4b2b8d58223fa22d6a7af5c94cfb36399b9641 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
@@ -70,6 +70,12 @@ public class EntityTNTPrimed extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
this.move(EnumMoveType.SELF, this.getMot());
|
|
||||||
+ // Paper start - Configurable TNT entity height nerf
|
|
||||||
+ if (this.world.paperConfig.entityTNTHeightNerf != 0 && this.locY() > this.world.paperConfig.entityTNTHeightNerf) {
|
|
||||||
+ this.die();
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
this.setMot(this.getMot().a(0.98D));
|
|
||||||
if (this.onGround) {
|
|
||||||
this.setMot(this.getMot().d(0.7D, -0.5D, 0.7D));
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartTNT.java b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartTNT.java
|
|
||||||
index ee31704ffb88ab68702657554d386e8ebfa05d03..4bec5c1d504b9456dafe1b76bdbb523d0a324abe 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartTNT.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartTNT.java
|
|
||||||
@@ -47,6 +47,12 @@ public class EntityMinecartTNT extends EntityMinecartAbstract {
|
|
||||||
public void tick() {
|
|
||||||
super.tick();
|
|
||||||
if (this.b > 0) {
|
|
||||||
+ // Paper start - Configurable TNT entity height nerf
|
|
||||||
+ if (this.world.paperConfig.entityTNTHeightNerf != 0 && this.locY() > this.world.paperConfig.entityTNTHeightNerf) {
|
|
||||||
+ this.die();
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
--this.b;
|
|
||||||
this.world.addParticle(Particles.SMOKE, this.locX(), this.locY() + 0.5D, this.locZ(), 0.0D, 0.0D, 0.0D);
|
|
||||||
} else if (this.b == 0) {
|
|
@ -1,117 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 14:32:43 -0600
|
|
||||||
Subject: [PATCH] Show 'Paper' in client crashes, server lists, and Mojang
|
|
||||||
stats
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/EULA.java b/src/main/java/net/minecraft/server/EULA.java
|
|
||||||
index a5171d28b960b12c2743ea68a36d747bc967697d..a0f53c9eff04a40780b3ba568dbfc5bbe9bd8504 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/EULA.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/EULA.java
|
|
||||||
@@ -72,7 +72,7 @@ public class EULA {
|
|
||||||
Properties properties = new Properties();
|
|
||||||
|
|
||||||
properties.setProperty("eula", "false");
|
|
||||||
- properties.store(outputstream, "By changing the setting below to TRUE you are indicating your agreement to our EULA (https://account.mojang.com/documents/minecraft_eula).");
|
|
||||||
+ properties.store(outputstream, "By changing the setting below to TRUE you are indicating your agreement to our EULA (https://account.mojang.com/documents/minecraft_eula).\nYou also agree that tacos are tasty, and the best food in the world."); // Paper - fix lag;
|
|
||||||
} catch (Throwable throwable1) {
|
|
||||||
throwable = throwable1;
|
|
||||||
throw throwable1;
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index 85c96e3f70a3ffecb6195a1d83053412eb180d78..c21790b4de698aa6f7fc4dadab64d791cd0562b6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -1342,7 +1342,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getServerModName() {
|
|
||||||
- return "Spigot"; // Spigot - Spigot > // CraftBukkit - cb > vanilla!
|
|
||||||
+ return "Paper"; //Paper - Paper > // Spigot - Spigot > // CraftBukkit - cb > vanilla!
|
|
||||||
}
|
|
||||||
|
|
||||||
public CrashReport b(CrashReport crashreport) {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index 2ad09749f3005c3eff143d83580e25910341aa6b..462633df2af43959fddf5b7a8ec43063abf7b14b 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -225,7 +225,7 @@ import org.yaml.snakeyaml.error.MarkedYAMLException;
|
|
||||||
import net.md_5.bungee.api.chat.BaseComponent; // Spigot
|
|
||||||
|
|
||||||
public final class CraftServer implements Server {
|
|
||||||
- private final String serverName = "CraftBukkit";
|
|
||||||
+ private final String serverName = "Paper"; // Paper
|
|
||||||
private final String serverVersion;
|
|
||||||
private final String bukkitVersion = Versioning.getBukkitVersion();
|
|
||||||
private final Logger logger = Logger.getLogger("Minecraft");
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/Main.java b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
index ce9f10f890a5866ab6208c7253b15b09fe323a81..e8c225fcd1a3fa5a7e1971683b1876dd6462a1e2 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/Main.java
|
|
||||||
@@ -206,12 +206,25 @@ public class Main {
|
|
||||||
deadline.add(Calendar.DAY_OF_YEAR, -28);
|
|
||||||
if (buildDate.before(deadline.getTime())) {
|
|
||||||
System.err.println("*** Error, this build is outdated ***");
|
|
||||||
- System.err.println("*** Please download a new build as per instructions from https://www.spigotmc.org/go/outdated-spigot ***");
|
|
||||||
+ System.err.println("*** Please download a new build as per instructions from https://papermc.io/downloads ***"); // Paper
|
|
||||||
System.err.println("*** Server will start in 20 seconds ***");
|
|
||||||
Thread.sleep(TimeUnit.SECONDS.toMillis(20));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Log Java and OS versioning to help with debugging plugin issues
|
|
||||||
+ java.lang.management.RuntimeMXBean runtimeMX = java.lang.management.ManagementFactory.getRuntimeMXBean();
|
|
||||||
+ java.lang.management.OperatingSystemMXBean osMX = java.lang.management.ManagementFactory.getOperatingSystemMXBean();
|
|
||||||
+ if (runtimeMX != null && osMX != null) {
|
|
||||||
+ String javaInfo = "Java " + runtimeMX.getSpecVersion() + " (" + runtimeMX.getVmName() + " " + runtimeMX.getVmVersion() + ")";
|
|
||||||
+ String osInfo = "Host: " + osMX.getName() + " " + osMX.getVersion() + " (" + osMX.getArch() + ")";
|
|
||||||
+
|
|
||||||
+ System.out.println("System Info: " + javaInfo + " " + osInfo);
|
|
||||||
+ } else {
|
|
||||||
+ System.out.println("Unable to read system info");
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
System.out.println("Loading libraries, please wait...");
|
|
||||||
net.minecraft.server.Main.main(options);
|
|
||||||
} catch (Throwable t) {
|
|
||||||
diff --git a/src/main/java/org/spigotmc/WatchdogThread.java b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
index b4eeeb0d001e692180874bd26385a0e786a8b752..1b3a14784cac8e855633fae6172ad5479ebe9877 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
@@ -19,7 +19,7 @@ public class WatchdogThread extends Thread
|
|
||||||
|
|
||||||
private WatchdogThread(long timeoutTime, boolean restart)
|
|
||||||
{
|
|
||||||
- super( "Spigot Watchdog Thread" );
|
|
||||||
+ super( "Paper Watchdog Thread" );
|
|
||||||
this.timeoutTime = timeoutTime;
|
|
||||||
this.restart = restart;
|
|
||||||
}
|
|
||||||
@@ -65,14 +65,14 @@ public class WatchdogThread extends Thread
|
|
||||||
{
|
|
||||||
Logger log = Bukkit.getServer().getLogger();
|
|
||||||
log.log( Level.SEVERE, "------------------------------" );
|
|
||||||
- log.log( Level.SEVERE, "The server has stopped responding! This is (probably) not a Spigot bug." );
|
|
||||||
+ log.log( Level.SEVERE, "The server has stopped responding! This is (probably) not a Paper bug." ); // Paper
|
|
||||||
log.log( Level.SEVERE, "If you see a plugin in the Server thread dump below, then please report it to that author" );
|
|
||||||
log.log( Level.SEVERE, "\t *Especially* if it looks like HTTP or MySQL operations are occurring" );
|
|
||||||
log.log( Level.SEVERE, "If you see a world save or edit, then it means you did far more than your server can handle at once" );
|
|
||||||
log.log( Level.SEVERE, "\t If this is the case, consider increasing timeout-time in spigot.yml but note that this will replace the crash with LARGE lag spikes" );
|
|
||||||
- log.log( Level.SEVERE, "If you are unsure or still think this is a Spigot bug, please report to https://www.spigotmc.org/" );
|
|
||||||
+ log.log( Level.SEVERE, "If you are unsure or still think this is a Paper bug, please report this to https://github.com/PaperMC/Paper/issues" );
|
|
||||||
log.log( Level.SEVERE, "Be sure to include ALL relevant console errors and Minecraft crash reports" );
|
|
||||||
- log.log( Level.SEVERE, "Spigot version: " + Bukkit.getServer().getVersion() );
|
|
||||||
+ log.log( Level.SEVERE, "Paper version: " + Bukkit.getServer().getVersion() );
|
|
||||||
//
|
|
||||||
if ( net.minecraft.world.level.World.lastPhysicsProblem != null )
|
|
||||||
{
|
|
||||||
@@ -82,7 +82,7 @@ public class WatchdogThread extends Thread
|
|
||||||
}
|
|
||||||
//
|
|
||||||
log.log( Level.SEVERE, "------------------------------" );
|
|
||||||
- log.log( Level.SEVERE, "Server thread dump (Look for plugins here before reporting to Spigot!):" );
|
|
||||||
+ log.log( Level.SEVERE, "Server thread dump (Look for plugins here before reporting to Paper!):" ); // Paper
|
|
||||||
dumpThread( ManagementFactory.getThreadMXBean().getThreadInfo( MinecraftServer.getServer().serverThread.getId(), Integer.MAX_VALUE ), log );
|
|
||||||
log.log( Level.SEVERE, "------------------------------" );
|
|
||||||
//
|
|
@ -1,150 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach@zachbr.io>
|
|
||||||
Date: Mon, 27 May 2019 03:40:05 -0500
|
|
||||||
Subject: [PATCH] Implement Paper VersionChecker
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java b/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..1a1b50e475b9ede544b2f6d0d36632b24b68898c
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java
|
|
||||||
@@ -0,0 +1,122 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import com.destroystokyo.paper.util.VersionFetcher;
|
|
||||||
+import com.google.common.base.Charsets;
|
|
||||||
+import com.google.common.io.Resources;
|
|
||||||
+import com.google.gson.*;
|
|
||||||
+import net.kyori.adventure.text.Component;
|
|
||||||
+import net.kyori.adventure.text.format.NamedTextColor;
|
|
||||||
+
|
|
||||||
+import javax.annotation.Nonnull;
|
|
||||||
+import javax.annotation.Nullable;
|
|
||||||
+import java.io.*;
|
|
||||||
+import java.net.HttpURLConnection;
|
|
||||||
+import java.net.URL;
|
|
||||||
+import java.util.stream.StreamSupport;
|
|
||||||
+
|
|
||||||
+public class PaperVersionFetcher implements VersionFetcher {
|
|
||||||
+ private static final java.util.regex.Pattern VER_PATTERN = java.util.regex.Pattern.compile("^([0-9\\.]*)\\-.*R"); // R is an anchor, will always give '-R' at end
|
|
||||||
+ private static final String GITHUB_BRANCH_NAME = "master";
|
|
||||||
+ private static @Nullable String mcVer;
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public long getCacheTime() {
|
|
||||||
+ return 720000;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Nonnull
|
|
||||||
+ @Override
|
|
||||||
+ public Component getVersionMessage(@Nonnull String serverVersion) {
|
|
||||||
+ String[] parts = serverVersion.substring("git-Paper-".length()).split("[-\\s]");
|
|
||||||
+ return getUpdateStatusMessage("PaperMC/Paper", GITHUB_BRANCH_NAME, parts[0]);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static @Nullable String getMinecraftVersion() {
|
|
||||||
+ if (mcVer == null) {
|
|
||||||
+ java.util.regex.Matcher matcher = VER_PATTERN.matcher(org.bukkit.Bukkit.getBukkitVersion());
|
|
||||||
+ if (matcher.find()) {
|
|
||||||
+ String result = matcher.group();
|
|
||||||
+ mcVer = result.substring(0, result.length() - 2); // strip 'R' anchor and trailing '-'
|
|
||||||
+ } else {
|
|
||||||
+ org.bukkit.Bukkit.getLogger().warning("Unable to match version to pattern! Report to PaperMC!");
|
|
||||||
+ org.bukkit.Bukkit.getLogger().warning("Pattern: " + VER_PATTERN.toString());
|
|
||||||
+ org.bukkit.Bukkit.getLogger().warning("Version: " + org.bukkit.Bukkit.getBukkitVersion());
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return mcVer;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static Component getUpdateStatusMessage(@Nonnull String repo, @Nonnull String branch, @Nonnull String versionInfo) {
|
|
||||||
+ int distance;
|
|
||||||
+ try {
|
|
||||||
+ int jenkinsBuild = Integer.parseInt(versionInfo);
|
|
||||||
+ distance = fetchDistanceFromSiteApi(jenkinsBuild, getMinecraftVersion());
|
|
||||||
+ } catch (NumberFormatException ignored) {
|
|
||||||
+ versionInfo = versionInfo.replace("\"", "");
|
|
||||||
+ distance = fetchDistanceFromGitHub(repo, branch, versionInfo);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ switch (distance) {
|
|
||||||
+ case -1:
|
|
||||||
+ return Component.text("Error obtaining version information", NamedTextColor.YELLOW);
|
|
||||||
+ case 0:
|
|
||||||
+ return Component.text("You are running the latest version", NamedTextColor.GREEN);
|
|
||||||
+ case -2:
|
|
||||||
+ return Component.text("Unknown version", NamedTextColor.YELLOW);
|
|
||||||
+ default:
|
|
||||||
+ return Component.text("You are " + distance + " version(s) behind", NamedTextColor.YELLOW);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static int fetchDistanceFromSiteApi(int jenkinsBuild, @Nullable String siteApiVersion) {
|
|
||||||
+ if (siteApiVersion == null) { return -1; }
|
|
||||||
+ try {
|
|
||||||
+ try (BufferedReader reader = Resources.asCharSource(
|
|
||||||
+ new URL("https://papermc.io/api/v2/projects/paper/versions/" + siteApiVersion),
|
|
||||||
+ Charsets.UTF_8
|
|
||||||
+ ).openBufferedStream()) {
|
|
||||||
+ JsonObject json = new Gson().fromJson(reader, JsonObject.class);
|
|
||||||
+ JsonArray builds = json.getAsJsonArray("builds");
|
|
||||||
+ int latest = StreamSupport.stream(builds.spliterator(), false)
|
|
||||||
+ .mapToInt(e -> e.getAsInt())
|
|
||||||
+ .max()
|
|
||||||
+ .getAsInt();
|
|
||||||
+ return latest - jenkinsBuild;
|
|
||||||
+ } catch (JsonSyntaxException ex) {
|
|
||||||
+ ex.printStackTrace();
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ } catch (IOException e) {
|
|
||||||
+ e.printStackTrace();
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Contributed by Techcable <Techcable@outlook.com> in GH-65
|
|
||||||
+ private static int fetchDistanceFromGitHub(@Nonnull String repo, @Nonnull String branch, @Nonnull String hash) {
|
|
||||||
+ try {
|
|
||||||
+ HttpURLConnection connection = (HttpURLConnection) new URL("https://api.github.com/repos/" + repo + "/compare/" + branch + "..." + hash).openConnection();
|
|
||||||
+ connection.connect();
|
|
||||||
+ if (connection.getResponseCode() == HttpURLConnection.HTTP_NOT_FOUND) return -2; // Unknown commit
|
|
||||||
+ try (BufferedReader reader = new BufferedReader(new InputStreamReader(connection.getInputStream(), Charsets.UTF_8))) {
|
|
||||||
+ JsonObject obj = new Gson().fromJson(reader, JsonObject.class);
|
|
||||||
+ String status = obj.get("status").getAsString();
|
|
||||||
+ switch (status) {
|
|
||||||
+ case "identical":
|
|
||||||
+ return 0;
|
|
||||||
+ case "behind":
|
|
||||||
+ return obj.get("behind_by").getAsInt();
|
|
||||||
+ default:
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ } catch (JsonSyntaxException | NumberFormatException e) {
|
|
||||||
+ e.printStackTrace();
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ } catch (IOException e) {
|
|
||||||
+ e.printStackTrace();
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/util/CraftMagicNumbers.java b/src/main/java/org/bukkit/craftbukkit/util/CraftMagicNumbers.java
|
|
||||||
index 8f737f63f280c00c1276bd1dc3ecf60448732ca8..8aa9e7796ea39c09a965750d06c3d358250f33b8 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/util/CraftMagicNumbers.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/util/CraftMagicNumbers.java
|
|
||||||
@@ -370,6 +370,11 @@ public final class CraftMagicNumbers implements UnsafeValues {
|
|
||||||
public String getTimingsServerName() {
|
|
||||||
return com.destroystokyo.paper.PaperConfig.timingsServerName;
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public com.destroystokyo.paper.util.VersionFetcher getVersionFetcher() {
|
|
||||||
+ return new com.destroystokyo.paper.PaperVersionFetcher();
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
/**
|
|
@ -1,215 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Kyle Wood <demonwav@gmail.com>
|
|
||||||
Date: Thu, 1 Mar 2018 19:37:52 -0600
|
|
||||||
Subject: [PATCH] Add version history to version command
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java b/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java
|
|
||||||
index 1a1b50e475b9ede544b2f6d0d36632b24b68898c..580bae0d414d371a07a6bfeefc41fdd989dc0083 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperVersionFetcher.java
|
|
||||||
@@ -5,7 +5,9 @@ import com.google.common.base.Charsets;
|
|
||||||
import com.google.common.io.Resources;
|
|
||||||
import com.google.gson.*;
|
|
||||||
import net.kyori.adventure.text.Component;
|
|
||||||
+import net.kyori.adventure.text.TextComponent;
|
|
||||||
import net.kyori.adventure.text.format.NamedTextColor;
|
|
||||||
+import net.kyori.adventure.text.format.TextDecoration;
|
|
||||||
|
|
||||||
import javax.annotation.Nonnull;
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
@@ -28,7 +30,10 @@ public class PaperVersionFetcher implements VersionFetcher {
|
|
||||||
@Override
|
|
||||||
public Component getVersionMessage(@Nonnull String serverVersion) {
|
|
||||||
String[] parts = serverVersion.substring("git-Paper-".length()).split("[-\\s]");
|
|
||||||
- return getUpdateStatusMessage("PaperMC/Paper", GITHUB_BRANCH_NAME, parts[0]);
|
|
||||||
+ final Component updateMessage = getUpdateStatusMessage("PaperMC/Paper", GITHUB_BRANCH_NAME, parts[0]);
|
|
||||||
+ final Component history = getHistory();
|
|
||||||
+
|
|
||||||
+ return history != null ? TextComponent.ofChildren(updateMessage, Component.newline(), history) : updateMessage;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static @Nullable String getMinecraftVersion() {
|
|
||||||
@@ -119,4 +124,19 @@ public class PaperVersionFetcher implements VersionFetcher {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ @Nullable
|
|
||||||
+ private Component getHistory() {
|
|
||||||
+ final VersionHistoryManager.VersionData data = VersionHistoryManager.INSTANCE.getVersionData();
|
|
||||||
+ if (data == null) {
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ final String oldVersion = data.getOldVersion();
|
|
||||||
+ if (oldVersion == null) {
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return Component.text("Previous version: " + oldVersion, NamedTextColor.GRAY, TextDecoration.ITALIC);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/VersionHistoryManager.java b/src/main/java/com/destroystokyo/paper/VersionHistoryManager.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..aac3f66cb23d260729c2a48d8710a9de2346aa22
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/VersionHistoryManager.java
|
|
||||||
@@ -0,0 +1,145 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import com.google.common.base.MoreObjects;
|
|
||||||
+import com.google.gson.Gson;
|
|
||||||
+import com.google.gson.JsonSyntaxException;
|
|
||||||
+import java.io.BufferedReader;
|
|
||||||
+import java.io.BufferedWriter;
|
|
||||||
+import java.io.IOException;
|
|
||||||
+import java.nio.charset.StandardCharsets;
|
|
||||||
+import java.nio.file.Files;
|
|
||||||
+import java.nio.file.Path;
|
|
||||||
+import java.nio.file.Paths;
|
|
||||||
+import java.nio.file.StandardOpenOption;
|
|
||||||
+import java.util.Objects;
|
|
||||||
+import java.util.logging.Level;
|
|
||||||
+import java.util.logging.Logger;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+
|
|
||||||
+import javax.annotation.Nonnull;
|
|
||||||
+import javax.annotation.Nullable;
|
|
||||||
+
|
|
||||||
+public enum VersionHistoryManager {
|
|
||||||
+ INSTANCE;
|
|
||||||
+
|
|
||||||
+ private final Gson gson = new Gson();
|
|
||||||
+
|
|
||||||
+ private final Logger logger = Bukkit.getLogger();
|
|
||||||
+
|
|
||||||
+ private VersionData currentData = null;
|
|
||||||
+
|
|
||||||
+ VersionHistoryManager() {
|
|
||||||
+ final Path path = Paths.get("version_history.json");
|
|
||||||
+
|
|
||||||
+ if (Files.exists(path)) {
|
|
||||||
+ // Basic file santiy checks
|
|
||||||
+ if (!Files.isRegularFile(path)) {
|
|
||||||
+ if (Files.isDirectory(path)) {
|
|
||||||
+ logger.severe(path + " is a directory, cannot be used for version history");
|
|
||||||
+ } else {
|
|
||||||
+ logger.severe(path + " is not a regular file, cannot be used for version history");
|
|
||||||
+ }
|
|
||||||
+ // We can't continue
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ try (final BufferedReader reader = Files.newBufferedReader(path, StandardCharsets.UTF_8)) {
|
|
||||||
+ currentData = gson.fromJson(reader, VersionData.class);
|
|
||||||
+ } catch (final IOException e) {
|
|
||||||
+ logger.log(Level.SEVERE, "Failed to read version history file '" + path + "'", e);
|
|
||||||
+ return;
|
|
||||||
+ } catch (final JsonSyntaxException e) {
|
|
||||||
+ logger.log(Level.SEVERE, "Invalid json syntax for file '" + path + "'", e);
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ final String version = Bukkit.getVersion();
|
|
||||||
+ if (version == null) {
|
|
||||||
+ logger.severe("Failed to retrieve current version");
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!version.equals(currentData.getCurrentVersion())) {
|
|
||||||
+ // The version appears to have changed
|
|
||||||
+ currentData.setOldVersion(currentData.getCurrentVersion());
|
|
||||||
+ currentData.setCurrentVersion(version);
|
|
||||||
+ writeFile(path);
|
|
||||||
+ }
|
|
||||||
+ } else {
|
|
||||||
+ // File doesn't exist, start fresh
|
|
||||||
+ currentData = new VersionData();
|
|
||||||
+ // oldVersion is null
|
|
||||||
+ currentData.setCurrentVersion(Bukkit.getVersion());
|
|
||||||
+ writeFile(path);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void writeFile(@Nonnull final Path path) {
|
|
||||||
+ try (final BufferedWriter writer = Files.newBufferedWriter(
|
|
||||||
+ path,
|
|
||||||
+ StandardCharsets.UTF_8,
|
|
||||||
+ StandardOpenOption.WRITE,
|
|
||||||
+ StandardOpenOption.CREATE,
|
|
||||||
+ StandardOpenOption.TRUNCATE_EXISTING
|
|
||||||
+ )) {
|
|
||||||
+ gson.toJson(currentData, writer);
|
|
||||||
+ } catch (final IOException e) {
|
|
||||||
+ logger.log(Level.SEVERE, "Failed to write to version history file", e);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Nullable
|
|
||||||
+ public VersionData getVersionData() {
|
|
||||||
+ return currentData;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public static class VersionData {
|
|
||||||
+ private String oldVersion;
|
|
||||||
+
|
|
||||||
+ private String currentVersion;
|
|
||||||
+
|
|
||||||
+ @Nullable
|
|
||||||
+ public String getOldVersion() {
|
|
||||||
+ return oldVersion;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void setOldVersion(@Nullable String oldVersion) {
|
|
||||||
+ this.oldVersion = oldVersion;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Nullable
|
|
||||||
+ public String getCurrentVersion() {
|
|
||||||
+ return currentVersion;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void setCurrentVersion(@Nullable String currentVersion) {
|
|
||||||
+ this.currentVersion = currentVersion;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public String toString() {
|
|
||||||
+ return MoreObjects.toStringHelper(this)
|
|
||||||
+ .add("oldVersion", oldVersion)
|
|
||||||
+ .add("currentVersion", currentVersion)
|
|
||||||
+ .toString();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public boolean equals(@Nullable Object o) {
|
|
||||||
+ if (this == o) {
|
|
||||||
+ return true;
|
|
||||||
+ }
|
|
||||||
+ if (o == null || getClass() != o.getClass()) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+ final VersionData versionData = (VersionData) o;
|
|
||||||
+ return Objects.equals(oldVersion, versionData.oldVersion) &&
|
|
||||||
+ Objects.equals(currentVersion, versionData.currentVersion);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public int hashCode() {
|
|
||||||
+ return Objects.hash(oldVersion, currentVersion);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
index 59942eb6bb6e8e1e9a988bce0d09757e575018b9..faf4d00bf288359db806913c4d2964324e8706b7 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
@@ -196,6 +196,7 @@ public class DedicatedServer extends MinecraftServer implements IMinecraftServer
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
com.destroystokyo.paper.PaperConfig.registerCommands();
|
|
||||||
+ com.destroystokyo.paper.VersionHistoryManager.INSTANCE.getClass(); // load version history now
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
this.setPVP(dedicatedserverproperties.pvp);
|
|
@ -1,157 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Tue, 1 Mar 2016 14:47:52 -0600
|
|
||||||
Subject: [PATCH] Player affects spawning API
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index 20e4ff812960a54872f2fea8fe6baf7bb1ef077d..cae9da158f54438d2a397665c7ce964f6f755469 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -1354,6 +1354,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
return MathHelper.c(f * f + f1 * f1 + f2 * f2);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public double getDistanceSquared(double x, double y, double z) { return h(x, y, z); } // Paper - OBFHELPER
|
|
||||||
public double h(double d0, double d1, double d2) {
|
|
||||||
double d3 = this.locX() - d0;
|
|
||||||
double d4 = this.locY() - d1;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityInsentient.java b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
index 6e30fc88fa7a3ff00c9b4b78842c3a533649bd50..31bb5df7bc63c993230bf595bd9b66bfaadb4d50 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityInsentient.java
|
|
||||||
@@ -756,7 +756,7 @@ public abstract class EntityInsentient extends EntityLiving {
|
|
||||||
if (this.world.getDifficulty() == EnumDifficulty.PEACEFUL && this.L()) {
|
|
||||||
this.die();
|
|
||||||
} else if (!this.isPersistent() && !this.isSpecialPersistence()) {
|
|
||||||
- EntityHuman entityhuman = this.world.findNearbyPlayer(this, -1.0D);
|
|
||||||
+ EntityHuman entityhuman = this.world.findNearbyPlayer(this, -1.0D, IEntitySelector.affectsSpawning); // Paper
|
|
||||||
|
|
||||||
if (entityhuman != null) {
|
|
||||||
double d0 = entityhuman.h((Entity) this); // CraftBukkit - decompile error
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/IEntitySelector.java b/src/main/java/net/minecraft/world/entity/IEntitySelector.java
|
|
||||||
index 6e78192a5898df017d96acba845a288011d24e35..dfcfdb31ca9531913d705aaaf85fb67399cfdc8c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/IEntitySelector.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/IEntitySelector.java
|
|
||||||
@@ -29,6 +29,12 @@ public final class IEntitySelector {
|
|
||||||
return !entity.isSpectator();
|
|
||||||
};
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ public static final Predicate<Entity> affectsSpawning = (entity) -> {
|
|
||||||
+ return !entity.isSpectator() && entity.isAlive() && (entity instanceof EntityPlayer) && ((EntityPlayer) entity).affectsSpawning;
|
|
||||||
+ };
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
public static Predicate<Entity> a(double d0, double d1, double d2, double d3) {
|
|
||||||
double d4 = d3 * d3;
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/monster/EntitySilverfish.java b/src/main/java/net/minecraft/world/entity/monster/EntitySilverfish.java
|
|
||||||
index 0b3b430766fba602e74727f78173567ca10fabc6..e1fcb1be102822e87eaf7757fbd64a516b2f58ac 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/monster/EntitySilverfish.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/monster/EntitySilverfish.java
|
|
||||||
@@ -122,7 +122,7 @@ public class EntitySilverfish extends EntityMonster {
|
|
||||||
if (c(entitytypes, generatoraccess, enummobspawn, blockposition, random)) {
|
|
||||||
EntityHuman entityhuman = generatoraccess.a((double) blockposition.getX() + 0.5D, (double) blockposition.getY() + 0.5D, (double) blockposition.getZ() + 0.5D, 5.0D, true);
|
|
||||||
|
|
||||||
- return entityhuman == null;
|
|
||||||
+ return !(entityhuman != null && !entityhuman.affectsSpawning) && entityhuman == null; // Paper - Affects Spawning API
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/player/EntityHuman.java b/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
index 1017ee73b8617ce2b6734469fa49aaff7563c2b1..f42e16589476c1bd10b13214dda5ac7bb3e52131 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
@@ -164,6 +164,9 @@ public abstract class EntityHuman extends EntityLiving {
|
|
||||||
private final ItemCooldown bM;
|
|
||||||
@Nullable
|
|
||||||
public EntityFishingHook hookedFish;
|
|
||||||
+ // Paper start
|
|
||||||
+ public boolean affectsSpawning = true;
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
public boolean fauxSleeping;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/IEntityAccess.java b/src/main/java/net/minecraft/world/level/IEntityAccess.java
|
|
||||||
index 4ece69851e7b05016f52c291ce911eb791cf3a23..6d5d4c3df65995b9a13b66d070ba08d553cc98a2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/IEntityAccess.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/IEntityAccess.java
|
|
||||||
@@ -92,8 +92,9 @@ public interface IEntityAccess {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Nullable
|
|
||||||
- default EntityHuman a(double d0, double d1, double d2, double d3, @Nullable Predicate<Entity> predicate) {
|
|
||||||
+ default EntityHuman findNearbyPlayer(Entity entity, double d0, @Nullable Predicate<Entity> predicate) { return this.findNearbyPlayer(entity.locX(), entity.locY(), entity.locZ(), d0, predicate); } // Paper
|
|
||||||
+ @Nullable default EntityHuman findNearbyPlayer(double d0, double d1, double d2, double d3, @Nullable Predicate<Entity> predicate) { return a(d0, d1, d2, d3, predicate); } // Paper - OBFHELPER
|
|
||||||
+ @Nullable default EntityHuman a(double d0, double d1, double d2, double d3, @Nullable Predicate<Entity> predicate) { // Paper
|
|
||||||
double d4 = -1.0D;
|
|
||||||
EntityHuman entityhuman = null;
|
|
||||||
Iterator iterator = this.getPlayers().iterator();
|
|
||||||
@@ -126,6 +127,27 @@ public interface IEntityAccess {
|
|
||||||
return this.a(d0, d1, d2, d3, predicate);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper end
|
|
||||||
+ default boolean isAffectsSpawningPlayerNearby(double d0, double d1, double d2, double d3) {
|
|
||||||
+ Iterator iterator = this.getPlayers().iterator();
|
|
||||||
+ double d4;
|
|
||||||
+ do {
|
|
||||||
+ EntityHuman entityhuman;
|
|
||||||
+ do {
|
|
||||||
+ if (!iterator.hasNext()) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ entityhuman = (EntityHuman) iterator.next();
|
|
||||||
+ } while (!IEntitySelector.affectsSpawning.test(entityhuman));
|
|
||||||
+
|
|
||||||
+ d4 = entityhuman.getDistanceSquared(d0, d1, d2);
|
|
||||||
+ } while (d3 >= 0.0D && d4 >= d3 * d3);
|
|
||||||
+
|
|
||||||
+ return true;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
default boolean isPlayerNearby(double d0, double d1, double d2, double d3) {
|
|
||||||
Iterator iterator = this.getPlayers().iterator();
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java b/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
index 33cba4e475edc0573b901f70c61d3659fd63ad62..8d8b03074df1635946f81bec0feae18d2f3e20aa 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
@@ -66,7 +66,7 @@ public abstract class MobSpawnerAbstract {
|
|
||||||
private boolean h() {
|
|
||||||
BlockPosition blockposition = this.b();
|
|
||||||
|
|
||||||
- return this.a().isPlayerNearby((double) blockposition.getX() + 0.5D, (double) blockposition.getY() + 0.5D, (double) blockposition.getZ() + 0.5D, (double) this.requiredPlayerRange);
|
|
||||||
+ return this.a().isAffectsSpawningPlayerNearby((double) blockposition.getX() + 0.5D, (double) blockposition.getY() + 0.5D, (double) blockposition.getZ() + 0.5D, (double) this.requiredPlayerRange); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
public void c() {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index 6ab14bccb1fcd108931bf7ec331e60f652e0b42d..cc471418b37a745ecea1af964e81bc0362cf7d94 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -1770,8 +1770,20 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
@Override
|
|
||||||
public String getLocale() {
|
|
||||||
return getHandle().locale;
|
|
||||||
+
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ public void setAffectsSpawning(boolean affects) {
|
|
||||||
+ this.getHandle().affectsSpawning = affects;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ @Override
|
|
||||||
+ public boolean getAffectsSpawning() {
|
|
||||||
+ return this.getHandle().affectsSpawning;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
@Override
|
|
||||||
public void updateCommands() {
|
|
||||||
if (getHandle().playerConnection == null) return;
|
|
@ -1,33 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Byteflux <byte@byteflux.net>
|
|
||||||
Date: Tue, 1 Mar 2016 15:08:03 -0600
|
|
||||||
Subject: [PATCH] Remove invalid mob spawner tile entities
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index ac576d268b23148089d404cb22d8c2f9d1a79d6e..a2d80c2c8e4f080f60746548f75631c5946ba8e2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -40,9 +40,11 @@ import net.minecraft.world.level.TickListChunk;
|
|
||||||
import net.minecraft.world.level.TickListEmpty;
|
|
||||||
import net.minecraft.world.level.World;
|
|
||||||
import net.minecraft.world.level.block.Block;
|
|
||||||
+import net.minecraft.world.level.block.BlockMobSpawner;
|
|
||||||
import net.minecraft.world.level.block.Blocks;
|
|
||||||
import net.minecraft.world.level.block.ITileEntity;
|
|
||||||
import net.minecraft.world.level.block.entity.TileEntity;
|
|
||||||
+import net.minecraft.world.level.block.entity.TileEntityMobSpawner;
|
|
||||||
import net.minecraft.world.level.block.state.IBlockData;
|
|
||||||
import net.minecraft.world.level.levelgen.ChunkProviderDebug;
|
|
||||||
import net.minecraft.world.level.levelgen.HeightMap;
|
|
||||||
@@ -648,6 +650,10 @@ public class Chunk implements IChunkAccess {
|
|
||||||
}
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
+ // Paper start - Remove invalid mob spawner tile entities
|
|
||||||
+ } else if (tileentity instanceof TileEntityMobSpawner && !(getBlockData(blockposition.getX(), blockposition.getY(), blockposition.getZ()).getBlock() instanceof BlockMobSpawner)) {
|
|
||||||
+ this.tileEntities.remove(blockposition);
|
|
||||||
+ // Paper end
|
|
||||||
} else {
|
|
||||||
System.out.println("Attempted to place a tile entity (" + tileentity + ") at " + tileentity.getPosition().getX() + "," + tileentity.getPosition().getY() + "," + tileentity.getPosition().getZ()
|
|
||||||
+ " (" + getType(blockposition) + ") where there was no entity tile!");
|
|
@ -1,248 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 8 Mar 2015 22:55:25 -0600
|
|
||||||
Subject: [PATCH] Optimize TileEntity Ticking
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/co/aikar/timings/TimingsExport.java b/src/main/java/co/aikar/timings/TimingsExport.java
|
|
||||||
index d4ebcf8f66197299256bd6b65710a1488c90ea41..c9164dfdb27ddf3709129c8aec54903a1df121ff 100644
|
|
||||||
--- a/src/main/java/co/aikar/timings/TimingsExport.java
|
|
||||||
+++ b/src/main/java/co/aikar/timings/TimingsExport.java
|
|
||||||
@@ -109,7 +109,7 @@ public class TimingsExport extends Thread {
|
|
||||||
pair("end", System.currentTimeMillis() / 1000),
|
|
||||||
pair("online-mode", Bukkit.getServer().getOnlineMode()),
|
|
||||||
pair("sampletime", (System.currentTimeMillis() - TimingsManager.timingStart) / 1000),
|
|
||||||
- pair("datapacks", toArrayMapper(MinecraftServer.getServer().getResourcePackRepository().d(), pack -> {
|
|
||||||
+ pair("datapacks", toArrayMapper(MinecraftServer.getServer().getResourcePackRepository().e(), pack -> {
|
|
||||||
// Don't feel like obf helper'ing these, non fatal if its temp missed.
|
|
||||||
return ChatColor.stripColor(CraftChatMessage.fromComponent(pack.a(true)));
|
|
||||||
}))
|
|
||||||
@@ -148,8 +148,8 @@ public class TimingsExport extends Thread {
|
|
||||||
);
|
|
||||||
|
|
||||||
parent.put("worlds", toObjectMapper(MinecraftServer.getServer().getWorlds(), world -> {
|
|
||||||
- if (world.getWorldData().getName().equals("worldeditregentempworld")) return null;
|
|
||||||
- return pair(world.getWorldData().getName(), createObject(
|
|
||||||
+ if (world.getWorld().getName().equals("worldeditregentempworld")) return null;
|
|
||||||
+ return pair(world.getWorld().getName(), createObject(
|
|
||||||
pair("gamerules", toObjectMapper(world.getWorld().getGameRules(), rule -> {
|
|
||||||
return pair(rule, world.getWorld().getGameRuleValue(rule));
|
|
||||||
})),
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockChest.java b/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
index 60e7dc1910ae9214d84d65b011cfec278b6b32ae..b229faad99120c67b089f7680d800fbe594fe7da 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
@@ -54,8 +54,8 @@ import net.minecraft.world.phys.shapes.VoxelShapeCollision;
|
|
||||||
public class BlockChest extends BlockChestAbstract<TileEntityChest> implements IBlockWaterlogged {
|
|
||||||
|
|
||||||
public static final BlockStateDirection FACING = BlockFacingHorizontal.FACING;
|
|
||||||
- public static final BlockStateEnum<BlockPropertyChestType> c = BlockProperties.aF;
|
|
||||||
- public static final BlockStateBoolean d = BlockProperties.C;
|
|
||||||
+ public static final BlockStateEnum<BlockPropertyChestType> c = BlockProperties.aF; public static final BlockStateEnum<BlockPropertyChestType> CHEST_TYPE_PROPERTY = c; // Paper - OBFHELPER
|
|
||||||
+ public static final BlockStateBoolean d = BlockProperties.C; public static final BlockStateBoolean waterlogged() { return d; } // Paper OBFHELPER
|
|
||||||
protected static final VoxelShape e = Block.a(1.0D, 0.0D, 0.0D, 15.0D, 14.0D, 15.0D);
|
|
||||||
protected static final VoxelShape f = Block.a(1.0D, 0.0D, 1.0D, 15.0D, 14.0D, 16.0D);
|
|
||||||
protected static final VoxelShape g = Block.a(0.0D, 0.0D, 1.0D, 15.0D, 14.0D, 15.0D);
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntityChest.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntityChest.java
|
|
||||||
index eed95b370d1d624ffc6b7a35357b7028ec58c584..51167d776c710decb0107bebcb35bdf43103772b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntityChest.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntityChest.java
|
|
||||||
@@ -8,6 +8,7 @@ import net.minecraft.core.NonNullList;
|
|
||||||
import net.minecraft.nbt.NBTTagCompound;
|
|
||||||
import net.minecraft.network.chat.ChatMessage;
|
|
||||||
import net.minecraft.network.chat.IChatBaseComponent;
|
|
||||||
+import net.minecraft.server.MCUtil;
|
|
||||||
import net.minecraft.sounds.SoundCategory;
|
|
||||||
import net.minecraft.sounds.SoundEffect;
|
|
||||||
import net.minecraft.sounds.SoundEffects;
|
|
||||||
@@ -33,7 +34,7 @@ import org.bukkit.craftbukkit.entity.CraftHumanEntity;
|
|
||||||
import org.bukkit.entity.HumanEntity;
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
-public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
+public class TileEntityChest extends TileEntityLootable { // Paper - Remove ITickable
|
|
||||||
|
|
||||||
private NonNullList<ItemStack> items;
|
|
||||||
protected float a;
|
|
||||||
@@ -111,14 +112,20 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
return nbttagcompound;
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
public void tick() {
|
|
||||||
int i = this.position.getX();
|
|
||||||
int j = this.position.getY();
|
|
||||||
int k = this.position.getZ();
|
|
||||||
|
|
||||||
++this.j;
|
|
||||||
- this.viewingCount = a(this.world, this, this.j, i, j, k, this.viewingCount);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void doOpenLogic() {
|
|
||||||
+ int i = this.position.getX();
|
|
||||||
+ int j = this.position.getY();
|
|
||||||
+ int k = this.position.getZ();
|
|
||||||
+
|
|
||||||
+ //this.viewingCount = a(this.world, this, this.j, i, j, k, this.viewingCount); // Paper - check is faulty given our logic is called before active container set
|
|
||||||
this.b = this.a;
|
|
||||||
float f = 0.1F;
|
|
||||||
|
|
||||||
@@ -132,8 +139,11 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
if (this.viewingCount > 0 && this.a == 0.0F) {
|
|
||||||
this.playOpenSound(SoundEffects.BLOCK_CHEST_OPEN);
|
|
||||||
}
|
|
||||||
+ }
|
|
||||||
|
|
||||||
- if (this.viewingCount == 0 && this.a > 0.0F || this.viewingCount > 0 && this.a < 1.0F) {
|
|
||||||
+ public void doCloseLogic() {
|
|
||||||
+ if (this.viewingCount == 0 /* && this.a > 0.0F || this.viewingCount > 0 && this.a < 1.0F */) { // Paper - disable all but player count check
|
|
||||||
+ /* // Paper - disable animation stuff
|
|
||||||
float f1 = this.a;
|
|
||||||
|
|
||||||
if (this.viewingCount > 0) {
|
|
||||||
@@ -149,8 +159,11 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
float f2 = 0.5F;
|
|
||||||
|
|
||||||
if (this.a < 0.5F && f1 >= 0.5F) {
|
|
||||||
+ */
|
|
||||||
+ MCUtil.scheduleTask(10, () -> {
|
|
||||||
this.playOpenSound(SoundEffects.BLOCK_CHEST_CLOSE);
|
|
||||||
- }
|
|
||||||
+ }, "Chest Sounds");
|
|
||||||
+ //} // Paper end
|
|
||||||
|
|
||||||
if (this.a < 0.0F) {
|
|
||||||
this.a = 0.0F;
|
|
||||||
@@ -189,6 +202,7 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void playOpenSound(SoundEffect soundeffect) {
|
|
||||||
+ if (!this.getBlock().contains(BlockChest.CHEST_TYPE_PROPERTY)) { return; } // Paper - this can be delayed, double check exists - Fixes GH-2074
|
|
||||||
BlockPropertyChestType blockpropertychesttype = (BlockPropertyChestType) this.getBlock().get(BlockChest.c);
|
|
||||||
|
|
||||||
if (blockpropertychesttype != BlockPropertyChestType.LEFT) {
|
|
||||||
@@ -227,6 +241,7 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
|
|
||||||
++this.viewingCount;
|
|
||||||
if (this.world == null) return; // CraftBukkit
|
|
||||||
+ doOpenLogic(); // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start - Call redstone event
|
|
||||||
if (this.getBlock().getBlock() == Blocks.TRAPPED_CHEST) {
|
|
||||||
@@ -249,6 +264,7 @@ public class TileEntityChest extends TileEntityLootable implements ITickable {
|
|
||||||
--this.viewingCount;
|
|
||||||
|
|
||||||
// CraftBukkit start - Call redstone event
|
|
||||||
+ doCloseLogic(); // Paper
|
|
||||||
if (this.getBlock().getBlock() == Blocks.TRAPPED_CHEST) {
|
|
||||||
int newPower = Math.max(0, Math.min(15, this.viewingCount));
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntityEnderChest.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntityEnderChest.java
|
|
||||||
index 930f1bd091d9754f7ca5d9e36cdf49b2be03eb23..2bc4213c70be47ca8bbc24898cc92e43f4228821 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntityEnderChest.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntityEnderChest.java
|
|
||||||
@@ -1,11 +1,12 @@
|
|
||||||
package net.minecraft.world.level.block.entity;
|
|
||||||
|
|
||||||
+import net.minecraft.server.MCUtil;
|
|
||||||
import net.minecraft.sounds.SoundCategory;
|
|
||||||
import net.minecraft.sounds.SoundEffects;
|
|
||||||
import net.minecraft.world.entity.player.EntityHuman;
|
|
||||||
import net.minecraft.world.level.block.Blocks;
|
|
||||||
|
|
||||||
-public class TileEntityEnderChest extends TileEntity implements ITickable {
|
|
||||||
+public class TileEntityEnderChest extends TileEntity { // Paper - Remove ITickable
|
|
||||||
|
|
||||||
public float a;
|
|
||||||
public float b;
|
|
||||||
@@ -16,18 +17,28 @@ public class TileEntityEnderChest extends TileEntity implements ITickable {
|
|
||||||
super(TileEntityTypes.ENDER_CHEST);
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
public void tick() {
|
|
||||||
if (++this.g % 20 * 4 == 0) {
|
|
||||||
this.world.playBlockAction(this.position, Blocks.ENDER_CHEST, 1, this.c);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.b = this.a;
|
|
||||||
+ /* // Paper
|
|
||||||
int i = this.position.getX();
|
|
||||||
int j = this.position.getY();
|
|
||||||
int k = this.position.getZ();
|
|
||||||
float f = 0.1F;
|
|
||||||
double d0;
|
|
||||||
+ // Paper start
|
|
||||||
+ */
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void doOpenLogic() {
|
|
||||||
+ int i = this.position.getX();
|
|
||||||
+ int j = this.position.getY();
|
|
||||||
+ int k = this.position.getZ();
|
|
||||||
+ double d0;
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
if (this.c > 0 && this.a == 0.0F) {
|
|
||||||
double d1 = (double) i + 0.5D;
|
|
||||||
@@ -35,8 +46,17 @@ public class TileEntityEnderChest extends TileEntity implements ITickable {
|
|
||||||
d0 = (double) k + 0.5D;
|
|
||||||
this.world.playSound((EntityHuman) null, d1, (double) j + 0.5D, d0, SoundEffects.BLOCK_ENDER_CHEST_OPEN, SoundCategory.BLOCKS, 0.5F, this.world.random.nextFloat() * 0.1F + 0.9F);
|
|
||||||
}
|
|
||||||
+ // Paper start
|
|
||||||
+ }
|
|
||||||
|
|
||||||
- if (this.c == 0 && this.a > 0.0F || this.c > 0 && this.a < 1.0F) {
|
|
||||||
+ private void doCloseLogic() {
|
|
||||||
+ int i = this.position.getX();
|
|
||||||
+ int j = this.position.getY();
|
|
||||||
+ int k = this.position.getZ();
|
|
||||||
+ double d0;
|
|
||||||
+
|
|
||||||
+ if (this.c == 0) { /* && this.a > 0.0F || this.c > 0 && this.a < 1.0F) {
|
|
||||||
+ // Paper end
|
|
||||||
float f1 = this.a;
|
|
||||||
|
|
||||||
if (this.c > 0) {
|
|
||||||
@@ -52,11 +72,14 @@ public class TileEntityEnderChest extends TileEntity implements ITickable {
|
|
||||||
float f2 = 0.5F;
|
|
||||||
|
|
||||||
if (this.a < 0.5F && f1 >= 0.5F) {
|
|
||||||
+ // Paper start
|
|
||||||
+ */
|
|
||||||
d0 = (double) i + 0.5D;
|
|
||||||
double d2 = (double) k + 0.5D;
|
|
||||||
|
|
||||||
+ MCUtil.scheduleTask(10, () -> {
|
|
||||||
this.world.playSound((EntityHuman) null, d0, (double) j + 0.5D, d2, SoundEffects.BLOCK_ENDER_CHEST_CLOSE, SoundCategory.BLOCKS, 0.5F, this.world.random.nextFloat() * 0.1F + 0.9F);
|
|
||||||
- }
|
|
||||||
+ }, "Chest Sounds");
|
|
||||||
|
|
||||||
if (this.a < 0.0F) {
|
|
||||||
this.a = 0.0F;
|
|
||||||
@@ -84,11 +107,13 @@ public class TileEntityEnderChest extends TileEntity implements ITickable {
|
|
||||||
public void d() {
|
|
||||||
++this.c;
|
|
||||||
this.world.playBlockAction(this.position, Blocks.ENDER_CHEST, 1, this.c);
|
|
||||||
+ doOpenLogic(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
public void f() {
|
|
||||||
--this.c;
|
|
||||||
this.world.playBlockAction(this.position, Blocks.ENDER_CHEST, 1, this.c);
|
|
||||||
+ doCloseLogic(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean a(EntityHuman entityhuman) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/state/IBlockDataHolder.java b/src/main/java/net/minecraft/world/level/block/state/IBlockDataHolder.java
|
|
||||||
index e4b59a85ee9b435b2e86d4c7d78b7224773f6967..ba046cffdd8331c7e0427f19fa54d0c7a99077d9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/state/IBlockDataHolder.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/state/IBlockDataHolder.java
|
|
||||||
@@ -84,6 +84,7 @@ public abstract class IBlockDataHolder<O, S> {
|
|
||||||
return Collections.unmodifiableCollection(this.b.keySet());
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public <T extends Comparable<T>> boolean contains(IBlockState<T> iblockstate) { return this.b(iblockstate); } // Paper - OBFHELPER
|
|
||||||
public <T extends Comparable<T>> boolean b(IBlockState<T> iblockstate) {
|
|
||||||
return this.b.containsKey(iblockstate);
|
|
||||||
}
|
|
@ -1,208 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Tue, 1 Mar 2016 23:09:29 -0600
|
|
||||||
Subject: [PATCH] Further improve server tick loop
|
|
||||||
|
|
||||||
Improves how the catchup buffer is handled, allowing it to roll both ways
|
|
||||||
increasing the effeciency of the thread sleep so it only will sleep once.
|
|
||||||
|
|
||||||
Also increases the buffer of the catchup to ensure server stays at 20 TPS unless extreme conditions
|
|
||||||
|
|
||||||
Previous implementation did not calculate TPS correctly.
|
|
||||||
Switch to a realistic rolling average and factor in std deviation as an extra reporting variable
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index c21790b4de698aa6f7fc4dadab64d791cd0562b6..fb0d985b5c977a7c63701484678b75928d9ee382 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -253,7 +253,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
public org.bukkit.command.ConsoleCommandSender console;
|
|
||||||
public org.bukkit.command.RemoteConsoleCommandSender remoteConsole;
|
|
||||||
public ConsoleReader reader;
|
|
||||||
- public static int currentTick = (int) (System.currentTimeMillis() / 50);
|
|
||||||
+ public static int currentTick = 0; // Paper - Further improve tick loop
|
|
||||||
public java.util.Queue<Runnable> processQueue = new java.util.concurrent.ConcurrentLinkedQueue<Runnable>();
|
|
||||||
public int autosavePeriod;
|
|
||||||
public CommandDispatcher vanillaCommandDispatcher;
|
|
||||||
@@ -262,7 +262,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
// Spigot start
|
|
||||||
public static final int TPS = 20;
|
|
||||||
public static final int TICK_TIME = 1000000000 / TPS;
|
|
||||||
- private static final int SAMPLE_INTERVAL = 100;
|
|
||||||
+ private static final int SAMPLE_INTERVAL = 20; // Paper
|
|
||||||
public final double[] recentTps = new double[ 3 ];
|
|
||||||
public final SlackActivityAccountant slackActivityAccountant = new SlackActivityAccountant();
|
|
||||||
// Spigot end
|
|
||||||
@@ -925,6 +925,57 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
{
|
|
||||||
return ( avg * exp ) + ( tps * ( 1 - exp ) );
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ // Paper start - Further improve server tick loop
|
|
||||||
+ private static final long SEC_IN_NANO = 1000000000;
|
|
||||||
+ private static final long MAX_CATCHUP_BUFFER = TICK_TIME * TPS * 60L;
|
|
||||||
+ private long lastTick = 0;
|
|
||||||
+ private long catchupTime = 0;
|
|
||||||
+ public final RollingAverage tps1 = new RollingAverage(60);
|
|
||||||
+ public final RollingAverage tps5 = new RollingAverage(60 * 5);
|
|
||||||
+ public final RollingAverage tps15 = new RollingAverage(60 * 15);
|
|
||||||
+
|
|
||||||
+ public static class RollingAverage {
|
|
||||||
+ private final int size;
|
|
||||||
+ private long time;
|
|
||||||
+ private java.math.BigDecimal total;
|
|
||||||
+ private int index = 0;
|
|
||||||
+ private final java.math.BigDecimal[] samples;
|
|
||||||
+ private final long[] times;
|
|
||||||
+
|
|
||||||
+ RollingAverage(int size) {
|
|
||||||
+ this.size = size;
|
|
||||||
+ this.time = size * SEC_IN_NANO;
|
|
||||||
+ this.total = dec(TPS).multiply(dec(SEC_IN_NANO)).multiply(dec(size));
|
|
||||||
+ this.samples = new java.math.BigDecimal[size];
|
|
||||||
+ this.times = new long[size];
|
|
||||||
+ for (int i = 0; i < size; i++) {
|
|
||||||
+ this.samples[i] = dec(TPS);
|
|
||||||
+ this.times[i] = SEC_IN_NANO;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static java.math.BigDecimal dec(long t) {
|
|
||||||
+ return new java.math.BigDecimal(t);
|
|
||||||
+ }
|
|
||||||
+ public void add(java.math.BigDecimal x, long t) {
|
|
||||||
+ time -= times[index];
|
|
||||||
+ total = total.subtract(samples[index].multiply(dec(times[index])));
|
|
||||||
+ samples[index] = x;
|
|
||||||
+ times[index] = t;
|
|
||||||
+ time += t;
|
|
||||||
+ total = total.add(x.multiply(dec(t)));
|
|
||||||
+ if (++index == size) {
|
|
||||||
+ index = 0;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public double getAverage() {
|
|
||||||
+ return total.divide(dec(time), 30, java.math.RoundingMode.HALF_UP).doubleValue();
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ private static final java.math.BigDecimal TPS_BASE = new java.math.BigDecimal(1E9).multiply(new java.math.BigDecimal(SAMPLE_INTERVAL));
|
|
||||||
+ // Paper End
|
|
||||||
// Spigot End
|
|
||||||
|
|
||||||
protected void w() {
|
|
||||||
@@ -937,30 +988,38 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
|
|
||||||
// Spigot start
|
|
||||||
Arrays.fill( recentTps, 20 );
|
|
||||||
- long curTime, tickSection = SystemUtils.getMonotonicMillis(), tickCount = 1;
|
|
||||||
+ long start = System.nanoTime(), curTime, tickSection = start; // Paper - Further improve server tick loop
|
|
||||||
+ lastTick = start - TICK_TIME; // Paper
|
|
||||||
while (this.isRunning) {
|
|
||||||
- long i = (curTime = SystemUtils.getMonotonicMillis()) - this.nextTick;
|
|
||||||
+ long i = ((curTime = System.nanoTime()) / (1000L * 1000L)) - this.nextTick; // Paper
|
|
||||||
|
|
||||||
if (i > 5000L && this.nextTick - this.lastOverloadTime >= 30000L) { // CraftBukkit
|
|
||||||
long j = i / 50L;
|
|
||||||
|
|
||||||
if (server.getWarnOnOverload()) // CraftBukkit
|
|
||||||
- MinecraftServer.LOGGER.warn("Can't keep up! Is the server overloaded? Running {}ms or {} ticks behind", i, j);
|
|
||||||
+ MinecraftServer.LOGGER.warn("Can't keep up! Is the server overloaded? Running {}ms or {} ticks behind", i, j);
|
|
||||||
this.nextTick += j * 50L;
|
|
||||||
this.lastOverloadTime = this.nextTick;
|
|
||||||
}
|
|
||||||
|
|
||||||
- if ( tickCount++ % SAMPLE_INTERVAL == 0 )
|
|
||||||
+ if ( ++MinecraftServer.currentTick % SAMPLE_INTERVAL == 0 )
|
|
||||||
{
|
|
||||||
- double currentTps = 1E3 / ( curTime - tickSection ) * SAMPLE_INTERVAL;
|
|
||||||
- recentTps[0] = calcTps( recentTps[0], 0.92, currentTps ); // 1/exp(5sec/1min)
|
|
||||||
- recentTps[1] = calcTps( recentTps[1], 0.9835, currentTps ); // 1/exp(5sec/5min)
|
|
||||||
- recentTps[2] = calcTps( recentTps[2], 0.9945, currentTps ); // 1/exp(5sec/15min)
|
|
||||||
+ final long diff = curTime - tickSection;
|
|
||||||
+ java.math.BigDecimal currentTps = TPS_BASE.divide(new java.math.BigDecimal(diff), 30, java.math.RoundingMode.HALF_UP);
|
|
||||||
+ tps1.add(currentTps, diff);
|
|
||||||
+ tps5.add(currentTps, diff);
|
|
||||||
+ tps15.add(currentTps, diff);
|
|
||||||
+ // Backwards compat with bad plugins
|
|
||||||
+ recentTps[0] = tps1.getAverage();
|
|
||||||
+ recentTps[1] = tps5.getAverage();
|
|
||||||
+ recentTps[2] = tps15.getAverage();
|
|
||||||
+ // Paper end
|
|
||||||
tickSection = curTime;
|
|
||||||
}
|
|
||||||
// Spigot end
|
|
||||||
|
|
||||||
- MinecraftServer.currentTick = (int) (System.currentTimeMillis() / 50); // CraftBukkit
|
|
||||||
+ //MinecraftServer.currentTick = (int) (System.currentTimeMillis() / 50); // CraftBukkit // Paper - don't overwrite current tick time
|
|
||||||
+ lastTick = curTime;
|
|
||||||
this.nextTick += 50L;
|
|
||||||
GameProfilerTick gameprofilertick = GameProfilerTick.a("Server");
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index 462633df2af43959fddf5b7a8ec43063abf7b14b..b4b592faaa699b6e62f77ddf57b239c5e15948ff 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -2122,6 +2122,17 @@ public final class CraftServer implements Server {
|
|
||||||
return CraftMagicNumbers.INSTANCE;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper - Add getTPS API - Further improve tick loop
|
|
||||||
+ @Override
|
|
||||||
+ public double[] getTPS() {
|
|
||||||
+ return new double[] {
|
|
||||||
+ net.minecraft.server.MinecraftServer.getServer().tps1.getAverage(),
|
|
||||||
+ net.minecraft.server.MinecraftServer.getServer().tps5.getAverage(),
|
|
||||||
+ net.minecraft.server.MinecraftServer.getServer().tps15.getAverage()
|
|
||||||
+ };
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
// Spigot start
|
|
||||||
private final org.bukkit.Server.Spigot spigot = new org.bukkit.Server.Spigot()
|
|
||||||
{
|
|
||||||
diff --git a/src/main/java/org/spigotmc/TicksPerSecondCommand.java b/src/main/java/org/spigotmc/TicksPerSecondCommand.java
|
|
||||||
index f5b6dec1cbe7501ce2ee9125920e810bc94670cc..e62890433ffbe0b4e48942fe6c38b599a19e58fd 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/TicksPerSecondCommand.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/TicksPerSecondCommand.java
|
|
||||||
@@ -24,22 +24,30 @@ public class TicksPerSecondCommand extends Command
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
- StringBuilder sb = new StringBuilder( ChatColor.GOLD + "TPS from last 1m, 5m, 15m: " );
|
|
||||||
- for ( double tps : MinecraftServer.getServer().recentTps )
|
|
||||||
- {
|
|
||||||
- sb.append( format( tps ) );
|
|
||||||
- sb.append( ", " );
|
|
||||||
+ // Paper start - Further improve tick handling
|
|
||||||
+ double[] tps = org.bukkit.Bukkit.getTPS();
|
|
||||||
+ String[] tpsAvg = new String[tps.length];
|
|
||||||
+
|
|
||||||
+ for ( int i = 0; i < tps.length; i++) {
|
|
||||||
+ tpsAvg[i] = format( tps[i] );
|
|
||||||
+ }
|
|
||||||
+ sender.sendMessage(ChatColor.GOLD + "TPS from last 1m, 5m, 15m: " + org.apache.commons.lang.StringUtils.join(tpsAvg, ", "));
|
|
||||||
+ if (args.length > 0 && args[0].equals("mem") && sender.hasPermission("bukkit.command.tpsmemory")) {
|
|
||||||
+ sender.sendMessage(ChatColor.GOLD + "Current Memory Usage: " + ChatColor.GREEN + ((Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / (1024 * 1024)) + "/" + (Runtime.getRuntime().totalMemory() / (1024 * 1024)) + " mb (Max: " + (Runtime.getRuntime().maxMemory() / (1024 * 1024)) + " mb)");
|
|
||||||
+ if (!hasShownMemoryWarning) {
|
|
||||||
+ sender.sendMessage(ChatColor.RED + "Warning: " + ChatColor.GOLD + " Memory usage on modern garbage collectors is not a stable value and it is perfectly normal to see it reach max. Please do not pay it much attention.");
|
|
||||||
+ hasShownMemoryWarning = true;
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
- sender.sendMessage( sb.substring( 0, sb.length() - 2 ) );
|
|
||||||
- sender.sendMessage(ChatColor.GOLD + "Current Memory Usage: " + ChatColor.GREEN + ((Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / (1024 * 1024)) + "/" + (Runtime.getRuntime().totalMemory() / (1024 * 1024)) + " mb (Max: "
|
|
||||||
- + (Runtime.getRuntime().maxMemory() / (1024 * 1024)) + " mb)");
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
- private String format(double tps)
|
|
||||||
+ private boolean hasShownMemoryWarning; // Paper
|
|
||||||
+ private static String format(double tps) // Paper - Made static
|
|
||||||
{
|
|
||||||
return ( ( tps > 18.0 ) ? ChatColor.GREEN : ( tps > 16.0 ) ? ChatColor.YELLOW : ChatColor.RED ).toString()
|
|
||||||
- + ( ( tps > 20.0 ) ? "*" : "" ) + Math.min( Math.round( tps * 100.0 ) / 100.0, 20.0 );
|
|
||||||
+ + ( ( tps > 21.0 ) ? "*" : "" ) + Math.min( Math.round( tps * 100.0 ) / 100.0, 20.0 ); // Paper - only print * at 21, we commonly peak to 20.02 as the tick sleep is not accurate enough, stop the noise
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,25 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 23:12:03 -0600
|
|
||||||
Subject: [PATCH] Only refresh abilities if needed
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index cc471418b37a745ecea1af964e81bc0362cf7d94..fb792de46ff80a6bad77a47954861cddfd17f2d9 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -1439,12 +1439,13 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFlying(boolean value) {
|
|
||||||
+ boolean needsUpdate = getHandle().abilities.isFlying != value; // Paper - Only refresh abilities if needed
|
|
||||||
if (!getAllowFlight() && value) {
|
|
||||||
throw new IllegalArgumentException("Cannot make player fly if getAllowFlight() is false");
|
|
||||||
}
|
|
||||||
|
|
||||||
getHandle().abilities.isFlying = value;
|
|
||||||
- getHandle().updateAbilities();
|
|
||||||
+ if (needsUpdate) getHandle().updateAbilities(); // Paper - Only refresh abilities if needed
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
@ -1,140 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Byteflux <byte@byteflux.net>
|
|
||||||
Date: Tue, 1 Mar 2016 23:45:08 -0600
|
|
||||||
Subject: [PATCH] Entity Origin API
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/nbt/NBTTagList.java b/src/main/java/net/minecraft/nbt/NBTTagList.java
|
|
||||||
index 4f6f6f51f9807bafa88482c0fe776c8b163107d7..ce6572df63c4e7341708aee60330fb214a3fe416 100644
|
|
||||||
--- a/src/main/java/net/minecraft/nbt/NBTTagList.java
|
|
||||||
+++ b/src/main/java/net/minecraft/nbt/NBTTagList.java
|
|
||||||
@@ -190,6 +190,7 @@ public class NBTTagList extends NBTList<NBTBase> {
|
|
||||||
return new int[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public final double getDoubleAt(int i) { return this.h(i); } // Paper - OBFHELPER
|
|
||||||
public double h(int i) {
|
|
||||||
if (i >= 0 && i < this.list.size()) {
|
|
||||||
NBTBase nbtbase = (NBTBase) this.list.get(i);
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index cf38d517821659e25e786a805e229ef2d626d75f..26d461196b4a998b445f8c6e67fd7ec0606344f6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -1246,6 +1246,11 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
this.navigators.add(((EntityInsentient) entity).getNavigation());
|
|
||||||
}
|
|
||||||
entity.valid = true; // CraftBukkit
|
|
||||||
+ // Paper start - Set origin location when the entity is being added to the world
|
|
||||||
+ if (entity.origin == null) {
|
|
||||||
+ entity.origin = entity.getBukkitEntity().getLocation();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index cae9da158f54438d2a397665c7ce964f6f755469..f6f0d551e22ff085935c1543bf84392de0368214 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -247,6 +247,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
public org.bukkit.projectiles.ProjectileSource projectileSource; // For projectiles only
|
|
||||||
public boolean forceExplosionKnockback; // SPIGOT-949
|
|
||||||
public boolean persistentInvisibility = false;
|
|
||||||
+ public org.bukkit.Location origin; // Paper
|
|
||||||
// Spigot start
|
|
||||||
public final org.spigotmc.ActivationRange.ActivationType activationType = org.spigotmc.ActivationRange.initializeEntityActivationType(this);
|
|
||||||
public final boolean defaultActivationState;
|
|
||||||
@@ -1625,6 +1626,12 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
this.bukkitEntity.storeBukkitValues(nbttagcompound);
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
+ // Paper start - Save the entity's origin location
|
|
||||||
+ if (this.origin != null) {
|
|
||||||
+ nbttagcompound.setUUID("Paper.OriginWorld", origin.getWorld().getUID());
|
|
||||||
+ nbttagcompound.set("Paper.Origin", this.createList(origin.getX(), origin.getY(), origin.getZ()));
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
return nbttagcompound;
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
CrashReport crashreport = CrashReport.a(throwable, "Saving entity NBT");
|
|
||||||
@@ -1747,6 +1754,17 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
+ // Paper start - Restore the entity's origin location
|
|
||||||
+ NBTTagList originTag = nbttagcompound.getList("Paper.Origin", 6);
|
|
||||||
+ if (!originTag.isEmpty()) {
|
|
||||||
+ org.bukkit.World originWorld = world.getWorld();
|
|
||||||
+ if (nbttagcompound.hasKey("Paper.OriginWorld")) {
|
|
||||||
+ originWorld = Bukkit.getWorld(nbttagcompound.getUUID("Paper.OriginWorld"));
|
|
||||||
+ }
|
|
||||||
+ origin = new org.bukkit.Location(originWorld, originTag.getDoubleAt(0), originTag.getDoubleAt(1), originTag.getDoubleAt(2));
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
CrashReport crashreport = CrashReport.a(throwable, "Loading entity NBT");
|
|
||||||
CrashReportSystemDetails crashreportsystemdetails = crashreport.a("Entity being loaded");
|
|
||||||
@@ -1808,6 +1826,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
|
|
||||||
protected abstract void saveData(NBTTagCompound nbttagcompound);
|
|
||||||
|
|
||||||
+ protected final NBTTagList createList(double... adouble) { return a(adouble); } // Paper - OBFHELPER
|
|
||||||
protected NBTTagList a(double... adouble) {
|
|
||||||
NBTTagList nbttaglist = new NBTTagList();
|
|
||||||
double[] adouble1 = adouble;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java b/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
index 901522f24b8bc58861e46eda400dbab92bb6401d..3f10e41b18e09186635fd6f7c653b04db7b39d8e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/item/EntityFallingBlock.java
|
|
||||||
@@ -293,6 +293,14 @@ public class EntityFallingBlock extends Entity {
|
|
||||||
this.block = Blocks.SAND.getBlockData();
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Try and load origin location from the old NBT tags for backwards compatibility
|
|
||||||
+ if (nbttagcompound.hasKey("SourceLoc_x")) {
|
|
||||||
+ int srcX = nbttagcompound.getInt("SourceLoc_x");
|
|
||||||
+ int srcY = nbttagcompound.getInt("SourceLoc_y");
|
|
||||||
+ int srcZ = nbttagcompound.getInt("SourceLoc_z");
|
|
||||||
+ origin = new org.bukkit.Location(world.getWorld(), srcX, srcY, srcZ);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
public void a(boolean flag) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java b/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
index 4f4b2b8d58223fa22d6a7af5c94cfb36399b9641..535e7d7297d81026b8586d5049b72fa65519b464 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/item/EntityTNTPrimed.java
|
|
||||||
@@ -120,6 +120,14 @@ public class EntityTNTPrimed extends Entity {
|
|
||||||
@Override
|
|
||||||
protected void loadData(NBTTagCompound nbttagcompound) {
|
|
||||||
this.setFuseTicks(nbttagcompound.getShort("Fuse"));
|
|
||||||
+ // Paper start - Try and load origin location from the old NBT tags for backwards compatibility
|
|
||||||
+ if (nbttagcompound.hasKey("SourceLoc_x")) {
|
|
||||||
+ int srcX = nbttagcompound.getInt("SourceLoc_x");
|
|
||||||
+ int srcY = nbttagcompound.getInt("SourceLoc_y");
|
|
||||||
+ int srcZ = nbttagcompound.getInt("SourceLoc_z");
|
|
||||||
+ origin = new org.bukkit.Location(world.getWorld(), srcX, srcY, srcZ);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
index 3cf81734c8580f4d88ea97b6ac737a370b413c84..220bad90bbb9a90c3f23562bf0fb109fce379682 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
@@ -1062,4 +1062,12 @@ public abstract class CraftEntity implements org.bukkit.entity.Entity {
|
|
||||||
return spigot;
|
|
||||||
}
|
|
||||||
// Spigot end
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ @Override
|
|
||||||
+ public Location getOrigin() {
|
|
||||||
+ Location origin = getHandle().origin;
|
|
||||||
+ return origin == null ? null : origin.clone();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
@ -1,65 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Tue, 1 Mar 2016 23:52:34 -0600
|
|
||||||
Subject: [PATCH] Prevent tile entity and entity crashes
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 62735e7908f86e31f7a89ce2071ddebe176c0385..87e37c38825ad20fc11f41ea2a4512753266d2b4 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -737,11 +737,13 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
|
|
||||||
gameprofilerfiller.exit();
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
- CrashReport crashreport = CrashReport.a(throwable, "Ticking block entity");
|
|
||||||
- CrashReportSystemDetails crashreportsystemdetails = crashreport.a("Block entity being ticked");
|
|
||||||
-
|
|
||||||
- tileentity.a(crashreportsystemdetails);
|
|
||||||
- throw new ReportedException(crashreport);
|
|
||||||
+ // Paper start - Prevent tile entity and entity crashes
|
|
||||||
+ System.err.println("TileEntity threw exception at " + tileentity.world.getWorld().getName() + ":" + tileentity.position.getX() + "," + tileentity.position.getY() + "," + tileentity.position.getZ());
|
|
||||||
+ throwable.printStackTrace();
|
|
||||||
+ tilesThisCycle--;
|
|
||||||
+ this.tileEntityListTick.remove(tileTickPosition--);
|
|
||||||
+ continue;
|
|
||||||
+ // Paper end
|
|
||||||
// Spigot start
|
|
||||||
} finally {
|
|
||||||
tileentity.tickTimer.stopTiming();
|
|
||||||
@@ -806,11 +808,12 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
try {
|
|
||||||
consumer.accept(entity);
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
- CrashReport crashreport = CrashReport.a(throwable, "Ticking entity");
|
|
||||||
- CrashReportSystemDetails crashreportsystemdetails = crashreport.a("Entity being ticked");
|
|
||||||
-
|
|
||||||
- entity.appendEntityCrashDetails(crashreportsystemdetails);
|
|
||||||
- throw new ReportedException(crashreport);
|
|
||||||
+ // Paper start - Prevent tile entity and entity crashes
|
|
||||||
+ System.err.println("Entity threw exception at " + entity.world.getWorld().getName() + ":" + entity.locX() + "," + entity.locY() + "," + entity.locZ());
|
|
||||||
+ throwable.printStackTrace();
|
|
||||||
+ entity.dead = true;
|
|
||||||
+ return;
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
index 9ebd91e1309938f81583eb3d4dd97fd39bcc930a..58789a6e285c31947508deae37caefe7e182278c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntity.java
|
|
||||||
@@ -208,7 +208,12 @@ public abstract class TileEntity implements net.minecraft.server.KeyedObject { /
|
|
||||||
return IRegistry.BLOCK_ENTITY_TYPE.getKey(this.getTileType()) + " // " + this.getClass().getCanonicalName();
|
|
||||||
});
|
|
||||||
if (this.world != null) {
|
|
||||||
- CrashReportSystemDetails.a(crashreportsystemdetails, this.position, this.getBlock());
|
|
||||||
+ // Paper start - Prevent TileEntity and Entity crashes
|
|
||||||
+ IBlockData block = this.getBlock();
|
|
||||||
+ if (block != null) {
|
|
||||||
+ CrashReportSystemDetails.a(crashreportsystemdetails, this.position, block);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
CrashReportSystemDetails.a(crashreportsystemdetails, this.position, this.world.getType(this.position));
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,89 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 1 Mar 2016 23:58:50 -0600
|
|
||||||
Subject: [PATCH] Configurable top of nether void damage
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index d16ae924bcbe31c964f7fb448757c748e5c4418c..4bba6977a0287837b8927718c040ac61463f0469 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -134,4 +134,19 @@ public class PaperWorldConfig {
|
|
||||||
if (fallingBlockHeightNerf != 0) log("Falling Block Height Limit set to Y: " + fallingBlockHeightNerf);
|
|
||||||
if (entityTNTHeightNerf != 0) log("TNT Entity Height Limit set to Y: " + entityTNTHeightNerf);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int netherVoidTopDamageHeight;
|
|
||||||
+ public boolean doNetherTopVoidDamage() { return netherVoidTopDamageHeight > 0; }
|
|
||||||
+ private void netherVoidTopDamageHeight() {
|
|
||||||
+ netherVoidTopDamageHeight = getInt("nether-ceiling-void-damage-height", 0);
|
|
||||||
+ log("Top of the nether void damage height: " + netherVoidTopDamageHeight);
|
|
||||||
+
|
|
||||||
+ if (PaperConfig.version < 18) {
|
|
||||||
+ boolean legacy = getBoolean("nether-ceiling-void-damage", false);
|
|
||||||
+ if (legacy) {
|
|
||||||
+ netherVoidTopDamageHeight = 128;
|
|
||||||
+ set("nether-ceiling-void-damage-height", netherVoidTopDamageHeight);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index f6f0d551e22ff085935c1543bf84392de0368214..df2582c3b00977d799b189214d7d4f30ded5b66d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -500,9 +500,16 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
this.fallDistance *= 0.5F;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Configurable nether ceiling damage
|
|
||||||
+
|
|
||||||
+ // Extracted to own function
|
|
||||||
+ /*
|
|
||||||
if (this.locY() < -64.0D) {
|
|
||||||
this.an();
|
|
||||||
}
|
|
||||||
+ */
|
|
||||||
+ this.performVoidDamage();
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
if (!this.world.isClientSide) {
|
|
||||||
this.setFlag(0, this.fireTicks > 0);
|
|
||||||
@@ -595,6 +602,17 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
this.setFireTicks(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ protected void performVoidDamage() {
|
|
||||||
+ if (this.locY() < -64.0D || (this.world.getWorld().getEnvironment() == org.bukkit.World.Environment.NETHER
|
|
||||||
+ && world.paperConfig.doNetherTopVoidDamage()
|
|
||||||
+ && this.locY() >= world.paperConfig.netherVoidTopDamageHeight)) {
|
|
||||||
+ this.doVoidDamage();
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
+ protected final void doVoidDamage() { this.an(); } // Paper - OBFHELPER
|
|
||||||
protected void an() {
|
|
||||||
this.die();
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartAbstract.java b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartAbstract.java
|
|
||||||
index 37785b27c3d9ffd010f09f53b2ca09941f609872..1f94cc096d95129d85a6278b1e369729df93d27d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartAbstract.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartAbstract.java
|
|
||||||
@@ -330,9 +330,15 @@ public abstract class EntityMinecartAbstract extends Entity {
|
|
||||||
this.setDamage(this.getDamage() - 1.0F);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Configurable nether ceiling damage
|
|
||||||
+ // Extracted to own function
|
|
||||||
+ /*
|
|
||||||
if (this.locY() < -64.0D) {
|
|
||||||
this.an();
|
|
||||||
}
|
|
||||||
+ */
|
|
||||||
+ this.performVoidDamage();
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
// this.doPortalTick(); // CraftBukkit - handled in postTick
|
|
||||||
if (this.world.isClientSide) {
|
|
@ -1,19 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Wed, 2 Mar 2016 00:03:55 -0600
|
|
||||||
Subject: [PATCH] Check online mode before converting and renaming player data
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/storage/WorldNBTStorage.java b/src/main/java/net/minecraft/world/level/storage/WorldNBTStorage.java
|
|
||||||
index c3a1b5943b0dad8701d566c45b9b474dac7e5c8a..191c9e9a00b9871038f60d54bc22620322f6bdbd 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/storage/WorldNBTStorage.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/storage/WorldNBTStorage.java
|
|
||||||
@@ -56,7 +56,7 @@ public class WorldNBTStorage {
|
|
||||||
File file = new File(this.playerDir, entityhuman.getUniqueIDString() + ".dat");
|
|
||||||
// Spigot Start
|
|
||||||
boolean usingWrongFile = false;
|
|
||||||
- if ( !file.exists() )
|
|
||||||
+ if ( org.bukkit.Bukkit.getOnlineMode() && !file.exists() ) // Paper - Check online mode first
|
|
||||||
{
|
|
||||||
file = new File( this.playerDir, java.util.UUID.nameUUIDFromBytes( ( "OfflinePlayer:" + entityhuman.getName() ).getBytes( "UTF-8" ) ).toString() + ".dat");
|
|
||||||
if ( file.exists() )
|
|
@ -1,18 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Wed, 2 Mar 2016 00:32:25 -0600
|
|
||||||
Subject: [PATCH] Always tick falling blocks
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/spigotmc/ActivationRange.java b/src/main/java/org/spigotmc/ActivationRange.java
|
|
||||||
index 69c5d4e51ebf747d931fadc819973e36f001f5bc..58d22363124a9343188d8c19476e5a92f2f0b80b 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/ActivationRange.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/ActivationRange.java
|
|
||||||
@@ -91,6 +91,7 @@ public class ActivationRange
|
|
||||||
|| entity instanceof EntityFireball
|
|
||||||
|| entity instanceof EntityLightning
|
|
||||||
|| entity instanceof EntityTNTPrimed
|
|
||||||
+ || entity instanceof EntityFallingBlock // Paper - Always tick falling blocks
|
|
||||||
|| entity instanceof EntityEnderCrystal
|
|
||||||
|| entity instanceof EntityFireworks
|
|
||||||
|| entity instanceof EntityThrownTrident )
|
|
@ -1,42 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: DoctorDark <doctordark11@gmail.com>
|
|
||||||
Date: Wed, 16 Mar 2016 02:21:39 -0500
|
|
||||||
Subject: [PATCH] Configurable end credits
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 4bba6977a0287837b8927718c040ac61463f0469..e6e18f309dc09ea9416ea37dcc697ddc2b571a96 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -149,4 +149,10 @@ public class PaperWorldConfig {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableEndCredits;
|
|
||||||
+ private void disableEndCredits() {
|
|
||||||
+ disableEndCredits = getBoolean("game-mechanics.disable-end-credits", false);
|
|
||||||
+ log("End credits disabled: " + disableEndCredits);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/EntityPlayer.java b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
index 5ef8b66cf266488df75ce7399596f75273b90761..808bb68b3c5115b1219a65d0dd253bd60d543652 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
@@ -191,7 +191,7 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
private long ca = SystemUtils.getMonotonicMillis();
|
|
||||||
private Entity spectatedEntity;
|
|
||||||
public boolean worldChangeInvuln;
|
|
||||||
- private boolean cd;
|
|
||||||
+ private boolean cd; private void setHasSeenCredits(boolean has) { this.cd = has; } // Paper - OBFHELPER
|
|
||||||
private final RecipeBookServer recipeBook = new RecipeBookServer();
|
|
||||||
private Vec3D cf;
|
|
||||||
private int cg;
|
|
||||||
@@ -896,6 +896,7 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
this.decouple();
|
|
||||||
this.getWorldServer().removePlayer(this);
|
|
||||||
if (!this.viewingCredits) {
|
|
||||||
+ if (world.paperConfig.disableEndCredits) this.setHasSeenCredits(true); // Paper - Toggle to always disable end credits
|
|
||||||
this.viewingCredits = true;
|
|
||||||
this.playerConnection.sendPacket(new PacketPlayOutGameStateChange(PacketPlayOutGameStateChange.e, this.cd ? 0.0F : 1.0F));
|
|
||||||
this.cd = true;
|
|
@ -1,19 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Iceee <andrew@opticgaming.tv>
|
|
||||||
Date: Wed, 2 Mar 2016 01:39:52 -0600
|
|
||||||
Subject: [PATCH] Fix lag from explosions processing dead entities
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/Explosion.java b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
index 9c1ad56e4362cd86e7ffe2aef7fd9ec301cf9002..7786a06ba09aacaa70c346e85a9eeed9f2ffec6e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
@@ -182,7 +182,7 @@ public class Explosion {
|
|
||||||
int i1 = MathHelper.floor(this.posY + (double) f2 + 1.0D);
|
|
||||||
int j1 = MathHelper.floor(this.posZ - (double) f2 - 1.0D);
|
|
||||||
int k1 = MathHelper.floor(this.posZ + (double) f2 + 1.0D);
|
|
||||||
- List<Entity> list = this.world.getEntities(this.source, new AxisAlignedBB((double) i, (double) l, (double) j1, (double) j, (double) i1, (double) k1));
|
|
||||||
+ List<Entity> list = this.world.getEntities(this.source, new AxisAlignedBB((double) i, (double) l, (double) j1, (double) j, (double) i1, (double) k1), (com.google.common.base.Predicate<Entity>) entity -> entity.isAlive() && !entity.isSpectator()); // Paper - Fix lag from explosions processing dead entities
|
|
||||||
Vec3D vec3d = new Vec3D(this.posX, this.posY, this.posZ);
|
|
||||||
|
|
||||||
for (int l1 = 0; l1 < list.size(); ++l1) {
|
|
@ -1,148 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Byteflux <byte@byteflux.net>
|
|
||||||
Date: Wed, 2 Mar 2016 11:59:48 -0600
|
|
||||||
Subject: [PATCH] Optimize explosions
|
|
||||||
|
|
||||||
The process of determining an entity's exposure from explosions can be
|
|
||||||
expensive when there are hundreds or more entities in range.
|
|
||||||
|
|
||||||
This patch adds a per-tick cache that is used for storing and retrieving
|
|
||||||
an entity's exposure during an explosion.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index e6e18f309dc09ea9416ea37dcc697ddc2b571a96..4881b03d470646843bad1bc343eb6a6ab9072d8e 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -155,4 +155,10 @@ public class PaperWorldConfig {
|
|
||||||
disableEndCredits = getBoolean("game-mechanics.disable-end-credits", false);
|
|
||||||
log("End credits disabled: " + disableEndCredits);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean optimizeExplosions;
|
|
||||||
+ private void optimizeExplosions() {
|
|
||||||
+ optimizeExplosions = getBoolean("optimize-explosions", false);
|
|
||||||
+ log("Optimize explosions: " + optimizeExplosions);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index fb0d985b5c977a7c63701484678b75928d9ee382..ed71de473d461528d74ca5b95c33b97e98128aff 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -1326,6 +1326,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
|
|
||||||
this.methodProfiler.exit();
|
|
||||||
this.methodProfiler.exit();
|
|
||||||
+ worldserver.explosionDensityCache.clear(); // Paper - Optimize explosions
|
|
||||||
}
|
|
||||||
|
|
||||||
this.methodProfiler.exitEnter("connection");
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/Explosion.java b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
index 7786a06ba09aacaa70c346e85a9eeed9f2ffec6e..6a7af2c0c3c294b10c6ddbf98babb0f30d7d5f56 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
@@ -201,7 +201,7 @@ public class Explosion {
|
|
||||||
d8 /= d11;
|
|
||||||
d9 /= d11;
|
|
||||||
d10 /= d11;
|
|
||||||
- double d12 = (double) a(vec3d, entity);
|
|
||||||
+ double d12 = this.getBlockDensity(vec3d, entity); // Paper - Optimize explosions
|
|
||||||
double d13 = (1.0D - d7) * d12;
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
@@ -420,4 +420,84 @@ public class Explosion {
|
|
||||||
|
|
||||||
private Effect() {}
|
|
||||||
}
|
|
||||||
+ // Paper start - Optimize explosions
|
|
||||||
+ private float getBlockDensity(Vec3D vec3d, Entity entity) {
|
|
||||||
+ if (!this.world.paperConfig.optimizeExplosions) {
|
|
||||||
+ return a(vec3d, entity);
|
|
||||||
+ }
|
|
||||||
+ CacheKey key = new CacheKey(this, entity.getBoundingBox());
|
|
||||||
+ Float blockDensity = this.world.explosionDensityCache.get(key);
|
|
||||||
+ if (blockDensity == null) {
|
|
||||||
+ blockDensity = a(vec3d, entity);
|
|
||||||
+ this.world.explosionDensityCache.put(key, blockDensity);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return blockDensity;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ static class CacheKey {
|
|
||||||
+ private final World world;
|
|
||||||
+ private final double posX, posY, posZ;
|
|
||||||
+ private final double minX, minY, minZ;
|
|
||||||
+ private final double maxX, maxY, maxZ;
|
|
||||||
+
|
|
||||||
+ public CacheKey(Explosion explosion, AxisAlignedBB aabb) {
|
|
||||||
+ this.world = explosion.world;
|
|
||||||
+ this.posX = explosion.posX;
|
|
||||||
+ this.posY = explosion.posY;
|
|
||||||
+ this.posZ = explosion.posZ;
|
|
||||||
+ this.minX = aabb.minX;
|
|
||||||
+ this.minY = aabb.minY;
|
|
||||||
+ this.minZ = aabb.minZ;
|
|
||||||
+ this.maxX = aabb.maxX;
|
|
||||||
+ this.maxY = aabb.maxY;
|
|
||||||
+ this.maxZ = aabb.maxZ;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public boolean equals(Object o) {
|
|
||||||
+ if (this == o) return true;
|
|
||||||
+ if (o == null || getClass() != o.getClass()) return false;
|
|
||||||
+
|
|
||||||
+ CacheKey cacheKey = (CacheKey) o;
|
|
||||||
+
|
|
||||||
+ if (Double.compare(cacheKey.posX, posX) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.posY, posY) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.posZ, posZ) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.minX, minX) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.minY, minY) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.minZ, minZ) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.maxX, maxX) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.maxY, maxY) != 0) return false;
|
|
||||||
+ if (Double.compare(cacheKey.maxZ, maxZ) != 0) return false;
|
|
||||||
+ return world.equals(cacheKey.world);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public int hashCode() {
|
|
||||||
+ int result;
|
|
||||||
+ long temp;
|
|
||||||
+ result = world.hashCode();
|
|
||||||
+ temp = Double.doubleToLongBits(posX);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(posY);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(posZ);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(minX);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(minY);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(minZ);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(maxX);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(maxY);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ temp = Double.doubleToLongBits(maxZ);
|
|
||||||
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
||||||
+ return result;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 87e37c38825ad20fc11f41ea2a4512753266d2b4..06455fb3916e347c075c6aa84977dca7e25760c2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -136,6 +136,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
private org.spigotmc.TickLimiter entityLimiter;
|
|
||||||
private org.spigotmc.TickLimiter tileLimiter;
|
|
||||||
private int tileTickPosition;
|
|
||||||
+ public final Map<Explosion.CacheKey, Float> explosionDensityCache = new HashMap<>(); // Paper - Optimize explosions
|
|
||||||
|
|
||||||
public CraftWorld getWorld() {
|
|
||||||
return this.world;
|
|
@ -1,69 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 14:48:03 -0600
|
|
||||||
Subject: [PATCH] Disable explosion knockback
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 4881b03d470646843bad1bc343eb6a6ab9072d8e..2222c1bb5f8625eee4d88946e4bfdfa2fe598977 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -161,4 +161,9 @@ public class PaperWorldConfig {
|
|
||||||
optimizeExplosions = getBoolean("optimize-explosions", false);
|
|
||||||
log("Optimize explosions: " + optimizeExplosions);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableExplosionKnockback;
|
|
||||||
+ private void disableExplosionKnockback(){
|
|
||||||
+ disableExplosionKnockback = getBoolean("disable-explosion-knockback", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityLiving.java b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
index e771d84c972cea4ca1b4b5d25cd573cfcbce4579..49e95369882847c90ee7417abea6270386cd622f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
@@ -1283,6 +1283,7 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
+ boolean knockbackCancelled = world.paperConfig.disableExplosionKnockback && damagesource.isExplosion() && this instanceof EntityHuman; // Paper - Disable explosion knockback
|
|
||||||
if (flag1) {
|
|
||||||
if (flag) {
|
|
||||||
this.world.broadcastEntityEffect(this, (byte) 29);
|
|
||||||
@@ -1301,6 +1302,7 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
b0 = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ if (!knockbackCancelled) // Paper - Disable explosion knockback
|
|
||||||
this.world.broadcastEntityEffect(this, b0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1324,6 +1326,7 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
+ if (knockbackCancelled) this.world.broadcastEntityEffect(this, (byte) 2); // Paper - Disable explosion knockback
|
|
||||||
if (this.dl()) {
|
|
||||||
if (!this.f(damagesource)) {
|
|
||||||
SoundEffect soundeffect = this.getSoundDeath();
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/Explosion.java b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
index 6a7af2c0c3c294b10c6ddbf98babb0f30d7d5f56..618cf4e0d71b4b04085807314e79a02785f8a498 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/Explosion.java
|
|
||||||
@@ -217,14 +217,14 @@ public class Explosion {
|
|
||||||
double d14 = d13;
|
|
||||||
|
|
||||||
if (entity instanceof EntityLiving) {
|
|
||||||
- d14 = EnchantmentProtection.a((EntityLiving) entity, d13);
|
|
||||||
+ d14 = entity instanceof EntityHuman && world.paperConfig.disableExplosionKnockback ? 0 : EnchantmentProtection.a((EntityLiving) entity, d13); // Paper - Disable explosion knockback
|
|
||||||
}
|
|
||||||
|
|
||||||
entity.setMot(entity.getMot().add(d8 * d14, d9 * d14, d10 * d14));
|
|
||||||
if (entity instanceof EntityHuman) {
|
|
||||||
EntityHuman entityhuman = (EntityHuman) entity;
|
|
||||||
|
|
||||||
- if (!entityhuman.isSpectator() && (!entityhuman.isCreative() || !entityhuman.abilities.isFlying)) {
|
|
||||||
+ if (!entityhuman.isSpectator() && (!entityhuman.isCreative() || !entityhuman.abilities.isFlying) && !world.paperConfig.disableExplosionKnockback) { // Paper - Disable explosion knockback
|
|
||||||
this.n.put(entityhuman, new Vec3D(d8 * d13, d9 * d13, d10 * d13));
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 14:52:43 -0600
|
|
||||||
Subject: [PATCH] Disable thunder
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 2222c1bb5f8625eee4d88946e4bfdfa2fe598977..083e421f8496b5336af473b108498ed28b984774 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -166,4 +166,9 @@ public class PaperWorldConfig {
|
|
||||||
private void disableExplosionKnockback(){
|
|
||||||
disableExplosionKnockback = getBoolean("disable-explosion-knockback", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableThunder;
|
|
||||||
+ private void disableThunder() {
|
|
||||||
+ disableThunder = getBoolean("disable-thunder", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 26d461196b4a998b445f8c6e67fd7ec0606344f6..8845b5c6dc9ebff30e8aa3abd3c905addaeb3e1d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -586,7 +586,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
gameprofilerfiller.enter("thunder");
|
|
||||||
BlockPosition blockposition;
|
|
||||||
|
|
||||||
- if (flag && this.W() && this.random.nextInt(100000) == 0) {
|
|
||||||
+ if (!this.paperConfig.disableThunder && flag && this.W() && this.random.nextInt(100000) == 0) { // Paper - Disable thunder
|
|
||||||
blockposition = this.a(this.a(j, 0, k, 15));
|
|
||||||
if (this.isRainingAt(blockposition)) {
|
|
||||||
DifficultyDamageScaler difficultydamagescaler = this.getDamageScaler(blockposition);
|
|
@ -1,33 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 14:57:24 -0600
|
|
||||||
Subject: [PATCH] Disable ice and snow
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 083e421f8496b5336af473b108498ed28b984774..2f7a5a4a5a7b29750cfd777e0bc5d19a14e93fa2 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -171,4 +171,9 @@ public class PaperWorldConfig {
|
|
||||||
private void disableThunder() {
|
|
||||||
disableThunder = getBoolean("disable-thunder", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableIceAndSnow;
|
|
||||||
+ private void disableIceAndSnow(){
|
|
||||||
+ disableIceAndSnow = getBoolean("disable-ice-and-snow", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 8845b5c6dc9ebff30e8aa3abd3c905addaeb3e1d..a1769df6a3f6150d322f145199caba3839871dff 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -610,7 +610,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
}
|
|
||||||
|
|
||||||
gameprofilerfiller.exitEnter("iceandsnow");
|
|
||||||
- if (this.random.nextInt(16) == 0) {
|
|
||||||
+ if (!this.paperConfig.disableIceAndSnow && this.random.nextInt(16) == 0) { // Paper - Disable ice and snow
|
|
||||||
blockposition = this.getHighestBlockYAt(HeightMap.Type.MOTION_BLOCKING, this.a(j, 0, k, 15));
|
|
||||||
BlockPosition blockposition1 = blockposition.down();
|
|
||||||
BiomeBase biomebase = this.getBiome(blockposition);
|
|
@ -1,65 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 15:03:53 -0600
|
|
||||||
Subject: [PATCH] Configurable mob spawner tick rate
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 2f7a5a4a5a7b29750cfd777e0bc5d19a14e93fa2..4de86b09c6bc3c1974ce61b550ccb73d37f6f170 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -176,4 +176,9 @@ public class PaperWorldConfig {
|
|
||||||
private void disableIceAndSnow(){
|
|
||||||
disableIceAndSnow = getBoolean("disable-ice-and-snow", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int mobSpawnerTickRate;
|
|
||||||
+ private void mobSpawnerTickRate() {
|
|
||||||
+ mobSpawnerTickRate = getInt("mob-spawner-tick-rate", 1);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java b/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
index 8d8b03074df1635946f81bec0feae18d2f3e20aa..76c98d576d3e567ec4482b30219f5a9107cb9703 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/MobSpawnerAbstract.java
|
|
||||||
@@ -41,6 +41,7 @@ public abstract class MobSpawnerAbstract {
|
|
||||||
public int maxNearbyEntities = 6;
|
|
||||||
public int requiredPlayerRange = 16;
|
|
||||||
public int spawnRange = 4;
|
|
||||||
+ private int tickDelay = 0; // Paper
|
|
||||||
|
|
||||||
public MobSpawnerAbstract() {}
|
|
||||||
|
|
||||||
@@ -70,6 +71,10 @@ public abstract class MobSpawnerAbstract {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void c() {
|
|
||||||
+ // Paper start - Configurable mob spawner tick rate
|
|
||||||
+ if (spawnDelay > 0 && --tickDelay > 0) return;
|
|
||||||
+ tickDelay = this.a().paperConfig.mobSpawnerTickRate;
|
|
||||||
+ // Paper end
|
|
||||||
if (!this.h()) {
|
|
||||||
this.f = this.e;
|
|
||||||
} else {
|
|
||||||
@@ -84,18 +89,18 @@ public abstract class MobSpawnerAbstract {
|
|
||||||
world.addParticle(Particles.SMOKE, d0, d1, d2, 0.0D, 0.0D, 0.0D);
|
|
||||||
world.addParticle(Particles.FLAME, d0, d1, d2, 0.0D, 0.0D, 0.0D);
|
|
||||||
if (this.spawnDelay > 0) {
|
|
||||||
- --this.spawnDelay;
|
|
||||||
+ this.spawnDelay -= tickDelay; // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
this.f = this.e;
|
|
||||||
this.e = (this.e + (double) (1000.0F / ((float) this.spawnDelay + 200.0F))) % 360.0D;
|
|
||||||
} else {
|
|
||||||
- if (this.spawnDelay == -1) {
|
|
||||||
+ if (this.spawnDelay < -tickDelay) { // Paper
|
|
||||||
this.i();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.spawnDelay > 0) {
|
|
||||||
- --this.spawnDelay;
|
|
||||||
+ this.spawnDelay -= tickDelay; // Paper
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
@ -1,108 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Wed, 2 Mar 2016 23:13:07 -0600
|
|
||||||
Subject: [PATCH] Send absolute position the first time an entity is seen
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/EntityTrackerEntry.java b/src/main/java/net/minecraft/server/level/EntityTrackerEntry.java
|
|
||||||
index 9ad74b380a92e3a563e1a891e81401d8b4707bcf..beb0beb716869978be6bc5a78ce3b6cf785c5aee 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/EntityTrackerEntry.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/EntityTrackerEntry.java
|
|
||||||
@@ -4,6 +4,7 @@ import com.google.common.collect.Lists;
|
|
||||||
import com.mojang.datafixers.util.Pair;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.Collections;
|
|
||||||
+import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Set;
|
|
||||||
@@ -52,7 +53,7 @@ public class EntityTrackerEntry {
|
|
||||||
private final Entity tracker;
|
|
||||||
private final int d;
|
|
||||||
private final boolean e;
|
|
||||||
- private final Consumer<Packet<?>> f;
|
|
||||||
+ private final Consumer<Packet<?>> f; private Consumer<Packet<?>> getPacketConsumer() { return f; } // Paper - OBFHELPER
|
|
||||||
private long xLoc;
|
|
||||||
private long yLoc;
|
|
||||||
private long zLoc;
|
|
||||||
@@ -67,8 +68,23 @@ public class EntityTrackerEntry {
|
|
||||||
private boolean r;
|
|
||||||
// CraftBukkit start
|
|
||||||
private final Set<EntityPlayer> trackedPlayers;
|
|
||||||
+ // Paper start
|
|
||||||
+ private java.util.Map<EntityPlayer, Boolean> trackedPlayerMap = null;
|
|
||||||
+
|
|
||||||
+ /**
|
|
||||||
+ * Requested in https://github.com/PaperMC/Paper/issues/1537 to allow intercepting packets
|
|
||||||
+ */
|
|
||||||
+ public void sendPlayerPacket(EntityPlayer player, Packet packet) {
|
|
||||||
+ player.playerConnection.sendPacket(packet);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public EntityTrackerEntry(WorldServer worldserver, Entity entity, int i, boolean flag, Consumer<Packet<?>> consumer, java.util.Map<EntityPlayer, Boolean> trackedPlayers) {
|
|
||||||
+ this(worldserver, entity, i, flag, consumer, trackedPlayers.keySet());
|
|
||||||
+ trackedPlayerMap = trackedPlayers;
|
|
||||||
+ }
|
|
||||||
|
|
||||||
public EntityTrackerEntry(WorldServer worldserver, Entity entity, int i, boolean flag, Consumer<Packet<?>> consumer, Set<EntityPlayer> trackedPlayers) {
|
|
||||||
+ // Paper end
|
|
||||||
this.trackedPlayers = trackedPlayers;
|
|
||||||
// CraftBukkit end
|
|
||||||
this.m = Vec3D.ORIGIN;
|
|
||||||
@@ -189,7 +205,25 @@ public class EntityTrackerEntry {
|
|
||||||
}
|
|
||||||
|
|
||||||
if (packet1 != null) {
|
|
||||||
- this.f.accept(packet1);
|
|
||||||
+ // paper start
|
|
||||||
+ if (trackedPlayerMap == null || packet1 instanceof PacketPlayOutEntityTeleport) {
|
|
||||||
+ this.f.accept((packet1));
|
|
||||||
+ } else {
|
|
||||||
+ PacketPlayOutEntityTeleport teleportPacket = null;
|
|
||||||
+
|
|
||||||
+ for (java.util.Map.Entry<EntityPlayer, Boolean> viewer : trackedPlayerMap.entrySet()) {
|
|
||||||
+ if (viewer.getValue()) {
|
|
||||||
+ viewer.setValue(false);
|
|
||||||
+ if (teleportPacket == null) {
|
|
||||||
+ teleportPacket = new PacketPlayOutEntityTeleport(this.tracker);
|
|
||||||
+ }
|
|
||||||
+ sendPlayerPacket(viewer.getKey(), teleportPacket);
|
|
||||||
+ } else {
|
|
||||||
+ sendPlayerPacket(viewer.getKey(), packet1);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
this.c();
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/PlayerChunkMap.java b/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
index 62245fa420390dc0a70ba9a95505dc46cd8aa64a..788a45d5426f0752509442aec2d28b1f32f63cb1 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
@@ -1303,10 +1303,14 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
|
||||||
private final Entity tracker;
|
|
||||||
private final int trackingDistance;
|
|
||||||
private SectionPosition e;
|
|
||||||
- public final Set<EntityPlayer> trackedPlayers = Sets.newHashSet();
|
|
||||||
+ // Paper start
|
|
||||||
+ // Replace trackedPlayers Set with a Map. The value is true until the player receives
|
|
||||||
+ // their first update (which is forced to have absolute coordinates), false afterward.
|
|
||||||
+ public java.util.Map<EntityPlayer, Boolean> trackedPlayerMap = new java.util.HashMap<>();
|
|
||||||
+ public Set<EntityPlayer> trackedPlayers = trackedPlayerMap.keySet();
|
|
||||||
|
|
||||||
public EntityTracker(Entity entity, int i, int j, boolean flag) {
|
|
||||||
- this.trackerEntry = new EntityTrackerEntry(PlayerChunkMap.this.world, entity, j, flag, this::broadcast, trackedPlayers); // CraftBukkit
|
|
||||||
+ this.trackerEntry = new EntityTrackerEntry(PlayerChunkMap.this.world, entity, j, flag, this::broadcast, trackedPlayerMap); // CraftBukkit // Paper
|
|
||||||
this.tracker = entity;
|
|
||||||
this.trackingDistance = i;
|
|
||||||
this.e = SectionPosition.a(entity);
|
|
||||||
@@ -1388,7 +1392,7 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
|
||||||
entityplayer.removeQueue.remove(Integer.valueOf(this.tracker.getId()));
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
- if (flag1 && this.trackedPlayers.add(entityplayer)) {
|
|
||||||
+ if (flag1 && this.trackedPlayerMap.putIfAbsent(entityplayer, true) == null) { // Paper
|
|
||||||
this.trackerEntry.b(entityplayer);
|
|
||||||
}
|
|
||||||
} else if (this.trackedPlayers.remove(entityplayer)) {
|
|
@ -1,67 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Byteflux <byte@byteflux.net>
|
|
||||||
Date: Wed, 2 Mar 2016 23:30:53 -0600
|
|
||||||
Subject: [PATCH] Add BeaconEffectEvent
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntityBeacon.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntityBeacon.java
|
|
||||||
index c6914c8d2a3d1057c98537a3538097d3ac6149e0..4098357d60165a4c670a7bc5134abf66178124c6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntityBeacon.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntityBeacon.java
|
|
||||||
@@ -42,6 +42,11 @@ import net.minecraft.world.phys.AxisAlignedBB;
|
|
||||||
import org.bukkit.craftbukkit.potion.CraftPotionUtil;
|
|
||||||
import org.bukkit.potion.PotionEffect;
|
|
||||||
// CraftBukkit end
|
|
||||||
+// Paper start
|
|
||||||
+import org.bukkit.craftbukkit.event.CraftEventFactory;
|
|
||||||
+import org.bukkit.entity.Player;
|
|
||||||
+import com.destroystokyo.paper.event.block.BeaconEffectEvent;
|
|
||||||
+// Paper end
|
|
||||||
|
|
||||||
public class TileEntityBeacon extends TileEntity implements ITileInventory, ITickable {
|
|
||||||
|
|
||||||
@@ -268,14 +273,30 @@ public class TileEntityBeacon extends TileEntity implements ITileInventory, ITic
|
|
||||||
}
|
|
||||||
|
|
||||||
private void applyEffect(List list, MobEffectList effects, int i, int b0) {
|
|
||||||
+ // Paper - BeaconEffectEvent
|
|
||||||
+ applyEffect(list, effects, i, b0, true);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void applyEffect(List list, MobEffectList effects, int i, int b0, boolean isPrimary) {
|
|
||||||
+ // Paper - BeaconEffectEvent
|
|
||||||
{
|
|
||||||
Iterator iterator = list.iterator();
|
|
||||||
|
|
||||||
EntityHuman entityhuman;
|
|
||||||
|
|
||||||
+ // Paper start - BeaconEffectEvent
|
|
||||||
+ org.bukkit.block.Block block = world.getWorld().getBlockAt(position.getX(), position.getY(), position.getZ());
|
|
||||||
+ PotionEffect effect = CraftPotionUtil.toBukkit(new MobEffect(effects, i, b0, true, true));
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
while (iterator.hasNext()) {
|
|
||||||
entityhuman = (EntityHuman) iterator.next();
|
|
||||||
- entityhuman.addEffect(new MobEffect(effects, i, b0, true, true), org.bukkit.event.entity.EntityPotionEffectEvent.Cause.BEACON);
|
|
||||||
+
|
|
||||||
+ // Paper start - BeaconEffectEvent
|
|
||||||
+ BeaconEffectEvent event = new BeaconEffectEvent(block, effect, (Player) entityhuman.getBukkitEntity(), isPrimary);
|
|
||||||
+ if (CraftEventFactory.callEvent(event).isCancelled()) continue;
|
|
||||||
+ entityhuman.addEffect(new MobEffect(CraftPotionUtil.fromBukkit(event.getEffect())), org.bukkit.event.entity.EntityPotionEffectEvent.Cause.BEACON);
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -298,10 +319,10 @@ public class TileEntityBeacon extends TileEntity implements ITileInventory, ITic
|
|
||||||
int i = getLevel();
|
|
||||||
List list = getHumansInRange();
|
|
||||||
|
|
||||||
- applyEffect(list, this.primaryEffect, i, b0);
|
|
||||||
+ applyEffect(list, this.primaryEffect, i, b0, true); // Paper - BeaconEffectEvent
|
|
||||||
|
|
||||||
if (hasSecondaryEffect()) {
|
|
||||||
- applyEffect(list, this.secondaryEffect, i, 0);
|
|
||||||
+ applyEffect(list, this.secondaryEffect, i, 0, false); // Paper - BeaconEffectEvent
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,46 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 23:34:44 -0600
|
|
||||||
Subject: [PATCH] Configurable container update tick rate
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 4de86b09c6bc3c1974ce61b550ccb73d37f6f170..5a4c3a8c511f22c8c3240c9c7cd83a65119c1054 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -181,4 +181,9 @@ public class PaperWorldConfig {
|
|
||||||
private void mobSpawnerTickRate() {
|
|
||||||
mobSpawnerTickRate = getInt("mob-spawner-tick-rate", 1);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int containerUpdateTickRate;
|
|
||||||
+ private void containerUpdateTickRate() {
|
|
||||||
+ containerUpdateTickRate = getInt("container-update-tick-rate", 1);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/EntityPlayer.java b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
index 808bb68b3c5115b1219a65d0dd253bd60d543652..cda0e7f8f9a9d66ac4e5a3f52609a4271bf0c4b5 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
@@ -210,6 +210,7 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
public boolean e;
|
|
||||||
public int ping;
|
|
||||||
public boolean viewingCredits;
|
|
||||||
+ private int containerUpdateDelay; // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
public String displayName;
|
|
||||||
@@ -534,7 +535,12 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
--this.noDamageTicks;
|
|
||||||
}
|
|
||||||
|
|
||||||
- this.activeContainer.c();
|
|
||||||
+ // Paper start - Configurable container update tick rate
|
|
||||||
+ if (--containerUpdateDelay <= 0) {
|
|
||||||
+ this.activeContainer.c();
|
|
||||||
+ containerUpdateDelay = world.paperConfig.containerUpdateTickRate;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
if (!this.world.isClientSide && !this.activeContainer.canUse(this)) {
|
|
||||||
this.closeInventory();
|
|
||||||
this.activeContainer = this.defaultContainer;
|
|
@ -1,25 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Techcable <Techcable@outlook.com>
|
|
||||||
Date: Wed, 2 Mar 2016 23:42:37 -0600
|
|
||||||
Subject: [PATCH] Use UserCache for player heads
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaSkull.java b/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaSkull.java
|
|
||||||
index b8cac55c9fdcebe7703f179d25ad8cfb15c78a0e..dff67a48961399f3746f99b4f2363724bfe51c36 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaSkull.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaSkull.java
|
|
||||||
@@ -166,7 +166,13 @@ class CraftMetaSkull extends CraftMetaItem implements SkullMeta {
|
|
||||||
if (name == null) {
|
|
||||||
setProfile(null);
|
|
||||||
} else {
|
|
||||||
- setProfile(new GameProfile(null, name));
|
|
||||||
+ // Paper start - Use Online Players Skull
|
|
||||||
+ GameProfile newProfile = null;
|
|
||||||
+ net.minecraft.server.EntityPlayer player = net.minecraft.server.MinecraftServer.getServer().getPlayerList().getPlayer(name);
|
|
||||||
+ if (player != null) newProfile = player.getProfile();
|
|
||||||
+ if (newProfile == null) newProfile = new GameProfile(null, name);
|
|
||||||
+ setProfile(newProfile);
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
@ -1,21 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Wed, 2 Mar 2016 23:45:17 -0600
|
|
||||||
Subject: [PATCH] Disable spigot tick limiters
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 06455fb3916e347c075c6aa84977dca7e25760c2..26d9388badb0328735abf0319a42bd633c90dc4b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -707,9 +707,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
// Spigot start
|
|
||||||
// Iterator iterator = this.tileEntityListTick.iterator();
|
|
||||||
int tilesThisCycle = 0;
|
|
||||||
- for (tileLimiter.initTick();
|
|
||||||
- tilesThisCycle < tileEntityListTick.size() && (tilesThisCycle % 10 != 0 || tileLimiter.shouldContinue());
|
|
||||||
- tileTickPosition++, tilesThisCycle++) {
|
|
||||||
+ for (tileTickPosition = 0; tileTickPosition < tileEntityListTick.size(); tileTickPosition++) { // Paper - Disable tick limiters
|
|
||||||
tileTickPosition = (tileTickPosition < tileEntityListTick.size()) ? tileTickPosition : 0;
|
|
||||||
TileEntity tileentity = (TileEntity) this.tileEntityListTick.get(tileTickPosition);
|
|
||||||
// Spigot start
|
|
@ -1,48 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Steve Anton <anxuiz.nx@gmail.com>
|
|
||||||
Date: Thu, 3 Mar 2016 00:09:38 -0600
|
|
||||||
Subject: [PATCH] Add PlayerInitialSpawnEvent
|
|
||||||
|
|
||||||
For modifying a player's initial spawn location as they join the server
|
|
||||||
|
|
||||||
This is a duplicate API from spigot, so use our duplicate subclass and
|
|
||||||
improve setPosition to use raw
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/players/PlayerList.java b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
index 6ebd4ec781aa215c2b941261250c15c87c223cab..46c516b9ff089a3c885d635244942fd5a6ecf132 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
@@ -213,7 +213,7 @@ public abstract class PlayerList {
|
|
||||||
|
|
||||||
// Spigot start - spawn location event
|
|
||||||
Player bukkitPlayer = entityplayer.getBukkitEntity();
|
|
||||||
- org.spigotmc.event.player.PlayerSpawnLocationEvent ev = new org.spigotmc.event.player.PlayerSpawnLocationEvent(bukkitPlayer, bukkitPlayer.getLocation());
|
|
||||||
+ org.spigotmc.event.player.PlayerSpawnLocationEvent ev = new com.destroystokyo.paper.event.player.PlayerInitialSpawnEvent(bukkitPlayer, bukkitPlayer.getLocation()); // Paper use our duplicate event
|
|
||||||
cserver.getPluginManager().callEvent(ev);
|
|
||||||
|
|
||||||
Location loc = ev.getSpawnLocation();
|
|
||||||
@@ -221,7 +221,10 @@ public abstract class PlayerList {
|
|
||||||
|
|
||||||
entityplayer.spawnIn(worldserver1);
|
|
||||||
entityplayer.playerInteractManager.a((WorldServer) entityplayer.world);
|
|
||||||
- entityplayer.setLocation(loc.getX(), loc.getY(), loc.getZ(), loc.getYaw(), loc.getPitch());
|
|
||||||
+ // Paper start - set raw so we aren't fully joined to the world (not added to chunk or world)
|
|
||||||
+ entityplayer.setPositionRaw(loc.getX(), loc.getY(), loc.getZ());
|
|
||||||
+ entityplayer.setYawPitch(loc.getYaw(), loc.getPitch());
|
|
||||||
+ // Paper end
|
|
||||||
// Spigot end
|
|
||||||
|
|
||||||
// CraftBukkit - Moved message to after join
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index df2582c3b00977d799b189214d7d4f30ded5b66d..77cc0aeb979369df2156f8fb916067f608b61ebf 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -398,7 +398,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
return d1 * d1 + d2 * d2 + d3 * d3 < d0 * d0;
|
|
||||||
}
|
|
||||||
|
|
||||||
- protected void setYawPitch(float f, float f1) {
|
|
||||||
+ public void setYawPitch(float f, float f1) { // Paper - protected -> public
|
|
||||||
// CraftBukkit start - yaw was sometimes set to NaN, so we need to set it back to 0
|
|
||||||
if (Float.isNaN(f)) {
|
|
||||||
f = 0;
|
|
@ -1,37 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 3 Mar 2016 01:13:45 -0600
|
|
||||||
Subject: [PATCH] Configurable Disabling Cat Chest Detection
|
|
||||||
|
|
||||||
Offers a gameplay feature to stop cats from blocking chests
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 5a4c3a8c511f22c8c3240c9c7cd83a65119c1054..70e074cdf2087e638af8e0f3878d0ef8eb7305cc 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -186,4 +186,9 @@ public class PaperWorldConfig {
|
|
||||||
private void containerUpdateTickRate() {
|
|
||||||
containerUpdateTickRate = getInt("container-update-tick-rate", 1);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableChestCatDetection;
|
|
||||||
+ private void disableChestCatDetection() {
|
|
||||||
+ disableChestCatDetection = getBoolean("game-mechanics.disable-chest-cat-detection", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockChest.java b/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
index b229faad99120c67b089f7680d800fbe594fe7da..b2c29cff5883868cb56a4e376ab946ac929abc94 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockChest.java
|
|
||||||
@@ -312,6 +312,11 @@ public class BlockChest extends BlockChestAbstract<TileEntityChest> implements I
|
|
||||||
}
|
|
||||||
|
|
||||||
private static boolean b(GeneratorAccess generatoraccess, BlockPosition blockposition) {
|
|
||||||
+ // Paper start - Option to disable chest cat detection
|
|
||||||
+ if (((World) generatoraccess).paperConfig.disableChestCatDetection) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
List<EntityCat> list = generatoraccess.a(EntityCat.class, new AxisAlignedBB((double) blockposition.getX(), (double) (blockposition.getY() + 1), (double) blockposition.getZ(), (double) (blockposition.getX() + 1), (double) (blockposition.getY() + 2), (double) (blockposition.getZ() + 1)));
|
|
||||||
|
|
||||||
if (!list.isEmpty()) {
|
|
@ -1,119 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 3 Mar 2016 01:17:12 -0600
|
|
||||||
Subject: [PATCH] Ensure commands are not ran async
|
|
||||||
|
|
||||||
Plugins calling Player.chat("/foo") or Server.dispatchCommand() could
|
|
||||||
trigger the server to execute a command while on another thread.
|
|
||||||
|
|
||||||
These commands would then process EXPECTING to be on the main thread, leaving to
|
|
||||||
very hard to trace concurrency issues.
|
|
||||||
|
|
||||||
This change will synchronize the command execution back to the main thread, causing a
|
|
||||||
big slowdown in execution but throwing an exception at same time to raise awareness
|
|
||||||
that it is happening so that plugin authors can fix their code to stop executing commands async.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/network/PlayerConnection.java b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
index 5db09e60c2ac1f4cb0da3190e57896ccae7c58a3..865d8efa2d480ae7edc286e3e79f2997a191ee5c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
@@ -1854,6 +1854,29 @@ public class PlayerConnection implements PacketListenerPlayIn {
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!async && s.startsWith("/")) {
|
|
||||||
+ // Paper Start
|
|
||||||
+ if (!org.spigotmc.AsyncCatcher.shuttingDown && !org.bukkit.Bukkit.isPrimaryThread()) {
|
|
||||||
+ final String fCommandLine = s;
|
|
||||||
+ MinecraftServer.LOGGER.log(org.apache.logging.log4j.Level.ERROR, "Command Dispatched Async: " + fCommandLine);
|
|
||||||
+ MinecraftServer.LOGGER.log(org.apache.logging.log4j.Level.ERROR, "Please notify author of plugin causing this execution to fix this bug! see: http://bit.ly/1oSiM6C", new Throwable());
|
|
||||||
+ Waitable wait = new Waitable() {
|
|
||||||
+ @Override
|
|
||||||
+ protected Object evaluate() {
|
|
||||||
+ chat(fCommandLine, false);
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ };
|
|
||||||
+ minecraftServer.processQueue.add(wait);
|
|
||||||
+ try {
|
|
||||||
+ wait.get();
|
|
||||||
+ return;
|
|
||||||
+ } catch (InterruptedException e) {
|
|
||||||
+ Thread.currentThread().interrupt(); // This is proper habit for java. If we aren't handling it, pass it on!
|
|
||||||
+ } catch (Exception e) {
|
|
||||||
+ throw new RuntimeException("Exception processing chat command", e.getCause());
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper End
|
|
||||||
this.handleCommand(s);
|
|
||||||
} else if (this.player.getChatFlags() == EnumChatVisibility.SYSTEM) {
|
|
||||||
// Do nothing, this is coming from a plugin
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index b4b592faaa699b6e62f77ddf57b239c5e15948ff..b7de5049dfd6807de1c84a9454b9097141399660 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -760,6 +760,29 @@ public final class CraftServer implements Server {
|
|
||||||
Validate.notNull(commandLine, "CommandLine cannot be null");
|
|
||||||
org.spigotmc.AsyncCatcher.catchOp("command dispatch"); // Spigot
|
|
||||||
|
|
||||||
+ // Paper Start
|
|
||||||
+ if (!org.spigotmc.AsyncCatcher.shuttingDown && !Bukkit.isPrimaryThread()) {
|
|
||||||
+ final CommandSender fSender = sender;
|
|
||||||
+ final String fCommandLine = commandLine;
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Command Dispatched Async: " + commandLine);
|
|
||||||
+ Bukkit.getLogger().log(Level.SEVERE, "Please notify author of plugin causing this execution to fix this bug! see: http://bit.ly/1oSiM6C", new Throwable());
|
|
||||||
+ org.bukkit.craftbukkit.util.Waitable<Boolean> wait = new org.bukkit.craftbukkit.util.Waitable<Boolean>() {
|
|
||||||
+ @Override
|
|
||||||
+ protected Boolean evaluate() {
|
|
||||||
+ return dispatchCommand(fSender, fCommandLine);
|
|
||||||
+ }
|
|
||||||
+ };
|
|
||||||
+ net.minecraft.server.MinecraftServer.getServer().processQueue.add(wait);
|
|
||||||
+ try {
|
|
||||||
+ return wait.get();
|
|
||||||
+ } catch (InterruptedException e) {
|
|
||||||
+ Thread.currentThread().interrupt(); // This is proper habit for java. If we aren't handling it, pass it on!
|
|
||||||
+ } catch (Exception e) {
|
|
||||||
+ throw new RuntimeException("Exception processing dispatch command", e.getCause());
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper End
|
|
||||||
+
|
|
||||||
if (commandMap.dispatch(sender, commandLine)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java b/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
|
|
||||||
index ddef523ea8762c927f37f7d16d581e43367e8c6b..70f8d42992aa348ef7b2d03d22cdd59d7c73f0fe 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
|
|
||||||
@@ -13,6 +13,7 @@ public class ServerShutdownThread extends Thread {
|
|
||||||
public void run() {
|
|
||||||
try {
|
|
||||||
org.spigotmc.AsyncCatcher.enabled = false; // Spigot
|
|
||||||
+ org.spigotmc.AsyncCatcher.shuttingDown = true; // Paper
|
|
||||||
server.close();
|
|
||||||
} finally {
|
|
||||||
try {
|
|
||||||
diff --git a/src/main/java/org/spigotmc/AsyncCatcher.java b/src/main/java/org/spigotmc/AsyncCatcher.java
|
|
||||||
index aeed7697254af17ffefe8e578353ad216e15f9f3..9f7d2ef932ab41cef5d3d0736d20a7c7e4a2c888 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/AsyncCatcher.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/AsyncCatcher.java
|
|
||||||
@@ -6,6 +6,7 @@ public class AsyncCatcher
|
|
||||||
{
|
|
||||||
|
|
||||||
public static boolean enabled = true;
|
|
||||||
+ public static boolean shuttingDown = false; // Paper
|
|
||||||
|
|
||||||
public static void catchOp(String reason)
|
|
||||||
{
|
|
||||||
diff --git a/src/main/java/org/spigotmc/RestartCommand.java b/src/main/java/org/spigotmc/RestartCommand.java
|
|
||||||
index d76e01d73da7413f192132134caf201d7780e3f1..a45155cdd7d7a302c119f75bfe2b428ae5e8ab47 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/RestartCommand.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/RestartCommand.java
|
|
||||||
@@ -43,6 +43,7 @@ public class RestartCommand extends Command
|
|
||||||
private static void restart(final String restartScript)
|
|
||||||
{
|
|
||||||
AsyncCatcher.enabled = false; // Disable async catcher incase it interferes with us
|
|
||||||
+ org.spigotmc.AsyncCatcher.shuttingDown = true; // Paper
|
|
||||||
try
|
|
||||||
{
|
|
||||||
String[] split = restartScript.split( " " );
|
|
@ -1,33 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: vemacs <d@nkmem.es>
|
|
||||||
Date: Thu, 3 Mar 2016 01:19:22 -0600
|
|
||||||
Subject: [PATCH] All chunks are slime spawn chunks toggle
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 70e074cdf2087e638af8e0f3878d0ef8eb7305cc..416a6760883cb40367535c7c5acd779742bb8af5 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -191,4 +191,9 @@ public class PaperWorldConfig {
|
|
||||||
private void disableChestCatDetection() {
|
|
||||||
disableChestCatDetection = getBoolean("game-mechanics.disable-chest-cat-detection", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean allChunksAreSlimeChunks;
|
|
||||||
+ private void allChunksAreSlimeChunks() {
|
|
||||||
+ allChunksAreSlimeChunks = getBoolean("all-chunks-are-slime-chunks", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/monster/EntitySlime.java b/src/main/java/net/minecraft/world/entity/monster/EntitySlime.java
|
|
||||||
index 292789d76da400d15d0742e2e0979f4ac6ec4b75..01d5b0db9a34d88172e8c7c84c4e1d0b2562217c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/monster/EntitySlime.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/monster/EntitySlime.java
|
|
||||||
@@ -325,7 +325,7 @@ public class EntitySlime extends EntityInsentient implements IMonster {
|
|
||||||
}
|
|
||||||
|
|
||||||
ChunkCoordIntPair chunkcoordintpair = new ChunkCoordIntPair(blockposition);
|
|
||||||
- boolean flag = SeededRandom.a(chunkcoordintpair.x, chunkcoordintpair.z, ((GeneratorAccessSeed) generatoraccess).getSeed(), generatoraccess.getMinecraftWorld().spigotConfig.slimeSeed).nextInt(10) == 0; // Spigot
|
|
||||||
+ boolean flag = generatoraccess.getMinecraftWorld().paperConfig.allChunksAreSlimeChunks || SeededRandom.a(chunkcoordintpair.x, chunkcoordintpair.z, ((GeneratorAccessSeed) generatoraccess).getSeed(), generatoraccess.getMinecraftWorld().spigotConfig.slimeSeed).nextInt(10) == 0; // Spigot // Paper
|
|
||||||
|
|
||||||
if (random.nextInt(10) == 0 && flag && blockposition.getY() < 40) {
|
|
||||||
return a(entitytypes, generatoraccess, enummobspawn, blockposition, random);
|
|
@ -1,18 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: kashike <kashike@vq.lc>
|
|
||||||
Date: Thu, 3 Mar 2016 02:15:57 -0600
|
|
||||||
Subject: [PATCH] Expose server CommandMap
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index b7de5049dfd6807de1c84a9454b9097141399660..f78f5e4f2c04b64dff1d2229a137c600f18e7051 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -1761,6 +1761,7 @@ public final class CraftServer implements Server {
|
|
||||||
return helpMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ @Override // Paper - add override
|
|
||||||
public SimpleCommandMap getCommandMap() {
|
|
||||||
return commandMap;
|
|
||||||
}
|
|
@ -1,22 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: kashike <kashike@vq.lc>
|
|
||||||
Date: Thu, 3 Mar 2016 02:18:39 -0600
|
|
||||||
Subject: [PATCH] Be a bit more informative in maxHealth exception
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
index da0a7a60cbabe0dafa9630b8dcba98e64dcc8d3a..d863fc9fa6b932b76a89871a09378a9c0697c108 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
@@ -111,7 +111,10 @@ public class CraftLivingEntity extends CraftEntity implements LivingEntity {
|
|
||||||
public void setHealth(double health) {
|
|
||||||
health = (float) health;
|
|
||||||
if ((health < 0) || (health > getMaxHealth())) {
|
|
||||||
- throw new IllegalArgumentException("Health must be between 0 and " + getMaxHealth() + "(" + health + ")");
|
|
||||||
+ // Paper - Be more informative
|
|
||||||
+ throw new IllegalArgumentException("Health must be between 0 and " + getMaxHealth() + ", but was " + health
|
|
||||||
+ + ". (attribute base value: " + this.getHandle().getAttributeInstance(GenericAttributes.MAX_HEALTH).getBaseValue()
|
|
||||||
+ + (this instanceof CraftPlayer ? ", player: " + this.getName() + ')' : ')'));
|
|
||||||
}
|
|
||||||
|
|
||||||
getHandle().setHealth((float) health);
|
|
@ -1,142 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Techcable <Techcable@outlook.com>
|
|
||||||
Date: Thu, 3 Mar 2016 02:32:10 -0600
|
|
||||||
Subject: [PATCH] Player Tab List and Title APIs
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/network/PacketDataSerializer.java b/src/main/java/net/minecraft/network/PacketDataSerializer.java
|
|
||||||
index 5f1c5dd7902f6cff5acae05e8c6bf58a1ba5bdf1..df459918c14589155a574730205cb35d463b8079 100644
|
|
||||||
--- a/src/main/java/net/minecraft/network/PacketDataSerializer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/network/PacketDataSerializer.java
|
|
||||||
@@ -171,6 +171,11 @@ public class PacketDataSerializer extends ByteBuf {
|
|
||||||
public PacketDataSerializer writeComponent(final net.kyori.adventure.text.Component component) {
|
|
||||||
return this.writeUtf(PaperAdventure.asJsonString(component, this.adventure$locale), 262144);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ @Deprecated
|
|
||||||
+ public PacketDataSerializer writeComponent(final net.md_5.bungee.api.chat.BaseComponent[] component) {
|
|
||||||
+ return this.writeUtf(net.md_5.bungee.chat.ComponentSerializer.toString(component), 262144);
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
public PacketDataSerializer a(IChatBaseComponent ichatbasecomponent) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/network/chat/IChatBaseComponent.java b/src/main/java/net/minecraft/network/chat/IChatBaseComponent.java
|
|
||||||
index e96fa348a37a39c381b6659f612232933686c2a7..a002125e454f8a86924e9010e0b20a95742fa04b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/network/chat/IChatBaseComponent.java
|
|
||||||
+++ b/src/main/java/net/minecraft/network/chat/IChatBaseComponent.java
|
|
||||||
@@ -364,6 +364,7 @@ public interface IChatBaseComponent extends Message, IChatFormatted, Iterable<IC
|
|
||||||
return IChatBaseComponent.ChatSerializer.a.toJsonTree(ichatbasecomponent);
|
|
||||||
}
|
|
||||||
|
|
||||||
+ @Nullable public static IChatBaseComponent jsonToComponent(String json) { return a(json);} // Paper - OBFHELPER
|
|
||||||
@Nullable
|
|
||||||
public static IChatMutableComponent a(String s) {
|
|
||||||
return (IChatMutableComponent) ChatDeserializer.a(IChatBaseComponent.ChatSerializer.a, s, IChatMutableComponent.class, false);
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index fb792de46ff80a6bad77a47954861cddfd17f2d9..1ad5863dc12b2288a38efed71b7fa4b84296d96d 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -1,5 +1,6 @@
|
|
||||||
package org.bukkit.craftbukkit.entity;
|
|
||||||
|
|
||||||
+import com.destroystokyo.paper.Title;
|
|
||||||
import com.google.common.base.Preconditions;
|
|
||||||
import com.google.common.collect.ImmutableSet;
|
|
||||||
import com.google.common.io.BaseEncoding;
|
|
||||||
@@ -238,6 +239,96 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ @Override
|
|
||||||
+ public void setPlayerListHeaderFooter(BaseComponent[] header, BaseComponent[] footer) {
|
|
||||||
+ if (header != null) {
|
|
||||||
+ String headerJson = net.md_5.bungee.chat.ComponentSerializer.toString(header);
|
|
||||||
+ playerListHeader = net.kyori.adventure.text.serializer.gson.GsonComponentSerializer.gson().deserialize(headerJson);
|
|
||||||
+ } else {
|
|
||||||
+ playerListHeader = null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (footer != null) {
|
|
||||||
+ String footerJson = net.md_5.bungee.chat.ComponentSerializer.toString(footer);
|
|
||||||
+ playerListFooter = net.kyori.adventure.text.serializer.gson.GsonComponentSerializer.gson().deserialize(footerJson);
|
|
||||||
+ } else {
|
|
||||||
+ playerListFooter = null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ updatePlayerListHeaderFooter();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setPlayerListHeaderFooter(BaseComponent header, BaseComponent footer) {
|
|
||||||
+ this.setPlayerListHeaderFooter(header == null ? null : new BaseComponent[]{header},
|
|
||||||
+ footer == null ? null : new BaseComponent[]{footer});
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setTitleTimes(int fadeInTicks, int stayTicks, int fadeOutTicks) {
|
|
||||||
+ getHandle().playerConnection.sendPacket(new PacketPlayOutTitle(PacketPlayOutTitle.EnumTitleAction.TIMES, (BaseComponent[]) null, fadeInTicks, stayTicks, fadeOutTicks));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setSubtitle(BaseComponent[] subtitle) {
|
|
||||||
+ getHandle().playerConnection.sendPacket(new PacketPlayOutTitle(PacketPlayOutTitle.EnumTitleAction.SUBTITLE, subtitle, 0, 0, 0));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setSubtitle(BaseComponent subtitle) {
|
|
||||||
+ setSubtitle(new BaseComponent[]{subtitle});
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void showTitle(BaseComponent[] title) {
|
|
||||||
+ getHandle().playerConnection.sendPacket(new PacketPlayOutTitle(PacketPlayOutTitle.EnumTitleAction.TITLE, title, 0, 0, 0));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void showTitle(BaseComponent title) {
|
|
||||||
+ showTitle(new BaseComponent[]{title});
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void showTitle(BaseComponent[] title, BaseComponent[] subtitle, int fadeInTicks, int stayTicks, int fadeOutTicks) {
|
|
||||||
+ setTitleTimes(fadeInTicks, stayTicks, fadeOutTicks);
|
|
||||||
+ setSubtitle(subtitle);
|
|
||||||
+ showTitle(title);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void showTitle(BaseComponent title, BaseComponent subtitle, int fadeInTicks, int stayTicks, int fadeOutTicks) {
|
|
||||||
+ setTitleTimes(fadeInTicks, stayTicks, fadeOutTicks);
|
|
||||||
+ setSubtitle(subtitle);
|
|
||||||
+ showTitle(title);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void sendTitle(Title title) {
|
|
||||||
+ Preconditions.checkNotNull(title, "Title is null");
|
|
||||||
+ setTitleTimes(title.getFadeIn(), title.getStay(), title.getFadeOut());
|
|
||||||
+ setSubtitle(title.getSubtitle() == null ? new BaseComponent[0] : title.getSubtitle());
|
|
||||||
+ showTitle(title.getTitle());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void updateTitle(Title title) {
|
|
||||||
+ Preconditions.checkNotNull(title, "Title is null");
|
|
||||||
+ setTitleTimes(title.getFadeIn(), title.getStay(), title.getFadeOut());
|
|
||||||
+ if (title.getSubtitle() != null) {
|
|
||||||
+ setSubtitle(title.getSubtitle());
|
|
||||||
+ }
|
|
||||||
+ showTitle(title.getTitle());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void hideTitle() {
|
|
||||||
+ getHandle().playerConnection.sendPacket(new PacketPlayOutTitle(PacketPlayOutTitle.EnumTitleAction.CLEAR, (BaseComponent[]) null, 0, 0, 0));
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
@Override
|
|
||||||
public String getDisplayName() {
|
|
||||||
if(true) return io.papermc.paper.adventure.DisplayNames.getLegacy(this); // Paper
|
|
@ -1,19 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Joseph Hirschfeld <joe@ibj.io>
|
|
||||||
Date: Thu, 3 Mar 2016 02:33:53 -0600
|
|
||||||
Subject: [PATCH] Ensure inv drag is in bounds
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/inventory/Container.java b/src/main/java/net/minecraft/world/inventory/Container.java
|
|
||||||
index 2410214a300407ef20ea14244db5db2ebede2759..e9733fd9dac89d31dbad391cb22a8c84216045db 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/inventory/Container.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/inventory/Container.java
|
|
||||||
@@ -239,7 +239,7 @@ public abstract class Container {
|
|
||||||
this.d();
|
|
||||||
}
|
|
||||||
} else if (this.h == 1) {
|
|
||||||
- Slot slot = (Slot) this.slots.get(i);
|
|
||||||
+ Slot slot = i < this.slots.size() ? this.slots.get(i) : null; // Paper - Ensure drag in bounds
|
|
||||||
|
|
||||||
itemstack1 = playerinventory.getCarried();
|
|
||||||
if (slot != null && a(slot, itemstack1, true) && slot.isAllowed(itemstack1) && (this.dragType == 2 || itemstack1.getCount() > this.i.size()) && this.b(slot)) {
|
|
@ -1,20 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Joseph Hirschfeld <joe@ibj.io>
|
|
||||||
Date: Thu, 3 Mar 2016 02:39:54 -0600
|
|
||||||
Subject: [PATCH] Change implementation of (tile)entity removal list
|
|
||||||
|
|
||||||
use sets for faster removal
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 26d9388badb0328735abf0319a42bd633c90dc4b..33469b719d679c65d4bcb8366008e6e107eb3a0b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -89,7 +89,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
public final List<TileEntity> tileEntityList = Lists.newArrayList();
|
|
||||||
public final List<TileEntity> tileEntityListTick = Lists.newArrayList();
|
|
||||||
protected final List<TileEntity> tileEntityListPending = Lists.newArrayList();
|
|
||||||
- protected final List<TileEntity> tileEntityListUnload = Lists.newArrayList();
|
|
||||||
+ protected final java.util.Set<TileEntity> tileEntityListUnload = com.google.common.collect.Sets.newHashSet();
|
|
||||||
public final Thread serverThread;
|
|
||||||
private final boolean debugWorld;
|
|
||||||
private int d;
|
|
@ -1,56 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Joseph Hirschfeld <joe@ibj.io>
|
|
||||||
Date: Thu, 3 Mar 2016 02:46:17 -0600
|
|
||||||
Subject: [PATCH] Add configurable portal search radius
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 416a6760883cb40367535c7c5acd779742bb8af5..670efbe53241a0ae32d618c83da601ccc1f26e37 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -196,4 +196,13 @@ public class PaperWorldConfig {
|
|
||||||
private void allChunksAreSlimeChunks() {
|
|
||||||
allChunksAreSlimeChunks = getBoolean("all-chunks-are-slime-chunks", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int portalSearchRadius;
|
|
||||||
+ public int portalCreateRadius;
|
|
||||||
+ public boolean portalSearchVanillaDimensionScaling;
|
|
||||||
+ private void portalSearchRadius() {
|
|
||||||
+ portalSearchRadius = getInt("portal-search-radius", 128);
|
|
||||||
+ portalCreateRadius = getInt("portal-create-radius", 16);
|
|
||||||
+ portalSearchVanillaDimensionScaling = getBoolean("portal-search-vanilla-dimension-scaling", true);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index 77cc0aeb979369df2156f8fb916067f608b61ebf..caae8bdae592a1ae4f99861088458d7461f4c97a 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -2618,7 +2618,13 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
double d4 = DimensionManager.a(this.world.getDimensionManager(), worldserver.getDimensionManager());
|
|
||||||
BlockPosition blockposition = new BlockPosition(MathHelper.a(this.locX() * d4, d0, d2), this.locY(), MathHelper.a(this.locZ() * d4, d1, d3));
|
|
||||||
// CraftBukkit start
|
|
||||||
- CraftPortalEvent event = callPortalEvent(this, worldserver, blockposition, PlayerTeleportEvent.TeleportCause.NETHER_PORTAL, flag2 ? 16 : 128, 16);
|
|
||||||
+ // Paper start
|
|
||||||
+ int portalSearchRadius = worldserver.paperConfig.portalSearchRadius;
|
|
||||||
+ if (world.paperConfig.portalSearchVanillaDimensionScaling && flag2) { // == THE_NETHER
|
|
||||||
+ portalSearchRadius = (int) (portalSearchRadius / worldserver.getDimensionManager().getCoordinateScale());
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+ CraftPortalEvent event = callPortalEvent(this, worldserver, blockposition, PlayerTeleportEvent.TeleportCause.NETHER_PORTAL, portalSearchRadius, worldserver.paperConfig.portalCreateRadius); // Paper start - configurable portal radius
|
|
||||||
if (event == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/portal/PortalTravelAgent.java b/src/main/java/net/minecraft/world/level/portal/PortalTravelAgent.java
|
|
||||||
index 7b8e8a7dae47ecc42a57e3f9444caa2ee5b1ef3b..77dfa7eaf178baa55041a829c9dec4851efeedfc 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/portal/PortalTravelAgent.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/portal/PortalTravelAgent.java
|
|
||||||
@@ -31,7 +31,7 @@ public class PortalTravelAgent {
|
|
||||||
|
|
||||||
public Optional<BlockUtil.Rectangle> findPortal(BlockPosition blockposition, boolean flag) {
|
|
||||||
// CraftBukkit start
|
|
||||||
- return findPortal(blockposition, flag ? 16 : 128); // Search Radius
|
|
||||||
+ return findPortal(blockposition, flag ? world.paperConfig.portalCreateRadius : world.paperConfig.portalSearchRadius); // Paper - search Radius
|
|
||||||
}
|
|
||||||
|
|
||||||
public Optional<BlockUtil.Rectangle> findPortal(BlockPosition blockposition, int i) {
|
|
@ -1,89 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Joseph Hirschfeld <joe@ibj.io>
|
|
||||||
Date: Thu, 3 Mar 2016 02:48:12 -0600
|
|
||||||
Subject: [PATCH] Add velocity warnings
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index f78f5e4f2c04b64dff1d2229a137c600f18e7051..22b4dec4944b7f823996645af95fbef2d1d8a83b 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -261,6 +261,7 @@ public final class CraftServer implements Server {
|
|
||||||
public boolean ignoreVanillaPermissions = false;
|
|
||||||
private final List<CraftPlayer> playerView;
|
|
||||||
public int reloadCount;
|
|
||||||
+ public static Exception excessiveVelEx; // Paper - Velocity warnings
|
|
||||||
|
|
||||||
static {
|
|
||||||
ConfigurationSerialization.registerClass(CraftOfflinePlayer.class);
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
index 220bad90bbb9a90c3f23562bf0fb109fce379682..a58626b1a0160983a738a45c8a1d411eb347e6a2 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
||||||
@@ -424,10 +424,41 @@ public abstract class CraftEntity implements org.bukkit.entity.Entity {
|
|
||||||
public void setVelocity(Vector velocity) {
|
|
||||||
Preconditions.checkArgument(velocity != null, "velocity");
|
|
||||||
velocity.checkFinite();
|
|
||||||
+ // Paper start - Warn server owners when plugins try to set super high velocities
|
|
||||||
+ if (!(this instanceof org.bukkit.entity.Projectile) && isUnsafeVelocity(velocity)) {
|
|
||||||
+ CraftServer.excessiveVelEx = new Exception("Excessive velocity set detected: tried to set velocity of entity " + entity.getName() + " id #" + getEntityId() + " to (" + velocity.getX() + "," + velocity.getY() + "," + velocity.getZ() + ").");
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
entity.setMot(CraftVector.toNMS(velocity));
|
|
||||||
entity.velocityChanged = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ /**
|
|
||||||
+ * Checks if the given velocity is not necessarily safe in all situations.
|
|
||||||
+ * This function returning true does not mean the velocity is dangerous or to be avoided, only that it may be
|
|
||||||
+ * a detriment to performance on the server.
|
|
||||||
+ *
|
|
||||||
+ * It is not to be used as a hard rule of any sort.
|
|
||||||
+ * Paper only uses it to warn server owners in watchdog crashes.
|
|
||||||
+ *
|
|
||||||
+ * @param vel incoming velocity to check
|
|
||||||
+ * @return if the velocity has the potential to be a performance detriment
|
|
||||||
+ */
|
|
||||||
+ private static boolean isUnsafeVelocity(Vector vel) {
|
|
||||||
+ final double x = vel.getX();
|
|
||||||
+ final double y = vel.getY();
|
|
||||||
+ final double z = vel.getZ();
|
|
||||||
+
|
|
||||||
+ if (x > 4 || x < -4 || y > 4 || y < -4 || z > 4 || z < -4) {
|
|
||||||
+ return true;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
@Override
|
|
||||||
public double getHeight() {
|
|
||||||
return getHandle().getHeight();
|
|
||||||
diff --git a/src/main/java/org/spigotmc/WatchdogThread.java b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
index 1b3a14784cac8e855633fae6172ad5479ebe9877..69e5054886b5858664fed333aca8c25a76e5cb11 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
@@ -80,7 +80,19 @@ public class WatchdogThread extends Thread
|
|
||||||
log.log( Level.SEVERE, "During the run of the server, a physics stackoverflow was supressed" );
|
|
||||||
log.log( Level.SEVERE, "near " + net.minecraft.world.level.World.lastPhysicsProblem );
|
|
||||||
}
|
|
||||||
- //
|
|
||||||
+ // Paper start - Warn in watchdog if an excessive velocity was ever set
|
|
||||||
+ if ( org.bukkit.craftbukkit.CraftServer.excessiveVelEx != null )
|
|
||||||
+ {
|
|
||||||
+ log.log( Level.SEVERE, "------------------------------" );
|
|
||||||
+ log.log( Level.SEVERE, "During the run of the server, a plugin set an excessive velocity on an entity" );
|
|
||||||
+ log.log( Level.SEVERE, "This may be the cause of the issue, or it may be entirely unrelated" );
|
|
||||||
+ log.log( Level.SEVERE, org.bukkit.craftbukkit.CraftServer.excessiveVelEx.getMessage());
|
|
||||||
+ for ( StackTraceElement stack : org.bukkit.craftbukkit.CraftServer.excessiveVelEx.getStackTrace() )
|
|
||||||
+ {
|
|
||||||
+ log.log( Level.SEVERE, "\t\t" + stack );
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
log.log( Level.SEVERE, "------------------------------" );
|
|
||||||
log.log( Level.SEVERE, "Server thread dump (Look for plugins here before reporting to Paper!):" ); // Paper
|
|
||||||
dumpThread( ManagementFactory.getThreadMXBean().getThreadInfo( MinecraftServer.getServer().serverThread.getId(), Integer.MAX_VALUE ), log );
|
|
@ -1,44 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Sudzzy <originmc@outlook.com>
|
|
||||||
Date: Thu, 3 Mar 2016 02:50:31 -0600
|
|
||||||
Subject: [PATCH] Configurable inter-world teleportation safety
|
|
||||||
|
|
||||||
People are able to abuse the way Bukkit handles teleportation across worlds since it provides a built in teleportation
|
|
||||||
safety check.
|
|
||||||
|
|
||||||
To abuse the safety check, players are required to get into a location deemed unsafe by Bukkit e.g. be within a chest
|
|
||||||
or door block. While they are in this block, they accept a teleport request from a player within a different world. Once
|
|
||||||
the player teleports, Minecraft will recursively search upwards for a safe location, this could eventually land within a
|
|
||||||
player's skybase.
|
|
||||||
|
|
||||||
Example setup to perform the glitch: http://puu.sh/ng3PC/cf072dcbdb.png
|
|
||||||
The wanted destination was on top of the emerald block however the player ended on top of the diamond block.
|
|
||||||
This only is the case if the player is teleporting between worlds.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 670efbe53241a0ae32d618c83da601ccc1f26e37..abbbe1786eb68af02f9d39650aad730ac44aac8a 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -205,4 +205,9 @@ public class PaperWorldConfig {
|
|
||||||
portalCreateRadius = getInt("portal-create-radius", 16);
|
|
||||||
portalSearchVanillaDimensionScaling = getBoolean("portal-search-vanilla-dimension-scaling", true);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean disableTeleportationSuffocationCheck;
|
|
||||||
+ private void disableTeleportationSuffocationCheck() {
|
|
||||||
+ disableTeleportationSuffocationCheck = getBoolean("disable-teleportation-suffocation-check", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index 1ad5863dc12b2288a38efed71b7fa4b84296d96d..f2228933719a2325a518be15237fedf56c994d1f 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -863,7 +863,7 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
if (fromWorld == toWorld) {
|
|
||||||
entity.playerConnection.teleport(to);
|
|
||||||
} else {
|
|
||||||
- server.getHandle().moveToWorld(entity, toWorld, true, to, true);
|
|
||||||
+ server.getHandle().moveToWorld(entity, toWorld, true, to, !toWorld.paperConfig.disableTeleportationSuffocationCheck); // Paper
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
@ -1,264 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Joseph Hirschfeld <joe@ibj.io>
|
|
||||||
Date: Thu, 3 Mar 2016 03:15:41 -0600
|
|
||||||
Subject: [PATCH] Add exception reporting event
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/ServerSchedulerReportingWrapper.java b/src/main/java/com/destroystokyo/paper/ServerSchedulerReportingWrapper.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..f699ce18ca044f813e194ef2786b7ea853ea86e7
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/ServerSchedulerReportingWrapper.java
|
|
||||||
@@ -0,0 +1,38 @@
|
|
||||||
+package com.destroystokyo.paper;
|
|
||||||
+
|
|
||||||
+import com.google.common.base.Preconditions;
|
|
||||||
+import org.bukkit.craftbukkit.scheduler.CraftTask;
|
|
||||||
+import com.destroystokyo.paper.event.server.ServerExceptionEvent;
|
|
||||||
+import com.destroystokyo.paper.exception.ServerSchedulerException;
|
|
||||||
+
|
|
||||||
+/**
|
|
||||||
+ * Reporting wrapper to catch exceptions not natively
|
|
||||||
+ */
|
|
||||||
+public class ServerSchedulerReportingWrapper implements Runnable {
|
|
||||||
+
|
|
||||||
+ private final CraftTask internalTask;
|
|
||||||
+
|
|
||||||
+ public ServerSchedulerReportingWrapper(CraftTask internalTask) {
|
|
||||||
+ this.internalTask = Preconditions.checkNotNull(internalTask, "internalTask");
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void run() {
|
|
||||||
+ try {
|
|
||||||
+ internalTask.run();
|
|
||||||
+ } catch (RuntimeException e) {
|
|
||||||
+ internalTask.getOwner().getServer().getPluginManager().callEvent(
|
|
||||||
+ new ServerExceptionEvent(new ServerSchedulerException(e, internalTask))
|
|
||||||
+ );
|
|
||||||
+ throw e;
|
|
||||||
+ } catch (Throwable t) {
|
|
||||||
+ internalTask.getOwner().getServer().getPluginManager().callEvent(
|
|
||||||
+ new ServerExceptionEvent(new ServerSchedulerException(t, internalTask))
|
|
||||||
+ ); //Do not rethrow, since it is not permitted with Runnable#run
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public CraftTask getInternalTask() {
|
|
||||||
+ return internalTask;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/PlayerChunkMap.java b/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
index 788a45d5426f0752509442aec2d28b1f32f63cb1..2511fbe7aa5ff1ace71b513d2938975e388295c6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/PlayerChunkMap.java
|
|
||||||
@@ -815,6 +815,7 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
|
||||||
return true;
|
|
||||||
} catch (Exception exception) {
|
|
||||||
PlayerChunkMap.LOGGER.error("Failed to save chunk {},{}", chunkcoordintpair.x, chunkcoordintpair.z, exception);
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/players/NameReferencingFileConverter.java b/src/main/java/net/minecraft/server/players/NameReferencingFileConverter.java
|
|
||||||
index 107979178e8be5ee6cf885d42f992fabf3bd00b0..8a343a857dc4661ba256e39cf391dd2c7a1cc970 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/players/NameReferencingFileConverter.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/players/NameReferencingFileConverter.java
|
|
||||||
@@ -1,5 +1,6 @@
|
|
||||||
package net.minecraft.server.players;
|
|
||||||
|
|
||||||
+import com.destroystokyo.paper.exception.ServerInternalException;
|
|
||||||
import com.google.common.collect.Lists;
|
|
||||||
import com.google.common.collect.Maps;
|
|
||||||
import com.google.common.io.Files;
|
|
||||||
@@ -363,6 +364,7 @@ public class NameReferencingFileConverter {
|
|
||||||
root = NBTCompressedStreamTools.a(new java.io.FileInputStream(file5));
|
|
||||||
} catch (Exception exception) {
|
|
||||||
exception.printStackTrace();
|
|
||||||
+ ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
if (root != null) {
|
|
||||||
@@ -376,6 +378,7 @@ public class NameReferencingFileConverter {
|
|
||||||
NBTCompressedStreamTools.a(root, new java.io.FileOutputStream(file2));
|
|
||||||
} catch (Exception exception) {
|
|
||||||
exception.printStackTrace();
|
|
||||||
+ ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/village/VillageSiege.java b/src/main/java/net/minecraft/world/entity/ai/village/VillageSiege.java
|
|
||||||
index 928ca3189af1ddaba797628a087cd6c6a9016f5c..eaa97eb11d893266253fb108249ced1e0e96a4dc 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/village/VillageSiege.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/village/VillageSiege.java
|
|
||||||
@@ -1,5 +1,7 @@
|
|
||||||
package net.minecraft.world.entity.ai.village;
|
|
||||||
|
|
||||||
+import com.destroystokyo.paper.exception.ServerInternalException;
|
|
||||||
+
|
|
||||||
import java.util.Iterator;
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
import net.minecraft.core.BaseBlockPosition;
|
|
||||||
@@ -119,6 +121,7 @@ public class VillageSiege implements MobSpawner {
|
|
||||||
entityzombie.prepare(worldserver, worldserver.getDamageScaler(entityzombie.getChunkCoordinates()), EnumMobSpawn.EVENT, (GroupDataEntity) null, (NBTTagCompound) null);
|
|
||||||
} catch (Exception exception) {
|
|
||||||
VillageSiege.LOGGER.warn("Failed to create zombie for village siege at {}", vec3d, exception);
|
|
||||||
+ ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/SpawnerCreature.java b/src/main/java/net/minecraft/world/level/SpawnerCreature.java
|
|
||||||
index 2d02b2fc502a0f7e541f7943ed647ff7177acee8..fd0595fd584046326eccacdf0a6afe40c5e84eed 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/SpawnerCreature.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/SpawnerCreature.java
|
|
||||||
@@ -301,6 +301,7 @@ public final class SpawnerCreature {
|
|
||||||
}
|
|
||||||
} catch (Exception exception) {
|
|
||||||
SpawnerCreature.LOGGER.warn("Failed to create mob", exception);
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -407,6 +408,7 @@ public final class SpawnerCreature {
|
|
||||||
entity = biomesettingsmobs_c.c.a((World) worldaccess.getMinecraftWorld());
|
|
||||||
} catch (Exception exception) {
|
|
||||||
SpawnerCreature.LOGGER.warn("Failed to create mob", exception);
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(exception); // Paper
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 33469b719d679c65d4bcb8366008e6e107eb3a0b..01cb0c8dd9875986e0c08371e876f0dba3f0cf5a 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -1,5 +1,10 @@
|
|
||||||
package net.minecraft.world.level;
|
|
||||||
|
|
||||||
+import co.aikar.timings.Timing;
|
|
||||||
+import co.aikar.timings.Timings;
|
|
||||||
+import com.destroystokyo.paper.event.server.ServerExceptionEvent;
|
|
||||||
+import com.destroystokyo.paper.exception.ServerInternalException;
|
|
||||||
+import com.google.common.base.MoreObjects;
|
|
||||||
import com.google.common.collect.Lists;
|
|
||||||
import com.mojang.serialization.Codec;
|
|
||||||
import java.io.IOException;
|
|
||||||
@@ -737,8 +742,11 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
gameprofilerfiller.exit();
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
// Paper start - Prevent tile entity and entity crashes
|
|
||||||
- System.err.println("TileEntity threw exception at " + tileentity.world.getWorld().getName() + ":" + tileentity.position.getX() + "," + tileentity.position.getY() + "," + tileentity.position.getZ());
|
|
||||||
+ String msg = "TileEntity threw exception at " + tileentity.getWorld().getWorld().getName() + ":" + tileentity.getPosition().getX() + "," + tileentity.getPosition().getY() + "," + tileentity.getPosition().getZ();
|
|
||||||
+ System.err.println(msg);
|
|
||||||
throwable.printStackTrace();
|
|
||||||
+ getServer().getPluginManager().callEvent(new ServerExceptionEvent(new ServerInternalException(msg, throwable)));
|
|
||||||
+ // Paper end
|
|
||||||
tilesThisCycle--;
|
|
||||||
this.tileEntityListTick.remove(tileTickPosition--);
|
|
||||||
continue;
|
|
||||||
@@ -808,8 +816,10 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
consumer.accept(entity);
|
|
||||||
} catch (Throwable throwable) {
|
|
||||||
// Paper start - Prevent tile entity and entity crashes
|
|
||||||
- System.err.println("Entity threw exception at " + entity.world.getWorld().getName() + ":" + entity.locX() + "," + entity.locY() + "," + entity.locZ());
|
|
||||||
+ String msg = "Entity threw exception at " + entity.world.getWorld().getName() + ":" + entity.locX() + "," + entity.locY() + "," + entity.locZ();
|
|
||||||
+ System.err.println(msg);
|
|
||||||
throwable.printStackTrace();
|
|
||||||
+ getServer().getPluginManager().callEvent(new ServerExceptionEvent(new ServerInternalException(msg, throwable)));
|
|
||||||
entity.dead = true;
|
|
||||||
return;
|
|
||||||
// Paper end
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index a2d80c2c8e4f080f60746548f75631c5946ba8e2..4b3de29b1a6e9d75b28962073c62bbe8d666165f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -1,5 +1,6 @@
|
|
||||||
package net.minecraft.world.level.chunk;
|
|
||||||
|
|
||||||
+import com.destroystokyo.paper.exception.ServerInternalException;
|
|
||||||
import com.google.common.collect.Maps;
|
|
||||||
import com.google.common.collect.Sets;
|
|
||||||
import it.unimi.dsi.fastutil.longs.LongOpenHashSet;
|
|
||||||
@@ -655,10 +656,15 @@ public class Chunk implements IChunkAccess {
|
|
||||||
this.tileEntities.remove(blockposition);
|
|
||||||
// Paper end
|
|
||||||
} else {
|
|
||||||
- System.out.println("Attempted to place a tile entity (" + tileentity + ") at " + tileentity.getPosition().getX() + "," + tileentity.getPosition().getY() + "," + tileentity.getPosition().getZ()
|
|
||||||
- + " (" + getType(blockposition) + ") where there was no entity tile!");
|
|
||||||
- System.out.println("Chunk coordinates: " + (this.loc.x * 16) + "," + (this.loc.z * 16));
|
|
||||||
- new Exception().printStackTrace();
|
|
||||||
+ // Paper start
|
|
||||||
+ ServerInternalException e = new ServerInternalException(
|
|
||||||
+ "Attempted to place a tile entity (" + tileentity + ") at " + tileentity.getPosition().getX() + ","
|
|
||||||
+ + tileentity.getPosition().getY() + "," + tileentity.getPosition().getZ()
|
|
||||||
+ + " (" + getType(blockposition) + ") where there was no entity tile!\n" +
|
|
||||||
+ "Chunk coordinates: " + (this.loc.x * 16) + "," + (this.loc.z * 16));
|
|
||||||
+ e.printStackTrace();
|
|
||||||
+ ServerInternalException.reportInternalException(e);
|
|
||||||
+ // Paper end
|
|
||||||
// CraftBukkit end
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
index d1b761055c508a4b80436b50a832e00d0449d8cb..1638f7902290e1bb233f11e5d0bbf83a9e863939 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
@@ -265,6 +265,7 @@ public class RegionFile implements AutoCloseable {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} catch (IOException ioexception) {
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(ioexception); // Paper
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -337,6 +338,7 @@ public class RegionFile implements AutoCloseable {
|
|
||||||
filechannel.write(bytebuffer);
|
|
||||||
} catch (Throwable throwable1) {
|
|
||||||
throwable = throwable1;
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(throwable); // Paper
|
|
||||||
throw throwable1;
|
|
||||||
} finally {
|
|
||||||
if (filechannel != null) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/storage/WorldPersistentData.java b/src/main/java/net/minecraft/world/level/storage/WorldPersistentData.java
|
|
||||||
index 3910daeaa177639fa8055301304634c2014dc20f..d61960d80599dc5e7b70cc990e4b0b174eb6e34e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/storage/WorldPersistentData.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/storage/WorldPersistentData.java
|
|
||||||
@@ -150,6 +150,7 @@ public class WorldPersistentData {
|
|
||||||
}
|
|
||||||
} catch (Throwable throwable6) {
|
|
||||||
throwable = throwable6;
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(throwable); // Paper
|
|
||||||
throw throwable6;
|
|
||||||
} finally {
|
|
||||||
if (fileinputstream != null) {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/scheduler/CraftScheduler.java b/src/main/java/org/bukkit/craftbukkit/scheduler/CraftScheduler.java
|
|
||||||
index ffe9cc1011226d604dc5499e7692e9a9a5132b72..9b6d9373abb59a30c2835ca891282d07559281f5 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/scheduler/CraftScheduler.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/scheduler/CraftScheduler.java
|
|
||||||
@@ -16,6 +16,9 @@ import java.util.concurrent.atomic.AtomicInteger;
|
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
|
||||||
import java.util.function.Consumer;
|
|
||||||
import java.util.logging.Level;
|
|
||||||
+import com.destroystokyo.paper.ServerSchedulerReportingWrapper;
|
|
||||||
+import com.destroystokyo.paper.event.server.ServerExceptionEvent;
|
|
||||||
+import com.destroystokyo.paper.exception.ServerSchedulerException;
|
|
||||||
import org.apache.commons.lang.Validate;
|
|
||||||
import org.bukkit.plugin.IllegalPluginAccessException;
|
|
||||||
import org.bukkit.plugin.Plugin;
|
|
||||||
@@ -419,6 +422,8 @@ public class CraftScheduler implements BukkitScheduler {
|
|
||||||
msg,
|
|
||||||
throwable);
|
|
||||||
}
|
|
||||||
+ org.bukkit.Bukkit.getServer().getPluginManager().callEvent(
|
|
||||||
+ new ServerExceptionEvent(new ServerSchedulerException(msg, throwable, task)));
|
|
||||||
// Paper end
|
|
||||||
} finally {
|
|
||||||
currentTask = null;
|
|
||||||
@@ -426,7 +431,7 @@ public class CraftScheduler implements BukkitScheduler {
|
|
||||||
parsePending();
|
|
||||||
} else {
|
|
||||||
debugTail = debugTail.setNext(new CraftAsyncDebugger(currentTick + RECENT_TICKS, task.getOwner(), task.getTaskClass()));
|
|
||||||
- executor.execute(task);
|
|
||||||
+ executor.execute(new ServerSchedulerReportingWrapper(task)); // Paper
|
|
||||||
// We don't need to parse pending
|
|
||||||
// (async tasks must live with race-conditions if they attempt to cancel between these few lines of code)
|
|
||||||
}
|
|
@ -1,27 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: kashike <kashike@vq.lc>
|
|
||||||
Date: Tue, 8 Mar 2016 18:28:43 -0800
|
|
||||||
Subject: [PATCH] Don't nest if we don't need to when cerealising text
|
|
||||||
components
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutChat.java b/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutChat.java
|
|
||||||
index edae451a54bfcd6b54e89c1619fb112a7763eb3b..f6a1c5ac9acb34b1ef2262721adbbb1a5b0feaf7 100644
|
|
||||||
--- a/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutChat.java
|
|
||||||
+++ b/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutChat.java
|
|
||||||
@@ -40,7 +40,14 @@ public class PacketPlayOutChat implements Packet<PacketListenerPlayOut> {
|
|
||||||
// Paper end
|
|
||||||
// Spigot start
|
|
||||||
if (components != null) {
|
|
||||||
- packetdataserializer.a(net.md_5.bungee.chat.ComponentSerializer.toString(components));
|
|
||||||
+ //packetdataserializer.a(net.md_5.bungee.chat.ComponentSerializer.toString(components)); // Paper - comment, replaced with below
|
|
||||||
+ // Paper start - don't nest if we don't need to so that we can preserve formatting
|
|
||||||
+ if (this.components.length == 1) {
|
|
||||||
+ packetdataserializer.a(net.md_5.bungee.chat.ComponentSerializer.toString(this.components[0]));
|
|
||||||
+ } else {
|
|
||||||
+ packetdataserializer.a(net.md_5.bungee.chat.ComponentSerializer.toString(this.components));
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
} else {
|
|
||||||
packetdataserializer.a(this.a);
|
|
||||||
}
|
|
@ -1,50 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Tue, 8 Mar 2016 23:25:45 -0500
|
|
||||||
Subject: [PATCH] Disable Scoreboards for non players by default
|
|
||||||
|
|
||||||
Entities collision is checking for scoreboards setting.
|
|
||||||
This is very heavy to do map lookups for every collision to check
|
|
||||||
this setting.
|
|
||||||
|
|
||||||
So avoid looking up scoreboards and short circuit to the "not on a team"
|
|
||||||
logic which is most likely to be true.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index abbbe1786eb68af02f9d39650aad730ac44aac8a..3ac2ac3db9b1c271b3c21930bb13716669ff64d3 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -210,4 +210,9 @@ public class PaperWorldConfig {
|
|
||||||
private void disableTeleportationSuffocationCheck() {
|
|
||||||
disableTeleportationSuffocationCheck = getBoolean("disable-teleportation-suffocation-check", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean nonPlayerEntitiesOnScoreboards = false;
|
|
||||||
+ private void nonPlayerEntitiesOnScoreboards() {
|
|
||||||
+ nonPlayerEntitiesOnScoreboards = getBoolean("allow-non-player-entities-on-scoreboards", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index caae8bdae592a1ae4f99861088458d7461f4c97a..bfced192c1e8fd3fa0250a0f93adfc061d7e71e5 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -2289,6 +2289,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public ScoreboardTeamBase getScoreboardTeam() {
|
|
||||||
+ if (!this.world.paperConfig.nonPlayerEntitiesOnScoreboards && !(this instanceof EntityHuman)) { return null; } // Paper
|
|
||||||
return this.world.getScoreboard().getPlayerTeam(this.getName());
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityLiving.java b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
index 49e95369882847c90ee7417abea6270386cd622f..70211129e6ab2f7cdb975adcb532be595bc3834f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
@@ -741,6 +741,7 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
if (nbttagcompound.hasKeyOfType("Team", 8)) {
|
|
||||||
String s = nbttagcompound.getString("Team");
|
|
||||||
ScoreboardTeam scoreboardteam = this.world.getScoreboard().getTeam(s);
|
|
||||||
+ if (!world.paperConfig.nonPlayerEntitiesOnScoreboards && !(this instanceof EntityHuman)) { scoreboardteam = null; } // Paper
|
|
||||||
boolean flag = scoreboardteam != null && this.world.getScoreboard().addPlayerToTeam(this.getUniqueIDString(), scoreboardteam);
|
|
||||||
|
|
||||||
if (!flag) {
|
|
@ -1,27 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: mrapple <tony@oc.tc>
|
|
||||||
Date: Sun, 25 Nov 2012 13:43:39 -0600
|
|
||||||
Subject: [PATCH] Add methods for working with arrows stuck in living entities
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
index d863fc9fa6b932b76a89871a09378a9c0697c108..c654026587bc9bf77b39f59a0c89991ac581da1e 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftLivingEntity.java
|
|
||||||
@@ -689,4 +689,16 @@ public class CraftLivingEntity extends CraftEntity implements LivingEntity {
|
|
||||||
getHandle().persistentInvisibility = invisible;
|
|
||||||
getHandle().setFlag(5, invisible);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ @Override
|
|
||||||
+ public int getArrowsStuck() {
|
|
||||||
+ return getHandle().getArrowCount();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setArrowsStuck(int arrows) {
|
|
||||||
+ getHandle().setArrowCount(arrows);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
@ -1,71 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Sat, 4 Apr 2015 23:17:52 -0400
|
|
||||||
Subject: [PATCH] Complete resource pack API
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/network/PlayerConnection.java b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
index 865d8efa2d480ae7edc286e3e79f2997a191ee5c..af0a7dc9464e1acf1451f45464bab8546207950e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
@@ -1605,7 +1605,11 @@ public class PlayerConnection implements PacketListenerPlayIn {
|
|
||||||
// CraftBukkit start
|
|
||||||
public void a(PacketPlayInResourcePackStatus packetplayinresourcepackstatus) {
|
|
||||||
PlayerConnectionUtils.ensureMainThread(packetplayinresourcepackstatus, this, this.player.getWorldServer());
|
|
||||||
- this.server.getPluginManager().callEvent(new PlayerResourcePackStatusEvent(getPlayer(), PlayerResourcePackStatusEvent.Status.values()[packetplayinresourcepackstatus.status.ordinal()]));
|
|
||||||
+ // Paper start
|
|
||||||
+ PlayerResourcePackStatusEvent.Status packStatus = PlayerResourcePackStatusEvent.Status.values()[packetplayinresourcepackstatus.status.ordinal()];
|
|
||||||
+ player.getBukkitEntity().setResourcePackStatus(packStatus);
|
|
||||||
+ this.server.getPluginManager().callEvent(new PlayerResourcePackStatusEvent(getPlayer(), packStatus));
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index f2228933719a2325a518be15237fedf56c994d1f..ed5680e6e0fcfbaf948bdede98d206cff2b26467 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -138,6 +138,10 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
private double health = 20;
|
|
||||||
private boolean scaledHealth = false;
|
|
||||||
private double healthScale = 20;
|
|
||||||
+ // Paper start
|
|
||||||
+ private org.bukkit.event.player.PlayerResourcePackStatusEvent.Status resourcePackStatus;
|
|
||||||
+ private String resourcePackHash;
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
public CraftPlayer(CraftServer server, EntityPlayer entity) {
|
|
||||||
super(server, entity);
|
|
||||||
@@ -1874,6 +1878,32 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
public boolean getAffectsSpawning() {
|
|
||||||
return this.getHandle().affectsSpawning;
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setResourcePack(String url, String hash) {
|
|
||||||
+ Validate.notNull(url, "Resource pack URL cannot be null");
|
|
||||||
+ Validate.notNull(hash, "Hash cannot be null");
|
|
||||||
+ this.getHandle().setResourcePack(url, hash);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public org.bukkit.event.player.PlayerResourcePackStatusEvent.Status getResourcePackStatus() {
|
|
||||||
+ return this.resourcePackStatus;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public String getResourcePackHash() {
|
|
||||||
+ return this.resourcePackHash;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public boolean hasResourcePack() {
|
|
||||||
+ return this.resourcePackStatus == org.bukkit.event.player.PlayerResourcePackStatusEvent.Status.SUCCESSFULLY_LOADED;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void setResourcePackStatus(org.bukkit.event.player.PlayerResourcePackStatusEvent.Status status) {
|
|
||||||
+ this.resourcePackStatus = status;
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
@Override
|
|
@ -1,59 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 4 Mar 2013 23:46:10 -0500
|
|
||||||
Subject: [PATCH] Chunk Save Reattempt
|
|
||||||
|
|
||||||
We commonly have "Stream Closed" errors on chunk saving, so this code should re-try to save the chunk in the event of failure and hopefully prevent rollbacks.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
index 1638f7902290e1bb233f11e5d0bbf83a9e863939..4bf3e0cb4602d33a2e00c502b1dd212032b22a8f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFile.java
|
|
||||||
@@ -265,7 +265,7 @@ public class RegionFile implements AutoCloseable {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} catch (IOException ioexception) {
|
|
||||||
- com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(ioexception); // Paper
|
|
||||||
+ com.destroystokyo.paper.util.SneakyThrow.sneaky(ioexception); // Paper - we want the upper try/catch to retry this
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
index de125077656f249d5cf9b76f07981b55e690e015..8310dd6bfc04b8ac0a51545baa3a264e6cb42eac 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
@@ -11,6 +11,7 @@ import java.io.IOException;
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
import net.minecraft.nbt.NBTCompressedStreamTools;
|
|
||||||
import net.minecraft.nbt.NBTTagCompound;
|
|
||||||
+import net.minecraft.server.MinecraftServer;
|
|
||||||
import net.minecraft.util.ExceptionSuppressor;
|
|
||||||
import net.minecraft.world.level.ChunkCoordIntPair;
|
|
||||||
|
|
||||||
@@ -92,6 +93,7 @@ public final class RegionFileCache implements AutoCloseable {
|
|
||||||
|
|
||||||
protected void write(ChunkCoordIntPair chunkcoordintpair, NBTTagCompound nbttagcompound) throws IOException {
|
|
||||||
RegionFile regionfile = this.getFile(chunkcoordintpair, false); // CraftBukkit
|
|
||||||
+ int attempts = 0; Exception laste = null; while (attempts++ < 5) { try { // Paper
|
|
||||||
DataOutputStream dataoutputstream = regionfile.c(chunkcoordintpair);
|
|
||||||
Throwable throwable = null;
|
|
||||||
|
|
||||||
@@ -115,6 +117,18 @@ public final class RegionFileCache implements AutoCloseable {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ return;
|
|
||||||
+ } catch (Exception ex) {
|
|
||||||
+ laste = ex;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (laste != null) {
|
|
||||||
+ com.destroystokyo.paper.exception.ServerInternalException.reportInternalException(laste);
|
|
||||||
+ MinecraftServer.LOGGER.error("Failed to save chunk", laste);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close() throws IOException {
|
|
@ -1,52 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Fri, 18 Mar 2016 13:17:38 -0400
|
|
||||||
Subject: [PATCH] Default loading permissions.yml before plugins
|
|
||||||
|
|
||||||
Under previous behavior, plugins were not able to check if a player had a permission
|
|
||||||
if it was defined in permissions.yml. there is no clean way for a plugin to fix that either.
|
|
||||||
|
|
||||||
This will change the order so that by default, permissions.yml loads BEFORE plugins instead of after.
|
|
||||||
|
|
||||||
This gives plugins expected permission checks.
|
|
||||||
|
|
||||||
It also helps improve the expected logic, as servers should set the initial defaults, and then let plugins
|
|
||||||
modify that. Under the previous logic, plugins were unable (cleanly) override permissions.yml.
|
|
||||||
|
|
||||||
A config option has been added for those who depend on the previous behavior, but I don't expect that.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
index 429b74474ced04d8dd8f038b8590b8dfe178bf4d..716f285e67019b8a62922d09c15883c99f9421aa 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -222,4 +222,9 @@ public class PaperConfig {
|
|
||||||
private static void useDisplayNameInQuit() {
|
|
||||||
useDisplayNameInQuit = getBoolean("use-display-name-in-quit-message", useDisplayNameInQuit);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public static boolean loadPermsBeforePlugins = true;
|
|
||||||
+ private static void loadPermsBeforePlugins() {
|
|
||||||
+ loadPermsBeforePlugins = getBoolean("settings.load-permissions-yml-before-plugins", true);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index 22b4dec4944b7f823996645af95fbef2d1d8a83b..6eec60b5739edb8f7278608e525b38bdac15bba8 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -398,6 +398,7 @@ public final class CraftServer implements Server {
|
|
||||||
if (type == PluginLoadOrder.STARTUP) {
|
|
||||||
helpMap.clear();
|
|
||||||
helpMap.initializeGeneralTopics();
|
|
||||||
+ if (com.destroystokyo.paper.PaperConfig.loadPermsBeforePlugins) loadCustomPermissions(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
Plugin[] plugins = pluginManager.getPlugins();
|
|
||||||
@@ -417,7 +418,7 @@ public final class CraftServer implements Server {
|
|
||||||
commandMap.registerServerAliases();
|
|
||||||
DefaultPermissions.registerCorePermissions();
|
|
||||||
CraftDefaultPermissions.registerCorePermissions();
|
|
||||||
- loadCustomPermissions();
|
|
||||||
+ if (!com.destroystokyo.paper.PaperConfig.loadPermsBeforePlugins) loadCustomPermissions(); // Paper
|
|
||||||
helpMap.initializeCommands();
|
|
||||||
syncCommands();
|
|
||||||
}
|
|
@ -1,35 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: William <admin@domnian.com>
|
|
||||||
Date: Fri, 18 Mar 2016 03:30:17 -0400
|
|
||||||
Subject: [PATCH] Allow Reloading of Custom Permissions
|
|
||||||
|
|
||||||
https://github.com/PaperMC/Paper/issues/49
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index 6eec60b5739edb8f7278608e525b38bdac15bba8..6d66a899912bfab2ce610fb6fb2dbec87cbb5790 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -2253,5 +2253,23 @@ public final class CraftServer implements Server {
|
|
||||||
}
|
|
||||||
return this.adventure$audiences;
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void reloadPermissions() {
|
|
||||||
+ pluginManager.clearPermissions();
|
|
||||||
+ if (com.destroystokyo.paper.PaperConfig.loadPermsBeforePlugins) loadCustomPermissions();
|
|
||||||
+ for (Plugin plugin : pluginManager.getPlugins()) {
|
|
||||||
+ for (Permission perm : plugin.getDescription().getPermissions()) {
|
|
||||||
+ try {
|
|
||||||
+ pluginManager.addPermission(perm);
|
|
||||||
+ } catch (IllegalArgumentException ex) {
|
|
||||||
+ getLogger().log(Level.WARNING, "Plugin " + plugin.getDescription().getFullName() + " tried to register permission '" + perm.getName() + "' but it's already registered", ex);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (!com.destroystokyo.paper.PaperConfig.loadPermsBeforePlugins) loadCustomPermissions();
|
|
||||||
+ DefaultPermissions.registerCorePermissions();
|
|
||||||
+ CraftDefaultPermissions.registerCorePermissions();
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
}
|
|
@ -1,31 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Fri, 18 Mar 2016 13:50:14 -0400
|
|
||||||
Subject: [PATCH] Remove Metadata on reload
|
|
||||||
|
|
||||||
Metadata is not meant to persist reload as things break badly with non primitive types
|
|
||||||
This will remove metadata on reload so it does not crash everything if a plugin uses it.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
index 6d66a899912bfab2ce610fb6fb2dbec87cbb5790..e670f484d454f7c706a0ec92c98087fa1373c492 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
||||||
@@ -871,8 +871,18 @@ public final class CraftServer implements Server {
|
|
||||||
world.paperConfig.init(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
+ Plugin[] pluginClone = pluginManager.getPlugins().clone(); // Paper
|
|
||||||
pluginManager.clearPlugins();
|
|
||||||
commandMap.clearCommands();
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ for (Plugin plugin : pluginClone) {
|
|
||||||
+ entityMetadata.removeAll(plugin);
|
|
||||||
+ worldMetadata.removeAll(plugin);
|
|
||||||
+ playerMetadata.removeAll(plugin);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
resetRecipes();
|
|
||||||
reloadData();
|
|
||||||
org.spigotmc.SpigotConfig.registerCommands(); // Spigot
|
|
@ -1,331 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 28 May 2015 23:00:19 -0400
|
|
||||||
Subject: [PATCH] Handle Item Meta Inconsistencies
|
|
||||||
|
|
||||||
First, Enchantment order would blow away seeing 2 items as the same,
|
|
||||||
however the Client forces enchantment list in a certain order, as well
|
|
||||||
as does the /enchant command. Anvils can insert it into forced order,
|
|
||||||
causing 2 same items to be considered different.
|
|
||||||
|
|
||||||
This change makes unhandled NBT Tags and Enchantments use a sorted tree map,
|
|
||||||
so they will always be in a consistent order.
|
|
||||||
|
|
||||||
Additionally, the old enchantment API was never updated when ItemMeta
|
|
||||||
was added, resulting in 2 different ways to modify an items enchantments.
|
|
||||||
|
|
||||||
For consistency, the old API methods now forward to use the
|
|
||||||
ItemMeta API equivalents, and should deprecate the old API's.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/item/ItemStack.java b/src/main/java/net/minecraft/world/item/ItemStack.java
|
|
||||||
index f1a780768e3f4bdb43a7ca6d7850befefb71bf57..201ba7250b298f4a91bc45f5954f54ae557305f2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/item/ItemStack.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/item/ItemStack.java
|
|
||||||
@@ -9,6 +9,8 @@ import com.mojang.serialization.Codec;
|
|
||||||
import com.mojang.serialization.codecs.RecordCodecBuilder;
|
|
||||||
import java.text.DecimalFormat;
|
|
||||||
import java.text.DecimalFormatSymbols;
|
|
||||||
+import java.util.Collections;
|
|
||||||
+import java.util.Comparator;
|
|
||||||
import java.util.Locale;
|
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.Optional;
|
|
||||||
@@ -120,6 +122,23 @@ public final class ItemStack {
|
|
||||||
private ShapeDetectorBlock n;
|
|
||||||
private boolean o;
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ private static final java.util.Comparator<? super NBTTagCompound> enchantSorter = java.util.Comparator.comparing(o -> o.getString("id"));
|
|
||||||
+ private void processEnchantOrder(NBTTagCompound tag) {
|
|
||||||
+ if (tag == null || !tag.hasKeyOfType("Enchantments", 9)) {
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ NBTTagList list = tag.getList("Enchantments", 10);
|
|
||||||
+ if (list.size() < 2) {
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ try {
|
|
||||||
+ //noinspection unchecked
|
|
||||||
+ list.sort((Comparator<? super NBTBase>) enchantSorter); // Paper
|
|
||||||
+ } catch (Exception ignored) {}
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
public ItemStack(IMaterial imaterial) {
|
|
||||||
this(imaterial, 1);
|
|
||||||
}
|
|
||||||
@@ -162,6 +181,7 @@ public final class ItemStack {
|
|
||||||
if (nbttagcompound.hasKeyOfType("tag", 10)) {
|
|
||||||
// CraftBukkit start - make defensive copy as this data may be coming from the save thread
|
|
||||||
this.tag = (NBTTagCompound) nbttagcompound.getCompound("tag").clone();
|
|
||||||
+ processEnchantOrder(this.tag); // Paper
|
|
||||||
this.getItem().b(this.tag);
|
|
||||||
// CraftBukkit end
|
|
||||||
}
|
|
||||||
@@ -680,6 +700,7 @@ public final class ItemStack {
|
|
||||||
// Paper end
|
|
||||||
public void setTag(@Nullable NBTTagCompound nbttagcompound) {
|
|
||||||
this.tag = nbttagcompound;
|
|
||||||
+ processEnchantOrder(this.tag); // Paper
|
|
||||||
if (this.getItem().usesDurability()) {
|
|
||||||
this.setDamage(this.getDamage());
|
|
||||||
}
|
|
||||||
@@ -770,6 +791,7 @@ public final class ItemStack {
|
|
||||||
nbttagcompound.setString("id", String.valueOf(IRegistry.ENCHANTMENT.getKey(enchantment)));
|
|
||||||
nbttagcompound.setShort("lvl", (short) ((byte) i));
|
|
||||||
nbttaglist.add(nbttagcompound);
|
|
||||||
+ processEnchantOrder(nbttagcompound); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean hasEnchantments() {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/inventory/CraftItemStack.java b/src/main/java/org/bukkit/craftbukkit/inventory/CraftItemStack.java
|
|
||||||
index 275b943a59ef28c831a068987e111e84ebba3bb7..7221ac52c9f66ae0af6f6cbf15c8d47f9c0291a0 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/inventory/CraftItemStack.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/inventory/CraftItemStack.java
|
|
||||||
@@ -178,28 +178,11 @@ public final class CraftItemStack extends ItemStack {
|
|
||||||
public void addUnsafeEnchantment(Enchantment ench, int level) {
|
|
||||||
Validate.notNull(ench, "Cannot add null enchantment");
|
|
||||||
|
|
||||||
- if (!makeTag(handle)) {
|
|
||||||
- return;
|
|
||||||
- }
|
|
||||||
- NBTTagList list = getEnchantmentList(handle);
|
|
||||||
- if (list == null) {
|
|
||||||
- list = new NBTTagList();
|
|
||||||
- handle.getTag().set(ENCHANTMENTS.NBT, list);
|
|
||||||
- }
|
|
||||||
- int size = list.size();
|
|
||||||
-
|
|
||||||
- for (int i = 0; i < size; i++) {
|
|
||||||
- NBTTagCompound tag = (NBTTagCompound) list.get(i);
|
|
||||||
- String id = tag.getString(ENCHANTMENTS_ID.NBT);
|
|
||||||
- if (id.equals(ench.getKey().toString())) {
|
|
||||||
- tag.setShort(ENCHANTMENTS_LVL.NBT, (short) level);
|
|
||||||
- return;
|
|
||||||
- }
|
|
||||||
- }
|
|
||||||
- NBTTagCompound tag = new NBTTagCompound();
|
|
||||||
- tag.setString(ENCHANTMENTS_ID.NBT, ench.getKey().toString());
|
|
||||||
- tag.setShort(ENCHANTMENTS_LVL.NBT, (short) level);
|
|
||||||
- list.add(tag);
|
|
||||||
+ // Paper start - Replace whole method
|
|
||||||
+ final ItemMeta itemMeta = getItemMeta();
|
|
||||||
+ itemMeta.addEnchant(ench, level, true);
|
|
||||||
+ setItemMeta(itemMeta);
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
static boolean makeTag(net.minecraft.world.item.ItemStack item) {
|
|
||||||
@@ -216,66 +199,33 @@ public final class CraftItemStack extends ItemStack {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean containsEnchantment(Enchantment ench) {
|
|
||||||
- return getEnchantmentLevel(ench) > 0;
|
|
||||||
+ return hasItemMeta() && getItemMeta().hasEnchant(ench); // Paper - use meta
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getEnchantmentLevel(Enchantment ench) {
|
|
||||||
- Validate.notNull(ench, "Cannot find null enchantment");
|
|
||||||
- if (handle == null) {
|
|
||||||
- return 0;
|
|
||||||
- }
|
|
||||||
- return EnchantmentManager.getEnchantmentLevel(CraftEnchantment.getRaw(ench), handle);
|
|
||||||
+ return hasItemMeta() ? getItemMeta().getEnchantLevel(ench) : 0; // Paper - replace entire method with meta
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int removeEnchantment(Enchantment ench) {
|
|
||||||
Validate.notNull(ench, "Cannot remove null enchantment");
|
|
||||||
|
|
||||||
- NBTTagList list = getEnchantmentList(handle), listCopy;
|
|
||||||
- if (list == null) {
|
|
||||||
- return 0;
|
|
||||||
- }
|
|
||||||
- int index = Integer.MIN_VALUE;
|
|
||||||
- int level = Integer.MIN_VALUE;
|
|
||||||
- int size = list.size();
|
|
||||||
-
|
|
||||||
- for (int i = 0; i < size; i++) {
|
|
||||||
- NBTTagCompound enchantment = (NBTTagCompound) list.get(i);
|
|
||||||
- String id = enchantment.getString(ENCHANTMENTS_ID.NBT);
|
|
||||||
- if (id.equals(ench.getKey().toString())) {
|
|
||||||
- index = i;
|
|
||||||
- level = 0xffff & enchantment.getShort(ENCHANTMENTS_LVL.NBT);
|
|
||||||
- break;
|
|
||||||
- }
|
|
||||||
- }
|
|
||||||
-
|
|
||||||
- if (index == Integer.MIN_VALUE) {
|
|
||||||
- return 0;
|
|
||||||
- }
|
|
||||||
- if (size == 1) {
|
|
||||||
- handle.getTag().remove(ENCHANTMENTS.NBT);
|
|
||||||
- if (handle.getTag().isEmpty()) {
|
|
||||||
- handle.setTag(null);
|
|
||||||
- }
|
|
||||||
- return level;
|
|
||||||
- }
|
|
||||||
-
|
|
||||||
- // This is workaround for not having an index removal
|
|
||||||
- listCopy = new NBTTagList();
|
|
||||||
- for (int i = 0; i < size; i++) {
|
|
||||||
- if (i != index) {
|
|
||||||
- listCopy.add(list.get(i));
|
|
||||||
- }
|
|
||||||
+ // Paper start - replace entire method
|
|
||||||
+ final ItemMeta itemMeta = getItemMeta();
|
|
||||||
+ int level = itemMeta.getEnchantLevel(ench);
|
|
||||||
+ if (level > 0) {
|
|
||||||
+ itemMeta.removeEnchant(ench);
|
|
||||||
+ setItemMeta(itemMeta);
|
|
||||||
}
|
|
||||||
- handle.getTag().set(ENCHANTMENTS.NBT, listCopy);
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
return level;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Map<Enchantment, Integer> getEnchantments() {
|
|
||||||
- return getEnchantments(handle);
|
|
||||||
+ return hasItemMeta() ? getItemMeta().getEnchants() : ImmutableMap.<Enchantment, Integer>of(); // Paper - use Item Meta
|
|
||||||
}
|
|
||||||
|
|
||||||
static Map<Enchantment, Integer> getEnchantments(net.minecraft.world.item.ItemStack item) {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaItem.java b/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaItem.java
|
|
||||||
index 7a11b2ddfa4244459253c918315aaab78ef2eb4a..57a6e66866ea82caccbbbfd55948a081f50f6bbe 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaItem.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/inventory/CraftMetaItem.java
|
|
||||||
@@ -6,6 +6,7 @@ import com.google.common.collect.ImmutableList;
|
|
||||||
import com.google.common.collect.ImmutableMap;
|
|
||||||
import com.google.common.collect.ImmutableMultimap;
|
|
||||||
import com.google.common.collect.LinkedHashMultimap;
|
|
||||||
+import com.google.common.collect.ImmutableSortedMap; // Paper
|
|
||||||
import com.google.common.collect.Lists;
|
|
||||||
import com.google.common.collect.Multimap;
|
|
||||||
import com.google.common.collect.SetMultimap;
|
|
||||||
@@ -22,6 +23,7 @@ import java.lang.reflect.InvocationTargetException;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.Collection;
|
|
||||||
+import java.util.Comparator; // Paper
|
|
||||||
import java.util.EnumSet;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Iterator;
|
|
||||||
@@ -32,6 +34,7 @@ import java.util.Map;
|
|
||||||
import java.util.NoSuchElementException;
|
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.Set;
|
|
||||||
+import java.util.TreeMap; // Paper
|
|
||||||
import java.util.logging.Level;
|
|
||||||
import java.util.logging.Logger;
|
|
||||||
import javax.annotation.Nonnull;
|
|
||||||
@@ -271,7 +274,7 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
private List<String> lore; // null and empty are two different states internally
|
|
||||||
private Integer customModelData;
|
|
||||||
private NBTTagCompound blockData;
|
|
||||||
- private Map<Enchantment, Integer> enchantments;
|
|
||||||
+ private EnchantmentMap enchantments; // Paper
|
|
||||||
private Multimap<Attribute, AttributeModifier> attributeModifiers;
|
|
||||||
private int repairCost;
|
|
||||||
private int hideFlag;
|
|
||||||
@@ -282,7 +285,7 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
private static final CraftPersistentDataTypeRegistry DATA_TYPE_REGISTRY = new CraftPersistentDataTypeRegistry();
|
|
||||||
|
|
||||||
private NBTTagCompound internalTag;
|
|
||||||
- private final Map<String, NBTBase> unhandledTags = new HashMap<String, NBTBase>();
|
|
||||||
+ private final Map<String, NBTBase> unhandledTags = new TreeMap<>(); // Paper
|
|
||||||
private CraftPersistentDataContainer persistentDataContainer = new CraftPersistentDataContainer(DATA_TYPE_REGISTRY);
|
|
||||||
|
|
||||||
private int version = CraftMagicNumbers.INSTANCE.getDataVersion(); // Internal use only
|
|
||||||
@@ -303,7 +306,7 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
this.blockData = meta.blockData;
|
|
||||||
|
|
||||||
if (meta.enchantments != null) { // Spigot
|
|
||||||
- this.enchantments = new LinkedHashMap<Enchantment, Integer>(meta.enchantments);
|
|
||||||
+ this.enchantments = new EnchantmentMap(meta.enchantments); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
if (meta.hasAttributeModifiers()) {
|
|
||||||
@@ -386,13 +389,13 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- static Map<Enchantment, Integer> buildEnchantments(NBTTagCompound tag, ItemMetaKey key) {
|
|
||||||
+ static EnchantmentMap buildEnchantments(NBTTagCompound tag, ItemMetaKey key) { // Paper
|
|
||||||
if (!tag.hasKey(key.NBT)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
NBTTagList ench = tag.getList(key.NBT, CraftMagicNumbers.NBT.TAG_COMPOUND);
|
|
||||||
- Map<Enchantment, Integer> enchantments = new LinkedHashMap<Enchantment, Integer>(ench.size());
|
|
||||||
+ EnchantmentMap enchantments = new EnchantmentMap(); // Paper
|
|
||||||
|
|
||||||
for (int i = 0; i < ench.size(); i++) {
|
|
||||||
String id = ((NBTTagCompound) ench.get(i)).getString(ENCHANTMENTS_ID.NBT);
|
|
||||||
@@ -545,13 +548,13 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- static Map<Enchantment, Integer> buildEnchantments(Map<String, Object> map, ItemMetaKey key) {
|
|
||||||
+ static EnchantmentMap buildEnchantments(Map<String, Object> map, ItemMetaKey key) { // Paper
|
|
||||||
Map<?, ?> ench = SerializableMeta.getObject(Map.class, map, key.BUKKIT, true);
|
|
||||||
if (ench == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
- Map<Enchantment, Integer> enchantments = new LinkedHashMap<Enchantment, Integer>(ench.size());
|
|
||||||
+ EnchantmentMap enchantments = new EnchantmentMap(); // Paper
|
|
||||||
for (Map.Entry<?, ?> entry : ench.entrySet()) {
|
|
||||||
// Doctor older enchants
|
|
||||||
String enchantKey = entry.getKey().toString();
|
|
||||||
@@ -827,14 +830,14 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Map<Enchantment, Integer> getEnchants() {
|
|
||||||
- return hasEnchants() ? ImmutableMap.copyOf(enchantments) : ImmutableMap.<Enchantment, Integer>of();
|
|
||||||
+ return hasEnchants() ? ImmutableSortedMap.copyOfSorted(enchantments) : ImmutableMap.<Enchantment, Integer>of(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean addEnchant(Enchantment ench, int level, boolean ignoreRestrictions) {
|
|
||||||
Validate.notNull(ench, "Enchantment cannot be null");
|
|
||||||
if (enchantments == null) {
|
|
||||||
- enchantments = new LinkedHashMap<Enchantment, Integer>(4);
|
|
||||||
+ enchantments = new EnchantmentMap(); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ignoreRestrictions || level >= ench.getStartLevel() && level <= ench.getMaxLevel()) {
|
|
||||||
@@ -1215,7 +1218,7 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
clone.customModelData = this.customModelData;
|
|
||||||
clone.blockData = this.blockData;
|
|
||||||
if (this.enchantments != null) {
|
|
||||||
- clone.enchantments = new LinkedHashMap<Enchantment, Integer>(this.enchantments);
|
|
||||||
+ clone.enchantments = new EnchantmentMap(this.enchantments); // Paper
|
|
||||||
}
|
|
||||||
if (this.hasAttributeModifiers()) {
|
|
||||||
clone.attributeModifiers = LinkedHashMultimap.create(this.attributeModifiers);
|
|
||||||
@@ -1447,4 +1450,22 @@ class CraftMetaItem implements ItemMeta, Damageable, Repairable, BlockDataMeta {
|
|
||||||
return HANDLED_TAGS;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ // Paper start
|
|
||||||
+ private static class EnchantmentMap extends TreeMap<Enchantment, Integer> {
|
|
||||||
+ private EnchantmentMap(Map<Enchantment, Integer> enchantments) {
|
|
||||||
+ this();
|
|
||||||
+ putAll(enchantments);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private EnchantmentMap() {
|
|
||||||
+ super(Comparator.comparing(o -> o.getKey().toString()));
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public EnchantmentMap clone() {
|
|
||||||
+ return (EnchantmentMap) super.clone();
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
}
|
|
@ -1,44 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Fri, 18 Mar 2016 15:12:22 -0400
|
|
||||||
Subject: [PATCH] Configurable Non Player Arrow Despawn Rate
|
|
||||||
|
|
||||||
Can set a much shorter despawn rate for arrows that players can not pick up.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 3ac2ac3db9b1c271b3c21930bb13716669ff64d3..3c78d3234054ce2dc46ef77decb6adb0cbd10620 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -215,4 +215,19 @@ public class PaperWorldConfig {
|
|
||||||
private void nonPlayerEntitiesOnScoreboards() {
|
|
||||||
nonPlayerEntitiesOnScoreboards = getBoolean("allow-non-player-entities-on-scoreboards", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int nonPlayerArrowDespawnRate = -1;
|
|
||||||
+ public int creativeArrowDespawnRate = -1;
|
|
||||||
+ private void nonPlayerArrowDespawnRate() {
|
|
||||||
+ nonPlayerArrowDespawnRate = getInt("non-player-arrow-despawn-rate", -1);
|
|
||||||
+ if (nonPlayerArrowDespawnRate == -1) {
|
|
||||||
+ nonPlayerArrowDespawnRate = spigotConfig.arrowDespawnRate;
|
|
||||||
+ }
|
|
||||||
+ creativeArrowDespawnRate = getInt("creative-arrow-despawn-rate", -1);
|
|
||||||
+ if (creativeArrowDespawnRate == -1) {
|
|
||||||
+ creativeArrowDespawnRate = spigotConfig.arrowDespawnRate;
|
|
||||||
+ }
|
|
||||||
+ log("Non Player Arrow Despawn Rate: " + nonPlayerArrowDespawnRate);
|
|
||||||
+ log("Creative Arrow Despawn Rate: " + creativeArrowDespawnRate);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/projectile/EntityArrow.java b/src/main/java/net/minecraft/world/entity/projectile/EntityArrow.java
|
|
||||||
index 04af85e0ccb7a98e0796afcdcce33f8595b1db8c..7868259a94766a6100d7b278c4296dde0a7f9397 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/projectile/EntityArrow.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/projectile/EntityArrow.java
|
|
||||||
@@ -283,7 +283,7 @@ public abstract class EntityArrow extends IProjectile {
|
|
||||||
|
|
||||||
protected void h() {
|
|
||||||
++this.despawnCounter;
|
|
||||||
- if (this.despawnCounter >= ((this instanceof EntityThrownTrident) ? world.spigotConfig.tridentDespawnRate : world.spigotConfig.arrowDespawnRate)) { // Spigot
|
|
||||||
+ if (this.despawnCounter >= (fromPlayer == PickupStatus.CREATIVE_ONLY ? world.paperConfig.creativeArrowDespawnRate : (fromPlayer == PickupStatus.DISALLOWED ? world.paperConfig.nonPlayerArrowDespawnRate : ((this instanceof EntityThrownTrident) ? world.spigotConfig.tridentDespawnRate : world.spigotConfig.arrowDespawnRate)))) { // Spigot // Paper - TODO: Extract this to init?
|
|
||||||
this.die();
|
|
||||||
}
|
|
||||||
|
|
@ -1,65 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Fri, 18 Mar 2016 20:16:03 -0400
|
|
||||||
Subject: [PATCH] Add World Util Methods
|
|
||||||
|
|
||||||
Methods that can be used for other patches to help improve logic.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index a1769df6a3f6150d322f145199caba3839871dff..3ea4ec748c229031a5f0d973988bb20e55679971 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -197,7 +197,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
public final Convertable.ConversionSession convertable;
|
|
||||||
public final UUID uuid;
|
|
||||||
|
|
||||||
- public Chunk getChunkIfLoaded(int x, int z) {
|
|
||||||
+ @Override public Chunk getChunkIfLoaded(int x, int z) { // Paper - this was added in world too but keeping here for NMS ABI
|
|
||||||
return this.chunkProvider.getChunkAt(x, z, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 01cb0c8dd9875986e0c08371e876f0dba3f0cf5a..a570998e4ef6c3ff83403881bf1d24c8cbcfcf67 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -297,11 +297,27 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
- public Fluid getFluidIfLoaded(BlockPosition blockposition) {
|
|
||||||
+ public final Fluid getFluidIfLoaded(BlockPosition blockposition) {
|
|
||||||
IChunkAccess chunk = this.getChunkIfLoadedImmediately(blockposition.getX() >> 4, blockposition.getZ() >> 4);
|
|
||||||
|
|
||||||
return chunk == null ? null : chunk.getFluid(blockposition);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public final boolean isLoadedAndInBounds(BlockPosition blockposition) { // Paper - final for inline
|
|
||||||
+ return getWorldBorder().isInBounds(blockposition) && getChunkIfLoadedImmediately(blockposition.getX() >> 4, blockposition.getZ() >> 4) != null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public Chunk getChunkIfLoaded(int x, int z) { // Overridden in WorldServer for ABI compat which has final
|
|
||||||
+ return ((WorldServer) this).getChunkProvider().getChunkAtIfLoadedImmediately(x, z);
|
|
||||||
+ }
|
|
||||||
+ public final Chunk getChunkIfLoaded(BlockPosition blockposition) {
|
|
||||||
+ return ((WorldServer) this).getChunkProvider().getChunkAtIfLoadedImmediately(blockposition.getX() >> 4, blockposition.getZ() >> 4);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // reduces need to do isLoaded before getType
|
|
||||||
+ public final IBlockData getTypeIfLoadedAndInBounds(BlockPosition blockposition) {
|
|
||||||
+ return getWorldBorder().isInBounds(blockposition) ? getTypeIfLoaded(blockposition) : null;
|
|
||||||
+ }
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
@Override
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/border/WorldBorder.java b/src/main/java/net/minecraft/world/level/border/WorldBorder.java
|
|
||||||
index 2bb03f1cb9671a7754a68059219f783d4508eeb9..f16c76df5d7b184d57f4cc397f069eac9cc430cb 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/border/WorldBorder.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/border/WorldBorder.java
|
|
||||||
@@ -31,6 +31,7 @@ public class WorldBorder {
|
|
||||||
|
|
||||||
public WorldBorder() {}
|
|
||||||
|
|
||||||
+ public final boolean isInBounds(BlockPosition blockposition) { return this.a(blockposition); } // Paper - OBFHELPER
|
|
||||||
public boolean a(BlockPosition blockposition) {
|
|
||||||
return (double) (blockposition.getX() + 1) > this.e() && (double) blockposition.getX() < this.g() && (double) (blockposition.getZ() + 1) > this.f() && (double) blockposition.getZ() < this.h();
|
|
||||||
}
|
|
@ -1,48 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Sun, 21 Jun 2015 15:07:20 -0400
|
|
||||||
Subject: [PATCH] Custom replacement for eaten items
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityLiving.java b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
index 70211129e6ab2f7cdb975adcb532be595bc3834f..759592fc6a4654c0760ff8a1d7f3b87c364b045a 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
@@ -3205,9 +3205,10 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
this.b(this.activeItem, 16);
|
|
||||||
// CraftBukkit start - fire PlayerItemConsumeEvent
|
|
||||||
ItemStack itemstack;
|
|
||||||
+ PlayerItemConsumeEvent event = null; // Paper
|
|
||||||
if (this instanceof EntityPlayer) {
|
|
||||||
org.bukkit.inventory.ItemStack craftItem = CraftItemStack.asBukkitCopy(this.activeItem);
|
|
||||||
- PlayerItemConsumeEvent event = new PlayerItemConsumeEvent((Player) this.getBukkitEntity(), craftItem);
|
|
||||||
+ event = new PlayerItemConsumeEvent((Player) this.getBukkitEntity(), craftItem); // Paper
|
|
||||||
world.getServer().getPluginManager().callEvent(event);
|
|
||||||
|
|
||||||
if (event.isCancelled()) {
|
|
||||||
@@ -3221,6 +3222,13 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
} else {
|
|
||||||
itemstack = this.activeItem.a(this.world, this);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ // Paper start - save the default replacement item and change it if necessary
|
|
||||||
+ final ItemStack defaultReplacement = itemstack;
|
|
||||||
+ if (event != null && event.getReplacement() != null) {
|
|
||||||
+ itemstack = CraftItemStack.asNMSCopy(event.getReplacement());
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
if (itemstack != this.activeItem) {
|
|
||||||
@@ -3228,6 +3236,11 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
this.clearActiveItem();
|
|
||||||
+ // Paper start - if the replacement is anything but the default, update the client inventory
|
|
||||||
+ if (this instanceof EntityPlayer && !com.google.common.base.Objects.equal(defaultReplacement, itemstack)) {
|
|
||||||
+ ((EntityPlayer) this).getBukkitEntity().updateInventory();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,57 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 27 Sep 2015 01:18:02 -0400
|
|
||||||
Subject: [PATCH] handle NaN health/absorb values and repair bad data
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityLiving.java b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
index 759592fc6a4654c0760ff8a1d7f3b87c364b045a..375284b7e321cb03a7a30aedea165ca7a2fd1091 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
@@ -702,7 +702,13 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void loadData(NBTTagCompound nbttagcompound) {
|
|
||||||
- this.setAbsorptionHearts(nbttagcompound.getFloat("AbsorptionAmount"));
|
|
||||||
+ // Paper start - jvm keeps optimizing the setter
|
|
||||||
+ float absorptionAmount = nbttagcompound.getFloat("AbsorptionAmount");
|
|
||||||
+ if (Float.isNaN(absorptionAmount)) {
|
|
||||||
+ absorptionAmount = 0;
|
|
||||||
+ }
|
|
||||||
+ this.setAbsorptionHearts(absorptionAmount);
|
|
||||||
+ // Paper end
|
|
||||||
if (nbttagcompound.hasKeyOfType("Attributes", 9) && this.world != null && !this.world.isClientSide) {
|
|
||||||
this.getAttributeMap().a(nbttagcompound.getList("Attributes", 10));
|
|
||||||
}
|
|
||||||
@@ -1151,6 +1157,10 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setHealth(float f) {
|
|
||||||
+ // Paper start
|
|
||||||
+ if (Float.isNaN(f)) { f = getMaxHealth(); if (this.valid) {
|
|
||||||
+ System.err.println("[NAN-HEALTH] " + getName() + " had NaN health set");
|
|
||||||
+ } } // Paper end
|
|
||||||
// CraftBukkit start - Handle scaled health
|
|
||||||
if (this instanceof EntityPlayer) {
|
|
||||||
org.bukkit.craftbukkit.entity.CraftPlayer player = ((EntityPlayer) this).getBukkitEntity();
|
|
||||||
@@ -3045,7 +3055,7 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setAbsorptionHearts(float f) {
|
|
||||||
- if (f < 0.0F) {
|
|
||||||
+ if (f < 0.0F || Float.isNaN(f)) { // Paper
|
|
||||||
f = 0.0F;
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index ed5680e6e0fcfbaf948bdede98d206cff2b26467..ab7e731cb04a75b829f0cd24d7b94f7ddad166e8 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -1680,6 +1680,7 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setRealHealth(double health) {
|
|
||||||
+ if (Double.isNaN(health)) {return;} // Paper
|
|
||||||
this.health = health;
|
|
||||||
}
|
|
||||||
|
|
@ -1,42 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Tue, 22 Mar 2016 00:33:47 -0400
|
|
||||||
Subject: [PATCH] Use a Shared Random for Entities
|
|
||||||
|
|
||||||
Reduces memory usage and provides ensures more randomness, Especially since a lot of garbage entity objects get created.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index bfced192c1e8fd3fa0250a0f93adfc061d7e71e5..415935739716df3b8b3319aac19519d29aaa5776 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -143,6 +143,21 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
return tag.hasKey("Bukkit.updateLevel") && tag.getInt("Bukkit.updateLevel") >= level;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ public static Random SHARED_RANDOM = new Random() {
|
|
||||||
+ private boolean locked = false;
|
|
||||||
+ @Override
|
|
||||||
+ public synchronized void setSeed(long seed) {
|
|
||||||
+ if (locked) {
|
|
||||||
+ LogManager.getLogger().error("Ignoring setSeed on Entity.SHARED_RANDOM", new Throwable());
|
|
||||||
+ } else {
|
|
||||||
+ super.setSeed(seed);
|
|
||||||
+ locked = true;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ };
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
private CraftEntity bukkitEntity;
|
|
||||||
|
|
||||||
public CraftEntity getBukkitEntity() {
|
|
||||||
@@ -272,7 +287,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
this.x = Vec3D.ORIGIN;
|
|
||||||
this.am = 1.0F;
|
|
||||||
this.an = 1.0F;
|
|
||||||
- this.random = new Random();
|
|
||||||
+ this.random = SHARED_RANDOM; // Paper
|
|
||||||
this.fireTicks = -this.getMaxFireTicks();
|
|
||||||
this.M = new Object2DoubleArrayMap(2);
|
|
||||||
this.justCreated = true;
|
|
@ -1,36 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Tue, 22 Mar 2016 12:04:28 -0500
|
|
||||||
Subject: [PATCH] Configurable spawn chances for skeleton horses
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 3c78d3234054ce2dc46ef77decb6adb0cbd10620..cd64fb9d0c6d123e1c86cb33f12cd9cefc9f80d0 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -230,4 +230,12 @@ public class PaperWorldConfig {
|
|
||||||
log("Non Player Arrow Despawn Rate: " + nonPlayerArrowDespawnRate);
|
|
||||||
log("Creative Arrow Despawn Rate: " + creativeArrowDespawnRate);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public double skeleHorseSpawnChance;
|
|
||||||
+ private void skeleHorseSpawnChance() {
|
|
||||||
+ skeleHorseSpawnChance = getDouble("skeleton-horse-thunder-spawn-chance", 0.01D);
|
|
||||||
+ if (skeleHorseSpawnChance < 0) {
|
|
||||||
+ skeleHorseSpawnChance = 0.01D; // Vanilla value
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 3ea4ec748c229031a5f0d973988bb20e55679971..5cd864e58b85fa163489557437f5c2eec9f008b7 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -590,7 +590,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
blockposition = this.a(this.a(j, 0, k, 15));
|
|
||||||
if (this.isRainingAt(blockposition)) {
|
|
||||||
DifficultyDamageScaler difficultydamagescaler = this.getDamageScaler(blockposition);
|
|
||||||
- boolean flag1 = this.getGameRules().getBoolean(GameRules.DO_MOB_SPAWNING) && this.random.nextDouble() < (double) difficultydamagescaler.b() * 0.01D;
|
|
||||||
+ boolean flag1 = this.getGameRules().getBoolean(GameRules.DO_MOB_SPAWNING) && this.random.nextDouble() < (double) difficultydamagescaler.b() * paperConfig.skeleHorseSpawnChance; // Paper
|
|
||||||
|
|
||||||
if (flag1) {
|
|
||||||
EntityHorseSkeleton entityhorseskeleton = (EntityHorseSkeleton) EntityTypes.SKELETON_HORSE.a((World) this);
|
|
@ -1,206 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 3 Mar 2016 02:07:55 -0600
|
|
||||||
Subject: [PATCH] Optimize isValidLocation, getType and getBlockData for
|
|
||||||
inlining
|
|
||||||
|
|
||||||
Hot methods, so reduce # of instructions for the method.
|
|
||||||
|
|
||||||
Move is valid location test to the BlockPosition class so that it can access local variables.
|
|
||||||
|
|
||||||
Replace all calls to the new place to the unnecessary forward.
|
|
||||||
|
|
||||||
Optimize getType and getBlockData to manually inline and optimize the calls
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/core/BaseBlockPosition.java b/src/main/java/net/minecraft/core/BaseBlockPosition.java
|
|
||||||
index 25fdd55a7548cfaa45a541ad77f22f33c33e7471..4b56683336fdab06804efdc8ca1f7c130b77291f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/core/BaseBlockPosition.java
|
|
||||||
+++ b/src/main/java/net/minecraft/core/BaseBlockPosition.java
|
|
||||||
@@ -22,6 +22,15 @@ public class BaseBlockPosition implements Comparable<BaseBlockPosition> {
|
|
||||||
private int b;public final void setY(final int y) { this.b = y; } // Paper - OBFHELPER
|
|
||||||
private int e;public final void setZ(final int z) { this.e = z; } // Paper - OBFHELPER
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ public boolean isValidLocation() {
|
|
||||||
+ return getX() >= -30000000 && getZ() >= -30000000 && getX() < 30000000 && getZ() < 30000000 && getY() >= 0 && getY() < 256;
|
|
||||||
+ }
|
|
||||||
+ public boolean isInvalidYLocation() {
|
|
||||||
+ return b < 0 || b >= 256;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
public BaseBlockPosition(int i, int j, int k) {
|
|
||||||
this.a = i;
|
|
||||||
this.b = j;
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index a570998e4ef6c3ff83403881bf1d24c8cbcfcf67..a22be13b097052b2a88707c9436b88c84298e46b 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -239,7 +239,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean isValidLocation(BlockPosition blockposition) {
|
|
||||||
- return !isOutsideWorld(blockposition) && D(blockposition);
|
|
||||||
+ return blockposition.isValidLocation(); // Paper - use better/optimized check
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean l(BlockPosition blockposition) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index 4b3de29b1a6e9d75b28962073c62bbe8d666165f..fdc491f978560c394eec22116572585f9bbdec9f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -348,12 +348,27 @@ public class Chunk implements IChunkAccess {
|
|
||||||
return this.sections;
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
+ // Paper start - Optimize getBlockData to reduce instructions
|
|
||||||
+ public final IBlockData getBlockData(BlockPosition pos) { return getBlockData(pos.getX(), pos.getY(), pos.getZ()); } // Paper
|
|
||||||
public IBlockData getType(BlockPosition blockposition) {
|
|
||||||
- int i = blockposition.getX();
|
|
||||||
- int j = blockposition.getY();
|
|
||||||
- int k = blockposition.getZ();
|
|
||||||
+ return this.getBlockData(blockposition.getX(), blockposition.getY(), blockposition.getZ());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public IBlockData getType(final int x, final int y, final int z) {
|
|
||||||
+ return getBlockData(x, y, z);
|
|
||||||
+ }
|
|
||||||
+ public final IBlockData getBlockData(final int x, final int y, final int z) {
|
|
||||||
+ // Method body / logic copied from below
|
|
||||||
+ final int i = y >> 4;
|
|
||||||
+ if (y < 0 || i >= this.sections.length || this.sections[i] == null || this.sections[i].nonEmptyBlockCount == 0) {
|
|
||||||
+ return Blocks.AIR.getBlockData();
|
|
||||||
+ }
|
|
||||||
+ // Inlined ChunkSection.getType() and DataPaletteBlock.a(int,int,int)
|
|
||||||
+ return this.sections[i].blockIds.a((y & 15) << 8 | (z & 15) << 4 | x & 15);
|
|
||||||
+ }
|
|
||||||
|
|
||||||
+ public IBlockData getBlockData_unused(int i, int j, int k) {
|
|
||||||
+ // Paper end
|
|
||||||
if (this.world.isDebugWorld()) {
|
|
||||||
IBlockData iblockdata = null;
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/ChunkEmpty.java b/src/main/java/net/minecraft/world/level/chunk/ChunkEmpty.java
|
|
||||||
index 395d21afaabcbd99f9ce0551d647f5db9507a518..89efd0b68b04457e1cd617dcc8bb1a6ea1c4717c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/ChunkEmpty.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/ChunkEmpty.java
|
|
||||||
@@ -23,7 +23,7 @@ import net.minecraft.world.phys.AxisAlignedBB;
|
|
||||||
|
|
||||||
public class ChunkEmpty extends Chunk {
|
|
||||||
|
|
||||||
- private static final BiomeBase[] b = (BiomeBase[]) SystemUtils.a((Object) (new BiomeBase[BiomeStorage.a]), (abiomebase) -> {
|
|
||||||
+ private static final BiomeBase[] b = SystemUtils.a((new BiomeBase[BiomeStorage.a]), (abiomebase) -> { // Paper - decompile error
|
|
||||||
Arrays.fill(abiomebase, BiomeRegistry.a);
|
|
||||||
});
|
|
||||||
|
|
||||||
@@ -31,6 +31,11 @@ public class ChunkEmpty extends Chunk {
|
|
||||||
super(world, chunkcoordintpair, new BiomeStorage(world.r().b(IRegistry.ay), ChunkEmpty.b));
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ @Override public IBlockData getType(int x, int y, int z) {
|
|
||||||
+ return Blocks.VOID_AIR.getBlockData();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
@Override
|
|
||||||
public IBlockData getType(BlockPosition blockposition) {
|
|
||||||
return Blocks.VOID_AIR.getBlockData();
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/ChunkSection.java b/src/main/java/net/minecraft/world/level/chunk/ChunkSection.java
|
|
||||||
index a4e2eb1a753e8fcb48982d78fe80e505bce5c476..eea4a30428293eaf7afbe303a37adec60b44c2b4 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/ChunkSection.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/ChunkSection.java
|
|
||||||
@@ -13,10 +13,10 @@ public class ChunkSection {
|
|
||||||
|
|
||||||
public static final DataPalette<IBlockData> GLOBAL_PALETTE = new DataPaletteGlobal<>(Block.REGISTRY_ID, Blocks.AIR.getBlockData());
|
|
||||||
private final int yPos;
|
|
||||||
- private short nonEmptyBlockCount;
|
|
||||||
+ short nonEmptyBlockCount; // Paper - package-private
|
|
||||||
private short tickingBlockCount;
|
|
||||||
private short e;
|
|
||||||
- private final DataPaletteBlock<IBlockData> blockIds;
|
|
||||||
+ final DataPaletteBlock<IBlockData> blockIds; // Paper - package-private
|
|
||||||
|
|
||||||
public ChunkSection(int i) {
|
|
||||||
this(i, (short) 0, (short) 0, (short) 0);
|
|
||||||
@@ -30,8 +30,8 @@ public class ChunkSection {
|
|
||||||
this.blockIds = new DataPaletteBlock<>(ChunkSection.GLOBAL_PALETTE, Block.REGISTRY_ID, GameProfileSerializer::c, GameProfileSerializer::a, Blocks.AIR.getBlockData());
|
|
||||||
}
|
|
||||||
|
|
||||||
- public IBlockData getType(int i, int j, int k) {
|
|
||||||
- return (IBlockData) this.blockIds.a(i, j, k);
|
|
||||||
+ public final IBlockData getType(int i, int j, int k) { // Paper
|
|
||||||
+ return this.blockIds.a(j << 8 | k << 4 | i); // Paper - inline
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fluid b(int i, int j, int k) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/DataPaletteBlock.java b/src/main/java/net/minecraft/world/level/chunk/DataPaletteBlock.java
|
|
||||||
index e397b871b846c3a90bc75d0e1cf0683b6a3d0ca9..8928157b01bb4f0dfe043732777b33708c23cda7 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/DataPaletteBlock.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/DataPaletteBlock.java
|
|
||||||
@@ -133,7 +133,7 @@ public class DataPaletteBlock<T> implements DataPaletteExpandable<T> {
|
|
||||||
}
|
|
||||||
|
|
||||||
public T a(int i, int j, int k) {
|
|
||||||
- return this.a(b(i, j, k));
|
|
||||||
+ return this.a(j << 8 | k << 4 | i); // Paper - inline
|
|
||||||
}
|
|
||||||
|
|
||||||
protected T a(int i) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/IChunkAccess.java b/src/main/java/net/minecraft/world/level/chunk/IChunkAccess.java
|
|
||||||
index 2cd04abd72f1135446182ad6294003e526f99a4b..e570dc58efa56bd0aa5ada5575b4054ee38d505e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/IChunkAccess.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/IChunkAccess.java
|
|
||||||
@@ -25,6 +25,7 @@ import org.apache.logging.log4j.LogManager;
|
|
||||||
|
|
||||||
public interface IChunkAccess extends IBlockAccess, IStructureAccess {
|
|
||||||
|
|
||||||
+ IBlockData getType(final int x, final int y, final int z); // Paper
|
|
||||||
@Nullable
|
|
||||||
IBlockData setType(BlockPosition blockposition, IBlockData iblockdata, boolean flag);
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/ProtoChunk.java b/src/main/java/net/minecraft/world/level/chunk/ProtoChunk.java
|
|
||||||
index 7572ca53a5cca8ca5085d18c24048b85dda4daa9..9eeb99a21a6ed7f71ff64cf4cfdff646d31abbcf 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/ProtoChunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/ProtoChunk.java
|
|
||||||
@@ -113,16 +113,18 @@ public class ProtoChunk implements IChunkAccess {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public IBlockData getType(BlockPosition blockposition) {
|
|
||||||
- int i = blockposition.getY();
|
|
||||||
-
|
|
||||||
- if (World.b(i)) {
|
|
||||||
+ return getType(blockposition.getX(), blockposition.getY(), blockposition.getZ());
|
|
||||||
+ }
|
|
||||||
+ // Paper start
|
|
||||||
+ public IBlockData getType(final int x, final int y, final int z) {
|
|
||||||
+ if (y < 0 || y >= 256) {
|
|
||||||
return Blocks.VOID_AIR.getBlockData();
|
|
||||||
} else {
|
|
||||||
- ChunkSection chunksection = this.getSections()[i >> 4];
|
|
||||||
-
|
|
||||||
- return ChunkSection.a(chunksection) ? Blocks.AIR.getBlockData() : chunksection.getType(blockposition.getX() & 15, i & 15, blockposition.getZ() & 15);
|
|
||||||
+ ChunkSection chunksection = this.getSections()[y >> 4];
|
|
||||||
+ return chunksection == Chunk.EMPTY_CHUNK_SECTION || chunksection.c() ? Blocks.AIR.getBlockData() : chunksection.getType(x & 15, y & 15, z & 15);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Fluid getFluid(BlockPosition blockposition) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/ProtoChunkExtension.java b/src/main/java/net/minecraft/world/level/chunk/ProtoChunkExtension.java
|
|
||||||
index c059d3d055c35b492680556e8605966e2caaf7fd..9351e6ba541d440c485b6e4a3209170c5756e31e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/ProtoChunkExtension.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/ProtoChunkExtension.java
|
|
||||||
@@ -42,6 +42,11 @@ public class ProtoChunkExtension extends ProtoChunk {
|
|
||||||
public IBlockData getType(BlockPosition blockposition) {
|
|
||||||
return this.a.getType(blockposition);
|
|
||||||
}
|
|
||||||
+ // Paper start
|
|
||||||
+ public final IBlockData getType(final int x, final int y, final int z) {
|
|
||||||
+ return this.a.getBlockData(x, y, z);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Fluid getFluid(BlockPosition blockposition) {
|
|
@ -1,95 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 28 Mar 2016 19:55:45 -0400
|
|
||||||
Subject: [PATCH] Only process BlockPhysicsEvent if a plugin has a listener
|
|
||||||
|
|
||||||
Saves on some object allocation and processing when no plugin listens to this
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index ed71de473d461528d74ca5b95c33b97e98128aff..b141c744b8ffbc37b09cb4347c4051a77bb7049e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -1292,6 +1292,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
|
|
||||||
while (iterator.hasNext()) {
|
|
||||||
WorldServer worldserver = (WorldServer) iterator.next();
|
|
||||||
+ worldserver.hasPhysicsEvent = org.bukkit.event.block.BlockPhysicsEvent.getHandlerList().getRegisteredListeners().length > 0; // Paper
|
|
||||||
|
|
||||||
this.methodProfiler.a(() -> {
|
|
||||||
return worldserver + " " + worldserver.getDimensionKey().a();
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 5cd864e58b85fa163489557437f5c2eec9f008b7..a82affb9ffd0b2a513dcbf29402e99b49ed95d63 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -196,6 +196,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
private int tickPosition;
|
|
||||||
public final Convertable.ConversionSession convertable;
|
|
||||||
public final UUID uuid;
|
|
||||||
+ public boolean hasPhysicsEvent = true; // Paper
|
|
||||||
|
|
||||||
@Override public Chunk getChunkIfLoaded(int x, int z) { // Paper - this was added in world too but keeping here for NMS ABI
|
|
||||||
return this.chunkProvider.getChunkAt(x, z, false);
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index a22be13b097052b2a88707c9436b88c84298e46b..9236e480d21340d4295caa16dae34363e182f483 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -458,7 +458,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
// CraftBukkit start
|
|
||||||
iblockdata1.b(this, blockposition, k, j - 1); // Don't call an event for the old block to limit event spam
|
|
||||||
CraftWorld world = ((WorldServer) this).getWorld();
|
|
||||||
- if (world != null) {
|
|
||||||
+ if (world != null && ((WorldServer)this).hasPhysicsEvent) { // Paper
|
|
||||||
BlockPhysicsEvent event = new BlockPhysicsEvent(world.getBlockAt(blockposition.getX(), blockposition.getY(), blockposition.getZ()), CraftBlockData.fromData(iblockdata));
|
|
||||||
this.getServer().getPluginManager().callEvent(event);
|
|
||||||
|
|
||||||
@@ -560,7 +560,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
try {
|
|
||||||
// CraftBukkit start
|
|
||||||
CraftWorld world = ((WorldServer) this).getWorld();
|
|
||||||
- if (world != null) {
|
|
||||||
+ if (world != null && ((WorldServer)this).hasPhysicsEvent) { // Paper
|
|
||||||
BlockPhysicsEvent event = new BlockPhysicsEvent(world.getBlockAt(blockposition.getX(), blockposition.getY(), blockposition.getZ()), CraftBlockData.fromData(iblockdata), world.getBlockAt(blockposition1.getX(), blockposition1.getY(), blockposition1.getZ()));
|
|
||||||
this.getServer().getPluginManager().callEvent(event);
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockPlant.java b/src/main/java/net/minecraft/world/level/block/BlockPlant.java
|
|
||||||
index 33a5c5a4dc1478ab211dbb2e09df87570b06644f..97dfe5c5e3ea1d9691de87ffbf4b1a29a83a65b4 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockPlant.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockPlant.java
|
|
||||||
@@ -2,6 +2,7 @@ package net.minecraft.world.level.block;
|
|
||||||
|
|
||||||
import net.minecraft.core.BlockPosition;
|
|
||||||
import net.minecraft.core.EnumDirection;
|
|
||||||
+import net.minecraft.server.level.WorldServer;
|
|
||||||
import net.minecraft.world.level.GeneratorAccess;
|
|
||||||
import net.minecraft.world.level.IBlockAccess;
|
|
||||||
import net.minecraft.world.level.IWorldReader;
|
|
||||||
@@ -23,7 +24,7 @@ public class BlockPlant extends Block {
|
|
||||||
public IBlockData updateState(IBlockData iblockdata, EnumDirection enumdirection, IBlockData iblockdata1, GeneratorAccess generatoraccess, BlockPosition blockposition, BlockPosition blockposition1) {
|
|
||||||
// CraftBukkit start
|
|
||||||
if (!iblockdata.canPlace(generatoraccess, blockposition)) {
|
|
||||||
- if (!org.bukkit.craftbukkit.event.CraftEventFactory.callBlockPhysicsEvent(generatoraccess, blockposition).isCancelled()) {
|
|
||||||
+ if (!(generatoraccess instanceof WorldServer && ((WorldServer) generatoraccess).hasPhysicsEvent) || !org.bukkit.craftbukkit.event.CraftEventFactory.callBlockPhysicsEvent(generatoraccess, blockposition).isCancelled()) { // Paper
|
|
||||||
return Blocks.AIR.getBlockData();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockTallPlant.java b/src/main/java/net/minecraft/world/level/block/BlockTallPlant.java
|
|
||||||
index ca22187625f7ac6c43b663fd4d66cbf0c943c655..1a5d29ecc9edc52bac14ed5d05ef5376fd5b8a9c 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockTallPlant.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockTallPlant.java
|
|
||||||
@@ -3,6 +3,7 @@ package net.minecraft.world.level.block;
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
import net.minecraft.core.BlockPosition;
|
|
||||||
import net.minecraft.core.EnumDirection;
|
|
||||||
+import net.minecraft.server.level.WorldServer;
|
|
||||||
import net.minecraft.world.entity.EntityLiving;
|
|
||||||
import net.minecraft.world.entity.player.EntityHuman;
|
|
||||||
import net.minecraft.world.item.ItemStack;
|
|
||||||
@@ -83,7 +84,7 @@ public class BlockTallPlant extends BlockPlant {
|
|
||||||
|
|
||||||
protected static void b(World world, BlockPosition blockposition, IBlockData iblockdata, EntityHuman entityhuman) {
|
|
||||||
// CraftBukkit start
|
|
||||||
- if (org.bukkit.craftbukkit.event.CraftEventFactory.callBlockPhysicsEvent(world, blockposition).isCancelled()) {
|
|
||||||
+ if (((WorldServer)world).hasPhysicsEvent && org.bukkit.craftbukkit.event.CraftEventFactory.callBlockPhysicsEvent(world, blockposition).isCancelled()) { // Paper
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
@ -1,27 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 28 Mar 2016 20:32:58 -0400
|
|
||||||
Subject: [PATCH] Entity AddTo/RemoveFrom World Events
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index a82affb9ffd0b2a513dcbf29402e99b49ed95d63..24a45e5ac9b17feb528e9a047d1ad1761569ebfa 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -1214,7 +1214,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
if (entity instanceof EntityInsentient) {
|
|
||||||
this.navigators.remove(((EntityInsentient) entity).getNavigation());
|
|
||||||
}
|
|
||||||
-
|
|
||||||
+ new com.destroystokyo.paper.event.entity.EntityRemoveFromWorldEvent(entity.getBukkitEntity()).callEvent(); // Paper - fire while valid
|
|
||||||
entity.valid = false; // CraftBukkit
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1252,6 +1252,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
entity.origin = entity.getBukkitEntity().getLocation();
|
|
||||||
}
|
|
||||||
// Paper end
|
|
||||||
+ new com.destroystokyo.paper.event.entity.EntityAddToWorldEvent(entity.getBukkitEntity()).callEvent(); // Paper - fire while valid
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,44 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 28 Mar 2016 20:46:14 -0400
|
|
||||||
Subject: [PATCH] Configurable Chunk Inhabited Time
|
|
||||||
|
|
||||||
Vanilla stores how long a chunk has been active on a server, and dynamically scales some
|
|
||||||
aspects of vanilla gameplay to this factor.
|
|
||||||
|
|
||||||
For people who want all chunks to be treated equally, you can chose a fixed value.
|
|
||||||
|
|
||||||
This allows to fine-tune vanilla gameplay.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index cd64fb9d0c6d123e1c86cb33f12cd9cefc9f80d0..74ba5dbb83c13ce1721619b755036a7864a1fb90 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -238,4 +238,14 @@ public class PaperWorldConfig {
|
|
||||||
skeleHorseSpawnChance = 0.01D; // Vanilla value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int fixedInhabitedTime;
|
|
||||||
+ private void fixedInhabitedTime() {
|
|
||||||
+ if (PaperConfig.version < 16) {
|
|
||||||
+ if (!config.getBoolean("world-settings.default.use-chunk-inhabited-timer", true)) config.set("world-settings.default.fixed-chunk-inhabited-time", 0);
|
|
||||||
+ if (!config.getBoolean("world-settings." + worldName + ".use-chunk-inhabited-timer", true)) config.set("world-settings." + worldName + ".fixed-chunk-inhabited-time", 0);
|
|
||||||
+ set("use-chunk-inhabited-timer", null);
|
|
||||||
+ }
|
|
||||||
+ fixedInhabitedTime = getInt("fixed-chunk-inhabited-time", -1);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/Chunk.java b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
index fdc491f978560c394eec22116572585f9bbdec9f..b6898cd6e6117fef65198db32b98a64c806811d4 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/Chunk.java
|
|
||||||
@@ -1022,7 +1022,7 @@ public class Chunk implements IChunkAccess {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public long getInhabitedTime() {
|
|
||||||
- return this.inhabitedTime;
|
|
||||||
+ return world.paperConfig.fixedInhabitedTime < 0 ? this.inhabitedTime : world.paperConfig.fixedInhabitedTime; // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
@ -1,107 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 28 Mar 2016 21:22:26 -0400
|
|
||||||
Subject: [PATCH] EntityPathfindEvent
|
|
||||||
|
|
||||||
Fires when an Entity decides to start moving to a location.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/navigation/Navigation.java b/src/main/java/net/minecraft/world/entity/ai/navigation/Navigation.java
|
|
||||||
index 942e03578836524ba746bc37699677eb06cc7803..703d06b2b29f1500301d82df78dc377141085145 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/navigation/Navigation.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/navigation/Navigation.java
|
|
||||||
@@ -75,7 +75,7 @@ public class Navigation extends NavigationAbstract {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public PathEntity a(Entity entity, int i) {
|
|
||||||
- return this.a(entity.getChunkCoordinates(), i);
|
|
||||||
+ return this.a(entity.getChunkCoordinates(), entity, i); // Paper - Forward target entity
|
|
||||||
}
|
|
||||||
|
|
||||||
private int u() {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
index 8848a7552a0ef3944560a71f71620c6bd0f08c10..58225877ce4f2533c19d34e143ae374dc289bce5 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
@@ -10,6 +10,7 @@ import net.minecraft.core.BaseBlockPosition;
|
|
||||||
import net.minecraft.core.BlockPosition;
|
|
||||||
import net.minecraft.core.IPosition;
|
|
||||||
import net.minecraft.network.protocol.game.PacketDebug;
|
|
||||||
+import net.minecraft.server.MCUtil;
|
|
||||||
import net.minecraft.util.MathHelper;
|
|
||||||
import net.minecraft.world.entity.Entity;
|
|
||||||
import net.minecraft.world.entity.EntityInsentient;
|
|
||||||
@@ -28,7 +29,7 @@ import net.minecraft.world.phys.Vec3D;
|
|
||||||
|
|
||||||
public abstract class NavigationAbstract {
|
|
||||||
|
|
||||||
- protected final EntityInsentient a;
|
|
||||||
+ protected final EntityInsentient a; public Entity getEntity() { return a; } // Paper - OBFHELPER
|
|
||||||
protected final World b;
|
|
||||||
@Nullable
|
|
||||||
protected PathEntity c;
|
|
||||||
@@ -115,16 +116,26 @@ public abstract class NavigationAbstract {
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public PathEntity a(BlockPosition blockposition, int i) {
|
|
||||||
- return this.a(ImmutableSet.of(blockposition), 8, false, i);
|
|
||||||
+ // Paper start - add target parameter
|
|
||||||
+ return this.a(blockposition, null, i);
|
|
||||||
+ }
|
|
||||||
+ @Nullable public PathEntity a(BlockPosition blockposition, Entity target, int i) {
|
|
||||||
+ return this.a(ImmutableSet.of(blockposition), target, 8, false, i);
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public PathEntity a(Entity entity, int i) {
|
|
||||||
- return this.a(ImmutableSet.of(entity.getChunkCoordinates()), 16, true, i);
|
|
||||||
+ return this.a(ImmutableSet.of(entity.getChunkCoordinates()), entity, 16, true, i); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
+ // Paper start - Add target
|
|
||||||
protected PathEntity a(Set<BlockPosition> set, int i, boolean flag, int j) {
|
|
||||||
+ return this.a(set, null, i, flag, j);
|
|
||||||
+ }
|
|
||||||
+ @Nullable protected PathEntity a(Set<BlockPosition> set, Entity target, int i, boolean flag, int j) {
|
|
||||||
+ // Paper end
|
|
||||||
if (set.isEmpty()) {
|
|
||||||
return null;
|
|
||||||
} else if (this.a.locY() < 0.0D) {
|
|
||||||
@@ -134,6 +145,23 @@ public abstract class NavigationAbstract {
|
|
||||||
} else if (this.c != null && !this.c.c() && set.contains(this.p)) {
|
|
||||||
return this.c;
|
|
||||||
} else {
|
|
||||||
+ // Paper start - Pathfind event
|
|
||||||
+ boolean copiedSet = false;
|
|
||||||
+ for (BlockPosition possibleTarget : set) {
|
|
||||||
+ if (!new com.destroystokyo.paper.event.entity.EntityPathfindEvent(getEntity().getBukkitEntity(),
|
|
||||||
+ MCUtil.toLocation(getEntity().world, possibleTarget), target == null ? null : target.getBukkitEntity()).callEvent()) {
|
|
||||||
+ if (!copiedSet) {
|
|
||||||
+ copiedSet = true;
|
|
||||||
+ set = new java.util.HashSet<>(set);
|
|
||||||
+ }
|
|
||||||
+ // note: since we copy the set this remove call is safe, since we're iterating over the old copy
|
|
||||||
+ set.remove(possibleTarget);
|
|
||||||
+ if (set.isEmpty()) {
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
this.b.getMethodProfiler().enter("pathfind");
|
|
||||||
float f = (float) this.a.b(GenericAttributes.FOLLOW_RANGE);
|
|
||||||
BlockPosition blockposition = flag ? this.a.getChunkCoordinates().up() : this.a.getChunkCoordinates();
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationFlying.java b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationFlying.java
|
|
||||||
index 2e1efe7a048f64d494260d10a4ae5dba86af5e6c..f5664b8c0762f775f3cd106d156eb74b48bcedc2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationFlying.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationFlying.java
|
|
||||||
@@ -37,7 +37,7 @@ public class NavigationFlying extends NavigationAbstract {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public PathEntity a(Entity entity, int i) {
|
|
||||||
- return this.a(entity.getChunkCoordinates(), i);
|
|
||||||
+ return this.a(entity.getChunkCoordinates(), entity, i); // Paper - Forward target entity
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
@ -1,39 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Antony Riley <antony@cyberiantiger.org>
|
|
||||||
Date: Tue, 29 Mar 2016 08:22:55 +0300
|
|
||||||
Subject: [PATCH] Sanitise RegionFileCache and make configurable.
|
|
||||||
|
|
||||||
RegionFileCache prior to this patch would close every single open region
|
|
||||||
file upon reaching a size of 256.
|
|
||||||
This patch modifies that behaviour so it closes the the least recently
|
|
||||||
used RegionFile.
|
|
||||||
The implementation uses a LinkedHashMap as an LRU cache (modified from HashMap).
|
|
||||||
The maximum size of the RegionFileCache is also made configurable.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
index 716f285e67019b8a62922d09c15883c99f9421aa..439dcc6effdc91830d2b7ede9063982998b37120 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -227,4 +227,9 @@ public class PaperConfig {
|
|
||||||
private static void loadPermsBeforePlugins() {
|
|
||||||
loadPermsBeforePlugins = getBoolean("settings.load-permissions-yml-before-plugins", true);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public static int regionFileCacheSize = 256;
|
|
||||||
+ private static void regionFileCacheSize() {
|
|
||||||
+ regionFileCacheSize = Math.max(getInt("settings.region-file-cache-size", 256), 4);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
index 8310dd6bfc04b8ac0a51545baa3a264e6cb42eac..75b10a3755392870d8f5b51239a09a0e7fd75a42 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/chunk/storage/RegionFileCache.java
|
|
||||||
@@ -33,7 +33,7 @@ public final class RegionFileCache implements AutoCloseable {
|
|
||||||
if (regionfile != null) {
|
|
||||||
return regionfile;
|
|
||||||
} else {
|
|
||||||
- if (this.cache.size() >= 256) {
|
|
||||||
+ if (this.cache.size() >= com.destroystokyo.paper.PaperConfig.regionFileCacheSize) { // Paper - configurable
|
|
||||||
((RegionFile) this.cache.removeLast()).close();
|
|
||||||
}
|
|
||||||
|
|
@ -1,68 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 31 Mar 2016 19:17:58 -0400
|
|
||||||
Subject: [PATCH] Do not load chunks for Pathfinding
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
index 58225877ce4f2533c19d34e143ae374dc289bce5..d71a6e5991629ce59c8529d7cc8064960e385236 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/ai/navigation/NavigationAbstract.java
|
|
||||||
@@ -48,7 +48,7 @@ public abstract class NavigationAbstract {
|
|
||||||
private BlockPosition p;
|
|
||||||
private int q;
|
|
||||||
private float r;
|
|
||||||
- private final Pathfinder s;
|
|
||||||
+ private final Pathfinder s; public Pathfinder getPathfinder() { return this.s; } // Paper - OBFHELPER
|
|
||||||
private boolean t;
|
|
||||||
|
|
||||||
public NavigationAbstract(EntityInsentient entityinsentient, World world) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/pathfinder/Pathfinder.java b/src/main/java/net/minecraft/world/level/pathfinder/Pathfinder.java
|
|
||||||
index 39cd22a820fdc4c75aefb625b45b0c8c6ce1f199..5784be69098805e4d550a0923ac8daa5aada73f9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/pathfinder/Pathfinder.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/pathfinder/Pathfinder.java
|
|
||||||
@@ -20,7 +20,7 @@ public class Pathfinder {
|
|
||||||
|
|
||||||
private final PathPoint[] a = new PathPoint[32];
|
|
||||||
private final int b;
|
|
||||||
- private final PathfinderAbstract c;
|
|
||||||
+ private final PathfinderAbstract c; public PathfinderAbstract getPathfinder() { return this.c; } // Paper - OBFHELPER
|
|
||||||
private final Path d = new Path();
|
|
||||||
|
|
||||||
public Pathfinder(PathfinderAbstract pathfinderabstract, int i) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/pathfinder/PathfinderNormal.java b/src/main/java/net/minecraft/world/level/pathfinder/PathfinderNormal.java
|
|
||||||
index ed9c1dfbc84b9573784e6531186b3cd9513ddf75..d14f2800237c2a80912bf6f2d418a9ba9031070d 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/pathfinder/PathfinderNormal.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/pathfinder/PathfinderNormal.java
|
|
||||||
@@ -479,7 +479,12 @@ public class PathfinderNormal extends PathfinderAbstract {
|
|
||||||
for (int j1 = -1; j1 <= 1; ++j1) {
|
|
||||||
if (l != 0 || j1 != 0) {
|
|
||||||
blockposition_mutableblockposition.d(i + l, j + i1, k + j1);
|
|
||||||
- IBlockData iblockdata = iblockaccess.getType(blockposition_mutableblockposition);
|
|
||||||
+ // Paper start
|
|
||||||
+ IBlockData iblockdata = iblockaccess.getTypeIfLoaded(blockposition_mutableblockposition);
|
|
||||||
+ if (iblockdata == null) {
|
|
||||||
+ pathtype = PathType.BLOCKED;
|
|
||||||
+ } else {
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
if (iblockdata.a(Blocks.CACTUS)) {
|
|
||||||
return PathType.DANGER_CACTUS;
|
|
||||||
@@ -496,6 +501,7 @@ public class PathfinderNormal extends PathfinderAbstract {
|
|
||||||
if (iblockaccess.getFluid(blockposition_mutableblockposition).a((Tag) TagsFluid.WATER)) {
|
|
||||||
return PathType.WATER_BORDER;
|
|
||||||
}
|
|
||||||
+ } // Paper
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -505,7 +511,8 @@ public class PathfinderNormal extends PathfinderAbstract {
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static PathType b(IBlockAccess iblockaccess, BlockPosition blockposition) {
|
|
||||||
- IBlockData iblockdata = iblockaccess.getType(blockposition);
|
|
||||||
+ IBlockData iblockdata = iblockaccess.getTypeIfLoaded(blockposition); // Paper
|
|
||||||
+ if (iblockdata == null) return PathType.BLOCKED; // Paper
|
|
||||||
Block block = iblockdata.getBlock();
|
|
||||||
Material material = iblockdata.getMaterial();
|
|
||||||
|
|
@ -1,40 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Sat, 2 Apr 2016 05:09:16 -0400
|
|
||||||
Subject: [PATCH] Add PlayerUseUnknownEntityEvent
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/network/protocol/game/PacketPlayInUseEntity.java b/src/main/java/net/minecraft/network/protocol/game/PacketPlayInUseEntity.java
|
|
||||||
index 9f3f8568ef9484ba226deaa6429f819c325b7a26..ce63f3e5ac4d1a4311c0ebeb7574d999d45987d9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/network/protocol/game/PacketPlayInUseEntity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/network/protocol/game/PacketPlayInUseEntity.java
|
|
||||||
@@ -11,7 +11,7 @@ import net.minecraft.world.phys.Vec3D;
|
|
||||||
|
|
||||||
public class PacketPlayInUseEntity implements Packet<PacketListenerPlayIn> {
|
|
||||||
|
|
||||||
- private int a;
|
|
||||||
+ private int a; public int getEntityId() { return this.a; } // Paper - add accessor
|
|
||||||
private PacketPlayInUseEntity.EnumEntityUseAction action;
|
|
||||||
private Vec3D c;
|
|
||||||
private EnumHand d;
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/network/PlayerConnection.java b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
index af0a7dc9464e1acf1451f45464bab8546207950e..d3938e1fbab0a01ae5045858b1f421e041b768e2 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/network/PlayerConnection.java
|
|
||||||
@@ -2200,6 +2200,16 @@ public class PlayerConnection implements PacketListenerPlayIn {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
+ // Paper start - fire event
|
|
||||||
+ else {
|
|
||||||
+ this.server.getPluginManager().callEvent(new com.destroystokyo.paper.event.player.PlayerUseUnknownEntityEvent(
|
|
||||||
+ this.getPlayer(),
|
|
||||||
+ packetplayinuseentity.getEntityId(),
|
|
||||||
+ packetplayinuseentity.b() == PacketPlayInUseEntity.EnumEntityUseAction.ATTACK,
|
|
||||||
+ packetplayinuseentity.c() == EnumHand.MAIN_HAND ? EquipmentSlot.HAND : EquipmentSlot.OFF_HAND
|
|
||||||
+ ));
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Jedediah Smith <jedediah@silencegreys.com>
|
|
||||||
Date: Sat, 2 Apr 2016 20:37:03 -0400
|
|
||||||
Subject: [PATCH] Fix reducedDebugInfo not initialized on client
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/players/PlayerList.java b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
index 46c516b9ff089a3c885d635244942fd5a6ecf132..9d327f1255dd9d6b11840f7bb8ffc302bc33dbcf 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
@@ -246,6 +246,7 @@ public abstract class PlayerList {
|
|
||||||
playerconnection.sendPacket(new PacketPlayOutHeldItemSlot(entityplayer.inventory.itemInHandIndex));
|
|
||||||
playerconnection.sendPacket(new PacketPlayOutRecipeUpdate(this.server.getCraftingManager().b()));
|
|
||||||
playerconnection.sendPacket(new PacketPlayOutTags(this.server.getTagRegistry()));
|
|
||||||
+ playerconnection.sendPacket(new PacketPlayOutEntityStatus(entityplayer, (byte) (worldserver1.getGameRules().getBoolean(GameRules.REDUCED_DEBUG_INFO) ? 22 : 23))); // Paper - fix this rule not being initialized on the client
|
|
||||||
this.d(entityplayer);
|
|
||||||
entityplayer.getStatisticManager().c();
|
|
||||||
entityplayer.getRecipeBook().a(entityplayer);
|
|
@ -1,41 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 3 Apr 2016 16:28:17 -0400
|
|
||||||
Subject: [PATCH] Configurable Grass Spread Tick Rate
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 74ba5dbb83c13ce1721619b755036a7864a1fb90..db2dddd12f54e6d15916c4cee623676541de37fb 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -248,4 +248,10 @@ public class PaperWorldConfig {
|
|
||||||
}
|
|
||||||
fixedInhabitedTime = getInt("fixed-chunk-inhabited-time", -1);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public int grassUpdateRate = 1;
|
|
||||||
+ private void grassUpdateRate() {
|
|
||||||
+ grassUpdateRate = Math.max(0, getInt("grass-spread-tick-rate", grassUpdateRate));
|
|
||||||
+ log("Grass Spread Tick Rate: " + grassUpdateRate);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockDirtSnowSpreadable.java b/src/main/java/net/minecraft/world/level/block/BlockDirtSnowSpreadable.java
|
|
||||||
index a98392f06e66959ec1b75df8d2ecf3b5267980af..712596420af83e6e1b9d147ae2fd8d8a1f36e1b9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockDirtSnowSpreadable.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockDirtSnowSpreadable.java
|
|
||||||
@@ -3,6 +3,7 @@ package net.minecraft.world.level.block;
|
|
||||||
import java.util.Random;
|
|
||||||
import net.minecraft.core.BlockPosition;
|
|
||||||
import net.minecraft.core.EnumDirection;
|
|
||||||
+import net.minecraft.server.MinecraftServer;
|
|
||||||
import net.minecraft.server.level.WorldServer;
|
|
||||||
import net.minecraft.tags.Tag;
|
|
||||||
import net.minecraft.tags.TagsFluid;
|
|
||||||
@@ -41,6 +42,7 @@ public abstract class BlockDirtSnowSpreadable extends BlockDirtSnow {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void tick(IBlockData iblockdata, WorldServer worldserver, BlockPosition blockposition, Random random) {
|
|
||||||
+ if (this instanceof BlockGrass && worldserver.paperConfig.grassUpdateRate != 1 && (worldserver.paperConfig.grassUpdateRate < 1 || (MinecraftServer.currentTick + blockposition.hashCode()) % worldserver.paperConfig.grassUpdateRate != 0)) { return; } // Paper
|
|
||||||
if (!b(iblockdata, (IWorldReader) worldserver, blockposition)) {
|
|
||||||
// CraftBukkit start
|
|
||||||
if (org.bukkit.craftbukkit.event.CraftEventFactory.callBlockFadeEvent(worldserver, blockposition, Blocks.DIRT.getBlockData()).isCancelled()) {
|
|
@ -1,18 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 3 Apr 2016 17:48:50 -0400
|
|
||||||
Subject: [PATCH] Fix Cancelling BlockPlaceEvent triggering physics
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 9236e480d21340d4295caa16dae34363e182f483..8da8141c2320c0c1a9b95826a9be2dbe22e11c14 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -518,6 +518,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
public void b(BlockPosition blockposition, IBlockData iblockdata, IBlockData iblockdata1) {}
|
|
||||||
|
|
||||||
public void applyPhysics(BlockPosition blockposition, Block block) {
|
|
||||||
+ if (captureBlockStates) { return; } // Paper - Cancel all physics during placement
|
|
||||||
this.a(blockposition.west(), block, blockposition);
|
|
||||||
this.a(blockposition.east(), block, blockposition);
|
|
||||||
this.a(blockposition.down(), block, blockposition);
|
|
@ -1,84 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Tue, 5 Apr 2016 21:38:58 -0400
|
|
||||||
Subject: [PATCH] Optimize DataBits
|
|
||||||
|
|
||||||
Remove Debug checks as these are super hot and causing noticeable hits
|
|
||||||
|
|
||||||
Before: http://i.imgur.com/nQsMzAE.png
|
|
||||||
After: http://i.imgur.com/nJ46crB.png
|
|
||||||
|
|
||||||
Optimize redundant converting of static fields into an unsigned long each call by precomputing it in ctor
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/util/DataBits.java b/src/main/java/net/minecraft/util/DataBits.java
|
|
||||||
index 0c0576c8730069fb5364d8383dec8ab7e698658d..c4f3b680512fb15cea01ad12d0a00c6e60bf34b7 100644
|
|
||||||
--- a/src/main/java/net/minecraft/util/DataBits.java
|
|
||||||
+++ b/src/main/java/net/minecraft/util/DataBits.java
|
|
||||||
@@ -13,8 +13,8 @@ public class DataBits {
|
|
||||||
private final long d;
|
|
||||||
private final int e;
|
|
||||||
private final int f;
|
|
||||||
- private final int g;
|
|
||||||
- private final int h;
|
|
||||||
+ private final int g;private final long g_unsigned; // Paper - referenced in b(int) with 2 Integer.toUnsignedLong calls
|
|
||||||
+ private final int h;private final long h_unsigned; // Paper
|
|
||||||
private final int i;
|
|
||||||
|
|
||||||
public DataBits(int i, int j) {
|
|
||||||
@@ -29,8 +29,8 @@ public class DataBits {
|
|
||||||
this.f = (char) (64 / i);
|
|
||||||
int k = 3 * (this.f - 1);
|
|
||||||
|
|
||||||
- this.g = DataBits.a[k + 0];
|
|
||||||
- this.h = DataBits.a[k + 1];
|
|
||||||
+ this.g = DataBits.a[k + 0]; this.g_unsigned = Integer.toUnsignedLong(this.g); // Paper
|
|
||||||
+ this.h = DataBits.a[k + 1]; this.h_unsigned = Integer.toUnsignedLong(this.h); // Paper
|
|
||||||
this.i = DataBits.a[k + 2];
|
|
||||||
int l = (j + this.f - 1) / this.f;
|
|
||||||
|
|
||||||
@@ -47,15 +47,15 @@ public class DataBits {
|
|
||||||
}
|
|
||||||
|
|
||||||
private int b(int i) {
|
|
||||||
- long j = Integer.toUnsignedLong(this.g);
|
|
||||||
- long k = Integer.toUnsignedLong(this.h);
|
|
||||||
+ //long j = Integer.toUnsignedLong(this.g); // Paper
|
|
||||||
+ //long k = Integer.toUnsignedLong(this.h); // Paper
|
|
||||||
|
|
||||||
- return (int) ((long) i * j + k >> 32 >> this.i);
|
|
||||||
+ return (int) ((long) i * this.g_unsigned + this.h_unsigned >> 32 >> this.i); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
- public int a(int i, int j) {
|
|
||||||
- Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i);
|
|
||||||
- Validate.inclusiveBetween(0L, this.d, (long) j);
|
|
||||||
+ public final int a(int i, int j) { // Paper - make final for inline
|
|
||||||
+ //Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i); // Paper
|
|
||||||
+ //Validate.inclusiveBetween(0L, this.d, (long) j); // Paper
|
|
||||||
int k = this.b(i);
|
|
||||||
long l = this.b[k];
|
|
||||||
int i1 = (i - k * this.f) * this.c;
|
|
||||||
@@ -65,9 +65,9 @@ public class DataBits {
|
|
||||||
return j1;
|
|
||||||
}
|
|
||||||
|
|
||||||
- public void b(int i, int j) {
|
|
||||||
- Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i);
|
|
||||||
- Validate.inclusiveBetween(0L, this.d, (long) j);
|
|
||||||
+ public final void b(int i, int j) { // Paper - make final for inline
|
|
||||||
+ //Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i); // Paper
|
|
||||||
+ //Validate.inclusiveBetween(0L, this.d, (long) j); // Paper
|
|
||||||
int k = this.b(i);
|
|
||||||
long l = this.b[k];
|
|
||||||
int i1 = (i - k * this.f) * this.c;
|
|
||||||
@@ -75,8 +75,8 @@ public class DataBits {
|
|
||||||
this.b[k] = l & ~(this.d << i1) | ((long) j & this.d) << i1;
|
|
||||||
}
|
|
||||||
|
|
||||||
- public int a(int i) {
|
|
||||||
- Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i);
|
|
||||||
+ public final int a(int i) { // Paper - make final for inline
|
|
||||||
+ //Validate.inclusiveBetween(0L, (long) (this.e - 1), (long) i); // Paper
|
|
||||||
int j = this.b(i);
|
|
||||||
long k = this.b[j];
|
|
||||||
int l = (i - j * this.f) * this.c;
|
|
@ -1,66 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Wed, 6 Apr 2016 01:04:23 -0500
|
|
||||||
Subject: [PATCH] Option to use vanilla per-world scoreboard coloring on names
|
|
||||||
|
|
||||||
This change is basically a bandaid to fix CB's complete and utter lack
|
|
||||||
of support for vanilla scoreboard name modifications.
|
|
||||||
|
|
||||||
In the future, finding a way to merge the vanilla expectations in with
|
|
||||||
bukkit's concept of a display name would be preferable. There was a PR
|
|
||||||
for this on CB at one point but I can't find it. We may need to do this
|
|
||||||
ourselves at some point in the future.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index db2dddd12f54e6d15916c4cee623676541de37fb..1942f5224aaebb18adb591d6f70a419cfc1a7bdd 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -254,4 +254,9 @@ public class PaperWorldConfig {
|
|
||||||
grassUpdateRate = Math.max(0, getInt("grass-spread-tick-rate", grassUpdateRate));
|
|
||||||
log("Grass Spread Tick Rate: " + grassUpdateRate);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean useVanillaScoreboardColoring;
|
|
||||||
+ private void useVanillaScoreboardColoring() {
|
|
||||||
+ useVanillaScoreboardColoring = getBoolean("use-vanilla-world-scoreboard-name-coloring", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/io/papermc/paper/adventure/ChatProcessor.java b/src/main/java/io/papermc/paper/adventure/ChatProcessor.java
|
|
||||||
index 909968952a7ae2aa0196f12d1b3177cade380db2..1fcb01abc93c3c6ad172f209f55421d8b98629d5 100644
|
|
||||||
--- a/src/main/java/io/papermc/paper/adventure/ChatProcessor.java
|
|
||||||
+++ b/src/main/java/io/papermc/paper/adventure/ChatProcessor.java
|
|
||||||
@@ -17,7 +17,11 @@ import net.kyori.adventure.text.event.ClickEvent;
|
|
||||||
import net.minecraft.network.chat.IChatBaseComponent;
|
|
||||||
import net.minecraft.server.MinecraftServer;
|
|
||||||
import net.minecraft.server.level.EntityPlayer;
|
|
||||||
+import net.minecraft.world.scores.ScoreboardTeam;
|
|
||||||
+import net.minecraft.world.scores.ScoreboardTeamBase;
|
|
||||||
import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.ChatColor;
|
|
||||||
+import org.bukkit.craftbukkit.CraftWorld;
|
|
||||||
import org.bukkit.craftbukkit.entity.CraftPlayer;
|
|
||||||
import org.bukkit.craftbukkit.util.LazyPlayerSet;
|
|
||||||
import org.bukkit.craftbukkit.util.Waitable;
|
|
||||||
@@ -179,10 +183,22 @@ public final class ChatProcessor {
|
|
||||||
}
|
|
||||||
|
|
||||||
private static String legacyDisplayName(final CraftPlayer player) {
|
|
||||||
+ if (((CraftWorld) player.getWorld()).getHandle().paperConfig.useVanillaScoreboardColoring) {
|
|
||||||
+ final EntityPlayer ep = player.getHandle();
|
|
||||||
+ IChatBaseComponent name = ep.getDisplayName();
|
|
||||||
+ final ScoreboardTeamBase team = ep.getScoreboardTeam();
|
|
||||||
+ if (team != null) {
|
|
||||||
+ name = team.getFormattedName(name);
|
|
||||||
+ }
|
|
||||||
+ return PaperAdventure.LEGACY_SECTION_UXRC.serialize(PaperAdventure.asAdventure(name)) + ChatColor.RESET;
|
|
||||||
+ }
|
|
||||||
return player.getDisplayName();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Component displayName(final CraftPlayer player) {
|
|
||||||
+ if (((CraftWorld) player.getWorld()).getHandle().paperConfig.useVanillaScoreboardColoring) {
|
|
||||||
+ return PaperAdventure.asAdventure(ScoreboardTeam.a(player.getHandle().getScoreboardTeam(), player.getHandle().getDisplayName()));
|
|
||||||
+ }
|
|
||||||
return player.displayName();
|
|
||||||
}
|
|
||||||
|
|
@ -1,29 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Sun, 10 Apr 2016 03:23:32 -0500
|
|
||||||
Subject: [PATCH] Workaround for setting passengers on players
|
|
||||||
|
|
||||||
SPIGOT-1915 & GH-114
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index ab7e731cb04a75b829f0cd24d7b94f7ddad166e8..4bb57229c045956bab631982e12c0fc420db450e 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -872,6 +872,17 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Ugly workaround for SPIGOT-1915 & GH-114
|
|
||||||
+ @Override
|
|
||||||
+ public boolean setPassenger(org.bukkit.entity.Entity passenger) {
|
|
||||||
+ boolean wasSet = super.setPassenger(passenger);
|
|
||||||
+ if (wasSet) {
|
|
||||||
+ this.getHandle().playerConnection.sendPacket(new net.minecraft.network.protocol.game.PacketPlayOutMount(this.getHandle()));
|
|
||||||
+ }
|
|
||||||
+ return wasSet;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
@Override
|
|
||||||
public void setSneaking(boolean sneak) {
|
|
||||||
getHandle().setSneaking(sneak);
|
|
@ -1,90 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 13 Apr 2016 00:25:28 -0400
|
|
||||||
Subject: [PATCH] Remove unused World Tile Entity List
|
|
||||||
|
|
||||||
Massive hit to performance and it is completely unnecessary.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 24a45e5ac9b17feb528e9a047d1ad1761569ebfa..75424a5dd6db57dded3b6d895e6b5b102e91c77e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -1721,7 +1721,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
}
|
|
||||||
|
|
||||||
bufferedwriter.write(String.format("entities: %d\n", this.entitiesById.size()));
|
|
||||||
- bufferedwriter.write(String.format("block_entities: %d\n", this.tileEntityList.size()));
|
|
||||||
+ bufferedwriter.write(String.format("block_entities: %d\n", this.tileEntityListTick.size())); // Paper - remove unused list
|
|
||||||
bufferedwriter.write(String.format("block_ticks: %d\n", this.getBlockTickList().a()));
|
|
||||||
bufferedwriter.write(String.format("fluid_ticks: %d\n", this.getFluidTickList().a()));
|
|
||||||
bufferedwriter.write("distance_manager: " + playerchunkmap.e().c() + "\n");
|
|
||||||
@@ -1860,7 +1860,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
|
|
||||||
private void a(Writer writer) throws IOException {
|
|
||||||
CSVWriter csvwriter = CSVWriter.a().a("x").a("y").a("z").a("type").a(writer);
|
|
||||||
- Iterator iterator = this.tileEntityList.iterator();
|
|
||||||
+ Iterator iterator = this.tileEntityListTick.iterator(); // Paper - remove unused list
|
|
||||||
|
|
||||||
while (iterator.hasNext()) {
|
|
||||||
TileEntity tileentity = (TileEntity) iterator.next();
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/World.java b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
index 8da8141c2320c0c1a9b95826a9be2dbe22e11c14..cad86b0273c05767f78bcb3bdfaa9ea01e26af4e 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/World.java
|
|
||||||
@@ -91,7 +91,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
public static final ResourceKey<World> THE_NETHER = ResourceKey.a(IRegistry.L, new MinecraftKey("the_nether"));
|
|
||||||
public static final ResourceKey<World> THE_END = ResourceKey.a(IRegistry.L, new MinecraftKey("the_end"));
|
|
||||||
private static final EnumDirection[] a = EnumDirection.values();
|
|
||||||
- public final List<TileEntity> tileEntityList = Lists.newArrayList();
|
|
||||||
+ //public final List<TileEntity> tileEntityList = Lists.newArrayList(); // Paper - remove unused list
|
|
||||||
public final List<TileEntity> tileEntityListTick = Lists.newArrayList();
|
|
||||||
protected final List<TileEntity> tileEntityListPending = Lists.newArrayList();
|
|
||||||
protected final java.util.Set<TileEntity> tileEntityListUnload = com.google.common.collect.Sets.newHashSet();
|
|
||||||
@@ -683,9 +683,9 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
}, tileentity::getPosition});
|
|
||||||
}
|
|
||||||
|
|
||||||
- boolean flag = this.tileEntityList.add(tileentity);
|
|
||||||
+ boolean flag = true; // Paper - remove unused list
|
|
||||||
|
|
||||||
- if (flag && tileentity instanceof ITickable) {
|
|
||||||
+ if (flag && tileentity instanceof ITickable && !this.tileEntityListTick.contains(tileentity)) { // Paper
|
|
||||||
this.tileEntityListTick.add(tileentity);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -721,7 +721,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
timings.tileEntityTick.startTiming(); // Spigot
|
|
||||||
if (!this.tileEntityListUnload.isEmpty()) {
|
|
||||||
this.tileEntityListTick.removeAll(this.tileEntityListUnload);
|
|
||||||
- this.tileEntityList.removeAll(this.tileEntityListUnload);
|
|
||||||
+ //this.tileEntityList.removeAll(this.tileEntityListUnload); // Paper - remove unused list
|
|
||||||
this.tileEntityListUnload.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -781,7 +781,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
tilesThisCycle--;
|
|
||||||
this.tileEntityListTick.remove(tileTickPosition--);
|
|
||||||
// Spigot end
|
|
||||||
- this.tileEntityList.remove(tileentity);
|
|
||||||
+ //this.tileEntityList.remove(tileentity); // Paper - remove unused list
|
|
||||||
if (this.isLoaded(tileentity.getPosition())) {
|
|
||||||
this.getChunkAtWorldCoords(tileentity.getPosition()).removeTileEntity(tileentity.getPosition());
|
|
||||||
}
|
|
||||||
@@ -811,7 +811,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
this.notify(tileentity1.getPosition(), iblockdata, iblockdata, 3);
|
|
||||||
// CraftBukkit start
|
|
||||||
// From above, don't screw this up - SPIGOT-1746
|
|
||||||
- if (!this.tileEntityList.contains(tileentity1)) {
|
|
||||||
+ if (true) { // Paper - remove unused list
|
|
||||||
this.a(tileentity1);
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
@@ -957,7 +957,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
|
||||||
} else {
|
|
||||||
if (tileentity != null) {
|
|
||||||
this.tileEntityListPending.remove(tileentity);
|
|
||||||
- this.tileEntityList.remove(tileentity);
|
|
||||||
+ //this.tileEntityList.remove(tileentity); // Paper - remove unused list
|
|
||||||
this.tileEntityListTick.remove(tileentity);
|
|
||||||
}
|
|
||||||
|
|
@ -1,28 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 13 Apr 2016 00:30:10 -0400
|
|
||||||
Subject: [PATCH] Don't tick Skulls - unused code
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntitySkull.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntitySkull.java
|
|
||||||
index 87a5f352c8a6336c65008d6e21a771fd6332773c..22217f24b4a87f10b6d5a3e37d23a1164af84ace 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntitySkull.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntitySkull.java
|
|
||||||
@@ -33,7 +33,7 @@ import net.minecraft.server.MinecraftServer;
|
|
||||||
import net.minecraft.world.entity.player.EntityHuman;
|
|
||||||
// Spigot end
|
|
||||||
|
|
||||||
-public class TileEntitySkull extends TileEntity implements ITickable {
|
|
||||||
+public class TileEntitySkull extends TileEntity /*implements ITickable*/ { // Paper - remove tickable
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
private static UserCache userCache;
|
|
||||||
@@ -136,7 +136,7 @@ public class TileEntitySkull extends TileEntity implements ITickable {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Override
|
|
||||||
+ // Paper - remove override
|
|
||||||
public void tick() {
|
|
||||||
IBlockData iblockdata = this.getBlock();
|
|
||||||
|
|
@ -1,131 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Wed, 13 Apr 2016 02:10:49 -0400
|
|
||||||
Subject: [PATCH] Configurable Player Collision
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
index 439dcc6effdc91830d2b7ede9063982998b37120..504efea7b6f50a0d17f4f353781953dfb18bdeca 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -232,4 +232,9 @@ public class PaperConfig {
|
|
||||||
private static void regionFileCacheSize() {
|
|
||||||
regionFileCacheSize = Math.max(getInt("settings.region-file-cache-size", 256), 4);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public static boolean enablePlayerCollisions = true;
|
|
||||||
+ private static void enablePlayerCollisions() {
|
|
||||||
+ enablePlayerCollisions = getBoolean("settings.enable-player-collisions", true);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutScoreboardTeam.java b/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutScoreboardTeam.java
|
|
||||||
index bc40f2cbe1645fd60c4cee106b90f17cd043d32d..c1bb5c325286119891e8d68ce8f7328c99edb486 100644
|
|
||||||
--- a/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutScoreboardTeam.java
|
|
||||||
+++ b/src/main/java/net/minecraft/network/protocol/game/PacketPlayOutScoreboardTeam.java
|
|
||||||
@@ -112,7 +112,7 @@ public class PacketPlayOutScoreboardTeam implements Packet<PacketListenerPlayOut
|
|
||||||
packetdataserializer.a(this.b);
|
|
||||||
packetdataserializer.writeByte(this.j);
|
|
||||||
packetdataserializer.a(this.e);
|
|
||||||
- packetdataserializer.a(this.f);
|
|
||||||
+ packetdataserializer.a(!com.destroystokyo.paper.PaperConfig.enablePlayerCollisions ? "never" : this.f); // Paper
|
|
||||||
packetdataserializer.a((Enum) this.g);
|
|
||||||
packetdataserializer.a(this.c);
|
|
||||||
packetdataserializer.a(this.d);
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index b141c744b8ffbc37b09cb4347c4051a77bb7049e..8df60cedc1198916dfce8fcea7ca4a49f98429ba 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -151,6 +151,7 @@ import net.minecraft.world.phys.Vec2F;
|
|
||||||
import net.minecraft.world.phys.Vec3D;
|
|
||||||
import net.minecraft.world.scores.PersistentScoreboard;
|
|
||||||
import net.minecraft.world.scores.Scoreboard;
|
|
||||||
+import net.minecraft.world.scores.ScoreboardTeam;
|
|
||||||
import org.apache.commons.lang3.Validate;
|
|
||||||
import org.apache.logging.log4j.LogManager;
|
|
||||||
import org.apache.logging.log4j.Logger;
|
|
||||||
@@ -551,6 +552,20 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
this.server.getPluginManager().callEvent(new org.bukkit.event.world.WorldLoadEvent(worldserver.getWorld()));
|
|
||||||
}
|
|
||||||
|
|
||||||
+ // Paper start - Handle collideRule team for player collision toggle
|
|
||||||
+ final Scoreboard scoreboard = this.getScoreboard();
|
|
||||||
+ final java.util.Collection<String> toRemove = scoreboard.getTeams().stream().filter(team -> team.getName().startsWith("collideRule_")).map(ScoreboardTeam::getName).collect(java.util.stream.Collectors.toList());
|
|
||||||
+ for (String teamName : toRemove) {
|
|
||||||
+ scoreboard.removeTeam(scoreboard.getTeam(teamName)); // Clean up after ourselves
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!com.destroystokyo.paper.PaperConfig.enablePlayerCollisions) {
|
|
||||||
+ this.getPlayerList().collideRuleTeamName = org.apache.commons.lang3.StringUtils.left("collideRule_" + java.util.concurrent.ThreadLocalRandom.current().nextInt(), 16);
|
|
||||||
+ ScoreboardTeam collideTeam = scoreboard.createTeam(this.getPlayerList().collideRuleTeamName);
|
|
||||||
+ collideTeam.setCanSeeFriendlyInvisibles(false); // Because we want to mimic them not being on a team at all
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
this.server.enablePlugins(org.bukkit.plugin.PluginLoadOrder.POSTWORLD);
|
|
||||||
this.server.getPluginManager().callEvent(new ServerLoadEvent(ServerLoadEvent.LoadType.STARTUP));
|
|
||||||
this.serverConnection.acceptConnections();
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/players/PlayerList.java b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
index 9d327f1255dd9d6b11840f7bb8ffc302bc33dbcf..49f1aed92fd5fa46b74a979f317a2eb0672991e9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/players/PlayerList.java
|
|
||||||
@@ -86,6 +86,7 @@ import net.minecraft.world.level.storage.SavedFile;
|
|
||||||
import net.minecraft.world.level.storage.WorldData;
|
|
||||||
import net.minecraft.world.level.storage.WorldNBTStorage;
|
|
||||||
import net.minecraft.world.phys.Vec3D;
|
|
||||||
+import net.minecraft.world.scores.Scoreboard;
|
|
||||||
import net.minecraft.world.scores.ScoreboardObjective;
|
|
||||||
import net.minecraft.world.scores.ScoreboardTeam;
|
|
||||||
import net.minecraft.world.scores.ScoreboardTeamBase;
|
|
||||||
@@ -145,6 +146,7 @@ public abstract class PlayerList {
|
|
||||||
// CraftBukkit start
|
|
||||||
private CraftServer cserver;
|
|
||||||
private final Map<String,EntityPlayer> playersByName = new java.util.HashMap<>();
|
|
||||||
+ public @Nullable String collideRuleTeamName; // Paper - Team name used for collideRule
|
|
||||||
|
|
||||||
public PlayerList(MinecraftServer minecraftserver, IRegistryCustom.Dimension iregistrycustom_dimension, WorldNBTStorage worldnbtstorage, int i) {
|
|
||||||
this.cserver = minecraftserver.server = new CraftServer((DedicatedServer) minecraftserver, this);
|
|
||||||
@@ -376,6 +378,13 @@ public abstract class PlayerList {
|
|
||||||
}
|
|
||||||
|
|
||||||
entityplayer.syncInventory();
|
|
||||||
+ // Paper start - Add to collideRule team if needed
|
|
||||||
+ final Scoreboard scoreboard = this.getServer().getWorldServer(World.OVERWORLD).getScoreboard();
|
|
||||||
+ final ScoreboardTeam collideRuleTeam = scoreboard.getTeam(collideRuleTeamName);
|
|
||||||
+ if (this.collideRuleTeamName != null && collideRuleTeam != null && entityplayer.getScoreboardTeam() == null) {
|
|
||||||
+ scoreboard.addPlayerToTeam(entityplayer.getName(), collideRuleTeam);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
// CraftBukkit - Moved from above, added world
|
|
||||||
PlayerList.LOGGER.info("{}[{}] logged in with entity id {} at ([{}]{}, {}, {})", entityplayer.getDisplayName().getString(), s1, entityplayer.getId(), worldserver1.worldDataServer.getName(), entityplayer.locX(), entityplayer.locY(), entityplayer.locZ());
|
|
||||||
}
|
|
||||||
@@ -496,6 +505,16 @@ public abstract class PlayerList {
|
|
||||||
entityplayer.playerTick(); // SPIGOT-924
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
+ // Paper start - Remove from collideRule team if needed
|
|
||||||
+ if (this.collideRuleTeamName != null) {
|
|
||||||
+ final Scoreboard scoreBoard = this.server.getWorldServer(World.OVERWORLD).getScoreboard();
|
|
||||||
+ final ScoreboardTeam team = scoreBoard.getTeam(this.collideRuleTeamName);
|
|
||||||
+ if (entityplayer.getScoreboardTeam() == team && team != null) {
|
|
||||||
+ scoreBoard.removePlayerFromTeam(entityplayer.getName(), team);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
+
|
|
||||||
this.savePlayerFile(entityplayer);
|
|
||||||
if (entityplayer.isPassenger()) {
|
|
||||||
Entity entity = entityplayer.getRootVehicle();
|
|
||||||
@@ -1144,6 +1163,13 @@ public abstract class PlayerList {
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
||||||
+ // Paper start - Remove collideRule team if it exists
|
|
||||||
+ if (this.collideRuleTeamName != null) {
|
|
||||||
+ final Scoreboard scoreboard = this.getServer().getWorldServer(World.OVERWORLD).getScoreboard();
|
|
||||||
+ final ScoreboardTeam team = scoreboard.getTeam(this.collideRuleTeamName);
|
|
||||||
+ if (team != null) scoreboard.removeTeam(team);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
@ -1,57 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: kashike <kashike@vq.lc>
|
|
||||||
Date: Wed, 13 Apr 2016 20:21:38 -0700
|
|
||||||
Subject: [PATCH] Add handshake event to allow plugins to handle client
|
|
||||||
handshaking logic themselves
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/network/HandshakeListener.java b/src/main/java/net/minecraft/server/network/HandshakeListener.java
|
|
||||||
index 9b24afa4f4fe41d2261b16aaecec2144ac7d049f..1164782be686c91379494b236ebded817a002a6f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/network/HandshakeListener.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/network/HandshakeListener.java
|
|
||||||
@@ -29,7 +29,7 @@ public class HandshakeListener implements PacketHandshakingInListener {
|
|
||||||
// CraftBukkit end
|
|
||||||
private static final IChatBaseComponent a = new ChatComponentText("Ignoring status request");
|
|
||||||
private final MinecraftServer b;
|
|
||||||
- private final NetworkManager c;
|
|
||||||
+ private final NetworkManager c; final NetworkManager getNetworkManager() { return this.c; } // Paper - OBFHELPER
|
|
||||||
|
|
||||||
public HandshakeListener(MinecraftServer minecraftserver, NetworkManager networkmanager) {
|
|
||||||
this.b = minecraftserver;
|
|
||||||
@@ -88,8 +88,35 @@ public class HandshakeListener implements PacketHandshakingInListener {
|
|
||||||
this.c.close(chatmessage);
|
|
||||||
} else {
|
|
||||||
this.c.setPacketListener(new LoginListener(this.b, this.c));
|
|
||||||
+ // Paper start - handshake event
|
|
||||||
+ boolean proxyLogicEnabled = org.spigotmc.SpigotConfig.bungee;
|
|
||||||
+ boolean handledByEvent = false;
|
|
||||||
+ // Try and handle the handshake through the event
|
|
||||||
+ if (com.destroystokyo.paper.event.player.PlayerHandshakeEvent.getHandlerList().getRegisteredListeners().length != 0) { // Hello? Can you hear me?
|
|
||||||
+ java.net.SocketAddress socketAddress = this.getNetworkManager().socketAddress;
|
|
||||||
+ String hostnameOfRemote = socketAddress instanceof java.net.InetSocketAddress ? ((java.net.InetSocketAddress) socketAddress).getHostString() : InetAddress.getLoopbackAddress().getHostAddress();
|
|
||||||
+ com.destroystokyo.paper.event.player.PlayerHandshakeEvent event = new com.destroystokyo.paper.event.player.PlayerHandshakeEvent(packethandshakinginsetprotocol.hostname, hostnameOfRemote, !proxyLogicEnabled);
|
|
||||||
+ if (event.callEvent()) {
|
|
||||||
+ // If we've failed somehow, let the client know so and go no further.
|
|
||||||
+ if (event.isFailed()) {
|
|
||||||
+ chatmessage = new ChatMessage(event.getFailMessage());
|
|
||||||
+ this.getNetworkManager().sendPacket(new PacketLoginOutDisconnect(chatmessage));
|
|
||||||
+ this.getNetworkManager().close(chatmessage);
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (event.getServerHostname() != null) packethandshakinginsetprotocol.hostname = event.getServerHostname();
|
|
||||||
+ if (event.getSocketAddressHostname() != null) this.getNetworkManager().socketAddress = new java.net.InetSocketAddress(event.getSocketAddressHostname(), socketAddress instanceof java.net.InetSocketAddress ? ((java.net.InetSocketAddress) socketAddress).getPort() : 0);
|
|
||||||
+ this.getNetworkManager().spoofedUUID = event.getUniqueId();
|
|
||||||
+ this.getNetworkManager().spoofedProfile = gson.fromJson(event.getPropertiesJson(), com.mojang.authlib.properties.Property[].class);
|
|
||||||
+ handledByEvent = true; // Hooray, we did it!
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Don't try and handle default logic if it's been handled by the event.
|
|
||||||
+ if (!handledByEvent && proxyLogicEnabled) {
|
|
||||||
+ // Paper end
|
|
||||||
// Spigot Start
|
|
||||||
- if (org.spigotmc.SpigotConfig.bungee) {
|
|
||||||
+ //if (org.spigotmc.SpigotConfig.bungee) { // Paper - comment out, we check above!
|
|
||||||
String[] split = packethandshakinginsetprotocol.hostname.split("\00");
|
|
||||||
if ( ( split.length == 3 || split.length == 4 ) && ( HOST_PATTERN.matcher( split[1] ).matches() ) ) {
|
|
||||||
packethandshakinginsetprotocol.hostname = split[0];
|
|
@ -1,59 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sat, 16 Apr 2016 00:39:33 -0400
|
|
||||||
Subject: [PATCH] Configurable RCON IP address
|
|
||||||
|
|
||||||
For servers with multiple IP's, ability to bind to a specific interface.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/dedicated/DedicatedServerProperties.java b/src/main/java/net/minecraft/server/dedicated/DedicatedServerProperties.java
|
|
||||||
index 0ecff9f5e2ba444b196d80da341ff851dd5ce26f..b7cf02301c02ed0a6b696384e656426762ae2105 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/dedicated/DedicatedServerProperties.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/dedicated/DedicatedServerProperties.java
|
|
||||||
@@ -67,6 +67,8 @@ public class DedicatedServerProperties extends PropertyManager<DedicatedServerPr
|
|
||||||
public final PropertyManager<DedicatedServerProperties>.EditableProperty<Boolean> whiteList;
|
|
||||||
public final GeneratorSettings generatorSettings;
|
|
||||||
|
|
||||||
+ public final String rconIp; // Paper - Add rcon ip
|
|
||||||
+
|
|
||||||
// CraftBukkit start
|
|
||||||
public DedicatedServerProperties(Properties properties, IRegistryCustom iregistrycustom, OptionSet optionset) {
|
|
||||||
super(properties, optionset);
|
|
||||||
@@ -118,6 +120,10 @@ public class DedicatedServerProperties extends PropertyManager<DedicatedServerPr
|
|
||||||
this.textFilteringConfig = this.getString("text-filtering-config", "");
|
|
||||||
this.playerIdleTimeout = this.b("player-idle-timeout", 0);
|
|
||||||
this.whiteList = this.b("white-list", false);
|
|
||||||
+ // Paper start - Configurable rcon ip
|
|
||||||
+ final String rconIp = this.getSettingIfExists("rcon.ip");
|
|
||||||
+ this.rconIp = rconIp == null ? this.serverIp : rconIp;
|
|
||||||
+ // Paper end
|
|
||||||
this.generatorSettings = GeneratorSettings.a(iregistrycustom, properties);
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/dedicated/PropertyManager.java b/src/main/java/net/minecraft/server/dedicated/PropertyManager.java
|
|
||||||
index cb85d8bc1b81f9e5fa95c4dfeffbe689667f56a2..03dd6dc3a1aa8d1be6990c750f96e8a280bc1f25 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/dedicated/PropertyManager.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/dedicated/PropertyManager.java
|
|
||||||
@@ -129,8 +129,8 @@ public abstract class PropertyManager<T extends PropertyManager<T>> {
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
- @Nullable
|
|
||||||
- private String c(String s) {
|
|
||||||
+ @Nullable String getSettingIfExists(final String path) { return this.c(path); } // Paper - OBFHELPER
|
|
||||||
+ @Nullable private String c(String s) { // Paper - OBFHELPER
|
|
||||||
return (String) getOverride(s, this.properties.getProperty(s)); // CraftBukkit
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/rcon/thread/RemoteControlListener.java b/src/main/java/net/minecraft/server/rcon/thread/RemoteControlListener.java
|
|
||||||
index 797a450a08da1b799e32fae2a71a7a50bb90d127..3b3e21d1d86629d6c5e06108e53d1c5e807074d8 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/rcon/thread/RemoteControlListener.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/rcon/thread/RemoteControlListener.java
|
|
||||||
@@ -62,7 +62,7 @@ public class RemoteControlListener extends RemoteConnectionThread {
|
|
||||||
@Nullable
|
|
||||||
public static RemoteControlListener a(IMinecraftServer iminecraftserver) {
|
|
||||||
DedicatedServerProperties dedicatedserverproperties = iminecraftserver.getDedicatedServerProperties();
|
|
||||||
- String s = iminecraftserver.h_();
|
|
||||||
+ String s = dedicatedserverproperties.rconIp; // Paper - Configurable rcon ip
|
|
||||||
|
|
||||||
if (s.isEmpty()) {
|
|
||||||
s = "0.0.0.0";
|
|
@ -1,82 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 17 Apr 2016 17:27:09 -0400
|
|
||||||
Subject: [PATCH] Prevent Fire from loading chunks & wrongly spread
|
|
||||||
|
|
||||||
This causes the nether to spam unload/reload chunks, plus overall
|
|
||||||
bad behavior.
|
|
||||||
|
|
||||||
This also stops fire from spreading to illegal locations.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockFire.java b/src/main/java/net/minecraft/world/level/block/BlockFire.java
|
|
||||||
index ee5400fd3e493e1f0518a9e47ddbc997e7a0fb92..c22fad0038fdb0769e23db782e3341206fbd80f9 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockFire.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockFire.java
|
|
||||||
@@ -135,7 +135,7 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
BlockStateBoolean blockstateboolean = (BlockStateBoolean) BlockFire.h.get(enumdirection);
|
|
||||||
|
|
||||||
if (blockstateboolean != null) {
|
|
||||||
- iblockdata1 = (IBlockData) iblockdata1.set(blockstateboolean, this.e(iblockaccess.getType(blockposition.shift(enumdirection))));
|
|
||||||
+ iblockdata1 = (IBlockData) iblockdata1.set(blockstateboolean, this.e(iblockaccess.getTypeIfLoaded(blockposition.shift(enumdirection)))); // Paper - prevent chunk loads
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -215,6 +215,7 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
}
|
|
||||||
|
|
||||||
blockposition_mutableblockposition.a((BaseBlockPosition) blockposition, l, j1, i1);
|
|
||||||
+ if (blockposition_mutableblockposition.isInvalidYLocation() || !worldserver.isLoaded(blockposition_mutableblockposition)) continue; // Paper
|
|
||||||
int l1 = this.a((IWorldReader) worldserver, (BlockPosition) blockposition_mutableblockposition);
|
|
||||||
|
|
||||||
if (l1 > 0) {
|
|
||||||
@@ -260,10 +261,16 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
}
|
|
||||||
|
|
||||||
private void trySpread(World world, BlockPosition blockposition, int i, Random random, int j, BlockPosition sourceposition) { // CraftBukkit add sourceposition
|
|
||||||
- int k = this.getBurnChance(world.getType(blockposition));
|
|
||||||
+ // Paper start
|
|
||||||
+ final IBlockData iblockdata = world.getTypeIfLoaded(blockposition);
|
|
||||||
+ if (iblockdata == null) {
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ int k = this.getBurnChance(iblockdata);
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
if (random.nextInt(i) < k) {
|
|
||||||
- IBlockData iblockdata = world.getType(blockposition);
|
|
||||||
+ //IBlockData iblockdata = world.getType(blockposition); // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
org.bukkit.block.Block theBlock = world.getWorld().getBlockAt(blockposition.getX(), blockposition.getY(), blockposition.getZ());
|
|
||||||
@@ -309,7 +316,7 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
for (int j = 0; j < i; ++j) {
|
|
||||||
EnumDirection enumdirection = aenumdirection[j];
|
|
||||||
|
|
||||||
- if (this.e(iblockaccess.getType(blockposition.shift(enumdirection)))) {
|
|
||||||
+ if (this.e(iblockaccess.getTypeIfLoaded(blockposition.shift(enumdirection)))) { // Paper - prevent chunk loads
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -327,7 +334,12 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
|
|
||||||
for (int k = 0; k < j; ++k) {
|
|
||||||
EnumDirection enumdirection = aenumdirection[k];
|
|
||||||
- IBlockData iblockdata = iworldreader.getType(blockposition.shift(enumdirection));
|
|
||||||
+ // Paper start
|
|
||||||
+ IBlockData iblockdata = iworldreader.getTypeIfLoaded(blockposition.shift(enumdirection));
|
|
||||||
+ if (iblockdata == null) {
|
|
||||||
+ continue;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
i = Math.max(this.getFlameChance(iblockdata), i);
|
|
||||||
}
|
|
||||||
@@ -338,7 +350,7 @@ public class BlockFire extends BlockFireAbstract {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected boolean e(IBlockData iblockdata) {
|
|
||||||
- return this.getFlameChance(iblockdata) > 0;
|
|
||||||
+ return iblockdata != null && this.getFlameChance(iblockdata) > 0; // Paper - iblockdata can be nullable if chunk is unloaded now
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
@ -1,48 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Isaac Moore <rmsy@me.com>
|
|
||||||
Date: Tue, 19 Apr 2016 14:09:31 -0500
|
|
||||||
Subject: [PATCH] Implement PlayerLocaleChangeEvent
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/EntityPlayer.java b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
index cda0e7f8f9a9d66ac4e5a3f52609a4271bf0c4b5..3f1f8f7fe9452f66b18a08ef480d50ef78ad3351 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/EntityPlayer.java
|
|
||||||
@@ -1695,7 +1695,7 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
- public String locale = "en_us"; // CraftBukkit - add, lowercase
|
|
||||||
+ public String locale = null; // CraftBukkit - add, lowercase // Paper - default to null
|
|
||||||
public java.util.Locale adventure$locale = java.util.Locale.US; // Paper
|
|
||||||
public void a(PacketPlayInSettings packetplayinsettings) {
|
|
||||||
// CraftBukkit start
|
|
||||||
@@ -1703,9 +1703,10 @@ public class EntityPlayer extends EntityHuman implements ICrafting {
|
|
||||||
PlayerChangedMainHandEvent event = new PlayerChangedMainHandEvent(getBukkitEntity(), getMainHand() == EnumMainHand.LEFT ? MainHand.LEFT : MainHand.RIGHT);
|
|
||||||
this.server.server.getPluginManager().callEvent(event);
|
|
||||||
}
|
|
||||||
- if (!this.locale.equals(packetplayinsettings.locale)) {
|
|
||||||
+ if (this.locale == null || !this.locale.equals(packetplayinsettings.locale)) { // Paper - check for null
|
|
||||||
PlayerLocaleChangeEvent event = new PlayerLocaleChangeEvent(getBukkitEntity(), packetplayinsettings.locale);
|
|
||||||
this.server.server.getPluginManager().callEvent(event);
|
|
||||||
+ new com.destroystokyo.paper.event.player.PlayerLocaleChangeEvent(this.getBukkitEntity(), this.locale, packetplayinsettings.locale).callEvent(); // Paper
|
|
||||||
}
|
|
||||||
this.locale = packetplayinsettings.locale;
|
|
||||||
// Paper start
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
index 4bb57229c045956bab631982e12c0fc420db450e..5698ef90bfeceec37eaf7f23361246ef125b3cd1 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftPlayer.java
|
|
||||||
@@ -1877,8 +1877,10 @@ public class CraftPlayer extends CraftHumanEntity implements Player {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getLocale() {
|
|
||||||
- return getHandle().locale;
|
|
||||||
-
|
|
||||||
+ // Paper start - Locale change event
|
|
||||||
+ final String locale = getHandle().locale;
|
|
||||||
+ return locale != null ? locale : "en_us";
|
|
||||||
+ // Paper end
|
|
||||||
}
|
|
||||||
|
|
||||||
// Paper start
|
|
@ -1,42 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Fri, 22 Apr 2016 01:43:11 -0500
|
|
||||||
Subject: [PATCH] EntityRegainHealthEvent isFastRegen API
|
|
||||||
|
|
||||||
Don't even get me started
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/EntityLiving.java b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
index 375284b7e321cb03a7a30aedea165ca7a2fd1091..db5c0d2da9ed9993ee83adccfa74e77f4e364f2a 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/EntityLiving.java
|
|
||||||
@@ -1130,10 +1130,16 @@ public abstract class EntityLiving extends Entity {
|
|
||||||
}
|
|
||||||
|
|
||||||
public void heal(float f, EntityRegainHealthEvent.RegainReason regainReason) {
|
|
||||||
+ // Paper start - Forward
|
|
||||||
+ heal(f, regainReason, false);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public void heal(float f, EntityRegainHealthEvent.RegainReason regainReason, boolean isFastRegen) {
|
|
||||||
+ // Paper end
|
|
||||||
float f1 = this.getHealth();
|
|
||||||
|
|
||||||
if (f1 > 0.0F) {
|
|
||||||
- EntityRegainHealthEvent event = new EntityRegainHealthEvent(this.getBukkitEntity(), f, regainReason);
|
|
||||||
+ EntityRegainHealthEvent event = new EntityRegainHealthEvent(this.getBukkitEntity(), f, regainReason, isFastRegen); // Paper
|
|
||||||
// Suppress during worldgen
|
|
||||||
if (this.valid) {
|
|
||||||
this.world.getServer().getPluginManager().callEvent(event);
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/food/FoodMetaData.java b/src/main/java/net/minecraft/world/food/FoodMetaData.java
|
|
||||||
index 7ed321acba6d46159f7d67b8d10a0a3e06ac88a9..e455b25b0809af15f6fde957121d0110da7eb08f 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/food/FoodMetaData.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/food/FoodMetaData.java
|
|
||||||
@@ -87,7 +87,7 @@ public class FoodMetaData {
|
|
||||||
if (this.foodTickTimer >= this.saturatedRegenRate) { // CraftBukkit
|
|
||||||
float f = Math.min(this.saturationLevel, 6.0F);
|
|
||||||
|
|
||||||
- entityhuman.heal(f / 6.0F, org.bukkit.event.entity.EntityRegainHealthEvent.RegainReason.SATIATED); // CraftBukkit - added RegainReason
|
|
||||||
+ entityhuman.heal(f / 6.0F, org.bukkit.event.entity.EntityRegainHealthEvent.RegainReason.SATIATED, true); // CraftBukkit - added RegainReason // Paper - This is fast regen
|
|
||||||
// this.a(f); CraftBukkit - EntityExhaustionEvent
|
|
||||||
entityhuman.applyExhaustion(f, org.bukkit.event.entity.EntityExhaustionEvent.ExhaustionReason.REGEN); // CraftBukkit - EntityExhaustionEvent
|
|
||||||
this.foodTickTimer = 0;
|
|
@ -1,52 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: kashike <kashike@vq.lc>
|
|
||||||
Date: Thu, 21 Apr 2016 23:51:55 -0700
|
|
||||||
Subject: [PATCH] Add ability to configure frosted_ice properties
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 1942f5224aaebb18adb591d6f70a419cfc1a7bdd..5baccb8d50c135ab20c38ffd0690f585514ce5af 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -259,4 +259,14 @@ public class PaperWorldConfig {
|
|
||||||
private void useVanillaScoreboardColoring() {
|
|
||||||
useVanillaScoreboardColoring = getBoolean("use-vanilla-world-scoreboard-name-coloring", false);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean frostedIceEnabled = true;
|
|
||||||
+ public int frostedIceDelayMin = 20;
|
|
||||||
+ public int frostedIceDelayMax = 40;
|
|
||||||
+ private void frostedIce() {
|
|
||||||
+ this.frostedIceEnabled = this.getBoolean("frosted-ice.enabled", this.frostedIceEnabled);
|
|
||||||
+ this.frostedIceDelayMin = this.getInt("frosted-ice.delay.min", this.frostedIceDelayMin);
|
|
||||||
+ this.frostedIceDelayMax = this.getInt("frosted-ice.delay.max", this.frostedIceDelayMax);
|
|
||||||
+ log("Frosted Ice: " + (this.frostedIceEnabled ? "enabled" : "disabled") + " / delay: min=" + this.frostedIceDelayMin + ", max=" + this.frostedIceDelayMax);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/BlockIceFrost.java b/src/main/java/net/minecraft/world/level/block/BlockIceFrost.java
|
|
||||||
index 7239a30bd4a5dc4ed09802eea8f7126485ebb635..e32e94868386ff06ff29254e6cc3bee9b446a293 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/BlockIceFrost.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/BlockIceFrost.java
|
|
||||||
@@ -30,6 +30,7 @@ public class BlockIceFrost extends BlockIce {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void tickAlways(IBlockData iblockdata, WorldServer worldserver, BlockPosition blockposition, Random random) {
|
|
||||||
+ if (!worldserver.paperConfig.frostedIceEnabled) return; // Paper - add ability to disable frosted ice
|
|
||||||
if ((random.nextInt(3) == 0 || this.a(worldserver, blockposition, 4)) && worldserver.getLightLevel(blockposition) > 11 - (Integer) iblockdata.get(BlockIceFrost.a) - iblockdata.b((IBlockAccess) worldserver, blockposition) && this.e(iblockdata, (World) worldserver, blockposition)) {
|
|
||||||
BlockPosition.MutableBlockPosition blockposition_mutableblockposition = new BlockPosition.MutableBlockPosition();
|
|
||||||
EnumDirection[] aenumdirection = EnumDirection.values();
|
|
||||||
@@ -42,12 +43,12 @@ public class BlockIceFrost extends BlockIce {
|
|
||||||
IBlockData iblockdata1 = worldserver.getType(blockposition_mutableblockposition);
|
|
||||||
|
|
||||||
if (iblockdata1.a((Block) this) && !this.e(iblockdata1, (World) worldserver, blockposition_mutableblockposition)) {
|
|
||||||
- worldserver.getBlockTickList().a(blockposition_mutableblockposition, this, MathHelper.nextInt(random, 20, 40));
|
|
||||||
+ worldserver.getBlockTickList().a(blockposition_mutableblockposition, this, MathHelper.nextInt(random, worldserver.paperConfig.frostedIceDelayMin, worldserver.paperConfig.frostedIceDelayMax)); // Paper - use configurable min/max delay
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
|
||||||
- worldserver.getBlockTickList().a(blockposition, this, MathHelper.nextInt(random, 20, 40));
|
|
||||||
+ worldserver.getBlockTickList().a(blockposition, this, MathHelper.nextInt(random, worldserver.paperConfig.frostedIceDelayMin, worldserver.paperConfig.frostedIceDelayMax)); // Paper - use configurable min/max delay
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,36 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Thu, 28 Apr 2016 00:57:27 -0400
|
|
||||||
Subject: [PATCH] remove null possibility for getServer singleton
|
|
||||||
|
|
||||||
to stop IDE complaining about potential NPE
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index 8df60cedc1198916dfce8fcea7ca4a49f98429ba..ed00ee8e56e6ca38a1ac689458c4675eff6e3eea 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -180,6 +180,7 @@ import org.spigotmc.SlackActivityAccountant; // Spigot
|
|
||||||
|
|
||||||
public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTask> implements IMojangStatistics, ICommandListener, AutoCloseable {
|
|
||||||
|
|
||||||
+ private static MinecraftServer SERVER; // Paper
|
|
||||||
public static final Logger LOGGER = LogManager.getLogger();
|
|
||||||
public static final File b = new File("usercache.json");
|
|
||||||
public static final WorldSettings c = new WorldSettings("Demo World", EnumGamemode.SURVIVAL, false, EnumDifficulty.NORMAL, false, new GameRules(), DataPackConfiguration.a);
|
|
||||||
@@ -286,6 +287,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
|
|
||||||
public MinecraftServer(OptionSet options, DataPackConfiguration datapackconfiguration, Thread thread, IRegistryCustom.Dimension iregistrycustom_dimension, Convertable.ConversionSession convertable_conversionsession, SaveData savedata, ResourcePackRepository resourcepackrepository, Proxy proxy, DataFixer datafixer, DataPackResources datapackresources, MinecraftSessionService minecraftsessionservice, GameProfileRepository gameprofilerepository, UserCache usercache, WorldLoadListenerFactory worldloadlistenerfactory) {
|
|
||||||
super("Server");
|
|
||||||
+ SERVER = this; // Paper - better singleton
|
|
||||||
this.m = new GameProfilerSwitcher(SystemUtils.a, this::ai);
|
|
||||||
this.methodProfiler = GameProfilerDisabled.a;
|
|
||||||
this.serverPing = new ServerPing();
|
|
||||||
@@ -2163,7 +2165,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
|
|
||||||
@Deprecated
|
|
||||||
public static MinecraftServer getServer() {
|
|
||||||
- return (Bukkit.getServer() instanceof CraftServer) ? ((CraftServer) Bukkit.getServer()).getServer() : null;
|
|
||||||
+ return SERVER; // Paper
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
|
|
@ -1,144 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Fri, 29 Apr 2016 20:02:00 -0400
|
|
||||||
Subject: [PATCH] Improve Maps (in item frames) performance and bug fixes
|
|
||||||
|
|
||||||
Maps used a modified version of rendering to support plugin controlled
|
|
||||||
imaging on maps. The Craft Map Renderer is much slower than Vanilla,
|
|
||||||
causing maps in item frames to cause a noticeable hit on server performance.
|
|
||||||
|
|
||||||
This updates the map system to not use the Craft system if we detect that no
|
|
||||||
custom renderers are in use, defaulting to the much simpler Vanilla system.
|
|
||||||
|
|
||||||
Additionally, numerous issues to player position tracking on maps has been fixed.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/level/WorldServer.java b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
index 75424a5dd6db57dded3b6d895e6b5b102e91c77e..c7652d6bff7630e2eefbb4c3b0deb6e17b9c98d0 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/level/WorldServer.java
|
|
||||||
@@ -1170,6 +1170,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
|
|
||||||
{
|
|
||||||
if ( iter.next().trackee == entity )
|
|
||||||
{
|
|
||||||
+ map.decorations.remove(entity.getDisplayName().getString()); // Paper
|
|
||||||
iter.remove();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/player/EntityHuman.java b/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
index f42e16589476c1bd10b13214dda5ac7bb3e52131..e3e3426a00128b56d523bb43a59b814b915ad0ff 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/player/EntityHuman.java
|
|
||||||
@@ -86,6 +86,7 @@ import net.minecraft.world.item.ItemElytra;
|
|
||||||
import net.minecraft.world.item.ItemProjectileWeapon;
|
|
||||||
import net.minecraft.world.item.ItemStack;
|
|
||||||
import net.minecraft.world.item.ItemSword;
|
|
||||||
+import net.minecraft.world.item.ItemWorldMap;
|
|
||||||
import net.minecraft.world.item.Items;
|
|
||||||
import net.minecraft.world.item.crafting.IRecipe;
|
|
||||||
import net.minecraft.world.item.enchantment.EnchantmentManager;
|
|
||||||
@@ -104,6 +105,7 @@ import net.minecraft.world.level.block.entity.TileEntitySign;
|
|
||||||
import net.minecraft.world.level.block.entity.TileEntityStructure;
|
|
||||||
import net.minecraft.world.level.block.state.IBlockData;
|
|
||||||
import net.minecraft.world.level.block.state.pattern.ShapeDetectorBlock;
|
|
||||||
+import net.minecraft.world.level.saveddata.maps.WorldMap;
|
|
||||||
import net.minecraft.world.phys.AxisAlignedBB;
|
|
||||||
import net.minecraft.world.phys.Vec3D;
|
|
||||||
import net.minecraft.world.scores.Scoreboard;
|
|
||||||
@@ -689,6 +691,12 @@ public abstract class EntityHuman extends EntityLiving {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
// CraftBukkit end
|
|
||||||
+ // Paper start - remove player from map on drop
|
|
||||||
+ if (itemstack.getItem() == Items.FILLED_MAP) {
|
|
||||||
+ WorldMap worldmap = ItemWorldMap.getSavedMap(itemstack, this.world);
|
|
||||||
+ worldmap.updateSeenPlayers(this, itemstack);
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
return entityitem;
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/saveddata/maps/WorldMap.java b/src/main/java/net/minecraft/world/level/saveddata/maps/WorldMap.java
|
|
||||||
index 3f057f0bd23bc1c693c8f04ee8acd6626c620008..d470af1814af332595c1c0beb1cdc552e186a6bb 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/saveddata/maps/WorldMap.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/saveddata/maps/WorldMap.java
|
|
||||||
@@ -57,6 +57,7 @@ public class WorldMap extends PersistentBase {
|
|
||||||
private final Map<String, MapIconBanner> m = Maps.newHashMap();
|
|
||||||
public final Map<String, MapIcon> decorations = Maps.newLinkedHashMap();
|
|
||||||
private final Map<String, WorldMapFrame> n = Maps.newHashMap();
|
|
||||||
+ private org.bukkit.craftbukkit.map.RenderData vanillaRender = new org.bukkit.craftbukkit.map.RenderData(); // Paper
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
public final CraftMapView mapView;
|
|
||||||
@@ -69,6 +70,7 @@ public class WorldMap extends PersistentBase {
|
|
||||||
// CraftBukkit start
|
|
||||||
mapView = new CraftMapView(this);
|
|
||||||
server = (CraftServer) org.bukkit.Bukkit.getServer();
|
|
||||||
+ vanillaRender.buffer = colors; // Paper
|
|
||||||
// CraftBukkit end
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -136,6 +138,7 @@ public class WorldMap extends PersistentBase {
|
|
||||||
this.m.put(mapiconbanner.f(), mapiconbanner);
|
|
||||||
this.a(mapiconbanner.c(), (GeneratorAccess) null, mapiconbanner.f(), (double) mapiconbanner.a().getX(), (double) mapiconbanner.a().getZ(), 180.0D, mapiconbanner.d());
|
|
||||||
}
|
|
||||||
+ this.vanillaRender.buffer = colors; // Paper
|
|
||||||
|
|
||||||
NBTTagList nbttaglist1 = nbttagcompound.getList("frames", 10);
|
|
||||||
|
|
||||||
@@ -216,6 +219,7 @@ public class WorldMap extends PersistentBase {
|
|
||||||
this.b();
|
|
||||||
}
|
|
||||||
|
|
||||||
+ public void updateSeenPlayers(EntityHuman entityhuman, ItemStack itemstack) { this.a(entityhuman, itemstack); } // Paper - OBFHELPER
|
|
||||||
public void a(EntityHuman entityhuman, ItemStack itemstack) {
|
|
||||||
if (!this.humans.containsKey(entityhuman)) {
|
|
||||||
WorldMap.WorldMapHumanTracker worldmap_worldmaphumantracker = new WorldMap.WorldMapHumanTracker(entityhuman);
|
|
||||||
@@ -451,6 +455,21 @@ public class WorldMap extends PersistentBase {
|
|
||||||
|
|
||||||
public class WorldMapHumanTracker {
|
|
||||||
|
|
||||||
+ // Paper start
|
|
||||||
+ private void addSeenPlayers(java.util.Collection<MapIcon> icons) {
|
|
||||||
+ org.bukkit.entity.Player player = (org.bukkit.entity.Player) trackee.getBukkitEntity();
|
|
||||||
+ WorldMap.this.decorations.forEach((name, mapIcon) -> {
|
|
||||||
+ // If this cursor is for a player check visibility with vanish system
|
|
||||||
+ org.bukkit.entity.Player other = org.bukkit.Bukkit.getPlayerExact(name); // Spigot
|
|
||||||
+ if (other == null || player.canSee(other)) {
|
|
||||||
+ icons.add(mapIcon);
|
|
||||||
+ }
|
|
||||||
+ });
|
|
||||||
+ }
|
|
||||||
+ private boolean shouldUseVanillaMap() {
|
|
||||||
+ return mapView.getRenderers().size() == 1 && mapView.getRenderers().get(0).getClass() == org.bukkit.craftbukkit.map.CraftMapRenderer.class;
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
public final EntityHuman trackee;
|
|
||||||
private boolean d = true;
|
|
||||||
private int e;
|
|
||||||
@@ -467,9 +486,12 @@ public class WorldMap extends PersistentBase {
|
|
||||||
@Nullable
|
|
||||||
public Packet<?> a(ItemStack itemstack) {
|
|
||||||
// CraftBukkit start
|
|
||||||
- org.bukkit.craftbukkit.map.RenderData render = WorldMap.this.mapView.render((org.bukkit.craftbukkit.entity.CraftPlayer) this.trackee.getBukkitEntity()); // CraftBukkit
|
|
||||||
+ if (!this.d && this.i % 5 != 0) { this.i++; return null; } // Paper - this won't end up sending, so don't render it!
|
|
||||||
+ boolean vanillaMaps = shouldUseVanillaMap(); // Paper
|
|
||||||
+ org.bukkit.craftbukkit.map.RenderData render = !vanillaMaps ? WorldMap.this.mapView.render((org.bukkit.craftbukkit.entity.CraftPlayer) this.trackee.getBukkitEntity()) : WorldMap.this.vanillaRender; // CraftBukkit // Paper
|
|
||||||
|
|
||||||
java.util.Collection<MapIcon> icons = new java.util.ArrayList<MapIcon>();
|
|
||||||
+ if (vanillaMaps) addSeenPlayers(icons); // Paper
|
|
||||||
|
|
||||||
for ( org.bukkit.map.MapCursor cursor : render.cursors) {
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/map/RenderData.java b/src/main/java/org/bukkit/craftbukkit/map/RenderData.java
|
|
||||||
index 256a131781721c86dd6cdbc329335964570cbe8c..5768cd512ec166f1e8d1f4a28792015347297c3f 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/map/RenderData.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/map/RenderData.java
|
|
||||||
@@ -5,7 +5,7 @@ import org.bukkit.map.MapCursor;
|
|
||||||
|
|
||||||
public class RenderData {
|
|
||||||
|
|
||||||
- public final byte[] buffer;
|
|
||||||
+ public byte[] buffer; // Paper
|
|
||||||
public final ArrayList<MapCursor> cursors;
|
|
||||||
|
|
||||||
public RenderData() {
|
|
@ -1,736 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sun, 1 May 2016 21:19:14 -0400
|
|
||||||
Subject: [PATCH] LootTable API & Replenishable Lootables Feature
|
|
||||||
|
|
||||||
Provides an API to control the loot table for an object.
|
|
||||||
Also provides a feature that any Lootable Inventory (Chests in Structures)
|
|
||||||
can automatically replenish after a given time.
|
|
||||||
|
|
||||||
This feature is good for long term worlds so that newer players
|
|
||||||
do not suffer with "Every chest has been looted"
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
index 5baccb8d50c135ab20c38ffd0690f585514ce5af..eb04fdb172a50ec1f5b7fe78fa0e7655246abd60 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
||||||
@@ -269,4 +269,26 @@ public class PaperWorldConfig {
|
|
||||||
this.frostedIceDelayMax = this.getInt("frosted-ice.delay.max", this.frostedIceDelayMax);
|
|
||||||
log("Frosted Ice: " + (this.frostedIceEnabled ? "enabled" : "disabled") + " / delay: min=" + this.frostedIceDelayMin + ", max=" + this.frostedIceDelayMax);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public boolean autoReplenishLootables;
|
|
||||||
+ public boolean restrictPlayerReloot;
|
|
||||||
+ public boolean changeLootTableSeedOnFill;
|
|
||||||
+ public int maxLootableRefills;
|
|
||||||
+ public int lootableRegenMin;
|
|
||||||
+ public int lootableRegenMax;
|
|
||||||
+ private void enhancedLootables() {
|
|
||||||
+ autoReplenishLootables = getBoolean("lootables.auto-replenish", false);
|
|
||||||
+ restrictPlayerReloot = getBoolean("lootables.restrict-player-reloot", true);
|
|
||||||
+ changeLootTableSeedOnFill = getBoolean("lootables.reset-seed-on-fill", true);
|
|
||||||
+ maxLootableRefills = getInt("lootables.max-refills", -1);
|
|
||||||
+ lootableRegenMin = PaperConfig.getSeconds(getString("lootables.refresh-min", "12h"));
|
|
||||||
+ lootableRegenMax = PaperConfig.getSeconds(getString("lootables.refresh-max", "2d"));
|
|
||||||
+ if (autoReplenishLootables) {
|
|
||||||
+ log("Lootables: Replenishing every " +
|
|
||||||
+ PaperConfig.timeSummary(lootableRegenMin) + " to " +
|
|
||||||
+ PaperConfig.timeSummary(lootableRegenMax) +
|
|
||||||
+ (restrictPlayerReloot ? " (restricting reloot)" : "")
|
|
||||||
+ );
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperLootableBlockInventory.java b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableBlockInventory.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..5dfc3c8008d64ad4ed71b4904c897f5005491349
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableBlockInventory.java
|
|
||||||
@@ -0,0 +1,33 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import net.minecraft.core.BlockPosition;
|
|
||||||
+import net.minecraft.world.level.World;
|
|
||||||
+import net.minecraft.world.level.block.entity.TileEntityLootable;
|
|
||||||
+import org.bukkit.Chunk;
|
|
||||||
+import org.bukkit.block.Block;
|
|
||||||
+
|
|
||||||
+public interface PaperLootableBlockInventory extends LootableBlockInventory, PaperLootableInventory {
|
|
||||||
+
|
|
||||||
+ TileEntityLootable getTileEntity();
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default LootableInventory getAPILootableInventory() {
|
|
||||||
+ return this;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default World getNMSWorld() {
|
|
||||||
+ return getTileEntity().getWorld();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ default Block getBlock() {
|
|
||||||
+ final BlockPosition position = getTileEntity().getPosition();
|
|
||||||
+ final Chunk bukkitChunk = getTileEntity().getWorld().getChunkAtWorldCoords(position).bukkitChunk;
|
|
||||||
+ return bukkitChunk.getBlock(position.getX(), position.getY(), position.getZ());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default PaperLootableInventoryData getLootableData() {
|
|
||||||
+ return getTileEntity().lootableData;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperLootableEntityInventory.java b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableEntityInventory.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..019a06fa2b43cacd3bbd4d58aba71b3728f37581
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableEntityInventory.java
|
|
||||||
@@ -0,0 +1,28 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import net.minecraft.world.level.World;
|
|
||||||
+import org.bukkit.entity.Entity;
|
|
||||||
+
|
|
||||||
+public interface PaperLootableEntityInventory extends LootableEntityInventory, PaperLootableInventory {
|
|
||||||
+
|
|
||||||
+ net.minecraft.world.entity.Entity getHandle();
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default LootableInventory getAPILootableInventory() {
|
|
||||||
+ return this;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ default Entity getEntity() {
|
|
||||||
+ return getHandle().getBukkitEntity();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default World getNMSWorld() {
|
|
||||||
+ return getHandle().getWorld();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default PaperLootableInventoryData getLootableData() {
|
|
||||||
+ return getHandle().lootableData;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventory.java b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventory.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..59e8aea749bbba079e3304d9a5854280db2692e9
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventory.java
|
|
||||||
@@ -0,0 +1,71 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import net.minecraft.world.level.World;
|
|
||||||
+import org.bukkit.loot.Lootable;
|
|
||||||
+
|
|
||||||
+import java.util.UUID;
|
|
||||||
+
|
|
||||||
+public interface PaperLootableInventory extends LootableInventory, Lootable {
|
|
||||||
+
|
|
||||||
+ PaperLootableInventoryData getLootableData();
|
|
||||||
+ LootableInventory getAPILootableInventory();
|
|
||||||
+
|
|
||||||
+ World getNMSWorld();
|
|
||||||
+
|
|
||||||
+ default org.bukkit.World getBukkitWorld() {
|
|
||||||
+ return getNMSWorld().getWorld();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default boolean isRefillEnabled() {
|
|
||||||
+ return getNMSWorld().paperConfig.autoReplenishLootables;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default boolean hasBeenFilled() {
|
|
||||||
+ return getLastFilled() != -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default boolean hasPlayerLooted(UUID player) {
|
|
||||||
+ return getLootableData().hasPlayerLooted(player);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default Long getLastLooted(UUID player) {
|
|
||||||
+ return getLootableData().getLastLooted(player);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default boolean setHasPlayerLooted(UUID player, boolean looted) {
|
|
||||||
+ final boolean hasLooted = hasPlayerLooted(player);
|
|
||||||
+ if (hasLooted != looted) {
|
|
||||||
+ getLootableData().setPlayerLootedState(player, looted);
|
|
||||||
+ }
|
|
||||||
+ return hasLooted;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default boolean hasPendingRefill() {
|
|
||||||
+ long nextRefill = getLootableData().getNextRefill();
|
|
||||||
+ return nextRefill != -1 && nextRefill > getLootableData().getLastFill();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default long getLastFilled() {
|
|
||||||
+ return getLootableData().getLastFill();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default long getNextRefill() {
|
|
||||||
+ return getLootableData().getNextRefill();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ default long setNextRefill(long refillAt) {
|
|
||||||
+ if (refillAt < -1) {
|
|
||||||
+ refillAt = -1;
|
|
||||||
+ }
|
|
||||||
+ return getLootableData().setNextRefill(refillAt);
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventoryData.java b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventoryData.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..904332454ede006f4ee33337d46b11674d78bef7
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperLootableInventoryData.java
|
|
||||||
@@ -0,0 +1,181 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import com.destroystokyo.paper.PaperWorldConfig;
|
|
||||||
+import net.minecraft.nbt.NBTTagCompound;
|
|
||||||
+import net.minecraft.nbt.NBTTagList;
|
|
||||||
+import net.minecraft.world.entity.player.EntityHuman;
|
|
||||||
+import org.bukkit.entity.Player;
|
|
||||||
+import org.bukkit.loot.LootTable;
|
|
||||||
+
|
|
||||||
+import javax.annotation.Nullable;
|
|
||||||
+import java.util.HashMap;
|
|
||||||
+import java.util.Map;
|
|
||||||
+import java.util.Random;
|
|
||||||
+import java.util.UUID;
|
|
||||||
+
|
|
||||||
+public class PaperLootableInventoryData {
|
|
||||||
+
|
|
||||||
+ private static final Random RANDOM = new Random();
|
|
||||||
+
|
|
||||||
+ private long lastFill = -1;
|
|
||||||
+ private long nextRefill = -1;
|
|
||||||
+ private int numRefills = 0;
|
|
||||||
+ private Map<UUID, Long> lootedPlayers;
|
|
||||||
+ private final PaperLootableInventory lootable;
|
|
||||||
+
|
|
||||||
+ public PaperLootableInventoryData(PaperLootableInventory lootable) {
|
|
||||||
+ this.lootable = lootable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ long getLastFill() {
|
|
||||||
+ return this.lastFill;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ long getNextRefill() {
|
|
||||||
+ return this.nextRefill;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ long setNextRefill(long nextRefill) {
|
|
||||||
+ long prev = this.nextRefill;
|
|
||||||
+ this.nextRefill = nextRefill;
|
|
||||||
+ return prev;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ public boolean shouldReplenish(@Nullable EntityHuman player) {
|
|
||||||
+ LootTable table = this.lootable.getLootTable();
|
|
||||||
+
|
|
||||||
+ // No Loot Table associated
|
|
||||||
+ if (table == null) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // ALWAYS process the first fill or if the feature is disabled
|
|
||||||
+ if (this.lastFill == -1 || !this.lootable.getNMSWorld().paperConfig.autoReplenishLootables) {
|
|
||||||
+ return true;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Only process refills when a player is set
|
|
||||||
+ if (player == null) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Chest is not scheduled for refill
|
|
||||||
+ if (this.nextRefill == -1) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ final PaperWorldConfig paperConfig = this.lootable.getNMSWorld().paperConfig;
|
|
||||||
+
|
|
||||||
+ // Check if max refills has been hit
|
|
||||||
+ if (paperConfig.maxLootableRefills != -1 && this.numRefills >= paperConfig.maxLootableRefills) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ // Refill has not been reached
|
|
||||||
+ if (this.nextRefill > System.currentTimeMillis()) {
|
|
||||||
+ return false;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+ final Player bukkitPlayer = (Player) player.getBukkitEntity();
|
|
||||||
+ LootableInventoryReplenishEvent event = new LootableInventoryReplenishEvent(bukkitPlayer, lootable.getAPILootableInventory());
|
|
||||||
+ if (paperConfig.restrictPlayerReloot && hasPlayerLooted(player.getUniqueID())) {
|
|
||||||
+ event.setCancelled(true);
|
|
||||||
+ }
|
|
||||||
+ return event.callEvent();
|
|
||||||
+ }
|
|
||||||
+ public void processRefill(@Nullable EntityHuman player) {
|
|
||||||
+ this.lastFill = System.currentTimeMillis();
|
|
||||||
+ final PaperWorldConfig paperConfig = this.lootable.getNMSWorld().paperConfig;
|
|
||||||
+ if (paperConfig.autoReplenishLootables) {
|
|
||||||
+ int min = paperConfig.lootableRegenMin;
|
|
||||||
+ int max = paperConfig.lootableRegenMax;
|
|
||||||
+ this.nextRefill = this.lastFill + (min + RANDOM.nextInt(max - min + 1)) * 1000L;
|
|
||||||
+ this.numRefills++;
|
|
||||||
+ if (paperConfig.changeLootTableSeedOnFill) {
|
|
||||||
+ this.lootable.setSeed(0);
|
|
||||||
+ }
|
|
||||||
+ if (player != null) { // This means that numRefills can be incremented without a player being in the lootedPlayers list - Seems to be EntityMinecartChest specific
|
|
||||||
+ this.setPlayerLootedState(player.getUniqueID(), true);
|
|
||||||
+ }
|
|
||||||
+ } else {
|
|
||||||
+ this.lootable.clearLootTable();
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+ public void loadNbt(NBTTagCompound base) {
|
|
||||||
+ if (!base.hasKeyOfType("Paper.LootableData", 10)) { // 10 = compound
|
|
||||||
+ return;
|
|
||||||
+ }
|
|
||||||
+ NBTTagCompound comp = base.getCompound("Paper.LootableData");
|
|
||||||
+ if (comp.hasKey("lastFill")) {
|
|
||||||
+ this.lastFill = comp.getLong("lastFill");
|
|
||||||
+ }
|
|
||||||
+ if (comp.hasKey("nextRefill")) {
|
|
||||||
+ this.nextRefill = comp.getLong("nextRefill");
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (comp.hasKey("numRefills")) {
|
|
||||||
+ this.numRefills = comp.getInt("numRefills");
|
|
||||||
+ }
|
|
||||||
+ if (comp.hasKeyOfType("lootedPlayers", 9)) { // 9 = list
|
|
||||||
+ NBTTagList list = comp.getList("lootedPlayers", 10); // 10 = compound
|
|
||||||
+ final int size = list.size();
|
|
||||||
+ if (size > 0) {
|
|
||||||
+ this.lootedPlayers = new HashMap<>(list.size());
|
|
||||||
+ }
|
|
||||||
+ for (int i = 0; i < size; i++) {
|
|
||||||
+ final NBTTagCompound cmp = list.getCompound(i);
|
|
||||||
+ lootedPlayers.put(cmp.getUUID("UUID"), cmp.getLong("Time"));
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ public void saveNbt(NBTTagCompound base) {
|
|
||||||
+ NBTTagCompound comp = new NBTTagCompound();
|
|
||||||
+ if (this.nextRefill != -1) {
|
|
||||||
+ comp.setLong("nextRefill", this.nextRefill);
|
|
||||||
+ }
|
|
||||||
+ if (this.lastFill != -1) {
|
|
||||||
+ comp.setLong("lastFill", this.lastFill);
|
|
||||||
+ }
|
|
||||||
+ if (this.numRefills != 0) {
|
|
||||||
+ comp.setInt("numRefills", this.numRefills);
|
|
||||||
+ }
|
|
||||||
+ if (this.lootedPlayers != null && !this.lootedPlayers.isEmpty()) {
|
|
||||||
+ NBTTagList list = new NBTTagList();
|
|
||||||
+ for (Map.Entry<UUID, Long> entry : this.lootedPlayers.entrySet()) {
|
|
||||||
+ NBTTagCompound cmp = new NBTTagCompound();
|
|
||||||
+ cmp.setUUID("UUID", entry.getKey());
|
|
||||||
+ cmp.setLong("Time", entry.getValue());
|
|
||||||
+ list.add(cmp);
|
|
||||||
+ }
|
|
||||||
+ comp.set("lootedPlayers", list);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!comp.isEmpty()) {
|
|
||||||
+ base.set("Paper.LootableData", comp);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ void setPlayerLootedState(UUID player, boolean looted) {
|
|
||||||
+ if (looted && this.lootedPlayers == null) {
|
|
||||||
+ this.lootedPlayers = new HashMap<>();
|
|
||||||
+ }
|
|
||||||
+ if (looted) {
|
|
||||||
+ if (!this.lootedPlayers.containsKey(player)) {
|
|
||||||
+ this.lootedPlayers.put(player, System.currentTimeMillis());
|
|
||||||
+ }
|
|
||||||
+ } else if (this.lootedPlayers != null) {
|
|
||||||
+ this.lootedPlayers.remove(player);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ boolean hasPlayerLooted(UUID player) {
|
|
||||||
+ return this.lootedPlayers != null && this.lootedPlayers.containsKey(player);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ Long getLastLooted(UUID player) {
|
|
||||||
+ return lootedPlayers != null ? lootedPlayers.get(player) : null;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperMinecartLootableInventory.java b/src/main/java/com/destroystokyo/paper/loottable/PaperMinecartLootableInventory.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..c682bd7700d8103533026d46cfc63a7abde5a5f4
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperMinecartLootableInventory.java
|
|
||||||
@@ -0,0 +1,62 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import net.minecraft.world.entity.Entity;
|
|
||||||
+import net.minecraft.world.entity.vehicle.EntityMinecartContainer;
|
|
||||||
+import net.minecraft.world.level.World;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.craftbukkit.util.CraftNamespacedKey;
|
|
||||||
+
|
|
||||||
+public class PaperMinecartLootableInventory implements PaperLootableEntityInventory {
|
|
||||||
+
|
|
||||||
+ private EntityMinecartContainer entity;
|
|
||||||
+
|
|
||||||
+ public PaperMinecartLootableInventory(EntityMinecartContainer entity) {
|
|
||||||
+ this.entity = entity;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public org.bukkit.loot.LootTable getLootTable() {
|
|
||||||
+ return entity.lootTable != null ? Bukkit.getLootTable(CraftNamespacedKey.fromMinecraft(entity.lootTable)) : null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setLootTable(org.bukkit.loot.LootTable table, long seed) {
|
|
||||||
+ setLootTable(table);
|
|
||||||
+ setSeed(seed);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setSeed(long seed) {
|
|
||||||
+ entity.lootTableSeed = seed;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public long getSeed() {
|
|
||||||
+ return entity.lootTableSeed;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setLootTable(org.bukkit.loot.LootTable table) {
|
|
||||||
+ entity.lootTable = (table == null) ? null : CraftNamespacedKey.toMinecraft(table.getKey());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public PaperLootableInventoryData getLootableData() {
|
|
||||||
+ return entity.lootableData;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public Entity getHandle() {
|
|
||||||
+ return entity;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public LootableInventory getAPILootableInventory() {
|
|
||||||
+ return (LootableInventory) entity.getBukkitEntity();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public World getNMSWorld() {
|
|
||||||
+ return entity.world;
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/loottable/PaperTileEntityLootableInventory.java b/src/main/java/com/destroystokyo/paper/loottable/PaperTileEntityLootableInventory.java
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000000000000000000000000000000000..9dae34370d014a291f025f83b55e18bff4619a23
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/loottable/PaperTileEntityLootableInventory.java
|
|
||||||
@@ -0,0 +1,65 @@
|
|
||||||
+package com.destroystokyo.paper.loottable;
|
|
||||||
+
|
|
||||||
+import net.minecraft.server.MCUtil;
|
|
||||||
+import net.minecraft.world.level.World;
|
|
||||||
+import net.minecraft.world.level.block.entity.TileEntityLootable;
|
|
||||||
+import org.bukkit.Bukkit;
|
|
||||||
+import org.bukkit.craftbukkit.util.CraftNamespacedKey;
|
|
||||||
+
|
|
||||||
+public class PaperTileEntityLootableInventory implements PaperLootableBlockInventory {
|
|
||||||
+ private TileEntityLootable tileEntityLootable;
|
|
||||||
+
|
|
||||||
+ public PaperTileEntityLootableInventory(TileEntityLootable tileEntityLootable) {
|
|
||||||
+ this.tileEntityLootable = tileEntityLootable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public org.bukkit.loot.LootTable getLootTable() {
|
|
||||||
+ return tileEntityLootable.lootTable != null ? Bukkit.getLootTable(CraftNamespacedKey.fromMinecraft(tileEntityLootable.lootTable)) : null;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setLootTable(org.bukkit.loot.LootTable table, long seed) {
|
|
||||||
+ setLootTable(table);
|
|
||||||
+ setSeed(seed);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setLootTable(org.bukkit.loot.LootTable table) {
|
|
||||||
+ tileEntityLootable.lootTable = (table == null) ? null : CraftNamespacedKey.toMinecraft(table.getKey());
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public void setSeed(long seed) {
|
|
||||||
+ tileEntityLootable.lootTableSeed = seed;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public long getSeed() {
|
|
||||||
+ return tileEntityLootable.lootTableSeed;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public PaperLootableInventoryData getLootableData() {
|
|
||||||
+ return tileEntityLootable.lootableData;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public TileEntityLootable getTileEntity() {
|
|
||||||
+ return tileEntityLootable;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public LootableInventory getAPILootableInventory() {
|
|
||||||
+ World world = tileEntityLootable.getWorld();
|
|
||||||
+ if (world == null) {
|
|
||||||
+ return null;
|
|
||||||
+ }
|
|
||||||
+ return (LootableInventory) getBukkitWorld().getBlockAt(MCUtil.toLocation(world, tileEntityLootable.getPosition())).getState();
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ @Override
|
|
||||||
+ public World getNMSWorld() {
|
|
||||||
+ return tileEntityLootable.getWorld();
|
|
||||||
+ }
|
|
||||||
+}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/Entity.java b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
index 415935739716df3b8b3319aac19519d29aaa5776..092ee75f9527af25a48ab052659e3304986b50e0 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/Entity.java
|
|
||||||
@@ -158,6 +158,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, ne
|
|
||||||
};
|
|
||||||
// Paper end
|
|
||||||
|
|
||||||
+ public com.destroystokyo.paper.loottable.PaperLootableInventoryData lootableData; // Paper
|
|
||||||
private CraftEntity bukkitEntity;
|
|
||||||
|
|
||||||
public CraftEntity getBukkitEntity() {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartContainer.java b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartContainer.java
|
|
||||||
index c4b970c37b1792ac0022936f2df4740183621a0d..0166d11cb540a536390f486e1069d6119d8d23d6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartContainer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/entity/vehicle/EntityMinecartContainer.java
|
|
||||||
@@ -46,6 +46,7 @@ public abstract class EntityMinecartContainer extends EntityMinecartAbstract imp
|
|
||||||
public long lootTableSeed;
|
|
||||||
|
|
||||||
// CraftBukkit start
|
|
||||||
+ { this.lootableData = new com.destroystokyo.paper.loottable.PaperLootableInventoryData(new com.destroystokyo.paper.loottable.PaperMinecartLootableInventory(this)); } // Paper
|
|
||||||
public List<HumanEntity> transaction = new java.util.ArrayList<HumanEntity>();
|
|
||||||
private int maxStack = MAX_STACK;
|
|
||||||
|
|
||||||
@@ -203,12 +204,13 @@ public abstract class EntityMinecartContainer extends EntityMinecartAbstract imp
|
|
||||||
@Override
|
|
||||||
protected void saveData(NBTTagCompound nbttagcompound) {
|
|
||||||
super.saveData(nbttagcompound);
|
|
||||||
+ this.lootableData.saveNbt(nbttagcompound); // Paper
|
|
||||||
if (this.lootTable != null) {
|
|
||||||
nbttagcompound.setString("LootTable", this.lootTable.toString());
|
|
||||||
if (this.lootTableSeed != 0L) {
|
|
||||||
nbttagcompound.setLong("LootTableSeed", this.lootTableSeed);
|
|
||||||
}
|
|
||||||
- } else {
|
|
||||||
+ } if (true) { // Paper - Always save the items, Table may stick around
|
|
||||||
ContainerUtil.a(nbttagcompound, this.items);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -217,11 +219,12 @@ public abstract class EntityMinecartContainer extends EntityMinecartAbstract imp
|
|
||||||
@Override
|
|
||||||
protected void loadData(NBTTagCompound nbttagcompound) {
|
|
||||||
super.loadData(nbttagcompound);
|
|
||||||
+ this.lootableData.loadNbt(nbttagcompound); // Paper
|
|
||||||
this.items = NonNullList.a(this.getSize(), ItemStack.b);
|
|
||||||
if (nbttagcompound.hasKeyOfType("LootTable", 8)) {
|
|
||||||
this.lootTable = new MinecraftKey(nbttagcompound.getString("LootTable"));
|
|
||||||
this.lootTableSeed = nbttagcompound.getLong("LootTableSeed");
|
|
||||||
- } else {
|
|
||||||
+ } if (true) { // Paper - always load the items, table may still remain
|
|
||||||
ContainerUtil.b(nbttagcompound, this.items);
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -252,14 +255,15 @@ public abstract class EntityMinecartContainer extends EntityMinecartAbstract imp
|
|
||||||
}
|
|
||||||
|
|
||||||
public void d(@Nullable EntityHuman entityhuman) {
|
|
||||||
- if (this.lootTable != null && this.world.getMinecraftServer() != null) {
|
|
||||||
+ if (this.lootableData.shouldReplenish(entityhuman) && this.world.getMinecraftServer() != null) { // Paper
|
|
||||||
LootTable loottable = this.world.getMinecraftServer().getLootTableRegistry().getLootTable(this.lootTable);
|
|
||||||
|
|
||||||
if (entityhuman instanceof EntityPlayer) {
|
|
||||||
CriterionTriggers.N.a((EntityPlayer) entityhuman, this.lootTable);
|
|
||||||
}
|
|
||||||
|
|
||||||
- this.lootTable = null;
|
|
||||||
+ //this.lootTable = null; // Paper
|
|
||||||
+ this.lootableData.processRefill(entityhuman); // Paper
|
|
||||||
LootTableInfo.Builder loottableinfo_builder = (new LootTableInfo.Builder((WorldServer) this.world)).set(LootContextParameters.ORIGIN, this.getPositionVector()).a(this.lootTableSeed);
|
|
||||||
|
|
||||||
if (entityhuman != null) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/level/block/entity/TileEntityLootable.java b/src/main/java/net/minecraft/world/level/block/entity/TileEntityLootable.java
|
|
||||||
index 62e6833a90d7adae3c7df33e3bc73b4288e0370b..1508e267a38555820e2d31f3075adca185fbd4b6 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/level/block/entity/TileEntityLootable.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/level/block/entity/TileEntityLootable.java
|
|
||||||
@@ -27,6 +27,7 @@ public abstract class TileEntityLootable extends TileEntityContainer {
|
|
||||||
@Nullable
|
|
||||||
public MinecraftKey lootTable;
|
|
||||||
public long lootTableSeed;
|
|
||||||
+ public final com.destroystokyo.paper.loottable.PaperLootableInventoryData lootableData = new com.destroystokyo.paper.loottable.PaperLootableInventoryData(new com.destroystokyo.paper.loottable.PaperTileEntityLootableInventory(this)); // Paper
|
|
||||||
|
|
||||||
protected TileEntityLootable(TileEntityTypes<?> tileentitytypes) {
|
|
||||||
super(tileentitytypes);
|
|
||||||
@@ -42,16 +43,19 @@ public abstract class TileEntityLootable extends TileEntityContainer {
|
|
||||||
}
|
|
||||||
|
|
||||||
protected boolean b(NBTTagCompound nbttagcompound) {
|
|
||||||
+ this.lootableData.loadNbt(nbttagcompound); // Paper
|
|
||||||
if (nbttagcompound.hasKeyOfType("LootTable", 8)) {
|
|
||||||
this.lootTable = new MinecraftKey(nbttagcompound.getString("LootTable"));
|
|
||||||
+ try { org.bukkit.craftbukkit.util.CraftNamespacedKey.fromMinecraft(this.lootTable); } catch (IllegalArgumentException ex) { this.lootTable = null; } // Paper - validate
|
|
||||||
this.lootTableSeed = nbttagcompound.getLong("LootTableSeed");
|
|
||||||
- return true;
|
|
||||||
+ return false; // Paper - always load the items, table may still remain
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected boolean c(NBTTagCompound nbttagcompound) {
|
|
||||||
+ this.lootableData.saveNbt(nbttagcompound); // Paper
|
|
||||||
if (this.lootTable == null) {
|
|
||||||
return false;
|
|
||||||
} else {
|
|
||||||
@@ -60,19 +64,20 @@ public abstract class TileEntityLootable extends TileEntityContainer {
|
|
||||||
nbttagcompound.setLong("LootTableSeed", this.lootTableSeed);
|
|
||||||
}
|
|
||||||
|
|
||||||
- return true;
|
|
||||||
+ return false; // Paper - always save the items, table may still remain
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void d(@Nullable EntityHuman entityhuman) {
|
|
||||||
- if (this.lootTable != null && this.world.getMinecraftServer() != null) {
|
|
||||||
+ if (this.lootableData.shouldReplenish(entityhuman) && this.world.getMinecraftServer() != null) { // Paper
|
|
||||||
LootTable loottable = this.world.getMinecraftServer().getLootTableRegistry().getLootTable(this.lootTable);
|
|
||||||
|
|
||||||
if (entityhuman instanceof EntityPlayer) {
|
|
||||||
CriterionTriggers.N.a((EntityPlayer) entityhuman, this.lootTable);
|
|
||||||
}
|
|
||||||
|
|
||||||
- this.lootTable = null;
|
|
||||||
+ //this.lootTable = null; // Paper
|
|
||||||
+ this.lootableData.processRefill(entityhuman); // Paper
|
|
||||||
LootTableInfo.Builder loottableinfo_builder = (new LootTableInfo.Builder((WorldServer) this.world)).set(LootContextParameters.ORIGIN, Vec3D.a((BaseBlockPosition) this.position)).a(this.lootTableSeed);
|
|
||||||
|
|
||||||
if (entityhuman != null) {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/block/CraftBlockEntityState.java b/src/main/java/org/bukkit/craftbukkit/block/CraftBlockEntityState.java
|
|
||||||
index 524f27830752f424493c3ae8d793b871f6495594..dcf3f9265b0b00a7bbb9ff428e10da3c198ba08a 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/block/CraftBlockEntityState.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/block/CraftBlockEntityState.java
|
|
||||||
@@ -64,7 +64,7 @@ public class CraftBlockEntityState<T extends TileEntity> extends CraftBlockState
|
|
||||||
}
|
|
||||||
|
|
||||||
// gets the wrapped TileEntity
|
|
||||||
- protected T getTileEntity() {
|
|
||||||
+ public T getTileEntity() { // Paper - protected -> public
|
|
||||||
return tileEntity;
|
|
||||||
}
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/block/CraftChest.java b/src/main/java/org/bukkit/craftbukkit/block/CraftChest.java
|
|
||||||
index 486fa8937d644f59a770db163482259525a7e465..54eb170fd533b0e91572601268fcbc167ed9bb5c 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/block/CraftChest.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/block/CraftChest.java
|
|
||||||
@@ -12,8 +12,9 @@ import org.bukkit.craftbukkit.CraftWorld;
|
|
||||||
import org.bukkit.craftbukkit.inventory.CraftInventory;
|
|
||||||
import org.bukkit.craftbukkit.inventory.CraftInventoryDoubleChest;
|
|
||||||
import org.bukkit.inventory.Inventory;
|
|
||||||
+import com.destroystokyo.paper.loottable.PaperLootableBlockInventory; // Paper
|
|
||||||
|
|
||||||
-public class CraftChest extends CraftLootable<TileEntityChest> implements Chest {
|
|
||||||
+public class CraftChest extends CraftLootable<TileEntityChest> implements Chest, PaperLootableBlockInventory { // Paper
|
|
||||||
|
|
||||||
public CraftChest(final Block block) {
|
|
||||||
super(block, TileEntityChest.class);
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/block/CraftLootable.java b/src/main/java/org/bukkit/craftbukkit/block/CraftLootable.java
|
|
||||||
index f0a7e61a26c4668a9aa823d641f29bdecd42dd1f..3512054ede5fd1dd7605444e827e30a0be47f935 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/block/CraftLootable.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/block/CraftLootable.java
|
|
||||||
@@ -10,7 +10,7 @@ import org.bukkit.craftbukkit.util.CraftNamespacedKey;
|
|
||||||
import org.bukkit.loot.LootTable;
|
|
||||||
import org.bukkit.loot.Lootable;
|
|
||||||
|
|
||||||
-public abstract class CraftLootable<T extends TileEntityLootable> extends CraftContainer<T> implements Nameable, Lootable {
|
|
||||||
+public abstract class CraftLootable<T extends TileEntityLootable> extends CraftContainer<T> implements Nameable, Lootable, com.destroystokyo.paper.loottable.PaperLootableBlockInventory { // Paper
|
|
||||||
|
|
||||||
public CraftLootable(Block block, Class<T> tileEntityClass) {
|
|
||||||
super(block, tileEntityClass);
|
|
||||||
@@ -54,7 +54,7 @@ public abstract class CraftLootable<T extends TileEntityLootable> extends CraftC
|
|
||||||
setLootTable(getLootTable(), seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
- private void setLootTable(LootTable table, long seed) {
|
|
||||||
+ public void setLootTable(LootTable table, long seed) { // Paper - public
|
|
||||||
MinecraftKey key = (table == null) ? null : CraftNamespacedKey.toMinecraft(table.getKey());
|
|
||||||
getSnapshot().setLootTable(key, seed);
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartChest.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartChest.java
|
|
||||||
index cbd121c21adfaf098dadca33de16a2e68d83c19a..d9a2552782c9242fb84cc0c8309a614a44777509 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartChest.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartChest.java
|
|
||||||
@@ -8,7 +8,7 @@ import org.bukkit.entity.minecart.StorageMinecart;
|
|
||||||
import org.bukkit.inventory.Inventory;
|
|
||||||
|
|
||||||
@SuppressWarnings("deprecation")
|
|
||||||
-public class CraftMinecartChest extends CraftMinecartContainer implements StorageMinecart {
|
|
||||||
+public class CraftMinecartChest extends CraftMinecartContainer implements StorageMinecart, com.destroystokyo.paper.loottable.PaperLootableEntityInventory { // Paper
|
|
||||||
private final CraftInventory inventory;
|
|
||||||
|
|
||||||
public CraftMinecartChest(CraftServer server, EntityMinecartChest entity) {
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartContainer.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartContainer.java
|
|
||||||
index 5ffb8108f456c2f7f3ed1a25249baccb4cbf4add..bf8b5b25d1af0c5129261e10abf2866521b2c375 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartContainer.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartContainer.java
|
|
||||||
@@ -47,7 +47,7 @@ public abstract class CraftMinecartContainer extends CraftMinecart implements Lo
|
|
||||||
return getHandle().lootTableSeed;
|
|
||||||
}
|
|
||||||
|
|
||||||
- private void setLootTable(LootTable table, long seed) {
|
|
||||||
+ public void setLootTable(LootTable table, long seed) { // Paper
|
|
||||||
MinecraftKey newKey = (table == null) ? null : CraftNamespacedKey.toMinecraft(table.getKey());
|
|
||||||
getHandle().setLootTable(newKey, seed);
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartHopper.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartHopper.java
|
|
||||||
index 17a42aec76f32a28b0c9885c60d1ed50c6727161..bfdcf01d2c6570493e86330d56500427dbb23146 100644
|
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartHopper.java
|
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftMinecartHopper.java
|
|
||||||
@@ -7,7 +7,7 @@ import org.bukkit.entity.EntityType;
|
|
||||||
import org.bukkit.entity.minecart.HopperMinecart;
|
|
||||||
import org.bukkit.inventory.Inventory;
|
|
||||||
|
|
||||||
-public final class CraftMinecartHopper extends CraftMinecartContainer implements HopperMinecart {
|
|
||||||
+public final class CraftMinecartHopper extends CraftMinecartContainer implements HopperMinecart, com.destroystokyo.paper.loottable.PaperLootableEntityInventory { // Paper
|
|
||||||
private final CraftInventory inventory;
|
|
||||||
|
|
||||||
public CraftMinecartHopper(CraftServer server, EntityMinecartHopper entity) {
|
|
@ -1,32 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Sat, 7 May 2016 23:33:08 -0400
|
|
||||||
Subject: [PATCH] Don't save empty scoreboard teams to scoreboard.dat
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
index 504efea7b6f50a0d17f4f353781953dfb18bdeca..1b8e5671c9dc8c15ce33d351c1bb20f28919b9a2 100644
|
|
||||||
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
||||||
@@ -237,4 +237,9 @@ public class PaperConfig {
|
|
||||||
private static void enablePlayerCollisions() {
|
|
||||||
enablePlayerCollisions = getBoolean("settings.enable-player-collisions", true);
|
|
||||||
}
|
|
||||||
+
|
|
||||||
+ public static boolean saveEmptyScoreboardTeams = false;
|
|
||||||
+ private static void saveEmptyScoreboardTeams() {
|
|
||||||
+ saveEmptyScoreboardTeams = getBoolean("settings.save-empty-scoreboard-teams", false);
|
|
||||||
+ }
|
|
||||||
}
|
|
||||||
diff --git a/src/main/java/net/minecraft/world/scores/PersistentScoreboard.java b/src/main/java/net/minecraft/world/scores/PersistentScoreboard.java
|
|
||||||
index 3998565ccd87c966c0fb9e6757cd1861faa5bc15..52f27bdbd0df8bbbf2ad5144bc262b5093d83413 100644
|
|
||||||
--- a/src/main/java/net/minecraft/world/scores/PersistentScoreboard.java
|
|
||||||
+++ b/src/main/java/net/minecraft/world/scores/PersistentScoreboard.java
|
|
||||||
@@ -182,6 +182,7 @@ public class PersistentScoreboard extends PersistentBase {
|
|
||||||
|
|
||||||
while (iterator.hasNext()) {
|
|
||||||
ScoreboardTeam scoreboardteam = (ScoreboardTeam) iterator.next();
|
|
||||||
+ if (!com.destroystokyo.paper.PaperConfig.saveEmptyScoreboardTeams && scoreboardteam.getPlayerNameSet().isEmpty()) continue; // Paper
|
|
||||||
NBTTagCompound nbttagcompound = new NBTTagCompound();
|
|
||||||
|
|
||||||
nbttagcompound.setString("Name", scoreboardteam.getName());
|
|
@ -1,19 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Zach Brown <zach.brown@destroystokyo.com>
|
|
||||||
Date: Thu, 12 May 2016 23:02:58 -0500
|
|
||||||
Subject: [PATCH] System property for disabling watchdoge
|
|
||||||
|
|
||||||
|
|
||||||
diff --git a/src/main/java/org/spigotmc/WatchdogThread.java b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
index 69e5054886b5858664fed333aca8c25a76e5cb11..4e0291be4bd5876bb5b5f62ebfa156635d4c758f 100644
|
|
||||||
--- a/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
+++ b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
||||||
@@ -61,7 +61,7 @@ public class WatchdogThread extends Thread
|
|
||||||
while ( !stopping )
|
|
||||||
{
|
|
||||||
//
|
|
||||||
- if ( lastTick != 0 && timeoutTime > 0 && monotonicMillis() > lastTick + timeoutTime )
|
|
||||||
+ if ( lastTick != 0 && timeoutTime > 0 && monotonicMillis() > lastTick + timeoutTime && !Boolean.getBoolean("disable.watchdog")) // Paper - Add property to disable
|
|
||||||
{
|
|
||||||
Logger log = Bukkit.getServer().getLogger();
|
|
||||||
log.log( Level.SEVERE, "------------------------------" );
|
|
@ -1,117 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 16 May 2016 20:47:41 -0400
|
|
||||||
Subject: [PATCH] Optimize UserCache / Thread Safe
|
|
||||||
|
|
||||||
Because Techable keeps complaining about how this isn't thread safe,
|
|
||||||
easier to do this than replace the entire thing.
|
|
||||||
|
|
||||||
Additionally, move Saving of the User cache to be done async, incase
|
|
||||||
the user never changed the default setting for Spigot's save on stop only.
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
index ed00ee8e56e6ca38a1ac689458c4675eff6e3eea..20dffd9dc4cb3868638151b70535f353a3bf1a44 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
||||||
@@ -907,7 +907,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
|
||||||
} catch (java.lang.InterruptedException ignored) {} // Paper
|
|
||||||
if (org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly) {
|
|
||||||
LOGGER.info("Saving usercache.json");
|
|
||||||
- this.getUserCache().b();
|
|
||||||
+ this.getUserCache().b(false); // Paper
|
|
||||||
}
|
|
||||||
// Spigot end
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
index faf4d00bf288359db806913c4d2964324e8706b7..8ae72e8c8325d9b03803f29fcdd83a0ce8d34450 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/dedicated/DedicatedServer.java
|
|
||||||
@@ -251,7 +251,7 @@ public class DedicatedServer extends MinecraftServer implements IMinecraftServer
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.convertNames()) {
|
|
||||||
- this.getUserCache().b();
|
|
||||||
+ this.getUserCache().b(false); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!NameReferencingFileConverter.e(this)) {
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/players/UserCache.java b/src/main/java/net/minecraft/server/players/UserCache.java
|
|
||||||
index 5694ba68a142d735a7c61563ee22cd54da9b4cc9..39d1c379b781c08bfdd720cd6810a9c0bb9f0d09 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/players/UserCache.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/players/UserCache.java
|
|
||||||
@@ -36,6 +36,7 @@ import java.util.concurrent.atomic.AtomicLong;
|
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
|
||||||
import java.util.stream.Stream;
|
|
||||||
import javax.annotation.Nullable;
|
|
||||||
+import net.minecraft.server.MCUtil;
|
|
||||||
import net.minecraft.world.entity.player.EntityHuman;
|
|
||||||
import org.apache.logging.log4j.LogManager;
|
|
||||||
import org.apache.logging.log4j.Logger;
|
|
||||||
@@ -108,7 +109,7 @@ public class UserCache {
|
|
||||||
return UserCache.b;
|
|
||||||
}
|
|
||||||
|
|
||||||
- public void a(GameProfile gameprofile) {
|
|
||||||
+ public synchronized void a(GameProfile gameprofile) { // Paper - synchronize
|
|
||||||
Calendar calendar = Calendar.getInstance();
|
|
||||||
|
|
||||||
calendar.setTime(new Date());
|
|
||||||
@@ -117,7 +118,7 @@ public class UserCache {
|
|
||||||
UserCache.UserCacheEntry usercache_usercacheentry = new UserCache.UserCacheEntry(gameprofile, date);
|
|
||||||
|
|
||||||
this.a(usercache_usercacheentry);
|
|
||||||
- if( !org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly ) this.b(); // Spigot - skip saving if disabled
|
|
||||||
+ if( !org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly ) this.b(true); // Spigot - skip saving if disabled // Paper - async
|
|
||||||
}
|
|
||||||
|
|
||||||
private long d() {
|
|
||||||
@@ -125,7 +126,7 @@ public class UserCache {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
- public GameProfile getProfile(String s) {
|
|
||||||
+ public synchronized GameProfile getProfile(String s) { // Paper - synchronize
|
|
||||||
String s1 = s.toLowerCase(Locale.ROOT);
|
|
||||||
UserCache.UserCacheEntry usercache_usercacheentry = (UserCache.UserCacheEntry) this.c.get(s1);
|
|
||||||
boolean flag = false;
|
|
||||||
@@ -151,7 +152,7 @@ public class UserCache {
|
|
||||||
}
|
|
||||||
|
|
||||||
if (flag && !org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly) { // Spigot - skip saving if disabled
|
|
||||||
- this.b();
|
|
||||||
+ this.b(true); // Paper
|
|
||||||
}
|
|
||||||
|
|
||||||
return gameprofile;
|
|
||||||
@@ -233,7 +234,7 @@ public class UserCache {
|
|
||||||
return arraylist;
|
|
||||||
}
|
|
||||||
|
|
||||||
- public void b() {
|
|
||||||
+ public void b(boolean asyncSave) { // Paper
|
|
||||||
JsonArray jsonarray = new JsonArray();
|
|
||||||
DateFormat dateformat = e();
|
|
||||||
|
|
||||||
@@ -241,6 +242,7 @@ public class UserCache {
|
|
||||||
jsonarray.add(a(usercache_usercacheentry, dateformat));
|
|
||||||
});
|
|
||||||
String s = this.f.toJson(jsonarray);
|
|
||||||
+ Runnable save = () -> { // Paper
|
|
||||||
|
|
||||||
try {
|
|
||||||
BufferedWriter bufferedwriter = Files.newWriter(this.g, StandardCharsets.UTF_8);
|
|
||||||
@@ -268,6 +270,14 @@ public class UserCache {
|
|
||||||
} catch (IOException ioexception) {
|
|
||||||
;
|
|
||||||
}
|
|
||||||
+ // Paper start
|
|
||||||
+ };
|
|
||||||
+ if (asyncSave) {
|
|
||||||
+ MCUtil.scheduleAsyncTask(save);
|
|
||||||
+ } else {
|
|
||||||
+ save.run();
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
@ -1,45 +0,0 @@
|
|||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
||||||
From: Aikar <aikar@aikar.co>
|
|
||||||
Date: Mon, 16 May 2016 23:19:16 -0400
|
|
||||||
Subject: [PATCH] Avoid blocking on Network Manager creation
|
|
||||||
|
|
||||||
Per Paper issue 294
|
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/network/ServerConnection.java b/src/main/java/net/minecraft/server/network/ServerConnection.java
|
|
||||||
index f66a5ba901601c1d359a287861a2edd8e3a106a7..d992cb5cd827e0fe655809e1088939cdad9c2301 100644
|
|
||||||
--- a/src/main/java/net/minecraft/server/network/ServerConnection.java
|
|
||||||
+++ b/src/main/java/net/minecraft/server/network/ServerConnection.java
|
|
||||||
@@ -52,6 +52,15 @@ public class ServerConnection {
|
|
||||||
public volatile boolean c;
|
|
||||||
private final List<ChannelFuture> listeningChannels = Collections.synchronizedList(Lists.newArrayList());
|
|
||||||
private final List<NetworkManager> connectedChannels = Collections.synchronizedList(Lists.newArrayList());
|
|
||||||
+ // Paper start - prevent blocking on adding a new network manager while the server is ticking
|
|
||||||
+ private final java.util.Queue<NetworkManager> pending = new java.util.concurrent.ConcurrentLinkedQueue<>();
|
|
||||||
+ private void addPending() {
|
|
||||||
+ NetworkManager manager = null;
|
|
||||||
+ while ((manager = pending.poll()) != null) {
|
|
||||||
+ connectedChannels.add(manager);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ // Paper end
|
|
||||||
|
|
||||||
public ServerConnection(MinecraftServer minecraftserver) {
|
|
||||||
this.e = minecraftserver;
|
|
||||||
@@ -87,7 +96,8 @@ public class ServerConnection {
|
|
||||||
int j = ServerConnection.this.e.k();
|
|
||||||
Object object = j > 0 ? new NetworkManagerServer(j) : new NetworkManager(EnumProtocolDirection.SERVERBOUND);
|
|
||||||
|
|
||||||
- ServerConnection.this.connectedChannels.add((NetworkManager) object); // CraftBukkit - decompile error
|
|
||||||
+ //ServerConnection.this.connectedChannels.add((NetworkManager) object); // CraftBukkit - decompile error
|
|
||||||
+ pending.add((NetworkManager) object); // Paper
|
|
||||||
channel.pipeline().addLast("packet_handler", (ChannelHandler) object);
|
|
||||||
((NetworkManager) object).setPacketListener(new HandshakeListener(ServerConnection.this.e, (NetworkManager) object));
|
|
||||||
}
|
|
||||||
@@ -126,6 +136,7 @@ public class ServerConnection {
|
|
||||||
|
|
||||||
synchronized (this.connectedChannels) {
|
|
||||||
// Spigot Start
|
|
||||||
+ this.addPending(); // Paper
|
|
||||||
// This prevents players from 'gaming' the server, and strategically relogging to increase their position in the tick order
|
|
||||||
if ( org.spigotmc.SpigotConfig.playerShuffle > 0 && MinecraftServer.currentTick % org.spigotmc.SpigotConfig.playerShuffle == 0 )
|
|
||||||
{
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user