-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathPatternMatchHelper.java
More file actions
58 lines (51 loc) · 2.46 KB
/
PatternMatchHelper.java
File metadata and controls
58 lines (51 loc) · 2.46 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
/*
* SonarSource Scala
* Copyright (C) 2018-2025 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<IdentifierTree> 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));
}
}