From ab816bd1053541df004e67dfc4bcfc387ca99219 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 01:33:13 +0200 Subject: [PATCH 01/16] Support multiple SDKs and rebrand to rewrite-feature-flags --- build.gradle.kts | 2 + .../RemoveBooleanFlag.java} | 31 ++--- .../launchdarkly/ChangeVariationDefault.java | 2 +- .../launchdarkly/MigrateUserToContext.java | 2 +- .../launchdarkly/RemoveBoolVariation.java | 57 ++++++++ .../launchdarkly/package-info.java | 21 +++ .../launchdarkly/search/FindFeatureFlag.java | 2 +- .../launchdarkly/search/package-info.java | 21 +++ .../openfeature/RemoveGetBooleanValue.java | 57 ++++++++ .../openfeature}/package-info.java | 2 +- .../package-info.java | 2 +- .../resources/META-INF/rewrite/category.yml | 2 +- .../META-INF/rewrite/launchdarkly-6.yml | 4 +- .../META-INF/rewrite/launchdarkly-7.yml | 4 +- .../featureflags/RemoveBooleanFlagTest.java | 125 ++++++++++++++++++ .../ChangeVariationDefaultTest.java | 3 +- .../MigrateUserToContextTest.java | 3 +- .../launchdarkly/RemoveBoolVariationTest.java | 112 +--------------- .../UpgradeLaunchDarkly6Test.java | 4 +- .../UpgradeLaunchDarkly7Test.java | 6 +- .../search/FindFeatureFlagTest.java | 3 +- .../RemoveGetBooleanValueTest.java | 65 +++++++++ 22 files changed, 384 insertions(+), 146 deletions(-) rename src/main/java/org/openrewrite/{launchdarkly/RemoveBoolVariation.java => featureflags/RemoveBooleanFlag.java} (80%) rename src/main/java/org/openrewrite/{ => featureflags}/launchdarkly/ChangeVariationDefault.java (99%) rename src/main/java/org/openrewrite/{ => featureflags}/launchdarkly/MigrateUserToContext.java (99%) create mode 100644 src/main/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariation.java create mode 100644 src/main/java/org/openrewrite/featureflags/launchdarkly/package-info.java rename src/main/java/org/openrewrite/{ => featureflags}/launchdarkly/search/FindFeatureFlag.java (99%) create mode 100644 src/main/java/org/openrewrite/featureflags/launchdarkly/search/package-info.java create mode 100644 src/main/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValue.java rename src/main/java/org/openrewrite/{launchdarkly/search => featureflags/openfeature}/package-info.java (93%) rename src/main/java/org/openrewrite/{launchdarkly => featureflags}/package-info.java (95%) create mode 100644 src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/ChangeVariationDefaultTest.java (98%) rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/MigrateUserToContextTest.java (97%) rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/RemoveBoolVariationTest.java (71%) rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/UpgradeLaunchDarkly6Test.java (97%) rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/UpgradeLaunchDarkly7Test.java (96%) rename src/test/java/org/openrewrite/{ => featureflags}/launchdarkly/search/FindFeatureFlagTest.java (99%) create mode 100644 src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java diff --git a/build.gradle.kts b/build.gradle.kts index 8829e3f..7b5f74c 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -21,6 +21,8 @@ dependencies { testImplementation("org.junit.jupiter:junit-jupiter-engine:latest.release") + testImplementation("dev.openfeature:sdk:latest.release") + testRuntimeOnly("org.gradle:gradle-tooling-api:latest.release") } diff --git a/src/main/java/org/openrewrite/launchdarkly/RemoveBoolVariation.java b/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java similarity index 80% rename from src/main/java/org/openrewrite/launchdarkly/RemoveBoolVariation.java rename to src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java index b78ad16..32ea343 100644 --- a/src/main/java/org/openrewrite/launchdarkly/RemoveBoolVariation.java +++ b/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java @@ -13,15 +13,13 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags; import lombok.EqualsAndHashCode; import lombok.Value; import org.openrewrite.*; import org.openrewrite.analysis.constantfold.ConstantFold; import org.openrewrite.analysis.util.CursorUtil; -import org.openrewrite.internal.StringUtils; -import org.openrewrite.internal.lang.Nullable; import org.openrewrite.java.JavaVisitor; import org.openrewrite.java.MethodMatcher; import org.openrewrite.java.search.UsesMethod; @@ -34,24 +32,26 @@ import org.openrewrite.staticanalysis.RemoveUnusedPrivateFields; import org.openrewrite.staticanalysis.SimplifyConstantIfBranchExecution; -import java.util.Optional; - @Value @EqualsAndHashCode(callSuper = false) -public class RemoveBoolVariation extends Recipe { - - private static final String METHOD_PATTERN_BOOLVARIATION = "com.launchdarkly.sdk.server.LDClient boolVariation(String, com.launchdarkly.sdk.*, boolean)"; +public class RemoveBooleanFlag extends Recipe { @Override public String getDisplayName() { - return "Remove `boolVariation` for feature key"; + return "Remove a boolean feature flag for feature key"; } @Override public String getDescription() { - return "Replace `boolVariation` invocations for feature key with value, and simplify constant if branch execution."; + return "Replace method invocations for feature key with value, and simplify constant if branch execution."; } + @Option(displayName = "Method pattern", + description = "A method pattern to match against. If not specified, will match `LDClient` `boolVariation`. " + + "The first argument must be the feature key as `String`.", + example = "dev.openfeature.sdk.Client getBooleanValue(String, Boolean)") + String methodPattern; + @Option(displayName = "Feature flag key", description = "The key of the feature flag to remove.", example = "flag-key-123abc") @@ -62,18 +62,9 @@ public String getDescription() { example = "true") Boolean replacementValue; - @Option(displayName = "Method pattern", - description = "A method pattern to match against. If not specified, will match `LDClient` `boolVariation`. " + - "The first argument must be the feature key as `String`.", - example = METHOD_PATTERN_BOOLVARIATION, - required = false) - @Nullable - String methodPattern; - @Override public TreeVisitor getVisitor() { - String pattern = Optional.ofNullable(methodPattern).filter(StringUtils::isNotEmpty).orElse(METHOD_PATTERN_BOOLVARIATION); - final MethodMatcher methodMatcher = new MethodMatcher(pattern, true); + final MethodMatcher methodMatcher = new MethodMatcher(methodPattern, true); JavaVisitor visitor = new JavaVisitor() { @Override public J visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) { diff --git a/src/main/java/org/openrewrite/launchdarkly/ChangeVariationDefault.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefault.java similarity index 99% rename from src/main/java/org/openrewrite/launchdarkly/ChangeVariationDefault.java rename to src/main/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefault.java index f8ae98d..1a25ed1 100644 --- a/src/main/java/org/openrewrite/launchdarkly/ChangeVariationDefault.java +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefault.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import lombok.EqualsAndHashCode; import lombok.Value; diff --git a/src/main/java/org/openrewrite/launchdarkly/MigrateUserToContext.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContext.java similarity index 99% rename from src/main/java/org/openrewrite/launchdarkly/MigrateUserToContext.java rename to src/main/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContext.java index 2fcd38c..321ac95 100644 --- a/src/main/java/org/openrewrite/launchdarkly/MigrateUserToContext.java +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContext.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import lombok.EqualsAndHashCode; import lombok.Value; diff --git a/src/main/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariation.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariation.java new file mode 100644 index 0000000..32976f7 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariation.java @@ -0,0 +1,57 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.launchdarkly; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.featureflags.RemoveBooleanFlag; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class RemoveBoolVariation extends Recipe { + + @Override + public String getDisplayName() { + return "Remove LaunchDarkly's `boolVariation` for feature key"; + } + + @Override + public String getDescription() { + return "Replace `boolVariation` invocations for feature key with value, and simplify constant if branch execution."; + } + + @Option(displayName = "Feature flag key", + description = "The key of the feature flag to remove.", + example = "flag-key-123abc") + String featureKey; + + @Option(displayName = "Replacement value", + description = "The value to replace the feature flag check with.", + example = "true") + Boolean replacementValue; + + @Override + public List getRecipeList() { + return Collections.singletonList(new RemoveBooleanFlag( + "com.launchdarkly.sdk.server.LDClient boolVariation(String, com.launchdarkly.sdk.*, boolean)", + featureKey, replacementValue)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/launchdarkly/package-info.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/package-info.java new file mode 100644 index 0000000..94b620d --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.launchdarkly; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/launchdarkly/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java similarity index 99% rename from src/main/java/org/openrewrite/launchdarkly/search/FindFeatureFlag.java rename to src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java index 1947901..ad96720 100644 --- a/src/main/java/org/openrewrite/launchdarkly/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly.search; +package org.openrewrite.featureflags.launchdarkly.search; import lombok.EqualsAndHashCode; import lombok.Value; diff --git a/src/main/java/org/openrewrite/featureflags/launchdarkly/search/package-info.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/package-info.java new file mode 100644 index 0000000..08c2a07 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.launchdarkly.search; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValue.java b/src/main/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValue.java new file mode 100644 index 0000000..ad080bd --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValue.java @@ -0,0 +1,57 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.openfeature; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.featureflags.RemoveBooleanFlag; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class RemoveGetBooleanValue extends Recipe { + + @Option(displayName = "Feature flag key", + description = "The key of the feature flag to remove.", + example = "flag-key-123abc") + String featureKey; + + @Option(displayName = "Replacement value", + description = "The value to replace the feature flag check with.", + example = "true") + Boolean replacementValue; + + @Override + public String getDisplayName() { + return "Remove OpenFeature's `getBooleanValue` for feature key"; + } + + @Override + public String getDescription() { + return "Replace `getBooleanValue()` invocations for `featureKey` with `replacementValue`, and simplify constant if branch execution."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new RemoveBooleanFlag( + "dev.openfeature.sdk.Features getBooleanValue(String, Boolean)", + featureKey, replacementValue)); + } +} diff --git a/src/main/java/org/openrewrite/launchdarkly/search/package-info.java b/src/main/java/org/openrewrite/featureflags/openfeature/package-info.java similarity index 93% rename from src/main/java/org/openrewrite/launchdarkly/search/package-info.java rename to src/main/java/org/openrewrite/featureflags/openfeature/package-info.java index 14d74ed..9095013 100644 --- a/src/main/java/org/openrewrite/launchdarkly/search/package-info.java +++ b/src/main/java/org/openrewrite/featureflags/openfeature/package-info.java @@ -15,7 +15,7 @@ */ @NonNullApi @NonNullFields -package org.openrewrite.launchdarkly.search; +package org.openrewrite.featureflags.openfeature; import org.openrewrite.internal.lang.NonNullApi; import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/launchdarkly/package-info.java b/src/main/java/org/openrewrite/featureflags/package-info.java similarity index 95% rename from src/main/java/org/openrewrite/launchdarkly/package-info.java rename to src/main/java/org/openrewrite/featureflags/package-info.java index 46d4008..6ad96c3 100644 --- a/src/main/java/org/openrewrite/launchdarkly/package-info.java +++ b/src/main/java/org/openrewrite/featureflags/package-info.java @@ -15,7 +15,7 @@ */ @NonNullApi @NonNullFields -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags; import org.openrewrite.internal.lang.NonNullApi; import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/resources/META-INF/rewrite/category.yml b/src/main/resources/META-INF/rewrite/category.yml index 892dac7..8aaa686 100644 --- a/src/main/resources/META-INF/rewrite/category.yml +++ b/src/main/resources/META-INF/rewrite/category.yml @@ -17,5 +17,5 @@ --- type: specs.openrewrite.org/v1beta/category name: LaunchDarkly -packageName: org.openrewrite.launchdarkly +packageName: org.openrewrite.featureflags.launchdarkly description: Recipes to perform [LaunchDarkly](https://launchdarkly.com/) migration tasks. diff --git a/src/main/resources/META-INF/rewrite/launchdarkly-6.yml b/src/main/resources/META-INF/rewrite/launchdarkly-6.yml index 2068be9..acfe5c8 100644 --- a/src/main/resources/META-INF/rewrite/launchdarkly-6.yml +++ b/src/main/resources/META-INF/rewrite/launchdarkly-6.yml @@ -16,7 +16,7 @@ --- type: specs.openrewrite.org/v1beta/recipe -name: org.openrewrite.launchdarkly.UpgradeLaunchDarkly6 +name: org.openrewrite.featureflags.launchdarkly.UpgradeLaunchDarkly6 displayName: Migrate to LaunchDarkly 6.x description: This recipe will apply changes commonly needed when migrating to LaunchDarkly 6.x. recipeList: @@ -25,4 +25,4 @@ recipeList: groupId: com.launchdarkly artifactId: launchdarkly-java-server-sdk newVersion: 6.x - - org.openrewrite.launchdarkly.MigrateUserToContext + - org.openrewrite.featureflags.launchdarkly.MigrateUserToContext diff --git a/src/main/resources/META-INF/rewrite/launchdarkly-7.yml b/src/main/resources/META-INF/rewrite/launchdarkly-7.yml index 21686cd..2559789 100644 --- a/src/main/resources/META-INF/rewrite/launchdarkly-7.yml +++ b/src/main/resources/META-INF/rewrite/launchdarkly-7.yml @@ -16,11 +16,11 @@ --- type: specs.openrewrite.org/v1beta/recipe -name: org.openrewrite.launchdarkly.UpgradeLaunchDarkly7 +name: org.openrewrite.featureflags.launchdarkly.UpgradeLaunchDarkly7 displayName: Migrate to LaunchDarkly 7.x description: This recipe will apply changes commonly needed when migrating to LaunchDarkly 7.x. recipeList: - - org.openrewrite.launchdarkly.UpgradeLaunchDarkly6 + - org.openrewrite.featureflags.launchdarkly.UpgradeLaunchDarkly6 # https://docs.launchdarkly.com/sdk/server-side/java/migration-6-to-7 - org.openrewrite.java.dependencies.UpgradeDependencyVersion: groupId: com.launchdarkly diff --git a/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java new file mode 100644 index 0000000..a2d78f5 --- /dev/null +++ b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java @@ -0,0 +1,125 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags; + +import org.junit.jupiter.api.Test; +import org.openrewrite.Issue; +import org.openrewrite.test.RewriteTest; +import org.openrewrite.test.SourceSpec; + +import static org.openrewrite.java.Assertions.java; + +class RemoveBooleanFlagTest implements RewriteTest { + + @Test + void customMethodPatternForWrapper() { + rewriteRun( + spec -> spec.recipe(new RemoveBooleanFlag("com.acme.bank.CustomLaunchDarklyWrapper featureFlagEnabled(String, boolean)", "flag-key-123abc", true)), + // language=java + java( + """ + package com.acme.bank; + + public class CustomLaunchDarklyWrapper { + public boolean featureFlagEnabled(String key, boolean fallback) { + return fallback; + } + } + """ + ), + // language=java + java( + """ + import com.acme.bank.CustomLaunchDarklyWrapper; + class Foo { + private CustomLaunchDarklyWrapper wrapper = new CustomLaunchDarklyWrapper(); + void bar() { + if (wrapper.featureFlagEnabled("flag-key-123abc", false)) { + // Application code to show the feature + System.out.println("Feature is on"); + } + else { + // The code to run if the feature is off + System.out.println("Feature is off"); + } + } + } + """, + """ + class Foo { + void bar() { + // Application code to show the feature + System.out.println("Feature is on"); + } + } + """ + ) + ); + } + + @Test + @Issue("https://github.com/openrewrite/rewrite-launchdarkly/issues/23") + void customMethodPatternNoConstants() { + // language=java + rewriteRun( + spec -> spec.recipe(new RemoveBooleanFlag("com.osd.util.ToggleChecker isToggleEnabled(String, boolean)", "FEATURE_TOGGLE1", true)), + java( + """ + package com.osd.util; + import java.util.Map; + import java.util.HashMap; + + public class ToggleChecker { + public boolean isToggleEnabled(String toggleName, boolean fallback) { + Map toggleMap = new HashMap<>(); + toggleMap.put("FEATURE_TOGGLE1", true); + toggleMap.put("FEATURE_TOGGLE2", true); + toggleMap.put("FEATURE_TOGGLE3", false); + return toggleMap.containsKey(toggleName); + } + } + """, + SourceSpec::skip + ), + java( + """ + import com.osd.util.ToggleChecker; + class Foo { + private ToggleChecker checker = new ToggleChecker(); + void bar() { + if (checker.isToggleEnabled("FEATURE_TOGGLE1", false)) { + // Application code to show the feature + System.out.println("Feature is on"); + } + else { + // The code to run if the feature is off + System.out.println("Feature is off"); + } + } + } + """, + """ + class Foo { + void bar() { + // Application code to show the feature + System.out.println("Feature is on"); + } + } + """ + ) + ); + } +} \ No newline at end of file diff --git a/src/test/java/org/openrewrite/launchdarkly/ChangeVariationDefaultTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java similarity index 98% rename from src/test/java/org/openrewrite/launchdarkly/ChangeVariationDefaultTest.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java index 0a35fec..70f1640 100644 --- a/src/test/java/org/openrewrite/launchdarkly/ChangeVariationDefaultTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java @@ -13,12 +13,13 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import org.junit.jupiter.api.Nested; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; +import org.openrewrite.featureflags.launchdarkly.ChangeVariationDefault; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; diff --git a/src/test/java/org/openrewrite/launchdarkly/MigrateUserToContextTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java similarity index 97% rename from src/test/java/org/openrewrite/launchdarkly/MigrateUserToContextTest.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java index ea8b23d..c190a7a 100644 --- a/src/test/java/org/openrewrite/launchdarkly/MigrateUserToContextTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java @@ -13,11 +13,12 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; +import org.openrewrite.featureflags.launchdarkly.MigrateUserToContext; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; diff --git a/src/test/java/org/openrewrite/launchdarkly/RemoveBoolVariationTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java similarity index 71% rename from src/test/java/org/openrewrite/launchdarkly/RemoveBoolVariationTest.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java index 2a59a16..5cbc6e6 100644 --- a/src/test/java/org/openrewrite/launchdarkly/RemoveBoolVariationTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java @@ -13,12 +13,13 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; import org.openrewrite.Issue; +import org.openrewrite.featureflags.launchdarkly.RemoveBoolVariation; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; @@ -30,7 +31,7 @@ class RemoveBoolVariationTest implements RewriteTest { @Override public void defaults(RecipeSpec spec) { - spec.recipe(new RemoveBoolVariation("flag-key-123abc", true, null)) + spec.recipe(new RemoveBoolVariation("flag-key-123abc", true)) .parser(JavaParser.fromJavaVersion() .classpathFromResources(new InMemoryExecutionContext(), "launchdarkly-java-server-sdk-6")); } @@ -147,7 +148,7 @@ void bar() { @Test void disablePermanently() { rewriteRun( - spec -> spec.recipe(new RemoveBoolVariation("flag-key-123abc", false, null)), + spec -> spec.recipe(new RemoveBoolVariation("flag-key-123abc", false)), // language=java java( """ @@ -290,109 +291,4 @@ void bar() { ) ); } - - @Test - void customMethodPatternForWrapper() { - rewriteRun( - spec -> spec.recipe(new RemoveBoolVariation("flag-key-123abc", true, "com.acme.bank.CustomLaunchDarklyWrapper featureFlagEnabled(String, boolean)")), - // language=java - java( - """ - package com.acme.bank; - - import com.launchdarkly.sdk.LDContext; - import com.launchdarkly.sdk.server.LDClient; - - public class CustomLaunchDarklyWrapper { - private LDClient client = new LDClient("sdk-key-123abc"); - public boolean featureFlagEnabled(String key, boolean fallback) { - LDContext context = null; - return client.boolVariation(key, context, false); - } - } - """ - ), - // language=java - java( - """ - import com.acme.bank.CustomLaunchDarklyWrapper; - class Foo { - private CustomLaunchDarklyWrapper wrapper = new CustomLaunchDarklyWrapper(); - void bar() { - if (wrapper.featureFlagEnabled("flag-key-123abc", false)) { - // Application code to show the feature - System.out.println("Feature is on"); - } - else { - // The code to run if the feature is off - System.out.println("Feature is off"); - } - } - } - """, - """ - class Foo { - void bar() { - // Application code to show the feature - System.out.println("Feature is on"); - } - } - """ - ) - ); - } - - @Test - @Issue("https://github.com/openrewrite/rewrite-launchdarkly/issues/23") - void customMethodPatternNoConstants() { - RemoveBoolVariation featureToggle1 = new RemoveBoolVariation("FEATURE_TOGGLE1", true, "com.osd.util.ToggleChecker isToggleEnabled(String, boolean)"); - // language=java - rewriteRun( - spec -> spec.recipe(featureToggle1), - java( - """ - package com.osd.util; - import java.util.Map; - import java.util.HashMap; - - public class ToggleChecker { - public boolean isToggleEnabled(String toggleName, boolean fallback) { - Map toggleMap = new HashMap<>(); - toggleMap.put("FEATURE_TOGGLE1", true); - toggleMap.put("FEATURE_TOGGLE2", true); - toggleMap.put("FEATURE_TOGGLE3", false); - return toggleMap.containsKey(toggleName); - } - } - """, - SourceSpec::skip - ), - java( - """ - import com.osd.util.ToggleChecker; - class Foo { - private ToggleChecker checker = new ToggleChecker(); - void bar() { - if (checker.isToggleEnabled("FEATURE_TOGGLE1", false)) { - // Application code to show the feature - System.out.println("Feature is on"); - } - else { - // The code to run if the feature is off - System.out.println("Feature is off"); - } - } - } - """, - """ - class Foo { - void bar() { - // Application code to show the feature - System.out.println("Feature is on"); - } - } - """ - ) - ); - } } diff --git a/src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly6Test.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly6Test.java similarity index 97% rename from src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly6Test.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly6Test.java index d1fd1a1..0770412 100644 --- a/src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly6Test.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly6Test.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; @@ -33,7 +33,7 @@ class UpgradeLaunchDarkly6Test implements RewriteTest { @Override public void defaults(RecipeSpec spec) { - spec.recipeFromResource("/META-INF/rewrite/launchdarkly-6.yml", "org.openrewrite.launchdarkly.UpgradeLaunchDarkly6") + spec.recipeFromResource("/META-INF/rewrite/launchdarkly-6.yml", "org.openrewrite.featureflags.launchdarkly.UpgradeLaunchDarkly6") .parser(JavaParser.fromJavaVersion().classpathFromResources(new InMemoryExecutionContext(), "launchdarkly-java-server-sdk-5")); } diff --git a/src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly7Test.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly7Test.java similarity index 96% rename from src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly7Test.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly7Test.java index c740a15..78f4ffe 100644 --- a/src/test/java/org/openrewrite/launchdarkly/UpgradeLaunchDarkly7Test.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/UpgradeLaunchDarkly7Test.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly; +package org.openrewrite.featureflags.launchdarkly; import org.junit.jupiter.api.Nested; import org.junit.jupiter.api.Test; @@ -37,9 +37,9 @@ class UpgradeLaunchDarkly7Test implements RewriteTest { @Override public void defaults(RecipeSpec spec) { spec.recipe(Environment.builder() - .scanRuntimeClasspath("org.openrewrite.launchdarkly") + .scanRuntimeClasspath("org.openrewrite.featureflags") .build() - .activateRecipes("org.openrewrite.launchdarkly.UpgradeLaunchDarkly7")) + .activateRecipes("org.openrewrite.featureflags.launchdarkly.UpgradeLaunchDarkly7")) .parser(JavaParser.fromJavaVersion().classpathFromResources(new InMemoryExecutionContext(), "launchdarkly-java-server-sdk-6")); } diff --git a/src/test/java/org/openrewrite/launchdarkly/search/FindFeatureFlagTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java similarity index 99% rename from src/test/java/org/openrewrite/launchdarkly/search/FindFeatureFlagTest.java rename to src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java index 6e0286f..3b6fcea 100644 --- a/src/test/java/org/openrewrite/launchdarkly/search/FindFeatureFlagTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java @@ -13,11 +13,12 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.launchdarkly.search; +package org.openrewrite.featureflags.launchdarkly.search; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; +import org.openrewrite.featureflags.launchdarkly.search.FindFeatureFlag; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; diff --git a/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java b/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java new file mode 100644 index 0000000..9ca9c66 --- /dev/null +++ b/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java @@ -0,0 +1,65 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.openfeature; + +import org.junit.jupiter.api.Test; +import org.openrewrite.DocumentExample; +import org.openrewrite.java.JavaParser; +import org.openrewrite.test.RecipeSpec; +import org.openrewrite.test.RewriteTest; + +import static org.openrewrite.java.Assertions.java; + +class RemoveGetBooleanValueTest implements RewriteTest { + + @Override + public void defaults(RecipeSpec spec) { + spec.recipe(new RemoveGetBooleanValue("flag-key-123abc", true)) + .parser(JavaParser.fromJavaVersion().classpath("sdk-1.9.0")); + } + + @Test + @DocumentExample + void removeGetBooleanValue() { + rewriteRun( + //language=java + java( + """ + import dev.openfeature.sdk.Client; + + class Foo { + void bar(Client client) { + if (client.getBooleanValue("flag-key-123abc", false)) { + System.out.println("Feature enabled"); + } else { + System.out.println("Feature disabled"); + } + } + } + """, + """ + import dev.openfeature.sdk.Client; + + class Foo { + void bar(Client client) { + System.out.println("Feature enabled"); + } + } + """ + ) + ); + } +} \ No newline at end of file From 97b01c0c8f2b9e6d74d1e62495aec0831b4bc1e8 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 01:36:02 +0200 Subject: [PATCH 02/16] Add missing newlines --- .../org/openrewrite/featureflags/RemoveBooleanFlagTest.java | 2 +- .../featureflags/openfeature/RemoveGetBooleanValueTest.java | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java index a2d78f5..2d74802 100644 --- a/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java +++ b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java @@ -122,4 +122,4 @@ void bar() { ) ); } -} \ No newline at end of file +} diff --git a/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java b/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java index 9ca9c66..e8dde3c 100644 --- a/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java +++ b/src/test/java/org/openrewrite/featureflags/openfeature/RemoveGetBooleanValueTest.java @@ -62,4 +62,4 @@ void bar(Client client) { ) ); } -} \ No newline at end of file +} From 649c9e5d772463877e694522ed6c1ed0a6fc7f9f Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 22:27:43 +0200 Subject: [PATCH 03/16] Rebrand project already --- README.md | 12 ++++++------ settings.gradle.kts | 2 +- src/main/resources/META-INF/rewrite/category.yml | 10 ++++++++++ .../featureflags/RemoveBooleanFlagTest.java | 2 +- 4 files changed, 18 insertions(+), 8 deletions(-) diff --git a/README.md b/README.md index 46ce97e..1d48bec 100644 --- a/README.md +++ b/README.md @@ -1,16 +1,16 @@ ![Logo](https://github.com/openrewrite/rewrite/raw/main/doc/logo-oss.png) -### Migrate LaunchDarkly. Automatically. +### Migrate feature flags. Automatically. -[![ci](https://github.com/openrewrite/rewrite-launchdarkly/actions/workflows/ci.yml/badge.svg)](https://github.com/openrewrite/rewrite-launchdarkly/actions/workflows/ci.yml) -[![Apache 2.0](https://img.shields.io/github/license/openrewrite/rewrite-launchdarkly.svg)](https://www.apache.org/licenses/LICENSE-2.0) -[![Maven Central](https://img.shields.io/maven-central/v/org.openrewrite.recipe/rewrite-launchdarkly.svg)](https://mvnrepository.com/artifact/org.openrewrite.recipe/rewrite-launchdarkly) +[![ci](https://github.com/openrewrite/rewrite-feature-flags/actions/workflows/ci.yml/badge.svg)](https://github.com/openrewrite/rewrite-feature-flags/actions/workflows/ci.yml) +[![Apache 2.0](https://img.shields.io/github/license/openrewrite/rewrite-feature-flags.svg)](https://www.apache.org/licenses/LICENSE-2.0) +[![Maven Central](https://img.shields.io/maven-central/v/org.openrewrite.recipe/rewrite-feature-flags.svg)](https://mvnrepository.com/artifact/org.openrewrite.recipe/rewrite-feature-flags) [![Revved up by Develocity](https://img.shields.io/badge/Revved%20up%20by-Develocity-06A0CE?logo=Gradle&labelColor=02303A)](https://ge.openrewrite.org/scans) ### What is this? -This project implements a [Rewrite module](https://github.com/openrewrite/rewrite) that performs common tasks when migrating to new version of LaunchDarkly. +This project implements a [Rewrite module](https://github.com/openrewrite/rewrite) that performs common tasks related to feature flags, or migrating to new versions of LaunchDarkly. -Browse [a selection of recipes available through this module in the recipe catalog](https://docs.openrewrite.org/recipes/launchdarkly). +Browse [a selection of recipes available through this module in the recipe catalog](https://docs.openrewrite.org/recipes/featureflags). ## Contributing diff --git a/settings.gradle.kts b/settings.gradle.kts index 84afeb0..1251206 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -1,4 +1,4 @@ -rootProject.name = "rewrite-launchdarkly" +rootProject.name = "rewrite-feature-flags" plugins { id("com.gradle.develocity") version "latest.release" diff --git a/src/main/resources/META-INF/rewrite/category.yml b/src/main/resources/META-INF/rewrite/category.yml index 8aaa686..63214d5 100644 --- a/src/main/resources/META-INF/rewrite/category.yml +++ b/src/main/resources/META-INF/rewrite/category.yml @@ -14,8 +14,18 @@ # limitations under the License. # +--- +type: specs.openrewrite.org/v1beta/category +name: Feature flags +packageName: org.openrewrite.featureflags +description: Recipes to perform feature flag migration tasks. --- type: specs.openrewrite.org/v1beta/category name: LaunchDarkly packageName: org.openrewrite.featureflags.launchdarkly description: Recipes to perform [LaunchDarkly](https://launchdarkly.com/) migration tasks. +--- +type: specs.openrewrite.org/v1beta/category +name: OpenFeature +packageName: org.openrewrite.featureflags.openfeature +description: Recipes to perform [OpenFeature](https://openfeature.dev/) migration tasks. diff --git a/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java index 2d74802..117f613 100644 --- a/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java +++ b/src/test/java/org/openrewrite/featureflags/RemoveBooleanFlagTest.java @@ -71,7 +71,7 @@ void bar() { } @Test - @Issue("https://github.com/openrewrite/rewrite-launchdarkly/issues/23") + @Issue("https://github.com/openrewrite/rewrite-feature-flags/issues/23") void customMethodPatternNoConstants() { // language=java rewriteRun( From 427c3dc845ea99287aed75ba698d6672d0994e04 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:18:41 +0200 Subject: [PATCH 04/16] Add FF4j --- build.gradle.kts | 1 + .../featureflags/ff4j/RemoveCheck.java | 57 ++++++++++++++++ .../featureflags/ff4j/package-info.java | 21 ++++++ .../featureflags/ff4j/RemoveCheckTest.java | 65 +++++++++++++++++++ 4 files changed, 144 insertions(+) create mode 100644 src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java create mode 100644 src/main/java/org/openrewrite/featureflags/ff4j/package-info.java create mode 100644 src/test/java/org/openrewrite/featureflags/ff4j/RemoveCheckTest.java diff --git a/build.gradle.kts b/build.gradle.kts index 7b5f74c..7bde896 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,6 +22,7 @@ dependencies { testImplementation("org.junit.jupiter:junit-jupiter-engine:latest.release") testImplementation("dev.openfeature:sdk:latest.release") + testImplementation("org.ff4j:ff4j-core:2.0.0") // 2.1.x requires Java 21 testRuntimeOnly("org.gradle:gradle-tooling-api:latest.release") } diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java b/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java new file mode 100644 index 0000000..996b260 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java @@ -0,0 +1,57 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.ff4j; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.featureflags.RemoveBooleanFlag; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class RemoveCheck extends Recipe { + + @Option(displayName = "Feature flag key", + description = "The key of the feature flag to remove.", + example = "flag-key-123abc") + String featureKey; + + @Option(displayName = "Replacement value", + description = "The value to replace the feature flag check with.", + example = "true") + Boolean replacementValue; + + @Override + public String getDisplayName() { + return "Remove OpenFeature's `getBooleanValue` for feature key"; + } + + @Override + public String getDescription() { + return "Replace `getBooleanValue()` invocations for `featureKey` with `replacementValue`, and simplify constant if branch execution."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new RemoveBooleanFlag( + "org.ff4j.FF4j check(String)", + featureKey, replacementValue)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/package-info.java b/src/main/java/org/openrewrite/featureflags/ff4j/package-info.java new file mode 100644 index 0000000..44a39d9 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/ff4j/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.ff4j; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/test/java/org/openrewrite/featureflags/ff4j/RemoveCheckTest.java b/src/test/java/org/openrewrite/featureflags/ff4j/RemoveCheckTest.java new file mode 100644 index 0000000..de3e8a7 --- /dev/null +++ b/src/test/java/org/openrewrite/featureflags/ff4j/RemoveCheckTest.java @@ -0,0 +1,65 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.ff4j; + +import org.junit.jupiter.api.Test; +import org.openrewrite.DocumentExample; +import org.openrewrite.java.JavaParser; +import org.openrewrite.test.RecipeSpec; +import org.openrewrite.test.RewriteTest; + +import static org.openrewrite.java.Assertions.java; + +class RemoveCheckTest implements RewriteTest { + + @Override + public void defaults(RecipeSpec spec) { + spec.recipe(new RemoveCheck("flag-key-123abc", true)) + .parser(JavaParser.fromJavaVersion().classpath("ff4j-core")); + } + + @Test + @DocumentExample + void removeCheck() { + rewriteRun( + //language=java + java( + """ + import org.ff4j.FF4j; + + class Test { + void bar(FF4j ff4j) { + if (ff4j.check("flag-key-123abc")) { + System.out.println("Feature enabled"); + } else { + System.out.println("Feature disabled"); + } + } + } + """, + """ + import org.ff4j.FF4j; + + class Test { + void bar(FF4j ff4j) { + System.out.println("Feature enabled"); + } + } + """ + ) + ); + } +} From ec9e1e0371cefba75ed96d251588dcfa9b4ca0ea Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:19:31 +0200 Subject: [PATCH 05/16] Update description --- .../java/org/openrewrite/featureflags/ff4j/RemoveCheck.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java b/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java index 996b260..f5c77c8 100644 --- a/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java +++ b/src/main/java/org/openrewrite/featureflags/ff4j/RemoveCheck.java @@ -40,12 +40,12 @@ public class RemoveCheck extends Recipe { @Override public String getDisplayName() { - return "Remove OpenFeature's `getBooleanValue` for feature key"; + return "Remove FF4j's `check` for feature key"; } @Override public String getDescription() { - return "Replace `getBooleanValue()` invocations for `featureKey` with `replacementValue`, and simplify constant if branch execution."; + return "Replace `check()` invocations for `featureKey` with `replacementValue`, and simplify constant if branch execution."; } @Override From c0a6b16718fd80f486b467e38657afa3252210dd Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:26:26 +0200 Subject: [PATCH 06/16] Add Unleash --- build.gradle.kts | 1 + .../featureflags/unleash/RemoveIsEnabled.java | 57 ++++++++++++++++ .../featureflags/unleash/package-info.java | 21 ++++++ .../unleash/RemoveIsEnabledTest.java | 66 +++++++++++++++++++ 4 files changed, 145 insertions(+) create mode 100644 src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java create mode 100644 src/main/java/org/openrewrite/featureflags/unleash/package-info.java create mode 100644 src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java diff --git a/build.gradle.kts b/build.gradle.kts index 7bde896..55b8c98 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -22,6 +22,7 @@ dependencies { testImplementation("org.junit.jupiter:junit-jupiter-engine:latest.release") testImplementation("dev.openfeature:sdk:latest.release") + testImplementation("io.getunleash:unleash-client-java:latest.release") testImplementation("org.ff4j:ff4j-core:2.0.0") // 2.1.x requires Java 21 testRuntimeOnly("org.gradle:gradle-tooling-api:latest.release") diff --git a/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java b/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java new file mode 100644 index 0000000..1989f1a --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java @@ -0,0 +1,57 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.unleash; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.featureflags.RemoveBooleanFlag; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class RemoveIsEnabled extends Recipe { + + @Option(displayName = "Feature flag key", + description = "The key of the feature flag to remove.", + example = "flag-key-123abc") + String featureKey; + + @Option(displayName = "Replacement value", + description = "The value to replace the feature flag check with.", + example = "true") + Boolean replacementValue; + + @Override + public String getDisplayName() { + return "Remove Unleash's `isEnabled` for feature key"; + } + + @Override + public String getDescription() { + return "Replace `isEnabled()` invocations for `featureKey` with `replacementValue`, and simplify constant if branch execution."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new RemoveBooleanFlag( + "io.getunleash.Unleash isEnabled(String)", + featureKey, replacementValue)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/unleash/package-info.java b/src/main/java/org/openrewrite/featureflags/unleash/package-info.java new file mode 100644 index 0000000..4e6fb09 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/unleash/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.unleash; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java b/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java new file mode 100644 index 0000000..480a420 --- /dev/null +++ b/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java @@ -0,0 +1,66 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.unleash; + +import org.junit.jupiter.api.Test; +import org.openrewrite.DocumentExample; +import org.openrewrite.featureflags.ff4j.RemoveCheck; +import org.openrewrite.java.JavaParser; +import org.openrewrite.test.RecipeSpec; +import org.openrewrite.test.RewriteTest; + +import static org.openrewrite.java.Assertions.java; + +class RemoveIsEnabledTest implements RewriteTest { + + @Override + public void defaults(RecipeSpec spec) { + spec.recipe(new RemoveIsEnabled("flag-key-123abc", true)) + .parser(JavaParser.fromJavaVersion().classpath("unleash-client-java")); + } + + @Test + @DocumentExample + void removeIsEnabled() { + rewriteRun( + //language=java + java( + """ + import io.getunleash.Unleash; + + class Test { + void bar(Unleash unleash) { + if (unleash.isEnabled("flag-key-123abc")) { + System.out.println("Feature enabled"); + } else { + System.out.println("Feature disabled"); + } + } + } + """, + """ + import io.getunleash.Unleash; + + class Test { + void bar(Unleash unleash) { + System.out.println("Feature enabled"); + } + } + """ + ) + ); + } +} From e4a04ffdd91b6bffc95e7561cb6fa9fd010500f2 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:31:33 +0200 Subject: [PATCH 07/16] Add new categories --- src/main/resources/META-INF/rewrite/category.yml | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/src/main/resources/META-INF/rewrite/category.yml b/src/main/resources/META-INF/rewrite/category.yml index 63214d5..d2d9a56 100644 --- a/src/main/resources/META-INF/rewrite/category.yml +++ b/src/main/resources/META-INF/rewrite/category.yml @@ -27,5 +27,15 @@ description: Recipes to perform [LaunchDarkly](https://launchdarkly.com/) migrat --- type: specs.openrewrite.org/v1beta/category name: OpenFeature +packageName: org.openrewrite.featureflags.ff4j +description: Recipes to perform [FF4j](https://ff4j.org/) migration tasks. +--- +type: specs.openrewrite.org/v1beta/category +name: OpenFeature packageName: org.openrewrite.featureflags.openfeature description: Recipes to perform [OpenFeature](https://openfeature.dev/) migration tasks. +--- +type: specs.openrewrite.org/v1beta/category +name: OpenFeature +packageName: org.openrewrite.featureflags.unleash +description: Recipes to perform [Unleash](https://getunleash.io/) migration tasks. From 41bffc44e49e0d8fa1ffde0f2ed64315ce8252c9 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:39:41 +0200 Subject: [PATCH 08/16] Update `methodPattern` description --- .../java/org/openrewrite/featureflags/RemoveBooleanFlag.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java b/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java index 32ea343..cc53893 100644 --- a/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java +++ b/src/main/java/org/openrewrite/featureflags/RemoveBooleanFlag.java @@ -47,8 +47,7 @@ public String getDescription() { } @Option(displayName = "Method pattern", - description = "A method pattern to match against. If not specified, will match `LDClient` `boolVariation`. " + - "The first argument must be the feature key as `String`.", + description = "A method pattern to match against. The first argument must be the feature key as `String`.", example = "dev.openfeature.sdk.Client getBooleanValue(String, Boolean)") String methodPattern; From 6abf26be2a957c909b9ed5d8a6d8743b2add8b34 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:45:14 +0200 Subject: [PATCH 09/16] Add a Precondition to FindFeatureFlag --- .../launchdarkly/search/FindFeatureFlag.java | 22 +++++++++---------- 1 file changed, 10 insertions(+), 12 deletions(-) diff --git a/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java index ad96720..352f000 100644 --- a/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java @@ -17,10 +17,7 @@ import lombok.EqualsAndHashCode; import lombok.Value; -import org.openrewrite.ExecutionContext; -import org.openrewrite.Option; -import org.openrewrite.Recipe; -import org.openrewrite.TreeVisitor; +import org.openrewrite.*; import org.openrewrite.analysis.InvocationMatcher; import org.openrewrite.analysis.constantfold.ConstantFold; import org.openrewrite.analysis.dataflow.DataFlowNode; @@ -31,6 +28,7 @@ import org.openrewrite.internal.lang.Nullable; import org.openrewrite.java.JavaIsoVisitor; import org.openrewrite.java.MethodMatcher; +import org.openrewrite.java.search.UsesMethod; import org.openrewrite.java.tree.Expression; import org.openrewrite.java.tree.J; import org.openrewrite.marker.SearchResult; @@ -38,7 +36,7 @@ @Value @EqualsAndHashCode(callSuper = false) public class FindFeatureFlag extends Recipe { - @Option(displayName = "Flag Type", + @Option(displayName = "Flag type", description = "The feature flag's type.", example = "Bool", valid = {"Bool", "Double", "Int", "JsonValue", "Migration", "String"}, @@ -46,7 +44,7 @@ public class FindFeatureFlag extends Recipe { @Nullable FeatureFlagType flagType; - @Option(displayName = "Feature Key", + @Option(displayName = "Feature key", description = "The unique key for the feature flag.", example = "flag-key-123abc", required = false) @@ -65,12 +63,12 @@ public String getDescription() { @Override public TreeVisitor getVisitor() { - MethodMatcher launchDarklyClientMatcher = new MethodMatcher("com.launchdarkly.sdk.server.LDClient *Variation(..)"); - return new JavaIsoVisitor() { + MethodMatcher methodMatcher = new MethodMatcher("com.launchdarkly.sdk.server.LDClient *Variation(..)"); + return Preconditions.check(new UsesMethod<>(methodMatcher), new JavaIsoVisitor() { @Override public J.MethodInvocation visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) { J.MethodInvocation m = super.visitMethodInvocation(method, ctx); - if (!launchDarklyClientMatcher.matches(m)) { + if (!methodMatcher.matches(m)) { return m; } @@ -104,7 +102,7 @@ public Expression visitExpression(Expression expression, ExecutionContext ctx) { return e; } - InvocationMatcher matcher = InvocationMatcher.fromMethodMatcher(launchDarklyClientMatcher); + InvocationMatcher matcher = InvocationMatcher.fromMethodMatcher(methodMatcher); boolean found = Dataflow.startingAt(getCursor()) .findSinks(new DataFlowSpec() { @Override @@ -122,13 +120,13 @@ public boolean isSink(DataFlowNode sinkNode) { }).isSome(); if (found) { J.MethodInvocation m = getCursor().firstEnclosing(J.MethodInvocation.class); - if (launchDarklyClientMatcher.matches(m)) { + if (methodMatcher.matches(m)) { getCursor().putMessageOnFirstEnclosing(J.MethodInvocation.class, "feature.found", true); } } return e; } - }; + }); } public enum FeatureFlagType { From 9341b9dfbf906de4b51c28b6685fab310a79ad4d Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Mon, 5 Aug 2024 23:57:20 +0200 Subject: [PATCH 10/16] Extract method for feature flag flow --- .../launchdarkly/search/FindFeatureFlag.java | 19 +++++++++---------- 1 file changed, 9 insertions(+), 10 deletions(-) diff --git a/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java index 352f000..f0ef0c9 100644 --- a/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlag.java @@ -23,7 +23,6 @@ import org.openrewrite.analysis.dataflow.DataFlowNode; import org.openrewrite.analysis.dataflow.DataFlowSpec; import org.openrewrite.analysis.dataflow.Dataflow; -import org.openrewrite.analysis.trait.expr.Literal; import org.openrewrite.internal.StringUtils; import org.openrewrite.internal.lang.Nullable; import org.openrewrite.java.JavaIsoVisitor; @@ -98,12 +97,19 @@ public J.MethodInvocation visitMethodInvocation(J.MethodInvocation method, Execu @Override public Expression visitExpression(Expression expression, ExecutionContext ctx) { Expression e = super.visitExpression(expression, ctx); + if (findFeatureKeyFlow() && methodMatcher.matches(getCursor().firstEnclosing(J.MethodInvocation.class))) { + getCursor().putMessageOnFirstEnclosing(J.MethodInvocation.class, "feature.found", true); + } + return e; + } + + private boolean findFeatureKeyFlow() { if (StringUtils.isBlank(featureKey)) { - return e; + return false; } InvocationMatcher matcher = InvocationMatcher.fromMethodMatcher(methodMatcher); - boolean found = Dataflow.startingAt(getCursor()) + return Dataflow.startingAt(getCursor()) .findSinks(new DataFlowSpec() { @Override public boolean isSource(DataFlowNode srcNode) { @@ -118,13 +124,6 @@ public boolean isSink(DataFlowNode sinkNode) { return matcher.advanced().isFirstParameter(sinkNode.getCursor()); } }).isSome(); - if (found) { - J.MethodInvocation m = getCursor().firstEnclosing(J.MethodInvocation.class); - if (methodMatcher.matches(m)) { - getCursor().putMessageOnFirstEnclosing(J.MethodInvocation.class, "feature.found", true); - } - } - return e; } }); } From 9876c6416d14c932eee9930bf6340dc03ba039bb Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:06:09 +0200 Subject: [PATCH 11/16] Generic recipe to find feature flags --- .../featureflags/FindFeatureFlag.java | 116 +++++++++++++++ .../featureflags/FindFeatureFlagTest.java | 140 ++++++++++++++++++ 2 files changed, 256 insertions(+) create mode 100644 src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java create mode 100644 src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java diff --git a/src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java new file mode 100644 index 0000000..6b6431a --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java @@ -0,0 +1,116 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.*; +import org.openrewrite.analysis.InvocationMatcher; +import org.openrewrite.analysis.constantfold.ConstantFold; +import org.openrewrite.analysis.dataflow.DataFlowNode; +import org.openrewrite.analysis.dataflow.DataFlowSpec; +import org.openrewrite.analysis.dataflow.Dataflow; +import org.openrewrite.internal.StringUtils; +import org.openrewrite.internal.lang.Nullable; +import org.openrewrite.java.JavaIsoVisitor; +import org.openrewrite.java.MethodMatcher; +import org.openrewrite.java.search.UsesMethod; +import org.openrewrite.java.tree.Expression; +import org.openrewrite.java.tree.J; +import org.openrewrite.marker.SearchResult; + +@Value +@EqualsAndHashCode(callSuper = false) +public class FindFeatureFlag extends Recipe { + + @Option(displayName = "Method pattern", + description = "A method pattern to match against. The first argument must be the feature key as `String`.", + example = "dev.openfeature.sdk.Client getBooleanValue(String, Boolean)") + String methodPattern; + + @Option(displayName = "Feature key", + description = "The unique key for the feature flag.", + example = "flag-key-123abc", + required = false) + @Nullable + String featureKey; + + @Override + public String getDisplayName() { + return "Find a feature flag"; + } + + @Override + public String getDescription() { + return "Find a feature flag matching method pattern."; + } + + @Override + public TreeVisitor getVisitor() { + MethodMatcher methodMatcher = new MethodMatcher(methodPattern, true); + return Preconditions.check(new UsesMethod<>(methodMatcher), new JavaIsoVisitor() { + @Override + public J.MethodInvocation visitMethodInvocation(J.MethodInvocation method, ExecutionContext ctx) { + J.MethodInvocation m = super.visitMethodInvocation(method, ctx); + if (!methodMatcher.matches(m)) { + return m; + } + + if (StringUtils.isBlank(featureKey)) { + return SearchResult.found(m); + } + + Boolean matchesFeatureKey = getCursor().getMessage("feature.found"); + if (matchesFeatureKey != null && matchesFeatureKey) { + return SearchResult.found(m); + } + + return m; + } + + @Override + public Expression visitExpression(Expression expression, ExecutionContext ctx) { + Expression e = super.visitExpression(expression, ctx); + if (findFeatureKeyFlow() && methodMatcher.matches(getCursor().firstEnclosing(J.MethodInvocation.class))) { + getCursor().putMessageOnFirstEnclosing(J.MethodInvocation.class, "feature.found", true); + } + return e; + } + + private boolean findFeatureKeyFlow() { + if (StringUtils.isBlank(featureKey)){ + return false; + } + + InvocationMatcher matcher = InvocationMatcher.fromMethodMatcher(methodMatcher); + return Dataflow.startingAt(getCursor()) + .findSinks(new DataFlowSpec() { + @Override + public boolean isSource(DataFlowNode srcNode) { + return ConstantFold.findConstantLiteralValue(srcNode, String.class) + .map(featureKey::equals) + .orSome(false); + } + + @Override + public boolean isSink(DataFlowNode sinkNode) { + return matcher.advanced().isFirstParameter(sinkNode.getCursor()); + } + }).isSome(); + } + }); + } +} diff --git a/src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java b/src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java new file mode 100644 index 0000000..5c53acd --- /dev/null +++ b/src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java @@ -0,0 +1,140 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags; + +import org.junit.jupiter.api.Test; +import org.openrewrite.DocumentExample; +import org.openrewrite.java.JavaParser; +import org.openrewrite.test.RecipeSpec; +import org.openrewrite.test.RewriteTest; + +import static org.openrewrite.java.Assertions.java; + +class FindFeatureFlagTest implements RewriteTest { + + @Override + public void defaults(RecipeSpec spec) { + //language=java + spec.parser(JavaParser.fromJavaVersion().dependsOn( + """ + package com.acme; + public class FeatureFlag { + public boolean isEnabled(String key) { + return false; + } + } + """ + ) + ); + } + + @Test + @DocumentExample + void findFeatureFlagWithAnyKey() { + rewriteRun( + spec -> spec.recipe(new FindFeatureFlag("com.acme.FeatureFlag isEnabled(String)", null)), + //language=java + java( + """ + import com.acme.FeatureFlag; + + class Test { + public void a() { + FeatureFlag client = new FeatureFlag(); + boolean flagValue = client.isEnabled("flag-key-123abc"); + if (flagValue) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + } + } + """, + """ + import com.acme.FeatureFlag; + + class Test { + public void a() { + FeatureFlag client = new FeatureFlag(); + boolean flagValue = /*~~>*/client.isEnabled("flag-key-123abc"); + if (flagValue) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + } + } + """ + ) + ); + } + + @Test + void findFlagWithSpecificKeyThroughFlow() { + rewriteRun( + spec -> spec.recipe(new FindFeatureFlag("com.acme.FeatureFlag isEnabled(String)", "flag-key-123abc")), + //language=java + java( + """ + import com.acme.FeatureFlag; + + class Test { + private static final String FEATURE_FLAG = "flag-key-123abc"; + private static final String FEATURE2_FLAG = "flag-key-789def"; + public void a() { + FeatureFlag client = new FeatureFlag(); + boolean flagValue = client.isEnabled(FEATURE_FLAG); + if (flagValue) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + boolean flagValue2 = client.isEnabled(FEATURE2_FLAG); + if (flagValue2) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + } + } + """, + """ + import com.acme.FeatureFlag; + + class Test { + private static final String FEATURE_FLAG = "flag-key-123abc"; + private static final String FEATURE2_FLAG = "flag-key-789def"; + public void a() { + FeatureFlag client = new FeatureFlag(); + boolean flagValue = /*~~>*/client.isEnabled(FEATURE_FLAG); + if (flagValue) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + boolean flagValue2 = client.isEnabled(FEATURE2_FLAG); + if (flagValue2) { + // Application code to show the feature + } else { + // The code to run if the feature is off + } + } + } + """ + ) + ); + } +} From 42385b571a5eb25062d85638abe15295bd4db695 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:14:35 +0200 Subject: [PATCH 12/16] Cover additional methods for Unleashed --- .../org/openrewrite/featureflags/unleash/RemoveIsEnabled.java | 2 +- .../openrewrite/featureflags/unleash/RemoveIsEnabledTest.java | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java b/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java index 1989f1a..560f768 100644 --- a/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java +++ b/src/main/java/org/openrewrite/featureflags/unleash/RemoveIsEnabled.java @@ -51,7 +51,7 @@ public String getDescription() { @Override public List getRecipeList() { return Collections.singletonList(new RemoveBooleanFlag( - "io.getunleash.Unleash isEnabled(String)", + "io.getunleash.Unleash isEnabled(String, ..)", featureKey, replacementValue)); } } diff --git a/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java b/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java index 480a420..5ca35a1 100644 --- a/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java +++ b/src/test/java/org/openrewrite/featureflags/unleash/RemoveIsEnabledTest.java @@ -17,7 +17,6 @@ import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; -import org.openrewrite.featureflags.ff4j.RemoveCheck; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; From beead70bd0d22263c3c387ce6ffd1bf6ca7dca5c Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:14:49 +0200 Subject: [PATCH 13/16] Find feature flags in use for other SDKs --- .../ff4j/search/FindFeatureFlag.java | 53 +++++++++++++++++++ .../ff4j/search/package-info.java | 21 ++++++++ .../openfeature/search/FindFeatureFlag.java | 53 +++++++++++++++++++ .../openfeature/search/package-info.java | 21 ++++++++ .../unleash/search/FindFeatureFlag.java | 53 +++++++++++++++++++ .../unleash/search/package-info.java | 21 ++++++++ 6 files changed, 222 insertions(+) create mode 100644 src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java create mode 100644 src/main/java/org/openrewrite/featureflags/ff4j/search/package-info.java create mode 100644 src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java create mode 100644 src/main/java/org/openrewrite/featureflags/openfeature/search/package-info.java create mode 100644 src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java create mode 100644 src/main/java/org/openrewrite/featureflags/unleash/search/package-info.java diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java new file mode 100644 index 0000000..11cf60c --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java @@ -0,0 +1,53 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.ff4j.search; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.internal.lang.Nullable; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class FindFeatureFlag extends Recipe { + + @Option(displayName = "Feature key", + description = "The unique key for the feature flag.", + example = "flag-key-123abc", + required = false) + @Nullable + String featureKey; + + @Override + public String getDisplayName() { + return "Find a FF4j feature flag"; + } + + @Override + public String getDescription() { + return "Find a FF4j feature flag."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + "org.ff4j.FF4j check(String, ..)", featureKey)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/search/package-info.java b/src/main/java/org/openrewrite/featureflags/ff4j/search/package-info.java new file mode 100644 index 0000000..e11c0ca --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/ff4j/search/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.ff4j.search; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java new file mode 100644 index 0000000..7615cc2 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java @@ -0,0 +1,53 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.openfeature.search; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.internal.lang.Nullable; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class FindFeatureFlag extends Recipe { + + @Option(displayName = "Feature key", + description = "The unique key for the feature flag.", + example = "flag-key-123abc", + required = false) + @Nullable + String featureKey; + + @Override + public String getDisplayName() { + return "Find an OpenFeature feature flag"; + } + + @Override + public String getDescription() { + return "Find an OpenFeature feature flag."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + "dev.openfeature.sdk.Features get*Value(String, ..)", featureKey)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/openfeature/search/package-info.java b/src/main/java/org/openrewrite/featureflags/openfeature/search/package-info.java new file mode 100644 index 0000000..889fd7c --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/openfeature/search/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.openfeature.search; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java new file mode 100644 index 0000000..c4855e0 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java @@ -0,0 +1,53 @@ +/* + * Copyright 2024 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.openrewrite.featureflags.unleash.search; + +import lombok.EqualsAndHashCode; +import lombok.Value; +import org.openrewrite.Option; +import org.openrewrite.Recipe; +import org.openrewrite.internal.lang.Nullable; + +import java.util.Collections; +import java.util.List; + +@Value +@EqualsAndHashCode(callSuper = false) +public class FindFeatureFlag extends Recipe { + + @Option(displayName = "Feature key", + description = "The unique key for the feature flag.", + example = "flag-key-123abc", + required = false) + @Nullable + String featureKey; + + @Override + public String getDisplayName() { + return "Find an Unleash feature flag"; + } + + @Override + public String getDescription() { + return "Find an Unleash feature flag."; + } + + @Override + public List getRecipeList() { + return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + "io.getunleash.Unleash isEnabled(String, ..)", featureKey)); + } +} diff --git a/src/main/java/org/openrewrite/featureflags/unleash/search/package-info.java b/src/main/java/org/openrewrite/featureflags/unleash/search/package-info.java new file mode 100644 index 0000000..8e1ac2f --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/unleash/search/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.unleash.search; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; From e7b88e78b3b4950de3f59b340d362fe14e0f413f Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:24:48 +0200 Subject: [PATCH 14/16] Update description in build.gradle.kts --- build.gradle.kts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle.kts b/build.gradle.kts index 55b8c98..c1c4ca7 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -3,7 +3,7 @@ plugins { } group = "org.openrewrite.recipe" -description = "LaunchDarkly Migration" +description = "Feature flag migration" val rewriteVersion = rewriteRecipe.rewriteVersion.get() dependencies { From 8b790aa5289f784c088cbd60572bc0ca927bf603 Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:46:17 +0200 Subject: [PATCH 15/16] Organize imports --- .../featureflags/launchdarkly/ChangeVariationDefaultTest.java | 1 - .../featureflags/launchdarkly/MigrateUserToContextTest.java | 1 - .../featureflags/launchdarkly/RemoveBoolVariationTest.java | 3 --- .../featureflags/launchdarkly/search/FindFeatureFlagTest.java | 1 - 4 files changed, 6 deletions(-) diff --git a/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java index 70f1640..7191810 100644 --- a/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/ChangeVariationDefaultTest.java @@ -19,7 +19,6 @@ import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; -import org.openrewrite.featureflags.launchdarkly.ChangeVariationDefault; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; diff --git a/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java index c190a7a..c0d40e8 100644 --- a/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/MigrateUserToContextTest.java @@ -18,7 +18,6 @@ import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; -import org.openrewrite.featureflags.launchdarkly.MigrateUserToContext; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; diff --git a/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java index 5cbc6e6..b2bd753 100644 --- a/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/RemoveBoolVariationTest.java @@ -18,12 +18,9 @@ import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; -import org.openrewrite.Issue; -import org.openrewrite.featureflags.launchdarkly.RemoveBoolVariation; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; -import org.openrewrite.test.SourceSpec; import static org.openrewrite.java.Assertions.java; diff --git a/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java b/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java index 3b6fcea..cd33c66 100644 --- a/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java +++ b/src/test/java/org/openrewrite/featureflags/launchdarkly/search/FindFeatureFlagTest.java @@ -18,7 +18,6 @@ import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample; import org.openrewrite.InMemoryExecutionContext; -import org.openrewrite.featureflags.launchdarkly.search.FindFeatureFlag; import org.openrewrite.java.JavaParser; import org.openrewrite.test.RecipeSpec; import org.openrewrite.test.RewriteTest; From 3dae65c08ace1714d781feb8746683fd709c5fba Mon Sep 17 00:00:00 2001 From: Tim te Beek Date: Tue, 6 Aug 2024 00:47:43 +0200 Subject: [PATCH 16/16] Move generic FindFeatureFlag recipe to search package too --- .../ff4j/search/FindFeatureFlag.java | 2 +- .../openfeature/search/FindFeatureFlag.java | 2 +- .../{ => search}/FindFeatureFlag.java | 2 +- .../featureflags/search/package-info.java | 21 +++++++++++++++++++ .../unleash/search/FindFeatureFlag.java | 2 +- .../{ => search}/FindFeatureFlagTest.java | 2 +- 6 files changed, 26 insertions(+), 5 deletions(-) rename src/main/java/org/openrewrite/featureflags/{ => search}/FindFeatureFlag.java (99%) create mode 100644 src/main/java/org/openrewrite/featureflags/search/package-info.java rename src/test/java/org/openrewrite/featureflags/{ => search}/FindFeatureFlagTest.java (99%) diff --git a/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java index 11cf60c..65929fe 100644 --- a/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/ff4j/search/FindFeatureFlag.java @@ -47,7 +47,7 @@ public String getDescription() { @Override public List getRecipeList() { - return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + return Collections.singletonList(new org.openrewrite.featureflags.search.FindFeatureFlag( "org.ff4j.FF4j check(String, ..)", featureKey)); } } diff --git a/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java index 7615cc2..2b20714 100644 --- a/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/openfeature/search/FindFeatureFlag.java @@ -47,7 +47,7 @@ public String getDescription() { @Override public List getRecipeList() { - return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + return Collections.singletonList(new org.openrewrite.featureflags.search.FindFeatureFlag( "dev.openfeature.sdk.Features get*Value(String, ..)", featureKey)); } } diff --git a/src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/search/FindFeatureFlag.java similarity index 99% rename from src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java rename to src/main/java/org/openrewrite/featureflags/search/FindFeatureFlag.java index 6b6431a..91e7dce 100644 --- a/src/main/java/org/openrewrite/featureflags/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/search/FindFeatureFlag.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.featureflags; +package org.openrewrite.featureflags.search; import lombok.EqualsAndHashCode; import lombok.Value; diff --git a/src/main/java/org/openrewrite/featureflags/search/package-info.java b/src/main/java/org/openrewrite/featureflags/search/package-info.java new file mode 100644 index 0000000..45ac455 --- /dev/null +++ b/src/main/java/org/openrewrite/featureflags/search/package-info.java @@ -0,0 +1,21 @@ +/* + * Copyright 2023 the original author or authors. + *

+ * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + *

+ * https://www.apache.org/licenses/LICENSE-2.0 + *

+ * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +@NonNullApi +@NonNullFields +package org.openrewrite.featureflags.search; + +import org.openrewrite.internal.lang.NonNullApi; +import org.openrewrite.internal.lang.NonNullFields; diff --git a/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java b/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java index c4855e0..d58b3bb 100644 --- a/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java +++ b/src/main/java/org/openrewrite/featureflags/unleash/search/FindFeatureFlag.java @@ -47,7 +47,7 @@ public String getDescription() { @Override public List getRecipeList() { - return Collections.singletonList(new org.openrewrite.featureflags.FindFeatureFlag( + return Collections.singletonList(new org.openrewrite.featureflags.search.FindFeatureFlag( "io.getunleash.Unleash isEnabled(String, ..)", featureKey)); } } diff --git a/src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java b/src/test/java/org/openrewrite/featureflags/search/FindFeatureFlagTest.java similarity index 99% rename from src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java rename to src/test/java/org/openrewrite/featureflags/search/FindFeatureFlagTest.java index 5c53acd..506ea80 100644 --- a/src/test/java/org/openrewrite/featureflags/FindFeatureFlagTest.java +++ b/src/test/java/org/openrewrite/featureflags/search/FindFeatureFlagTest.java @@ -13,7 +13,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package org.openrewrite.featureflags; +package org.openrewrite.featureflags.search; import org.junit.jupiter.api.Test; import org.openrewrite.DocumentExample;