mirror of
https://github.com/BX-Team/DivineMC.git
synced 2025-12-21 07:49:18 +00:00
add sentry integration
This commit is contained in:
@@ -0,0 +1,132 @@
|
||||
package gg.pufferfish.pufferfish.sentry;
|
||||
|
||||
import com.google.common.reflect.TypeToken;
|
||||
import com.google.gson.Gson;
|
||||
import io.sentry.Breadcrumb;
|
||||
import io.sentry.Sentry;
|
||||
import io.sentry.SentryEvent;
|
||||
import io.sentry.SentryLevel;
|
||||
import io.sentry.protocol.Message;
|
||||
import io.sentry.protocol.User;
|
||||
import org.apache.logging.log4j.Level;
|
||||
import org.apache.logging.log4j.LogManager;
|
||||
import org.apache.logging.log4j.Marker;
|
||||
import org.apache.logging.log4j.core.LogEvent;
|
||||
import org.apache.logging.log4j.core.Logger;
|
||||
import org.apache.logging.log4j.core.appender.AbstractAppender;
|
||||
import org.apache.logging.log4j.core.filter.AbstractFilter;
|
||||
import org.bxteam.divinemc.DivineConfig;
|
||||
|
||||
import java.util.Map;
|
||||
|
||||
public class PufferfishSentryAppender extends AbstractAppender {
|
||||
private static final org.apache.logging.log4j.Logger LOGGER = LogManager.getLogger(PufferfishSentryAppender.class.getSimpleName());
|
||||
|
||||
private static final Gson GSON = new Gson();
|
||||
private final Level logLevel;
|
||||
|
||||
public PufferfishSentryAppender(Level logLevel) {
|
||||
super("PufferfishSentryAdapter", new SentryFilter(), null);
|
||||
this.logLevel = logLevel;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void append(LogEvent logEvent) {
|
||||
if (logEvent.getLevel().isMoreSpecificThan(logLevel) && (logEvent.getThrown() != null || !DivineConfig.onlyLogThrown)) {
|
||||
try {
|
||||
logException(logEvent);
|
||||
} catch (Exception e) {
|
||||
LOGGER.warn("Failed to log event with sentry", e);
|
||||
}
|
||||
} else {
|
||||
try {
|
||||
logBreadcrumb(logEvent);
|
||||
} catch (Exception e) {
|
||||
LOGGER.warn("Failed to log event with sentry", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void logException(LogEvent e) {
|
||||
SentryEvent event = new SentryEvent(e.getThrown());
|
||||
|
||||
Message sentryMessage = new Message();
|
||||
sentryMessage.setMessage(e.getMessage().getFormattedMessage());
|
||||
|
||||
event.setThrowable(e.getThrown());
|
||||
event.setLevel(getLevel(e.getLevel()));
|
||||
event.setLogger(e.getLoggerName());
|
||||
event.setTransaction(e.getLoggerName());
|
||||
event.setExtra("thread_name", e.getThreadName());
|
||||
|
||||
boolean hasContext = e.getContextData() != null;
|
||||
|
||||
if (hasContext && e.getContextData().containsKey("pufferfishsentry_playerid")) {
|
||||
User user = new User();
|
||||
user.setId(e.getContextData().getValue("pufferfishsentry_playerid"));
|
||||
user.setUsername(e.getContextData().getValue("pufferfishsentry_playername"));
|
||||
event.setUser(user);
|
||||
}
|
||||
|
||||
if (hasContext && e.getContextData().containsKey("pufferfishsentry_pluginname")) {
|
||||
event.setExtra("plugin.name", e.getContextData().getValue("pufferfishsentry_pluginname"));
|
||||
event.setExtra("plugin.version", e.getContextData().getValue("pufferfishsentry_pluginversion"));
|
||||
event.setTransaction(e.getContextData().getValue("pufferfishsentry_pluginname"));
|
||||
}
|
||||
|
||||
if (hasContext && e.getContextData().containsKey("pufferfishsentry_eventdata")) {
|
||||
Map<String, String> eventFields = GSON.fromJson((String) e.getContextData().getValue("pufferfishsentry_eventdata"), new TypeToken<Map<String, String>>() {
|
||||
}.getType());
|
||||
if (eventFields != null) {
|
||||
event.setExtra("event", eventFields);
|
||||
}
|
||||
}
|
||||
|
||||
Sentry.captureEvent(event);
|
||||
}
|
||||
|
||||
private void logBreadcrumb(LogEvent e) {
|
||||
Breadcrumb breadcrumb = new Breadcrumb();
|
||||
|
||||
breadcrumb.setLevel(getLevel(e.getLevel()));
|
||||
breadcrumb.setCategory(e.getLoggerName());
|
||||
breadcrumb.setType(e.getLoggerName());
|
||||
breadcrumb.setMessage(e.getMessage().getFormattedMessage());
|
||||
|
||||
Sentry.addBreadcrumb(breadcrumb);
|
||||
}
|
||||
|
||||
private SentryLevel getLevel(Level level) {
|
||||
return switch (level.getStandardLevel()) {
|
||||
case TRACE, DEBUG -> SentryLevel.DEBUG;
|
||||
case WARN -> SentryLevel.WARNING;
|
||||
case ERROR -> SentryLevel.ERROR;
|
||||
case FATAL -> SentryLevel.FATAL;
|
||||
default -> SentryLevel.INFO;
|
||||
};
|
||||
}
|
||||
|
||||
private static class SentryFilter extends AbstractFilter {
|
||||
|
||||
@Override
|
||||
public Result filter(Logger logger, org.apache.logging.log4j.Level level, Marker marker, String msg,
|
||||
Object... params) {
|
||||
return this.filter(logger.getName());
|
||||
}
|
||||
|
||||
@Override
|
||||
public Result filter(Logger logger, org.apache.logging.log4j.Level level, Marker marker, Object msg, Throwable t) {
|
||||
return this.filter(logger.getName());
|
||||
}
|
||||
|
||||
@Override
|
||||
public Result filter(LogEvent event) {
|
||||
return this.filter(event == null ? null : event.getLoggerName());
|
||||
}
|
||||
|
||||
private Result filter(String loggerName) {
|
||||
return loggerName != null && loggerName.startsWith("gg.castaway.pufferfish.sentry") ? Result.DENY
|
||||
: Result.NEUTRAL;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,42 @@
|
||||
package gg.pufferfish.pufferfish.sentry;
|
||||
|
||||
import io.sentry.Sentry;
|
||||
import org.apache.logging.log4j.Level;
|
||||
import org.apache.logging.log4j.LogManager;
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.bxteam.divinemc.DivineConfig;
|
||||
|
||||
public class SentryManager {
|
||||
private static final Logger LOGGER = LogManager.getLogger(SentryManager.class);
|
||||
private static boolean initialized = false;
|
||||
|
||||
private SentryManager() {
|
||||
throw new UnsupportedOperationException("This class cannot be instantiated.");
|
||||
}
|
||||
|
||||
public static synchronized void init(Level logLevel) {
|
||||
if (initialized) {
|
||||
return;
|
||||
}
|
||||
if (logLevel == null) {
|
||||
LOGGER.error("Invalid log level, defaulting to WARN.");
|
||||
logLevel = Level.WARN;
|
||||
}
|
||||
try {
|
||||
initialized = true;
|
||||
|
||||
Sentry.init(options -> {
|
||||
options.setDsn(DivineConfig.sentryDsn);
|
||||
options.setMaxBreadcrumbs(100);
|
||||
});
|
||||
|
||||
PufferfishSentryAppender appender = new PufferfishSentryAppender(logLevel);
|
||||
appender.start();
|
||||
((org.apache.logging.log4j.core.Logger) LogManager.getRootLogger()).addAppender(appender);
|
||||
LOGGER.info("Sentry logging started!");
|
||||
} catch (Exception e) {
|
||||
LOGGER.warn("Failed to initialize sentry!", e);
|
||||
initialized = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,6 +1,7 @@
|
||||
package org.bxteam.divinemc;
|
||||
|
||||
import com.google.common.base.Throwables;
|
||||
import org.apache.logging.log4j.Level;
|
||||
import org.apache.logging.log4j.LogManager;
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.bukkit.configuration.ConfigurationSection;
|
||||
@@ -323,6 +324,20 @@ public class DivineConfig {
|
||||
"Enables optimization that will offload much of the computational effort involved with spawning new mobs to a different thread.");
|
||||
}
|
||||
|
||||
public static String sentryDsn = "";
|
||||
public static String logLevel = "WARN";
|
||||
public static boolean onlyLogThrown = true;
|
||||
private static void sentrySettings() {
|
||||
sentryDsn = getString("settings.sentry.dsn", sentryDsn,
|
||||
"The DSN for Sentry, a service that provides real-time crash reporting that helps you monitor and fix crashes in real time. Leave blank to disable. Obtain link at https://sentry.io");
|
||||
logLevel = getString("settings.sentry.log-level", logLevel,
|
||||
"Logs with a level higher than or equal to this level will be recorded.");
|
||||
onlyLogThrown = getBoolean("settings.sentry.only-log-thrown", onlyLogThrown,
|
||||
"Only log Throwable exceptions to Sentry.");
|
||||
|
||||
if (sentryDsn != null && !sentryDsn.isBlank()) gg.pufferfish.pufferfish.sentry.SentryManager.init(Level.getLevel(logLevel));
|
||||
}
|
||||
|
||||
public static boolean disableDisconnectSpam = false;
|
||||
public static boolean connectionFlushQueueRewrite = false;
|
||||
public static boolean gracefulTeleportHandling = false;
|
||||
|
||||
Reference in New Issue
Block a user