From c926fb6f2ffb412422b05a0c27a69076c73e3ae3 Mon Sep 17 00:00:00 2001 From: viliambalaz Date: Mon, 23 Aug 2021 20:13:34 +0200 Subject: [PATCH 1/5] #280 Fix inforequests view mine --- .../apps/inforequests/tests/__init__.py | 2 +- .../tests/test_views/skip_test_mine.py | 70 ---------------- .../tests/test_views/test_mine.py | 84 +++++++++++++++++++ .../apps/inforequests/views/__init__.py | 4 + 4 files changed, 89 insertions(+), 71 deletions(-) delete mode 100644 chcemvediet/apps/inforequests/tests/test_views/skip_test_mine.py create mode 100644 chcemvediet/apps/inforequests/tests/test_views/test_mine.py diff --git a/chcemvediet/apps/inforequests/tests/__init__.py b/chcemvediet/apps/inforequests/tests/__init__.py index faabf39d7..f12989ed5 100644 --- a/chcemvediet/apps/inforequests/tests/__init__.py +++ b/chcemvediet/apps/inforequests/tests/__init__.py @@ -21,7 +21,7 @@ class InforequestsTestCaseMixin(ChcemvedietTestCaseMixin): @contextlib.contextmanager - def assertQueriesDuringRender(self, *patterns, **kwargs): + def assertQueriesDuringRender(self, patterns, **kwargs): u""" Use to assert that views prefetch all related models before rendering their templates. Views should prefetch their related models to prevent templates from making database diff --git a/chcemvediet/apps/inforequests/tests/test_views/skip_test_mine.py b/chcemvediet/apps/inforequests/tests/test_views/skip_test_mine.py deleted file mode 100644 index a84160615..000000000 --- a/chcemvediet/apps/inforequests/tests/test_views/skip_test_mine.py +++ /dev/null @@ -1,70 +0,0 @@ -# vim: expandtab -# -*- coding: utf-8 -*- -from django.test import TestCase - -from poleno.utils.test import ViewTestCaseMixin -from poleno.utils.urls import reverse - -from .. import InforequestsTestCaseMixin - - -class MineViewTest(InforequestsTestCaseMixin, ViewTestCaseMixin, TestCase): - u""" - Tests ``mine()`` view registered as "inforequests:mine". - """ - - def test_allowed_http_methods(self): - allowed = [u'HEAD', u'GET'] - self.assert_allowed_http_methods(allowed, reverse(u'inforequests:mine')) - - def test_anonymous_user_is_redirected(self): - self.assert_anonymous_user_is_redirected(reverse(u'inforequests:mine')) - - def test_authenticated_user_gets_inforequest_mine(self): - self._login_user() - response = self.client.get(reverse(u'inforequests:mine')) - self.assertEqual(response.status_code, 200) - self.assertTemplateUsed(response, u'inforequests/mine.html') - - def test_user_gets_only_his_inforequests_and_drafts(self): - drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] - drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] - inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] - inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] - closed1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] - closed2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] - - self._login_user(self.user1) - response = self.client.get(reverse(u'inforequests:mine')) - self.assertEqual(response.status_code, 200) - self.assertItemsEqual(response.context[u'inforequests'], inforequests1) - self.assertItemsEqual(response.context[u'drafts'], drafts1) - self.assertItemsEqual(response.context[u'closed_inforequests'], closed1) - - def test_with_user_with_no_his_inforequests_nor_drafts(self): - drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] - inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] - closed2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] - - self._login_user(self.user1) - response = self.client.get(reverse(u'inforequests:mine')) - self.assertEqual(response.status_code, 200) - self.assertItemsEqual(response.context[u'inforequests'], []) - self.assertItemsEqual(response.context[u'drafts'], []) - self.assertItemsEqual(response.context[u'closed_inforequests'], []) - - def test_related_models_are_prefetched_before_render(self): - drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] - inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] - closed1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] - - # Force view querysets to evaluate before calling render - def pre_mock_render(request, temaplate, context): - list(context[u'inforequests']) - list(context[u'drafts']) - list(context[u'closed_inforequests']) - - self._login_user(self.user1) - with self.assertQueriesDuringRender([], pre_mock_render=pre_mock_render): - response = self.client.get(reverse(u'inforequests:mine')) - self.assertEqual(response.status_code, 200) diff --git a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py new file mode 100644 index 000000000..fd43dc55a --- /dev/null +++ b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py @@ -0,0 +1,84 @@ +# vim: expandtab +# -*- coding: utf-8 -*- +from django.test import TestCase + +from poleno.utils.test import ViewTestCaseMixin +from poleno.utils.urls import reverse + +from .. import InforequestsTestCaseMixin +from ...models import Action + + +class MineViewTest(InforequestsTestCaseMixin, ViewTestCaseMixin, TestCase): + u""" + Tests ``mine()`` view registered as "inforequests:mine". + """ + + def _create_successful_inforequest(self, applicant): + inforequest, _, _ = self._create_inforequest_scenario(applicant, {u'closed': True}, + (u'disclosure', dict(disclosure_level=Action.DISCLOSURE_LEVELS.PARTIAL))) + return inforequest + + + def test_allowed_http_methods(self): + allowed = [u'HEAD', u'GET'] + self.assert_allowed_http_methods(allowed, reverse(u'inforequests:mine')) + + def test_anonymous_user_is_redirected(self): + self.assert_anonymous_user_is_redirected(reverse(u'inforequests:mine')) + + def test_authenticated_user_gets_inforequest_mine(self): + self._login_user(self.user1) + response = self.client.get(reverse(u'inforequests:mine')) + self.assertEqual(response.status_code, 200) + self.assertTemplateUsed(response, u'inforequests/mine/mine.html') + + def test_user_gets_only_his_inforequests_and_drafts(self): + drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] + drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] + pending_inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] + pending_inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] + successful_inforequests1 = [self._create_successful_inforequest(applicant=self.user1) for i in range(3)] + successful_inforequests2 = [self._create_successful_inforequest(applicant=self.user2) for i in range(3)] + unsuccessful_inforequests1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] + unsuccessful_inforequests2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] + + self._login_user(self.user1) + response = self.client.get(reverse(u'inforequests:mine')) + self.assertEqual(response.status_code, 200) + self.assertItemsEqual(response.context[u'pending_inforequests'], pending_inforequests1) + self.assertItemsEqual(response.context[u'drafts'], drafts1) + self.assertItemsEqual(response.context[u'successful_inforequests'], successful_inforequests1) + self.assertItemsEqual(response.context[u'unsuccessful_inforequests'], unsuccessful_inforequests1) + + def test_with_user_with_no_his_inforequests_nor_drafts(self): + drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] + pending_inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] + successful_inforequests2 = [self._create_successful_inforequest(applicant=self.user2) for i in range(3)] + unsuccessful_inforequests2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] + + self._login_user(self.user1) + response = self.client.get(reverse(u'inforequests:mine')) + self.assertEqual(response.status_code, 200) + self.assertItemsEqual(response.context[u'pending_inforequests'], []) + self.assertItemsEqual(response.context[u'drafts'], []) + self.assertItemsEqual(response.context[u'successful_inforequests'], []) + self.assertItemsEqual(response.context[u'unsuccessful_inforequests'], []) + + def test_related_models_are_prefetched_before_render(self): + drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] + pending_inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] + successful_inforequests1 = [self._create_successful_inforequest(applicant=self.user1) for i in range(3)] + unsuccessful_inforequests1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] + + # Force view querysets to evaluate before calling render + def pre_mock_render(request, template, context): + list(context[u'pending_inforequests']) + list(context[u'drafts']) + list(context[u'successful_inforequests']) + list(context[u'unsuccessful_inforequests']) + + self._login_user(self.user1) + with self.assertQueriesDuringRender([], pre_mock_render=pre_mock_render): + response = self.client.get(reverse(u'inforequests:mine')) + self.assertEqual(response.status_code, 200) diff --git a/chcemvediet/apps/inforequests/views/__init__.py b/chcemvediet/apps/inforequests/views/__init__.py index 299bd1d6a..5b0b3dca6 100644 --- a/chcemvediet/apps/inforequests/views/__init__.py +++ b/chcemvediet/apps/inforequests/views/__init__.py @@ -1,6 +1,10 @@ # vim: expandtab # -*- coding: utf-8 -*- +from django.shortcuts import render # To let tests mock this function. + +from poleno.utils.template import render_to_string # To let tests mock this function. + from .inforequest import inforequest_index from .inforequest import inforequest_mine from .inforequest import inforequest_create From 525eea56f48bd186ad97063a89477ccdb1fc53c6 Mon Sep 17 00:00:00 2001 From: viliambalaz Date: Fri, 1 Oct 2021 17:30:04 +0200 Subject: [PATCH 2/5] #280 Set correct path to mocked render and render_to_string --- .../apps/inforequests/tests/__init__.py | 11 ++-- .../tests/test_views/test_mine.py | 56 ++++++++++++++++++- .../apps/inforequests/views/__init__.py | 4 -- 3 files changed, 61 insertions(+), 10 deletions(-) diff --git a/chcemvediet/apps/inforequests/tests/__init__.py b/chcemvediet/apps/inforequests/tests/__init__.py index f12989ed5..a7f303bbf 100644 --- a/chcemvediet/apps/inforequests/tests/__init__.py +++ b/chcemvediet/apps/inforequests/tests/__init__.py @@ -21,7 +21,7 @@ class InforequestsTestCaseMixin(ChcemvedietTestCaseMixin): @contextlib.contextmanager - def assertQueriesDuringRender(self, patterns, **kwargs): + def assertQueriesDuringRender(self, *patterns, **kwargs): u""" Use to assert that views prefetch all related models before rendering their templates. Views should prefetch their related models to prevent templates from making database @@ -39,15 +39,16 @@ def mock_render(*args, **kwargs): return res def mock_render_to_string(*args, **kwargs): if pre_mock_render_to_string: # pragma: no cover - self.pre_mock_render_to_string(*args, **kwargs) + pre_mock_render_to_string(*args, **kwargs) with CaptureQueriesContext(connection) as captured: res = render_to_string(*args, **kwargs) queries.append(captured) return res - with mock.patch(u'chcemvediet.apps.inforequests.views.render', mock_render): - with mock.patch(u'chcemvediet.apps.inforequests.views.render_to_string', mock_render_to_string): - yield + with mock.patch(u'chcemvediet.apps.inforequests.views.inforequest.render', mock_render): + with mock.patch(u'chcemvediet.apps.inforequests.views.shortcuts.render', mock_render): + with mock.patch(u'chcemvediet.apps.inforequests.views.shortcuts.render_to_string', mock_render_to_string): + yield self.assertEqual(len(queries), len(patterns), u'%d renders executed, %d expected' % (len(queries), len(patterns))) for render_queries, render_patterns in zip(queries, patterns): diff --git a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py index fd43dc55a..c1532ae3a 100644 --- a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py +++ b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py @@ -2,6 +2,7 @@ # -*- coding: utf-8 -*- from django.test import TestCase +from poleno.utils.misc import squeeze from poleno.utils.test import ViewTestCaseMixin from poleno.utils.urls import reverse @@ -79,6 +80,59 @@ def pre_mock_render(request, template, context): list(context[u'unsuccessful_inforequests']) self._login_user(self.user1) - with self.assertQueriesDuringRender([], pre_mock_render=pre_mock_render): + + # main/base/single_column.html + patterns = [ + u'FROM "accounts_profile" WHERE "accounts_profile"."user_id" = %s', + squeeze(u""" + FROM "mail_message" + INNER JOIN "inforequests_inforequestemail" ON \( "mail_message"."id" = "inforequests_inforequestemail"."email_id" \) + INNER JOIN "inforequests_inforequest" ON \( "inforequests_inforequestemail"."inforequest_id" = "inforequests_inforequest"."id" \) + WHERE \("inforequests_inforequestemail"."type" = %s + AND "inforequests_inforequest"."applicant_id" = %s + AND "inforequests_inforequest"."closed" = %s\) + """), + u'FROM "invitations_invitationsupply" WHERE "invitations_invitationsupply"."user_id" = %s', + ] + + # pending_inforequests + patterns.extend([ + squeeze(u""" + FROM "inforequests_action" + INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) + WHERE "inforequests_branch"."inforequest_id" = %s + """) for _ in range(4) + ]) + + # successful_inforequests + patterns.extend([ + u'FROM "inforequests_action" WHERE "inforequests_action"."branch_id" = %s', + u'FROM "inforequests_branch" WHERE "inforequests_branch"."inforequest_id" = %s', + squeeze(u""" + FROM "inforequests_action" + INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) + WHERE "inforequests_branch"."inforequest_id" = %s + """), + squeeze(u""" + FROM "inforequests_action" + INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) + WHERE "inforequests_branch"."inforequest_id" = %s + """), + ] * 3) + + # unsuccessful_inforequests + patterns.extend([ + squeeze(u""" + FROM "inforequests_action" + INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) + WHERE "inforequests_branch"."inforequest_id" = %s + """), + squeeze(u""" + FROM "inforequests_action" + INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) + WHERE "inforequests_branch"."inforequest_id" = %s + """), + ] * 3) + with self.assertQueriesDuringRender(patterns, pre_mock_render=pre_mock_render): response = self.client.get(reverse(u'inforequests:mine')) self.assertEqual(response.status_code, 200) diff --git a/chcemvediet/apps/inforequests/views/__init__.py b/chcemvediet/apps/inforequests/views/__init__.py index 5b0b3dca6..299bd1d6a 100644 --- a/chcemvediet/apps/inforequests/views/__init__.py +++ b/chcemvediet/apps/inforequests/views/__init__.py @@ -1,10 +1,6 @@ # vim: expandtab # -*- coding: utf-8 -*- -from django.shortcuts import render # To let tests mock this function. - -from poleno.utils.template import render_to_string # To let tests mock this function. - from .inforequest import inforequest_index from .inforequest import inforequest_mine from .inforequest import inforequest_create From 1e772bfbe9f39aa7ba869e82346157bd8d0f28ee Mon Sep 17 00:00:00 2001 From: viliambalaz Date: Sat, 30 Oct 2021 17:21:17 +0200 Subject: [PATCH 3/5] #280 Add review suggestions --- .../apps/inforequests/tests/__init__.py | 7 -- .../tests/test_views/test_mine.py | 73 ++++++++++++------- chcemvediet/tests/__init__.py | 11 +++ 3 files changed, 56 insertions(+), 35 deletions(-) diff --git a/chcemvediet/apps/inforequests/tests/__init__.py b/chcemvediet/apps/inforequests/tests/__init__.py index a7f303bbf..e7cea54f5 100644 --- a/chcemvediet/apps/inforequests/tests/__init__.py +++ b/chcemvediet/apps/inforequests/tests/__init__.py @@ -71,16 +71,9 @@ def _pre_setup(self): self.settings_override = override_settings( MEDIA_ROOT=self.tempdir.path, EMAIL_BACKEND=u'poleno.mail.backend.EmailBackend', - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), ) self.settings_override.enable() - self.user1 = self._create_user() - self.user2 = self._create_user() - self.obligee1 = self._create_obligee() - self.obligee2 = self._create_obligee() - self.obligee3 = self._create_obligee() - def _post_teardown(self): self.settings_override.disable() self.tempdir.cleanup() diff --git a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py index c1532ae3a..14c354cf0 100644 --- a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py +++ b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py @@ -15,11 +15,16 @@ class MineViewTest(InforequestsTestCaseMixin, ViewTestCaseMixin, TestCase): Tests ``mine()`` view registered as "inforequests:mine". """ + def _create_pending_inforequest(self, applicant): + return self._create_inforequest(applicant=applicant) + def _create_successful_inforequest(self, applicant): inforequest, _, _ = self._create_inforequest_scenario(applicant, {u'closed': True}, (u'disclosure', dict(disclosure_level=Action.DISCLOSURE_LEVELS.PARTIAL))) return inforequest + def _create_unsuccessful_inforequest(self, applicant): + return self._create_inforequest(applicant=applicant, closed=True) def test_allowed_http_methods(self): allowed = [u'HEAD', u'GET'] @@ -37,12 +42,12 @@ def test_authenticated_user_gets_inforequest_mine(self): def test_user_gets_only_his_inforequests_and_drafts(self): drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] - pending_inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] - pending_inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] + pending_inforequests1 = [self._create_pending_inforequest(applicant=self.user1) for i in range(4)] + pending_inforequests2 = [self._create_pending_inforequest(applicant=self.user2) for i in range(5)] successful_inforequests1 = [self._create_successful_inforequest(applicant=self.user1) for i in range(3)] successful_inforequests2 = [self._create_successful_inforequest(applicant=self.user2) for i in range(3)] - unsuccessful_inforequests1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] - unsuccessful_inforequests2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] + unsuccessful_inforequests1 = [self._create_unsuccessful_inforequest(applicant=self.user1) for i in range(3)] + unsuccessful_inforequests2 = [self._create_unsuccessful_inforequest(applicant=self.user2) for i in range(3)] self._login_user(self.user1) response = self.client.get(reverse(u'inforequests:mine')) @@ -54,9 +59,9 @@ def test_user_gets_only_his_inforequests_and_drafts(self): def test_with_user_with_no_his_inforequests_nor_drafts(self): drafts2 = [self._create_inforequest_draft(applicant=self.user2) for i in range(3)] - pending_inforequests2 = [self._create_inforequest(applicant=self.user2) for i in range(5)] + pending_inforequests2 = [self._create_pending_inforequest(applicant=self.user2) for i in range(5)] successful_inforequests2 = [self._create_successful_inforequest(applicant=self.user2) for i in range(3)] - unsuccessful_inforequests2 = [self._create_inforequest(applicant=self.user2, closed=True) for i in range(3)] + unsuccessful_inforequests2 = [self._create_unsuccessful_inforequest(applicant=self.user2) for i in range(3)] self._login_user(self.user1) response = self.client.get(reverse(u'inforequests:mine')) @@ -68,9 +73,9 @@ def test_with_user_with_no_his_inforequests_nor_drafts(self): def test_related_models_are_prefetched_before_render(self): drafts1 = [self._create_inforequest_draft(applicant=self.user1) for i in range(5)] - pending_inforequests1 = [self._create_inforequest(applicant=self.user1) for i in range(4)] + pending_inforequests1 = [self._create_pending_inforequest(applicant=self.user1) for i in range(4)] successful_inforequests1 = [self._create_successful_inforequest(applicant=self.user1) for i in range(3)] - unsuccessful_inforequests1 = [self._create_inforequest(applicant=self.user1, closed=True) for i in range(3)] + unsuccessful_inforequests1 = [self._create_unsuccessful_inforequest(applicant=self.user1) for i in range(3)] # Force view querysets to evaluate before calling render def pre_mock_render(request, template, context): @@ -80,59 +85,71 @@ def pre_mock_render(request, template, context): list(context[u'unsuccessful_inforequests']) self._login_user(self.user1) - - # main/base/single_column.html - patterns = [ - u'FROM "accounts_profile" WHERE "accounts_profile"."user_id" = %s', + patterns_single_column = [ + u'FROM "accounts_profile" WHERE "accounts_profile"."user_id" = %s LIMIT 21', squeeze(u""" - FROM "mail_message" + SELECT COUNT\(\*\) FROM "mail_message" INNER JOIN "inforequests_inforequestemail" ON \( "mail_message"."id" = "inforequests_inforequestemail"."email_id" \) INNER JOIN "inforequests_inforequest" ON \( "inforequests_inforequestemail"."inforequest_id" = "inforequests_inforequest"."id" \) WHERE \("inforequests_inforequestemail"."type" = %s AND "inforequests_inforequest"."applicant_id" = %s AND "inforequests_inforequest"."closed" = %s\) """), - u'FROM "invitations_invitationsupply" WHERE "invitations_invitationsupply"."user_id" = %s', + u'FROM "invitations_invitationsupply" WHERE "invitations_invitationsupply"."user_id" = %s LIMIT 21', ] - - # pending_inforequests - patterns.extend([ + # TODO: Optimize N+1 queries in #419 + patterns_pending_inforequests = [ squeeze(u""" FROM "inforequests_action" INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) WHERE "inforequests_branch"."inforequest_id" = %s + ORDER BY "inforequests_action"."created" DESC, "inforequests_action"."id" DESC + LIMIT 1 """) for _ in range(4) - ]) - - # successful_inforequests - patterns.extend([ - u'FROM "inforequests_action" WHERE "inforequests_action"."branch_id" = %s', - u'FROM "inforequests_branch" WHERE "inforequests_branch"."inforequest_id" = %s', + ] + patterns_successful_inforequests = [ + squeeze(u""" + FROM "inforequests_action" WHERE "inforequests_action"."branch_id" = %s + ORDER BY "inforequests_action"."created" ASC, "inforequests_action"."id" ASC + """), + squeeze(u""" + FROM "inforequests_branch" WHERE "inforequests_branch"."inforequest_id" = %s + ORDER BY "inforequests_branch"."id" ASC"""), squeeze(u""" FROM "inforequests_action" INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) WHERE "inforequests_branch"."inforequest_id" = %s + ORDER BY "inforequests_action"."created" DESC, "inforequests_action"."id" DESC + LIMIT 1 """), squeeze(u""" + SELECT MAX\("inforequests_action"."disclosure_level"\) AS "disclosure_level__max" FROM "inforequests_action" INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) WHERE "inforequests_branch"."inforequest_id" = %s """), - ] * 3) - - # unsuccessful_inforequests - patterns.extend([ + ] * 3 + patterns_unsuccessful_inforequests = [ squeeze(u""" FROM "inforequests_action" INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) WHERE "inforequests_branch"."inforequest_id" = %s + ORDER BY "inforequests_action"."created" DESC, "inforequests_action"."id" DESC + LIMIT 1 """), squeeze(u""" + SELECT MAX\("inforequests_action"."disclosure_level"\) AS "disclosure_level__max" FROM "inforequests_action" INNER JOIN "inforequests_branch" ON \( "inforequests_action"."branch_id" = "inforequests_branch"."id" \) WHERE "inforequests_branch"."inforequest_id" = %s """), - ] * 3) + ] * 3 + patterns = list( + patterns_single_column + + patterns_pending_inforequests + + patterns_successful_inforequests + + patterns_unsuccessful_inforequests + ) with self.assertQueriesDuringRender(patterns, pre_mock_render=pre_mock_render): response = self.client.get(reverse(u'inforequests:mine')) self.assertEqual(response.status_code, 200) diff --git a/chcemvediet/tests/__init__.py b/chcemvediet/tests/__init__.py index 3671d7d75..9a274c701 100644 --- a/chcemvediet/tests/__init__.py +++ b/chcemvediet/tests/__init__.py @@ -11,6 +11,7 @@ from django.template import Context, Template from django.test import TestCase from django.test.runner import DiscoverRunner +from django.test.utils import override_settings from poleno.attachments.models import Attachment from poleno.mail.models import Message, Recipient @@ -35,9 +36,14 @@ def setup_test_environment(self, **kwargs): super(CustomTestRunner, self).setup_test_environment(**kwargs) settings.LANGUAGE_CODE = u'en' os.environ[u'RECAPTCHA_TESTING'] = u'True' + self.settings_override = override_settings( + PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), + ) + self.settings_override.enable() def teardown_test_environment(self, **kwargs): del os.environ[u'RECAPTCHA_TESTING'] + self.settings_override.disable() super(CustomTestRunner, self).teardown_test_environment(**kwargs) def run_tests(self, *args, **kwargs): @@ -62,6 +68,11 @@ def _pre_setup(self): self.inforequest = self._create_inforequest() self.branch = self._create_branch() self.action = self._create_action() + self.user1 = self._create_user() + self.user2 = self._create_user() + self.obligee1 = self._create_obligee() + self.obligee2 = self._create_obligee() + self.obligee3 = self._create_obligee() def _call_with_defaults(self, func, kwargs, defaults): From c56d2bff649db904987b59f05a7226b735e17a6b Mon Sep 17 00:00:00 2001 From: viliambalaz Date: Tue, 16 Nov 2021 14:51:20 +0100 Subject: [PATCH 4/5] #280 Set global password_hasher --- chcemvediet/apps/accounts/tests/__init__.py | 18 -------------- chcemvediet/apps/accounts/tests/test_forms.py | 10 +++----- .../apps/accounts/tests/test_models.py | 3 +-- chcemvediet/apps/accounts/tests/test_views.py | 7 +----- .../tests/test_views/test_mine.py | 4 ++-- chcemvediet/tests/__init__.py | 24 +++++++++---------- chcemvediet/tests/test_cron.py | 6 ----- poleno/attachments/tests/test_forms.py | 1 - poleno/attachments/tests/test_models.py | 1 - poleno/attachments/tests/test_views.py | 1 - poleno/timewarp/tests/test_admin.py | 9 ------- poleno/utils/tests/test_views.py | 1 - 12 files changed, 19 insertions(+), 66 deletions(-) diff --git a/chcemvediet/apps/accounts/tests/__init__.py b/chcemvediet/apps/accounts/tests/__init__.py index d2f8e730a..e69de29bb 100644 --- a/chcemvediet/apps/accounts/tests/__init__.py +++ b/chcemvediet/apps/accounts/tests/__init__.py @@ -1,18 +0,0 @@ -# vim: expandtab -# -*- coding: utf-8 -*- -from django.test import TestCase -from django.test.utils import override_settings - - -class AccountsTestCaseMixin(TestCase): - - def _pre_setup(self): - super(AccountsTestCaseMixin, self)._pre_setup() - self.settings_override = override_settings( - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), - ) - self.settings_override.enable() - - def _post_teardown(self): - self.settings_override.disable() - super(AccountsTestCaseMixin, self)._post_teardown() diff --git a/chcemvediet/apps/accounts/tests/test_forms.py b/chcemvediet/apps/accounts/tests/test_forms.py index 89f0a6f25..f6c0151bb 100644 --- a/chcemvediet/apps/accounts/tests/test_forms.py +++ b/chcemvediet/apps/accounts/tests/test_forms.py @@ -1,17 +1,13 @@ # vim: expandtab # -*- coding: utf-8 -*- -import os - import lxml.html from django.contrib.auth.models import User from django.test import TestCase from poleno.utils.urls import reverse -from . import AccountsTestCaseMixin - -class LoginFormTest(AccountsTestCaseMixin, TestCase): +class LoginFormTest(TestCase): u""" Tests ``allauth`` ``account_login`` view using ``LoginForm`` form registered as "account_login". Does not check ``account_login`` functionality, only checks functionality @@ -51,7 +47,7 @@ def test_recaptcha_field_is_required(self): response = self.client.post(reverse(u'account_login'), data, follow=True) self.assertFormError(response, u'form', u'recaptcha', u'This field is required.') -class SignupFormTest(AccountsTestCaseMixin, TestCase): +class SignupFormTest(TestCase): u""" Tests ``allauth`` ``account_signup`` view using ``SignupForm`` form registered as "account_signup". Does not check ``account_signup`` functionality, only checks functionality @@ -207,7 +203,7 @@ def test_recaptcha_field_is_required(self): response = self.client.post(reverse(u'account_signup'), data, follow=True) self.assertFormError(response, u'form', u'recaptcha', u'This field is required.') -class ResetPasswordFormTest(AccountsTestCaseMixin, TestCase): +class ResetPasswordFormTest(TestCase): u""" Tests ``allauth`` ``password_reset`` view using ``ResetPasswordForm`` form registered as "password_reset". Does not check ``password_reset`` functionality, only checks functionality diff --git a/chcemvediet/apps/accounts/tests/test_models.py b/chcemvediet/apps/accounts/tests/test_models.py index 53b6fce56..c32b4055a 100644 --- a/chcemvediet/apps/accounts/tests/test_models.py +++ b/chcemvediet/apps/accounts/tests/test_models.py @@ -4,12 +4,11 @@ from django.contrib.auth.models import User from django.test import TestCase -from . import AccountsTestCaseMixin from ..models import Profile from ..signals import create_profile_on_user_post_save -class ProfileModelTest(AccountsTestCaseMixin, TestCase): +class ProfileModelTest(TestCase): u""" Tests ``Profile`` model. """ diff --git a/chcemvediet/apps/accounts/tests/test_views.py b/chcemvediet/apps/accounts/tests/test_views.py index 392c6e96d..9613ad3eb 100644 --- a/chcemvediet/apps/accounts/tests/test_views.py +++ b/chcemvediet/apps/accounts/tests/test_views.py @@ -1,18 +1,13 @@ # vim: expandtab # -*- coding: utf-8 -*- -import functools -import mock - from django.contrib.auth.models import User from django.test import TestCase from poleno.utils.test import ViewTestCaseMixin from poleno.utils.urls import reverse -from . import AccountsTestCaseMixin - -class ProfileViewTest(AccountsTestCaseMixin, ViewTestCaseMixin, TestCase): +class ProfileViewTest(ViewTestCaseMixin, TestCase): u""" Tests ``profile()`` view registered as "accounts:profile". """ diff --git a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py index 14c354cf0..863114f2a 100644 --- a/chcemvediet/apps/inforequests/tests/test_views/test_mine.py +++ b/chcemvediet/apps/inforequests/tests/test_views/test_mine.py @@ -105,8 +105,8 @@ def pre_mock_render(request, template, context): WHERE "inforequests_branch"."inforequest_id" = %s ORDER BY "inforequests_action"."created" DESC, "inforequests_action"."id" DESC LIMIT 1 - """) for _ in range(4) - ] + """) + ] * 4 patterns_successful_inforequests = [ squeeze(u""" FROM "inforequests_action" WHERE "inforequests_action"."branch_id" = %s diff --git a/chcemvediet/tests/__init__.py b/chcemvediet/tests/__init__.py index 9a274c701..5436cc299 100644 --- a/chcemvediet/tests/__init__.py +++ b/chcemvediet/tests/__init__.py @@ -11,7 +11,6 @@ from django.template import Context, Template from django.test import TestCase from django.test.runner import DiscoverRunner -from django.test.utils import override_settings from poleno.attachments.models import Attachment from poleno.mail.models import Message, Recipient @@ -28,6 +27,7 @@ class CustomTestRunner(DiscoverRunner): -- Disabled logging while testing. Source: http://stackoverflow.com/questions/5255657/how-can-i-disable-logging-while-running-unit-tests-in-python-django -- Forced language code 'en' + -- Used MD5 password hasher, it's much faster then the production hasher. -- Mocked google recaptcha. Source: https://pypi.org/project/django-recaptcha/1.0.6/#unit-testing """ @@ -35,15 +35,11 @@ class CustomTestRunner(DiscoverRunner): def setup_test_environment(self, **kwargs): super(CustomTestRunner, self).setup_test_environment(**kwargs) settings.LANGUAGE_CODE = u'en' + settings.PASSWORD_HASHERS = [u'django.contrib.auth.hashers.MD5PasswordHasher'] os.environ[u'RECAPTCHA_TESTING'] = u'True' - self.settings_override = override_settings( - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), - ) - self.settings_override.enable() def teardown_test_environment(self, **kwargs): del os.environ[u'RECAPTCHA_TESTING'] - self.settings_override.disable() super(CustomTestRunner, self).teardown_test_environment(**kwargs) def run_tests(self, *args, **kwargs): @@ -56,6 +52,8 @@ def _pre_setup(self): super(ChcemvedietTestCaseMixin, self)._pre_setup() self.counter = itertools.count() self.user = self._create_user() + self.user1 = self._create_user() + self.user2 = self._create_user() self.message = self._create_message() self.recipient = self._create_recipient() self.region = self._create_region() @@ -63,16 +61,14 @@ def _pre_setup(self): self.municipality = self._create_municipality() self.neighbourhood = self._create_neighbourhood() self.obligee = self._create_obligee() + self.obligee1 = self._create_obligee() + self.obligee2 = self._create_obligee() + self.obligee3 = self._create_obligee() self.tag = self._create_obligee_tag() self.group = self._create_obligee_group() self.inforequest = self._create_inforequest() self.branch = self._create_branch() self.action = self._create_action() - self.user1 = self._create_user() - self.user2 = self._create_user() - self.obligee1 = self._create_obligee() - self.obligee2 = self._create_obligee() - self.obligee3 = self._create_obligee() def _call_with_defaults(self, func, kwargs, defaults): @@ -112,7 +108,11 @@ def _create_user(self, **kwargs): def _login_user(self, user=None, password=u'default_testing_secret'): if user is None: user = self.user - self.client.login(username=user.username, password=password) + if not self.client.login(username=user.username, password=password): + raise Exception( + u'The provided credentials are incorrect:\nusername: {} password: {}.'.format( + user.username, password + )) def _logout_user(self): self.client.logout() diff --git a/chcemvediet/tests/test_cron.py b/chcemvediet/tests/test_cron.py index caf124a8d..3d1baeb3d 100644 --- a/chcemvediet/tests/test_cron.py +++ b/chcemvediet/tests/test_cron.py @@ -3,7 +3,6 @@ from django.contrib.auth.models import User from django.contrib.sessions.models import Session from django.test import TestCase -from django.test.utils import override_settings from poleno.timewarp import timewarp from poleno.utils.date import utc_datetime_from_local @@ -20,13 +19,8 @@ def _pre_setup(self): super(ClearExpiredSessionsCronjobTest, self)._pre_setup() timewarp.enable() timewarp.reset() - self.settings_override = override_settings( - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), - ) - self.settings_override.enable() def _post_teardown(self): - self.settings_override.disable() timewarp.reset() super(ClearExpiredSessionsCronjobTest, self)._post_teardown() diff --git a/poleno/attachments/tests/test_forms.py b/poleno/attachments/tests/test_forms.py index 764a1485c..71ff939be 100644 --- a/poleno/attachments/tests/test_forms.py +++ b/poleno/attachments/tests/test_forms.py @@ -38,7 +38,6 @@ def setUp(self): self.settings_override = override_settings( MEDIA_ROOT=self.tempdir.path, - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), ) self.settings_override.enable() diff --git a/poleno/attachments/tests/test_models.py b/poleno/attachments/tests/test_models.py index 282805f39..12ed8b2f4 100644 --- a/poleno/attachments/tests/test_models.py +++ b/poleno/attachments/tests/test_models.py @@ -30,7 +30,6 @@ def setUp(self): self.settings_override = override_settings( MEDIA_ROOT=self.tempdir.path, - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), ) self.settings_override.enable() diff --git a/poleno/attachments/tests/test_views.py b/poleno/attachments/tests/test_views.py index 3da57bbf1..e0b9a8543 100644 --- a/poleno/attachments/tests/test_views.py +++ b/poleno/attachments/tests/test_views.py @@ -42,7 +42,6 @@ def setUp(self): self.settings_override = override_settings( MEDIA_ROOT=self.tempdir.path, - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), ) self.settings_override.enable() diff --git a/poleno/timewarp/tests/test_admin.py b/poleno/timewarp/tests/test_admin.py index cccc90971..c1ec5bf12 100644 --- a/poleno/timewarp/tests/test_admin.py +++ b/poleno/timewarp/tests/test_admin.py @@ -5,7 +5,6 @@ from django.http import HttpResponse from django.contrib.auth.models import User from django.test import TestCase -from django.test.utils import override_settings from poleno.utils.urls import reverse @@ -21,20 +20,12 @@ class TimewarpAdminTest(TestCase): def setUp(self): timewarp.enable() timewarp.reset() - - self.settings_override = override_settings( - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), - ) - self.settings_override.enable() - self.admin = User.objects.create_superuser(username=u'admin', email=u'admin@example.com', password=u'top_secret') self.client.login(username=u'admin', password=u'top_secret') def tearDown(self): timewarp.reset() - self.settings_override.disable() - def _parse_dt(self, value): dt = datetime.datetime.strptime(value, u'%Y-%m-%d %H:%M:%S') diff --git a/poleno/utils/tests/test_views.py b/poleno/utils/tests/test_views.py index 99eeb32d7..4542598b1 100644 --- a/poleno/utils/tests/test_views.py +++ b/poleno/utils/tests/test_views.py @@ -66,7 +66,6 @@ def login_required_with_exception_view(request): def setUp(self): self.settings_override = override_settings( - PASSWORD_HASHERS=(u'django.contrib.auth.hashers.MD5PasswordHasher',), TEMPLATE_LOADERS=(u'django.template.loaders.filesystem.Loader',), ) self.settings_override.enable() From d07cb6f6f02e6a0d2c0e62219078e94bd4763543 Mon Sep 17 00:00:00 2001 From: viliambalaz Date: Tue, 23 Nov 2021 13:45:43 +0100 Subject: [PATCH 5/5] #280 Use assertTrue instead raising exception --- chcemvediet/tests/__init__.py | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/chcemvediet/tests/__init__.py b/chcemvediet/tests/__init__.py index 5436cc299..1c399e149 100644 --- a/chcemvediet/tests/__init__.py +++ b/chcemvediet/tests/__init__.py @@ -108,11 +108,10 @@ def _create_user(self, **kwargs): def _login_user(self, user=None, password=u'default_testing_secret'): if user is None: user = self.user - if not self.client.login(username=user.username, password=password): - raise Exception( - u'The provided credentials are incorrect:\nusername: {} password: {}.'.format( - user.username, password - )) + logged_in = self.client.login(username=user.username, password=password) + self.assertTrue(logged_in, u'The provided credentials are incorrect:\nusername: {} password: {}.'.format( + user.username, password + )) def _logout_user(self): self.client.logout()