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

kanela.agent.Kanela Maven / Gradle / Ivy

/*
 * =========================================================================================
 * Copyright © 2013-2021 the kamon project 
 *
 * 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 kanela.agent;

import io.vavr.collection.List;
import kanela.agent.api.instrumentation.listener.InstrumentationRegistryListener;
import kanela.agent.builder.KanelaFileTransformer;
import kanela.agent.circuitbreaker.SystemThroughputCircuitBreaker;
import kanela.agent.reinstrument.Reinstrumenter;
import kanela.agent.util.BootstrapInjector;
import kanela.agent.util.banner.KanelaBanner;
import kanela.agent.util.classloader.InstrumentationClassPath;
import kanela.agent.util.classloader.PreInitializeClasses;
import kanela.agent.util.conf.KanelaConfiguration;
import kanela.agent.util.jvm.OldGarbageCollectorListener;
import kanela.agent.util.log.Logger;
import lombok.val;

import java.lang.instrument.Instrumentation;

import static kanela.agent.util.Execution.runWithTimeSpent;

final public class Kanela {

  private static String loadedPropertyName = "kanela.loaded";
  private static volatile Instrumentation instrumentation;
  private static volatile List installedTransformers = List.empty();

  /**
   * Entry point when the Kanela agent is added with the -javaagent command line option.
   */
  public static void premain(final String args, final Instrumentation instrumentation) {
      Kanela.start(args, instrumentation, false);
  }

  /**
   * Entry point when the Kanela agent is attached at runtime.
   */
  public static void agentmain(final String args, final Instrumentation instrumentation) {
      Kanela.start(args, instrumentation, true);
  }

  /**
   * Scans the instrumentation class path for modules and installs them on the JVM.
   */
  public static void start(final String arguments, final Instrumentation instrumentation, boolean isRuntimeAttach) {

      // This ensures that we will not load Kanela more than once on the same JVM.
      if(Kanela.instrumentation == null) {
          // We keep the reference in case we will need to reload the agent.
          Kanela.instrumentation = instrumentation;

          runWithTimeSpent(() -> {
              InstrumentationClassPath.build().use(instrumentationClassLoader -> {
                  PreInitializeClasses.preInitializeClasses(instrumentationClassLoader);

                  BootstrapInjector.injectJar(instrumentation, "bootstrap");

                  val configuration = KanelaConfiguration.from(instrumentationClassLoader);
                  Logger.configureLogger(configuration);

                  if (isRuntimeAttach) configuration.runtimeAttach();
                  KanelaBanner.show(configuration);

                  installedTransformers = InstrumentationLoader.load(instrumentation, instrumentationClassLoader, configuration);
                  Reinstrumenter.attach(instrumentation, configuration, installedTransformers);
                  OldGarbageCollectorListener.attach(configuration.getOldGarbageCollectorConfig());
                  SystemThroughputCircuitBreaker.attach(configuration.getCircuitBreakerConfig());
                  updateLoadedSystemProperty();
              });
          });
      }
  }

  /**
   * Removes all instrumentation modules already applied by Kanela and re-scans the instrumentation class path for
   * modules to apply them on the current JVM.
   */
  public static void reload() {
    Kanela.reload(false);
  }

  /**
   * Removes all instrumentation modules already applied by Kanela and re-scans the instrumentation class path for
   * modules to apply them on the current JVM.
   */
  public static void reload(boolean clearRegistry) {

    // We will only proceed to reload if Kanela was properly started already.
    if(Kanela.instrumentation != null) {
        if(clearRegistry)
          InstrumentationRegistryListener.instance().clear();

        InstrumentationClassPath.build().use(instrumentationClassLoader -> {
            installedTransformers.forEach(transformer -> instrumentation.removeTransformer(transformer.getClassFileTransformer()));
            installedTransformers = List.empty();

            val configuration = KanelaConfiguration.from(instrumentationClassLoader);
            Logger.configureLogger(configuration);
            installedTransformers = InstrumentationLoader.load(instrumentation, instrumentationClassLoader, configuration);
        });
    }
  }

  /**
   * Sets a System property indicating that Kanela has been loaded. That property is meant to be used by external
   * libraries that might want to check whether Kanela was loaded or not.
   */
  private static void updateLoadedSystemProperty() {
    System.setProperty(loadedPropertyName, "true");
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy