7

Compare commits

...

39 Commits

Author SHA1 Message Date
02e06bd9f3 Merge branch 'release/0.11.1' 2018-03-01 16:26:35 +02:00
c7ad3251cf Version 0.11.1 2018-03-01 16:26:20 +02:00
cb8b7da496 Merge pull request #23 from praekeltfoundation/feature/populate-static-segments-from-db-at-creation
Populate static segments from database
2018-03-01 16:23:10 +02:00
0efd3ae937 Update tests for new static segment population 2018-02-26 14:34:02 +02:00
d335e4fd7b Populate static segments even if the count is 0 2018-02-26 14:31:56 +02:00
db2f82967e Only loop through users once when saving static segments
When saving a new static segment we count the matching users and
populate the segment from the database. Each of these requires us
to loop through all of the users, so it's better to do them at the
same time.
2018-02-25 15:43:12 +02:00
37243365a7 Merge branch 'develop' into feature/populate-static-segments-from-db-at-creation 2018-02-25 15:08:36 +02:00
43a2b590b4 Merge tag '0.11.0' into develop
Bug Fix: Query rule should not be static
Enable retrieval of user data for static rules through csv download
2018-02-23 17:02:53 +02:00
cc1dd337bb Merge branch 'release/0.11.0' 2018-02-23 17:02:34 +02:00
a677846ff7 Bump version to 0.11.0 2018-02-23 17:02:17 +02:00
7d7861b862 Merge pull request #22 from praekeltfoundation/feature/SAS-72-download-cvs-of-user-data
Download CSV of users in a segment
2018-02-23 16:50:52 +02:00
8e854d0abe Merge branch 'develop' into feature/SAS-72-download-cvs-of-user-data 2018-02-23 14:05:31 +02:00
0051061d96 Merge pull request #21 from praekeltfoundation/feature/SAS-72-get-user-info-for-visit-count-rule
Get user info from visit count rule
2018-02-23 14:02:15 +02:00
f898dfe017 Actually add tests for segment users view 2018-02-22 16:12:24 +02:00
8ced5bd81c Fix flake8 2018-02-22 15:15:01 +02:00
9a86b0c8cc Add tests for segment users view 2018-02-22 15:14:34 +02:00
9408f90789 Use mock for testing matching user count
The fake class was causing other tests to fail because it inherits from AbstractBaseRule but isn't in the database.
I removed it and replaced it with mocked calls
2018-02-22 14:23:14 +02:00
ba6056e3f8 Link to download CSV of users in segment 2018-02-22 14:21:43 +02:00
fdc0a7f2e1 Get user info for Visit count rule 2018-02-21 19:08:29 +02:00
12b0cd9231 Make visit count session retrieval seperate method 2018-02-21 19:07:35 +02:00
330557be8d Make VisitCountRule.test_user actually test with only a user 2018-02-21 18:48:44 +02:00
aa917dee9c Merge pull request #20 from praekeltfoundation/feature/SAS-90-revert-query-rule-to-not-be-static
Remove static flag for Query Rule
2018-02-21 15:54:36 +02:00
364cb1a7e6 Query rule should not be static 2018-02-20 15:08:12 +02:00
8f789b3e17 Fill static segments with users from the database at creation 2018-02-15 13:20:48 +02:00
bedbe06c65 Merge branch 'release/0.10.9' into develop 2018-02-13 14:17:55 +02:00
362f15e5ff version bump 0.10.9 2018-02-13 14:17:29 +02:00
8a0dba2efb Merge pull request #18 from praekeltfoundation/feature/issue-18-Display-the-number-of-users-in-a-static-segment-SAS-99
Display the number of users in a static segment SAS-99
2018-02-13 14:11:56 +02:00
59f4877e04 add localize filter 2018-02-13 13:57:00 +02:00
2ff29cc375 get the number of users in a static segment from static_users variable 2018-02-13 13:47:45 +02:00
8527e6ff23 Merge tag '0.10.8' into develop
Don't add users to exclude list for dynamic segments
Store segments a user is excluded from in the session
2018-02-13 10:13:24 +02:00
d7c07cb238 Merge branch 'release/0.10.8' 2018-02-13 10:12:45 +02:00
6e83366df6 Bump version to 0.10.8 2018-02-13 10:12:35 +02:00
55364f8906 Merge pull request #16 from praekeltfoundation/feature/SAS-78-fix-sampling-dynamic-segments
Fix sampling for Dynamic segments
2018-02-13 09:59:28 +02:00
4fd0b30c66 Check rules test skipped if segment excluded by session 2018-02-12 18:56:13 +02:00
c909852b08 Add tests 2018-02-12 18:01:01 +02:00
ea1ecc2a98 Get excluded segments from session and don't check them again 2018-02-12 18:00:38 +02:00
0f0aecf673 Store excluded segments in the session object 2018-02-12 17:57:36 +02:00
c11960f921 Only store excluded users for static segments 2018-02-12 16:58:20 +02:00
37d49dcdfb Merge tag '0.10.7' into develop
Bug Fix: Ensure static segment members are show the survey immediately
Records users excluded by randomisation on the segment
Don't re-check excluded users
2018-02-09 17:01:02 +02:00
14 changed files with 351 additions and 195 deletions

