Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add a context manager object and a method to call said object #357

Open
wants to merge 32 commits into
base: master
Choose a base branch
from
Open
Changes from 25 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
8d1ae15
quick context manager prototype
Oct 20, 2020
529f0a4
adding some defaults
Oct 20, 2020
5bac851
default param added
Oct 20, 2020
4eeed1a
assignment, duh
Oct 20, 2020
4714ba4
debugging
Oct 20, 2020
021661d
dont delete feature flag data if exception was raise
Oct 20, 2020
2401f78
some quick refactoring to flush data when getting out of scope
Oct 20, 2020
00e4967
adding user default params
Oct 20, 2020
9c51dcc
Utilize linked list structure for scope of context manager
Oct 28, 2020
642cc20
Create watch wrapper method and rename FeatureFlags
Nov 4, 2020
ea80652
Revert renamte of FeatureFlags
Nov 4, 2020
83b77b5
Generalize FeatureFlags context manager
Nov 6, 2020
9e4c34a
Remove unnecessary re-assignment
Nov 9, 2020
5b63fc0
Clear out scopes after report_exc_info is called
Nov 9, 2020
0483523
Merge branch 'master' into context-manager-ld
Nov 10, 2020
ae692bf
Clear scope on exit only
Nov 11, 2020
fa6cf69
Change reference to scope to tag and force extra_data to be a dict
Nov 11, 2020
e7c0b5c
Attach the tag to the exception on exit
Nov 12, 2020
8d89888
Refactor tags check
Nov 17, 2020
9ca6df6
Implement specific feature flags interface
Nov 17, 2020
48eaa57
change context manager object to take in dict
Nov 17, 2020
e5fb916
Make feature flags stack thread safe
Nov 17, 2020
5bae76e
Use tags instead of feature flags
Nov 18, 2020
7378835
Add check for attr
Nov 18, 2020
69b035e
Model feature flags with tags
Nov 20, 2020
430fd29
Fix typo in comment
Nov 24, 2020
5eb2da2
Add test cases for the feature_flag context manager
Nov 25, 2020
00c89ec
Use [0][0] to index mocked object
Nov 25, 2020
25045f9
Remove feature_flag.data.order
Nov 25, 2020
bcece26
use self.tags rather than self.tag
Nov 30, 2020
1153529
Initialize _tags in thread_local on append and value
Dec 1, 2020
d3894db
Move flattening to _LocalTags and rename _tags
Dec 4, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
111 changes: 111 additions & 0 deletions rollbar/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -536,6 +536,49 @@ def wait(f=None):
return f()


def feature_flag(flag_key, variation=None, user=None):
"""
A context manager interface that sets the tags used to model a feature flag.

key: the key of the feature flag.
variation: (optional) the evaluated feature flag variation.
user: (optional) the user being evaluated.

Example usage:

with rollbar.featureflag('flag1', variation=True, user='[email protected]'):
ajtran marked this conversation as resolved.
Show resolved Hide resolved
code()

Tags generated from the above example:

[
{'key': feature_flag.key', 'value': 'flag1'},
{'key': feature_flag.data.flag1.order', 'value': 0},
{'key': feature_flag.data.flag1.variation', 'value': True},
{'key': feature_flag.data.flag1.user, 'value': '[email protected]'}
]
"""
flag_key_tag = _create_tag('feature_flag.key', flag_key)

# create the feature flag order tag
order_key = _feature_flag_data_key(flag_key, 'order')
ajtran marked this conversation as resolved.
Show resolved Hide resolved
order_tag = _create_tag(order_key, len(_tags.value))

tags = [flag_key_tag, order_tag]

if variation is not None:
variation_key = _feature_flag_data_key(flag_key, 'variation')
variation_tag = _create_tag(variation_key, variation)
tags.append(variation_tag)

if user is not None:
user_key = _feature_flag_data_key(flag_key, 'user')
user_tag = _create_tag(user_key, user)
tags.append(user_tag)

return _TagManager(tags)


