11
11
import dev .openfeature .contrib .providers .gofeatureflag .bean .GoFeatureFlagRequest ;
12
12
import dev .openfeature .contrib .providers .gofeatureflag .bean .GoFeatureFlagResponse ;
13
13
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 ;
17
14
import dev .openfeature .contrib .providers .gofeatureflag .exception .InvalidEndpoint ;
18
15
import dev .openfeature .contrib .providers .gofeatureflag .exception .InvalidOptions ;
19
16
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 ;
20
19
import dev .openfeature .sdk .ErrorCode ;
21
20
import dev .openfeature .sdk .EvaluationContext ;
22
21
import dev .openfeature .sdk .FeatureProvider ;
32
31
import dev .openfeature .sdk .exceptions .OpenFeatureError ;
33
32
import dev .openfeature .sdk .exceptions .ProviderNotReadyError ;
34
33
import dev .openfeature .sdk .exceptions .TypeMismatchError ;
34
+ import edu .umd .cs .findbugs .annotations .SuppressFBWarnings ;
35
35
import lombok .AccessLevel ;
36
36
import lombok .Getter ;
37
37
import lombok .extern .slf4j .Slf4j ;
46
46
47
47
import java .io .IOException ;
48
48
import java .time .Duration ;
49
+ import java .util .ArrayList ;
49
50
import java .util .List ;
50
51
import java .util .Map ;
51
52
import java .util .concurrent .TimeUnit ;
52
- import java .util .function .Consumer ;
53
53
54
54
import static dev .openfeature .sdk .Value .objectToValue ;
55
55
import static java .net .HttpURLConnection .HTTP_BAD_REQUEST ;
56
- import static java .net .HttpURLConnection .HTTP_OK ;
57
56
import static java .net .HttpURLConnection .HTTP_UNAUTHORIZED ;
58
57
59
58
/**
60
59
* GoFeatureFlagProvider is the JAVA provider implementation for the feature flag solution GO Feature Flag.
61
60
*/
62
61
@ Slf4j
63
62
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 ;
66
63
public static final long DEFAULT_CACHE_TTL_MS = 1000 ;
67
64
public static final int DEFAULT_CACHE_CONCURRENCY_LEVEL = 1 ;
68
65
public static final int DEFAULT_CACHE_INITIAL_CAPACITY = 100 ;
69
66
public static final int DEFAULT_CACHE_MAXIMUM_SIZE = 10000 ;
67
+ public static final ObjectMapper requestMapper = new ObjectMapper ();
70
68
protected static final String CACHED_REASON = Reason .CACHED .name ();
71
69
private static final String NAME = "GO Feature Flag Provider" ;
72
- private static final ObjectMapper requestMapper = new ObjectMapper ();
73
70
private static final ObjectMapper responseMapper = new ObjectMapper ()
74
71
.configure (DeserializationFeature .FAIL_ON_UNKNOWN_PROPERTIES , false );
75
72
private final GoFeatureFlagProviderOptions options ;
73
+ private DataCollectorHook dataCollectorHook ;
74
+ private final List <Hook > hooks = new ArrayList <>();
76
75
private HttpUrl parsedEndpoint ;
77
76
// httpClient is the instance of the OkHttpClient used by the provider
78
77
private OkHttpClient httpClient ;
79
78
// apiKey contains the token to use while calling GO Feature Flag relay proxy
80
79
private String apiKey ;
81
80
@ Getter (AccessLevel .PROTECTED )
82
81
private Cache <String , ProviderEvaluation <?>> cache ;
83
- @ Getter (AccessLevel .PROTECTED )
84
- private EventsPublisher <Event > eventsPublisher ;
85
82
private ProviderState state = ProviderState .NOT_READY ;
86
83
87
84
/**
@@ -107,21 +104,9 @@ private void validateInputOptions(GoFeatureFlagProviderOptions options) throws I
107
104
throw new InvalidOptions ("No options provided" );
108
105
}
109
106
110
- if (options .getEndpoint () == null || "" . equals ( options .getEndpoint ())) {
107
+ if (options .getEndpoint () == null || options .getEndpoint (). isEmpty ( )) {
111
108
throw new InvalidEndpoint ("endpoint is a mandatory field when initializing the provider" );
112
109
}
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
- }
125
110
}
126
111
127
112
/**
@@ -143,8 +128,9 @@ public Metadata getMetadata() {
143
128
}
144
129
145
130
@ Override
131
+ @ SuppressFBWarnings ({"EI_EXPOSE_REP" })
146
132
public List <Hook > getProviderHooks () {
147
- return FeatureProvider . super . getProviderHooks () ;
133
+ return this . hooks ;
148
134
}
149
135
150
136
@ Override
@@ -222,17 +208,15 @@ public void initialize(EvaluationContext evaluationContext) throws Exception {
222
208
this .apiKey = options .getApiKey ();
223
209
boolean enableCache = options .getEnableCache () == null || options .getEnableCache ();
224
210
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 );
236
220
}
237
221
state = ProviderState .READY ;
238
222
log .info ("finishing initializing provider, state: {}" , state );
@@ -287,7 +271,6 @@ private <T> ProviderEvaluation<T> getEvaluation(
287
271
return proxyRes .getProviderEvaluation ();
288
272
}
289
273
cachedProviderEvaluation .setReason (CACHED_REASON );
290
- addCacheEvaluationEvent (key , defaultValue , user , cachedProviderEvaluation );
291
274
292
275
if (cachedProviderEvaluation .getValue ().getClass () != expectedType ) {
293
276
throw new InvalidTypeInCache (expectedType , cachedProviderEvaluation .getValue ().getClass ());
@@ -302,30 +285,6 @@ private <T> ProviderEvaluation<T> getEvaluation(
302
285
}
303
286
}
304
287
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
-
329
288
/**
330
289
* resolveEvaluationGoFeatureFlagProxy is calling the GO Feature Flag API to retrieve the flag value.
331
290
*
@@ -474,59 +433,12 @@ private <T> T convertValue(Object value, Class<?> expectedType) {
474
433
return (T ) objectToValue (value );
475
434
}
476
435
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
- }
520
436
521
437
@ Override
522
438
public void shutdown () {
523
439
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 ();
530
442
}
531
443
}
532
444
}
0 commit comments