add prefetching, add test cases checking number of queries

This commit is contained in:
Mira Weller
2024-07-25 14:37:19 +02:00
parent 5087e654e2
commit 9337ad1f70
2 changed files with 178 additions and 4 deletions

View File

@@ -74,7 +74,7 @@ class CrossSellingService:
(c, products_qs, discount_info) for (c, products_qs, discount_info) in
(
(c, *self._get_visible_items_for_category(c))
for c in self.event.categories.filter(cross_selling_mode__isnull=False)
for c in self.event.categories.filter(cross_selling_mode__isnull=False).prefetch_related('items')
)
if products_qs is not None
]
@@ -96,7 +96,7 @@ class CrossSellingService:
match = set(match.pk for match in category.cross_selling_match_products.only('pk')) # TODO prefetch this
return (category.items.all(), {}) if any(pos.item.pk in match for pos in self.cartpositions) else (None, {})
if category.cross_selling_condition == 'discounts':
my_item_pks = category.items.values_list('pk', flat=True)
my_item_pks = [item.id for item in category.items.all()] #category.items.values_list('pk', flat=True)
potential_discount_items = {
item.pk: (max_count, discount_rule)
for item, max_count, discount_rule in self._potential_discounts_by_item_for_current_cart

View File

@@ -30,6 +30,7 @@ from django_scopes import scopes_disabled
from pretix.base.models import CartPosition, Discount, Event, Organizer
from pretix.base.services.cross_selling import CrossSellingService
from tests import assert_num_queries
@pytest.fixture
@@ -227,6 +228,7 @@ def setup_items(event, category_name, category_type, cross_selling_condition, *i
item = cat.items.create(event=event, name=name, default_price=price)
quota = event.quotas.create()
quota.items.add(item)
quota.save()
def split_table(txt):
@@ -237,7 +239,7 @@ def split_table(txt):
]
def check_cart_behaviour(event, cart_contents, recommendations):
def check_cart_behaviour(event, cart_contents, recommendations, expect_num_queries=None):
cart_contents = split_table(cart_contents)
positions = [
CartPosition(
@@ -248,8 +250,15 @@ def check_cart_behaviour(event, cart_contents, recommendations):
]
expected_recommendations = split_table(recommendations)
event.organizer.get_cache().clear()
event.get_cache().clear()
event = Event.objects.get(pk=event.pk)
service = CrossSellingService(event, event.organizer.sales_channels.get(identifier='web'), positions, None)
result = service.get_data()
if expect_num_queries:
with assert_num_queries(expect_num_queries):
result = service.get_data()
else:
result = service.get_data()
result_recommendations = [
[str(category.name), str(item.name), str(item.original_price.gross.quantize(Decimal('0.00'))),
str(item.display_price.gross.quantize(Decimal('0.00'))), str(item.order_max)]
@@ -457,3 +466,168 @@ def test_five_tickets_one_free(event):
recommendations=''' Price Discounted Price Max Count
'''
)
@scopes_disabled()
@pytest.mark.django_db
@pytest.mark.parametrize("itemcount", [3, 10, 50])
def test_query_count_many_items(event, itemcount):
setup_items(event, 'Tickets', 'both', 'discounts',
*[(f'Ticket {n}', '42.00') for n in range(itemcount)]
)
make_discount('For every 5 of Ticket 1, get 100% discount on 1 of Ticket 2.', event)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
''',
expect_num_queries=8,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Tickets Ticket 2 42.00 0.00 1
''',
expect_num_queries=13,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
Ticket 1 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Tickets Ticket 2 42.00 0.00 1
''',
expect_num_queries=13,
)
@scopes_disabled()
@pytest.mark.django_db
@pytest.mark.parametrize("catcount", [1, 10, 50])
def test_query_count_many_categories_and_discounts(event, catcount):
for n in range(1, catcount + 1):
setup_items(event, f'Category {n}', 'both', 'discounts',
(f'Ticket {n}-A', '42.00'),
(f'Ticket {n}-B', '42.00'),
)
make_discount(f'For every 5 of Ticket {n}-A, get 100% discount on 1 of Ticket {n}-B.', event)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
''',
expect_num_queries=8,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Category 1 Ticket 1-B 42.00 0.00 1
''',
expect_num_queries=13,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Category 1 Ticket 1-B 42.00 0.00 1
''',
expect_num_queries=13,
)
@scopes_disabled()
@pytest.mark.django_db
@pytest.mark.parametrize("catcount", [2, 10, 50])
def test_query_count_many_cartpos(event, catcount):
for n in range(1, catcount + 1):
setup_items(event, f'Category {n}', 'both', 'discounts',
(f'Ticket {n}-A', '42.00'),
(f'Ticket {n}-B', '42.00'),
)
make_discount(f'For every 5 of Ticket {n}-A, get 100% discount on 1 of Ticket {n}-B.', event)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
''',
expect_num_queries=8,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Category 1 Ticket 1-B 42.00 0.00 1
''',
expect_num_queries=13,
)
check_cart_behaviour(
event,
cart_contents=''' Price Discounted
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 1-A 42.00 42.00
Ticket 2-A 42.00 42.00
Ticket 2-A 42.00 42.00
Ticket 2-A 42.00 42.00
Ticket 2-A 42.00 42.00
Ticket 2-A 42.00 42.00
''',
recommendations=''' Price Discounted Price Max Count
Category 1 Ticket 1-B 42.00 0.00 1
Category 2 Ticket 2-B 42.00 0.00 1
''',
expect_num_queries=18,
)