Skip to content

Commit ed98b85

Browse files
authored
Overhaul logging integration. (#123)
* Overhaul logging integration. * Update README to reflect more modern install conventions. * Refactor SimpleLogger/NoOpLogger to use standard python logging. * Update tests. * Fix logger deprecation warning test.
1 parent cca95a6 commit ed98b85

16 files changed

+866
-594
lines changed

.gitignore

+4
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,16 @@ MANIFEST
99
dist
1010

1111
# Output of running coverage locally
12+
cover
1213
.coverage
1314

1415
# From python-testapp.git
1516
.idea/
1617
.virt/
1718
datafile.json
1819

20+
# vim swap files
21+
*.swp
22+
1923
# OSX folder metadata
2024
*.DS_Store

README.md

+6-9
Original file line numberDiff line numberDiff line change
@@ -26,23 +26,20 @@ See the Optimizely Full Stack [developer documentation](http://developers.optimi
2626

2727
### Building the SDK
2828

29-
Build the SDK using the following command:
29+
Build and install the SDK with pip, using the following command:
3030

3131
```
32-
python setup.py sdist
32+
pip install -e .
3333
```
3434

35-
This will create a tarball under `dist/`
35+
### Unit tests
3636

37-
You can then install the SDK and its dependencies with:
37+
##### Running all tests
3838

39+
To get test dependencies installed, use a modified version of the install command:
3940
```
40-
pip install dist/optimizely-sdk-{VERSION}.tar.gz
41+
pip install -e .[test]
4142
```
42-
43-
### Unit tests
44-
45-
##### Running all tests
4643
You can run all unit tests with:
4744

4845
```

optimizely/bucketer.py

+21-11
Original file line numberDiff line numberDiff line change
@@ -80,9 +80,10 @@ def find_bucket(self, bucketing_id, parent_id, traffic_allocations):
8080

8181
bucketing_key = BUCKETING_ID_TEMPLATE.format(bucketing_id=bucketing_id, parent_id=parent_id)
8282
bucketing_number = self._generate_bucket_value(bucketing_key)
83-
self.config.logger.log(enums.LogLevels.DEBUG,
84-
'Assigned bucket %s to user with bucketing ID "%s".' % (bucketing_number,
85-
bucketing_id))
83+
self.config.logger.debug('Assigned bucket %s to user with bucketing ID "%s".' % (
84+
bucketing_number,
85+
bucketing_id
86+
))
8687

8788
for traffic_allocation in traffic_allocations:
8889
current_end_of_range = traffic_allocation.get('endOfRange')
@@ -115,24 +116,33 @@ def bucket(self, experiment, user_id, bucketing_id):
115116

116117
user_experiment_id = self.find_bucket(bucketing_id, experiment.groupId, group.trafficAllocation)
117118
if not user_experiment_id:
118-
self.config.logger.log(enums.LogLevels.INFO, 'User "%s" is in no experiment.' % user_id)
119+
self.config.logger.info('User "%s" is in no experiment.' % user_id)
119120
return None
120121

121122
if user_experiment_id != experiment.id:
122-
self.config.logger.log(enums.LogLevels.INFO, 'User "%s" is not in experiment "%s" of group %s.' %
123-
(user_id, experiment.key, experiment.groupId))
123+
self.config.logger.info('User "%s" is not in experiment "%s" of group %s.' % (
124+
user_id,
125+
experiment.key,
126+
experiment.groupId
127+
))
124128
return None
125129

126-
self.config.logger.log(enums.LogLevels.INFO, 'User "%s" is in experiment %s of group %s.' %
127-
(user_id, experiment.key, experiment.groupId))
130+
self.config.logger.info('User "%s" is in experiment %s of group %s.' % (
131+
user_id,
132+
experiment.key,
133+
experiment.groupId
134+
))
128135

129136
# Bucket user if not in white-list and in group (if any)
130137
variation_id = self.find_bucket(bucketing_id, experiment.id, experiment.trafficAllocation)
131138
if variation_id:
132139
variation = self.config.get_variation_from_id(experiment.key, variation_id)
133-
self.config.logger.log(enums.LogLevels.INFO, 'User "%s" is in variation "%s" of experiment %s.' %
134-
(user_id, variation.key, experiment.key))
140+
self.config.logger.info('User "%s" is in variation "%s" of experiment %s.' % (
141+
user_id,
142+
variation.key,
143+
experiment.key
144+
))
135145
return variation
136146

137-
self.config.logger.log(enums.LogLevels.INFO, 'User "%s" is in no variation.' % user_id)
147+
self.config.logger.info('User "%s" is in no variation.' % user_id)
138148
return None

optimizely/decision_service.py

+47-44
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@
1111
# See the License for the specific language governing permissions and
1212
# limitations under the License.
1313

14-
import sys
1514
from collections import namedtuple
1615

1716
from . import bucketer
@@ -21,7 +20,6 @@
2120
from .helpers import validator
2221
from .user_profile import UserProfile
2322

24-
2523
Decision = namedtuple('Decision', 'experiment variation source')
2624
DECISION_SOURCE_EXPERIMENT = 'experiment'
2725
DECISION_SOURCE_ROLLOUT = 'rollout'
@@ -68,8 +66,7 @@ def get_forced_variation(self, experiment, user_id):
6866
variation_key = forced_variations.get(user_id)
6967
variation = self.config.get_variation_from_key(experiment.key, variation_key)
7068
if variation:
71-
self.config.logger.log(enums.LogLevels.INFO,
72-
'User "%s" is forced in variation "%s".' % (user_id, variation_key))
69+
self.logger.info('User "%s" is forced in variation "%s".' % (user_id, variation_key))
7370
return variation
7471

7572
return None
@@ -91,9 +88,11 @@ def get_stored_variation(self, experiment, user_profile):
9188
if variation_id:
9289
variation = self.config.get_variation_from_id(experiment.key, variation_id)
9390
if variation:
94-
self.config.logger.log(enums.LogLevels.INFO,
95-
'Found a stored decision. User "%s" is in variation "%s" of experiment "%s".' %
96-
(user_id, variation.key, experiment.key))
91+
self.logger.info('Found a stored decision. User "%s" is in variation "%s" of experiment "%s".' % (
92+
user_id,
93+
variation.key,
94+
experiment.key
95+
))
9796
return variation
9897

9998
return None
@@ -119,7 +118,7 @@ def get_variation(self, experiment, user_id, attributes, ignore_user_profile=Fal
119118

120119
# Check if experiment is running
121120
if not experiment_helper.is_experiment_running(experiment):
122-
self.logger.log(enums.LogLevels.INFO, 'Experiment "%s" is not running.' % experiment.key)
121+
self.logger.info('Experiment "%s" is not running.' % experiment.key)
123122
return None
124123

125124
# Check if the user is forced into a variation
@@ -138,11 +137,7 @@ def get_variation(self, experiment, user_id, attributes, ignore_user_profile=Fal
138137
try:
139138
retrieved_profile = self.user_profile_service.lookup(user_id)
140139
except:
141-
error = sys.exc_info()[1]
142-
self.logger.log(
143-
enums.LogLevels.ERROR,
144-
'Unable to retrieve user profile for user "%s" as lookup failed. Error: %s' % (user_id, str(error))
145-
)
140+
self.logger.exception('Unable to retrieve user profile for user "%s" as lookup failed.' % user_id)
146141
retrieved_profile = None
147142

148143
if validator.is_user_profile_valid(retrieved_profile):
@@ -151,14 +146,14 @@ def get_variation(self, experiment, user_id, attributes, ignore_user_profile=Fal
151146
if variation:
152147
return variation
153148
else:
154-
self.logger.log(enums.LogLevels.WARNING, 'User profile has invalid format.')
149+
self.logger.warning('User profile has invalid format.')
155150

156151
# Bucket user and store the new decision
157152
if not audience_helper.is_user_in_experiment(self.config, experiment, attributes):
158-
self.logger.log(
159-
enums.LogLevels.INFO,
160-
'User "%s" does not meet conditions to be in experiment "%s".' % (user_id, experiment.key)
161-
)
153+
self.logger.info('User "%s" does not meet conditions to be in experiment "%s".' % (
154+
user_id,
155+
experiment.key
156+
))
162157
return None
163158

164159
# Determine bucketing ID to be used
@@ -172,9 +167,7 @@ def get_variation(self, experiment, user_id, attributes, ignore_user_profile=Fal
172167
user_profile.save_variation_for_experiment(experiment.id, variation.id)
173168
self.user_profile_service.save(user_profile.__dict__)
174169
except:
175-
error = sys.exc_info()[1]
176-
self.logger.log(enums.LogLevels.ERROR,
177-
'Unable to save user profile for user "%s". Error: %s' % (user_id, str(error)))
170+
self.logger.exception('Unable to save user profile for user "%s".' % user_id)
178171
return variation
179172

180173
return None
@@ -199,25 +192,27 @@ def get_variation_for_rollout(self, rollout, user_id, attributes=None):
199192

200193
# Check if user meets audience conditions for targeting rule
201194
if not audience_helper.is_user_in_experiment(self.config, experiment, attributes):
202-
self.logger.log(
203-
enums.LogLevels.DEBUG,
204-
'User "%s" does not meet conditions for targeting rule %s.' % (user_id, idx + 1)
205-
)
195+
self.logger.debug('User "%s" does not meet conditions for targeting rule %s.' % (
196+
user_id,
197+
idx + 1
198+
))
206199
continue
207200

208-
self.logger.log(enums.LogLevels.DEBUG, 'User "%s" meets conditions for targeting rule %s.' % (user_id, idx + 1))
201+
self.logger.debug('User "%s" meets conditions for targeting rule %s.' % (user_id, idx + 1))
209202
# Determine bucketing ID to be used
210203
bucketing_id = self._get_bucketing_id(user_id, attributes)
211204
variation = self.bucketer.bucket(experiment, user_id, bucketing_id)
212205
if variation:
213-
self.logger.log(enums.LogLevels.DEBUG,
214-
'User "%s" is in variation %s of experiment %s.' % (user_id, variation.key, experiment.key))
206+
self.logger.debug('User "%s" is in variation %s of experiment %s.' % (
207+
user_id,
208+
variation.key,
209+
experiment.key
210+
))
215211
return Decision(experiment, variation, DECISION_SOURCE_ROLLOUT)
216212
else:
217213
# Evaluate no further rules
218-
self.logger.log(enums.LogLevels.DEBUG,
219-
'User "%s" is not in the traffic group for the targeting else. '
220-
'Checking "Everyone Else" rule now.' % user_id)
214+
self.logger.debug('User "%s" is not in the traffic group for the targeting else. '
215+
'Checking "Everyone Else" rule now.' % user_id)
221216
break
222217

223218
# Evaluate last rule i.e. "Everyone Else" rule
@@ -229,8 +224,7 @@ def get_variation_for_rollout(self, rollout, user_id, attributes=None):
229224
bucketing_id = self._get_bucketing_id(user_id, attributes)
230225
variation = self.bucketer.bucket(everyone_else_experiment, user_id, bucketing_id)
231226
if variation:
232-
self.logger.log(enums.LogLevels.DEBUG,
233-
'User "%s" meets conditions for targeting rule "Everyone Else".' % user_id)
227+
self.logger.debug('User "%s" meets conditions for targeting rule "Everyone Else".' % user_id)
234228
return Decision(everyone_else_experiment, variation, DECISION_SOURCE_ROLLOUT)
235229

236230
return Decision(None, None, DECISION_SOURCE_ROLLOUT)
@@ -250,14 +244,17 @@ def get_experiment_in_group(self, group, bucketing_id):
250244
if experiment_id:
251245
experiment = self.config.get_experiment_from_id(experiment_id)
252246
if experiment:
253-
self.logger.log(enums.LogLevels.INFO,
254-
'User with bucketing ID "%s" is in experiment %s of group %s.' %
255-
(bucketing_id, experiment.key, group.id))
247+
self.logger.info('User with bucketing ID "%s" is in experiment %s of group %s.' % (
248+
bucketing_id,
249+
experiment.key,
250+
group.id
251+
))
256252
return experiment
257253

258-
self.logger.log(enums.LogLevels.INFO,
259-
'User with bucketing ID "%s" is not in any experiments of group %s.' %
260-
(bucketing_id, group.id))
254+
self.logger.info('User with bucketing ID "%s" is not in any experiments of group %s.' % (
255+
bucketing_id,
256+
group.id
257+
))
261258

262259
return None
263260

@@ -286,10 +283,13 @@ def get_variation_for_feature(self, feature, user_id, attributes=None):
286283
variation = self.get_variation(experiment, user_id, attributes)
287284

288285
if variation:
289-
self.logger.log(enums.LogLevels.DEBUG,
290-
'User "%s" is in variation %s of experiment %s.' % (user_id, variation.key, experiment.key))
286+
self.logger.debug('User "%s" is in variation %s of experiment %s.' % (
287+
user_id,
288+
variation.key,
289+
experiment.key
290+
))
291291
else:
292-
self.logger.log(enums.LogLevels.ERROR, enums.Errors.INVALID_GROUP_ID_ERROR.format('_get_variation_for_feature'))
292+
self.logger.error(enums.Errors.INVALID_GROUP_ID_ERROR.format('_get_variation_for_feature'))
293293

294294
# Next check if the feature is being experimented on
295295
elif feature.experimentIds:
@@ -299,8 +299,11 @@ def get_variation_for_feature(self, feature, user_id, attributes=None):
299299
variation = self.get_variation(experiment, user_id, attributes)
300300

301301
if variation:
302-
self.logger.log(enums.LogLevels.DEBUG,
303-
'User "%s" is in variation %s of experiment %s.' % (user_id, variation.key, experiment.key))
302+
self.logger.debug('User "%s" is in variation %s of experiment %s.' % (
303+
user_id,
304+
variation.key,
305+
experiment.key
306+
))
304307

305308
# Next check if user is part of a rollout
306309
if not variation and feature.rolloutId:

0 commit comments

Comments
 (0)