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

io.codemodder.codemods.HardenProcessCreationCodemod Maven / Gradle / Ivy

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

import com.contrastsecurity.sarif.Result;
import com.github.javaparser.ast.CompilationUnit;
import com.github.javaparser.ast.Node;
import com.github.javaparser.ast.NodeList;
import com.github.javaparser.ast.body.FieldDeclaration;
import com.github.javaparser.ast.body.VariableDeclarator;
import com.github.javaparser.ast.expr.Expression;
import com.github.javaparser.ast.expr.MethodCallExpr;
import com.github.javaparser.ast.expr.NameExpr;
import com.github.javaparser.ast.expr.SimpleName;
import io.codemodder.*;
import io.codemodder.ast.ASTTransforms;
import io.codemodder.javaparser.ChangesResult;
import io.codemodder.providers.sarif.semgrep.SemgrepScan;
import io.github.pixee.security.SystemCommand;
import java.util.List;
import javax.inject.Inject;

/** Harden new process creation. */
@Codemod(
    id = "pixee:java/harden-process-creation",
    importance = Importance.HIGH,
    reviewGuidance = ReviewGuidance.MERGE_WITHOUT_REVIEW)
public final class HardenProcessCreationCodemod
    extends SarifPluginJavaParserChanger {

  @Inject
  public HardenProcessCreationCodemod(
      @SemgrepScan(ruleId = "harden-process-creation") final RuleSarif sarif) {
    super(sarif, MethodCallExpr.class);
  }

  @Override
  public ChangesResult onResultFound(
      final CodemodInvocationContext context,
      final CompilationUnit cu,
      final MethodCallExpr methodCallExpr,
      final Result result) {

    final List fieldDeclarations = cu.findAll(FieldDeclaration.class);
    final List constantFieldDeclarations =
        getConstantFieldDeclarations(fieldDeclarations);
    final List constantNames = extractVariableNames(constantFieldDeclarations);

    final List allArgumentsExpressions =
        ArgumentExpressionExtractor.extractExpressions(methodCallExpr);

    final List onlyVariableExpressions =
        allArgumentsExpressions.stream()
            .filter(Expression::isNameExpr)
            .filter(
                expression -> {
                  final NameExpr nameExpr = (NameExpr) expression;
                  return !constantNames.contains(nameExpr.getName());
                })
            .toList();

    final boolean containsOnlyConstantsAndHardcodedValues = onlyVariableExpressions.isEmpty();

    if (containsOnlyConstantsAndHardcodedValues) {
      return ChangesResult.noChanges;
    }

    Node parent = methodCallExpr.getParentNode().get();
    Expression scope = methodCallExpr.getScope().get();
    ASTTransforms.addImportIfMissing(cu, SystemCommand.class);
    NameExpr callbackClass = new NameExpr(SystemCommand.class.getSimpleName());
    MethodCallExpr safeExpression = new MethodCallExpr(callbackClass, "runCommand");
    NodeList nodeList = new NodeList<>();
    nodeList.add(scope);
    nodeList.addAll(methodCallExpr.getArguments());
    safeExpression.setArguments(nodeList);

    parent.replace(methodCallExpr, safeExpression);
    return ChangesResult.changesAppliedWith(dependencies);
  }

  private static final List dependencies =
      List.of(DependencyGAV.JAVA_SECURITY_TOOLKIT);

  private List extractVariableNames(List fieldDeclarations) {
    return fieldDeclarations.stream()
        .flatMap(fieldDeclaration -> fieldDeclaration.getVariables().stream())
        .map(VariableDeclarator::getName)
        .toList();
  }

  private List getConstantFieldDeclarations(
      final List fieldDeclarations) {

    return fieldDeclarations.stream()
        .filter(FieldDeclaration::isFinal) // Check if the fieldDeclaration has the 'final' modifier
        .filter(
            fieldDeclaration ->
                fieldDeclaration.getVariables().stream()
                    .allMatch(
                        variable ->
                            variable
                                .getInitializer()
                                .isPresent())) // Check if all variables have initializers
        .toList();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy