diff options
-rw-r--r-- | sonar-plugin-api/src/main/java/org/sonar/api/utils/log/Logger.java | 37 | ||||
-rw-r--r-- | sonar-plugin-api/src/test/java/org/sonar/api/utils/log/LogTesterTest.java | 53 |
2 files changed, 89 insertions, 1 deletions
diff --git a/sonar-plugin-api/src/main/java/org/sonar/api/utils/log/Logger.java b/sonar-plugin-api/src/main/java/org/sonar/api/utils/log/Logger.java index 36665b990a6..3fd95e6ec55 100644 --- a/sonar-plugin-api/src/main/java/org/sonar/api/utils/log/Logger.java +++ b/sonar-plugin-api/src/main/java/org/sonar/api/utils/log/Logger.java @@ -19,6 +19,7 @@ */ package org.sonar.api.utils.log; +import java.util.function.Supplier; import javax.annotation.Nullable; /** @@ -60,6 +61,23 @@ public interface Logger { void trace(String msg); /** + * Logs a TRACE message, which is only to be constructed if the logging level + * is such that the message will actually be logged. + * <p> + * TRACE messages must + * be valuable for diagnosing production problems. They must not be used for development debugging. + * They can significantly slow down performances. The standard use-case is logging of + * SQL and Elasticsearch requests. + * @param msgSupplier A function, which when called, produces the desired log message + * @since 6.3 + */ + default void trace(Supplier<String> msgSupplier) { + if (isTraceEnabled()) { + trace(msgSupplier.get()); + } + } + + /** * Logs an TRACE parameterized message according to the specified format and argument. Example: * <code>trace("Value is {}", value)</code> * <p> @@ -99,12 +117,29 @@ public interface Logger { boolean isDebugEnabled(); /** - * Logs a DEBUG message. Debug messages must + * Logs a DEBUG message. + * <p> + * DEBUG messages must * be valuable for diagnosing production problems. They must not be used for development debugging. */ void debug(String msg); /** + * Logs a DEBUG message which is only to be constructed if the logging level + * is such that the message will actually be logged. + * <p> + * DEBUG messages must + * be valuable for diagnosing production problems. They must not be used for development debugging. + * @param msgSupplier A function, which when called, produces the desired log message + * @since 6.3 + */ + default void debug(Supplier<String> msgSupplier) { + if (isDebugEnabled()) { + debug(msgSupplier.get()); + } + } + + /** * Logs an DEBUG parameterized message according to the specified format and argument. Example: * <code>debug("Value is {}", value)</code> * <p> diff --git a/sonar-plugin-api/src/test/java/org/sonar/api/utils/log/LogTesterTest.java b/sonar-plugin-api/src/test/java/org/sonar/api/utils/log/LogTesterTest.java index 0521837bc3b..e94f797ac44 100644 --- a/sonar-plugin-api/src/test/java/org/sonar/api/utils/log/LogTesterTest.java +++ b/sonar-plugin-api/src/test/java/org/sonar/api/utils/log/LogTesterTest.java @@ -19,6 +19,7 @@ */ package org.sonar.api.utils.log; +import java.util.concurrent.atomic.AtomicBoolean; import org.junit.Test; import static org.assertj.core.api.Assertions.assertThat; @@ -63,4 +64,56 @@ public class LogTesterTest { underTest.after(); assertThat(LogInterceptors.get()).isSameAs(NullInterceptor.NULL_INSTANCE); } + + @Test + public void use_suppliers() throws Throwable { + // when LogTester is used, then info logs are enabled by default + underTest.before(); + AtomicBoolean touchedTrace = new AtomicBoolean(); + AtomicBoolean touchedDebug = new AtomicBoolean(); + Loggers.get("logger1").trace(() -> { + touchedTrace.set(true); + return "a trace information"; + }); + Loggers.get("logger1").debug(() -> { + touchedDebug.set(true); + return "a debug information"; + }); + + assertThat(underTest.logs()).isEmpty(); + assertThat(touchedTrace.get()).isFalse(); + assertThat(touchedDebug.get()).isFalse(); + + // change level to DEBUG + underTest.setLevel(LoggerLevel.DEBUG); + Loggers.get("logger1").trace(() -> { + touchedTrace.set(true); + return "a trace information"; + }); + Loggers.get("logger1").debug(() -> { + touchedDebug.set(true); + return "a debug information"; + }); + + assertThat(underTest.logs()).containsOnly("a debug information"); + assertThat(touchedTrace.get()).isFalse(); + assertThat(touchedDebug.get()).isTrue(); + touchedDebug.set(false); + underTest.logs().clear(); + + // change level to TRACE + underTest.setLevel(LoggerLevel.TRACE); + Loggers.get("logger1").trace(() -> { + touchedTrace.set(true); + return "a trace information"; + }); + Loggers.get("logger1").debug(() -> { + touchedDebug.set(true); + return "a debug information"; + }); + + assertThat(underTest.logs()).containsExactly("a trace information", "a debug information"); + assertThat(touchedTrace.get()).isTrue(); + assertThat(touchedDebug.get()).isTrue(); + } } |