mirror of
https://github.com/plan-player-analytics/Plan.git
synced 2025-01-12 19:30:44 +01:00
Moved page parsing under utilities/html/pages
This commit is contained in:
parent
7d389a1fc1
commit
bac1f4f7bf
@ -15,10 +15,10 @@ import com.djrapitops.plan.system.info.request.CacheRequest;
|
|||||||
import com.djrapitops.plan.system.info.request.GenerateInspectPageRequest;
|
import com.djrapitops.plan.system.info.request.GenerateInspectPageRequest;
|
||||||
import com.djrapitops.plan.system.info.request.InfoRequest;
|
import com.djrapitops.plan.system.info.request.InfoRequest;
|
||||||
import com.djrapitops.plan.system.info.server.ServerInfo;
|
import com.djrapitops.plan.system.info.server.ServerInfo;
|
||||||
import com.djrapitops.plan.system.webserver.pages.parsing.NetworkPage;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.cache.PageId;
|
import com.djrapitops.plan.system.webserver.response.cache.PageId;
|
||||||
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
||||||
import com.djrapitops.plan.system.webserver.response.pages.AnalysisPageResponse;
|
import com.djrapitops.plan.system.webserver.response.pages.AnalysisPageResponse;
|
||||||
|
import com.djrapitops.plan.utilities.html.pages.NetworkPage;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* InfoSystem for Bungee.
|
* InfoSystem for Bungee.
|
||||||
|
@ -12,9 +12,9 @@ import com.djrapitops.plan.data.store.containers.AnalysisContainer;
|
|||||||
import com.djrapitops.plan.system.cache.CacheSystem;
|
import com.djrapitops.plan.system.cache.CacheSystem;
|
||||||
import com.djrapitops.plan.system.info.InfoSystem;
|
import com.djrapitops.plan.system.info.InfoSystem;
|
||||||
import com.djrapitops.plan.system.info.server.ServerInfo;
|
import com.djrapitops.plan.system.info.server.ServerInfo;
|
||||||
import com.djrapitops.plan.system.webserver.pages.parsing.AnalysisPage;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.DefaultResponses;
|
import com.djrapitops.plan.system.webserver.response.DefaultResponses;
|
||||||
import com.djrapitops.plan.system.webserver.response.Response;
|
import com.djrapitops.plan.system.webserver.response.Response;
|
||||||
|
import com.djrapitops.plan.utilities.html.pages.AnalysisPage;
|
||||||
import com.djrapitops.plugin.api.utility.log.Log;
|
import com.djrapitops.plugin.api.utility.log.Log;
|
||||||
import com.djrapitops.plugin.utilities.Verify;
|
import com.djrapitops.plugin.utilities.Verify;
|
||||||
|
|
||||||
|
@ -8,10 +8,10 @@ import com.djrapitops.plan.api.exceptions.ParseException;
|
|||||||
import com.djrapitops.plan.api.exceptions.connection.*;
|
import com.djrapitops.plan.api.exceptions.connection.*;
|
||||||
import com.djrapitops.plan.api.exceptions.database.DBException;
|
import com.djrapitops.plan.api.exceptions.database.DBException;
|
||||||
import com.djrapitops.plan.system.info.InfoSystem;
|
import com.djrapitops.plan.system.info.InfoSystem;
|
||||||
import com.djrapitops.plan.system.webserver.pages.parsing.InspectPage;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.DefaultResponses;
|
import com.djrapitops.plan.system.webserver.response.DefaultResponses;
|
||||||
import com.djrapitops.plan.system.webserver.response.Response;
|
import com.djrapitops.plan.system.webserver.response.Response;
|
||||||
import com.djrapitops.plan.system.webserver.response.errors.NotFoundResponse;
|
import com.djrapitops.plan.system.webserver.response.errors.NotFoundResponse;
|
||||||
|
import com.djrapitops.plan.utilities.html.pages.InspectPage;
|
||||||
import com.djrapitops.plugin.utilities.Verify;
|
import com.djrapitops.plugin.utilities.Verify;
|
||||||
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
@ -5,11 +5,11 @@ import com.djrapitops.plan.api.exceptions.connection.NoServersException;
|
|||||||
import com.djrapitops.plan.api.exceptions.connection.WebException;
|
import com.djrapitops.plan.api.exceptions.connection.WebException;
|
||||||
import com.djrapitops.plan.system.info.InfoSystem;
|
import com.djrapitops.plan.system.info.InfoSystem;
|
||||||
import com.djrapitops.plan.system.processing.Processing;
|
import com.djrapitops.plan.system.processing.Processing;
|
||||||
import com.djrapitops.plan.system.webserver.pages.parsing.AnalysisPage;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.Response;
|
import com.djrapitops.plan.system.webserver.response.Response;
|
||||||
import com.djrapitops.plan.system.webserver.response.cache.PageId;
|
import com.djrapitops.plan.system.webserver.response.cache.PageId;
|
||||||
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
||||||
import com.djrapitops.plan.system.webserver.response.errors.NotFoundResponse;
|
import com.djrapitops.plan.system.webserver.response.errors.NotFoundResponse;
|
||||||
|
import com.djrapitops.plan.utilities.html.pages.AnalysisPage;
|
||||||
import com.djrapitops.plugin.api.utility.log.Log;
|
import com.djrapitops.plugin.api.utility.log.Log;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
@ -4,40 +4,9 @@
|
|||||||
*/
|
*/
|
||||||
package com.djrapitops.plan.system.webserver.response.pages;
|
package com.djrapitops.plan.system.webserver.response.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.PlanPlugin;
|
|
||||||
import com.djrapitops.plan.data.container.Session;
|
|
||||||
import com.djrapitops.plan.data.store.CachingSupplier;
|
|
||||||
import com.djrapitops.plan.data.store.Key;
|
|
||||||
import com.djrapitops.plan.data.store.keys.SessionKeys;
|
|
||||||
import com.djrapitops.plan.data.store.mutators.formatting.Formatter;
|
|
||||||
import com.djrapitops.plan.data.store.mutators.formatting.Formatters;
|
|
||||||
import com.djrapitops.plan.data.store.objects.DateHolder;
|
|
||||||
import com.djrapitops.plan.system.cache.CacheSystem;
|
|
||||||
import com.djrapitops.plan.system.cache.SessionCache;
|
|
||||||
import com.djrapitops.plan.system.database.databases.Database;
|
|
||||||
import com.djrapitops.plan.system.info.connection.ConnectionLog;
|
|
||||||
import com.djrapitops.plan.system.info.connection.ConnectionSystem;
|
|
||||||
import com.djrapitops.plan.system.info.server.Server;
|
|
||||||
import com.djrapitops.plan.system.info.server.ServerInfo;
|
|
||||||
import com.djrapitops.plan.system.info.server.properties.ServerProperties;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.errors.ErrorResponse;
|
import com.djrapitops.plan.system.webserver.response.errors.ErrorResponse;
|
||||||
import com.djrapitops.plan.utilities.file.FileUtil;
|
|
||||||
import com.djrapitops.plan.utilities.html.Html;
|
|
||||||
import com.djrapitops.plan.utilities.html.HtmlStructure;
|
|
||||||
import com.djrapitops.plan.utilities.html.icon.Icon;
|
import com.djrapitops.plan.utilities.html.icon.Icon;
|
||||||
import com.djrapitops.plan.utilities.html.structure.TabsElement;
|
import com.djrapitops.plan.utilities.html.pages.DebugPage;
|
||||||
import com.djrapitops.plugin.api.Benchmark;
|
|
||||||
import com.djrapitops.plugin.api.utility.log.ErrorLogger;
|
|
||||||
import com.djrapitops.plugin.api.utility.log.Log;
|
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.lang.management.ManagementFactory;
|
|
||||||
import java.lang.management.RuntimeMXBean;
|
|
||||||
import java.nio.charset.Charset;
|
|
||||||
import java.util.*;
|
|
||||||
import java.util.function.Supplier;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WebServer response for /debug-page used for easing issue reporting.
|
* WebServer response for /debug-page used for easing issue reporting.
|
||||||
@ -49,288 +18,8 @@ public class DebugPageResponse extends ErrorResponse {
|
|||||||
public DebugPageResponse() {
|
public DebugPageResponse() {
|
||||||
super.setHeader("HTTP/1.1 200 OK");
|
super.setHeader("HTTP/1.1 200 OK");
|
||||||
super.setTitle(Icon.called("bug") + " Debug Information");
|
super.setTitle(Icon.called("bug") + " Debug Information");
|
||||||
super.setParagraph(buildContent());
|
super.setParagraph(new DebugPage().toHtml());
|
||||||
replacePlaceholders();
|
replacePlaceholders();
|
||||||
}
|
}
|
||||||
|
|
||||||
private String buildContent() {
|
|
||||||
StringBuilder preContent = new StringBuilder();
|
|
||||||
|
|
||||||
String issueLink = Html.LINK_EXTERNAL.parse("https://github.com/Rsl1122/Plan-PlayerAnalytics/issues/new", "Create new issue on Github");
|
|
||||||
String hastebinLink = Html.LINK_EXTERNAL.parse("https://hastebin.com/", "Create a new hastebin paste");
|
|
||||||
|
|
||||||
preContent.append("<p>")
|
|
||||||
.append(HtmlStructure.separateWithDots(issueLink, hastebinLink)).append("<br><br>")
|
|
||||||
.append("This page contains debug information for an issue ticket. You can copy it directly into the issue, the info is pre-formatted.")
|
|
||||||
.append("</p>");
|
|
||||||
|
|
||||||
TabsElement.Tab info = new TabsElement.Tab(Icon.called("server") + " Server Information", createServerInfoContent());
|
|
||||||
TabsElement.Tab errors = new TabsElement.Tab(Icon.called("exclamation-circle") + " Errors", createErrorContent());
|
|
||||||
TabsElement.Tab debugLog = new TabsElement.Tab(Icon.called("bug") + " Debug Log", createDebugLogContent());
|
|
||||||
TabsElement.Tab config = new TabsElement.Tab(Icon.called("cogs") + " Plan Config", createConfigContent());
|
|
||||||
TabsElement.Tab caches = new TabsElement.Tab(Icon.called("archive") + " Plan Caches", createCacheContent());
|
|
||||||
|
|
||||||
TabsElement tabs = new TabsElement(info, errors, debugLog, config, caches);
|
|
||||||
|
|
||||||
return preContent + tabs.toHtmlFull();
|
|
||||||
}
|
|
||||||
|
|
||||||
private String createCacheContent() {
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
appendResponseCache(content);
|
|
||||||
appendSessionCache(content);
|
|
||||||
appendDataContainerCache(content);
|
|
||||||
return content.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendResponseCache(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
content.append("<pre>### Cached Responses:<br><br>");
|
|
||||||
List<String> cacheKeys = new ArrayList<>(ResponseCache.getCacheKeys());
|
|
||||||
if (cacheKeys.isEmpty()) {
|
|
||||||
content.append("Empty");
|
|
||||||
}
|
|
||||||
Collections.sort(cacheKeys);
|
|
||||||
for (String cacheKey : cacheKeys) {
|
|
||||||
content.append("- ").append(cacheKey).append("<br>");
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendSessionCache(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
content.append("<pre>### Session Cache:<br><br>");
|
|
||||||
content.append("UUID | Session Started <br>")
|
|
||||||
.append("-- | -- <br>");
|
|
||||||
Formatter<Long> timeStamp = Formatters.yearLongValue();
|
|
||||||
Set<Map.Entry<UUID, Session>> sessions = SessionCache.getActiveSessions().entrySet();
|
|
||||||
if (sessions.isEmpty()) {
|
|
||||||
content.append("Empty");
|
|
||||||
}
|
|
||||||
for (Map.Entry<UUID, Session> entry : sessions) {
|
|
||||||
UUID uuid = entry.getKey();
|
|
||||||
String start = entry.getValue().getValue(SessionKeys.START).map(timeStamp).orElse("Unknown");
|
|
||||||
content.append(uuid.toString()).append(" | ").append(start).append("<br>");
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendDataContainerCache(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
content.append("<pre>### DataContainer Cache:<br><br>");
|
|
||||||
|
|
||||||
content.append("Key | Is Cached | Cache Time <br>")
|
|
||||||
.append("-- | -- | -- <br>");
|
|
||||||
Formatter<Long> timeStamp = Formatters.yearLongValue();
|
|
||||||
Set<Map.Entry<Key, Supplier>> dataContainers = CacheSystem.getInstance().getDataContainerCache().getMap().entrySet();
|
|
||||||
if (dataContainers.isEmpty()) {
|
|
||||||
content.append("Empty");
|
|
||||||
}
|
|
||||||
for (Map.Entry<Key, Supplier> entry : dataContainers) {
|
|
||||||
String keyName = entry.getKey().getKeyName();
|
|
||||||
Supplier supplier = entry.getValue();
|
|
||||||
if (supplier instanceof CachingSupplier) {
|
|
||||||
CachingSupplier cachingSupplier = (CachingSupplier) supplier;
|
|
||||||
boolean isCached = cachingSupplier.isCached();
|
|
||||||
String cacheText = isCached ? "Yes" : "No";
|
|
||||||
String cacheTime = isCached ? timeStamp.apply(cachingSupplier.getCacheTime()) : "-";
|
|
||||||
content.append(keyName).append(" | ").append(cacheText).append(" | ").append(cacheTime).append("<br>");
|
|
||||||
} else {
|
|
||||||
content.append(keyName).append(" | ").append("Non-caching Supplier").append(" | ").append("-").append("<br>");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private String createConfigContent() {
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
appendConfig(content);
|
|
||||||
return content.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private String createDebugLogContent() {
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
appendDebugLog(content);
|
|
||||||
return content.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private String createErrorContent() {
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
appendLoggedErrors(content);
|
|
||||||
return content.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private String createServerInfoContent() {
|
|
||||||
StringBuilder content = new StringBuilder();
|
|
||||||
|
|
||||||
appendServerInformation(content);
|
|
||||||
appendConnectionLog(content);
|
|
||||||
appendBenchmarks(content);
|
|
||||||
|
|
||||||
return content.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendConnectionLog(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
Map<String, Map<String, ConnectionLog.Entry>> logEntries = ConnectionLog.getLogEntries();
|
|
||||||
|
|
||||||
content.append("<pre>### Connection Log:<br><br>");
|
|
||||||
content.append("Server Address | Request Type | Response | Sent<br>")
|
|
||||||
.append("-- | -- | -- | --<br>");
|
|
||||||
|
|
||||||
Formatter<DateHolder> formatter = Formatters.second();
|
|
||||||
|
|
||||||
if (logEntries.isEmpty()) {
|
|
||||||
content.append("**No Connections Logged**<br>");
|
|
||||||
}
|
|
||||||
for (Map.Entry<String, Map<String, ConnectionLog.Entry>> entry : logEntries.entrySet()) {
|
|
||||||
String address = entry.getKey();
|
|
||||||
Map<String, ConnectionLog.Entry> requests = entry.getValue();
|
|
||||||
for (Map.Entry<String, ConnectionLog.Entry> requestEntry : requests.entrySet()) {
|
|
||||||
String infoRequest = requestEntry.getKey();
|
|
||||||
ConnectionLog.Entry logEntry = requestEntry.getValue();
|
|
||||||
|
|
||||||
content.append(address).append(" | ")
|
|
||||||
.append(infoRequest).append(" | ")
|
|
||||||
.append(logEntry.getResponseCode()).append(" | ")
|
|
||||||
.append(formatter.apply(logEntry)).append("<br>");
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
|
|
||||||
content.append("<pre>### Servers:<br><br>");
|
|
||||||
List<Server> servers = ConnectionSystem.getInstance().getBukkitServers();
|
|
||||||
content.append("Server Name | Address | UUID <br>")
|
|
||||||
.append("-- | -- | --<br>");
|
|
||||||
for (Server server : servers) {
|
|
||||||
content.append(server.getName()).append(" | ")
|
|
||||||
.append(server.getWebAddress()).append(" | ")
|
|
||||||
.append(server.getUuid()).append("<br>");
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendServerInformation(StringBuilder content) {
|
|
||||||
PlanPlugin plugin = PlanPlugin.getInstance();
|
|
||||||
ServerProperties variable = ServerInfo.getServerProperties();
|
|
||||||
|
|
||||||
content.append("<pre>### Server Information<br>")
|
|
||||||
.append("**Plan Version:** ").append(plugin.getVersion()).append("<br>");
|
|
||||||
|
|
||||||
content.append("**Server:** ");
|
|
||||||
content.append(variable.getName())
|
|
||||||
.append(" ").append(variable.getImplVersion())
|
|
||||||
.append(" ").append(variable.getVersion());
|
|
||||||
content.append("<br>");
|
|
||||||
|
|
||||||
Database database = Database.getActive();
|
|
||||||
content.append("**Database:** ").append(database.getName());
|
|
||||||
content.append("<br><br>");
|
|
||||||
|
|
||||||
RuntimeMXBean runtimeMxBean = ManagementFactory.getRuntimeMXBean();
|
|
||||||
Properties properties = System.getProperties();
|
|
||||||
|
|
||||||
String osName = properties.getProperty("os.name");
|
|
||||||
String osVersion = properties.getProperty("os.version");
|
|
||||||
String osArch = properties.getProperty("os.arch");
|
|
||||||
|
|
||||||
String javaVendor = properties.getProperty("java.vendor");
|
|
||||||
String javaVersion = properties.getProperty("java.version");
|
|
||||||
|
|
||||||
String javaVMVendor = properties.getProperty("java.vm.vendor");
|
|
||||||
String javaVMName = properties.getProperty("java.vm.name");
|
|
||||||
String javaVMVersion = properties.getProperty("java.vm.version");
|
|
||||||
List<String> javaVMFlags = runtimeMxBean.getInputArguments();
|
|
||||||
|
|
||||||
content.append("**Operating SubSystem:** ").append(osName).append(" (").append(osArch)
|
|
||||||
.append(") version ").append(osVersion).append("<br>");
|
|
||||||
|
|
||||||
content.append("**Java Version:** ").append(javaVersion).append(", ").append(javaVendor).append("<br>");
|
|
||||||
content.append("**Java VM Version:** ").append(javaVMName).append(" version ").append(javaVMVersion)
|
|
||||||
.append(", ").append(javaVMVendor).append("<br>");
|
|
||||||
content.append("**Java VM Flags:** ").append(javaVMFlags).append("<br>");
|
|
||||||
|
|
||||||
content.append("</pre>");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendConfig(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
File configFile = new File(PlanPlugin.getInstance().getDataFolder(), "config.yml");
|
|
||||||
if (configFile.exists()) {
|
|
||||||
content.append("<pre>### config.yml<br>```<br>");
|
|
||||||
FileUtil.lines(configFile, Charset.forName("UTF-8"))
|
|
||||||
.stream().filter(line -> !line.toLowerCase().contains("pass") && !line.toLowerCase().contains("secret"))
|
|
||||||
.forEach(line -> content.append(line).append("<br>"));
|
|
||||||
content.append("```</pre>");
|
|
||||||
}
|
|
||||||
} catch (IOException e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendBenchmarks(StringBuilder content) {
|
|
||||||
content.append("<pre>### Benchmarks<br>```<br>");
|
|
||||||
try {
|
|
||||||
for (String line : Benchmark.getAverages().asStringArray()) {
|
|
||||||
content.append(line).append("<br>");
|
|
||||||
}
|
|
||||||
} catch (Exception e) {
|
|
||||||
content.append("Exception on Benchmark.getAverages().asStringArray()");
|
|
||||||
}
|
|
||||||
content.append("```</pre>");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendLoggedErrors(StringBuilder content) {
|
|
||||||
try {
|
|
||||||
content.append("<pre>### Logged Errors<br>");
|
|
||||||
|
|
||||||
SortedMap<String, List<String>> errors = ErrorLogger.getLoggedErrors(PlanPlugin.getInstance());
|
|
||||||
|
|
||||||
if (!errors.isEmpty()) {
|
|
||||||
List<String> errorLines = new ArrayList<>();
|
|
||||||
for (Map.Entry<String, List<String>> entry : errors.entrySet()) {
|
|
||||||
StringBuilder errorLineBuilder = new StringBuilder();
|
|
||||||
for (String line : entry.getValue()) {
|
|
||||||
errorLineBuilder.append(line).append("<br>");
|
|
||||||
}
|
|
||||||
String error = errorLineBuilder.toString();
|
|
||||||
if (!errorLines.contains(error)) {
|
|
||||||
errorLines.add(error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (String error : errorLines) {
|
|
||||||
content.append("</pre><pre>```<br>")
|
|
||||||
.append(error)
|
|
||||||
.append("```");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
content.append("**No Errors logged.**<br>");
|
|
||||||
}
|
|
||||||
content.append("</pre>");
|
|
||||||
} catch (IOException e) {
|
|
||||||
Log.toLog(this.getClass(), e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void appendDebugLog(StringBuilder content) {
|
|
||||||
content.append("<pre>### Debug Log<br>```<br>");
|
|
||||||
for (String line : Log.getDebugLogInMemory()) {
|
|
||||||
content.append(line).append("<br>");
|
|
||||||
}
|
|
||||||
content.append("```</pre>");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
package com.djrapitops.plan.system.webserver.response.pages;
|
package com.djrapitops.plan.system.webserver.response.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
||||||
import com.djrapitops.plan.system.webserver.pages.parsing.PlayersPage;
|
|
||||||
import com.djrapitops.plan.system.webserver.response.Response;
|
import com.djrapitops.plan.system.webserver.response.Response;
|
||||||
import com.djrapitops.plan.system.webserver.response.errors.InternalErrorResponse;
|
import com.djrapitops.plan.system.webserver.response.errors.InternalErrorResponse;
|
||||||
|
import com.djrapitops.plan.utilities.html.pages.PlayersPage;
|
||||||
import com.djrapitops.plugin.api.utility.log.Log;
|
import com.djrapitops.plugin.api.utility.log.Log;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
* License is provided in the jar as LICENSE also here:
|
* License is provided in the jar as LICENSE also here:
|
||||||
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
||||||
*/
|
*/
|
||||||
package com.djrapitops.plan.system.webserver.pages.parsing;
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
||||||
import com.djrapitops.plan.data.store.containers.AnalysisContainer;
|
import com.djrapitops.plan.data.store.containers.AnalysisContainer;
|
@ -0,0 +1,325 @@
|
|||||||
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
|
import com.djrapitops.plan.PlanPlugin;
|
||||||
|
import com.djrapitops.plan.data.container.Session;
|
||||||
|
import com.djrapitops.plan.data.store.CachingSupplier;
|
||||||
|
import com.djrapitops.plan.data.store.Key;
|
||||||
|
import com.djrapitops.plan.data.store.keys.SessionKeys;
|
||||||
|
import com.djrapitops.plan.data.store.mutators.formatting.Formatter;
|
||||||
|
import com.djrapitops.plan.data.store.mutators.formatting.Formatters;
|
||||||
|
import com.djrapitops.plan.data.store.objects.DateHolder;
|
||||||
|
import com.djrapitops.plan.system.cache.CacheSystem;
|
||||||
|
import com.djrapitops.plan.system.cache.SessionCache;
|
||||||
|
import com.djrapitops.plan.system.database.databases.Database;
|
||||||
|
import com.djrapitops.plan.system.info.connection.ConnectionLog;
|
||||||
|
import com.djrapitops.plan.system.info.connection.ConnectionSystem;
|
||||||
|
import com.djrapitops.plan.system.info.server.Server;
|
||||||
|
import com.djrapitops.plan.system.info.server.ServerInfo;
|
||||||
|
import com.djrapitops.plan.system.info.server.properties.ServerProperties;
|
||||||
|
import com.djrapitops.plan.system.webserver.response.cache.ResponseCache;
|
||||||
|
import com.djrapitops.plan.utilities.file.FileUtil;
|
||||||
|
import com.djrapitops.plan.utilities.html.Html;
|
||||||
|
import com.djrapitops.plan.utilities.html.HtmlStructure;
|
||||||
|
import com.djrapitops.plan.utilities.html.icon.Icon;
|
||||||
|
import com.djrapitops.plan.utilities.html.structure.TabsElement;
|
||||||
|
import com.djrapitops.plugin.api.Benchmark;
|
||||||
|
import com.djrapitops.plugin.api.utility.log.ErrorLogger;
|
||||||
|
import com.djrapitops.plugin.api.utility.log.Log;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.management.ManagementFactory;
|
||||||
|
import java.lang.management.RuntimeMXBean;
|
||||||
|
import java.nio.charset.Charset;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Html parsing for the Debug page.
|
||||||
|
*
|
||||||
|
* @author Rsl1122
|
||||||
|
*/
|
||||||
|
public class DebugPage implements Page {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toHtml() {
|
||||||
|
StringBuilder preContent = new StringBuilder();
|
||||||
|
|
||||||
|
String issueLink = Html.LINK_EXTERNAL.parse("https://github.com/Rsl1122/Plan-PlayerAnalytics/issues/new", "Create new issue on Github");
|
||||||
|
String hastebinLink = Html.LINK_EXTERNAL.parse("https://hastebin.com/", "Create a new hastebin paste");
|
||||||
|
|
||||||
|
preContent.append("<p>")
|
||||||
|
.append(HtmlStructure.separateWithDots(issueLink, hastebinLink)).append("<br><br>")
|
||||||
|
.append("This page contains debug information for an issue ticket. You can copy it directly into the issue, the info is pre-formatted.")
|
||||||
|
.append("</p>");
|
||||||
|
|
||||||
|
TabsElement.Tab info = new TabsElement.Tab(Icon.called("server") + " Server Information", createServerInfoContent());
|
||||||
|
TabsElement.Tab errors = new TabsElement.Tab(Icon.called("exclamation-circle") + " Errors", createErrorContent());
|
||||||
|
TabsElement.Tab debugLog = new TabsElement.Tab(Icon.called("bug") + " Debug Log", createDebugLogContent());
|
||||||
|
TabsElement.Tab config = new TabsElement.Tab(Icon.called("cogs") + " Plan Config", createConfigContent());
|
||||||
|
TabsElement.Tab caches = new TabsElement.Tab(Icon.called("archive") + " Plan Caches", createCacheContent());
|
||||||
|
|
||||||
|
TabsElement tabs = new TabsElement(info, errors, debugLog, config, caches);
|
||||||
|
|
||||||
|
return preContent + tabs.toHtmlFull();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String createCacheContent() {
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
appendResponseCache(content);
|
||||||
|
appendSessionCache(content);
|
||||||
|
appendDataContainerCache(content);
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendResponseCache(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
content.append("<pre>### Cached Responses:<br><br>");
|
||||||
|
List<String> cacheKeys = new ArrayList<>(ResponseCache.getCacheKeys());
|
||||||
|
if (cacheKeys.isEmpty()) {
|
||||||
|
content.append("Empty");
|
||||||
|
}
|
||||||
|
Collections.sort(cacheKeys);
|
||||||
|
for (String cacheKey : cacheKeys) {
|
||||||
|
content.append("- ").append(cacheKey).append("<br>");
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendSessionCache(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
content.append("<pre>### Session Cache:<br><br>");
|
||||||
|
content.append("UUID | Session Started <br>")
|
||||||
|
.append("-- | -- <br>");
|
||||||
|
Formatter<Long> timeStamp = Formatters.yearLongValue();
|
||||||
|
Set<Map.Entry<UUID, Session>> sessions = SessionCache.getActiveSessions().entrySet();
|
||||||
|
if (sessions.isEmpty()) {
|
||||||
|
content.append("Empty");
|
||||||
|
}
|
||||||
|
for (Map.Entry<UUID, Session> entry : sessions) {
|
||||||
|
UUID uuid = entry.getKey();
|
||||||
|
String start = entry.getValue().getValue(SessionKeys.START).map(timeStamp).orElse("Unknown");
|
||||||
|
content.append(uuid.toString()).append(" | ").append(start).append("<br>");
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendDataContainerCache(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
content.append("<pre>### DataContainer Cache:<br><br>");
|
||||||
|
|
||||||
|
content.append("Key | Is Cached | Cache Time <br>")
|
||||||
|
.append("-- | -- | -- <br>");
|
||||||
|
Formatter<Long> timeStamp = Formatters.yearLongValue();
|
||||||
|
Set<Map.Entry<Key, Supplier>> dataContainers = CacheSystem.getInstance().getDataContainerCache().getMap().entrySet();
|
||||||
|
if (dataContainers.isEmpty()) {
|
||||||
|
content.append("Empty");
|
||||||
|
}
|
||||||
|
for (Map.Entry<Key, Supplier> entry : dataContainers) {
|
||||||
|
String keyName = entry.getKey().getKeyName();
|
||||||
|
Supplier supplier = entry.getValue();
|
||||||
|
if (supplier instanceof CachingSupplier) {
|
||||||
|
CachingSupplier cachingSupplier = (CachingSupplier) supplier;
|
||||||
|
boolean isCached = cachingSupplier.isCached();
|
||||||
|
String cacheText = isCached ? "Yes" : "No";
|
||||||
|
String cacheTime = isCached ? timeStamp.apply(cachingSupplier.getCacheTime()) : "-";
|
||||||
|
content.append(keyName).append(" | ").append(cacheText).append(" | ").append(cacheTime).append("<br>");
|
||||||
|
} else {
|
||||||
|
content.append(keyName).append(" | ").append("Non-caching Supplier").append(" | ").append("-").append("<br>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String createConfigContent() {
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
appendConfig(content);
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String createDebugLogContent() {
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
appendDebugLog(content);
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String createErrorContent() {
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
appendLoggedErrors(content);
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private String createServerInfoContent() {
|
||||||
|
StringBuilder content = new StringBuilder();
|
||||||
|
|
||||||
|
appendServerInformation(content);
|
||||||
|
appendConnectionLog(content);
|
||||||
|
appendBenchmarks(content);
|
||||||
|
|
||||||
|
return content.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendConnectionLog(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
Map<String, Map<String, ConnectionLog.Entry>> logEntries = ConnectionLog.getLogEntries();
|
||||||
|
|
||||||
|
content.append("<pre>### Connection Log:<br><br>");
|
||||||
|
content.append("Server Address | Request Type | Response | Sent<br>")
|
||||||
|
.append("-- | -- | -- | --<br>");
|
||||||
|
|
||||||
|
Formatter<DateHolder> formatter = Formatters.second();
|
||||||
|
|
||||||
|
if (logEntries.isEmpty()) {
|
||||||
|
content.append("**No Connections Logged**<br>");
|
||||||
|
}
|
||||||
|
for (Map.Entry<String, Map<String, ConnectionLog.Entry>> entry : logEntries.entrySet()) {
|
||||||
|
String address = entry.getKey();
|
||||||
|
Map<String, ConnectionLog.Entry> requests = entry.getValue();
|
||||||
|
for (Map.Entry<String, ConnectionLog.Entry> requestEntry : requests.entrySet()) {
|
||||||
|
String infoRequest = requestEntry.getKey();
|
||||||
|
ConnectionLog.Entry logEntry = requestEntry.getValue();
|
||||||
|
|
||||||
|
content.append(address).append(" | ")
|
||||||
|
.append(infoRequest).append(" | ")
|
||||||
|
.append(logEntry.getResponseCode()).append(" | ")
|
||||||
|
.append(formatter.apply(logEntry)).append("<br>");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
|
||||||
|
content.append("<pre>### Servers:<br><br>");
|
||||||
|
List<Server> servers = ConnectionSystem.getInstance().getBukkitServers();
|
||||||
|
content.append("Server Name | Address | UUID <br>")
|
||||||
|
.append("-- | -- | --<br>");
|
||||||
|
for (Server server : servers) {
|
||||||
|
content.append(server.getName()).append(" | ")
|
||||||
|
.append(server.getWebAddress()).append(" | ")
|
||||||
|
.append(server.getUuid()).append("<br>");
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendServerInformation(StringBuilder content) {
|
||||||
|
PlanPlugin plugin = PlanPlugin.getInstance();
|
||||||
|
ServerProperties variable = ServerInfo.getServerProperties();
|
||||||
|
|
||||||
|
content.append("<pre>### Server Information<br>")
|
||||||
|
.append("**Plan Version:** ").append(plugin.getVersion()).append("<br>");
|
||||||
|
|
||||||
|
content.append("**Server:** ");
|
||||||
|
content.append(variable.getName())
|
||||||
|
.append(" ").append(variable.getImplVersion())
|
||||||
|
.append(" ").append(variable.getVersion());
|
||||||
|
content.append("<br>");
|
||||||
|
|
||||||
|
Database database = Database.getActive();
|
||||||
|
content.append("**Database:** ").append(database.getName());
|
||||||
|
content.append("<br><br>");
|
||||||
|
|
||||||
|
RuntimeMXBean runtimeMxBean = ManagementFactory.getRuntimeMXBean();
|
||||||
|
Properties properties = System.getProperties();
|
||||||
|
|
||||||
|
String osName = properties.getProperty("os.name");
|
||||||
|
String osVersion = properties.getProperty("os.version");
|
||||||
|
String osArch = properties.getProperty("os.arch");
|
||||||
|
|
||||||
|
String javaVendor = properties.getProperty("java.vendor");
|
||||||
|
String javaVersion = properties.getProperty("java.version");
|
||||||
|
|
||||||
|
String javaVMVendor = properties.getProperty("java.vm.vendor");
|
||||||
|
String javaVMName = properties.getProperty("java.vm.name");
|
||||||
|
String javaVMVersion = properties.getProperty("java.vm.version");
|
||||||
|
List<String> javaVMFlags = runtimeMxBean.getInputArguments();
|
||||||
|
|
||||||
|
content.append("**Operating SubSystem:** ").append(osName).append(" (").append(osArch)
|
||||||
|
.append(") version ").append(osVersion).append("<br>");
|
||||||
|
|
||||||
|
content.append("**Java Version:** ").append(javaVersion).append(", ").append(javaVendor).append("<br>");
|
||||||
|
content.append("**Java VM Version:** ").append(javaVMName).append(" version ").append(javaVMVersion)
|
||||||
|
.append(", ").append(javaVMVendor).append("<br>");
|
||||||
|
content.append("**Java VM Flags:** ").append(javaVMFlags).append("<br>");
|
||||||
|
|
||||||
|
content.append("</pre>");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendConfig(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
File configFile = new File(PlanPlugin.getInstance().getDataFolder(), "config.yml");
|
||||||
|
if (configFile.exists()) {
|
||||||
|
content.append("<pre>### config.yml<br>```<br>");
|
||||||
|
FileUtil.lines(configFile, Charset.forName("UTF-8"))
|
||||||
|
.stream().filter(line -> !line.toLowerCase().contains("pass") && !line.toLowerCase().contains("secret"))
|
||||||
|
.forEach(line -> content.append(line).append("<br>"));
|
||||||
|
content.append("```</pre>");
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendBenchmarks(StringBuilder content) {
|
||||||
|
content.append("<pre>### Benchmarks<br>```<br>");
|
||||||
|
try {
|
||||||
|
for (String line : Benchmark.getAverages().asStringArray()) {
|
||||||
|
content.append(line).append("<br>");
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
content.append("Exception on Benchmark.getAverages().asStringArray()");
|
||||||
|
}
|
||||||
|
content.append("```</pre>");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendLoggedErrors(StringBuilder content) {
|
||||||
|
try {
|
||||||
|
content.append("<pre>### Logged Errors<br>");
|
||||||
|
|
||||||
|
SortedMap<String, List<String>> errors = ErrorLogger.getLoggedErrors(PlanPlugin.getInstance());
|
||||||
|
|
||||||
|
if (!errors.isEmpty()) {
|
||||||
|
List<String> errorLines = new ArrayList<>();
|
||||||
|
for (Map.Entry<String, List<String>> entry : errors.entrySet()) {
|
||||||
|
StringBuilder errorLineBuilder = new StringBuilder();
|
||||||
|
for (String line : entry.getValue()) {
|
||||||
|
errorLineBuilder.append(line).append("<br>");
|
||||||
|
}
|
||||||
|
String error = errorLineBuilder.toString();
|
||||||
|
if (!errorLines.contains(error)) {
|
||||||
|
errorLines.add(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (String error : errorLines) {
|
||||||
|
content.append("</pre><pre>```<br>")
|
||||||
|
.append(error)
|
||||||
|
.append("```");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
content.append("**No Errors logged.**<br>");
|
||||||
|
}
|
||||||
|
content.append("</pre>");
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.toLog(this.getClass(), e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void appendDebugLog(StringBuilder content) {
|
||||||
|
content.append("<pre>### Debug Log<br>```<br>");
|
||||||
|
for (String line : Log.getDebugLogInMemory()) {
|
||||||
|
content.append(line).append("<br>");
|
||||||
|
}
|
||||||
|
content.append("```</pre>");
|
||||||
|
}
|
||||||
|
}
|
@ -2,7 +2,7 @@
|
|||||||
* License is provided in the jar as LICENSE also here:
|
* License is provided in the jar as LICENSE also here:
|
||||||
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
||||||
*/
|
*/
|
||||||
package com.djrapitops.plan.system.webserver.pages.parsing;
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
||||||
import com.djrapitops.plan.data.container.Session;
|
import com.djrapitops.plan.data.container.Session;
|
@ -2,7 +2,7 @@
|
|||||||
* License is provided in the jar as LICENSE also here:
|
* License is provided in the jar as LICENSE also here:
|
||||||
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
||||||
*/
|
*/
|
||||||
package com.djrapitops.plan.system.webserver.pages.parsing;
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
||||||
import com.djrapitops.plan.data.store.containers.NetworkContainer;
|
import com.djrapitops.plan.data.store.containers.NetworkContainer;
|
@ -2,7 +2,7 @@
|
|||||||
* License is provided in the jar as LICENSE also here:
|
* License is provided in the jar as LICENSE also here:
|
||||||
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
* https://github.com/Rsl1122/Plan-PlayerAnalytics/blob/master/Plan/src/main/resources/LICENSE
|
||||||
*/
|
*/
|
||||||
package com.djrapitops.plan.system.webserver.pages.parsing;
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
||||||
|
|
@ -1,4 +1,4 @@
|
|||||||
package com.djrapitops.plan.system.webserver.pages.parsing;
|
package com.djrapitops.plan.utilities.html.pages;
|
||||||
|
|
||||||
import com.djrapitops.plan.PlanPlugin;
|
import com.djrapitops.plan.PlanPlugin;
|
||||||
import com.djrapitops.plan.api.exceptions.ParseException;
|
import com.djrapitops.plan.api.exceptions.ParseException;
|
Loading…
Reference in New Issue
Block a user