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

co.aikar.timings.Timings Maven / Gradle / Ivy

/*
 * This file is licensed under the MIT License (MIT).
 *
 * Copyright (c) 2014 Daniel Ennis 
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
package co.aikar.timings;

import cn.nukkit.Server;
import cn.nukkit.api.PowerNukkitOnly;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import cn.nukkit.blockentity.BlockEntity;
import cn.nukkit.command.Command;
import cn.nukkit.entity.Entity;
import cn.nukkit.event.Event;
import cn.nukkit.event.Listener;
import cn.nukkit.network.protocol.DataPacket;
import cn.nukkit.plugin.EventExecutor;
import cn.nukkit.plugin.MethodEventExecutor;
import cn.nukkit.plugin.Plugin;
import cn.nukkit.scheduler.PluginTask;
import cn.nukkit.scheduler.TaskHandler;
import lombok.extern.log4j.Log4j2;

import java.util.HashSet;
import java.util.Queue;
import java.util.Set;

import static co.aikar.timings.TimingIdentifier.DEFAULT_GROUP;

@Log4j2
public final class Timings {
    private static final boolean timingsCompletelyClosed;
    private static boolean timingsEnabled = false;
    private static boolean verboseEnabled = false;
    private static boolean privacy = false;
    private static Set ignoredConfigSections = new HashSet<>();

    private static final int MAX_HISTORY_FRAMES = 12;
    private static int historyInterval = -1;
    private static int historyLength = -1;
    //当timings完全关闭时代替使用的占位Timer
    //Placeholder Timer to use instead when timings are completely off
    public static final EmptyTiming emptyTimer;
    public static final FullServerTickTiming fullServerTickTimer;
    public static final Timing timingsTickTimer;
    public static final Timing pluginEventTimer;

    public static final Timing connectionTimer;
    public static final Timing schedulerTimer;
    public static final Timing schedulerAsyncTimer;
    public static final Timing schedulerSyncTimer;
    public static final Timing commandTimer;
    public static final Timing serverCommandTimer;
    public static final Timing levelSaveTimer;

    public static final Timing playerNetworkSendTimer;
    public static final Timing playerNetworkReceiveTimer;
    public static final Timing playerChunkOrderTimer;
    public static final Timing playerChunkSendTimer;
    public static final Timing playerCommandTimer;

    public static final Timing tickEntityTimer;
    public static final Timing tickBlockEntityTimer;
    public static final Timing entityMoveTimer;
    public static final Timing entityBaseTickTimer;
    public static final Timing livingEntityBaseTickTimer;

    public static final Timing generationTimer;
    public static final Timing populationTimer;
    public static final Timing generationCallbackTimer;

    public static final Timing permissibleCalculationTimer;
    public static final Timing permissionDefaultTimer;

    static {
        if (Server.getInstance() != null) {
            timingsCompletelyClosed = Server.getInstance().getConfig("timings.completely-close", true);
            if (!timingsCompletelyClosed) {
                setTimingsEnabled(Server.getInstance().getConfig("timings.enabled", false));
                setVerboseEnabled(Server.getInstance().getConfig("timings.verbose", false));
            } else {
                setTimingsEnabled(false);
                setVerboseEnabled(false);
            }
            setHistoryInterval(Server.getInstance().getConfig("timings.history-interval", 6000));
            setHistoryLength(Server.getInstance().getConfig("timings.history-length", 72000));
            privacy = Server.getInstance().getConfig("timings.privacy", false);
            ignoredConfigSections.addAll(Server.getInstance().getConfig().getStringList("timings.ignore"));
        } else timingsCompletelyClosed = true;

        if (!timingsCompletelyClosed) log.debug("Timings: \n" +
                        "Enabled - {}\n" +
                        "Verbose - {}\n" +
                        "History Interval - {}\n" +
                        "History Length - {}",
                isTimingsEnabled(), isVerboseEnabled(), getHistoryInterval(), getHistoryLength());

        emptyTimer = new EmptyTiming();
        fullServerTickTimer = new FullServerTickTiming();
        
        timingsTickTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming(DEFAULT_GROUP.name, "Timings Tick", fullServerTickTimer);
        pluginEventTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Plugin Events");

        connectionTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Connection Handler");
        schedulerTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Scheduler");
        schedulerAsyncTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## Scheduler - Async Tasks");
        schedulerSyncTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## Scheduler - Sync Tasks");
        commandTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Commands");
        serverCommandTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Server Command");
        levelSaveTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Level Save");

        playerNetworkSendTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Player Network Send");
        playerNetworkReceiveTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Player Network Receive");
        playerChunkOrderTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Player Order Chunks");
        playerChunkSendTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Player Send Chunks");
        playerCommandTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Player Command");

        tickEntityTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## Entity Tick");
        tickBlockEntityTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## BlockEntity Tick");
        entityMoveTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## Entity Move");
        entityBaseTickTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## Entity Base Tick");
        livingEntityBaseTickTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("## LivingEntity Base Tick");

        generationTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Level Generation");
        populationTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Level Population");
        generationCallbackTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Level Generation Callback");

        permissibleCalculationTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Permissible Calculation");
        permissionDefaultTimer = timingsCompletelyClosed ? emptyTimer : TimingsManager.getTiming("Default Permission Calculation");
    }

    /**
     * Initialize the static fields.
     */
    @PowerNukkitOnly
    @Since("1.5.1.0-PN")
    public static void init() {
        // code is already executed on 
    }

    /**
     * Timings是否被完全关闭了
     * @return boolean
     */
    @PowerNukkitXOnly
    @Since("1.19.50-r3")
    public static boolean isTimingsCloseCompletely() {
        return timingsCompletelyClosed;
    }

    public static boolean isTimingsEnabled() {
        return timingsEnabled;
    }

    public static void setTimingsEnabled(boolean enabled) {
        if (timingsCompletelyClosed) return;
        timingsEnabled = enabled;
        TimingsManager.reset();
    }

    public static boolean isVerboseEnabled() {
        return verboseEnabled;
    }

    public static void setVerboseEnabled(boolean enabled) {
        verboseEnabled = enabled;
        TimingsManager.needsRecheckEnabled = true;
    }

    public static boolean isPrivacy() {
        return privacy;
    }

    public static Set getIgnoredConfigSections() {
        return ignoredConfigSections;
    }

    public static int getHistoryInterval() {
        return historyInterval;
    }

    public static void setHistoryInterval(int interval) {
        historyInterval = Math.max(20 * 60, interval);
        //Recheck the history length with the new Interval
        if (historyLength != -1) {
            setHistoryLength(historyLength);
        }
    }

    public static int getHistoryLength() {
        return historyLength;
    }

    public static void setHistoryLength(int length) {
        //Cap at 12 History Frames, 1 hour at 5 minute frames.
        int maxLength = historyInterval * MAX_HISTORY_FRAMES;
        //For special cases of servers with special permission to bypass the max.
        //This max helps keep data file sizes reasonable for processing on Aikar's Timing parser side.
        //Setting this will not help you bypass the max unless Aikar has added an exception on the API side.
        if (Server.getInstance().getConfig().getBoolean("timings.bypass-max", false)) {
            maxLength = Integer.MAX_VALUE;
        }

        historyLength = Math.max(Math.min(maxLength, length), historyInterval);

        Queue oldQueue = TimingsManager.HISTORY;
        int frames = (getHistoryLength() / getHistoryInterval());
        if (length > maxLength) {
            log.warn(
                    "Timings Length too high. Requested {}, max is {}"
                            + ". To get longer history, you must increase your interval. Set Interval to {}"
                            + " to achieve this length.",
                    length, maxLength, Math.ceil((float) length / MAX_HISTORY_FRAMES));
        }

        TimingsManager.HISTORY = new TimingsManager.BoundedQueue<>(frames);
        TimingsManager.HISTORY.addAll(oldQueue);
    }

    public static void reset() {
        TimingsManager.reset();
    }


    public static Timing getCommandTiming(Command command) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        return TimingsManager.getTiming(DEFAULT_GROUP.name, "Command: " + command.getLabel(), commandTimer);
    }

    public static Timing getTaskTiming(TaskHandler handler, long period) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        String repeating = " ";
        if (period > 0) {
            repeating += "(interval:" + period + ")";
        } else {
            repeating += "(Single)";
        }

        if (handler.getTask() instanceof PluginTask) {
            String owner = ((PluginTask) handler.getTask()).getOwner().getName();
            return TimingsManager.getTiming(owner, "PluginTask: " + handler.getTaskId() + repeating, schedulerSyncTimer, false);
        } else if (!handler.isAsynchronous()) {
            return TimingsManager.getTiming(DEFAULT_GROUP.name, "Task: " + handler.getTaskId() + repeating, schedulerSyncTimer, false);
        } else {
            return null;
        }
    }

    public static Timing getPluginEventTiming(Class event, Listener listener, EventExecutor executor, Plugin plugin) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        Timing group = TimingsManager.getTiming(plugin.getName(), "Combined Total", pluginEventTimer);

        return TimingsManager.getTiming(plugin.getName(), "Event: " + listener.getClass().getName() + "."
                + (executor instanceof MethodEventExecutor ? ((MethodEventExecutor) executor).getMethod().getName() : "???")
                + " (" + event.getSimpleName() + ")", group);
    }

    public static Timing getEntityTiming(Entity entity) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        return TimingsManager.getTiming(DEFAULT_GROUP.name, "## Entity Tick: " + entity.getClass().getSimpleName(), tickEntityTimer);
    }

    public static Timing getBlockEntityTiming(BlockEntity blockEntity) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        return TimingsManager.getTiming(DEFAULT_GROUP.name, "## BlockEntity Tick: " + blockEntity.getClass().getSimpleName(), tickBlockEntityTimer);
    }

    public static Timing getReceiveDataPacketTiming(DataPacket pk) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        return TimingsManager.getTiming(DEFAULT_GROUP.name, "## Receive Packet: " + pk.getClass().getSimpleName(), playerNetworkReceiveTimer);
    }

    public static Timing getSendDataPacketTiming(DataPacket pk) {
        if (Timings.isTimingsCloseCompletely()) return Timings.emptyTimer;
        return TimingsManager.getTiming(DEFAULT_GROUP.name, "## Send Packet: " + pk.getClass().getSimpleName(), playerNetworkSendTimer);
    }

    public static void stopServer() {
        setTimingsEnabled(false);
        TimingsManager.recheckEnabled();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy