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

io.opentelemetry.instrumentation.testing.util.ContextStorageCloser Maven / Gradle / Ivy

There is a newer version: 2.10.0-alpha
Show newest version
/*
 * Copyright The OpenTelemetry Authors
 * SPDX-License-Identifier: Apache-2.0
 */

package io.opentelemetry.instrumentation.testing.util;

import static org.awaitility.Awaitility.await;

import io.opentelemetry.context.ContextStorage;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.time.Duration;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * Helper class for closing context storage with retry in case there are strict context check
 * failures.
 */
public final class ContextStorageCloser {

  private ContextStorageCloser() {}

  public static void close(ContextStorage storage) throws Exception {
    if (storage instanceof AutoCloseable) {
      cleanup((AutoCloseable) storage);
    }
  }

  private static void cleanup(AutoCloseable storage) throws Exception {
    ContextRestorer restorer = ContextRestorer.create((ContextStorage) storage);
    if (restorer == null) {
      storage.close();
      return;
    }

    // If close is called before all request processing threads have completed we might get false
    // positive notifications for leaked scopes when strict context checks are enabled. Here we
    // retry close when scope leak was reported.
    await()
        .ignoreException(AssertionError.class)
        .atMost(Duration.ofSeconds(10))
        .pollInterval(Duration.ofSeconds(1))
        .pollDelay(Duration.ZERO)
        .until(() -> restorer.runWithRestore(storage));
  }

  // Helper class that allows for retrying ContextStorage close operation by restoring
  // ContextStorage to the state where it was before close was called.
  private abstract static class ContextRestorer {
    abstract void restore();

    @SuppressWarnings("SystemOut")
    boolean runWithRestore(AutoCloseable target) {
      try {
        target.close();
        return true;
      } catch (Throwable throwable) {
        restore();
        if (throwable instanceof AssertionError) {
          System.err.println();
          for (Map.Entry threadEntry :
              Thread.getAllStackTraces().entrySet()) {
            System.err.println("Thread " + threadEntry.getKey());
            for (StackTraceElement stackTraceElement : threadEntry.getValue()) {
              System.err.println("\t" + stackTraceElement);
            }
            System.err.println();
          }
          throw (AssertionError) throwable;
        }
        throw new IllegalStateException(throwable);
      }
    }

    static ContextRestorer create(ContextStorage storage)
        throws NoSuchFieldException, IllegalAccessException {
      Object strictContextStorage = getStrictContextStorage(storage);
      if (strictContextStorage == null) {
        return null;
      }

      Object pendingScopes = getStrictContextStoragePendingScopes(strictContextStorage);
      Field mapField = pendingScopes.getClass().getDeclaredField("map");
      mapField.setAccessible(true);
      @SuppressWarnings("unchecked")
      ConcurrentHashMap map =
          (ConcurrentHashMap) mapField.get(pendingScopes);
      Map copy = new HashMap<>(map);

      return new ContextRestorer() {
        @Override
        void restore() {
          map.putAll(copy);
        }
      };
    }

    private static Object getStrictContextStoragePendingScopes(Object strictContextStorage)
        throws NoSuchFieldException, IllegalAccessException {
      Field field = strictContextStorage.getClass().getDeclaredField("pendingScopes");
      field.setAccessible(true);
      return field.get(strictContextStorage);
    }

    private static Object getStrictContextStorage(ContextStorage storage)
        throws NoSuchFieldException, IllegalAccessException {
      // in library instrumentation tests we already have access to StrictContextStorage
      if (storage.getClass().getName().contains("StrictContextStorage")) {
        return storage;
      }
      // in javaagent tests the storage we get is wrapped by opentelemetry api bridge, find the
      // actual storage
      Object contextStorage = getAgentContextStorage();
      if (contextStorage == null) {
        return null;
      }
      contextStorage = unwrapStrictContextStressor(contextStorage);
      Class contextStorageClass = contextStorage.getClass();
      if (contextStorageClass.getName().contains("StrictContextStorage")) {
        return contextStorage;
      }
      return null;
    }

    private static Object getAgentContextStorage() {
      try {
        Class contextStorageClass =
            Class.forName(
                "io.opentelemetry.javaagent.shaded.io.opentelemetry.context.ContextStorage");
        Method method = contextStorageClass.getDeclaredMethod("get");
        return method.invoke(null);
      } catch (Exception exception) {
        return null;
      }
    }

    private static Object unwrapStrictContextStressor(Object contextStorage)
        throws NoSuchFieldException, IllegalAccessException {
      Class contextStorageClass = contextStorage.getClass();
      if (contextStorageClass.getName().contains("StrictContextStressor")) {
        Field field = contextStorageClass.getDeclaredField("contextStorage");
        field.setAccessible(true);
        return field.get(contextStorage);
      }
      return contextStorage;
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy