diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/CheckHelper.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/CheckHelper.java index 3f18abb..b9544f4 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/CheckHelper.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/CheckHelper.java @@ -19,12 +19,58 @@ import io.ecocode.ios.swift.antlr.generated.Swift5Parser; import org.antlr.v4.runtime.tree.ParseTree; +import org.antlr.v4.runtime.tree.TerminalNodeImpl; public class CheckHelper { + private static final String METHOD_CALL_PATTERN = ".%s("; + public static final String END_OF_FILE = ""; + private CheckHelper() { } + /** + * Utility method to check if the given import is present in the parse tree + * + * @param tree the parse tree to check + * @param importName the import name to check for + * @return true if the import is effectively present in the parse tree + */ public static boolean isImportExisting(ParseTree tree, String importName) { - return (tree instanceof Swift5Parser.Import_declarationContext && tree.getText().contains(importName)); + return (tree instanceof Swift5Parser.Import_declarationContext && + tree.getText().contains(importName) + ); + } + + /** + * Utility method to check if a call to the given method or function is present in the parse tree + * + * @param tree the parse tree to check + * @param methodName the method name to check for call + * @return true if the method is effectively called in the parse tree + */ + public static boolean isFunctionCalled(ParseTree tree, String methodName) { + return (tree instanceof Swift5Parser.ExpressionContext && + tree.getText().contains(String.format(METHOD_CALL_PATTERN, methodName))); + } + /** + * Utility method to check if the given expression is present in the parse tree + * + * @param tree the parse tree to check + * @param expression the expression to check for presence + * @return true if the expression is effectively used in the parse tree + */ + public static boolean isExpressionPresent(ParseTree tree, String expression) { + return (tree instanceof Swift5Parser.ExpressionContext && + tree.getText().contains(expression)); + } + + /** + * Utility method to check if the parse tree represents the end of the analyzed file + * + * @param tree the parse tree to check + * @return true if the end of file is reached within the parse tree + */ + public static boolean isEndOfFile(ParseTree tree){ + return tree instanceof TerminalNodeImpl && tree.getText().equals(END_OF_FILE); } } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/camera/CameraLeakCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/camera/CameraLeakCheck.java index 7a6f181..c0da5a9 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/camera/CameraLeakCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/camera/CameraLeakCheck.java @@ -23,6 +23,9 @@ import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isEndOfFile; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; + @Rule(key = "EC512") public class CameraLeakCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Any started capture session should be stopped."; @@ -32,17 +35,16 @@ public class CameraLeakCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext && tree.getText().contains("startRunning")) { + if (isExpressionPresent(tree,"startRunning")) { id = (Swift5Parser.ExpressionContext) tree; captureSessionStarted = true; } - if (tree instanceof Swift5Parser.ExpressionContext - && (tree.getText().contains("stopRunning"))) { + if (isExpressionPresent(tree,"stopRunning")) { captureSessionStopped = true; } - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (captureSessionStarted && !captureSessionStopped) { this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocation.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocation.java index 1fb403b..c437822 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocation.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocation.java @@ -23,7 +23,7 @@ import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; -import static io.ecocode.ios.swift.checks.CheckHelper.isImportExisting; +import static io.ecocode.ios.swift.checks.CheckHelper.*; @Rule(key = "EC524") public class ThriftyGeolocation extends SwiftRuleCheck { @@ -39,13 +39,11 @@ public void apply(ParseTree tree) { importExist = true; } - if (!geolocationUpdated - && tree instanceof Swift5Parser.ExpressionContext - && (tree.getText().contains("desiredAccuracy") || tree.getText().contains("CLActivityType"))) { - geolocationUpdated = true; - } + geolocationUpdated = geolocationUpdated || + (isExpressionPresent(tree, "desiredAccuracy") || + isExpressionPresent(tree, "CLActivityType")); - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (importExist && !geolocationUpdated) { this.recordIssue(importTree.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/IdleTimerDisabledCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/IdleTimerDisabledCheck.java index 6d4177a..1a01627 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/IdleTimerDisabledCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/IdleTimerDisabledCheck.java @@ -22,6 +22,8 @@ import org.antlr.v4.runtime.tree.ParseTree; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; + /** * Check the use of "UIApplication.shared.isIdleTimerDisabled" and triggers when set to true. */ @@ -31,12 +33,9 @@ public class IdleTimerDisabledCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - - if (tree instanceof Swift5Parser.ExpressionContext) { + if (isExpressionPresent(tree,"UIApplication.shared.isIdleTimerDisabled=true")) { Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; - if (id.getText().equals("UIApplication.shared.isIdleTimerDisabled=true")) { - this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); - } + this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } } } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/RigidAlarmCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/RigidAlarmCheck.java index 43f14cd..6168acb 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/RigidAlarmCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/idleness/RigidAlarmCheck.java @@ -38,5 +38,4 @@ public void apply(ParseTree tree) { } } } - } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/motionsensor/MotionSensorUpdateRateCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/motionsensor/MotionSensorUpdateRateCheck.java index 98ea747..64efd11 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/motionsensor/MotionSensorUpdateRateCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/motionsensor/MotionSensorUpdateRateCheck.java @@ -26,7 +26,7 @@ import java.util.Arrays; import java.util.List; -import static io.ecocode.ios.swift.checks.CheckHelper.isImportExisting; +import static io.ecocode.ios.swift.checks.CheckHelper.*; @Rule(key="EC534") public class MotionSensorUpdateRateCheck extends SwiftRuleCheck { @@ -43,12 +43,10 @@ public void apply(ParseTree tree) { importTree = (Swift5Parser.Import_declarationContext) tree; importExist = true; } + sensorRateUpdated = sensorRateUpdated || sensorRateUpdateExpressions.stream() + .anyMatch(exp -> isExpressionPresent(tree, exp)); - if (!sensorRateUpdated && tree instanceof Swift5Parser.ExpressionContext) { - sensorRateUpdated = sensorRateUpdateExpressions.stream().anyMatch(exp -> tree.getText().contains(exp)); - } - - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (importExist && !sensorRateUpdated) { this.recordIssue(importTree.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AnimationFreeCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AnimationFreeCheck.java index 4d94a0a..dbc5bda 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AnimationFreeCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AnimationFreeCheck.java @@ -25,6 +25,8 @@ import java.util.Arrays; import java.util.List; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; + /** * Check the use of "UIScreen.main.brightness" and triggers when set. @@ -43,16 +45,12 @@ public class AnimationFreeCheck extends SwiftRuleCheck { public void apply(ParseTree tree) { if (tree instanceof Swift5Parser.ExpressionContext) { - Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; - String expressionText = id.getText(); - boolean containsAnimationMethod = ANIMATION_METHODS.stream() - .anyMatch(expressionText::contains); - + .anyMatch(animation -> isExpressionPresent(tree, animation)); boolean containsUISwiftAnimationMethod = SWIFTUI_ANIMATION_METHODS.stream() - .anyMatch(expressionText::contains); - + .anyMatch(animation -> isExpressionPresent(tree, animation)); if (containsAnimationMethod || containsUISwiftAnimationMethod) { + Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AudioRecorderLeakCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AudioRecorderLeakCheck.java index c067b64..d6e2987 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AudioRecorderLeakCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/AudioRecorderLeakCheck.java @@ -23,6 +23,8 @@ import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.*; + @Rule(key = "EC515") public class AudioRecorderLeakCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Any audio recording started should be stopped."; @@ -30,35 +32,36 @@ public class AudioRecorderLeakCheck extends SwiftRuleCheck { private boolean audioRecorderStarted = false; private boolean audioRecorderStopped = false; private boolean importExist = false; + private boolean isAudioRecorderUsed = false; @Override public void apply(ParseTree tree) { - if (!importExist && tree instanceof Swift5Parser.ExpressionContext && tree.getText().contains("AVAudioRecorder")) { - importExist = true; - } + importExist = importExist || isImportExisting(tree, "AVFoundation"); + isAudioRecorderUsed = isAudioRecorderUsed || (importExist && isExpressionPresent(tree, "AVAudioRecorder")); - if (importExist) { + if (isAudioRecorderUsed) { findStartedButNotStoppedAudioRecord(tree); } } private void findStartedButNotStoppedAudioRecord(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext && tree.getText().contains("record()")) { + if (isFunctionCalled(tree, "record")) { id = (Swift5Parser.ExpressionContext) tree; audioRecorderStarted = true; } - if (tree instanceof Swift5Parser.ExpressionContext && (tree.getText().contains("stop()"))) { + if (isFunctionCalled(tree, "stop")) { audioRecorderStopped = true; } - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (audioRecorderStarted && !audioRecorderStopped) { this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } audioRecorderStarted = false; audioRecorderStopped = false; + isAudioRecorderUsed = false; importExist = false; } } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/BrightnessOverrideCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/BrightnessOverrideCheck.java index 7eef415..8f8ab24 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/BrightnessOverrideCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/BrightnessOverrideCheck.java @@ -22,6 +22,8 @@ import org.antlr.v4.runtime.tree.ParseTree; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; + /** * Check the use of "UIScreen.main.brightness" and triggers when set. */ @@ -31,11 +33,9 @@ public class BrightnessOverrideCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext) { - Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; - if (id.getText().contains("UIScreen.main.brightness")) { + if (isExpressionPresent(tree,"UIScreen.main.brightness")) { + Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } } } -} diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/FeedbackGeneratorUsageCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/FeedbackGeneratorUsageCheck.java index 5ae62e9..833ac82 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/FeedbackGeneratorUsageCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/FeedbackGeneratorUsageCheck.java @@ -19,17 +19,19 @@ import io.ecocode.ios.swift.SwiftRuleCheck; import io.ecocode.ios.swift.antlr.generated.Swift5Parser; -import io.ecocode.ios.swift.checks.CheckHelper; import org.antlr.v4.runtime.tree.ParseTree; -import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; import java.util.Objects; +import static io.ecocode.ios.swift.checks.CheckHelper.*; + @Rule(key = "EC528") public class FeedbackGeneratorUsageCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Avoid using the device vibrator to use less energy."; public static final String UI_KIT = "UIKit"; + public static final String IMPACT_OCCURRED = "impactOccurred"; + public static final String UI_IMPACT_FEEDBACK_GENERATOR_CLASS = "UIImpactFeedbackGenerator"; protected boolean isUIKitImported; protected Swift5Parser.ExpressionContext id; @@ -40,16 +42,15 @@ public class FeedbackGeneratorUsageCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - isUIKitImported = isUIKitImported || CheckHelper.isImportExisting(tree, UI_KIT); + isUIKitImported = isUIKitImported || isImportExisting(tree, UI_KIT); isFeedbackGeneratorInstantiated = isFeedbackGeneratorInstantiated || (isUIKitImported && - tree instanceof Swift5Parser.ExpressionContext && - (tree.getText().contains("UIImpactFeedbackGenerator"))); + isExpressionPresent(tree, UI_IMPACT_FEEDBACK_GENERATOR_CLASS)); isImpactMethodCalled = isImpactMethodCalled || (isFeedbackGeneratorInstantiated && - (tree.getText().contains(".impactOccurred("))); + isFunctionCalled(tree, IMPACT_OCCURRED)); if (Objects.isNull(id) && tree instanceof Swift5Parser.ExpressionContext && @@ -58,7 +59,7 @@ public void apply(ParseTree tree) { id = (Swift5Parser.ExpressionContext) tree; } - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (isImpactMethodCalled) { this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationLeakCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationLeakCheck.java index 6cf1e83..2e22d34 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationLeakCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationLeakCheck.java @@ -20,13 +20,17 @@ import io.ecocode.ios.swift.SwiftRuleCheck; import io.ecocode.ios.swift.antlr.generated.Swift5Parser; import org.antlr.v4.runtime.tree.ParseTree; -import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isEndOfFile; +import static io.ecocode.ios.swift.checks.CheckHelper.isFunctionCalled; + @Rule(key = "EC513") public class LocationLeakCheck extends SwiftRuleCheck { - private static final String DEFAULT_ISSUE_MESSAGE = "calls must be carefully paired: CLLocationManager.startUpdatingLocation() and CLLocationManager.stopUpdatingLocation()"; + public static final String START_UPDATING_LOCATION_METHOD = "startUpdatingLocation"; + public static final String STOP_UPDATING_LOCATION_METHOD = "stopUpdatingLocation"; + private static final String DEFAULT_ISSUE_MESSAGE = "calls must be carefully paired: CLLocationManager." + START_UPDATING_LOCATION_METHOD + "() and CLLocationManager." + STOP_UPDATING_LOCATION_METHOD + "()"; protected boolean firstCallExist = false; protected boolean secondCallExist = false; protected Swift5Parser.ExpressionContext id; @@ -34,16 +38,16 @@ public class LocationLeakCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext && (tree.getText().contains(".startUpdatingLocation()"))) { + if (isFunctionCalled(tree, START_UPDATING_LOCATION_METHOD)) { firstCallExist = true; id = (Swift5Parser.ExpressionContext) tree; } - if (tree instanceof Swift5Parser.ExpressionContext && (tree.getText().contains(".stopUpdatingLocation()"))) { + if (isFunctionCalled(tree, STOP_UPDATING_LOCATION_METHOD)) { secondCallExist = true; } - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (firstCallExist && !secondCallExist) { this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationUpdatesDisabledCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationUpdatesDisabledCheck.java index 31419e4..47f7d52 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationUpdatesDisabledCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/LocationUpdatesDisabledCheck.java @@ -22,6 +22,8 @@ import org.antlr.v4.runtime.tree.ParseTree; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; + /** * Check the use of "CLLocationManager#pausesLocationUpdatesAutomatically" and triggers when set to false. @@ -29,13 +31,13 @@ @Rule(key = "EC533") public class LocationUpdatesDisabledCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Do not disable location updates pause, unless absolutely necessary"; + public static final String LOCATION_UPDATES_DISABLE_EXPRESSION = ".pausesLocationUpdatesAutomatically=false"; + @Override public void apply(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext) { + if(isExpressionPresent(tree, LOCATION_UPDATES_DISABLE_EXPRESSION)){ Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; - if (id.getText().contains(".pausesLocationUpdatesAutomatically=false")) { - this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); - } + this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } } } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/MotionSensorLeakCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/MotionSensorLeakCheck.java index 9281adc..6e45ce1 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/MotionSensorLeakCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/MotionSensorLeakCheck.java @@ -23,6 +23,9 @@ import org.antlr.v4.runtime.tree.TerminalNodeImpl; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isEndOfFile; +import static io.ecocode.ios.swift.checks.CheckHelper.isFunctionCalled; + @Rule(key = "EC514") public class MotionSensorLeakCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Any motion sensor started should be stopped."; @@ -32,17 +35,16 @@ public class MotionSensorLeakCheck extends SwiftRuleCheck { @Override public void apply(ParseTree tree) { - if (tree instanceof Swift5Parser.ExpressionContext && tree.getText().contains("startAccelerometerUpdates")) { + if (isFunctionCalled(tree,"startAccelerometerUpdates")) { id = (Swift5Parser.ExpressionContext) tree; motionSensorStarted = true; } - if (tree instanceof Swift5Parser.ExpressionContext - && (tree.getText().contains("stopAccelerometerUpdates"))) { + if (isFunctionCalled(tree,"stopAccelerometerUpdates")) { motionSensorStopped = true; } - if (tree instanceof TerminalNodeImpl && tree.getText().equals("")) { + if (isEndOfFile(tree)) { if (motionSensorStarted && !motionSensorStopped) { this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } diff --git a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/TorchFreeCheck.java b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/TorchFreeCheck.java index 954edec..7630d18 100644 --- a/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/TorchFreeCheck.java +++ b/swift-lang/src/main/java/io/ecocode/ios/swift/checks/sobriety/TorchFreeCheck.java @@ -22,23 +22,25 @@ import org.antlr.v4.runtime.tree.ParseTree; import org.sonar.check.Rule; +import static io.ecocode.ios.swift.checks.CheckHelper.isExpressionPresent; +import static io.ecocode.ios.swift.checks.CheckHelper.isFunctionCalled; + /** * Check the use of "AVCaptureTorchMode.on", "setTorchModeOn(level: Float)", or "torchMode = .on" and triggers when set to true. */ @Rule(key = "EC530") public class TorchFreeCheck extends SwiftRuleCheck { private static final String DEFAULT_ISSUE_MESSAGE = "Usage of `AVCaptureDevice#torchMode` or `AVCaptureDevice#setTorchModeOn(level:)` must absolutely be avoided"; + @Override public void apply(ParseTree tree) { - - if (tree instanceof Swift5Parser.ExpressionContext) { - Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; - String expressionText = id.getText(); - if (expressionText.contains("AVCaptureTorchMode.on") || - expressionText.contains("setTorchModeOn") || - expressionText.contains("torchMode=.on")) { + if ( + isExpressionPresent(tree, "AVCaptureTorchMode.on") || + isExpressionPresent(tree, "torchMode=.on") || + isFunctionCalled(tree, "setTorchModeOn") + ) { + Swift5Parser.ExpressionContext id = (Swift5Parser.ExpressionContext) tree; this.recordIssue(id.getStart().getStartIndex(), DEFAULT_ISSUE_MESSAGE); } } - } } diff --git a/swift-lang/src/test/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocationCheckTest.java b/swift-lang/src/test/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocationCheckTest.java index 99687e0..1a01dab 100644 --- a/swift-lang/src/test/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocationCheckTest.java +++ b/swift-lang/src/test/java/io/ecocode/ios/swift/checks/geolocalisation/ThriftyGeolocationCheckTest.java @@ -47,4 +47,10 @@ public void Geo_no_trigger() { SensorContextTester context = CheckTestHelper.analyzeTestFile("checks/geolocalisation/Geolocalisation_no_trigger.swift"); assertThat(context.allIssues()).isEmpty(); } + + @Test + public void Geo_no_trigger_on_desired_accuracy() { + SensorContextTester context = CheckTestHelper.analyzeTestFile("checks/geolocalisation/Geolocalisation_no_trigger_desired_accuracy.swift"); + assertThat(context.allIssues()).isEmpty(); + } } diff --git a/swift-lang/src/test/resources/checks/geolocalisation/Geolocalisation_no_trigger_desired_accuracy.swift b/swift-lang/src/test/resources/checks/geolocalisation/Geolocalisation_no_trigger_desired_accuracy.swift new file mode 100644 index 0000000..11d84b6 --- /dev/null +++ b/swift-lang/src/test/resources/checks/geolocalisation/Geolocalisation_no_trigger_desired_accuracy.swift @@ -0,0 +1,15 @@ +import Foundation +import SwiftUI +import CLLocationManager + +final class AppDelegate: NSObject, UIApplicationDelegate { + func application( + _ application: UIApplication, + didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? = nil + ) -> Bool { + + // Should not trigger + CLLocationManager.desiredAccuracy = 2 + return true + } +}