class ApiException(Exception):
"""
This exception will be raised if there was a problem decoding the
Expand Down Expand Up @@ -702,6 +745,12 @@ def _report_exc_info(exc_info, request, extra_data, payload_data, level=None):
if extra_trace_data and not extra_data:
data['custom'] = extra_trace_data

# if there are tags attached to the exception, reverse the order, and append to `_tags`
# stack to send the full chain of tags to Rollbar.
tags = _tags.value + getattr(exc_info[1], '_rollbar_tags', [])[::-1]
if tags:
data['tags'] = _flatten_nested_lists(tags)

request = _get_actual_request(request)
_add_request_data(data, request)
_add_person_data(data, request)
Expand Down Expand Up @@ -788,6 +837,9 @@ def _report_message(message, level, request, extra_data, payload_data):
_add_lambda_context_data(data)
data['server'] = _build_server_data()

if _tags.value:
data['tags'] = _flatten_nested_lists(_tags.value)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What _flatten_nested_lists() is doing? isn't _tags already a flatten list and not nested?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

no, _tags.value at this point would look something like

[
  [
    {'key': 'feature_flag.name', 'value': 'feature-1'},
    {'key': 'feature_flag.data.feature-1.order', 'value': 0}
  ],
  [
    {'key': 'feature_flag.name', 'value': 'feature-2'},
    {'key': 'feature_flag.data.feature-2.order', 'value': 1},
    {'key': 'feature_flag.data.feature-2.variation', 'value': True}
  ]
]

where each inner list keeps the tags used to represent a feature flag together. I thought this was necessary because when an exception occurs in the scope of a context manager and we pop and attach the tags to the exception.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok I see. I think we need to change the tag manager so it appends each tag to the list:

class _TagManager(object):
    """
    Context manager object that interfaces with the `_tags` stack:
        On enter, puts the tag object at top of the stack.
        On exit, pops off the top element of the stack.
          - If there is an exception, attach the tag object to the exception
            for rebuilding of the `_tags` stack before reporting.
    """
    def __init__(self, tag):
        self.tag = tag

    def __enter__(self):
        _tags.append(self.tag)

That self.tag should be self.tags and we shouldn't do append() but maybe extend() so _tags is always flatten

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure I agree with using extend or keeping _tags always flatten. In the __exit__ method:

    def __exit__(self, exc_type, exc_value, traceback):

        if exc_value:
            if not hasattr(exc_value, '_rollbar_tags'):
                exc_value._rollbar_tags = []

            exc_value._rollbar_tags.append(self.tag)

        _tags.pop()

we pop from _tags whenever we exit the context manager. If this was flattened, we'd have to keep track of the number of times we should pop.

Well, I guess this is something we can do. Would something like this be preferred?

    def __exit__(self, exc_type, exc_value, traceback):

        if exc_value:
            if not hasattr(exc_value, '_rollbar_tags'):
                exc_value._rollbar_tags = []

            exc_value._rollbar_tags.append(self.tag)

        for tag in self.tag:
            _tags.pop()

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh ok, I see what you mean. You are right although maybe we can:

  • rename _tags to something different that represents better that they are "context tags" or similar
  • Add a method in _LocalTags that gives you the flatten tags. That way _tags will hide the internal structure and I think the code that uses it can be cleaner since will not deal with flattening the list, etc...

What you think?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oooo I like that! Let me see if I cam implement it :D

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

done


if payload_data:
data = dict_merge(data, payload_data, silence_errors=True)

Expand Down Expand Up @@ -1606,3 +1658,62 @@ def _wsgi_extract_user_ip(environ):
if real_ip:
return real_ip
return environ['REMOTE_ADDR']


def _create_tag(key, value):
return {'key': key, 'value': value}


def _feature_flag_data_key(flag_key, attribute):
return 'feature_flag.data.%s.%s' % (flag_key, attribute)


class _LocalTags(object):
"""
An object to ensure thread safety.
"""
def __init__(self):
self._registry = threading.local()
self._registry.tags = []

def append(self, value):
self._registry.tags.append(value)

def pop(self):
self._registry.tags.pop()

@property
def value(self):
if hasattr(self._registry, 'tags'):
return self._registry.tags

return []
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why or when would be the case where _LocalTags is initialized, but does not have a tags attribute in self._registry?

removing this if condition, fails this test https://github.com/rollbar/pyrollbar/blob/master/rollbar/test/test_rollbar.py#L289

with this:

test_report_exception_with_same_exception_as_cause (rollbar.test.test_rollbar.RollbarTest) ... ERROR:rollbar:Exception while reporting exc_info to Rollbar. AttributeError("'thread._local' object has no attribute 'tags'",)
Traceback (most recent call last):
  File "/Users/ajtran/Development/tmp/test-launchdarkly/pyrollbar/rollbar/__init__.py", line 439, in report_exc_info
    return _report_exc_info(exc_info, request, extra_data, payload_data, level=level)
  File "/Users/ajtran/Development/tmp/test-launchdarkly/pyrollbar/rollbar/__init__.py", line 722, in _report_exc_info
    tags = _tags.value + getattr(exc_info[1], '_rollbar_tags', [])[::-1]
  File "/Users/ajtran/Development/tmp/test-launchdarkly/pyrollbar/rollbar/__init__.py", line 1652, in value
    return self._registry.tags
AttributeError: 'thread._local' object has no attribute 'tags'

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you should try to get the root exception that is triggering that report and see at which moment that's happening

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah. Ok ok. I think I see what is happening right now. For this test test_report_exception_with_same_exception_as_cause, we are running the report_exc_info inside a threading.Thread and when we do that, tags is not set in that thread._local, thus this runtime error when trying to call report_exc_info. I think I might know a way around this.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've resolved this by initializing self._registry.tags in the append and value method of _LocalTags.



_tags = _LocalTags()


class _TagManager(object):
"""
Context manager object that interfaces with the `_tags` stack:

On enter, puts the tag object at top of the stack.
On exit, pops off the top element of the stack.
- If there is an exception, attach the tag object to the exception
for rebuilding of the `_tags` stack before reporting.
"""
def __init__(self, tag):
self.tag = tag

def __enter__(self):
_tags.append(self.tag)

def __exit__(self, exc_type, exc_value, traceback):

if exc_value:
if not hasattr(exc_value, '_rollbar_tags'):
exc_value._rollbar_tags = []

exc_value._rollbar_tags.append(self.tag)

_tags.pop()