test_views.py 9.99 KB
Newer Older
1 2
# pylint: disable=missing-docstring,maybe-no-member

David Baumgold committed
3
from mock import patch, sentinel
4

5
from django.contrib.auth.models import User
6
from django.test.client import RequestFactory
7
from django.test.utils import override_settings
8

9 10 11 12
from track import views
from track.middleware import TrackMiddleware
from track.tests import EventTrackingTestCase, FROZEN_TIME
from openedx.core.lib.tests.assertions.events import assert_event_matches
13

14

15
class TestTrackViews(EventTrackingTestCase):
16 17

    def setUp(self):
18 19
        super(TestTrackViews, self).setUp()

20 21
        self.request_factory = RequestFactory()

22
        patcher = patch('track.views.tracker', autospec=True)
23 24 25 26 27 28 29 30 31 32 33 34 35 36
        self.mock_tracker = patcher.start()
        self.addCleanup(patcher.stop)

        self.path_with_course = '/courses/foo/bar/baz/xmod/'
        self.url_with_course = 'http://www.edx.org' + self.path_with_course

        self.event = {
            sentinel.key: sentinel.value
        }

    def test_user_track(self):
        request = self.request_factory.get('/event', {
            'page': self.url_with_course,
            'event_type': sentinel.event_type,
37
            'event': '{}'
38 39
        })

40 41 42
        views.user_track(request)

        actual_event = self.get_event()
43 44 45 46
        expected_event = {
            'context': {
                'course_id': 'foo/bar/baz',
                'org_id': 'foo',
47 48 49
                'event_source': 'browser',
                'page': self.url_with_course,
                'username': 'anonymous'
50
            },
51 52 53
            'data': {},
            'timestamp': FROZEN_TIME,
            'name': str(sentinel.event_type)
54
        }
55
        assert_event_matches(expected_event, actual_event)
56

57 58 59
    def test_user_track_with_missing_values(self):
        request = self.request_factory.get('/event')

60 61 62
        views.user_track(request)

        actual_event = self.get_event()
63 64 65 66
        expected_event = {
            'context': {
                'course_id': '',
                'org_id': '',
67 68 69
                'event_source': 'browser',
                'page': '',
                'username': 'anonymous'
70
            },
71 72 73
            'data': {},
            'timestamp': FROZEN_TIME,
            'name': 'unknown'
74
        }
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
        assert_event_matches(expected_event, actual_event)

        views.user_track(request)

    def test_user_track_with_empty_event(self):
        request = self.request_factory.get('/event', {
            'page': self.url_with_course,
            'event_type': sentinel.event_type,
            'event': ''
        })

        views.user_track(request)

        actual_event = self.get_event()
        expected_event = {
            'context': {
                'course_id': 'foo/bar/baz',
                'org_id': 'foo',
                'event_source': 'browser',
                'page': self.url_with_course,
                'username': 'anonymous'
            },
            'data': {},
            'timestamp': FROZEN_TIME,
            'name': str(sentinel.event_type)
        }
        assert_event_matches(expected_event, actual_event)

    @override_settings(
        EVENT_TRACKING_PROCESSORS=[{'ENGINE': 'track.shim.LegacyFieldMappingProcessor'}],
    )
    def test_user_track_with_middleware_and_processors(self):
        self.recreate_tracker()
108

109
        middleware = TrackMiddleware()
110 111
        payload = '{"foo": "bar"}'
        user_id = 1
112 113 114
        request = self.request_factory.get('/event', {
            'page': self.url_with_course,
            'event_type': sentinel.event_type,
115
            'event': payload
116
        })
117 118 119 120 121 122
        request.user = User.objects.create(pk=user_id, username=str(sentinel.username))
        request.META['REMOTE_ADDR'] = '10.0.0.1'
        request.META['HTTP_REFERER'] = str(sentinel.referer)
        request.META['HTTP_ACCEPT_LANGUAGE'] = str(sentinel.accept_language)
        request.META['HTTP_USER_AGENT'] = str(sentinel.user_agent)
        request.META['SERVER_NAME'] = 'testserver2'
123 124 125 126 127
        middleware.process_request(request)
        try:
            views.user_track(request)

            expected_event = {
128 129 130
                'accept_language': str(sentinel.accept_language),
                'referer': str(sentinel.referer),
                'username': str(sentinel.username),
131
                'session': '',
132
                'ip': '10.0.0.1',
133 134
                'event_source': 'browser',
                'event_type': str(sentinel.event_type),
135 136 137
                'name': str(sentinel.event_type),
                'event': payload,
                'agent': str(sentinel.user_agent),
138
                'page': self.url_with_course,
139 140
                'time': FROZEN_TIME,
                'host': 'testserver2',
141 142 143
                'context': {
                    'course_id': 'foo/bar/baz',
                    'org_id': 'foo',
144
                    'user_id': user_id,
145 146 147 148 149 150
                    'path': u'/event'
                },
            }
        finally:
            middleware.process_response(request, None)

151 152
        actual_event = self.get_event()
        assert_event_matches(expected_event, actual_event)
153

154 155 156 157 158
    def test_server_track(self):
        request = self.request_factory.get(self.path_with_course)
        views.server_track(request, str(sentinel.event_type), '{}')

        expected_event = {
159 160
            'accept_language': '',
            'referer': '',
161 162 163 164 165 166 167
            'username': 'anonymous',
            'ip': '127.0.0.1',
            'event_source': 'server',
            'event_type': str(sentinel.event_type),
            'event': '{}',
            'agent': '',
            'page': None,
168
            'time': FROZEN_TIME,
169 170 171
            'host': 'testserver',
            'context': {},
        }
172 173 174 175 176 177
        self.assert_mock_tracker_call_matches(expected_event)

    def assert_mock_tracker_call_matches(self, expected_event):
        self.assertEqual(len(self.mock_tracker.send.mock_calls), 1)
        actual_event = self.mock_tracker.send.mock_calls[0][1][0]
        assert_event_matches(expected_event, actual_event)
178

179 180 181 182 183 184 185 186 187 188
    def test_server_track_with_middleware(self):
        middleware = TrackMiddleware()
        request = self.request_factory.get(self.path_with_course)
        middleware.process_request(request)
        # The middleware emits an event, reset the mock to ignore it since we aren't testing that feature.
        self.mock_tracker.reset_mock()
        try:
            views.server_track(request, str(sentinel.event_type), '{}')

            expected_event = {
189 190
                'accept_language': '',
                'referer': '',
191 192 193 194 195 196 197
                'username': 'anonymous',
                'ip': '127.0.0.1',
                'event_source': 'server',
                'event_type': str(sentinel.event_type),
                'event': '{}',
                'agent': '',
                'page': None,
198
                'time': FROZEN_TIME,
199
                'host': 'testserver',
200 201 202 203 204 205 206 207 208 209
                'context': {
                    'user_id': '',
                    'course_id': u'foo/bar/baz',
                    'org_id': 'foo',
                    'path': u'/courses/foo/bar/baz/xmod/'
                },
            }
        finally:
            middleware.process_response(request, None)

210
        self.assert_mock_tracker_call_matches(expected_event)
211 212 213 214 215 216 217 218 219 220 221 222

    def test_server_track_with_middleware_and_google_analytics_cookie(self):
        middleware = TrackMiddleware()
        request = self.request_factory.get(self.path_with_course)
        request.COOKIES['_ga'] = 'GA1.2.1033501218.1368477899'
        middleware.process_request(request)
        # The middleware emits an event, reset the mock to ignore it since we aren't testing that feature.
        self.mock_tracker.reset_mock()
        try:
            views.server_track(request, str(sentinel.event_type), '{}')

            expected_event = {
223 224
                'accept_language': '',
                'referer': '',
225 226 227 228 229 230 231
                'username': 'anonymous',
                'ip': '127.0.0.1',
                'event_source': 'server',
                'event_type': str(sentinel.event_type),
                'event': '{}',
                'agent': '',
                'page': None,
232
                'time': FROZEN_TIME,
233
                'host': 'testserver',
234 235 236 237 238 239 240 241 242 243
                'context': {
                    'user_id': '',
                    'course_id': u'foo/bar/baz',
                    'org_id': 'foo',
                    'path': u'/courses/foo/bar/baz/xmod/'
                },
            }
        finally:
            middleware.process_response(request, None)

244
        self.assert_mock_tracker_call_matches(expected_event)
245

246 247 248 249 250
    def test_server_track_with_no_request(self):
        request = None
        views.server_track(request, str(sentinel.event_type), '{}')

        expected_event = {
251 252
            'accept_language': '',
            'referer': '',
253 254 255 256 257 258 259
            'username': 'anonymous',
            'ip': '',
            'event_source': 'server',
            'event_type': str(sentinel.event_type),
            'event': '{}',
            'agent': '',
            'page': None,
260
            'time': FROZEN_TIME,
261 262 263
            'host': '',
            'context': {},
        }
264
        self.assert_mock_tracker_call_matches(expected_event)
265 266 267

    def test_task_track(self):
        request_info = {
268 269
            'accept_language': '',
            'referer': '',
270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291
            'username': 'anonymous',
            'ip': '127.0.0.1',
            'agent': 'agent',
            'host': 'testserver',
        }

        task_info = {
            sentinel.task_key: sentinel.task_value
        }
        expected_event_data = dict(task_info)
        expected_event_data.update(self.event)

        views.task_track(request_info, task_info, str(sentinel.event_type), self.event)

        expected_event = {
            'username': 'anonymous',
            'ip': '127.0.0.1',
            'event_source': 'task',
            'event_type': str(sentinel.event_type),
            'event': expected_event_data,
            'agent': 'agent',
            'page': None,
292
            'time': FROZEN_TIME,
293 294 295 296 297 298
            'host': 'testserver',
            'context': {
                'course_id': '',
                'org_id': ''
            },
        }
299
        self.assert_mock_tracker_call_matches(expected_event)