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

jvmMain.com.giancarlobuenaflor.kflogger.context.ScopedLoggingContexts Maven / Gradle / Ivy

/*
 * Copyright (C) 2021 The Flogger Authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.giancarlobuenaflor.kflogger.context;

import static java.util.concurrent.TimeUnit.MINUTES;

import com.giancarlobuenaflor.kflogger.FluentLogger;
import com.giancarlobuenaflor.kflogger.MetadataKey;
import com.giancarlobuenaflor.kflogger.StackSize;
import com.google.errorprone.annotations.CanIgnoreReturnValue;

/**
 * Static methods equivalent to the instance methods on {@link ScopedLoggingContext} but which
 * always operate on the current {@link ScopedLoggingContext} that would be returned by {@link
 * ScopedLoggingContext#getInstance}.
 */
public final class ScopedLoggingContexts {
  private static final FluentLogger logger = FluentLogger.forEnclosingClass();

  private static boolean warnOnFailure(boolean wasSuccessful) {
    if (!wasSuccessful && !ScopedLoggingContext.getInstance().isNoOp()) {
      logger
          .atWarning()
          .atMostEvery(5, MINUTES)
          .withStackTrace(StackSize.SMALL)
          .log(
              "***** An attempt to add metadata to the current logging context failed. *****\n"
                  + "Calls to static methods in 'ScopedLoggingContexts' may fail when there is no"
                  + " existing context available.\n"
                  + "To ensure metadata is available to log statements, create a new context via"
                  + " 'ScopedLoggingContexts.newContext()' and add metadata to it explicitly.\n");
    }
    return wasSuccessful;
  }

  private ScopedLoggingContexts() {}

  /**
   * Creates a new {@link ScopedLoggingContext.Builder} to which additional logging metadata can be
   * attached before being installed or used to wrap some existing code.
   *
   * 
{@code
   * Foo result = ScopedLoggingContexts.newContext()
   *     .withTags(Tags.of("my_tag", someValue))
   *     .call(MyClass::doFoo);
   * }
*/ public static ScopedLoggingContext.Builder newContext() { return ScopedLoggingContext.getInstance().newContext(); } /** * Adds tags by modifying the current context (if one exists). * *

Warning: It is always better to create a new context via {@link #newContext()} rather than * attempting to modify an existing context. In order of preference you should: * *

    *
  1. Call or wrap a new context with metadata added to it. *
  2. {@link ScopedLoggingContext.Builder#install install()} a new context and close it when * you it exits (e.g. if you are using callbacks to listen to state changes in a task). * However it is vital that the returned {@link ScopedLoggingContext.LogContextCloseable} is * always closed. *
  3. Call this method and check that it succeeded (e.g. logging a warning if it fails). *
* *

The given tags are merged with those of the modifed context but existing tags will not be * overwritten or removed. This is deliberate since two pieces of code may not know about each * other and could accidentally use the same tag name; in that situation it's important that both * tag values are preserved. * *

Furthermore, the types of data allowed for tag values are strictly controlled. This is also * very deliberate since these tags must be efficiently added to every log statement and so it's * important that they resulting string representation is reliably cacheable and can be calculated * without invoking arbitrary code (e.g. the {@code toString()} method of some unknown user type). * * @return false if there is no current context, or scoped contexts are not supported. */ @CanIgnoreReturnValue public static boolean addTags(Tags tags) { return warnOnFailure(ScopedLoggingContext.getInstance().addTags(tags)); } /** * Adds a single metadata key/value pair to the current context. * *

Warning: It is always better to create a new context via {@link #newContext()} rather than * attempting to modify an existing context. In order of preference you should: * *

    *
  1. Call or wrap a new context with metadata added to it. *
  2. {@link ScopedLoggingContext.Builder#install install()} a new context and close it when * you it exits (e.g. if you are using callbacks to listen to state changes in a task). * However it is vital that the returned {@link ScopedLoggingContext.LogContextCloseable} is * always closed. *
  3. Call this method and check that it succeeded (e.g. logging a warning if it fails). *
* *

Unlike {@link Tags}, which have a well defined value ordering, independent of the order in * which values were added, context metadata preserves the order of addition. As such, it is not * advised to add values for the same metadata key from multiple threads, since that may create * non-deterministic ordering. It is recommended (where possible) to add metadata when building a * new context, rather than adding it to context visible to multiple threads. */ @CanIgnoreReturnValue public static boolean addMetadata(MetadataKey key, T value) { return warnOnFailure(ScopedLoggingContext.getInstance().addMetadata(key, value)); } /** * Applies the given log level map to the current context. * *

Warning: It is always better to create a new context via {@link #newContext()} rather than * attempting to modify an existing context. In order of preference you should: * *

    *
  1. Call or wrap a new context with metadata added to it. *
  2. {@link ScopedLoggingContext.Builder#install install()} a new context and close it when * you it exits (e.g. if you are using callbacks to listen to state changes in a task). * However it is vital that the returned {@link ScopedLoggingContext.LogContextCloseable} is * always closed. *
  3. Call this method and check that it succeeded (e.g. logging a warning if it fails). *
* * Log level settings are merged with any existing setting from the current (or parent) contexts * such that logging will be enabled for a log statement if: * *
    *
  • It was enabled by the given map. *
  • It was already enabled by the current context. *
* *

The effects of this call will be undone only when the current context terminates. * * @return false if there is no current context, or scoped contexts are not supported. */ @CanIgnoreReturnValue public static boolean applyLogLevelMap(LogLevelMap logLevelMap) { return warnOnFailure(ScopedLoggingContext.getInstance().applyLogLevelMap(logLevelMap)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy