Skip to content

Commit a34ad58

Browse files
chore: GOFF Move data collector into a hook (open-feature#437)
Signed-off-by: Thomas Poignant <[email protected]>
1 parent d3804c3 commit a34ad58

File tree

10 files changed

+659
-356
lines changed

10 files changed

+659
-356
lines changed

providers/go-feature-flag/src/main/java/dev/openfeature/contrib/providers/gofeatureflag/EvaluationResponse.java

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
/**
88
* EvaluationResponse wrapping the provider evaluation.
9+
*
910
* @param <T> evaluation type
1011
*/
1112
@Builder

providers/go-feature-flag/src/main/java/dev/openfeature/contrib/providers/gofeatureflag/GoFeatureFlagProvider.java

+21-109
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,11 @@
1111
import dev.openfeature.contrib.providers.gofeatureflag.bean.GoFeatureFlagRequest;
1212
import dev.openfeature.contrib.providers.gofeatureflag.bean.GoFeatureFlagResponse;
1313
import dev.openfeature.contrib.providers.gofeatureflag.bean.GoFeatureFlagUser;
14-
import dev.openfeature.contrib.providers.gofeatureflag.events.Event;
15-
import dev.openfeature.contrib.providers.gofeatureflag.events.Events;
16-
import dev.openfeature.contrib.providers.gofeatureflag.events.EventsPublisher;
1714
import dev.openfeature.contrib.providers.gofeatureflag.exception.InvalidEndpoint;
1815
import dev.openfeature.contrib.providers.gofeatureflag.exception.InvalidOptions;
1916
import dev.openfeature.contrib.providers.gofeatureflag.exception.InvalidTypeInCache;
17+
import dev.openfeature.contrib.providers.gofeatureflag.hook.DataCollectorHook;
18+
import dev.openfeature.contrib.providers.gofeatureflag.hook.DataCollectorHookOptions;
2019
import dev.openfeature.sdk.ErrorCode;
2120
import dev.openfeature.sdk.EvaluationContext;
2221
import dev.openfeature.sdk.FeatureProvider;
@@ -32,6 +31,7 @@
3231
import dev.openfeature.sdk.exceptions.OpenFeatureError;
3332
import dev.openfeature.sdk.exceptions.ProviderNotReadyError;
3433
import dev.openfeature.sdk.exceptions.TypeMismatchError;
34+
import edu.umd.cs.findbugs.annotations.SuppressFBWarnings;
3535
import lombok.AccessLevel;
3636
import lombok.Getter;
3737
import lombok.extern.slf4j.Slf4j;
@@ -46,42 +46,39 @@
4646

4747
import java.io.IOException;
4848
import java.time.Duration;
49+
import java.util.ArrayList;
4950
import java.util.List;
5051
import java.util.Map;
5152
import java.util.concurrent.TimeUnit;
52-
import java.util.function.Consumer;
5353

5454
import static dev.openfeature.sdk.Value.objectToValue;
5555
import static java.net.HttpURLConnection.HTTP_BAD_REQUEST;
56-
import static java.net.HttpURLConnection.HTTP_OK;
5756
import static java.net.HttpURLConnection.HTTP_UNAUTHORIZED;
5857

5958
/**
6059
* GoFeatureFlagProvider is the JAVA provider implementation for the feature flag solution GO Feature Flag.
6160
*/
6261
@Slf4j
6362
public class GoFeatureFlagProvider implements FeatureProvider {
64-
public static final long DEFAULT_FLUSH_INTERVAL_MS = Duration.ofMinutes(1).toMillis();
65-
public static final int DEFAULT_MAX_PENDING_EVENTS = 10000;
6663
public static final long DEFAULT_CACHE_TTL_MS = 1000;
6764
public static final int DEFAULT_CACHE_CONCURRENCY_LEVEL = 1;
6865
public static final int DEFAULT_CACHE_INITIAL_CAPACITY = 100;
6966
public static final int DEFAULT_CACHE_MAXIMUM_SIZE = 10000;
67+
public static final ObjectMapper requestMapper = new ObjectMapper();
7068
protected static final String CACHED_REASON = Reason.CACHED.name();
7169
private static final String NAME = "GO Feature Flag Provider";
72-
private static final ObjectMapper requestMapper = new ObjectMapper();
7370
private static final ObjectMapper responseMapper = new ObjectMapper()
7471
.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
7572
private final GoFeatureFlagProviderOptions options;
73+
private DataCollectorHook dataCollectorHook;
74+
private final List<Hook> hooks = new ArrayList<>();
7675
private HttpUrl parsedEndpoint;
7776
// httpClient is the instance of the OkHttpClient used by the provider
7877
private OkHttpClient httpClient;
7978
// apiKey contains the token to use while calling GO Feature Flag relay proxy
8079
private String apiKey;
8180
@Getter(AccessLevel.PROTECTED)
8281
private Cache<String, ProviderEvaluation<?>> cache;
83-
@Getter(AccessLevel.PROTECTED)
84-
private EventsPublisher<Event> eventsPublisher;
8582
private ProviderState state = ProviderState.NOT_READY;
8683

8784
/**
@@ -107,21 +104,9 @@ private void validateInputOptions(GoFeatureFlagProviderOptions options) throws I
107104
throw new InvalidOptions("No options provided");
108105
}
109106

110-
if (options.getEndpoint() == null || "".equals(options.getEndpoint())) {
107+
if (options.getEndpoint() == null || options.getEndpoint().isEmpty()) {
111108
throw new InvalidEndpoint("endpoint is a mandatory field when initializing the provider");
112109
}
113-
114-
if (options.getFlushIntervalMs() != null && options.getFlushIntervalMs() <= 0) {
115-
throw new InvalidOptions("flushIntervalMs must be larger than 0");
116-
}
117-
118-
if (Boolean.FALSE.equals(options.getEnableCache()) && options.getFlushIntervalMs() != null) {
119-
throw new InvalidOptions("flushIntervalMs not used when cache is disabled");
120-
}
121-
122-
if (options.getMaxPendingEvents() != null && options.getMaxPendingEvents() <= 0) {
123-
throw new InvalidOptions("maxPendingEvents must be larger than 0");
124-
}
125110
}
126111

127112
/**
@@ -143,8 +128,9 @@ public Metadata getMetadata() {
143128
}
144129

145130
@Override
131+
@SuppressFBWarnings({"EI_EXPOSE_REP"})
146132
public List<Hook> getProviderHooks() {
147-
return FeatureProvider.super.getProviderHooks();
133+
return this.hooks;
148134
}
149135

150136
@Override
@@ -222,17 +208,15 @@ public void initialize(EvaluationContext evaluationContext) throws Exception {
222208
this.apiKey = options.getApiKey();
223209
boolean enableCache = options.getEnableCache() == null || options.getEnableCache();
224210
if (enableCache) {
225-
if (options.getCacheBuilder() != null) {
226-
this.cache = options.getCacheBuilder().build();
227-
} else {
228-
this.cache = buildDefaultCache();
229-
}
230-
long flushIntervalMs = options.getFlushIntervalMs() == null
231-
? DEFAULT_FLUSH_INTERVAL_MS : options.getFlushIntervalMs();
232-
int maxPendingEvents = options.getMaxPendingEvents() == null
233-
? DEFAULT_MAX_PENDING_EVENTS : options.getMaxPendingEvents();
234-
Consumer<List<Event>> publisher = this::publishEvents;
235-
eventsPublisher = new EventsPublisher<>(publisher, flushIntervalMs, maxPendingEvents);
211+
this.cache = options.getCacheBuilder() != null ? options.getCacheBuilder().build() : buildDefaultCache();
212+
this.dataCollectorHook = new DataCollectorHook(DataCollectorHookOptions.builder()
213+
.flushIntervalMs(options.getFlushIntervalMs())
214+
.parsedEndpoint(parsedEndpoint)
215+
.maxPendingEvents(options.getMaxPendingEvents())
216+
.apiKey(options.getApiKey())
217+
.httpClient(this.httpClient)
218+
.build());
219+
this.hooks.add(this.dataCollectorHook);
236220
}
237221
state = ProviderState.READY;
238222
log.info("finishing initializing provider, state: {}", state);
@@ -287,7 +271,6 @@ private <T> ProviderEvaluation<T> getEvaluation(
287271
return proxyRes.getProviderEvaluation();
288272
}
289273
cachedProviderEvaluation.setReason(CACHED_REASON);
290-
addCacheEvaluationEvent(key, defaultValue, user, cachedProviderEvaluation);
291274

292275
if (cachedProviderEvaluation.getValue().getClass() != expectedType) {
293276
throw new InvalidTypeInCache(expectedType, cachedProviderEvaluation.getValue().getClass());
@@ -302,30 +285,6 @@ private <T> ProviderEvaluation<T> getEvaluation(
302285
}
303286
}
304287

305-
/**
306-
* addCacheEvaluationEvent is adding an event to the list of event to send to GO Feature Flag.
307-
*
308-
* @param key - name of the feature flag
309-
* @param defaultValue - value used if something is not working as expected
310-
* @param user - user (containing EvaluationContext) used for the request
311-
* @param providerEvaluation - object containing the evaluation response for openfeature
312-
* @param <T> the type of your evaluation
313-
*/
314-
private <T> void addCacheEvaluationEvent(String key, T defaultValue, GoFeatureFlagUser user,
315-
ProviderEvaluation<?> providerEvaluation) {
316-
eventsPublisher.add(Event.builder()
317-
.key(key)
318-
.kind("feature")
319-
.contextKind(user.isAnonymous() ? "anonymousUser" : "user")
320-
.defaultValue(defaultValue)
321-
.variation(providerEvaluation.getVariant())
322-
.value(providerEvaluation.getValue())
323-
.userKey(user.getKey())
324-
.creationDate(System.currentTimeMillis())
325-
.build()
326-
);
327-
}
328-
329288
/**
330289
* resolveEvaluationGoFeatureFlagProxy is calling the GO Feature Flag API to retrieve the flag value.
331290
*
@@ -474,59 +433,12 @@ private <T> T convertValue(Object value, Class<?> expectedType) {
474433
return (T) objectToValue(value);
475434
}
476435

477-
/**
478-
* publishEvents is calling the GO Feature Flag data/collector api to store the flag usage for analytics.
479-
*
480-
* @param eventsList - list of the event to send to GO Feature Flag
481-
*/
482-
private void publishEvents(List<Event> eventsList) {
483-
try {
484-
Events events = new Events(eventsList);
485-
HttpUrl url = this.parsedEndpoint.newBuilder()
486-
.addEncodedPathSegment("v1")
487-
.addEncodedPathSegment("data")
488-
.addEncodedPathSegment("collector")
489-
.build();
490-
491-
Request.Builder reqBuilder = new Request.Builder()
492-
.url(url)
493-
.addHeader("Content-Type", "application/json")
494-
.post(RequestBody.create(
495-
requestMapper.writeValueAsBytes(events),
496-
MediaType.get("application/json; charset=utf-8")));
497-
498-
if (this.apiKey != null && !"".equals(this.apiKey)) {
499-
reqBuilder.addHeader("Authorization", "Bearer " + this.apiKey);
500-
}
501-
502-
try (Response response = this.httpClient.newCall(reqBuilder.build()).execute()) {
503-
if (response.code() == HTTP_UNAUTHORIZED) {
504-
throw new GeneralError("Unauthorized");
505-
}
506-
if (response.code() >= HTTP_BAD_REQUEST) {
507-
throw new GeneralError("Bad request: " + response.body());
508-
}
509-
510-
if (response.code() == HTTP_OK) {
511-
log.info("Published {} events successfully: {}", eventsList.size(), response.body());
512-
}
513-
} catch (IOException e) {
514-
throw new GeneralError("Impossible to send the usage data to GO Feature Flag", e);
515-
}
516-
} catch (JsonProcessingException e) {
517-
throw new GeneralError("Impossible to convert data collector events", e);
518-
}
519-
}
520436

521437
@Override
522438
public void shutdown() {
523439
log.info("shutdown");
524-
try {
525-
if (eventsPublisher != null) {
526-
eventsPublisher.shutdown();
527-
}
528-
} catch (Exception e) {
529-
log.error("error publishing events on shutdown", e);
440+
if (this.dataCollectorHook != null) {
441+
this.dataCollectorHook.shutdown();
530442
}
531443
}
532444
}

providers/go-feature-flag/src/main/java/dev/openfeature/contrib/providers/gofeatureflag/GoFeatureFlagProviderOptions.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -48,9 +48,9 @@ public class GoFeatureFlagProviderOptions {
4848
private String apiKey;
4949

5050
/**
51-
* (optional) If cache custom configuration is wanted, you should provide
52-
* a cache builder.
53-
* Default: null
51+
* (optional) If cache custom configuration is wanted, you should provide
52+
* a cache builder.
53+
* Default: null
5454
*/
5555
private CacheBuilder<String, ProviderEvaluation<?>> cacheBuilder;
5656

providers/go-feature-flag/src/main/java/dev/openfeature/contrib/providers/gofeatureflag/concurrent/ConcurrentUtils.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,9 @@ public class ConcurrentUtils {
1919
/**
2020
* Graceful shutdown a thread pool. <br>
2121
* See <a href="https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/ExecutorService.html">
22-
* https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/ExecutorService.html</a>
23-
* @param pool thread pool
22+
* https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/ExecutorService.html</a>
23+
*
24+
* @param pool thread pool
2425
* @param timeoutSeconds grace period timeout in seconds - timeout can be twice than this value, as first it
2526
* waits for existing tasks to terminate, then waits for cancelled tasks to terminate.
2627
*/

0 commit comments

Comments
 (0)