16
16
from six import string_types
17
17
18
18
from . import bucketer
19
+ from .decision .optimizely_decide_option import OptimizelyDecideOption
19
20
from .helpers import audience as audience_helper
20
21
from .helpers import enums
21
22
from .helpers import experiment as experiment_helper
22
23
from .helpers import validator
23
24
from .optimizely_user_context import OptimizelyUserContext
24
- from .decision .optimizely_decide_option import OptimizelyDecideOption
25
25
from .user_profile import UserProfile
26
26
27
27
Decision = namedtuple ('Decision' , 'experiment variation source' )
@@ -44,14 +44,14 @@ def __init__(self, logger, user_profile_service):
44
44
def _get_bucketing_id (self , user_id , attributes ):
45
45
""" Helper method to determine bucketing ID for the user.
46
46
47
- Args:
48
- user_id: ID for user.
49
- attributes: Dict representing user attributes. May consist of bucketing ID to be used.
47
+ Args:
48
+ user_id: ID for user.
49
+ attributes: Dict representing user attributes. May consist of bucketing ID to be used.
50
50
51
- Returns:
52
- String representing bucketing ID if it is a String type in attributes else return user ID
53
- array of log messages representing decision making.
54
- """
51
+ Returns:
52
+ String representing bucketing ID if it is a String type in attributes else return user ID
53
+ array of log messages representing decision making.
54
+ """
55
55
decide_reasons = []
56
56
attributes = attributes or {}
57
57
bucketing_id = attributes .get (enums .ControlAttributes .BUCKETING_ID )
@@ -68,15 +68,15 @@ def _get_bucketing_id(self, user_id, attributes):
68
68
def set_forced_variation (self , project_config , experiment_key , user_id , variation_key ):
69
69
""" Sets users to a map of experiments to forced variations.
70
70
71
- Args:
72
- project_config: Instance of ProjectConfig.
73
- experiment_key: Key for experiment.
74
- user_id: The user ID.
75
- variation_key: Key for variation. If None, then clear the existing experiment-to-variation mapping.
71
+ Args:
72
+ project_config: Instance of ProjectConfig.
73
+ experiment_key: Key for experiment.
74
+ user_id: The user ID.
75
+ variation_key: Key for variation. If None, then clear the existing experiment-to-variation mapping.
76
76
77
- Returns:
78
- A boolean value that indicates if the set completed successfully.
79
- """
77
+ Returns:
78
+ A boolean value that indicates if the set completed successfully.
79
+ """
80
80
experiment = project_config .get_experiment_from_key (experiment_key )
81
81
if not experiment :
82
82
# The invalid experiment key will be logged inside this call.
@@ -126,15 +126,15 @@ def set_forced_variation(self, project_config, experiment_key, user_id, variatio
126
126
def get_forced_variation (self , project_config , experiment_key , user_id ):
127
127
""" Gets the forced variation key for the given user and experiment.
128
128
129
- Args:
130
- project_config: Instance of ProjectConfig.
131
- experiment_key: Key for experiment.
132
- user_id: The user ID.
129
+ Args:
130
+ project_config: Instance of ProjectConfig.
131
+ experiment_key: Key for experiment.
132
+ user_id: The user ID.
133
133
134
- Returns:
135
- The variation which the given user and experiment should be forced into and
136
- array of log messages representing decision making.
137
- """
134
+ Returns:
135
+ The variation which the given user and experiment should be forced into and
136
+ array of log messages representing decision making.
137
+ """
138
138
decide_reasons = []
139
139
if user_id not in self .forced_variation_map :
140
140
message = 'User "%s" is not in the forced variation map.' % user_id
@@ -174,15 +174,15 @@ def get_whitelisted_variation(self, project_config, experiment, user_id):
174
174
""" Determine if a user is forced into a variation (through whitelisting)
175
175
for the given experiment and return that variation.
176
176
177
- Args:
178
- project_config: Instance of ProjectConfig.
179
- experiment: Object representing the experiment for which user is to be bucketed.
180
- user_id: ID for the user.
177
+ Args:
178
+ project_config: Instance of ProjectConfig.
179
+ experiment: Object representing the experiment for which user is to be bucketed.
180
+ user_id: ID for the user.
181
181
182
- Returns:
183
- Variation in which the user with ID user_id is forced into. None if no variation and
184
- array of log messages representing decision making.
185
- """
182
+ Returns:
183
+ Variation in which the user with ID user_id is forced into. None if no variation and
184
+ array of log messages representing decision making.
185
+ """
186
186
decide_reasons = []
187
187
forced_variations = experiment .forcedVariations
188
188
@@ -202,14 +202,14 @@ def get_whitelisted_variation(self, project_config, experiment, user_id):
202
202
def get_stored_variation (self , project_config , experiment , user_profile ):
203
203
""" Determine if the user has a stored variation available for the given experiment and return that.
204
204
205
- Args:
206
- project_config: Instance of ProjectConfig.
207
- experiment: Object representing the experiment for which user is to be bucketed.
208
- user_profile: UserProfile object representing the user's profile.
205
+ Args:
206
+ project_config: Instance of ProjectConfig.
207
+ experiment: Object representing the experiment for which user is to be bucketed.
208
+ user_profile: UserProfile object representing the user's profile.
209
209
210
- Returns:
211
- Variation if available. None otherwise.
212
- """
210
+ Returns:
211
+ Variation if available. None otherwise.
212
+ """
213
213
user_id = user_profile .user_id
214
214
variation_id = user_profile .get_variation_for_experiment (experiment .id )
215
215
@@ -228,22 +228,22 @@ def get_stored_variation(self, project_config, experiment, user_profile):
228
228
def get_variation (self , project_config , experiment , user_context , options = None ):
229
229
""" Top-level function to help determine variation user should be put in.
230
230
231
- First, check if experiment is running.
232
- Second, check if user is forced in a variation.
233
- Third, check if there is a stored decision for the user and return the corresponding variation.
234
- Fourth, figure out if user is in the experiment by evaluating audience conditions if any.
235
- Fifth, bucket the user and return the variation.
236
-
237
- Args:
238
- project_config: Instance of ProjectConfig.
239
- experiment: Experiment for which user variation needs to be determined.
240
- user_context: contains user id and attributes
241
- options: Decide options.
242
-
243
- Returns:
244
- Variation user should see. None if user is not in experiment or experiment is not running
245
- And an array of log messages representing decision making.
246
- """
231
+ First, check if experiment is running.
232
+ Second, check if user is forced in a variation.
233
+ Third, check if there is a stored decision for the user and return the corresponding variation.
234
+ Fourth, figure out if user is in the experiment by evaluating audience conditions if any.
235
+ Fifth, bucket the user and return the variation.
236
+
237
+ Args:
238
+ project_config: Instance of ProjectConfig.
239
+ experiment: Experiment for which user variation needs to be determined.
240
+ user_context: contains user id and attributes
241
+ options: Decide options.
242
+
243
+ Returns:
244
+ Variation user should see. None if user is not in experiment or experiment is not running
245
+ And an array of log messages representing decision making.
246
+ """
247
247
user_id = user_context .user_id
248
248
attributes = user_context .get_user_attributes ()
249
249
@@ -333,21 +333,21 @@ def get_variation(self, project_config, experiment, user_context, options=None):
333
333
decide_reasons .append (message )
334
334
return None , decide_reasons
335
335
336
- def get_variation_for_rollout (self , project_config , feature , user , options ):
336
+ def get_variation_for_rollout (self , project_config , feature , user ):
337
337
""" Determine which experiment/variation the user is in for a given rollout.
338
338
Returns the variation of the first experiment the user qualifies for.
339
339
340
- Args:
341
- project_config: Instance of ProjectConfig.
342
- flagKey: Feature key.
343
- rollout: Rollout for which we are getting the variation.
344
- user: ID and attributes for user.
345
- options: Decide options.
340
+ Args:
341
+ project_config: Instance of ProjectConfig.
342
+ flagKey: Feature key.
343
+ rollout: Rollout for which we are getting the variation.
344
+ user: ID and attributes for user.
345
+ options: Decide options.
346
346
347
- Returns:
348
- Decision namedtuple consisting of experiment and variation for the user and
349
- array of log messages representing decision making.
350
- """
347
+ Returns:
348
+ Decision namedtuple consisting of experiment and variation for the user and
349
+ array of log messages representing decision making.
350
+ """
351
351
decide_reasons = []
352
352
353
353
if not feature or not feature .rolloutId :
@@ -366,8 +366,7 @@ def get_variation_for_rollout(self, project_config, feature, user, options):
366
366
while index < len (rollout_rules ):
367
367
decision_response , reasons_received = self .get_variation_from_delivery_rule (project_config ,
368
368
feature ,
369
- rollout_rules , index , user ,
370
- options )
369
+ rollout_rules , index , user )
371
370
372
371
decide_reasons += reasons_received
373
372
@@ -406,8 +405,7 @@ def get_variation_from_experiment_rule(self, config, flag_key, rule, user, optio
406
405
optimizely_decision_context = OptimizelyUserContext .OptimizelyDecisionContext (flag_key , rule .key )
407
406
408
407
forced_decision_variation , reasons_received = user .find_validated_forced_decision (
409
- optimizely_decision_context ,
410
- options )
408
+ optimizely_decision_context )
411
409
decide_reasons += reasons_received
412
410
413
411
if forced_decision_variation :
@@ -418,7 +416,7 @@ def get_variation_from_experiment_rule(self, config, flag_key, rule, user, optio
418
416
decide_reasons += variation_reasons
419
417
return decision_variation , decide_reasons
420
418
421
- def get_variation_from_delivery_rule (self , config , feature , rules , rule_index , user , options ):
419
+ def get_variation_from_delivery_rule (self , config , feature , rules , rule_index , user ):
422
420
""" Checks for delivery rule if decision is forced and returns it.
423
421
Otherwise returns a regular decision.
424
422
@@ -428,7 +426,6 @@ def get_variation_from_delivery_rule(self, config, feature, rules, rule_index, u
428
426
rules: Experiment rule.
429
427
rule_index: integer index of the rule in the list.
430
428
user: ID and attributes for user.
431
- options: Decide options.
432
429
433
430
Returns:
434
431
If forced decision, it returns namedtuple consisting of forced_decision_variation and skip_to_everyone_else
@@ -444,8 +441,7 @@ def get_variation_from_delivery_rule(self, config, feature, rules, rule_index, u
444
441
# check forced decision first
445
442
rule = rules [rule_index ]
446
443
optimizely_decision_context = OptimizelyUserContext .OptimizelyDecisionContext (feature .key , rule .key )
447
- forced_decision_variation , reasons_received = user .find_validated_forced_decision (optimizely_decision_context ,
448
- options )
444
+ forced_decision_variation , reasons_received = user .find_validated_forced_decision (optimizely_decision_context )
449
445
450
446
decide_reasons += reasons_received
451
447
@@ -504,15 +500,15 @@ def get_variation_from_delivery_rule(self, config, feature, rules, rule_index, u
504
500
def get_variation_for_feature (self , project_config , feature , user_context , options = None ):
505
501
""" Returns the experiment/variation the user is bucketed in for the given feature.
506
502
507
- Args:
508
- project_config: Instance of ProjectConfig.
509
- feature: Feature for which we are determining if it is enabled or not for the given user.
510
- user: user context for user.
511
- attributes: Dict representing user attributes.
512
- options: Decide options.
503
+ Args:
504
+ project_config: Instance of ProjectConfig.
505
+ feature: Feature for which we are determining if it is enabled or not for the given user.
506
+ user: user context for user.
507
+ attributes: Dict representing user attributes.
508
+ options: Decide options.
513
509
514
- Returns:
515
- Decision namedtuple consisting of experiment and variation for the user.
510
+ Returns:
511
+ Decision namedtuple consisting of experiment and variation for the user.
516
512
"""
517
513
decide_reasons = []
518
514
@@ -528,8 +524,4 @@ def get_variation_for_feature(self, project_config, feature, user_context, optio
528
524
if variation :
529
525
return Decision (experiment , variation , enums .DecisionSources .FEATURE_TEST ), decide_reasons
530
526
531
- # Next check if user is part of a rollout
532
- if feature .rolloutId :
533
- return self .get_variation_for_rollout (project_config , feature , user_context , options )
534
- else :
535
- return Decision (None , None , enums .DecisionSources .ROLLOUT ), decide_reasons
527
+ return self .get_variation_for_rollout (project_config , feature , user_context )
0 commit comments