From e221402b38840665741643d4e0a87e0c4b055d36 Mon Sep 17 00:00:00 2001 From: Mark Phelps <209477+markphelps@users.noreply.github.com> Date: Tue, 1 Oct 2024 15:59:54 -0400 Subject: [PATCH] fix(flipt): set variant attachment as value for object evaluation (#956) Signed-off-by: Mark Phelps <209477+markphelps@users.noreply.github.com> --- providers/flipt/README.md | 3 +- providers/flipt/pom.xml | 15 +++++++ .../providers/flipt/FliptProvider.java | 44 ++++++++++++++----- .../providers/flipt/FliptProviderTest.java | 28 +++++++++--- .../src/test/resources/variant-object.json | 6 +++ 5 files changed, 80 insertions(+), 16 deletions(-) create mode 100644 providers/flipt/src/test/resources/variant-object.json diff --git a/providers/flipt/README.md b/providers/flipt/README.md index 482df09af..1e95c5e99 100644 --- a/providers/flipt/README.md +++ b/providers/flipt/README.md @@ -19,7 +19,8 @@ ## Concepts * Boolean evaluation gets feature boolean evaluation / enabled status. -* Non-boolean evaluation gets feature variant key. +* Object evaluation gets variant attachment. +* Other evaluations gets feature variant key. ## Usage diff --git a/providers/flipt/pom.xml b/providers/flipt/pom.xml index 6960ddc5c..29bb603a9 100644 --- a/providers/flipt/pom.xml +++ b/providers/flipt/pom.xml @@ -20,6 +20,15 @@ false + + + markphelps + Mark Phelps + Flipt Software + https://flipt.io/ + + + io.flipt @@ -33,6 +42,12 @@ 2.0.16 + + com.fasterxml.jackson.core + jackson-databind + 2.17.2 + + com.github.tomakehurst wiremock-jre8 diff --git a/providers/flipt/src/main/java/dev/openfeature/contrib/providers/flipt/FliptProvider.java b/providers/flipt/src/main/java/dev/openfeature/contrib/providers/flipt/FliptProvider.java index 5977f9689..98ff216ad 100644 --- a/providers/flipt/src/main/java/dev/openfeature/contrib/providers/flipt/FliptProvider.java +++ b/providers/flipt/src/main/java/dev/openfeature/contrib/providers/flipt/FliptProvider.java @@ -40,7 +40,11 @@ public class FliptProvider extends EventProvider { @Getter private FliptClient fliptClient; - private AtomicBoolean isInitialized = new AtomicBoolean(false); + @Setter(AccessLevel.PROTECTED) + @Getter + private ProviderState state = ProviderState.NOT_READY; + + private final AtomicBoolean isInitialized = new AtomicBoolean(false); /** * Constructor. @@ -96,7 +100,8 @@ public ProviderEvaluation getBooleanEvaluation(String key, Boolean defa @Override public ProviderEvaluation getStringEvaluation(String key, String defaultValue, EvaluationContext ctx) { - ProviderEvaluation valueProviderEvaluation = getObjectEvaluation(key, new Value(defaultValue), ctx); + ProviderEvaluation valueProviderEvaluation = + evaluateVariant(String.class, key, new Value(defaultValue), ctx); return ProviderEvaluation.builder() .value(valueProviderEvaluation.getValue().asString()) .variant(valueProviderEvaluation.getVariant()) @@ -108,7 +113,8 @@ public ProviderEvaluation getStringEvaluation(String key, String default @Override public ProviderEvaluation getIntegerEvaluation(String key, Integer defaultValue, EvaluationContext ctx) { - ProviderEvaluation valueProviderEvaluation = getObjectEvaluation(key, new Value(defaultValue), ctx); + ProviderEvaluation valueProviderEvaluation = + evaluateVariant(Integer.class, key, new Value(defaultValue), ctx); Integer value = getIntegerValue(valueProviderEvaluation, defaultValue); return ProviderEvaluation.builder() .value(value) @@ -130,7 +136,8 @@ private static Integer getIntegerValue(ProviderEvaluation valueProviderEv @Override public ProviderEvaluation getDoubleEvaluation(String key, Double defaultValue, EvaluationContext ctx) { - ProviderEvaluation valueProviderEvaluation = getObjectEvaluation(key, new Value(defaultValue), ctx); + ProviderEvaluation valueProviderEvaluation = + evaluateVariant(Double.class, key, new Value(defaultValue), ctx); Double value = getDoubleValue(valueProviderEvaluation, defaultValue); return ProviderEvaluation.builder() .value(value) @@ -152,6 +159,18 @@ private static Double getDoubleValue(ProviderEvaluation valueProviderEval @Override public ProviderEvaluation getObjectEvaluation(String key, Value defaultValue, EvaluationContext ctx) { + return evaluateVariant(Value.class, key, defaultValue, ctx); + } + + private ProviderEvaluation evaluateVariant(Class clazz, String key, Value defaultValue, + EvaluationContext ctx) { + if (!ProviderState.READY.equals(state)) { + if (ProviderState.NOT_READY.equals(state)) { + throw new ProviderNotReadyError(PROVIDER_NOT_YET_INITIALIZED); + } + throw new GeneralError(UNKNOWN_ERROR); + } + Map contextMap = ContextTransformer.transform(ctx); EvaluationRequest request = EvaluationRequest.builder().namespaceKey(fliptProviderConfig.getNamespace()) .flagKey(key).entityId(ctx.getTargetingKey()).context(contextMap).build(); @@ -172,17 +191,22 @@ public ProviderEvaluation getObjectEvaluation(String key, Value defaultVa .build(); } + Value value = new Value(response.getVariantKey()); ImmutableMetadata.ImmutableMetadataBuilder flagMetadataBuilder = ImmutableMetadata.builder(); - if (response.getVariantAttachment() != null) { + if (response.getVariantAttachment() != null && !response.getVariantAttachment().isEmpty()) { flagMetadataBuilder.addString("variant-attachment", response.getVariantAttachment()); + + if (clazz.isAssignableFrom(Value.class)) { + value = new Value(response.getVariantAttachment()); + } } return ProviderEvaluation.builder() - .value(new Value(response.getVariantKey())) - .variant(response.getVariantKey()) - .reason(TARGETING_MATCH.name()) - .flagMetadata(flagMetadataBuilder.build()) - .build(); + .value(value) + .variant(response.getVariantKey()) + .reason(TARGETING_MATCH.name()) + .flagMetadata(flagMetadataBuilder.build()) + .build(); } @Override diff --git a/providers/flipt/src/test/java/dev/openfeature/contrib/providers/flipt/FliptProviderTest.java b/providers/flipt/src/test/java/dev/openfeature/contrib/providers/flipt/FliptProviderTest.java index 6cd62eabc..fb3c3da53 100644 --- a/providers/flipt/src/test/java/dev/openfeature/contrib/providers/flipt/FliptProviderTest.java +++ b/providers/flipt/src/test/java/dev/openfeature/contrib/providers/flipt/FliptProviderTest.java @@ -14,6 +14,7 @@ import dev.openfeature.sdk.ProviderEvaluation; import dev.openfeature.sdk.ProviderEventDetails; import dev.openfeature.sdk.ProviderState; +import dev.openfeature.sdk.Value; import dev.openfeature.sdk.exceptions.GeneralError; import dev.openfeature.sdk.exceptions.ProviderNotReadyError; import lombok.SneakyThrows; @@ -31,8 +32,7 @@ import static com.github.tomakehurst.wiremock.client.WireMock.post; import static com.github.tomakehurst.wiremock.client.WireMock.stubFor; import static com.github.tomakehurst.wiremock.client.WireMock.urlEqualTo; -import static org.junit.jupiter.api.Assertions.assertEquals; -import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.*; /** * FliptProvider test, based on APIs mocking. @@ -50,9 +50,10 @@ class FliptProviderTest { public static final Double DOUBLE_FLAG_VALUE = 1.23; public static final String USERS_FLAG_NAME = "users-flag"; public static final String TARGETING_KEY = "targeting_key"; + public static final String OBJECT_FLAG_NAME = "object-flag"; + private static FliptProvider fliptProvider; private static Client client; - private String apiUrl; @BeforeAll @@ -101,7 +102,6 @@ void getBooleanEvaluation() { mockFliptAPI("/evaluate/v1/boolean", "boolean.json", FLAG_NAME); MutableContext evaluationContext = new MutableContext(); evaluationContext.setTargetingKey(TARGETING_KEY); - assertEquals(true, fliptProvider.getBooleanEvaluation(FLAG_NAME, false, evaluationContext).getValue()); assertEquals(true, client.getBooleanValue(FLAG_NAME, false, evaluationContext)); assertEquals(false, client.getBooleanValue("non-existing", false, evaluationContext)); } @@ -171,6 +171,24 @@ void getEvaluationMetadataTest() { assertEquals("attachment-1", flagMetadata.getString("variant-attachment")); FlagEvaluationDetails nonExistingFlagEvaluation = client.getStringDetails("non-existing", "", evaluationContext); - assertEquals(null, nonExistingFlagEvaluation.getFlagMetadata().getBoolean("variant-attachment")); + assertNull(nonExistingFlagEvaluation.getFlagMetadata().getBoolean("variant-attachment")); + } + + @SneakyThrows + @Test + void getObjectEvaluationTest() { + mockFliptAPI("/evaluate/v1/variant", "variant-object.json", OBJECT_FLAG_NAME); + MutableContext evaluationContext = new MutableContext(); + evaluationContext.setTargetingKey(TARGETING_KEY); + evaluationContext.add("userId", "object"); + + Value expectedValue = new Value("{\"key1\":\"value1\",\"key2\":42,\"key3\":true}"); + Value emptyValue = new Value(); + + assertEquals(expectedValue, client.getObjectValue(OBJECT_FLAG_NAME, emptyValue, evaluationContext)); + assertEquals(emptyValue, client.getObjectValue("non-existing", emptyValue, evaluationContext)); + + // non-object flag value + assertEquals(emptyValue, client.getObjectValue(VARIANT_FLAG_NAME, emptyValue, evaluationContext)); } } \ No newline at end of file diff --git a/providers/flipt/src/test/resources/variant-object.json b/providers/flipt/src/test/resources/variant-object.json new file mode 100644 index 000000000..abf764532 --- /dev/null +++ b/providers/flipt/src/test/resources/variant-object.json @@ -0,0 +1,6 @@ +{ + "flagKey": "object-flag", + "match": true, + "variantKey": "object-variant", + "variantAttachment": "{\"key1\":\"value1\",\"key2\":42,\"key3\":true}" +} \ No newline at end of file