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

org.sonarsource.scala.checks.PatternMatchHelper Maven / Gradle / Ivy

/*
 * SonarSource Scala
 * Copyright (C) 2018-2024 SonarSource SA
 * mailto:info AT sonarsource DOT com
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the Sonar Source-Available License Version 1, as published by SonarSource SA.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the Sonar Source-Available License for more details.
 *
 * You should have received a copy of the Sonar Source-Available License
 * along with this program; if not, see https://sonarsource.com/license/ssal/
 */
package org.sonarsource.scala.checks;

import java.util.Optional;
import org.sonarsource.slang.api.IdentifierTree;
import org.sonarsource.slang.api.MatchCaseTree;
import org.sonarsource.slang.api.NativeTree;
import org.sonarsource.slang.api.Tree;

class PatternMatchHelper {

  private static final String PATTERN_VARIABLE = "ScalaNativeKind(class scala.meta.Pat$Var$PatVarImpl)";

  private PatternMatchHelper() {
    // only for static usage
  }

  static boolean hasPatternMatchedVariable(MatchCaseTree caseTree) {
    Tree caseCondition = caseTree.expression();
    Tree caseBody = caseTree.body();
    return caseCondition != null && caseBody != null && caseConditionContainsVariableDeclarationUsedInTheBody(caseCondition, caseBody);
  }

  static boolean caseConditionContainsVariableDeclarationUsedInTheBody(Tree caseCondition, Tree caseBody) {
    if (caseCondition instanceof NativeTree &&
      PATTERN_VARIABLE.equals(((NativeTree) caseCondition).nativeKind().toString())) {
      Optional variableDeclaration = caseCondition.descendants()
        .filter(IdentifierTree.class::isInstance)
        .map(IdentifierTree.class::cast)
        .findFirst();
      return variableDeclaration.filter(variable -> variableIsUsedIn(variable, caseBody)).isPresent();
    }
    return caseCondition.children().stream()
        .anyMatch(caseConditionChild -> caseConditionContainsVariableDeclarationUsedInTheBody(caseConditionChild, caseBody));
  }

  private static boolean variableIsUsedIn(IdentifierTree variableDeclaration, Tree caseBody) {
    if (caseBody instanceof IdentifierTree) {
      return ((IdentifierTree) caseBody).name().equals(variableDeclaration.name());
    }
    return caseBody.children().stream().anyMatch(caseBodyChild -> variableIsUsedIn(variableDeclaration, caseBodyChild));
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy