mirror of
https://github.com/pretix/pretix.git
synced 2026-05-03 14:54:04 +00:00
* Add data shredders for PII * First working shredder * Add more shredders * Add new shredders and download confirmation * tmp * PayPal, Stripe, banktransfer * Add icon to logs * Untested payment log shredders * Add waiting list shredder * First tests * Add tests for shredders * Improve templats, link to shredder * Test payment info shredders * More tests * Documentation * Fix enabled flag in payment provider overview * Fix minor issues
This commit is contained in:
355
src/tests/base/test_shredders.py
Normal file
355
src/tests/base/test_shredders.py
Normal file
@@ -0,0 +1,355 @@
|
||||
import json
|
||||
import os
|
||||
from datetime import timedelta
|
||||
from decimal import Decimal
|
||||
|
||||
import pytest
|
||||
from django.core.files.base import ContentFile
|
||||
from django.utils.timezone import now
|
||||
|
||||
from pretix.base.models import (
|
||||
CachedCombinedTicket, CachedTicket, Event, InvoiceAddress, Order,
|
||||
OrderPosition, Organizer, QuestionAnswer,
|
||||
)
|
||||
from pretix.base.services.invoices import generate_invoice, invoice_pdf_task
|
||||
from pretix.base.services.tickets import generate, generate_order
|
||||
from pretix.base.shredder import (
|
||||
AttendeeNameShredder, CachedTicketShredder, EmailAddressShredder,
|
||||
InvoiceAddressShredder, InvoiceShredder, PaymentInfoShredder,
|
||||
QuestionAnswerShredder, WaitingListShredder, shred_constraints,
|
||||
)
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def event():
|
||||
o = Organizer.objects.create(name='Dummy', slug='dummy')
|
||||
event = Event.objects.create(
|
||||
organizer=o, name='Dummy', slug='dummy',
|
||||
date_from=now(), plugins='pretix.plugins.banktransfer,pretix.plugins.ticketoutputpdf'
|
||||
)
|
||||
return event
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def item(event):
|
||||
return event.items.create(
|
||||
name='Early-bird ticket',
|
||||
category=None, default_price=23,
|
||||
admission=True
|
||||
)
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def order(event, item):
|
||||
o = Order.objects.create(
|
||||
code='FOO', event=event, email='dummy@dummy.test',
|
||||
status=Order.STATUS_PENDING,
|
||||
datetime=now(), expires=now() + timedelta(days=10),
|
||||
total=14, payment_provider='banktransfer', locale='en'
|
||||
)
|
||||
event.settings.set('attendee_names_asked', True)
|
||||
event.settings.set('locales', ['en', 'de'])
|
||||
OrderPosition.objects.create(
|
||||
order=o,
|
||||
item=item,
|
||||
variation=None,
|
||||
price=Decimal("14"),
|
||||
attendee_name="Peter",
|
||||
attendee_email="foo@example.org"
|
||||
)
|
||||
return o
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def question(event, item):
|
||||
q = event.questions.create(question="T-Shirt size", type="C", identifier="ABC")
|
||||
q.items.add(item)
|
||||
q.options.create(answer="XL", identifier="LVETRWVU")
|
||||
return q
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_email_shredder(event, order):
|
||||
l1 = order.log_action(
|
||||
'pretix.event.order.email.expired',
|
||||
data={
|
||||
'recipient': 'dummy@dummy.test',
|
||||
'message': 'Hello Peter@,',
|
||||
'subject': 'Foo'
|
||||
}
|
||||
)
|
||||
l2 = order.log_action(
|
||||
'pretix.event.order.contact.changed',
|
||||
data={
|
||||
'old_email': 'dummy@dummy.test',
|
||||
'new_email': 'foo@bar.com',
|
||||
}
|
||||
)
|
||||
|
||||
s = EmailAddressShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert json.loads(f[0][2]) == {
|
||||
order.code: 'dummy@dummy.test'
|
||||
}
|
||||
assert json.loads(f[1][2]) == {
|
||||
'{}-{}'.format(order.code, 1): 'foo@example.org'
|
||||
}
|
||||
s.shred_data()
|
||||
order.refresh_from_db()
|
||||
assert order.email is None
|
||||
assert order.positions.first().attendee_email is None
|
||||
l1.refresh_from_db()
|
||||
assert '@' not in l1.data
|
||||
assert 'Foo' not in l1.data
|
||||
l2.refresh_from_db()
|
||||
assert '@' not in l2.data
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_waitinglist_shredder(event, item):
|
||||
q = event.quotas.create(size=5)
|
||||
q.items.add(item)
|
||||
wle = event.waitinglistentries.create(
|
||||
item=item, email='foo@example.org'
|
||||
)
|
||||
wle.send_voucher()
|
||||
assert '@' in wle.voucher.comment
|
||||
assert '@' in wle.voucher.all_logentries().last().data
|
||||
s = WaitingListShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert json.loads(f[0][2]) == [
|
||||
{
|
||||
'id': wle.pk,
|
||||
'item': item.pk,
|
||||
'variation': None,
|
||||
'subevent': None,
|
||||
'voucher': wle.voucher.pk,
|
||||
'created': wle.created.isoformat().replace('+00:00', 'Z'),
|
||||
'locale': 'en',
|
||||
'email': 'foo@example.org'
|
||||
}
|
||||
]
|
||||
s.shred_data()
|
||||
wle.refresh_from_db()
|
||||
wle.voucher.refresh_from_db()
|
||||
assert '@' not in wle.email
|
||||
assert '@' not in wle.voucher.comment
|
||||
assert '@' not in wle.voucher.all_logentries().last().data
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_attendee_name_shredder(event, order):
|
||||
l1 = order.log_action(
|
||||
'pretix.event.order.modified',
|
||||
data={
|
||||
"data": [{"attendee_name": "Hans", "question_1": "Test"}],
|
||||
"invoice_data": {"name": "Foo"}
|
||||
}
|
||||
)
|
||||
|
||||
s = AttendeeNameShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert json.loads(f[0][2]) == {
|
||||
'{}-{}'.format(order.code, 1): 'Peter'
|
||||
}
|
||||
s.shred_data()
|
||||
order.refresh_from_db()
|
||||
assert order.positions.first().attendee_name is None
|
||||
l1.refresh_from_db()
|
||||
assert 'Hans' not in l1.data
|
||||
assert 'Foo' in l1.data
|
||||
assert 'Test' in l1.data
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_invoice_address_shredder(event, order):
|
||||
l1 = order.log_action(
|
||||
'pretix.event.order.modified',
|
||||
data={
|
||||
"data": [{"attendee_name": "Hans", "question_1": "Test"}],
|
||||
"invoice_data": {"name": "Peter", "country": "DE", "is_business": False, "internal_reference": "",
|
||||
"company": "ACME", "street": "Sesam Street", "city": "Sample City", "zipcode": "12345"}
|
||||
}
|
||||
)
|
||||
ia = InvoiceAddress.objects.create(company='Acme Company', street='221B Baker Street',
|
||||
zipcode='12345', city='London', country='UK',
|
||||
order=order)
|
||||
s = InvoiceAddressShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert json.loads(f[0][2]) == {
|
||||
order.code: {
|
||||
'city': 'London',
|
||||
'company': 'Acme Company',
|
||||
'country': 'UK',
|
||||
'internal_reference': '',
|
||||
'is_business': False,
|
||||
'last_modified': ia.last_modified.isoformat().replace('+00:00', 'Z'),
|
||||
'name': '',
|
||||
'street': '221B Baker Street',
|
||||
'vat_id': '',
|
||||
'vat_id_validated': False,
|
||||
'zipcode': '12345'
|
||||
}
|
||||
}
|
||||
s.shred_data()
|
||||
order.refresh_from_db()
|
||||
assert not InvoiceAddress.objects.filter(order=order).exists()
|
||||
l1.refresh_from_db()
|
||||
assert l1.parsed_data == {
|
||||
"data": [{"attendee_name": "Hans", "question_1": "Test"}],
|
||||
"invoice_data": {"name": "█", "country": "█", "is_business": False, "internal_reference": "", "company": "█",
|
||||
"street": "█", "city": "█", "zipcode": "█"}
|
||||
}
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_question_answer_shredder(event, order, question):
|
||||
opt = question.options.first()
|
||||
l1 = order.log_action(
|
||||
'pretix.event.order.modified',
|
||||
data={
|
||||
"data": [
|
||||
{
|
||||
"attendee_name": "Hans",
|
||||
"question_%d" % question.pk: [{"id": opt.pk, "type": "QuestionOption"}]
|
||||
}
|
||||
],
|
||||
}
|
||||
)
|
||||
qa = QuestionAnswer.objects.create(
|
||||
orderposition=order.positions.first(),
|
||||
question=question,
|
||||
answer='S'
|
||||
)
|
||||
qa.file.save('foo.pdf', ContentFile('foo'))
|
||||
fname = qa.file.path
|
||||
assert os.path.exists(fname)
|
||||
qa.options.add(opt)
|
||||
s = QuestionAnswerShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert json.loads(f[0][2]) == {
|
||||
'{}-1'.format(order.code): [{
|
||||
'question': question.pk,
|
||||
'answer': 'S',
|
||||
'question_identifier': question.identifier,
|
||||
'options': [opt.pk],
|
||||
'option_identifiers': [opt.identifier],
|
||||
}]
|
||||
}
|
||||
s.shred_data()
|
||||
order.refresh_from_db()
|
||||
assert not os.path.exists(fname)
|
||||
assert not QuestionAnswer.objects.filter(pk=qa.pk).exists()
|
||||
l1.refresh_from_db()
|
||||
assert l1.parsed_data == {
|
||||
"data": [{"attendee_name": "Hans", "question_%d" % question.pk: "█"}],
|
||||
}
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_invoice_shredder(event, order):
|
||||
InvoiceAddress.objects.create(company='Acme Company', street='221B Baker Street',
|
||||
zipcode='12345', city='London', country='UK',
|
||||
order=order)
|
||||
inv = generate_invoice(order)
|
||||
invoice_pdf_task.apply(args=(inv.pk,))
|
||||
inv.refresh_from_db()
|
||||
assert inv.invoice_to == "Acme Company\n\n221B Baker Street\n12345 London"
|
||||
assert inv.file
|
||||
fname = inv.file.path
|
||||
assert os.path.exists(fname)
|
||||
s = InvoiceShredder(event)
|
||||
f = list(s.generate_files())
|
||||
assert len(f) == 1
|
||||
s.shred_data()
|
||||
inv.refresh_from_db()
|
||||
|
||||
assert "Acme" not in inv.invoice_to
|
||||
assert "icket" not in inv.lines.first().description
|
||||
assert not inv.file
|
||||
assert not os.path.exists(fname)
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_cached_tickets(event, order):
|
||||
generate(order.positions.first().pk, 'pdf')
|
||||
generate_order(order.pk, 'pdf')
|
||||
|
||||
ct = CachedTicket.objects.get(order_position=order.positions.first(), provider='pdf')
|
||||
cct = CachedCombinedTicket.objects.get(order=order, provider='pdf')
|
||||
assert ct.file
|
||||
assert cct.file
|
||||
ct_fname = ct.file.path
|
||||
cct_fname = cct.file.path
|
||||
assert os.path.exists(ct_fname)
|
||||
assert os.path.exists(cct_fname)
|
||||
s = CachedTicketShredder(event)
|
||||
assert s.generate_files() is None
|
||||
s.shred_data()
|
||||
|
||||
assert not CachedTicket.objects.filter(order_position=order.positions.first(), provider='pdf').exists()
|
||||
assert not CachedCombinedTicket.objects.filter(order=order, provider='pdf').exists()
|
||||
assert not os.path.exists(ct_fname)
|
||||
assert not os.path.exists(cct_fname)
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_payment_info_shredder(event, order):
|
||||
order.payment_info = json.dumps({
|
||||
'reference': 'Verwendungszweck 1',
|
||||
'date': '2018-05-01',
|
||||
'payer': 'Hans',
|
||||
'trans_id': 12
|
||||
})
|
||||
order.save()
|
||||
|
||||
s = PaymentInfoShredder(event)
|
||||
assert s.generate_files() is None
|
||||
s.shred_data()
|
||||
|
||||
order.refresh_from_db()
|
||||
assert json.loads(order.payment_info) == {
|
||||
'_shredded': True,
|
||||
'reference': '█',
|
||||
'date': '2018-05-01',
|
||||
'payer': '█',
|
||||
'trans_id': 12
|
||||
}
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_shred_constraint_offline(event):
|
||||
event.live = True
|
||||
event.date_from = now() - timedelta(days=365)
|
||||
assert shred_constraints(event)
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_shred_constraint_60_days(event):
|
||||
event.live = False
|
||||
event.date_from = now() - timedelta(days=62)
|
||||
event.date_to = now() - timedelta(days=62)
|
||||
assert shred_constraints(event) is None
|
||||
event.date_from = now() - timedelta(days=52)
|
||||
event.date_to = now() - timedelta(days=52)
|
||||
assert shred_constraints(event)
|
||||
event.date_from = now() - timedelta(days=62)
|
||||
event.date_to = now() - timedelta(days=52)
|
||||
assert shred_constraints(event)
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
def test_shred_constraint_60_days_subevents(event):
|
||||
event.has_subevents = True
|
||||
event.live = False
|
||||
|
||||
event.subevents.create(
|
||||
date_from=now() - timedelta(days=62),
|
||||
date_to=now() - timedelta(days=62)
|
||||
)
|
||||
assert shred_constraints(event) is None
|
||||
event.subevents.create(
|
||||
date_from=now() - timedelta(days=62),
|
||||
date_to=now() - timedelta(days=52)
|
||||
)
|
||||
assert shred_constraints(event)
|
||||
166
src/tests/control/test_shredders.py
Normal file
166
src/tests/control/test_shredders.py
Normal file
@@ -0,0 +1,166 @@
|
||||
import datetime
|
||||
import json
|
||||
from io import BytesIO
|
||||
from zipfile import ZipFile
|
||||
|
||||
from django.utils.timezone import now
|
||||
from tests.base import SoupTest
|
||||
|
||||
from pretix.base.models import Event, Order, Organizer, Team, User
|
||||
|
||||
|
||||
class EventShredderTest(SoupTest):
|
||||
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,pretix.plugins.stripe,tests.testdummy'
|
||||
)
|
||||
|
||||
t = Team.objects.create(organizer=self.orga1, can_create_events=True, can_change_event_settings=True,
|
||||
can_change_items=True, can_change_orders=True)
|
||||
t.members.add(self.user)
|
||||
t.limit_events.add(self.event1)
|
||||
self.order = Order.objects.create(
|
||||
code='FOO', event=self.event1, email='dummy@dummy.test',
|
||||
status=Order.STATUS_PENDING,
|
||||
datetime=now(), expires=now(),
|
||||
total=14, payment_provider='banktransfer', locale='en'
|
||||
)
|
||||
|
||||
self.client.login(email='dummy@dummy.dummy', password='dummy')
|
||||
|
||||
def test_shred_simple(self):
|
||||
doc = self.get_doc('/control/event/%s/%s/shredder/' % (self.orga1.slug, self.event1.slug))
|
||||
assert doc.select("input[value=order_emails]")
|
||||
assert doc.select("input[value=stripe_logs]")
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/export' % (self.orga1.slug, self.event1.slug), {
|
||||
'shredder': 'order_emails'
|
||||
})
|
||||
assert doc.select("a.btn-primary")[0].text.strip() == "Download data"
|
||||
dlink = doc.select("a.btn-primary")[0].attrs['href']
|
||||
zipfiler = self.client.get(dlink)
|
||||
with ZipFile(BytesIO(zipfiler.getvalue()), 'r') as zipfile:
|
||||
indexdata = json.loads(zipfile.read('index.json').decode())
|
||||
assert indexdata['shredders'] == ['order_emails']
|
||||
assert indexdata['organizer'] == 'ccc'
|
||||
assert indexdata['event'] == '30c3'
|
||||
assert zipfile.read('CONFIRM_CODE.txt').decode() == indexdata['confirm_code']
|
||||
|
||||
maildata = json.loads(zipfile.read('emails-by-order.json').decode())
|
||||
assert maildata == {
|
||||
'FOO': 'dummy@dummy.test'
|
||||
}
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/shred' % (self.orga1.slug, self.event1.slug), {
|
||||
'confirm_code': indexdata['confirm_code'],
|
||||
'file': doc.select("input[name=file]")[0].attrs['value'],
|
||||
'password': 'dummy'
|
||||
})
|
||||
assert doc.select('.alert-success')
|
||||
self.order.refresh_from_db()
|
||||
assert not self.order.email
|
||||
|
||||
def test_shred_password_wrong(self):
|
||||
doc = self.get_doc('/control/event/%s/%s/shredder/' % (self.orga1.slug, self.event1.slug))
|
||||
assert doc.select("input[value=order_emails]")
|
||||
assert doc.select("input[value=stripe_logs]")
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/export' % (self.orga1.slug, self.event1.slug), {
|
||||
'shredder': 'order_emails'
|
||||
})
|
||||
assert doc.select("a.btn-primary")[0].text.strip() == "Download data"
|
||||
dlink = doc.select("a.btn-primary")[0].attrs['href']
|
||||
zipfiler = self.client.get(dlink)
|
||||
with ZipFile(BytesIO(zipfiler.getvalue()), 'r') as zipfile:
|
||||
indexdata = json.loads(zipfile.read('index.json').decode())
|
||||
assert indexdata['shredders'] == ['order_emails']
|
||||
assert indexdata['organizer'] == 'ccc'
|
||||
assert indexdata['event'] == '30c3'
|
||||
assert zipfile.read('CONFIRM_CODE.txt').decode() == indexdata['confirm_code']
|
||||
|
||||
maildata = json.loads(zipfile.read('emails-by-order.json').decode())
|
||||
assert maildata == {
|
||||
'FOO': 'dummy@dummy.test'
|
||||
}
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/shred' % (self.orga1.slug, self.event1.slug), {
|
||||
'confirm_code': indexdata['confirm_code'],
|
||||
'file': doc.select("input[name=file]")[0].attrs['value'],
|
||||
'password': 'test'
|
||||
})
|
||||
assert doc.select('.alert-danger')
|
||||
self.order.refresh_from_db()
|
||||
assert self.order.email
|
||||
|
||||
def test_shred_confirm_code_wrong(self):
|
||||
doc = self.get_doc('/control/event/%s/%s/shredder/' % (self.orga1.slug, self.event1.slug))
|
||||
assert doc.select("input[value=order_emails]")
|
||||
assert doc.select("input[value=stripe_logs]")
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/export' % (self.orga1.slug, self.event1.slug), {
|
||||
'shredder': 'order_emails'
|
||||
})
|
||||
assert doc.select("a.btn-primary")[0].text.strip() == "Download data"
|
||||
dlink = doc.select("a.btn-primary")[0].attrs['href']
|
||||
zipfiler = self.client.get(dlink)
|
||||
with ZipFile(BytesIO(zipfiler.getvalue()), 'r') as zipfile:
|
||||
indexdata = json.loads(zipfile.read('index.json').decode())
|
||||
assert indexdata['shredders'] == ['order_emails']
|
||||
assert indexdata['organizer'] == 'ccc'
|
||||
assert indexdata['event'] == '30c3'
|
||||
assert zipfile.read('CONFIRM_CODE.txt').decode() == indexdata['confirm_code']
|
||||
|
||||
maildata = json.loads(zipfile.read('emails-by-order.json').decode())
|
||||
assert maildata == {
|
||||
'FOO': 'dummy@dummy.test'
|
||||
}
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/shred' % (self.orga1.slug, self.event1.slug), {
|
||||
'confirm_code': indexdata['confirm_code'][::-1] + 'A',
|
||||
'file': doc.select("input[name=file]")[0].attrs['value'],
|
||||
'password': 'dummy'
|
||||
})
|
||||
assert doc.select('.alert-danger')
|
||||
self.order.refresh_from_db()
|
||||
assert self.order.email
|
||||
|
||||
def test_shred_constraints(self):
|
||||
self.event1.live = True
|
||||
self.event1.save()
|
||||
doc = self.get_doc('/control/event/%s/%s/shredder/' % (self.orga1.slug, self.event1.slug))
|
||||
assert not doc.select("input[value=order_emails]")
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/export' % (self.orga1.slug, self.event1.slug), {
|
||||
'shredder': 'order_emails'
|
||||
})
|
||||
assert doc.select('.alert-danger')
|
||||
|
||||
def test_shred_something_happened(self):
|
||||
doc = self.get_doc('/control/event/%s/%s/shredder/' % (self.orga1.slug, self.event1.slug))
|
||||
assert doc.select("input[value=order_emails]")
|
||||
assert doc.select("input[value=stripe_logs]")
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/export' % (self.orga1.slug, self.event1.slug), {
|
||||
'shredder': 'order_emails'
|
||||
})
|
||||
assert doc.select("a.btn-primary")[0].text.strip() == "Download data"
|
||||
dlink = doc.select("a.btn-primary")[0].attrs['href']
|
||||
zipfiler = self.client.get(dlink)
|
||||
with ZipFile(BytesIO(zipfiler.getvalue()), 'r') as zipfile:
|
||||
indexdata = json.loads(zipfile.read('index.json').decode())
|
||||
assert indexdata['shredders'] == ['order_emails']
|
||||
assert indexdata['organizer'] == 'ccc'
|
||||
assert indexdata['event'] == '30c3'
|
||||
assert zipfile.read('CONFIRM_CODE.txt').decode() == indexdata['confirm_code']
|
||||
|
||||
maildata = json.loads(zipfile.read('emails-by-order.json').decode())
|
||||
assert maildata == {
|
||||
'FOO': 'dummy@dummy.test'
|
||||
}
|
||||
self.order.log_action('dummy')
|
||||
doc = self.post_doc('/control/event/%s/%s/shredder/shred' % (self.orga1.slug, self.event1.slug), {
|
||||
'confirm_code': indexdata['confirm_code'],
|
||||
'file': doc.select("input[name=file]")[0].attrs['value'],
|
||||
'password': 'dummy'
|
||||
})
|
||||
assert doc.select('.alert-danger')
|
||||
self.order.refresh_from_db()
|
||||
assert self.order.email
|
||||
Reference in New Issue
Block a user