2019-01-31 04:49:52 +00:00
|
|
|
From 7f98ea9490aaabaf4295ad4ef1f63ae99db850fd Mon Sep 17 00:00:00 2001
|
2018-08-12 13:27:23 -04:00
|
|
|
From: miclebrick <miclebrick@outlook.com>
|
|
|
|
Date: Wed, 8 Aug 2018 15:30:52 -0400
|
|
|
|
Subject: [PATCH] Add Early Warning Feature to WatchDog
|
|
|
|
|
|
|
|
Detect when the server has been hung for a long duration, and start printing
|
|
|
|
thread dumps at an interval until the point of crash.
|
|
|
|
|
|
|
|
This will help diagnose what was going on in that time before the crash.
|
|
|
|
|
|
|
|
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
2019-01-31 04:49:52 +00:00
|
|
|
index 975ebf75c..6bd59a886 100644
|
2018-08-12 13:27:23 -04:00
|
|
|
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
|
|
|
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
2018-10-18 20:44:59 -04:00
|
|
|
@@ -25,6 +25,7 @@ import org.bukkit.configuration.file.YamlConfiguration;
|
2018-08-12 13:27:23 -04:00
|
|
|
import co.aikar.timings.Timings;
|
|
|
|
import co.aikar.timings.TimingsManager;
|
2018-10-18 20:44:59 -04:00
|
|
|
import org.spigotmc.SpigotConfig;
|
2018-08-12 13:27:23 -04:00
|
|
|
+import org.spigotmc.WatchdogThread;
|
|
|
|
|
|
|
|
public class PaperConfig {
|
|
|
|
|
2018-12-14 20:17:27 -05:00
|
|
|
@@ -333,6 +334,14 @@ public class PaperConfig {
|
2018-08-12 13:27:23 -04:00
|
|
|
}
|
|
|
|
}
|
2018-08-12 13:27:23 -04:00
|
|
|
|
2018-08-12 13:27:23 -04:00
|
|
|
+ public static int watchdogPrintEarlyWarningEvery = 5000;
|
|
|
|
+ public static int watchdogPrintEarlyWarningDelay = 10000;
|
|
|
|
+ private static void watchdogEarlyWarning() {
|
|
|
|
+ watchdogPrintEarlyWarningEvery = getInt("settings.watchdog.early-warning-every", 5000);
|
|
|
|
+ watchdogPrintEarlyWarningDelay = getInt("settings.watchdog.early-warning-delay", 10000);
|
|
|
|
+ WatchdogThread.doStart(SpigotConfig.timeoutTime, SpigotConfig.restartOnCrash );
|
|
|
|
+ }
|
2018-08-12 13:27:23 -04:00
|
|
|
+
|
2018-08-29 17:26:24 +01:00
|
|
|
public static int tabSpamIncrement = 1;
|
2018-08-12 13:27:23 -04:00
|
|
|
public static int tabSpamLimit = 500;
|
|
|
|
private static void tabSpamLimiters() {
|
2018-08-12 13:27:23 -04:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2019-01-06 17:15:21 +00:00
|
|
|
index c562ba546..e993b4da6 100644
|
2018-08-12 13:27:23 -04:00
|
|
|
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2019-01-06 17:15:21 +00:00
|
|
|
@@ -792,6 +792,7 @@ public abstract class MinecraftServer implements IAsyncTaskHandler, IMojangStati
|
2018-08-26 14:11:49 -04:00
|
|
|
this.a(this.m);
|
2018-08-12 13:27:23 -04:00
|
|
|
|
|
|
|
// Spigot start
|
2018-08-16 18:11:35 -04:00
|
|
|
+ org.spigotmc.WatchdogThread.hasStarted = true; // Paper
|
2018-08-12 13:27:23 -04:00
|
|
|
Arrays.fill( recentTps, 20 );
|
2018-08-26 14:11:49 -04:00
|
|
|
long start = System.nanoTime(), curTime, wait, tickSection = start; // Paper - Further improve server tick loop
|
|
|
|
lastTick = start - TICK_TIME; // Paper
|
2018-12-12 17:41:11 -08:00
|
|
|
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
2019-01-31 04:49:52 +00:00
|
|
|
index 296e338c8..bc7d39e26 100644
|
2018-12-12 17:41:11 -08:00
|
|
|
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
|
|
|
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
2019-01-31 04:49:52 +00:00
|
|
|
@@ -752,6 +752,7 @@ public final class CraftServer implements Server {
|
2018-12-12 17:41:11 -08:00
|
|
|
|
|
|
|
@Override
|
|
|
|
public void reload() {
|
|
|
|
+ org.spigotmc.WatchdogThread.hasStarted = false; // Paper - Disable watchdog early timeout on reload
|
|
|
|
reloadCount++;
|
|
|
|
configuration = YamlConfiguration.loadConfiguration(getConfigFile());
|
|
|
|
commandsConfiguration = YamlConfiguration.loadConfiguration(getCommandsConfigFile());
|
2019-01-31 04:49:52 +00:00
|
|
|
@@ -857,6 +858,7 @@ public final class CraftServer implements Server {
|
2018-12-12 17:41:11 -08:00
|
|
|
enablePlugins(PluginLoadOrder.STARTUP);
|
|
|
|
enablePlugins(PluginLoadOrder.POSTWORLD);
|
|
|
|
getPluginManager().callEvent(new ServerLoadEvent(ServerLoadEvent.LoadType.RELOAD));
|
|
|
|
+ org.spigotmc.WatchdogThread.hasStarted = true; // Paper - Disable watchdog early timeout on reload
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2018-08-12 13:27:23 -04:00
|
|
|
diff --git a/src/main/java/org/spigotmc/SpigotConfig.java b/src/main/java/org/spigotmc/SpigotConfig.java
|
2019-01-01 03:15:55 +00:00
|
|
|
index 9c442dee2..fb09fb097 100644
|
2018-08-12 13:27:23 -04:00
|
|
|
--- a/src/main/java/org/spigotmc/SpigotConfig.java
|
|
|
|
+++ b/src/main/java/org/spigotmc/SpigotConfig.java
|
2019-01-01 03:15:55 +00:00
|
|
|
@@ -225,7 +225,7 @@ public class SpigotConfig
|
2018-08-12 13:27:23 -04:00
|
|
|
restartScript = getString( "settings.restart-script", restartScript );
|
|
|
|
restartMessage = transform( getString( "messages.restart", "Server is restarting" ) );
|
|
|
|
commands.put( "restart", new RestartCommand( "restart" ) );
|
|
|
|
- WatchdogThread.doStart( timeoutTime, restartOnCrash );
|
|
|
|
+ //WatchdogThread.doStart( timeoutTime, restartOnCrash ); // Paper - moved to PaperConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean bungee;
|
|
|
|
diff --git a/src/main/java/org/spigotmc/WatchdogThread.java b/src/main/java/org/spigotmc/WatchdogThread.java
|
2018-12-12 17:41:11 -08:00
|
|
|
index ed5f46bf6..9dba9510f 100644
|
2018-08-12 13:27:23 -04:00
|
|
|
--- a/src/main/java/org/spigotmc/WatchdogThread.java
|
|
|
|
+++ b/src/main/java/org/spigotmc/WatchdogThread.java
|
|
|
|
@@ -5,6 +5,7 @@ import java.lang.management.MonitorInfo;
|
|
|
|
import java.lang.management.ThreadInfo;
|
|
|
|
import java.util.logging.Level;
|
|
|
|
import java.util.logging.Logger;
|
|
|
|
+import com.destroystokyo.paper.PaperConfig;
|
|
|
|
import net.minecraft.server.MinecraftServer;
|
|
|
|
import org.bukkit.Bukkit;
|
|
|
|
|
|
|
|
@@ -13,6 +14,10 @@ public class WatchdogThread extends Thread
|
|
|
|
|
|
|
|
private static WatchdogThread instance;
|
|
|
|
private final long timeoutTime;
|
|
|
|
+ private final long earlyWarningEvery; // Paper - Timeout time for just printing a dump but not restarting
|
|
|
|
+ private final long earlyWarningDelay; // Paper
|
|
|
|
+ public static volatile boolean hasStarted; // Paper
|
|
|
|
+ private long lastEarlyWarning; // Paper - Keep track of short dump times to avoid spamming console with short dumps
|
|
|
|
private final boolean restart;
|
|
|
|
private volatile long lastTick;
|
|
|
|
private volatile boolean stopping;
|
|
|
|
@@ -22,6 +27,8 @@ public class WatchdogThread extends Thread
|
|
|
|
super( "Paper Watchdog Thread" );
|
|
|
|
this.timeoutTime = timeoutTime;
|
|
|
|
this.restart = restart;
|
|
|
|
+ earlyWarningEvery = Math.min(PaperConfig.watchdogPrintEarlyWarningEvery, timeoutTime); // Paper
|
|
|
|
+ earlyWarningDelay = Math.min(PaperConfig.watchdogPrintEarlyWarningDelay, timeoutTime); // Paper
|
|
|
|
}
|
|
|
|
|
2018-11-24 05:28:04 +00:00
|
|
|
private static long monotonicMillis()
|
|
|
|
@@ -56,10 +63,17 @@ public class WatchdogThread extends Thread
|
2018-08-16 18:11:35 -04:00
|
|
|
{
|
2018-08-12 13:27:23 -04:00
|
|
|
while ( !stopping )
|
|
|
|
{
|
2018-08-16 18:11:35 -04:00
|
|
|
- //
|
2018-11-24 05:28:04 +00:00
|
|
|
- if ( lastTick != 0 && monotonicMillis() > lastTick + timeoutTime && !Boolean.getBoolean("disable.watchdog")) // Paper - Add property to disable
|
2018-08-16 18:11:35 -04:00
|
|
|
+ // Paper start
|
2018-11-24 05:28:04 +00:00
|
|
|
+ Logger log = Bukkit.getServer().getLogger();
|
|
|
|
+ long currentTime = monotonicMillis();
|
2018-08-16 18:11:35 -04:00
|
|
|
+ if ( lastTick != 0 && currentTime > lastTick + earlyWarningEvery && !Boolean.getBoolean("disable.watchdog") )
|
2018-08-12 13:27:23 -04:00
|
|
|
{
|
2018-11-24 05:28:04 +00:00
|
|
|
- Logger log = Bukkit.getServer().getLogger();
|
2018-08-12 13:27:23 -04:00
|
|
|
+ boolean isLongTimeout = currentTime > lastTick + timeoutTime;
|
2018-08-12 13:27:23 -04:00
|
|
|
+ // Don't spam early warning dumps
|
2018-08-12 13:27:23 -04:00
|
|
|
+ if ( !isLongTimeout && (earlyWarningEvery <= 0 || !hasStarted || currentTime < lastEarlyWarning + earlyWarningEvery || currentTime < lastTick + earlyWarningDelay)) continue;
|
|
|
|
+ lastEarlyWarning = currentTime;
|
2018-11-24 05:28:04 +00:00
|
|
|
+ if (isLongTimeout) {
|
2018-08-12 13:27:23 -04:00
|
|
|
+ // Paper end
|
2018-11-24 05:28:04 +00:00
|
|
|
log.log( Level.SEVERE, "------------------------------" );
|
|
|
|
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" );
|
|
|
|
@@ -89,29 +103,46 @@ public class WatchdogThread extends Thread
|
2018-08-12 13:27:23 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Paper end
|
|
|
|
+ } else
|
|
|
|
+ {
|
2018-10-24 15:05:39 -04:00
|
|
|
+ log.log(Level.SEVERE, "--- DO NOT REPORT THIS TO PAPER - THIS IS NOT A BUG OR A CRASH - " + Bukkit.getServer().getVersion() + " ---");
|
2018-08-12 13:27:23 -04:00
|
|
|
+ log.log(Level.SEVERE, "The server has not responded for " + (currentTime - lastTick) / 1000 + " seconds! Creating thread dump");
|
2018-08-12 13:27:23 -04:00
|
|
|
+ }
|
|
|
|
+ // Paper end - Different message for short timeout
|
|
|
|
log.log( Level.SEVERE, "------------------------------" );
|
|
|
|
log.log( Level.SEVERE, "Server thread dump (Look for plugins here before reporting to Paper!):" );
|
|
|
|
dumpThread( ManagementFactory.getThreadMXBean().getThreadInfo( MinecraftServer.getServer().primaryThread.getId(), Integer.MAX_VALUE ), log );
|
|
|
|
log.log( Level.SEVERE, "------------------------------" );
|
|
|
|
//
|
|
|
|
+ // Paper start - Only print full dump on long timeouts
|
|
|
|
+ if ( isLongTimeout )
|
|
|
|
+ {
|
|
|
|
log.log( Level.SEVERE, "Entire Thread Dump:" );
|
|
|
|
ThreadInfo[] threads = ManagementFactory.getThreadMXBean().dumpAllThreads( true, true );
|
|
|
|
for ( ThreadInfo thread : threads )
|
2018-08-12 13:27:23 -04:00
|
|
|
{
|
|
|
|
dumpThread( thread, log );
|
|
|
|
}
|
|
|
|
+ } else {
|
|
|
|
+ log.log(Level.SEVERE, "--- DO NOT REPORT THIS TO PAPER - THIS IS NOT A BUG OR A CRASH ---");
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+
|
|
|
|
log.log( Level.SEVERE, "------------------------------" );
|
|
|
|
|
2018-08-17 09:09:08 -07:00
|
|
|
+ if ( isLongTimeout )
|
|
|
|
+ {
|
2018-08-12 13:27:23 -04:00
|
|
|
if ( restart )
|
|
|
|
{
|
2018-08-12 13:27:23 -04:00
|
|
|
RestartCommand.restart();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
+ } // Paper end
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
- sleep( 10000 );
|
|
|
|
+ sleep( 1000 ); // Paper - Reduce check time to every second instead of every ten seconds, more consistent and allows for short timeout
|
|
|
|
} catch ( InterruptedException ex )
|
|
|
|
{
|
|
|
|
interrupt();
|
|
|
|
--
|
2019-01-01 03:15:55 +00:00
|
|
|
2.20.1
|
2018-08-12 13:27:23 -04:00
|
|
|
|