All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.nukkit.command.defaults.DebugPasteCommand Maven / Gradle / Ivy

package cn.nukkit.command.defaults;

import cn.nukkit.Nukkit;
import cn.nukkit.Server;
import cn.nukkit.command.CapturingCommandSender;
import cn.nukkit.command.CommandSender;
import cn.nukkit.command.ConsoleCommandSender;
import cn.nukkit.command.data.CommandParameter;
import cn.nukkit.network.protocol.ProtocolInfo;
import cn.nukkit.plugin.Plugin;
import cn.nukkit.plugin.PluginDescription;
import cn.nukkit.scheduler.AsyncTask;
import cn.nukkit.utils.HumanStringComparator;
import cn.nukkit.utils.Utils;
import com.nimbusds.jose.util.IOUtils;
import lombok.extern.log4j.Log4j2;
import org.iq80.leveldb.util.FileUtils;

import org.jetbrains.annotations.NotNull;

import java.io.*;
import java.lang.management.ManagementFactory;
import java.net.HttpURLConnection;
import java.net.URL;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.attribute.BasicFileAttributes;
import java.nio.file.attribute.FileTime;
import java.text.SimpleDateFormat;
import java.util.Comparator;
import java.util.Date;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;

@Log4j2
public class DebugPasteCommand extends TestCommand implements CoreCommand {

    private static final String ENDPOINT = "https://debugpaste.powernukkit.org/paste.php";
    private static final String USER_AGENT = "PowerNukkit/" + Nukkit.VERSION;

    public DebugPasteCommand(String name) {
        super(name, "%nukkit.command.debug.description", "%nukkit.command.debug.usage");
        this.setPermission("nukkit.command.debug.perform");
        this.commandParameters.clear();
        this.commandParameters.put("clear", new CommandParameter[]{
                CommandParameter.newEnum("clear", new String[]{"clear"})
        });

        this.commandParameters.put("upload", new CommandParameter[]{
                CommandParameter.newEnum("upload", new String[]{"upload"}),
                CommandParameter.newEnum("last", true, new String[]{"last"})
        });

        this.commandParameters.put("default", CommandParameter.EMPTY_ARRAY);
    }

    private static boolean filterValidPastes(Path file) {
        String name = file.getFileName().toString();
        return name.startsWith("debugpaste-") && name.endsWith(".zip");
    }

    private static void clear(CommandSender sender) {
        Path dataPath = Paths.get(sender.getServer().getDataPath()).toAbsolutePath();
        Path pastesFolder = dataPath.resolve("debugpastes");
        try {
            AtomicInteger count = new AtomicInteger();
            if (Files.isDirectory(pastesFolder)) {
                try (Stream listing = Files.list(pastesFolder)) {
                    listing.filter(DebugPasteCommand::filterValidPastes)
                            .forEach(file -> {
                                try {
                                    Files.delete(file);
                                    count.incrementAndGet();
                                } catch (IOException e) {
                                    log.error("Could not delete {}", file, e);
                                }
                            });
                }
            }
            if (count.get() == 0) {
                sender.sendMessage("The debug pastes folder is already clean");
                return;
            }
            log.info("{} debug pastes were deleted by {}", count.get(), sender.getName());
            sender.sendMessage("All " + count.get() + " debug pastes were deleted");
        } catch (Exception e) {
            sender.sendMessage("Oh no! An error has occurred! " + e);
            log.error("Failed to delete {}", dataPath, e);
        }
    }

    private static void uploadLast(CommandSender sender) {
        Path dataPath = Paths.get(sender.getServer().getDataPath()).toAbsolutePath();
        Path pastesFolder = dataPath.resolve("debugpastes");
        Optional last = Optional.empty();
        try {
            try (Stream listing = Files.list(pastesFolder)) {
                last = listing.filter(Files::isRegularFile)
                        .filter(DebugPasteCommand::filterValidPastes)
                        .max(Comparator.comparing(file -> {
                            try {
                                BasicFileAttributes attributes = Files.readAttributes(file, BasicFileAttributes.class);
                                return attributes.creationTime().toMillis();
                            } catch (IOException e) {
                                throw new UncheckedIOException(e);
                            }
                        }));
            } catch (UncheckedIOException e) {
                throw e.getCause();
            }

            if (!last.isPresent()) {
                sender.sendMessage("No debug pastes was found. Try to run \"/debugpaste upload\" to create a new one and send right away.");
                return;
            }

            Path lastPath = last.get();
            Path urlPath = lastPath.resolveSibling(lastPath.getFileName() + ".url");
            if (Files.isRegularFile(urlPath)) {
                Optional url = Files.lines(urlPath).filter(line -> line.startsWith("URL=http") && line.length() > 8).findFirst();
                if (url.isPresent()) {
                    sender.sendMessage("The last debug paste " + lastPath.getFileName() + " was already uploaded to:");
                    String directUrl = url.get().substring(4);
                    sender.sendMessage(directUrl);

                    if (!(sender instanceof ConsoleCommandSender)) {
                        sender.sendMessage("The url is also being logged in the console for convenience, so you can do CTRL+C, CTRL+V if you have console access.");
                        log.info("The last debug paste {} was already uploaded to: {}", lastPath.getFileName(), directUrl);
                    }
                    return;
                }
                String fileName = urlPath.getFileName().toString();
                Files.move(urlPath, urlPath.resolveSibling(fileName.substring(0, fileName.length() - 4) + System.currentTimeMillis() + ".url"));
            }

            upload(sender, lastPath);
        } catch (IOException e) {
            log.error("Failed to find the last debug paste", e);
            sender.sendMessage("Sorry, an error has occurred. Check the logs for details. " + e);
        }
    }