18
CHANGES
View File

@ -1,3 +1,21 @@
0.11.1
==================
- Populate entirely static segments from registered Users not active Sessions
0.11.0
==================
- Bug Fix: Query rule should not be static
- Enable retrieval of user data for static rules through csv download
0.10.9
==================
- Bug Fix: Display the number of users in a static segment on dashboard
0.10.8
==================
- Don't add users to exclude list for dynamic segments
- Store segments a user is excluded from in the session
0.10.7 0.10.7
================== ==================
- Bug Fix: Ensure static segment members are show the survey immediately - Bug Fix: Ensure static segment members are show the survey immediately

View File

@ -55,10 +55,10 @@ author = 'Lab Digital BV'
# built documents. # built documents.
# #
# The short X.Y version. # The short X.Y version.
version = '0.10.7' version = '0.11.1'
# The full version, including alpha/beta/rc tags. # The full version, including alpha/beta/rc tags.
release = '0.10.7' release = '0.11.1'
# The language for content autogenerated by Sphinx. Refer to documentation # The language for content autogenerated by Sphinx. Refer to documentation
# for a list of supported languages. # for a list of supported languages.

View File

@ -1,5 +1,5 @@
[bumpversion] [bumpversion]
current_version = 0.10.7 current_version = 0.11.1
commit = true commit = true
tag = true tag = true
tag_name = {new_version} tag_name = {new_version}

View File

@ -32,7 +32,7 @@ with open('README.rst') as fh:
setup( setup(
name='wagtail-personalisation-molo', name='wagtail-personalisation-molo',
version='0.10.7', version='0.11.1',
description='A forked version of Wagtail add-on for showing personalized content', description='A forked version of Wagtail add-on for showing personalized content',
author='Praekelt.org', author='Praekelt.org',
author_email='dev@praekeltfoundation.org', author_email='dev@praekeltfoundation.org',

View File

@ -66,17 +66,21 @@ class SessionSegmentsAdapter(BaseSegmentsAdapter):
self.request.session.setdefault('segments', []) self.request.session.setdefault('segments', [])
self._segment_cache = None self._segment_cache = None
def get_segments(self): def get_segments(self, key="segments"):
"""Return the persistent segments stored in the request session. """Return the persistent segments stored in the request session.
:param key: The key under which the segments are stored
:type key: String
:returns: The segments in the request session :returns: The segments in the request session
:rtype: list of wagtail_personalisation.models.Segment or empty list :rtype: list of wagtail_personalisation.models.Segment or empty list
""" """
if self._segment_cache is not None: if key == "segments" and self._segment_cache is not None:
return self._segment_cache return self._segment_cache
raw_segments = self.request.session['segments'] if key not in self.request.session:
return []
raw_segments = self.request.session[key]
segment_ids = [segment['id'] for segment in raw_segments] segment_ids = [segment['id'] for segment in raw_segments]
segments = ( segments = (
@ -86,14 +90,17 @@ class SessionSegmentsAdapter(BaseSegmentsAdapter):
.in_bulk(segment_ids)) .in_bulk(segment_ids))
retval = [segments[pk] for pk in segment_ids if pk in segments] retval = [segments[pk] for pk in segment_ids if pk in segments]
if key == "segments":
self._segment_cache = retval self._segment_cache = retval
return retval return retval
def set_segments(self, segments): def set_segments(self, segments, key="segments"):
"""Set the currently active segments """Set the currently active segments
:param segments: The segments to set for the current request :param segments: The segments to set for the current request
:type segments: list of wagtail_personalisation.models.Segment :type segments: list of wagtail_personalisation.models.Segment
:param key: The key under which to store the segments. Optional
:type key: String
""" """
cache_segments = [] cache_segments = []
@ -108,7 +115,8 @@ class SessionSegmentsAdapter(BaseSegmentsAdapter):
serialized_segments.append(serialized) serialized_segments.append(serialized)
segment_ids.add(segment.pk) segment_ids.add(segment.pk)
self.request.session['segments'] = serialized_segments self.request.session[key] = serialized_segments
if key == "segments":
self._segment_cache = cache_segments self._segment_cache = cache_segments
def get_segment_by_id(self, segment_id): def get_segment_by_id(self, segment_id):
@ -171,13 +179,15 @@ class SessionSegmentsAdapter(BaseSegmentsAdapter):
rule_models = AbstractBaseRule.get_descendant_models() rule_models = AbstractBaseRule.get_descendant_models()
current_segments = self.get_segments() current_segments = self.get_segments()
excluded_segments = self.get_segments("excluded_segments")
# Run tests on all remaining enabled segments to verify applicability. # Run tests on all remaining enabled segments to verify applicability.
additional_segments = [] additional_segments = []
for segment in enabled_segments: for segment in enabled_segments:
if segment.is_static and segment.static_users.filter(id=self.request.user.id).exists(): if segment.is_static and segment.static_users.filter(id=self.request.user.id).exists():
additional_segments.append(segment) additional_segments.append(segment)
elif segment.excluded_users.filter(id=self.request.user.id).exists(): elif (segment.excluded_users.filter(id=self.request.user.id).exists() or
segment in excluded_segments):
continue continue
elif not segment.is_static or not segment.is_full: elif not segment.is_static or not segment.is_full:
segment_rules = [] segment_rules = []
@ -193,10 +203,13 @@ class SessionSegmentsAdapter(BaseSegmentsAdapter):
segment.static_users.add(self.request.user) segment.static_users.add(self.request.user)
additional_segments.append(segment) additional_segments.append(segment)
elif result: elif result:
if self.request.user.is_authenticated(): if segment.is_static and self.request.user.is_authenticated():
segment.excluded_users.add(self.request.user) segment.excluded_users.add(self.request.user)
else:
excluded_segments += [segment]
self.set_segments(current_segments + additional_segments) self.set_segments(current_segments + additional_segments)
self.set_segments(excluded_segments, "excluded_segments")
self.update_visit_count() self.update_visit_count()

View File

@ -13,4 +13,6 @@ urlpatterns = [
views.copy_page_view, name='copy_page'), views.copy_page_view, name='copy_page'),
url(r'^segment/toggle_segment_view/$', url(r'^segment/toggle_segment_view/$',
views.toggle_segment_view, name='toggle_segment_view'), views.toggle_segment_view, name='toggle_segment_view'),
url(r'^segment/users/(?P<segment_id>[0-9]+)$',
views.segment_user_data, name='segment_user_data'),
] ]

View File

@ -2,12 +2,10 @@ from __future__ import absolute_import, unicode_literals
from datetime import datetime from datetime import datetime
from importlib import import_module from importlib import import_module
from itertools import takewhile
from django.conf import settings from django.conf import settings
from django.contrib.auth import get_user_model from django.contrib.auth import get_user_model
from django.contrib.auth.models import AnonymousUser from django.contrib.auth.models import AnonymousUser
from django.contrib.sessions.models import Session
from django.contrib.staticfiles.templatetags.staticfiles import static from django.contrib.staticfiles.templatetags.staticfiles import static
from django.test.client import RequestFactory from django.test.client import RequestFactory
from django.utils.lru_cache import lru_cache from django.utils.lru_cache import lru_cache
@ -87,7 +85,7 @@ class SegmentAdminForm(WagtailAdminModelForm):
if not self.instance.is_static: if not self.instance.is_static:
self.instance.count = 0 self.instance.count = 0
if is_new: if is_new and self.instance.is_static and not self.instance.all_rules_static:
rules = [ rules = [
form.instance for formset in self.formsets.values() form.instance for formset in self.formsets.values()
for form in formset for form in formset
@ -108,23 +106,24 @@ class SegmentAdminForm(WagtailAdminModelForm):
users_to_add = [] users_to_add = []
users_to_exclude = [] users_to_exclude = []
sessions = Session.objects.iterator()
take_session = takewhile( User = get_user_model()
lambda x: instance.count == 0 or len(users_to_add) <= instance.count, users = User.objects.filter(is_active=True, is_staff=False)
sessions
) matched_count = 0
for session in take_session: for user in users.iterator():
session_data = session.get_decoded()
user = user_from_data(session_data.get('_auth_user_id'))
if user.is_authenticated():
request.user = user request.user = user
request.session = SessionStore(session_key=session.session_key)
passes = adapter._test_rules(instance.get_rules(), request, instance.match_any) passes = adapter._test_rules(instance.get_rules(), request, instance.match_any)
if passes and instance.randomise_into_segment(): if passes:
matched_count += 1
if instance.count == 0 or len(users_to_add) <= instance.count:
if instance.randomise_into_segment():
users_to_add.append(user) users_to_add.append(user)
elif passes: else:
users_to_exclude.append(user) users_to_exclude.append(user)
instance.matched_users_count = matched_count
instance.matched_count_updated_at = datetime.now()
instance.static_users.add(*users_to_add) instance.static_users.add(*users_to_add)
instance.excluded_users.add(*users_to_exclude) instance.excluded_users.add(*users_to_exclude)

View File

@ -2,17 +2,23 @@ from __future__ import absolute_import, unicode_literals
import re import re
from datetime import datetime from datetime import datetime
from importlib import import_module
from django.apps import apps from django.apps import apps
from django.conf import settings
from django.contrib.sessions.models import Session
from django.db import models from django.db import models
from django.template.defaultfilters import slugify from django.template.defaultfilters import slugify
from django.utils.encoding import force_text, python_2_unicode_compatible from django.utils.encoding import force_text, python_2_unicode_compatible
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from django.test.client import RequestFactory
from modelcluster.fields import ParentalKey from modelcluster.fields import ParentalKey
from user_agents import parse from user_agents import parse
from wagtail.wagtailadmin.edit_handlers import ( from wagtail.wagtailadmin.edit_handlers import (
FieldPanel, FieldRowPanel, PageChooserPanel) FieldPanel, FieldRowPanel, PageChooserPanel)
SessionStore = import_module(settings.SESSION_ENGINE).SessionStore
@python_2_unicode_compatible @python_2_unicode_compatible
class AbstractBaseRule(models.Model): class AbstractBaseRule(models.Model):
@ -220,18 +226,37 @@ class VisitCountRule(AbstractBaseRule):
class Meta: class Meta:
verbose_name = _('Visit count Rule') verbose_name = _('Visit count Rule')
def _get_user_session(self, user):
sessions = Session.objects.iterator()
for session in sessions:
session_data = session.get_decoded()
if session_data.get('_auth_user_id') == str(user.id):
return SessionStore(session_key=session.session_key)
return SessionStore()
def test_user(self, request, user=None): def test_user(self, request, user=None):
# Local import for cyclic import
from wagtail_personalisation.adapters import (
get_segment_adapter, SessionSegmentsAdapter, SEGMENT_ADAPTER_CLASS)
if user: if user:
# This rule currently does not support testing a user directly # Create a fake request so we can use the adapter
# TODO: Make this test a user directly when the rule uses request = RequestFactory().get('/')
# historical data request.user = user
# If we're using the session adapter check for an active session
if SEGMENT_ADAPTER_CLASS == SessionSegmentsAdapter:
request.session = self._get_user_session(user)
else:
request.session = SessionStore()
elif not request:
# Return false if we don't have a user or a request
return False return False
operator = self.operator operator = self.operator
segment_count = self.count segment_count = self.count
# Local import for cyclic import
from wagtail_personalisation.adapters import get_segment_adapter
adapter = get_segment_adapter(request) adapter = get_segment_adapter(request)
visit_count = adapter.get_visit_count(self.counted_page) visit_count = adapter.get_visit_count(self.counted_page)
@ -257,6 +282,28 @@ class VisitCountRule(AbstractBaseRule):
), ),
} }
def get_column_header(self):
return "Visit count - %s" % self.counted_page
def get_user_info_string(self, user):
# Local import for cyclic import
from wagtail_personalisation.adapters import (
get_segment_adapter, SessionSegmentsAdapter, SEGMENT_ADAPTER_CLASS)
# Create a fake request so we can use the adapter
request = RequestFactory().get('/')
request.user = user
# If we're using the session adapter check for an active session
if SEGMENT_ADAPTER_CLASS == SessionSegmentsAdapter:
request.session = self._get_user_session(user)
else:
request.session = SessionStore()
adapter = get_segment_adapter(request)
visit_count = adapter.get_visit_count(self.counted_page)
return str(visit_count)
class QueryRule(AbstractBaseRule): class QueryRule(AbstractBaseRule):
"""Query rule to segment users based on matching queries. """Query rule to segment users based on matching queries.
@ -266,7 +313,6 @@ class QueryRule(AbstractBaseRule):
""" """
icon = 'fa-link' icon = 'fa-link'
static = True
parameter = models.SlugField(_("The query parameter to search for"), parameter = models.SlugField(_("The query parameter to search for"),
max_length=20) max_length=20)
@ -281,13 +327,7 @@ class QueryRule(AbstractBaseRule):
class Meta: class Meta:
verbose_name = _('Query Rule') verbose_name = _('Query Rule')
def test_user(self, request, user=None): def test_user(self, request):
if user:
# This rule currently does not support testing a user directly
# TODO: Make this test a user directly if/when the rule uses
# historical data
return False
return request.GET.get(self.parameter, '') == self.value return request.GET.get(self.parameter, '') == self.value
def description(self): def description(self):

