1
1
/****************************************************************************
2
- * Copyright 2016-2023 , Optimizely, Inc. and contributors *
2
+ * Copyright 2016-2024 , Optimizely, Inc. and contributors *
3
3
* *
4
4
* Licensed under the Apache License, Version 2.0 (the "License"); *
5
5
* you may not use this file except in compliance with the License. *
42
42
import javax .annotation .Nonnull ;
43
43
import javax .annotation .Nullable ;
44
44
import javax .annotation .concurrent .ThreadSafe ;
45
+
45
46
import java .io .Closeable ;
46
47
import java .util .*;
48
+ import java .util .stream .Collectors ;
47
49
48
50
import static com .optimizely .ab .internal .SafetyUtils .tryClose ;
49
51
@@ -1194,55 +1196,39 @@ private OptimizelyUserContext createUserContextCopy(@Nonnull String userId, @Non
1194
1196
OptimizelyDecision decide (@ Nonnull OptimizelyUserContext user ,
1195
1197
@ Nonnull String key ,
1196
1198
@ Nonnull List <OptimizelyDecideOption > options ) {
1197
-
1198
1199
ProjectConfig projectConfig = getProjectConfig ();
1199
1200
if (projectConfig == null ) {
1200
1201
return OptimizelyDecision .newErrorDecision (key , user , DecisionMessage .SDK_NOT_READY .reason ());
1201
1202
}
1202
1203
1203
- FeatureFlag flag = projectConfig .getFeatureKeyMapping ().get (key );
1204
- if (flag == null ) {
1205
- return OptimizelyDecision .newErrorDecision (key , user , DecisionMessage .FLAG_KEY_INVALID .reason (key ));
1206
- }
1207
-
1208
- String userId = user .getUserId ();
1209
- Map <String , Object > attributes = user .getAttributes ();
1210
- Boolean decisionEventDispatched = false ;
1211
1204
List <OptimizelyDecideOption > allOptions = getAllOptions (options );
1212
- DecisionReasons decisionReasons = DefaultDecisionReasons . newInstance ( allOptions );
1205
+ allOptions . remove ( OptimizelyDecideOption . ENABLED_FLAGS_ONLY );
1213
1206
1214
- Map <String , ?> copiedAttributes = new HashMap <>(attributes );
1215
- FeatureDecision flagDecision ;
1216
-
1217
- // Check Forced Decision
1218
- OptimizelyDecisionContext optimizelyDecisionContext = new OptimizelyDecisionContext (flag .getKey (), null );
1219
- DecisionResponse <Variation > forcedDecisionVariation = decisionService .validatedForcedDecision (optimizelyDecisionContext , projectConfig , user );
1220
- decisionReasons .merge (forcedDecisionVariation .getReasons ());
1221
- if (forcedDecisionVariation .getResult () != null ) {
1222
- flagDecision = new FeatureDecision (null , forcedDecisionVariation .getResult (), FeatureDecision .DecisionSource .FEATURE_TEST );
1223
- } else {
1224
- // Regular decision
1225
- DecisionResponse <FeatureDecision > decisionVariation = decisionService .getVariationForFeature (
1226
- flag ,
1227
- user ,
1228
- projectConfig ,
1229
- allOptions );
1230
- flagDecision = decisionVariation .getResult ();
1231
- decisionReasons .merge (decisionVariation .getReasons ());
1232
- }
1207
+ return decideForKeys (user , Arrays .asList (key ), allOptions , true ).get (key );
1208
+ }
1209
+
1210
+ private OptimizelyDecision createOptimizelyDecision (
1211
+ OptimizelyUserContext user ,
1212
+ String flagKey ,
1213
+ FeatureDecision flagDecision ,
1214
+ DecisionReasons decisionReasons ,
1215
+ List <OptimizelyDecideOption > allOptions ,
1216
+ ProjectConfig projectConfig
1217
+ ) {
1218
+ String userId = user .getUserId ();
1233
1219
1234
1220
Boolean flagEnabled = false ;
1235
1221
if (flagDecision .variation != null ) {
1236
1222
if (flagDecision .variation .getFeatureEnabled ()) {
1237
1223
flagEnabled = true ;
1238
1224
}
1239
1225
}
1240
- logger .info ("Feature \" {}\" is enabled for user \" {}\" ? {}" , key , userId , flagEnabled );
1226
+ logger .info ("Feature \" {}\" is enabled for user \" {}\" ? {}" , flagKey , userId , flagEnabled );
1241
1227
1242
1228
Map <String , Object > variableMap = new HashMap <>();
1243
1229
if (!allOptions .contains (OptimizelyDecideOption .EXCLUDE_VARIABLES )) {
1244
1230
DecisionResponse <Map <String , Object >> decisionVariables = getDecisionVariableMap (
1245
- flag ,
1231
+ projectConfig . getFeatureKeyMapping (). get ( flagKey ) ,
1246
1232
flagDecision .variation ,
1247
1233
flagEnabled );
1248
1234
variableMap = decisionVariables .getResult ();
@@ -1261,22 +1247,28 @@ OptimizelyDecision decide(@Nonnull OptimizelyUserContext user,
1261
1247
// add to event metadata as well (currently set to experimentKey)
1262
1248
String ruleKey = flagDecision .experiment != null ? flagDecision .experiment .getKey () : null ;
1263
1249
1250
+
1251
+ Boolean decisionEventDispatched = false ;
1252
+
1253
+ Map <String , Object > attributes = user .getAttributes ();
1254
+ Map <String , ?> copiedAttributes = new HashMap <>(attributes );
1255
+
1264
1256
if (!allOptions .contains (OptimizelyDecideOption .DISABLE_DECISION_EVENT )) {
1265
1257
decisionEventDispatched = sendImpression (
1266
1258
projectConfig ,
1267
1259
flagDecision .experiment ,
1268
1260
userId ,
1269
1261
copiedAttributes ,
1270
1262
flagDecision .variation ,
1271
- key ,
1263
+ flagKey ,
1272
1264
decisionSource .toString (),
1273
1265
flagEnabled );
1274
1266
}
1275
1267
1276
1268
DecisionNotification decisionNotification = DecisionNotification .newFlagDecisionNotificationBuilder ()
1277
1269
.withUserId (userId )
1278
1270
.withAttributes (copiedAttributes )
1279
- .withFlagKey (key )
1271
+ .withFlagKey (flagKey )
1280
1272
.withEnabled (flagEnabled )
1281
1273
.withVariables (variableMap )
1282
1274
.withVariationKey (variationKey )
@@ -1291,30 +1283,84 @@ OptimizelyDecision decide(@Nonnull OptimizelyUserContext user,
1291
1283
flagEnabled ,
1292
1284
optimizelyJSON ,
1293
1285
ruleKey ,
1294
- key ,
1286
+ flagKey ,
1295
1287
user ,
1296
1288
reasonsToReport );
1297
1289
}
1298
1290
1299
1291
Map <String , OptimizelyDecision > decideForKeys (@ Nonnull OptimizelyUserContext user ,
1292
+ @ Nonnull List <String > keys ,
1293
+ @ Nonnull List <OptimizelyDecideOption > options ) {
1294
+ return decideForKeys (user , keys , options , false );
1295
+ }
1296
+
1297
+ private Map <String , OptimizelyDecision > decideForKeys (@ Nonnull OptimizelyUserContext user ,
1300
1298
@ Nonnull List <String > keys ,
1301
- @ Nonnull List <OptimizelyDecideOption > options ) {
1299
+ @ Nonnull List <OptimizelyDecideOption > options ,
1300
+ boolean ignoreDefaultOptions ) {
1302
1301
Map <String , OptimizelyDecision > decisionMap = new HashMap <>();
1303
1302
1304
1303
ProjectConfig projectConfig = getProjectConfig ();
1305
1304
if (projectConfig == null ) {
1306
- logger .error ("Optimizely instance is not valid, failing isFeatureEnabled call." );
1305
+ logger .error ("Optimizely instance is not valid, failing decideForKeys call." );
1307
1306
return decisionMap ;
1308
1307
}
1309
1308
1310
1309
if (keys .isEmpty ()) return decisionMap ;
1311
1310
1312
- List <OptimizelyDecideOption > allOptions = getAllOptions (options );
1311
+ List <OptimizelyDecideOption > allOptions = ignoreDefaultOptions ? options : getAllOptions (options );
1312
+
1313
+ Map <String , FeatureDecision > flagDecisions = new HashMap <>();
1314
+ Map <String , DecisionReasons > decisionReasonsMap = new HashMap <>();
1315
+
1316
+ List <FeatureFlag > flagsWithoutForcedDecision = new ArrayList <>();
1317
+
1318
+ List <String > validKeys = new ArrayList <>();
1313
1319
1314
1320
for (String key : keys ) {
1315
- OptimizelyDecision decision = decide (user , key , options );
1316
- if (!allOptions .contains (OptimizelyDecideOption .ENABLED_FLAGS_ONLY ) || decision .getEnabled ()) {
1317
- decisionMap .put (key , decision );
1321
+ FeatureFlag flag = projectConfig .getFeatureKeyMapping ().get (key );
1322
+ if (flag == null ) {
1323
+ decisionMap .put (key , OptimizelyDecision .newErrorDecision (key , user , DecisionMessage .FLAG_KEY_INVALID .reason (key )));
1324
+ continue ;
1325
+ }
1326
+
1327
+ validKeys .add (key );
1328
+
1329
+ DecisionReasons decisionReasons = DefaultDecisionReasons .newInstance (allOptions );
1330
+ decisionReasonsMap .put (key , decisionReasons );
1331
+
1332
+ OptimizelyDecisionContext optimizelyDecisionContext = new OptimizelyDecisionContext (key , null );
1333
+ DecisionResponse <Variation > forcedDecisionVariation = decisionService .validatedForcedDecision (optimizelyDecisionContext , projectConfig , user );
1334
+ decisionReasons .merge (forcedDecisionVariation .getReasons ());
1335
+
1336
+ if (forcedDecisionVariation .getResult () != null ) {
1337
+ flagDecisions .put (key ,
1338
+ new FeatureDecision (null , forcedDecisionVariation .getResult (), FeatureDecision .DecisionSource .FEATURE_TEST ));
1339
+ } else {
1340
+ flagsWithoutForcedDecision .add (flag );
1341
+ }
1342
+ }
1343
+
1344
+ List <DecisionResponse <FeatureDecision >> decisionList =
1345
+ decisionService .getVariationsForFeatureList (flagsWithoutForcedDecision , user , projectConfig , allOptions );
1346
+
1347
+ for (int i = 0 ; i < flagsWithoutForcedDecision .size (); i ++) {
1348
+ DecisionResponse <FeatureDecision > decision = decisionList .get (i );
1349
+ String flagKey = flagsWithoutForcedDecision .get (i ).getKey ();
1350
+ flagDecisions .put (flagKey , decision .getResult ());
1351
+ decisionReasonsMap .get (flagKey ).merge (decision .getReasons ());
1352
+ }
1353
+
1354
+ for (String key : validKeys ) {
1355
+ FeatureDecision flagDecision = flagDecisions .get (key );
1356
+ DecisionReasons decisionReasons = decisionReasonsMap .get ((key ));
1357
+
1358
+ OptimizelyDecision optimizelyDecision = createOptimizelyDecision (
1359
+ user , key , flagDecision , decisionReasons , allOptions , projectConfig
1360
+ );
1361
+
1362
+ if (!allOptions .contains (OptimizelyDecideOption .ENABLED_FLAGS_ONLY ) || optimizelyDecision .getEnabled ()) {
1363
+ decisionMap .put (key , optimizelyDecision );
1318
1364
}
1319
1365
}
1320
1366
0 commit comments