    private static void upload(CommandSender sender, Path zipPath) {
        sender.sendMessage("Uploading...");
        try {
            final URL url = new URL(ENDPOINT);
            final HttpURLConnection connection = (HttpURLConnection) url.openConnection();

            connection.setRequestMethod("PUT");
            connection.setRequestProperty("User-Agent", USER_AGENT);
            connection.setRequestProperty("Content-Type", "application/zip");
            connection.setDoOutput(true);

            try (BufferedOutputStream bos = new BufferedOutputStream(connection.getOutputStream())) {
                Files.copy(zipPath, bos);
            }

            int code = connection.getResponseCode();
            if (code != 201) {
                throw new IOException("The server responded with code " + code + ", expected 201");
            }

            try (BufferedInputStream in = new BufferedInputStream(connection.getInputStream());
                 InputStreamReader rd = connection.getContentEncoding() != null ? new InputStreamReader(in, connection.getContentEncoding()) : new InputStreamReader(in);
                 BufferedReader reader = new BufferedReader(rd)
            ) {
                String response = reader.readLine();
                if (log.isDebugEnabled()) {
                    StringBuilder sb = new StringBuilder().append(response);
                    String line;
                    while ((line = reader.readLine()) != null) {
                        sb.append(line).append("\n");
                    }
                    String fullReturn = sb.toString();
                    if (!fullReturn.equals(response)) {
                        log.debug(fullReturn);
                    }
                }

                URL publicUrl = new URL(response);
                log.info("The debug paste {} was uploaded to {}", zipPath.getFileName(), publicUrl);
                if (sender.isPlayer()) {
                    sender.sendMessage("Your paste was uploaded to: " + publicUrl);
                }
                Utils.writeFile(zipPath.resolveSibling(zipPath.getFileName() + ".url").toString(),
                        "[InternetShortcut]" + System.lineSeparator() +
                                "URL=" + publicUrl + System.lineSeparator());
            }
        } catch (Exception e) {
            log.error("Failed to upload the debugpaste {}", zipPath, e);
            sender.sendMessage("Failed to upload the debugpaste, the file is still available in your server directory.");
        }
    }

    @NotNull
    private static String eval(String... command) {
        try {
            try (InputStream in = Runtime.getRuntime().exec(command).getInputStream()) {
                return IOUtils.readInputStreamToString(in, Charset.defaultCharset()).trim();
            }
        } catch (Exception e) {
            return e.toString().trim();
        }
    }

