TestMDCAdapter.java

  1. package com.github.valfirst.slf4jtest;

  2. import java.util.Collections;
  3. import java.util.HashMap;
  4. import java.util.Map;
  5. import java.util.Objects;
  6. import java.util.TreeMap;
  7. import org.slf4j.MDC;
  8. import org.slf4j.helpers.BasicMDCAdapter;

  9. public class TestMDCAdapter extends BasicMDCAdapter {

  10.     private final ThreadLocal<Map<String, String>> value;
  11.     private final boolean initialEnable;
  12.     private final boolean initialInherit;
  13.     private final boolean initialReturnNullCopyWhenMdcNotSet;
  14.     private final boolean initialAllowNullValues;
  15.     private volatile boolean enable;
  16.     private volatile boolean inherit;
  17.     private volatile boolean returnNullCopyWhenMdcNotSet;
  18.     private volatile boolean allowNullValues;

  19.     public TestMDCAdapter() {
  20.         this(OverridableProperties.createUnchecked("slf4jtest"));
  21.     }

  22.     TestMDCAdapter(OverridableProperties properties) {
  23.         enable = initialEnable = getBooleanProperty(properties, "mdc.enable", true);
  24.         inherit = initialInherit = getBooleanProperty(properties, "mdc.inherit", false);
  25.         returnNullCopyWhenMdcNotSet =
  26.                 initialReturnNullCopyWhenMdcNotSet =
  27.                         getBooleanProperty(properties, "mdc.return.null.copy.when.mdc.not.set", false);
  28.         allowNullValues =
  29.                 initialAllowNullValues = getBooleanProperty(properties, "mdc.allow.null.values", true);

  30.         value =
  31.                 new InheritableThreadLocal<Map<String, String>>() {
  32.                     @Override
  33.                     protected Map<String, String> childValue(Map<String, String> parentValue) {
  34.                         if (enable && inherit && parentValue != null) {
  35.                             return new HashMap<>(parentValue);
  36.                         } else {
  37.                             return null;
  38.                         }
  39.                     }
  40.                 };
  41.     }

  42.     static boolean getBooleanProperty(
  43.             OverridableProperties properties, String propertyKey, boolean defaultValue) {
  44.         return Boolean.parseBoolean(properties.getProperty(propertyKey, String.valueOf(defaultValue)));
  45.     }

  46.     @Override
  47.     public void put(final String key, final String val) {
  48.         if (!enable) {
  49.             return;
  50.         }
  51.         if (key == null) {
  52.             throw new IllegalArgumentException("key cannot be null");
  53.         }
  54.         if (val == null && !allowNullValues) {
  55.             throw new IllegalArgumentException("val cannot be null");
  56.         }
  57.         Map<String, String> map = value.get();
  58.         if (map == null) {
  59.             map = new HashMap<>();
  60.             value.set(map);
  61.         }
  62.         map.put(key, val);
  63.     }

  64.     @Override
  65.     public String get(final String key) {
  66.         if (!enable) {
  67.             return null;
  68.         }
  69.         if (key == null) {
  70.             throw new IllegalArgumentException("key cannot be null");
  71.         }
  72.         Map<String, String> map = value.get();
  73.         if (map != null) {
  74.             return map.get(key);
  75.         } else {
  76.             return null;
  77.         }
  78.     }

  79.     @Override
  80.     public void remove(final String key) {
  81.         if (!enable) {
  82.             return;
  83.         }
  84.         if (key == null) {
  85.             throw new IllegalArgumentException("key cannot be null");
  86.         }
  87.         Map<String, String> map = value.get();
  88.         if (map != null) {
  89.             map.remove(key);
  90.         }
  91.     }

  92.     @Override
  93.     public void clear() {
  94.         if (!enable) {
  95.             return;
  96.         }
  97.         Map<String, String> map = value.get();
  98.         if (map == null) {
  99.             return;
  100.         }
  101.         map.clear();
  102.         value.remove();
  103.     }

  104.     /**
  105.      * Return a copy of the current thread's context map. {@code null} is returned if
  106.      *
  107.      * <ul>
  108.      *   <li>The MDC functionality is disabled, c.f. <code>setEnable</code>, or
  109.      *   <li>"return null when empty" is enabled, c.f. <code>setReturnNullCopyWhenMdcNotSet</code>,
  110.      *       and <code>put</code> has not been called.
  111.      * </ul>
  112.      *
  113.      * @return A copy of the current thread's context map.
  114.      */
  115.     @Override
  116.     public Map<String, String> getCopyOfContextMap() {
  117.         if (!enable) {
  118.             return null;
  119.         }
  120.         Map<String, String> map = value.get();
  121.         if (map == null) {
  122.             if (returnNullCopyWhenMdcNotSet) {
  123.                 return null;
  124.             } else {
  125.                 return new TreeMap<>();
  126.             }
  127.         } else {
  128.             return new TreeMap<>(map);
  129.         }
  130.     }

  131.     // Internal access
  132.     Map<String, String> getContextMap() {
  133.         Map<String, String> map = value.get();
  134.         return map == null ? Collections.emptySortedMap() : map;
  135.     }

  136.     @Override
  137.     public void setContextMap(final Map<String, String> contextMap) {
  138.         if (!enable) {
  139.             return;
  140.         }
  141.         clear();
  142.         if (contextMap == null) {
  143.             return;
  144.         }
  145.         if (contextMap.keySet().stream().anyMatch(Objects::isNull)) {
  146.             throw new IllegalArgumentException("key cannot be null");
  147.         }
  148.         if (!allowNullValues && contextMap.containsValue(null)) {
  149.             throw new IllegalArgumentException("val cannot be null");
  150.         }
  151.         value.set(new HashMap<>(contextMap));
  152.     }

  153.     /**
  154.      * Enable the MDC functionality for all threads.
  155.      *
  156.      * @param enable Whether to enable the MDC functionality. The default value is {@code true}.
  157.      */
  158.     public void setEnable(boolean enable) {
  159.         this.enable = enable;
  160.     }

  161.     /**
  162.      * Define whether child threads inherit a copy of the MDC from its parent thread. Note that the
  163.      * copy is taken when the {@link Thread} is constructed. This affects all threads.
  164.      *
  165.      * @param inherit Whether to enable inheritance. The default value is {@code false}.
  166.      */
  167.     public void setInherit(boolean inherit) {
  168.         this.inherit = inherit;
  169.     }

  170.     /**
  171.      * Define whether null values are allowed in the MDC. This affects all threads.
  172.      *
  173.      * @param allowNullValues Whether to allow nulls. The default value is {@code true}.
  174.      */
  175.     public void setAllowNullValues(boolean allowNullValues) {
  176.         this.allowNullValues = allowNullValues;
  177.     }

  178.     /**
  179.      * Define whether {@link #getCopyOfContextMap} returns {@code null} when no values have been set.
  180.      * This affects all threads.
  181.      *
  182.      * @param returnNullCopyWhenMdcNotSet Whether to return null. The default value is {@code false}.
  183.      *     If {@code false}, an empty map is returned instead.
  184.      */
  185.     public void setReturnNullCopyWhenMdcNotSet(boolean returnNullCopyWhenMdcNotSet) {
  186.         this.returnNullCopyWhenMdcNotSet = returnNullCopyWhenMdcNotSet;
  187.     }

  188.     /**
  189.      * Whether the MDC functionality is enabled.
  190.      *
  191.      * @return Whether the MDC functionality is enabled.
  192.      */
  193.     public boolean getEnable() {
  194.         return enable;
  195.     }

  196.     /**
  197.      * Whether child threads inherit a copy of the MDC from its parent thread.
  198.      *
  199.      * @return Whether inheritance is enabled.
  200.      */
  201.     public boolean getInherit() {
  202.         return inherit;
  203.     }

  204.     /**
  205.      * Whether null values are allowed in the MDC.
  206.      *
  207.      * @return Whether nulls are allowed.
  208.      */
  209.     public boolean getAllowNullValues() {
  210.         return allowNullValues;
  211.     }

  212.     /**
  213.      * Whether {@link #getCopyOfContextMap} returns {@code null} when no values have been set.
  214.      *
  215.      * @return Whether to return null.
  216.      */
  217.     public boolean getReturnNullCopyWhenMdcNotSet() {
  218.         return returnNullCopyWhenMdcNotSet;
  219.     }

  220.     /**
  221.      * Reset the options to values defined by the static configuration. This undoes to changes made
  222.      * using {@link #setEnable}, {@link #setInherit}, {@link #setAllowNullValues}, and {@link
  223.      * #setReturnNullCopyWhenMdcNotSet}.
  224.      */
  225.     public void restoreOptions() {
  226.         enable = initialEnable;
  227.         inherit = initialInherit;
  228.         returnNullCopyWhenMdcNotSet = initialReturnNullCopyWhenMdcNotSet;
  229.         allowNullValues = initialAllowNullValues;
  230.     }

  231.     /** Access the current MDC adapter. Used to call the option setting methods. */
  232.     @SuppressWarnings("unchecked")
  233.     public static TestMDCAdapter getInstance() {
  234.         return (TestMDCAdapter) MDC.getMDCAdapter();
  235.     }
  236. }