View File

@ -38,11 +38,11 @@
<li class="stat_card"> <li class="stat_card">
{% trans "This segment is Static" %} {% trans "This segment is Static" %}
<span class="icon icon-fa-user"> <span class="icon icon-fa-user">
{{ segment.sessions.count|localize }} {{ segment.static_users.count|localize }}
{% if segment.sessions.count < segment.count %} {% if segment.static_users.count < segment.count %}
/ {{ segment.count }} {% trans "member" %}{{ segment.count|pluralize }} / {{ segment.count }} {% trans "member" %}{{ segment.count|pluralize }}
{% else %} {% else %}
{% trans "member" %}{{ segment.sessions.count|pluralize }} {% trans "member" %}{{ segment.count|pluralize }}
{% endif %} {% endif %}
</span> </span>
</li> </li>
@ -103,6 +103,9 @@
<li><a href="{% url 'segment:toggle' segment.pk %}" title="{% trans "Disable this segment" %}">disable</a></li> <li><a href="{% url 'segment:toggle' segment.pk %}" title="{% trans "Disable this segment" %}">disable</a></li>
{% endif %} {% endif %}
<li><a href="{% url 'wagtail_personalisation_segment_modeladmin_edit' segment.pk %}" title="{% trans "Configure this segment" %}">configure this</a></li> <li><a href="{% url 'wagtail_personalisation_segment_modeladmin_edit' segment.pk %}" title="{% trans "Configure this segment" %}">configure this</a></li>
{% if segment.is_static %}
<li><a href="{% url 'segment:segment_user_data' segment.pk %}" title="{% trans "Download user info" %}">download users csv</a></li>
{% endif %}
</ul> </ul>
{% endif %} {% endif %}
</div> </div>

View File

@ -1,8 +1,9 @@
from __future__ import absolute_import, unicode_literals from __future__ import absolute_import, unicode_literals
import csv
from django import forms from django import forms
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.http import HttpResponseForbidden, HttpResponseRedirect from django.http import HttpResponse, HttpResponseForbidden, HttpResponseRedirect
from django.shortcuts import get_object_or_404 from django.shortcuts import get_object_or_404
from django.utils.translation import ugettext_lazy as _ from django.utils.translation import ugettext_lazy as _
from wagtail.contrib.modeladmin.options import ModelAdmin, modeladmin_register from wagtail.contrib.modeladmin.options import ModelAdmin, modeladmin_register
@ -139,3 +140,32 @@ def copy_page_view(request, page_id, segment_id):
return HttpResponseRedirect(edit_url) return HttpResponseRedirect(edit_url)
return HttpResponseForbidden() return HttpResponseForbidden()
# CSV download views
def segment_user_data(request, segment_id):
if request.user.has_perm('wagtailadmin.access_admin'):
segment = get_object_or_404(Segment, pk=segment_id)
response = HttpResponse(content_type='text/csv; charset=utf-8')
response['Content-Disposition'] = \
'attachment;filename=segment-%s-users.csv' % str(segment_id)
headers = ['Username']
for rule in segment.get_rules():
if rule.static:
headers.append(rule.get_column_header())
writer = csv.writer(response)
writer.writerow(headers)
for user in segment.static_users.all():
row = [user.username]
for rule in segment.get_rules():
if rule.static:
row.append(rule.get_user_info_string(user))
writer.writerow(row)
return response
return HttpResponseForbidden()

View File

@ -45,4 +45,3 @@ def test_query_rule_create():
assert query_rule.parameter == 'query' assert query_rule.parameter == 'query'
assert query_rule.value == 'value' assert query_rule.value == 'value'
assert query_rule.static

View File

@ -1,5 +1,8 @@
import pytest import pytest
from tests.factories.rule import VisitCountRuleFactory
from tests.factories.segment import SegmentFactory
@pytest.mark.django_db @pytest.mark.django_db
def test_visit_count(site, client): def test_visit_count(site, client):
@ -20,3 +23,50 @@ def test_visit_count(site, client):
visit_count = client.session['visit_count'] visit_count = client.session['visit_count']
assert visit_count[0]['count'] == 2 assert visit_count[0]['count'] == 2
assert visit_count[1]['count'] == 1 assert visit_count[1]['count'] == 1
@pytest.mark.django_db
def test_visit_count_call_test_user_with_user(site, client, user):
segment = SegmentFactory(name='VisitCount')
rule = VisitCountRuleFactory(counted_page=site.root_page, segment=segment)
session = client.session
session['visit_count'] = [{'path': '/', 'count': 2}]
session.save()
client.force_login(user)
assert rule.test_user(None, user)
@pytest.mark.django_db
def test_visit_count_call_test_user_with_user_or_request_fails(site, client, user):
segment = SegmentFactory(name='VisitCount')
rule = VisitCountRuleFactory(counted_page=site.root_page, segment=segment)
session = client.session
session['visit_count'] = [{'path': '/', 'count': 2}]
session.save()
client.force_login(user)
assert not rule.test_user(None)
@pytest.mark.django_db
def test_get_column_header(site):
segment = SegmentFactory(name='VisitCount')
rule = VisitCountRuleFactory(counted_page=site.root_page, segment=segment)
assert rule.get_column_header() == 'Visit count - Test page'
@pytest.mark.django_db
def test_get_user_info_string_returns_count(site, client, user):
segment = SegmentFactory(name='VisitCount')
rule = VisitCountRuleFactory(counted_page=site.root_page, segment=segment)
session = client.session
session['visit_count'] = [{'path': '/', 'count': 2}]
session.save()
client.force_login(user)
assert rule.get_user_info_string(user) == '2'

View File

@ -8,8 +8,7 @@ from django.forms.models import model_to_dict
from tests.factories.segment import SegmentFactory from tests.factories.segment import SegmentFactory
from wagtail_personalisation.forms import SegmentAdminForm from wagtail_personalisation.forms import SegmentAdminForm
from wagtail_personalisation.models import Segment from wagtail_personalisation.models import Segment
from wagtail_personalisation.rules import (AbstractBaseRule, TimeRule, from wagtail_personalisation.rules import TimeRule, VisitCountRule
VisitCountRule)
def form_with_data(segment, *rules): def form_with_data(segment, *rules):
@ -36,22 +35,18 @@ def form_with_data(segment, *rules):
@pytest.mark.django_db @pytest.mark.django_db
def test_session_added_to_static_segment_at_creation(site, client, user): def test_user_added_to_static_segment_at_creation(site, user, mocker):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = VisitCountRule(counted_page=site.root_page) rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
instance = form.save() instance = form.save()
assert user in instance.static_users.all() assert user in instance.static_users.all()
@pytest.mark.django_db @pytest.mark.django_db
def test_anonymous_user_not_added_to_static_segment_at_creation(site, client): def test_anonymous_user_not_added_to_static_segment_at_creation(site, client, mocker):
session = client.session session = client.session
session.save() session.save()
client.get(site.root_page.url) client.get(site.root_page.url)
@ -59,43 +54,32 @@ def test_anonymous_user_not_added_to_static_segment_at_creation(site, client):
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = VisitCountRule(counted_page=site.root_page) rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mock_test_rule = mocker.patch('wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules')
instance = form.save() instance = form.save()
assert not instance.static_users.all() assert not instance.static_users.all()
assert mock_test_rule.call_count == 0
@pytest.mark.django_db @pytest.mark.django_db
def test_match_any_correct_populates(site, client, django_user_model): def test_match_any_correct_populates(site, django_user_model, mocker):
user = django_user_model.objects.create(username='first') user = django_user_model.objects.create(username='first')
session = client.session
client.force_login(user)
client.get(site.root_page.url)
other_user = django_user_model.objects.create(username='second') other_user = django_user_model.objects.create(username='second')
client.cookies.clear()
second_session = client.session
other_page = site.root_page.get_last_child() other_page = site.root_page.get_last_child()
client.force_login(other_user)
client.get(other_page.url)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, match_any=True) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, match_any=True)
rule_1 = VisitCountRule(counted_page=site.root_page) rule_1 = VisitCountRule(counted_page=site.root_page)
rule_2 = VisitCountRule(counted_page=other_page) rule_2 = VisitCountRule(counted_page=other_page)
form = form_with_data(segment, rule_1, rule_2) form = form_with_data(segment, rule_1, rule_2)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', side_effect=[True, False, True, False])
instance = form.save() instance = form.save()
assert session.session_key != second_session.session_key
assert user in instance.static_users.all() assert user in instance.static_users.all()
assert other_user in instance.static_users.all() assert other_user in instance.static_users.all()
@pytest.mark.django_db @pytest.mark.django_db
def test_mixed_static_dynamic_session_doesnt_generate_at_creation(site, client, user): def test_mixed_static_dynamic_session_doesnt_generate_at_creation(site, mocker):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=1) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=1)
static_rule = VisitCountRule(counted_page=site.root_page) static_rule = VisitCountRule(counted_page=site.root_page)
non_static_rule = TimeRule( non_static_rule = TimeRule(
@ -103,9 +87,12 @@ def test_mixed_static_dynamic_session_doesnt_generate_at_creation(site, client,
end_time=datetime.time(23, 59, 59), end_time=datetime.time(23, 59, 59),
) )
form = form_with_data(segment, static_rule, non_static_rule) form = form_with_data(segment, static_rule, non_static_rule)
mock_test_rule = mocker.patch('wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules')
instance = form.save() instance = form.save()
assert not instance.static_users.all() assert not instance.static_users.all()
assert mock_test_rule.call_count == 0
@pytest.mark.django_db @pytest.mark.django_db
@ -181,12 +168,7 @@ def test_session_not_added_to_static_segment_after_full(site, client, django_use
@pytest.mark.django_db @pytest.mark.django_db
def test_sessions_not_added_to_static_segment_if_rule_not_static(client, site, user): def test_sessions_not_added_to_static_segment_if_rule_not_static(mocker):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=1) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=1)
rule = TimeRule( rule = TimeRule(
start_time=datetime.time(0, 0, 0), start_time=datetime.time(0, 0, 0),
@ -194,26 +176,27 @@ def test_sessions_not_added_to_static_segment_if_rule_not_static(client, site, u
segment=segment, segment=segment,
) )
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mock_test_rule = mocker.patch('wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules')
instance = form.save() instance = form.save()
assert not instance.static_users.all() assert not instance.static_users.all()
assert mock_test_rule.call_count == 0
@pytest.mark.django_db @pytest.mark.django_db
def test_does_not_calculate_the_segment_again(site, client, mocker, user): def test_does_not_calculate_the_segment_again(site, client, mocker, user):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=2) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, count=2)
rule = VisitCountRule(counted_page=site.root_page, segment=segment) rule = VisitCountRule(counted_page=site.root_page, segment=segment)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
instance = form.save() instance = form.save()
assert user in instance.static_users.all() assert user in instance.static_users.all()
mock_test_rule = mocker.patch('wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules') mock_test_rule = mocker.patch('wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules')
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url) client.get(site.root_page.url)
assert mock_test_rule.call_count == 0 assert mock_test_rule.call_count == 0
@ -332,6 +315,7 @@ def test_offered_dynamic_segment_if_random_is_below_percentage(site, client, moc
session.save() session.save()
client.get(site.root_page.url) client.get(site.root_page.url)
assert len(client.session['excluded_segments']) == 0
assert instance.id == client.session['segments'][0]['id'] assert instance.id == client.session['segments'][0]['id']
@ -341,7 +325,7 @@ def test_not_offered_dynamic_segment_if_random_is_above_percentage(site, client,
randomisation_percent=40) randomisation_percent=40)
rule = VisitCountRule(counted_page=site.root_page) rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
form.save() instance = form.save()
mocker.patch('random.randint', return_value=41) mocker.patch('random.randint', return_value=41)
session = client.session session = client.session
@ -349,6 +333,7 @@ def test_not_offered_dynamic_segment_if_random_is_above_percentage(site, client,
client.get(site.root_page.url) client.get(site.root_page.url)
assert len(client.session['segments']) == 0 assert len(client.session['segments']) == 0
assert instance.id == client.session['excluded_segments'][0]['id']
@pytest.mark.django_db @pytest.mark.django_db
@ -388,16 +373,12 @@ def test_always_in_segment_if_percentage_is_100(site, client, mocker, user):
@pytest.mark.django_db @pytest.mark.django_db
def test_not_added_to_static_segment_at_creation_if_random_above_percent(site, client, mocker, user): def test_not_added_to_static_segment_at_creation_if_random_above_percent(site, mocker, user):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
mocker.patch('random.randint', return_value=41) mocker.patch('random.randint', return_value=41)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, randomisation_percent=40) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, randomisation_percent=40)
rule = VisitCountRule(counted_page=site.root_page) rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
instance = form.save() instance = form.save()
assert user not in instance.static_users.all() assert user not in instance.static_users.all()
@ -405,16 +386,12 @@ def test_not_added_to_static_segment_at_creation_if_random_above_percent(site, c
@pytest.mark.django_db @pytest.mark.django_db
def test_added_to_static_segment_at_creation_if_random_below_percent(site, client, mocker, user): def test_added_to_static_segment_at_creation_if_random_below_percent(site, mocker, user):
session = client.session
session.save()
client.force_login(user)
client.get(site.root_page.url)
mocker.patch('random.randint', return_value=39) mocker.patch('random.randint', return_value=39)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC, randomisation_percent=40) segment = SegmentFactory.build(type=Segment.TYPE_STATIC, randomisation_percent=40)
rule = VisitCountRule(counted_page=site.root_page) rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
instance = form.save() instance = form.save()
assert user in instance.static_users.all() assert user in instance.static_users.all()
@ -446,7 +423,31 @@ def test_rules_check_skipped_if_user_in_excluded(site, client, mocker, user):
@pytest.mark.django_db @pytest.mark.django_db
def test_matched_user_count_added_to_segment_at_creation(site, client, mocker, django_user_model): def test_rules_check_skipped_if_dynamic_segment_in_excluded(site, client, mocker, user):
segment = SegmentFactory.build(type=Segment.TYPE_DYNAMIC,
randomisation_percent=100)
rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule)
instance = form.save()
instance.persistent = True
instance.save()
session = client.session
session['excluded_segments'] = [{'id': instance.pk}]
session.save()
mock_test_rule = mocker.patch(
'wagtail_personalisation.adapters.SessionSegmentsAdapter._test_rules')
client.force_login(user)
client.get(site.root_page.url)
assert mock_test_rule.call_count == 0
assert len(client.session['segments']) == 0
@pytest.mark.django_db
def test_matched_user_count_added_to_segment_at_creation(site, mocker, django_user_model):
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second') django_user_model.objects.create(username='second')
@ -454,6 +455,7 @@ def test_matched_user_count_added_to_segment_at_creation(site, client, mocker, d
rule = VisitCountRule() rule = VisitCountRule()
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
form.instance.type = Segment.TYPE_STATIC
mock_test_user = mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True) mock_test_user = mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
instance = form.save() instance = form.save()
@ -462,152 +464,97 @@ def test_matched_user_count_added_to_segment_at_creation(site, client, mocker, d
@pytest.mark.django_db @pytest.mark.django_db
def test_count_users_matching_static_rules(site, client, django_user_model): def test_count_users_matching_static_rules(site, client, mocker, django_user_model):
class TestStaticRule(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
return True
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second') django_user_model.objects.create(username='second')
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = TestStaticRule() rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
assert form.count_matching_users([rule], True) is 2 assert form.count_matching_users([rule], True) is 2
@pytest.mark.django_db @pytest.mark.django_db
def test_count_matching_users_excludes_staff(site, client, django_user_model): def test_count_matching_users_excludes_staff(site, client, mocker, django_user_model):
class TestStaticRule(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
return True
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second', is_staff=True) django_user_model.objects.create(username='second', is_staff=True)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = TestStaticRule() rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mock_test_user = mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
assert form.count_matching_users([rule], True) is 1 assert form.count_matching_users([rule], True) is 1
assert mock_test_user.call_count == 1
@pytest.mark.django_db @pytest.mark.django_db
def test_count_matching_users_excludes_inactive(site, client, django_user_model): def test_count_matching_users_excludes_inactive(site, client, mocker, django_user_model):
class TestStaticRule(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
return True
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second', is_active=False) django_user_model.objects.create(username='second', is_active=False)
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = TestStaticRule() rule = VisitCountRule(counted_page=site.root_page)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mock_test_user = mocker.patch('wagtail_personalisation.rules.VisitCountRule.test_user', return_value=True)
assert form.count_matching_users([rule], True) is 1 assert form.count_matching_users([rule], True) is 1
assert mock_test_user.call_count == 1
@pytest.mark.django_db @pytest.mark.django_db
def test_count_matching_users_only_counts_static_rules(site, client, django_user_model): def test_count_matching_users_only_counts_static_rules(site, client, mocker, django_user_model):
class TestStaticRule(AbstractBaseRule):
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
return True
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second') django_user_model.objects.create(username='second')
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
rule = TestStaticRule() rule = TimeRule(
start_time=datetime.time(0, 0, 0),
end_time=datetime.time(23, 59, 59),
segment=segment,
)
form = form_with_data(segment, rule) form = form_with_data(segment, rule)
mock_test_user = mocker.patch('wagtail_personalisation.rules.TimeRule.test_user')
assert form.count_matching_users([rule], True) is 0 assert form.count_matching_users([rule], True) is 0
assert mock_test_user.call_count == 0
@pytest.mark.django_db @pytest.mark.django_db
def test_count_matching_users_handles_match_any(site, client, django_user_model): def test_count_matching_users_handles_match_any(site, client, mocker, django_user_model):
class TestStaticRuleFirst(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
if user.username == 'first':
return True
return False
class TestStaticRuleSecond(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
if user.username == 'second':
return True
return False
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second') django_user_model.objects.create(username='second')
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
first_rule = TestStaticRuleFirst() first_rule = VisitCountRule(counted_page=site.root_page)
second_rule = TestStaticRuleSecond() other_page = site.root_page.get_last_child()
second_rule = VisitCountRule(counted_page=other_page)
form = form_with_data(segment, first_rule, second_rule) form = form_with_data(segment, first_rule, second_rule)
mock_test_user = mocker.patch(
'wagtail_personalisation.rules.VisitCountRule.test_user',
side_effect=[True, False, True, False])
assert form.count_matching_users([first_rule, second_rule], True) is 2 assert form.count_matching_users([first_rule, second_rule], True) is 2
mock_test_user.call_count == 4
@pytest.mark.django_db @pytest.mark.django_db
def test_count_matching_users_handles_match_all(site, client, django_user_model): def test_count_matching_users_handles_match_all(site, client, mocker, django_user_model):
class TestStaticRuleFirst(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
if user.username == 'first':
return True
return False
class TestStaticRuleContainsS(AbstractBaseRule):
static = True
class Meta:
app_label = 'wagtail_personalisation'
def test_user(self, request, user):
if 's' in user.username:
return True
return False
django_user_model.objects.create(username='first') django_user_model.objects.create(username='first')
django_user_model.objects.create(username='second') django_user_model.objects.create(username='second')
segment = SegmentFactory.build(type=Segment.TYPE_STATIC) segment = SegmentFactory.build(type=Segment.TYPE_STATIC)
first_rule = TestStaticRuleFirst() first_rule = VisitCountRule(counted_page=site.root_page)
s_rule = TestStaticRuleContainsS() other_page = site.root_page.get_last_child()
form = form_with_data(segment, first_rule, s_rule) second_rule = VisitCountRule(counted_page=other_page)
form = form_with_data(segment, first_rule, second_rule)
assert form.count_matching_users([first_rule, s_rule], False) is 1 mock_test_user = mocker.patch(
'wagtail_personalisation.rules.VisitCountRule.test_user',
side_effect=[True, True, False, True])
assert form.count_matching_users([first_rule, second_rule], False) is 1
mock_test_user.call_count == 4

55
tests/unit/test_views.py Normal file
View File

@ -0,0 +1,55 @@
import pytest
from django.contrib.auth.models import Permission
from django.core.urlresolvers import reverse
from wagtail_personalisation.models import Segment
from wagtail_personalisation.rules import VisitCountRule
@pytest.mark.django_db
def test_segment_user_data_view_requires_admin_access(site, client, django_user_model):
user = django_user_model.objects.create(username='first')
segment = Segment(type=Segment.TYPE_STATIC, count=1)
segment.save()
client.force_login(user)
url = reverse('segment:segment_user_data', args=(segment.id,))
response = client.get(url)
assert response.status_code == 302
assert response.url == '/admin/login/?next=%s' % url
@pytest.mark.django_db
def test_segment_user_data_view(site, client, mocker, django_user_model):
user1 = django_user_model.objects.create(username='first')
user2 = django_user_model.objects.create(username='second')
admin_user = django_user_model.objects.create(username='admin')
permission = Permission.objects.get(codename='access_admin')
admin_user.user_permissions.add(permission)
segment = Segment(type=Segment.TYPE_STATIC, count=1)
segment.save()
segment.static_users.add(user1)
segment.static_users.add(user2)
rule1 = VisitCountRule(counted_page=site.root_page, segment=segment)
rule2 = VisitCountRule(counted_page=site.root_page.get_last_child(),
segment=segment)
rule1.save()
rule2.save()
mocker.patch('wagtail_personalisation.rules.VisitCountRule.get_user_info_string',
side_effect=[3, 9, 0, 1])
client.force_login(admin_user)
response = client.get(
reverse('segment:segment_user_data', args=(segment.id,)))
assert response.status_code == 200
data_lines = response.content.decode().split("\n")
assert data_lines[0] == 'Username,Visit count - Test page,Visit count - Regular page\r'
assert data_lines[1] == 'first,3,9\r'
assert data_lines[2] == 'second,0,1\r'