    @Override
    public boolean execute(CommandSender sender, String commandLabel, String[] args) {
        if (!this.testPermission(sender)) {
            return false;
        }
        sender.sendMessage("The /debugpaste is executing, please wait...");
        Server server = Server.getInstance();
        if ((args.length != 1 || !"clear".equalsIgnoreCase(args[0])) &&
                TimeUnit.MILLISECONDS.toMinutes(System.currentTimeMillis() - server.getLaunchTime()) < 15) {
            sender.sendMessage("Tip: This command works better if you use it right after you experience an issue");
        }
        server.getScheduler().scheduleAsyncTask(new AsyncTask() {
            @Override
            public void onRun() {
                if (args.length == 1 && "clear".equalsIgnoreCase(args[0])) {
                    clear(sender);
                    return;
                }

                if (args.length == 2 && "upload".equalsIgnoreCase(args[0]) && "last".equalsIgnoreCase(args[1])) {
                    uploadLast(sender);
                    return;
                }

                String now = new SimpleDateFormat("yyyy-MM-dd'T'HH.mm.ss.SSSZ").format(new Date());
                Path zipPath;
                try {
                    Path dataPath = Paths.get(server.getDataPath()).toAbsolutePath();
                    Path dir = Files.createDirectories(dataPath.resolve("debugpastes/debugpaste-" + now)).toAbsolutePath();

                    Utils.writeFile(dir.resolve("thread-dump.txt").toString(), Utils.getAllThreadDumps());

                    CapturingCommandSender capturing = new CapturingCommandSender();
                    capturing.setOp(true);
                    new StatusCommand("status").execute(capturing, "status", new String[]{});
                    Utils.writeFile(dir.resolve("status.txt").toString(), capturing.getCleanCapture());

                    Optional secondMostLatest = StreamSupport.stream(Files.newDirectoryStream(dataPath.resolve("logs")).spliterator(), false)
                            .filter(path -> path.toString().toLowerCase().endsWith(".log.gz"))
                            .max(((a, b) -> {
                                FileTime aTime = null, bTime = null;
                                try {
                                    aTime = Files.readAttributes(a, BasicFileAttributes.class).creationTime();
                                } catch (IOException ignored) {
                                }
                                try {
                                    bTime = Files.readAttributes(b, BasicFileAttributes.class).creationTime();
                                } catch (IOException ignored) {
                                }
                                if (aTime == null && bTime != null) {
                                    return 1;
                                }
                                if (aTime != null && bTime == null) {
                                    return -1;
                                }
                                if (aTime == null) {
                                    return 0;
                                }
                                int comp = aTime.compareTo(bTime);
                                if (comp != 0) {
                                    return comp;
                                }
                                return HumanStringComparator.getInstance().compare(
                                        a.getFileName().toString().toLowerCase(),
                                        b.getFileName().toString().toLowerCase()
                                );
                            }));

                    Utils.copyFile(dataPath.resolve("logs/server.log").toFile(), dir.resolve("server-latest.log").toFile());
                    if (secondMostLatest.isPresent()) {
                        Utils.copyFile(secondMostLatest.get().toFile(), dir.resolve("server-second-most-latest.log.gz").toFile());
                    }

                    Utils.copyFile(dataPath.resolve("nukkit.yml").toFile(), dir.resolve("nukkit.yml").toFile());
                    Utils.copyFile(dataPath.resolve("server.properties").toFile(), dir.resolve("server.properties").toFile());

                    StringBuilder b = new StringBuilder();
                    b.append("\n# Server Information\n");
                    b.append("server.name: ").append(server.getName()).append('\n');
                    b.append("version.api: ").append(server.getApiVersion()).append('\n');
                    b.append("version.nukkit: ").append(server.getNukkitVersion()).append('\n');
                    b.append("version.git: ").append(server.getGitCommit()).append('\n');
                    b.append("version.codename: ").append(server.getCodename()).append('\n');
                    b.append("version.minecraft: ").append(server.getVersion()).append('\n');
                    b.append("version.protocol: ").append(ProtocolInfo.CURRENT_PROTOCOL).append('\n');
                    b.append("plugins:");
                    for (Plugin plugin : server.getPluginManager().getPlugins().values()) {
                        boolean enabled = plugin.isEnabled();
                        String name = plugin.getName();
                        PluginDescription desc = plugin.getDescription();
                        String version = desc.getVersion();
                        b.append("\n  ")
                                .append(name)
                                .append(":\n    ")
                                .append("version: '")
                                .append(version)
                                .append('\'')
                                .append("\n    enabled: ")
                                .append(enabled);
                    }
                    b.append("\n\n# Java Details\n");
                    Runtime runtime = Runtime.getRuntime();
                    b.append("memory.free: ").append(runtime.freeMemory()).append('\n');
                    b.append("memory.max: ").append(runtime.maxMemory()).append('\n');
                    b.append("cpu.runtime: ").append(ManagementFactory.getRuntimeMXBean().getUptime()).append('\n');
                    b.append("cpu.processors: ").append(runtime.availableProcessors()).append('\n');
                    b.append("java.specification.version: '").append(System.getProperty("java.specification.version")).append("'\n");
                    b.append("java.vendor: '").append(System.getProperty("java.vendor")).append("'\n");
                    b.append("java.version: '").append(System.getProperty("java.version")).append("'\n");
                    b.append("os.arch: '").append(System.getProperty("os.arch")).append("'\n");
                    b.append("os.name: '").append(System.getProperty("os.name")).append("'\n");
                    b.append("os.version: '").append(System.getProperty("os.version")).append("'\n");
                    b.append("ulimit:\n").append(eval("sh", "-c", "ulimit -a")).append("\n\n");
                    b.append("\n# Create a ticket: https://github.com/PowerNukkit/PowerNukkit/issues/new");

                    Utils.writeFile(dir.resolve("server-info.txt").toString(), b.toString());

                    zipPath = dir.resolveSibling(dir.getFileName() + ".zip");
                    Utils.zipFolder(dir, zipPath);
                    Path relative = dataPath.relativize(zipPath);
                    log.info("A debug paste was created at {}", relative);
                    if (sender.isPlayer()) {
                        sender.sendMessage("A debug paste has been saved in " + relative);
                    }
                    FileUtils.deleteRecursively(dir.toFile());
                } catch (IOException e) {
                    log.error("Failed to create a debugpaste in debugpastes/debugpaste-{}", now, e);
                    if (sender.isPlayer()) {
                        sender.sendMessage("An error has occurred: " + e);
                        sender.sendMessage("A partial paste might be available in debugpastes/debugpaste-" + now);
                    }
                    return;
                }

                if (args.length == 1 && "upload".equalsIgnoreCase(args[0])) {
                    upload(sender, zipPath);
                } else {
                    sender.sendMessage("Review the file and scrub the files as you wish, and run \"/debugpaste upload last\" to make it available online");
                }
            }
        });
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy