mirror of
https://github.com/pretix/pretix.git
synced 2026-05-17 17:14:04 +00:00
729 lines
28 KiB
Python
729 lines
28 KiB
Python
#
|
|
# This file is part of pretix (Community Edition).
|
|
#
|
|
# Copyright (C) 2014-2020 Raphael Michel and contributors
|
|
# Copyright (C) 2020-today pretix GmbH and contributors
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
|
|
# Public License as published by the Free Software Foundation in version 3 of the License.
|
|
#
|
|
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
|
|
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
|
|
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
|
|
# this file, see <https://pretix.eu/about/en/license>.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
|
|
# <https://www.gnu.org/licenses/>.
|
|
#
|
|
|
|
import datetime
|
|
from zoneinfo import ZoneInfo
|
|
|
|
import pytest
|
|
from django.core import mail as djmail
|
|
from django.db.models import F
|
|
from django.utils.timezone import now
|
|
from django_scopes import scopes_disabled
|
|
|
|
from pretix.base.models import Event, InvoiceAddress, Order
|
|
from pretix.base.services.checkin import perform_checkin
|
|
from pretix.plugins.sendmail.models import Rule, ScheduledMail
|
|
from pretix.plugins.sendmail.signals import sendmail_run_rules
|
|
|
|
|
|
@pytest.mark.django_db
|
|
def test_sendmail_rule_create_single(event):
|
|
dt = now()
|
|
r = Rule.objects.create(event=event, subject='dummy mail', template='mail body', send_date=dt)
|
|
|
|
mails = ScheduledMail.objects.filter(rule=r)
|
|
assert mails.count() == 1
|
|
|
|
mail = mails.get()
|
|
assert mail.computed_datetime == dt
|
|
|
|
|
|
dt_now = now()
|
|
NZ = ZoneInfo('NZ')
|
|
Berlin = ZoneInfo('Europe/Berlin')
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@pytest.mark.parametrize(
|
|
"event_from,event_to,event_tz,rule,expected",
|
|
[
|
|
|
|
# Tests for all possible configurations of relative times
|
|
( # "Absolute"
|
|
None,
|
|
None,
|
|
'UTC',
|
|
Rule(date_is_absolute=True, send_date=dt_now),
|
|
dt_now
|
|
),
|
|
( # "Relative, after event start"
|
|
None,
|
|
None,
|
|
'UTC',
|
|
Rule(date_is_absolute=False, offset_is_after=True, send_offset_days=1, send_offset_time=datetime.time(hour=9)),
|
|
(dt_now + datetime.timedelta(days=1)).replace(hour=9, minute=0, second=0, microsecond=0)
|
|
),
|
|
( # "Relative, before event start"
|
|
datetime.datetime(2021, 5, 17, 12, 14, 0, tzinfo=datetime.timezone.utc),
|
|
None,
|
|
'UTC',
|
|
Rule(date_is_absolute=False, send_offset_days=2, send_offset_time=datetime.time(hour=0)),
|
|
datetime.datetime(2021, 5, 15, 0, tzinfo=datetime.timezone.utc)
|
|
),
|
|
( # "Relative, after event end"
|
|
datetime.datetime(2021, 5, 17, 18, tzinfo=datetime.timezone.utc),
|
|
datetime.datetime(2021, 5, 18, 5, tzinfo=datetime.timezone.utc),
|
|
'UTC',
|
|
Rule(date_is_absolute=False, offset_to_event_end=True, offset_is_after=True, send_offset_days=1, send_offset_time=datetime.time(hour=10)),
|
|
datetime.datetime(2021, 5, 19, 10, tzinfo=datetime.timezone.utc)
|
|
),
|
|
( # "Relative, before event end"
|
|
datetime.datetime(2021, 5, 17, 18, tzinfo=datetime.timezone.utc),
|
|
datetime.datetime(2021, 5, 22, 5, tzinfo=datetime.timezone.utc),
|
|
'UTC',
|
|
Rule(date_is_absolute=False, offset_to_event_end=True, offset_is_after=False, send_offset_days=1, send_offset_time=datetime.time(hour=10)),
|
|
datetime.datetime(2021, 5, 21, 10, tzinfo=datetime.timezone.utc)
|
|
),
|
|
|
|
# Tests for timezone quirks
|
|
( # Test sending on leap day
|
|
datetime.datetime(2020, 2, 27, 9, tzinfo=datetime.timezone.utc),
|
|
None,
|
|
'UTC',
|
|
Rule(date_is_absolute=False, offset_is_after=True, send_offset_days=2, send_offset_time=datetime.time(hour=9)),
|
|
datetime.datetime(2020, 2, 29, 9, tzinfo=datetime.timezone.utc)
|
|
),
|
|
( # Test timezone far off from UTC
|
|
datetime.datetime(2021, 5, 17, 22, tzinfo=NZ),
|
|
None,
|
|
'NZ',
|
|
Rule(date_is_absolute=False, offset_is_after=True, send_offset_days=1, send_offset_time=datetime.time(hour=9)),
|
|
datetime.datetime(2021, 5, 18, 9, tzinfo=NZ)
|
|
),
|
|
( # Test across DST change
|
|
datetime.datetime(2021, 10, 29, 16, 30, tzinfo=Berlin),
|
|
None,
|
|
'Europe/Berlin',
|
|
Rule(date_is_absolute=False, offset_is_after=True, send_offset_days=4, send_offset_time=datetime.time(hour=2, minute=30)),
|
|
datetime.datetime(2021, 11, 2, 2, 30, tzinfo=Berlin)
|
|
),
|
|
( # Test ambiguous time at DST change
|
|
datetime.datetime(2021, 10, 29, 18, 30, tzinfo=Berlin),
|
|
None,
|
|
'Europe/Berlin',
|
|
Rule(date_is_absolute=False, offset_is_after=True, send_offset_days=2, send_offset_time=datetime.time(hour=2, minute=30)),
|
|
datetime.datetime(2021, 10, 31, 1, 30, tzinfo=datetime.timezone.utc)
|
|
),
|
|
( # Test non-existing time at DST change
|
|
datetime.datetime(2021, 3, 29, 14, 30, tzinfo=Berlin),
|
|
None,
|
|
'Europe/Berlin',
|
|
Rule(date_is_absolute=False, offset_is_after=False, send_offset_days=1, send_offset_time=datetime.time(hour=2, minute=30)),
|
|
datetime.datetime(2021, 3, 28, 1, 30, tzinfo=datetime.timezone.utc)
|
|
),
|
|
|
|
])
|
|
def test_sendmail_rule_send_time(event_from, event_to, event_tz, rule, expected, event):
|
|
if event_from:
|
|
event.date_from = event_from
|
|
event.save()
|
|
|
|
if event_to:
|
|
event.date_to = event_to
|
|
event.save()
|
|
|
|
event.settings.timezone = event_tz
|
|
|
|
rule.event = event
|
|
rule.save()
|
|
m = ScheduledMail.objects.filter(rule=rule).get()
|
|
|
|
assert m.computed_datetime.astimezone(event.timezone) == expected.astimezone(event.timezone)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_recompute(event):
|
|
event.has_subevents = True
|
|
event.save()
|
|
se1 = event.subevents.create(name="meow", date_from=dt_now)
|
|
|
|
rule = event.sendmail_rules.create(date_is_absolute=False, offset_is_after=False, send_offset_days=1,
|
|
send_offset_time=datetime.time(4, 30))
|
|
|
|
se1.date_from += datetime.timedelta(days=1)
|
|
se1.save()
|
|
|
|
expected = dt_now.replace(hour=4, minute=30, second=0, microsecond=0)
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
m = ScheduledMail.objects.filter(rule=rule).first()
|
|
assert m.computed_datetime.astimezone(datetime.timezone.utc) == expected
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@pytest.mark.parametrize('send_to,amount_mails,recipients', [
|
|
(Rule.CUSTOMERS, 1, ['dummy@dummy.test']),
|
|
(Rule.ATTENDEES, 1, ['meow@dummy.test']),
|
|
(Rule.BOTH, 2, ['dummy@dummy.test', 'meow@dummy.test']),
|
|
])
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_vs_pos(send_to, amount_mails, recipients, order, event, item):
|
|
djmail.outbox = []
|
|
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
order.event.sendmail_rules.create(date_is_absolute=True, send_date=dt_now - datetime.timedelta(hours=1),
|
|
send_to=send_to,
|
|
subject='meow', template='meow meow meow')
|
|
order.all_positions.create(item=item, price=0, attendee_email='meow@dummy.test')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == amount_mails
|
|
|
|
_recipients = [mail.to[0] for mail in djmail.outbox]
|
|
assert set(recipients) == set(_recipients)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_attendees_unset_mail(order, event, item):
|
|
djmail.outbox = []
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
order.all_positions.create(item=item, price=13)
|
|
order.event.sendmail_rules.create(date_is_absolute=True, send_date=dt_now - datetime.timedelta(hours=1),
|
|
send_to=Rule.ATTENDEES,
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
mail = djmail.outbox[0]
|
|
assert mail.to[0] == 'dummy@dummy.test'
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_both_same_email(order, event, item):
|
|
djmail.outbox = []
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
order.all_positions.create(item=item, price=13, attendee_email='dummy@dummy.test')
|
|
order.event.sendmail_rules.create(date_is_absolute=True, send_date=dt_now - datetime.timedelta(hours=1),
|
|
send_to=Rule.BOTH,
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_correct_subevent(order, event_series, subevent1, subevent2, item):
|
|
djmail.outbox = []
|
|
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
event_series.sendmail_rules.create(date_is_absolute=False, offset_is_after=False, send_offset_days=2,
|
|
send_offset_time=datetime.time(9, 30), send_to=Rule.ATTENDEES,
|
|
subject='meow', template='meow meow meow')
|
|
p1 = order.all_positions.create(item=item, price=13, attendee_email='se1@dummy.test', subevent=subevent1)
|
|
order.all_positions.create(item=item, price=23, attendee_email='se2@dummy.test', subevent=subevent2)
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
assert djmail.outbox[0].to[0] == p1.attendee_email
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_specified_subevent(order, event_series, subevent1, subevent2, item):
|
|
djmail.outbox = []
|
|
|
|
order.all_positions.create(item=item, price=13, subevent=subevent1)
|
|
order.all_positions.create(item=item, price=23, subevent=subevent2)
|
|
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
rule = event_series.sendmail_rules.create(date_is_absolute=False, offset_is_after=False, send_offset_days=1,
|
|
send_offset_time=datetime.time(0, 0), subevent=subevent1,
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
m = ScheduledMail.objects.filter(rule=rule).first()
|
|
assert m.subevent == subevent1
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_all_subevents(event_series, subevent1, subevent2, item):
|
|
djmail.outbox = []
|
|
|
|
o1 = Order.objects.create(event=item.event, status=Order.STATUS_PAID,
|
|
expires=now() + datetime.timedelta(hours=1),
|
|
total=13, code='DUMMY1', email='dummy1@dummy.test',
|
|
sales_channel=event_series.organizer.sales_channels.get(identifier="web"),
|
|
datetime=now(), locale='en')
|
|
o1.all_positions.create(item=item, price=13, subevent=subevent1)
|
|
o1.all_positions.create(item=item, price=13, subevent=subevent2)
|
|
o2 = Order.objects.create(event=item.event, status=Order.STATUS_PAID,
|
|
expires=now() + datetime.timedelta(hours=1),
|
|
total=13, code='DUMMY2', email='dummy2@dummy.test',
|
|
sales_channel=event_series.organizer.sales_channels.get(identifier="web"),
|
|
datetime=now(), locale='en')
|
|
o2.all_positions.create(item=item, price=23, subevent=subevent1)
|
|
o2.all_positions.create(item=item, price=23, subevent=subevent2)
|
|
|
|
rule = event_series.sendmail_rules.create(date_is_absolute=False, offset_is_after=False, send_offset_days=1,
|
|
send_offset_time=datetime.time(4, 30), subevent=None,
|
|
subject='meow', template='meow meow meow')
|
|
sendmail_run_rules(None)
|
|
|
|
ms = ScheduledMail.objects.filter(rule=rule)
|
|
assert len(ms) == 2 # one for each subevent
|
|
assert ms.get(subevent=subevent1) is not None
|
|
assert ms.get(subevent=subevent2) is not None
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_correct_products(event, order, item, item2):
|
|
djmail.outbox = []
|
|
|
|
order.status = order.STATUS_PAID
|
|
order.save()
|
|
|
|
rule = event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), send_to=Rule.ATTENDEES,
|
|
subject='meow', template='meow meow meow', all_products=False)
|
|
|
|
rule.limit_products.set([item])
|
|
rule.save()
|
|
|
|
p1 = order.all_positions.create(item=item, price=13, attendee_email='item1@dummy.test')
|
|
order.all_positions.create(item=item2, price=13, attendee_email='item2@dummy.test')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
assert djmail.outbox[0].to[0] == p1.attendee_email
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_not_checked_in_all_get_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="all",
|
|
subject='meow', template='meow meow meow')
|
|
|
|
djmail.outbox = []
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1, "email not sent"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_checked_in_all_get_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
p1 = order.all_positions.create(item=item, price=13, attendee_email='item1@dummy.test')
|
|
clist = event.checkin_lists.create(name="Default", all_products=True)
|
|
perform_checkin(p1, clist, {})
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="all",
|
|
subject='meow', template='meow meow meow')
|
|
|
|
djmail.outbox = []
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1, "email not sent"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_not_checked_in_no_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="checked_in",
|
|
subject='meow', template='meow meow meow')
|
|
|
|
# receives no mail when not checked in
|
|
djmail.outbox = []
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 0, "email sent unexpectedly"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_not_checked_in_get_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
order.all_positions.create(item=item, price=13, attendee_email='item1@dummy.test')
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="no_checkin",
|
|
subject='meow', template='meow meow meow')
|
|
|
|
# receives mail when not checked in
|
|
djmail.outbox = []
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1, "email not sent"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_checked_in_no_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
p1 = order.all_positions.create(item=item, price=13, attendee_email='item1@dummy.test')
|
|
clist = event.checkin_lists.create(name="Default", all_products=True)
|
|
|
|
# receives no mail when checked in
|
|
djmail.outbox = []
|
|
perform_checkin(p1, clist, {})
|
|
assert clist.checkin_count == 1
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="no_checkin",
|
|
subject='meow', template='meow meow meow')
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 0, "email sent unexpectedly"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_checked_in_get_mail(event, order, item):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
p1 = order.all_positions.create(item=item, price=13, attendee_email='item1@dummy.test')
|
|
clist = event.checkin_lists.create(name="Default", all_products=True)
|
|
|
|
# receives mail when checked in
|
|
djmail.outbox = []
|
|
perform_checkin(p1, clist, {})
|
|
assert clist.checkin_count == 1
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), checked_in_status="checked_in",
|
|
subject='meow', template='meow meow meow')
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1, "email not sent"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def run_restriction_test(event, order, restrictions_pass=[], restrictions_fail=[]):
|
|
for r in restrictions_pass:
|
|
djmail.outbox = []
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), restrict_to_status=[r],
|
|
subject='meow', template='meow meow meow')
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1, f"email not sent for {r}"
|
|
|
|
for r in restrictions_fail:
|
|
djmail.outbox = []
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1), restrict_to_status=[r],
|
|
subject='meow', template='meow meow meow')
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 0, f"email sent for {r} unexpectedly"
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_paid(event, order):
|
|
order.status = Order.STATUS_PAID
|
|
order.save()
|
|
order.valid_if_pending = False
|
|
restrictions_pass = ['p']
|
|
restrictions_fail = ['e', 'c', 'n__not_pending_approval_and_not_valid_if_pending', 'n__pending_approval',
|
|
'n__valid_if_pending', 'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_pending(event, order):
|
|
order.status = Order.STATUS_PENDING
|
|
order.require_approval = False
|
|
order.valid_if_pending = False
|
|
order.save()
|
|
restrictions_pass = ['n__not_pending_approval_and_not_valid_if_pending']
|
|
restrictions_fail = ['p', 'e', 'c', 'n__pending_approval', "n__valid_if_pending", 'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_valid_pending(event, order):
|
|
order.status = Order.STATUS_PENDING
|
|
order.require_approval = False
|
|
order.valid_if_pending = True
|
|
order.save()
|
|
restrictions_pass = ["n__valid_if_pending"]
|
|
restrictions_fail = ['p', 'e', 'c', 'n__not_pending_approval_and_not_valid_if_pending', 'n__pending_approval',
|
|
'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_approval_pending(event, order):
|
|
order.status = Order.STATUS_PENDING
|
|
order.require_approval = True
|
|
order.valid_if_pending = False
|
|
order.save()
|
|
restrictions_pass = ['n__pending_approval']
|
|
restrictions_fail = ['p', 'e', 'c', 'n__not_pending_approval_and_not_valid_if_pending', "n__valid_if_pending",
|
|
'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_overdue_pending(event, order):
|
|
event.settings.payment_term_expire_automatically = False
|
|
order.status = Order.STATUS_PENDING
|
|
order.require_approval = False
|
|
order.valid_if_pending = False
|
|
order.expires = order.expires - datetime.timedelta(days=15)
|
|
order.save()
|
|
restrictions_pass = ['n__pending_overdue', 'n__not_pending_approval_and_not_valid_if_pending']
|
|
restrictions_fail = ['p', 'e', 'c', 'n__pending_approval', "n__valid_if_pending"]
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_expired(event, order):
|
|
order.status = Order.STATUS_EXPIRED
|
|
order.save()
|
|
restrictions_pass = ['e']
|
|
restrictions_fail = ['p', 'c', 'n__not_pending_approval_and_not_valid_if_pending', 'n__pending_approval',
|
|
'n__valid_if_pending', 'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_restrictions_status_canceled(event, order):
|
|
order.status = Order.STATUS_CANCELED
|
|
order.save()
|
|
restrictions_pass = ['c']
|
|
restrictions_fail = ['p', 'e', 'n__not_pending_approval_and_not_valid_if_pending', 'n__pending_approval',
|
|
'n__valid_if_pending', 'n__pending_overdue']
|
|
|
|
run_restriction_test(event, order, restrictions_pass, restrictions_fail)
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_pending(event, order):
|
|
djmail.outbox = []
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_pending_excluded(event, order):
|
|
djmail.outbox = []
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', "n__valid_if_pending"],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 0
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_valid_if_pending(event, order):
|
|
order.valid_if_pending = True
|
|
order.status = Order.STATUS_PENDING
|
|
order.save()
|
|
djmail.outbox = []
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', "n__valid_if_pending"],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 1
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@pytest.mark.parametrize('status', [
|
|
Order.STATUS_EXPIRED,
|
|
Order.STATUS_CANCELED,
|
|
])
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_status(status, event, order):
|
|
djmail.outbox = []
|
|
|
|
order.status = status
|
|
order.save()
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', "n__valid_if_pending"],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 0
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_send_order_approval(event, order):
|
|
djmail.outbox = []
|
|
|
|
order.require_approval = True
|
|
order.save()
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
|
|
assert len(djmail.outbox) == 0
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_only_send_once(event, order):
|
|
djmail.outbox = []
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 1
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_only_live(event, order):
|
|
djmail.outbox = []
|
|
event.live = False
|
|
event.save()
|
|
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='meow meow meow')
|
|
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 0
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_rule_disabled(event, order):
|
|
djmail.outbox = []
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='meow meow meow', enabled=False)
|
|
|
|
sendmail_run_rules(None)
|
|
assert len(djmail.outbox) == 0
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_sendmail_context_localization(event, order, pos):
|
|
order.locale = 'de'
|
|
order.save()
|
|
event.settings.name_scheme = 'salutation_given_family'
|
|
InvoiceAddress.objects.create(
|
|
order=order,
|
|
name_parts={'_scheme': 'salutation_given_family', 'salutation': 'Mr', 'given_name': 'Max', 'family_name': 'Mustermann'}
|
|
)
|
|
|
|
djmail.outbox = []
|
|
event.sendmail_rules.create(send_date=dt_now - datetime.timedelta(hours=1),
|
|
restrict_to_status=['p', 'n__not_pending_approval_and_not_valid_if_pending',
|
|
'n__valid_if_pending'],
|
|
subject='meow', template='Hallo {name_for_salutation}')
|
|
|
|
sendmail_run_rules(None)
|
|
assert "Hallo Herr Mustermann" in djmail.outbox[0].body
|
|
|
|
|
|
@pytest.mark.django_db
|
|
@scopes_disabled()
|
|
def test_event_clone_ignores_rules_with_subevent(event, order, pos):
|
|
event.has_subevents = True
|
|
event.save()
|
|
se1 = event.subevents.create(name="subevent 1", date_from=dt_now)
|
|
|
|
event.sendmail_rules.create(date_is_absolute=False, send_offset_days=1, send_offset_time=datetime.time(hour=12),
|
|
subject='Mail To Subevent', template='TestBody', subevent=se1)
|
|
event.sendmail_rules.create(date_is_absolute=False, send_offset_days=1, send_offset_time=datetime.time(hour=12),
|
|
subject='Mail To All', template='TestBody')
|
|
|
|
assert event.sendmail_rules.count() == 2
|
|
assert event.scheduledmail_set.count() == 2
|
|
for rule in event.sendmail_rules.all():
|
|
assert rule.scheduledmail_set.count() == 1
|
|
|
|
copied_event = Event.objects.create(
|
|
organizer=event.organizer, name='Dummy2', slug='dummy2',
|
|
date_from=datetime.datetime(2022, 4, 15, 9, 0, 0, tzinfo=datetime.timezone.utc),
|
|
has_subevents=True
|
|
)
|
|
copied_event.copy_data_from(event)
|
|
copied_event.refresh_from_db()
|
|
event.refresh_from_db()
|
|
|
|
assert copied_event.sendmail_rules.count() == 1
|
|
assert copied_event.sendmail_rules.first().scheduledmail_set.count() == 0
|
|
assert str(copied_event.sendmail_rules.first().subject) == "Mail To All"
|
|
|
|
copied_event.subevents.create(name="subevent 1 in copied", date_from=dt_now)
|
|
assert copied_event.sendmail_rules.first().scheduledmail_set.count() == 1
|
|
|
|
# Double-Check: no scheduled mails and rules exist where subevent.event and event do not align
|
|
assert ScheduledMail.objects.filter(rule__subevent__isnull=False).exclude(rule__subevent__event=F('rule__event')).count() == 0
|
|
assert Rule.objects.filter(subevent__isnull=False).exclude(subevent__event=F('event')).count() == 0
|