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

io.codemodder.ParameterModule Maven / Gradle / Ivy

package io.codemodder;

import com.google.inject.AbstractModule;
import java.nio.file.Path;
import java.util.List;
import java.util.Objects;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.stream.Collectors;

/**
 * Responsible for binding {@link Parameter}s to codemods. Also performs validation to ensure
 * provided arguments are valid.
 */
final class ParameterModule extends AbstractModule {

  private final List parameterArguments;
  private final List injectableParameters;

  ParameterModule(
      final List codemodParameters,
      final List injectableParameters) {
    this.parameterArguments = Objects.requireNonNull(codemodParameters);
    this.injectableParameters = Objects.requireNonNull(injectableParameters);
  }

  @Override
  protected void configure() {

    // this code is a little confusing to read because the collision between the
    // java.lang.reflect.Parameter and io.codemodder.Parameter
    List codemodParameters =
        this.injectableParameters.stream()
            .filter(param -> param.isAnnotationPresent(CodemodParameter.class))
            .filter(param -> param.getType().equals(Parameter.class))
            .collect(Collectors.toUnmodifiableList());

    for (java.lang.reflect.Parameter param : codemodParameters) {
      CodemodParameter codemodParameter = param.getAnnotation(CodemodParameter.class);
      Codemod codemod =
          param.getDeclaringExecutable().getDeclaringClass().getAnnotation(Codemod.class);
      ParameterArgument parameterArgument =
          parameterArguments.stream()
              .filter(p -> p.codemodId().equals(codemod.id()))
              .findFirst()
              .orElse(null);

      Parameter parameter;
      if (parameterArgument != null) {
        parameter = new GivenParameter(codemodParameter, parameterArgument);
      } else {
        parameter = new NoValueProvidedParameter(codemodParameter);
      }
      String defaultValue = parameter.getDefaultValue();
      Pattern p = Pattern.compile(codemodParameter.validationPattern());
      Matcher matcher = p.matcher(defaultValue);
      if (!matcher.matches()) {
        throw new IllegalArgumentException(
            "invalid parameter default value: "
                + defaultValue
                + " for parameter: "
                + codemodParameter.name()
                + " in codemod: "
                + codemod.id()
                + " with validation pattern: "
                + codemodParameter.validationPattern());
      }

      bind(Parameter.class).annotatedWith(codemodParameter).toInstance(parameter);
    }
  }

  /**
   * This is the {@link Parameter} implementation for when no value is provided for a given
   * parameter.
   */
  private record NoValueProvidedParameter(CodemodParameter declaration) implements Parameter {

    private NoValueProvidedParameter {
      Objects.requireNonNull(declaration);
    }

    @Override
    public String getQuestion() {
      return declaration.question();
    }

    @Override
    public String getType() {
      return declaration.type().name().toLowerCase();
    }

    @Override
    public String getName() {
      return declaration.name();
    }

    @Override
    public String getValue(final Path path, final int currentLine) {
      return declaration.defaultValue();
    }

    @Override
    public String getLabel() {
      return declaration.label();
    }

    @Override
    public String getDefaultValue() {
      return declaration.defaultValue();
    }
  }

  /**
   * This is the {@link Parameter} implementation for when a value is provided for a given
   * parameter.
   */
  private record GivenParameter(CodemodParameter declaration, ParameterArgument parameterArgument)
      implements Parameter {

    private GivenParameter {
      Objects.requireNonNull(declaration);
      Objects.requireNonNull(parameterArgument);
    }

    @Override
    public String getQuestion() {
      return declaration.question();
    }

    @Override
    public String getType() {
      return declaration.type().name().toLowerCase();
    }

    @Override
    public String getName() {
      return declaration.name();
    }

    @Override
    public String getValue(final Path path, final int currentLine) {
      String file = parameterArgument.file();
      Integer line = parameterArgument.line();
      if (file == null && line == null) {
        // this is a default value for the whole run, so we return that
        return parameterArgument.value();
      }
      if (file == null || !file.equals(path.toString())) {
        // the parameter doesn't cover this file, so we return the default value
        return declaration.defaultValue();
      } else {
        // only return the value if the line isn't specified by the arg or it matches
        if (line == null || line == currentLine) {
          return parameterArgument.value();
        }
      }
      // if our parameter doesn't match, we return the default value
      return declaration.defaultValue();
    }

    @Override
    public String getLabel() {
      return declaration.label();
    }

    @Override
    public String getDefaultValue() {
      if (parameterArgument.file() == null && parameterArgument.line() == null) {
        return parameterArgument.value();
      }
      return declaration.defaultValue();
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy