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

io.codemodder.CodemodLoader Maven / Gradle / Ivy

There is a newer version: 0.98.6
Show newest version
package io.codemodder;

import com.google.inject.AbstractModule;
import com.google.inject.Guice;
import com.google.inject.Injector;
import io.github.classgraph.ClassGraph;
import io.github.classgraph.ClassInfoList;
import io.github.classgraph.ScanResult;
import java.lang.reflect.Executable;
import java.lang.reflect.Parameter;
import java.nio.file.Path;
import java.util.*;
import java.util.regex.Pattern;
import javax.inject.Inject;
import org.jetbrains.annotations.VisibleForTesting;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/** This type is responsible for loading codemods and the surrounding subsystem. */
public final class CodemodLoader {

  private final List codemods;

  public CodemodLoader(
      final List> unorderedCodemodTypes,
      final CodemodRegulator codemodRegulator,
      final Path repositoryDir,
      final List pathIncludes,
      final List pathExcludes,
      final List includedFiles,
      final Map> ruleSarifByTool,
      final List codemodParameters,
      final List sonarIssuesJsonFiles,
      final List sonarHotspotsJsonFiles,
      final Path defectDojoFindingsJsonFile,
      final Path contrastVulnerabilitiesXmlFilePath) {

    log.trace("Loading providers");

    // get all the providers ready for dependency injection & codemod instantiation
    final List providers =
        ServiceLoader.load(CodemodProvider.class).stream()
            .map(ServiceLoader.Provider::get)
            .toList();
    final Set allModules = new HashSet<>();

    // sort the codemods according to their priority
    List> orderedCodemodTypes = new ArrayList<>(unorderedCodemodTypes);

    // if there's an order from --codemod-includes, honor that
    Optional> desiredOrder = codemodRegulator.desiredCodemodIdOrder();
    if (desiredOrder.isPresent()) {
      orderedCodemodTypes.sort(
          (c1, c2) -> {
            String id1 = c1.getAnnotation(Codemod.class).id();
            String id2 = c2.getAnnotation(Codemod.class).id();
            int index1 = desiredOrder.get().indexOf(id1);
            int index2 = desiredOrder.get().indexOf(id2);
            return Integer.compare(index1, index2);
          });
    } else {
      // sort according to the codemod execution priority of each codemod type
      orderedCodemodTypes.sort(
          (c1, c2) -> {
            CodemodExecutionPriority p1 = c1.getAnnotation(Codemod.class).executionPriority();
            CodemodExecutionPriority p2 = c2.getAnnotation(Codemod.class).executionPriority();
            return CodemodExecutionPriority.priorityOrderComparator.compare(p1, p2);
          });
    }

    // get all the injectable parameters
    Set packagesScanned = new HashSet<>();
    List injectableParameters = new ArrayList<>();
    for (Class codemodType : orderedCodemodTypes) {
      String packageName = codemodType.getPackageName();
      if (!packagesScanned.contains(packageName)) {
        packagesScanned.add(packageName);
        final ClassInfoList classesWithMethodAnnotation;
        try (ScanResult scan =
            new ClassGraph()
                .enableAllInfo()
                .acceptPackagesNonRecursive(packageName)
                .removeTemporaryFilesAfterScan()
                .scan()) {
          classesWithMethodAnnotation = scan.getClassesWithMethodAnnotation(Inject.class);
          List> injectableClasses = classesWithMethodAnnotation.loadClasses();
          List targetedParams =
              injectableClasses.stream()
                  .map(Class::getDeclaredConstructors)
                  .flatMap(Arrays::stream)
                  .filter(constructor -> constructor.isAnnotationPresent(Inject.class))
                  .map(Executable::getParameters)
                  .flatMap(Arrays::stream)
                  .filter(param -> param.getAnnotations().length > 0)
                  .toList();
          injectableParameters.addAll(targetedParams);
        }
      }
    }

    // add default modules
    allModules.add(new CodeDirectoryModule(repositoryDir));
    allModules.add(new XPathStreamProcessorModule());
    allModules.add(new ParameterModule(codemodParameters, injectableParameters));

    // add all provider modules
    for (final CodemodProvider provider : providers) {
      final List wantsSarif = provider.wantsSarifToolNames();
      final var allWantedSarifs =
          wantsSarif.stream()
              .flatMap(toolName -> ruleSarifByTool.getOrDefault(toolName, List.of()).stream())
              .toList();
      log.trace("Loading modules from provider: {}", provider.getClass().getSimpleName());
      final Set modules =
          provider.getModules(
              repositoryDir,
              includedFiles,
              pathIncludes,
              pathExcludes,
              orderedCodemodTypes,
              allWantedSarifs,
              sonarIssuesJsonFiles,
              sonarHotspotsJsonFiles,
              defectDojoFindingsJsonFile,
              contrastVulnerabilitiesXmlFilePath);
      allModules.addAll(modules);
    }

    // record which changers are associated with which codemod ids
    final List codemods = new ArrayList<>();

    // validate and instantiate the codemods
    log.trace("Instantiating codemods");
    final Injector injector = Guice.createInjector(allModules);
    log.trace("Codemods instantiated");
    final Set codemodIds = new HashSet<>();
    for (final Class type : orderedCodemodTypes) {
      final Codemod codemodAnnotation = type.getAnnotation(Codemod.class);
      validateRequiredFields(codemodAnnotation);
      final CodeChanger codeChanger = injector.getInstance(type);
      final String codemodId = codemodAnnotation.id();
      if (codemodIds.contains(codemodId)) {
        throw new UnsupportedOperationException("multiple codemods under id: " + codemodId);
      }
      codemodIds.add(codemodId);
      if (codemodRegulator.isAllowed(codemodId)) {
        codemods.add(new CodemodIdPair(codemodId, codeChanger));
      }
    }
    this.codemods = Collections.unmodifiableList(codemods);
  }

  public List getCodemods() {
    return codemods;
  }

  private static void validateRequiredFields(final Codemod codemodAnnotation) {
    final String id = codemodAnnotation.id();
    if (!isValidCodemodId(id)) {
      throw new IllegalArgumentException("must have valid codemod id");
    }

    final ReviewGuidance reviewGuidance = codemodAnnotation.reviewGuidance();
    if (reviewGuidance == null) {
      throw new IllegalArgumentException("must have review guidance");
    }
  }

  @VisibleForTesting
  static boolean isValidCodemodId(final String codemodId) {
    return codemodIdPattern.matcher(codemodId).matches();
  }

  private static final Logger log = LoggerFactory.getLogger(CodemodLoader.class);

  private static final Pattern codemodIdPattern =
      Pattern.compile("^([A-Za-z0-9]+):(([A-Za-z0-9]+)/)+([A-Za-z0-9\\-\\.]+)$");
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy