mirror of
https://github.com/pretix/pretix.git
synced 2026-05-04 15:04:03 +00:00
* Add version notes to the docs * Adapt signal handling * Add UI * Add API * API and tests * Fix registry * Update doc/development/api/plugins.rst Co-authored-by: Felix Rindt <felix@rindt.me> * Fix failing tests * Apply suggestions from code review Co-authored-by: Richard Schreiber <schreiber@rami.io> * Update src/pretix/control/templates/pretixcontrol/organizers/plugin_events.html Co-authored-by: luelista <weller@rami.io> * Update src/pretix/control/templates/pretixcontrol/organizers/plugins.html Co-authored-by: luelista <weller@rami.io> * Update src/pretix/control/templates/pretixcontrol/organizers/plugins.html Co-authored-by: luelista <weller@rami.io> * Update src/pretix/control/navigation.py Co-authored-by: luelista <weller@rami.io> * Update src/pretix/control/urls.py Co-authored-by: luelista <weller@rami.io> * Apply suggestion from @wiffbi * REbase migration * Fix review note * Fix test cases * Remove plugin from all events if disabled on org level * Update doc/development/api/plugins.rst * Unify registries * Rebase migration --------- Co-authored-by: Felix Rindt <felix@rindt.me> Co-authored-by: Richard Schreiber <schreiber@rami.io> Co-authored-by: luelista <weller@rami.io>
1331 lines
62 KiB
Python
1331 lines
62 KiB
Python
#
|
|
# This file is part of pretix (Community Edition).
|
|
#
|
|
# Copyright (C) 2014-2020 Raphael Michel and contributors
|
|
# Copyright (C) 2020-2021 rami.io 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/>.
|
|
#
|
|
|
|
# This file is based on an earlier version of pretix which was released under the Apache License 2.0. The full text of
|
|
# the Apache License 2.0 can be obtained at <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
#
|
|
# This file may have since been changed and any changes are released under the terms of AGPLv3 as described above. A
|
|
# full history of changes and contributors is available at <https://github.com/pretix/pretix>.
|
|
#
|
|
# This file contains Apache-licensed contributions copyrighted by: Aiman Parvaiz, Maico Timmerman, Matthew Emerson,
|
|
# Tobias Kunze, jasonwaiting@live.hk, luto
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software distributed under the Apache License 2.0 is
|
|
# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations under the License.
|
|
|
|
import datetime
|
|
import time
|
|
from decimal import Decimal
|
|
from smtplib import SMTPResponseException
|
|
from zoneinfo import ZoneInfo
|
|
|
|
import pytest
|
|
from django.test.utils import override_settings
|
|
from django.utils.timezone import now
|
|
from django_scopes import scopes_disabled
|
|
from i18nfield.strings import LazyI18nString
|
|
from tests.base import SoupTest, extract_form_fields
|
|
|
|
from pretix.base.models import Event, LogEntry, Order, Organizer, Team, User
|
|
|
|
|
|
@pytest.fixture
|
|
def class_monkeypatch(request, monkeypatch):
|
|
request.cls.monkeypatch = monkeypatch
|
|
|
|
|
|
@pytest.mark.usefixtures("class_monkeypatch")
|
|
class EventsTest(SoupTest):
|
|
@scopes_disabled()
|
|
def setUp(self):
|
|
super().setUp()
|
|
self.user = User.objects.create_user('dummy@dummy.dummy', 'dummy')
|
|
self.orga1 = Organizer.objects.create(name='CCC', slug='ccc')
|
|
self.orga2 = Organizer.objects.create(name='MRM', slug='mrm')
|
|
self.event1 = Event.objects.create(
|
|
organizer=self.orga1, name='30C3', slug='30c3',
|
|
date_from=datetime.datetime(2013, 12, 26, tzinfo=datetime.timezone.utc),
|
|
plugins='pretix.plugins.banktransfer,tests.testdummy'
|
|
)
|
|
self.event2 = Event.objects.create(
|
|
organizer=self.orga1, name='31C3', slug='31c3',
|
|
date_from=datetime.datetime(2014, 12, 26, tzinfo=datetime.timezone.utc),
|
|
)
|
|
self.event3 = Event.objects.create(
|
|
organizer=self.orga2, name='MRMCD14', slug='mrmcd14',
|
|
date_from=datetime.datetime(2014, 9, 5, tzinfo=datetime.timezone.utc),
|
|
)
|
|
|
|
self.team1 = Team.objects.create(organizer=self.orga1, can_create_events=True, can_change_event_settings=True,
|
|
can_change_items=True)
|
|
self.team1.members.add(self.user)
|
|
self.team1.limit_events.add(self.event1)
|
|
|
|
self.team2 = Team.objects.create(organizer=self.orga1, can_change_event_settings=True, can_change_items=True,
|
|
can_change_orders=True, can_change_vouchers=True)
|
|
self.team2.members.add(self.user)
|
|
|
|
self.client.login(email='dummy@dummy.dummy', password='dummy')
|
|
|
|
def test_event_list(self):
|
|
doc = self.get_doc('/control/events/')
|
|
tabletext = doc.select("#page-wrapper .table")[0].text
|
|
self.assertIn("30C3", tabletext)
|
|
self.assertNotIn("31C3", tabletext)
|
|
self.assertNotIn("MRMCD14", tabletext)
|
|
|
|
def test_convenience_organizer_redirect(self):
|
|
resp = self.client.get('/control/event/%s/' % (self.orga1.slug))
|
|
self.assertRedirects(resp, '/control/organizer/%s/' % (self.orga1.slug))
|
|
|
|
def test_quick_setup_later(self):
|
|
with scopes_disabled():
|
|
self.event1.quotas.create(name='foo', size=2)
|
|
resp = self.client.get('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug))
|
|
self.assertRedirects(resp, '/control/event/%s/%s/' % (self.orga1.slug, self.event1.slug))
|
|
|
|
def test_quick_setup_total_quota(self):
|
|
doc = self.get_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug))
|
|
doc.select("[name=show_quota_left]")[0]['checked'] = "checked"
|
|
doc.select("[name=ticket_download]")[0]['checked'] = "checked"
|
|
doc.select("[name=contact_mail]")[0]['value'] = "test@example.org"
|
|
doc.select("[name=payment_banktransfer__enabled]")[0]['checked'] = "checked"
|
|
doc.select("[name=payment_banktransfer_bank_details_type]")[1]['checked'] = 'checked'
|
|
del doc.select("[name=payment_banktransfer_bank_details_type]")[0]['checked']
|
|
doc.select("[name*=payment_banktransfer_bank_details_0]")[0].contents[0].replace_with("Foo")
|
|
doc.select("[name=total_quota]")[0]['value'] = "300"
|
|
doc.select("[name=form-TOTAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-INITIAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-MIN_NUM_FORMS]")[0]['value'] = "0"
|
|
doc.select("[name=form-MAX_NUM_FORMS]")[0]['value'] = "1000"
|
|
doc.select("[name=form-0-name_0]")[0]['value'] = "Normal ticket"
|
|
doc.select("[name=form-0-default_price]")[0]['value'] = "13.90"
|
|
doc.select("[name=form-0-quota]")[0]['value'] = ""
|
|
doc.select("[name=form-1-name_0]")[0]['value'] = "Reduced ticket"
|
|
doc.select("[name=form-1-default_price]")[0]['value'] = "13.20"
|
|
doc.select("[name=form-1-quota]")[0]['value'] = ""
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert len(doc.select(".alert-success")) > 0
|
|
self.event1.refresh_from_db()
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.show_quota_left
|
|
assert self.event1.settings.contact_mail == "test@example.org"
|
|
assert self.event1.settings.ticket_download
|
|
assert self.event1.settings.ticketoutput_pdf__enabled
|
|
assert self.event1.settings.payment_banktransfer__enabled
|
|
assert self.event1.settings.get('payment_banktransfer_bank_details', as_type=LazyI18nString).localize('en') == "Foo"
|
|
assert 'pretix.plugins.banktransfer' in self.event1.plugins
|
|
with scopes_disabled():
|
|
assert self.event1.items.count() == 2
|
|
i = self.event1.items.first()
|
|
assert str(i.name) == "Normal ticket"
|
|
assert i.default_price == Decimal('13.90')
|
|
i = self.event1.items.last()
|
|
assert str(i.name) == "Reduced ticket"
|
|
assert i.default_price == Decimal('13.20')
|
|
assert self.event1.quotas.count() == 1
|
|
q = self.event1.quotas.first()
|
|
assert q.name == 'Tickets'
|
|
assert q.size == 300
|
|
assert q.items.count() == 2
|
|
|
|
def test_quick_setup_single_quota(self):
|
|
doc = self.get_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug))
|
|
doc.select("[name=show_quota_left]")[0]['checked'] = "checked"
|
|
doc.select("[name=ticket_download]")[0]['checked'] = "checked"
|
|
doc.select("[name=contact_mail]")[0]['value'] = "test@example.org"
|
|
doc.select("[name=payment_banktransfer__enabled]")[0]['checked'] = "checked"
|
|
doc.select("[name=payment_banktransfer_bank_details_type]")[1]['checked'] = 'checked'
|
|
del doc.select("[name=payment_banktransfer_bank_details_type]")[0]['checked']
|
|
doc.select("[name*=payment_banktransfer_bank_details_0]")[0].contents[0].replace_with("Foo")
|
|
doc.select("[name=total_quota]")[0]['value'] = ""
|
|
doc.select("[name=form-TOTAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-INITIAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-MIN_NUM_FORMS]")[0]['value'] = "0"
|
|
doc.select("[name=form-MAX_NUM_FORMS]")[0]['value'] = "1000"
|
|
doc.select("[name=form-0-name_0]")[0]['value'] = "Normal ticket"
|
|
doc.select("[name=form-0-default_price]")[0]['value'] = "13.90"
|
|
doc.select("[name=form-0-quota]")[0]['value'] = "100"
|
|
doc.select("[name=form-1-name_0]")[0]['value'] = "Reduced ticket"
|
|
doc.select("[name=form-1-default_price]")[0]['value'] = "13.20"
|
|
doc.select("[name=form-1-quota]")[0]['value'] = "50"
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert len(doc.select(".alert-success")) > 0
|
|
self.event1.refresh_from_db()
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.show_quota_left
|
|
assert self.event1.settings.contact_mail == "test@example.org"
|
|
assert self.event1.settings.ticket_download
|
|
assert self.event1.settings.ticketoutput_pdf__enabled
|
|
assert self.event1.settings.payment_banktransfer__enabled
|
|
assert self.event1.settings.get('payment_banktransfer_bank_details', as_type=LazyI18nString).localize('en') == "Foo"
|
|
assert 'pretix.plugins.banktransfer' in self.event1.plugins
|
|
with scopes_disabled():
|
|
assert self.event1.items.count() == 2
|
|
i = self.event1.items.first()
|
|
assert str(i.name) == "Normal ticket"
|
|
assert i.default_price == Decimal('13.90')
|
|
i = self.event1.items.last()
|
|
assert str(i.name) == "Reduced ticket"
|
|
assert i.default_price == Decimal('13.20')
|
|
assert self.event1.quotas.count() == 2
|
|
q = self.event1.quotas.first()
|
|
assert q.name == 'Normal ticket'
|
|
assert q.size == 100
|
|
assert q.items.count() == 1
|
|
q = self.event1.quotas.last()
|
|
assert q.name == 'Reduced ticket'
|
|
assert q.size == 50
|
|
assert q.items.count() == 1
|
|
|
|
def test_quick_setup_dual_quota(self):
|
|
doc = self.get_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug))
|
|
doc.select("[name=show_quota_left]")[0]['checked'] = "checked"
|
|
doc.select("[name=ticket_download]")[0]['checked'] = "checked"
|
|
doc.select("[name=contact_mail]")[0]['value'] = "test@example.org"
|
|
doc.select("[name=payment_banktransfer__enabled]")[0]['checked'] = "checked"
|
|
doc.select("[name=payment_banktransfer_bank_details_type]")[1]['checked'] = 'checked'
|
|
del doc.select("[name=payment_banktransfer_bank_details_type]")[0]['checked']
|
|
doc.select("[name*=payment_banktransfer_bank_details_0]")[0].contents[0].replace_with("Foo")
|
|
doc.select("[name=total_quota]")[0]['value'] = "120"
|
|
doc.select("[name=form-TOTAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-INITIAL_FORMS]")[0]['value'] = "2"
|
|
doc.select("[name=form-MIN_NUM_FORMS]")[0]['value'] = "0"
|
|
doc.select("[name=form-MAX_NUM_FORMS]")[0]['value'] = "1000"
|
|
doc.select("[name=form-0-name_0]")[0]['value'] = "Normal ticket"
|
|
doc.select("[name=form-0-default_price]")[0]['value'] = "13.90"
|
|
doc.select("[name=form-0-quota]")[0]['value'] = "100"
|
|
doc.select("[name=form-1-name_0]")[0]['value'] = "Reduced ticket"
|
|
doc.select("[name=form-1-default_price]")[0]['value'] = "13.20"
|
|
doc.select("[name=form-1-quota]")[0]['value'] = "50"
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/quickstart/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert len(doc.select(".alert-success")) > 0
|
|
self.event1.refresh_from_db()
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.show_quota_left
|
|
assert self.event1.settings.contact_mail == "test@example.org"
|
|
assert self.event1.settings.ticket_download
|
|
assert self.event1.settings.ticketoutput_pdf__enabled
|
|
assert self.event1.settings.payment_banktransfer__enabled
|
|
assert self.event1.settings.get('payment_banktransfer_bank_details', as_type=LazyI18nString).localize('en') == "Foo"
|
|
assert 'pretix.plugins.banktransfer' in self.event1.plugins
|
|
with scopes_disabled():
|
|
assert self.event1.items.count() == 2
|
|
i = self.event1.items.first()
|
|
assert str(i.name) == "Normal ticket"
|
|
assert i.default_price == Decimal('13.90')
|
|
i = self.event1.items.last()
|
|
assert str(i.name) == "Reduced ticket"
|
|
assert i.default_price == Decimal('13.20')
|
|
assert self.event1.quotas.count() == 3
|
|
q = self.event1.quotas.first()
|
|
assert q.name == 'Normal ticket'
|
|
assert q.size == 100
|
|
assert q.items.count() == 1
|
|
q = self.event1.quotas.last()
|
|
assert q.name == 'Tickets'
|
|
assert q.size == 120
|
|
assert q.items.count() == 2
|
|
|
|
def test_settings(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
doc.select("[name=date_to_0]")[0]['value'] = "2013-12-30"
|
|
doc.select("[name=date_to_1]")[0]['value'] = "17:00:00"
|
|
doc.select("[name=settings-max_items_per_order]")[0]['value'] = "12"
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert len(doc.select(".alert-success")) > 0
|
|
assert doc.select("[name=date_to_0]")[0]['value'] == "2013-12-30"
|
|
assert doc.select("[name=date_to_1]")[0]['value'] == "17:00:00"
|
|
assert doc.select("[name=settings-max_items_per_order]")[0]['value'] == "12"
|
|
|
|
def test_unchanged_settings_do_not_create_logentry(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert not LogEntry.objects.exists()
|
|
|
|
def test_settings_timezone(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
doc.select("[name=date_to_0]")[0]['value'] = "2013-12-30"
|
|
doc.select("[name=date_to_1]")[0]['value'] = "17:00:00"
|
|
doc.select("[name=settings-max_items_per_order]")[0]['value'] = "12"
|
|
doc.select("[name=settings-timezone]")[0]['value'] = "Asia/Tokyo"
|
|
doc.find('option', {"value": "Asia/Tokyo"})['selected'] = 'selected'
|
|
doc.find('option', {"value": "UTC"}).attrs.pop('selected')
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
extract_form_fields(doc.select('.container-fluid form')[0]))
|
|
assert len(doc.select(".alert-success")) > 0
|
|
# date_to should not be changed even though the timezone is changed
|
|
assert doc.select("[name=date_to_0]")[0]['value'] == "2013-12-30"
|
|
assert doc.select("[name=date_to_1]")[0]['value'] == "17:00:00"
|
|
assert 'selected' in doc.find('option', {"value": "Asia/Tokyo"}).attrs
|
|
assert doc.select("[name=settings-max_items_per_order]")[0]['value'] == "12"
|
|
|
|
self.event1.refresh_from_db()
|
|
# Asia/Tokyo -> GMT+9
|
|
assert self.event1.date_to.strftime('%Y-%m-%d %H:%M:%S') == "2013-12-30 08:00:00"
|
|
assert self.event1.settings.timezone == 'Asia/Tokyo'
|
|
|
|
def test_plugins(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug))
|
|
self.assertIn("Stripe", doc.select(".form-plugins")[0].text)
|
|
self.assertIn("Enable", doc.select("[name=\"plugin:pretix.plugins.stripe\"]")[0].text)
|
|
assert not doc.select("[name=\"plugin:tests.testdummyrestricted\"]")
|
|
assert not doc.select("[name=\"plugin:tests.testdummyhidden\"]")
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:pretix.plugins.stripe': 'enable'})
|
|
self.assertIn("Disable", doc.select("[name=\"plugin:pretix.plugins.stripe\"]")[0].text)
|
|
|
|
doc = self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:pretix.plugins.stripe': 'disable'})
|
|
self.assertIn("Enable", doc.select("[name=\"plugin:pretix.plugins.stripe\"]")[0].text)
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyhidden': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "testdummyhidden" not in self.event1.plugins
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyrestricted': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "testdummyrestricted" not in self.event1.plugins
|
|
|
|
self.orga1.settings.allowed_restricted_plugins = ["tests.testdummyrestricted"]
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyrestricted': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "testdummyrestricted" in self.event1.plugins
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyorga': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "testdummyorga" not in self.event1.plugins
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyhybrid': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "tests.testdummyhybrid" not in self.event1.plugins
|
|
self.orga1.refresh_from_db()
|
|
assert "tests.testdummyhybrid" not in self.orga1.plugins
|
|
|
|
t2 = Team.objects.create(organizer=self.orga1, can_change_organizer_settings=True)
|
|
t2.members.add(self.user)
|
|
|
|
self.post_doc('/control/event/%s/%s/settings/plugins' % (self.orga1.slug, self.event1.slug),
|
|
{'plugin:tests.testdummyhybrid': 'enable'})
|
|
self.event1.refresh_from_db()
|
|
assert "tests.testdummyhybrid" in self.event1.plugins
|
|
self.orga1.refresh_from_db()
|
|
assert "tests.testdummyhybrid" in self.orga1.plugins
|
|
|
|
def test_testmode_enable(self):
|
|
self.event1.testmode = False
|
|
self.event1.save()
|
|
self.post_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug),
|
|
{'testmode': 'true'})
|
|
self.event1.refresh_from_db()
|
|
assert self.event1.testmode
|
|
|
|
def test_testmode_disable(self):
|
|
with scopes_disabled():
|
|
o = Order.objects.create(
|
|
code='FOO', event=self.event1, email='dummy@dummy.test',
|
|
status=Order.STATUS_PENDING,
|
|
datetime=now(), expires=now() + datetime.timedelta(days=10),
|
|
total=14, locale='en', testmode=True,
|
|
sales_channel=self.event1.organizer.sales_channels.get(identifier="web"),
|
|
)
|
|
o2 = Order.objects.create(
|
|
code='FOO2', event=self.event1, email='dummy@dummy.test',
|
|
status=Order.STATUS_PENDING,
|
|
datetime=now(), expires=now() + datetime.timedelta(days=10),
|
|
total=14, locale='en',
|
|
sales_channel=self.event1.organizer.sales_channels.get(identifier="web"),
|
|
)
|
|
self.event1.testmode = True
|
|
self.event1.save()
|
|
self.post_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug),
|
|
{'testmode': 'false'})
|
|
self.event1.refresh_from_db()
|
|
assert not self.event1.testmode
|
|
with scopes_disabled():
|
|
assert Order.objects.filter(pk=o.pk).exists()
|
|
assert Order.objects.filter(pk=o2.pk).exists()
|
|
|
|
def test_testmode_disable_delete(self):
|
|
with scopes_disabled():
|
|
o = Order.objects.create(
|
|
code='FOO', event=self.event1, email='dummy@dummy.test',
|
|
status=Order.STATUS_PENDING,
|
|
datetime=now(), expires=now() + datetime.timedelta(days=10),
|
|
total=14, locale='en', testmode=True,
|
|
sales_channel=self.event1.organizer.sales_channels.get(identifier="web"),
|
|
)
|
|
o2 = Order.objects.create(
|
|
code='FOO2', event=self.event1, email='dummy@dummy.test',
|
|
status=Order.STATUS_PENDING,
|
|
datetime=now(), expires=now() + datetime.timedelta(days=10),
|
|
total=14, locale='en',
|
|
sales_channel=self.event1.organizer.sales_channels.get(identifier="web"),
|
|
)
|
|
self.event1.testmode = True
|
|
self.event1.save()
|
|
self.post_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug),
|
|
{'testmode': 'false', 'delete': 'yes'})
|
|
self.event1.refresh_from_db()
|
|
assert not self.event1.testmode
|
|
with scopes_disabled():
|
|
assert not Order.objects.filter(pk=o.pk).exists()
|
|
assert Order.objects.filter(pk=o2.pk).exists()
|
|
|
|
def test_live_disable(self):
|
|
self.event1.live = True
|
|
self.event1.save()
|
|
self.post_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug),
|
|
{'live': 'false'})
|
|
self.event1.refresh_from_db()
|
|
assert not self.event1.live
|
|
|
|
def test_live_ok(self):
|
|
with scopes_disabled():
|
|
self.event1.items.create(name='Test', default_price=5)
|
|
self.event1.settings.set('payment_banktransfer__enabled', True)
|
|
self.event1.quotas.create(name='Test quota')
|
|
doc = self.get_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug))
|
|
assert len(doc.select("input[name=live]"))
|
|
self.post_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug),
|
|
{'live': 'true'})
|
|
self.event1.refresh_from_db()
|
|
assert self.event1.live
|
|
|
|
def test_live_dont_require_payment_method_free(self):
|
|
with scopes_disabled():
|
|
self.event1.items.create(name='Test', default_price=0)
|
|
self.event1.settings.set('payment_banktransfer__enabled', False)
|
|
self.event1.quotas.create(name='Test quota')
|
|
doc = self.get_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug))
|
|
assert len(doc.select("input[name=live]"))
|
|
|
|
def test_live_require_payment_method(self):
|
|
with scopes_disabled():
|
|
self.event1.items.create(name='Test', default_price=5)
|
|
self.event1.settings.set('payment_banktransfer__enabled', False)
|
|
self.event1.quotas.create(name='Test quota')
|
|
doc = self.get_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug))
|
|
assert len(doc.select("input[name=live]")) == 0
|
|
|
|
def test_live_require_a_quota(self):
|
|
self.event1.settings.set('payment_banktransfer__enabled', True)
|
|
doc = self.get_doc('/control/event/%s/%s/live/' % (self.orga1.slug, self.event1.slug))
|
|
assert len(doc.select("input[name=live]")) == 0
|
|
|
|
def test_payment_settings_provider(self):
|
|
self.get_doc('/control/event/%s/%s/settings/payment/banktransfer' % (self.orga1.slug, self.event1.slug))
|
|
self.post_doc('/control/event/%s/%s/settings/payment/banktransfer' % (self.orga1.slug, self.event1.slug), {
|
|
'payment_banktransfer__enabled': 'true',
|
|
'payment_banktransfer_ack': 'true',
|
|
'payment_banktransfer__fee_abs': '12.23',
|
|
'payment_banktransfer_bank_details_type': 'other',
|
|
'payment_banktransfer_bank_details_0': 'Test',
|
|
'payment_banktransfer__restrict_to_sales_channels': ['web'],
|
|
})
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.get('payment_banktransfer__enabled', as_type=bool)
|
|
assert self.event1.settings.get('payment_banktransfer__fee_abs', as_type=Decimal) == Decimal('12.23')
|
|
|
|
def test_payment_settings(self):
|
|
self.get_doc('/control/event/%s/%s/settings/payment' % (self.orga1.slug, self.event1.slug))
|
|
self.post_doc('/control/event/%s/%s/settings/payment' % (self.orga1.slug, self.event1.slug), {
|
|
'payment_term_days': '2',
|
|
'payment_term_minutes': '30',
|
|
'payment_term_mode': 'days',
|
|
'tax_rule_payment': 'default',
|
|
})
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.get('payment_term_days', as_type=int) == 2
|
|
|
|
def test_payment_settings_last_date_payment_after_presale_end(self):
|
|
self.event1.presale_end = now()
|
|
self.event1.save(update_fields=['presale_end'])
|
|
doc = self.post_doc('/control/event/%s/%s/settings/payment' % (self.orga1.slug, self.event1.slug), {
|
|
'payment_term_days': '2',
|
|
'payment_term_last_0': 'absolute',
|
|
'payment_term_last_1': (self.event1.presale_end - datetime.timedelta(1)).strftime('%Y-%m-%d'),
|
|
'payment_term_last_2': '0',
|
|
'payment_term_last_3': 'date_from',
|
|
'tax_rule_payment': 'default',
|
|
})
|
|
assert doc.select('.alert-danger')
|
|
self.event1.presale_end = None
|
|
self.event1.save(update_fields=['presale_end'])
|
|
|
|
def test_payment_settings_relative_date_payment_after_presale_end(self):
|
|
self.event1.presale_end = self.event1.date_from - datetime.timedelta(days=5)
|
|
self.event1.save(update_fields=['presale_end'])
|
|
doc = self.post_doc('/control/event/%s/%s/settings/payment' % (self.orga1.slug, self.event1.slug), {
|
|
'payment_term_days': '2',
|
|
'payment_term_last_0': 'relative',
|
|
'payment_term_last_1': '',
|
|
'payment_term_last_2': '10',
|
|
'payment_term_last_3': 'date_from',
|
|
'tax_rule_payment': 'default',
|
|
})
|
|
assert doc.select('.alert-danger')
|
|
self.event1.presale_end = None
|
|
self.event1.save(update_fields=['presale_end'])
|
|
|
|
def test_invoice_settings(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/invoice' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['invoice_address_required'] = 'on'
|
|
doc = self.post_doc('/control/event/%s/%s/settings/invoice' % (self.orga1.slug, self.event1.slug),
|
|
data, follow=True)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.get('invoice_address_required', as_type=bool)
|
|
|
|
def test_display_settings(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['settings-primary_color'] = '#000000'
|
|
doc = self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
data, follow=True)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.get('primary_color') == '#000000'
|
|
|
|
def test_display_settings_do_not_override_parent(self):
|
|
self.orga1.settings.primary_color = '#000000'
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
doc = self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
data, follow=True)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert 'primary_color' not in self.event1.settings._cache()
|
|
assert self.event1.settings.primary_color == self.orga1.settings.primary_color
|
|
|
|
def test_display_settings_explicitly_override_parent(self):
|
|
self.orga1.settings.primary_color = '#000000'
|
|
|
|
doc = self.get_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['decouple'] = 'primary_color'
|
|
doc = self.post_doc('/control/event/%s/%s/settings/' % (self.orga1.slug, self.event1.slug),
|
|
data, follow=True)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert 'primary_color' in self.event1.settings._cache()
|
|
assert self.event1.settings.primary_color == self.orga1.settings.primary_color
|
|
|
|
def test_email_settings(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/email' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['mail_from_name'] = 'test'
|
|
doc = self.post_doc('/control/event/%s/%s/settings/email' % (self.orga1.slug, self.event1.slug),
|
|
data, follow=True)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.mail_from_name == "test"
|
|
|
|
def test_email_setup_system(self):
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'system'
|
|
},
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert "mail_from" not in self.orga1.settings._cache()
|
|
assert not self.event1.settings.smtp_use_custom
|
|
|
|
@override_settings(MAIL_CUSTOM_SENDER_VERIFICATION_REQUIRED=True, MAIL_CUSTOM_SENDER_SPF_STRING=False)
|
|
def test_email_setup_simple_with_verification(self):
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'simple',
|
|
'simple-mail_from': 'test@test.pretix.dev',
|
|
},
|
|
follow=True
|
|
)
|
|
self.event1.settings.flush()
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['verification'] = self.client.session[
|
|
f'sender_mail_verification_code_/control/event/{self.orga1.slug}/{self.event1.slug}/settings/email/setup_test@test.pretix.dev'
|
|
]
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
data,
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.mail_from == 'test@test.pretix.dev'
|
|
|
|
@override_settings(MAIL_CUSTOM_SENDER_VERIFICATION_REQUIRED=True, MAIL_CUSTOM_SENDER_SPF_STRING=False)
|
|
def test_email_setup_simple_with_verification_wrong_code(self):
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'simple',
|
|
'simple-mail_from': 'test@test.pretix.dev',
|
|
},
|
|
follow=True
|
|
)
|
|
self.event1.settings.flush()
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['verification'] = 'AAAA'
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
data,
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-danger')
|
|
self.event1.settings.flush()
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
|
|
@staticmethod
|
|
def _fake_spf_record(hostname):
|
|
return {
|
|
'test.pretix.dev': 'v=spf1 redirect=_spf.pretix.dev',
|
|
'_spf.pretix.dev': 'v=spf1 a mx include:level2.pretix.dev ~all',
|
|
'level2.pretix.dev': 'v=spf1 a mx +include:level3.pretix.dev include:spftest.pretix.dev '
|
|
'-include:level4.pretix.dev ~all',
|
|
'level3.pretix.dev': 'v=spf1 a mx include:test2.pretix.dev ~all',
|
|
'level4.pretix.dev': 'v=spf1 a mx include:test3.pretix.dev ~all',
|
|
'test2.pretix.dev': None,
|
|
'test3.pretix.dev': None,
|
|
'spftest.pretix.dev': None,
|
|
}[hostname]
|
|
|
|
@override_settings(MAIL_CUSTOM_SENDER_VERIFICATION_REQUIRED=False, MAIL_CUSTOM_SENDER_SPF_STRING="include:spftest.pretix.dev include:test2.pretix.dev")
|
|
def test_email_setup_no_verification_spf_success(self):
|
|
self.monkeypatch.setattr("pretix.control.views.mailsetup.get_spf_record", EventsTest._fake_spf_record)
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'simple',
|
|
'simple-mail_from': 'test@test.pretix.dev',
|
|
},
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
# not yet saved
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
data,
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.mail_from == 'test@test.pretix.dev'
|
|
|
|
@override_settings(MAIL_CUSTOM_SENDER_VERIFICATION_REQUIRED=False, MAIL_CUSTOM_SENDER_SPF_STRING="include:spftest.pretix.dev include:test3.pretix.dev")
|
|
def test_email_setup_no_verification_spf_warning(self):
|
|
self.monkeypatch.setattr("pretix.control.views.mailsetup.get_spf_record", EventsTest._fake_spf_record)
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'simple',
|
|
'simple-mail_from': 'test@test.pretix.dev',
|
|
},
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-danger')
|
|
self.event1.settings.flush()
|
|
# not yet saved
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
|
|
def test_email_setup_smtp(self):
|
|
self.monkeypatch.setattr("pretix.base.email.test_custom_smtp_backend", lambda b, a: None)
|
|
self.monkeypatch.setattr("socket.gethostbyname", lambda h: "8.8.8.8")
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'smtp',
|
|
'smtp-mail_from': 'test@test.pretix.dev',
|
|
'smtp-smtp_host': 'test.pretix.dev',
|
|
'smtp-smtp_port': '587',
|
|
},
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
# not yet saved
|
|
self.event1.settings.flush()
|
|
assert "smtp_use_custom" not in self.event1.settings._cache()
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
data,
|
|
follow=True
|
|
)
|
|
assert doc.select('.alert-success')
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.mail_from == 'test@test.pretix.dev'
|
|
assert self.event1.settings.smtp_host == 'test.pretix.dev'
|
|
assert self.event1.settings.smtp_port == 587
|
|
assert self.event1.settings.smtp_use_custom
|
|
|
|
def test_email_setup_smtp_failure(self):
|
|
def fail(a, b):
|
|
raise SMTPResponseException(400, 'Auth denied')
|
|
self.monkeypatch.setattr("pretix.base.email.test_custom_smtp_backend", fail)
|
|
self.monkeypatch.setattr("socket.gethostbyname", lambda h: "8.8.8.8")
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'smtp',
|
|
'smtp-mail_from': 'test@test.pretix.dev',
|
|
'smtp-smtp_host': 'test.pretix.dev',
|
|
'smtp-smtp_port': '587',
|
|
},
|
|
follow=True
|
|
)
|
|
assert 'Auth denied' in doc.select('.alert-danger')[0].text
|
|
# not yet saved
|
|
self.event1.settings.flush()
|
|
assert "smtp_use_custom" not in self.event1.settings._cache()
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
|
|
def test_email_setup_do_not_allow_private_ip_by_default(self):
|
|
doc = self.post_doc(
|
|
'/control/event/%s/%s/settings/email/setup' % (self.orga1.slug, self.event1.slug),
|
|
{
|
|
'mode': 'simple',
|
|
'smtp-mail_from': 'test@test.pretix.dev',
|
|
'smtp-smtp_host': '10.0.1.1',
|
|
'smtp-smtp_port': '587',
|
|
},
|
|
follow=True
|
|
)
|
|
assert doc.select('.has-error')
|
|
# not yet saved
|
|
self.event1.settings.flush()
|
|
assert "smtp_use_custom" not in self.event1.settings._cache()
|
|
assert "mail_from" not in self.event1.settings._cache()
|
|
|
|
def test_ticket_settings(self):
|
|
doc = self.get_doc('/control/event/%s/%s/settings/tickets' % (self.orga1.slug, self.event1.slug))
|
|
data = extract_form_fields(doc.select("form")[0])
|
|
data['ticket_download'] = 'on'
|
|
data['ticketoutput_testdummy__enabled'] = 'on'
|
|
self.post_doc('/control/event/%s/%s/settings/tickets' % (self.orga1.slug, self.event1.slug), data, follow=True)
|
|
self.event1.settings.flush()
|
|
assert self.event1.settings.get('ticket_download', as_type=bool)
|
|
|
|
def test_create_event_unauthorized(self):
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga2.pk,
|
|
'foundation-locales': ('en', 'de')
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
def test_create_invalid_default_language(self):
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('de',)
|
|
})
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from': '2016-12-27 10:00:00',
|
|
'basics-date_to': '2016-12-30 19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start': '2016-11-01 10:00:00',
|
|
'basics-presale_end': '2016-11-30 18:00:00',
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
def test_create_duplicate_slug(self):
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'event_wizard-current_step': 'foundation',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('de', 'en')
|
|
})
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '31c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
def test_create_event_success(self):
|
|
doc = self.get_doc('/control/events/add')
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('en', 'de')
|
|
})
|
|
assert doc.select("#id_basics-name_0")
|
|
assert doc.select("#id_basics-name_1")
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '19.00',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
'basics-team': '',
|
|
})
|
|
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'copy',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'copy-copy_from_event': ''
|
|
})
|
|
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.name == LazyI18nString({'de': '33C3', 'en': '33C3'})
|
|
assert ev.settings.locales == ['en', 'de']
|
|
assert ev.settings.locale == 'en'
|
|
assert ev.currency == 'EUR'
|
|
assert ev.settings.timezone == 'Europe/Berlin'
|
|
assert ev.organizer == self.orga1
|
|
assert ev.location == LazyI18nString({'de': 'Hamburg', 'en': 'Hamburg'})
|
|
assert Team.objects.filter(limit_events=ev, members=self.user).exists()
|
|
|
|
berlin_tz = ZoneInfo('Europe/Berlin')
|
|
assert ev.date_from == datetime.datetime(2016, 12, 27, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.date_to == datetime.datetime(2016, 12, 30, 19, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_start == datetime.datetime(2016, 11, 1, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_end == datetime.datetime(2016, 11, 30, 18, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
|
|
assert ev.tax_rules.filter(rate=Decimal('19.00')).exists()
|
|
|
|
def test_create_event_with_subevents_success(self):
|
|
doc = self.get_doc('/control/events/add')
|
|
tabletext = doc.select("form")[0].text
|
|
self.assertIn("CCC", tabletext)
|
|
self.assertNotIn("MRM", tabletext)
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'event_wizard-current_step': 'foundation',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('en', 'de'),
|
|
'foundation-has_subevents': 'on',
|
|
})
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
'basics-team': '',
|
|
})
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'copy',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'copy-copy_from_event': ''
|
|
})
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.has_subevents
|
|
assert ev.subevents.count() == 0
|
|
|
|
def test_create_event_copy_success(self):
|
|
with scopes_disabled():
|
|
tr = self.event1.tax_rules.create(
|
|
rate=19, name="VAT", default=True
|
|
)
|
|
q1 = self.event1.quotas.create(
|
|
name='Foo',
|
|
size=0,
|
|
)
|
|
self.event1.items.create(
|
|
name='Early-bird ticket',
|
|
category=None, default_price=23, tax_rule=tr,
|
|
admission=True, hidden_if_available=q1
|
|
)
|
|
doc = self.get_doc('/control/events/add')
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('en', 'de')
|
|
})
|
|
assert doc.select("#id_basics-name_0")
|
|
assert doc.select("#id_basics-name_1")
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '19.00',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
})
|
|
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'copy',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'copy-copy_from_event': self.event1.pk
|
|
})
|
|
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.name == LazyI18nString({'de': '33C3', 'en': '33C3'})
|
|
assert ev.settings.locales == ['en', 'de']
|
|
assert ev.settings.locale == 'en'
|
|
assert ev.currency == 'EUR'
|
|
assert ev.settings.timezone == 'Europe/Berlin'
|
|
assert ev.organizer == self.orga1
|
|
assert ev.location == LazyI18nString({'de': 'Hamburg', 'en': 'Hamburg'})
|
|
assert Team.objects.filter(limit_events=ev, members=self.user).exists()
|
|
assert ev.items.count() == 1
|
|
|
|
berlin_tz = ZoneInfo('Europe/Berlin')
|
|
assert ev.date_from == datetime.datetime(2016, 12, 27, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.date_to == datetime.datetime(2016, 12, 30, 19, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_start == datetime.datetime(2016, 11, 1, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_end == datetime.datetime(2016, 11, 30, 18, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
|
|
assert ev.tax_rules.filter(rate=Decimal('19.00')).count() == 1
|
|
i = ev.items.get()
|
|
assert i.hidden_if_available.name == "Foo"
|
|
assert i.hidden_if_available.event == ev
|
|
assert i.hidden_if_available.pk != q1.pk
|
|
|
|
def test_create_event_clone_success(self):
|
|
with scopes_disabled():
|
|
tr = self.event1.tax_rules.create(
|
|
rate=19, name="VAT", default=True
|
|
)
|
|
self.event1.items.create(
|
|
name='Early-bird ticket',
|
|
category=None, default_price=23, tax_rule=tr,
|
|
admission=True
|
|
)
|
|
doc = self.get_doc('/control/events/add?clone=' + str(self.event1.pk))
|
|
tabletext = doc.select("form")[0].text
|
|
self.assertIn("CCC", tabletext)
|
|
self.assertNotIn("MRM", tabletext)
|
|
|
|
doc = self.post_doc('/control/events/add?clone=' + str(self.event1.pk), {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': ('en', 'de')
|
|
})
|
|
assert doc.select("#id_basics-date_from_0")[0]['value'] == '2013-12-26'
|
|
|
|
doc = self.post_doc('/control/events/add?clone=' + str(self.event1.pk), {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-name_1': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-location_1': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '19.00',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
'basics-team': '',
|
|
})
|
|
|
|
assert not doc.select("#id_copy-copy_from_event_1")
|
|
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.name == LazyI18nString({'de': '33C3', 'en': '33C3'})
|
|
assert ev.settings.locales == ['en', 'de']
|
|
assert ev.settings.locale == 'en'
|
|
assert ev.currency == 'EUR'
|
|
assert ev.settings.timezone == 'Europe/Berlin'
|
|
assert ev.organizer == self.orga1
|
|
assert ev.location == LazyI18nString({'de': 'Hamburg', 'en': 'Hamburg'})
|
|
assert Team.objects.filter(limit_events=ev, members=self.user).exists()
|
|
assert ev.items.count() == 1
|
|
|
|
berlin_tz = ZoneInfo('Europe/Berlin')
|
|
assert ev.date_from == datetime.datetime(2016, 12, 27, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.date_to == datetime.datetime(2016, 12, 30, 19, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_start == datetime.datetime(2016, 11, 1, 10, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
assert ev.presale_end == datetime.datetime(2016, 11, 30, 18, 0, 0, tzinfo=berlin_tz).astimezone(datetime.timezone.utc)
|
|
|
|
assert ev.tax_rules.filter(rate=Decimal('19.00')).count() == 1
|
|
|
|
def test_create_event_only_date_from(self):
|
|
# date_to, presale_start & presale_end are optional fields
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': 'en'
|
|
})
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '',
|
|
'basics-date_to_1': '',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'UTC',
|
|
'basics-presale_start_0': '',
|
|
'basics-presale_start_1': '',
|
|
'basics-presale_end_0': '',
|
|
'basics-presale_end_1': '',
|
|
'basics-team': '',
|
|
})
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'copy',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'copy-copy_from_event': ''
|
|
})
|
|
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.name == LazyI18nString({'en': '33C3'})
|
|
assert ev.settings.locales == ['en']
|
|
assert ev.settings.locale == 'en'
|
|
assert ev.currency == 'EUR'
|
|
assert ev.settings.timezone == 'UTC'
|
|
assert ev.organizer == self.orga1
|
|
assert ev.location == LazyI18nString({'en': 'Hamburg'})
|
|
assert Team.objects.filter(limit_events=ev, members=self.user).exists()
|
|
assert ev.date_from == datetime.datetime(2016, 12, 27, 10, 0, 0, tzinfo=datetime.timezone.utc)
|
|
assert ev.date_to is None
|
|
assert ev.presale_start is None
|
|
assert ev.presale_end is None
|
|
|
|
def test_create_event_existing_team(self):
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': 'en'
|
|
})
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '',
|
|
'basics-date_to_1': '',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'UTC',
|
|
'basics-presale_start_0': '',
|
|
'basics-presale_start_1': '',
|
|
'basics-presale_end_0': '',
|
|
'basics-presale_end_1': '',
|
|
'basics-team': str(self.team2.pk),
|
|
})
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'copy',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'copy-copy_from_event': ''
|
|
})
|
|
|
|
with scopes_disabled():
|
|
ev = Event.objects.get(slug='33c3')
|
|
assert ev.name == LazyI18nString({'en': '33C3'})
|
|
assert ev.settings.locales == ['en']
|
|
assert ev.settings.locale == 'en'
|
|
assert ev.currency == 'EUR'
|
|
assert ev.settings.timezone == 'UTC'
|
|
assert ev.organizer == self.orga1
|
|
assert ev.location == LazyI18nString({'en': 'Hamburg'})
|
|
team = Team.objects.filter(limit_events=ev, members=self.user).first()
|
|
assert team == self.team2
|
|
assert ev.date_from == datetime.datetime(2016, 12, 27, 10, 0, 0, tzinfo=datetime.timezone.utc)
|
|
assert ev.date_to is None
|
|
assert ev.presale_start is None
|
|
assert ev.presale_end is None
|
|
|
|
def test_create_event_missing_date_from(self):
|
|
# date_from is mandatory
|
|
self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': 'en'
|
|
})
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-slug': '33c3',
|
|
'basics-date_from_0': '',
|
|
'basics-date_from_1': '',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-currency': 'EUR',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
def test_create_event_currency_symbol(self):
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': 'en'
|
|
})
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-slug': '31c4',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-currency': '$',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
def test_create_event_non_iso_currency(self):
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'foundation',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'foundation-organizer': self.orga1.pk,
|
|
'foundation-locales': 'en'
|
|
})
|
|
|
|
doc = self.post_doc('/control/events/add', {
|
|
'event_wizard-current_step': 'basics',
|
|
'event_wizard-prefix': 'event_wizard',
|
|
'basics-name_0': '33C3',
|
|
'basics-slug': '31c5',
|
|
'basics-date_from_0': '2016-12-27',
|
|
'basics-date_from_1': '10:00:00',
|
|
'basics-date_to_0': '2016-12-30',
|
|
'basics-date_to_1': '19:00:00',
|
|
'basics-location_0': 'Hamburg',
|
|
'basics-currency': 'ASD',
|
|
'basics-tax_rate': '',
|
|
'basics-no_taxes': 'on',
|
|
'basics-locale': 'en',
|
|
'basics-timezone': 'Europe/Berlin',
|
|
'basics-presale_start_0': '2016-11-01',
|
|
'basics-presale_start_1': '10:00:00',
|
|
'basics-presale_end_0': '2016-11-30',
|
|
'basics-presale_end_1': '18:00:00',
|
|
})
|
|
assert doc.select(".has-error")
|
|
|
|
|
|
class EventDeletionTest(SoupTest):
|
|
@scopes_disabled()
|
|
def setUp(self):
|
|
super().setUp()
|
|
self.user = User.objects.create_user('dummy@dummy.dummy', 'dummy')
|
|
self.orga1 = Organizer.objects.create(name='CCC', slug='ccc')
|
|
self.event1 = Event.objects.create(
|
|
organizer=self.orga1, name='30C3', slug='30c3',
|
|
date_from=datetime.datetime(2013, 12, 26, tzinfo=datetime.timezone.utc),
|
|
plugins='pretix.plugins.banktransfer,tests.testdummy',
|
|
has_subevents=False
|
|
)
|
|
|
|
t = Team.objects.create(organizer=self.orga1, can_create_events=True, can_change_event_settings=True,
|
|
can_change_items=True)
|
|
t.members.add(self.user)
|
|
t.limit_events.add(self.event1)
|
|
self.ticket = self.event1.items.create(name='Early-bird ticket',
|
|
category=None, default_price=23,
|
|
admission=True)
|
|
|
|
self.client.login(email='dummy@dummy.dummy', password='dummy')
|
|
|
|
def test_delete_allowed(self):
|
|
session = self.client.session
|
|
session['pretix_auth_login_time'] = int(time.time())
|
|
session.save()
|
|
self.client.post('/control/event/ccc/30c3/delete/', {
|
|
'slug': '30c3'
|
|
})
|
|
|
|
with scopes_disabled():
|
|
assert not self.orga1.events.exists()
|
|
|
|
def test_delete_wrong_slug(self):
|
|
self.post_doc('/control/event/ccc/30c3/delete/', {
|
|
'user_pw': 'dummy',
|
|
'slug': '31c3'
|
|
})
|
|
with scopes_disabled():
|
|
assert self.orga1.events.exists()
|
|
|
|
def test_delete_wrong_pw(self):
|
|
self.post_doc('/control/event/ccc/30c3/delete/', {
|
|
'user_pw': 'invalid',
|
|
'slug': '30c3'
|
|
})
|
|
with scopes_disabled():
|
|
assert self.orga1.events.exists()
|
|
|
|
def test_delete_orders(self):
|
|
with scopes_disabled():
|
|
Order.objects.create(
|
|
code='FOO', event=self.event1, email='dummy@dummy.test',
|
|
status=Order.STATUS_PENDING,
|
|
datetime=now(), expires=now(),
|
|
total=14, locale='en',
|
|
sales_channel=self.event1.organizer.sales_channels.get(identifier="web"),
|
|
)
|
|
self.post_doc('/control/event/ccc/30c3/delete/', {
|
|
'user_pw': 'dummy',
|
|
'slug': '30c3'
|
|
})
|
|
with scopes_disabled():
|
|
assert self.orga1.events.exists()
|