diff --git a/src/pretix/base/models/tax.py b/src/pretix/base/models/tax.py
index 4eda56f5a..34bbd93d1 100644
--- a/src/pretix/base/models/tax.py
+++ b/src/pretix/base/models/tax.py
@@ -114,7 +114,7 @@ EU_CURRENCIES = {
'RO': 'RON',
'SE': 'SEK'
}
-VAT_ID_COUNTRIES = EU_COUNTRIES | {'CH'}
+VAT_ID_COUNTRIES = EU_COUNTRIES | {'CH', 'NO'}
def is_eu_country(cc):
diff --git a/src/pretix/base/services/tax.py b/src/pretix/base/services/tax.py
index c2837d565..07255bfa0 100644
--- a/src/pretix/base/services/tax.py
+++ b/src/pretix/base/services/tax.py
@@ -22,9 +22,9 @@
import logging
import os
import re
-from urllib.error import HTTPError
+from xml.etree import ElementTree
-import vat_moss.errors
+import requests
import vat_moss.id
from django.conf import settings
from django.utils.translation import gettext_lazy as _
@@ -35,6 +35,16 @@ from zeep.exceptions import Fault
from pretix.base.models.tax import cc_to_vat_prefix, is_eu_country
logger = logging.getLogger(__name__)
+error_messages = {
+ 'unavailable': _(
+ 'Your VAT ID could not be checked, as the VAT checking service of '
+ 'your country is currently not available. We will therefore '
+ 'need to charge VAT on your invoice. You can get the tax amount '
+ 'back via the VAT reimbursement process.'
+ ),
+ 'invalid': _('This VAT ID is not valid. Please re-check your input.'),
+ 'country_mismatch': _('Your VAT ID does not match the selected country.'),
+}
class VATIDError(Exception):
@@ -50,33 +60,97 @@ class VATIDTemporaryError(VATIDError):
pass
-def _validate_vat_id_EU(vat_id, country_code):
- if vat_id[:2] != cc_to_vat_prefix(country_code):
- raise VATIDFinalError(_('Your VAT ID does not match the selected country.'))
+def _validate_vat_id_NO(vat_id, country_code):
+ # Inspired by vat_moss library
+ vat_id = vat_moss.id.normalize(vat_id)
+
+ if not vat_id or len(vat_id) < 3 or not re.match('^\\d{9}MVA$', vat_id[2:]):
+ raise VATIDFinalError(error_messages['invalid'])
+
+ organization_number = vat_id[2:].replace('MVA', '')
+ validation_url = 'https://data.brreg.no/enhetsregisteret/api/enheter/%s' % organization_number
try:
- result = vat_moss.id.validate(vat_id)
- if result:
- country_code, normalized_id, company_name = result
- return normalized_id
- except (vat_moss.errors.InvalidError, ValueError):
- raise VATIDFinalError(_('This VAT ID is not valid. Please re-check your input.'))
- except vat_moss.errors.WebServiceUnavailableError:
+ response = requests.get(validation_url, timeout=10)
+ if response.status_code in (404, 400):
+ raise VATIDFinalError(error_messages['invalid'])
+
+ response.raise_for_status()
+
+ info = response.json()
+ # This should never happen, but keeping it incase the API is changed
+ if 'organisasjonsnummer' not in info or info['organisasjonsnummer'] != organization_number:
+ logger.warning(
+ 'VAT ID checking failed for Norway due to missing or mismatching organisasjonsnummer in repsonse'
+ )
+ raise VATIDFinalError(error_messages['invalid'])
+ except requests.RequestException:
logger.exception('VAT ID checking failed for country {}'.format(country_code))
- raise VATIDTemporaryError(_(
- 'Your VAT ID could not be checked, as the VAT checking service of '
- 'your country is currently not available. We will therefore '
- 'need to charge VAT on your invoice. You can get the tax amount '
- 'back via the VAT reimbursement process.'
- ))
- except (vat_moss.errors.WebServiceError, HTTPError):
+ raise VATIDTemporaryError(error_messages['unavailable'])
+ else:
+ return vat_id
+
+
+def _validate_vat_id_EU(vat_id, country_code):
+ # Inspired by vat_moss library
+ vat_id = vat_moss.id.normalize(vat_id)
+ number = vat_id[2:]
+
+ if vat_id[:2] != cc_to_vat_prefix(country_code):
+ raise VATIDFinalError(error_messages['country_mismatch'])
+
+ if not re.match(vat_moss.id.ID_PATTERNS[country_code]['regex'], number):
+ raise VATIDFinalError(error_messages['invalid'])
+
+ payload = """
+
+
+
+
+ %s
+ %s
+
+
+
+ """.strip() % (country_code, number)
+
+ try:
+ response = requests.post(
+ 'https://ec.europa.eu/taxation_customs/vies/services/checkVatService',
+ data=payload,
+ timeout=10,
+ )
+ response.raise_for_status()
+
+ return_xml = response.text
+
+ try:
+ envelope = ElementTree.fromstring(return_xml)
+ except ElementTree.ParseError:
+ logger.error(
+ f'VAT ID checking failed for {country_code} due to XML parse error'
+ )
+ raise VATIDTemporaryError(error_messages['unavailable'])
+
+ namespaces = {
+ 'soap': 'http://schemas.xmlsoap.org/soap/envelope/',
+ 'vat': 'urn:ec.europa.eu:taxud:vies:services:checkVat:types'
+ }
+ valid_elements = envelope.findall('./soap:Body/vat:checkVatResponse/vat:valid', namespaces)
+ if not valid_elements:
+ logger.error(
+ f'VAT ID checking failed for {country_code} due to missing tag'
+ )
+ raise VATIDTemporaryError(error_messages['unavailable'])
+
+ if valid_elements[0].text.lower() != 'true':
+ raise VATIDFinalError(error_messages['invalid'])
+
+ except requests.RequestException:
logger.exception('VAT ID checking failed for country {}'.format(country_code))
- raise VATIDTemporaryError(_(
- 'Your VAT ID could not be checked, as the VAT checking service of '
- 'your country returned an incorrect result. We will therefore '
- 'need to charge VAT on your invoice. Please contact support to '
- 'resolve this manually.'
- ))
+ raise VATIDTemporaryError(error_messages['unavailable'])
+ else:
+ return vat_id
def _validate_vat_id_CH(vat_id, country_code):
@@ -85,10 +159,13 @@ def _validate_vat_id_CH(vat_id, country_code):
vat_id = re.sub('[^A-Z0-9]', '', vat_id.replace('HR', '').replace('MWST', ''))
try:
- transport = Transport(cache=SqliteCache(os.path.join(settings.CACHE_DIR, "validate_vat_id_ch_zeep_cache.db")))
+ transport = Transport(
+ cache=SqliteCache(os.path.join(settings.CACHE_DIR, "validate_vat_id_ch_zeep_cache.db")),
+ timeout=10
+ )
client = Client(
'https://www.uid-wse.admin.ch/V5.0/PublicServices.svc?wsdl',
- transport=transport
+ transport=transport,
)
result = client.service.ValidateUID(uid=vat_id)
except Fault as e:
@@ -125,10 +202,14 @@ def _validate_vat_id_CH(vat_id, country_code):
def validate_vat_id(vat_id, country_code):
+ if not vat_id:
+ return vat_id
country_code = str(country_code)
if is_eu_country(country_code):
return _validate_vat_id_EU(vat_id, country_code)
elif country_code == 'CH':
return _validate_vat_id_CH(vat_id, country_code)
+ elif country_code == 'NO':
+ return _validate_vat_id_NO(vat_id, country_code)
raise VATIDTemporaryError(f'VAT ID should not be entered for country {country_code}')
diff --git a/src/tests/base/test_vat_id_validation.py b/src/tests/base/test_vat_id_validation.py
new file mode 100644
index 000000000..de65b87fe
--- /dev/null
+++ b/src/tests/base/test_vat_id_validation.py
@@ -0,0 +1,169 @@
+import pytest
+import responses
+from requests import Timeout
+
+from pretix.base.services.tax import VATIDTemporaryError, validate_vat_id, VATIDFinalError
+
+
+def test_unknown_country():
+ with pytest.raises(VATIDTemporaryError):
+ validate_vat_id('TR12345', 'TR')
+
+
+@responses.activate
+def test_eu_invalid_format():
+ with pytest.raises(VATIDFinalError):
+ validate_vat_id('AT12345', 'AT')
+
+
+@responses.activate
+def test_eu_country_mismatch():
+ with pytest.raises(VATIDFinalError):
+ validate_vat_id('AT12345', 'DE')
+
+
+@responses.activate
+def test_eu_server_down():
+ def _callback(request):
+ raise Timeout
+
+ responses.add_callback(
+ responses.POST,
+ 'https://ec.europa.eu/taxation_customs/vies/services/checkVatService',
+ callback=_callback
+ )
+
+ with pytest.raises(VATIDTemporaryError):
+ validate_vat_id('ATU36801500', 'AT')
+
+
+@responses.activate
+def test_eu_server_error():
+ responses.add(
+ responses.POST,
+ 'https://ec.europa.eu/taxation_customs/vies/services/checkVatService',
+ body='error',
+ status=500
+ )
+
+ with pytest.raises(VATIDTemporaryError):
+ validate_vat_id('ATU36801500', 'AT')
+
+
+@responses.activate
+def test_eu_id_invalid():
+ responses.add(
+ responses.POST,
+ 'https://ec.europa.eu/taxation_customs/vies/services/checkVatService',
+ body="""
+
+
+ AT
+ U36801500
+ 2014-12-17+01:00
+ false
+ STADT WIEN
+ UNKNOWN
+
+
+ """,
+ status=200
+ )
+
+ with pytest.raises(VATIDFinalError):
+ validate_vat_id('ATU36801500', 'AT')
+
+
+@responses.activate
+def test_eu_id_valid():
+ responses.add(
+ responses.POST,
+ 'https://ec.europa.eu/taxation_customs/vies/services/checkVatService',
+ body="""
+
+
+ AT
+ U36801500
+ 2014-12-17+01:00
+ true
+ STADT WIEN
+ UNKNOWN
+
+
+ """,
+ status=200
+ )
+
+ assert validate_vat_id('ATU36801500', 'AT') == 'ATU36801500'
+
+
+@responses.activate
+def test_NO_invalid_format():
+ with pytest.raises(VATIDFinalError):
+ validate_vat_id('NO12345', 'NO')
+
+
+@responses.activate
+def test_NO_server_down():
+ def _callback(request):
+ raise Timeout
+
+ responses.add_callback(
+ responses.GET,
+ 'https://data.brreg.no/enhetsregisteret/api/enheter/974760673',
+ callback=_callback
+ )
+
+ with pytest.raises(VATIDTemporaryError):
+ validate_vat_id('NO974760673 MVA', 'NO')
+
+
+@responses.activate
+def test_NO_server_error():
+ responses.add(
+ responses.GET,
+ 'https://data.brreg.no/enhetsregisteret/api/enheter/974760673',
+ body='error',
+ status=500
+ )
+
+ with pytest.raises(VATIDTemporaryError):
+ validate_vat_id('NO974760673 MVA', 'NO')
+
+
+@responses.activate
+def test_NO_id_invalid():
+ responses.add(
+ responses.GET,
+ 'https://data.brreg.no/enhetsregisteret/api/enheter/974760673',
+ body="",
+ status=404
+ )
+
+ with pytest.raises(VATIDFinalError):
+ validate_vat_id('NO974760673 MVA', 'NO')
+
+
+@responses.activate
+def test_NO_id_valid():
+ responses.add(
+ responses.GET,
+ 'https://data.brreg.no/enhetsregisteret/api/enheter/974760673',
+ body='{"organisasjonsnummer":"974760673","navn":"REGISTERENHETEN I BRØNNØYSUND","organisasjonsform":{"kode":'
+ '"ORGL","beskrivelse":"Organisasjonsledd","_links":{"self":{"href":"https://data.brreg.no/enhetsregisteret/api/'
+ 'organisasjonsformer/ORGL"}}},"hjemmeside":"www.brreg.no","postadresse":{"land":"Norge","landkode":"NO","postn'
+ 'ummer":"8910","poststed":"BRØNNØYSUND","adresse":["Postboks 900"],"kommune":"BRØNNØY","kommunenummer":"1813"}'
+ ',"registreringsdatoEnhetsregisteret":"1995-08-09","registrertIMvaregisteret":false,"naeringskode1":{"beskrivels'
+ 'e":"Generell offentlig administrasjon","kode":"84.110"},"antallAnsatte":455,"overordnetEnhet":"912660680","for'
+ 'retningsadresse":{"land":"Norge","landkode":"NO","postnummer":"8900","poststed":"BRØNNØYSUND","adresse":["Havn'
+ 'egata 48"],"kommune":"BRØNNØY","kommunenummer":"1813"},"institusjonellSektorkode":{"kode":"6100","beskrivelse'
+ '":"Statsforvaltningen"},"registrertIForetaksregisteret":false,"registrertIStiftelsesregisteret":false,"registr'
+ 'ertIFrivillighetsregisteret":false,"konkurs":false,"underAvvikling":false,"underTvangsavviklingEllerTvangsopp'
+ 'losning":false,"maalform":"Bokmål","_links":{"self":{"href":"https://data.brreg.no/enhetsregisteret/api/enheter'
+ '/974760673"},"overordnetEnhet":{"href":"https://data.brreg.no/enhetsregisteret/api/enheter/912660680"}}}',
+ status=200
+ )
+
+ assert validate_vat_id('NO974760673 MVA', 'NO') == 'NO974760673MVA'
+
+# No tests for CH currently since it's harder to mock Zeep
diff --git a/src/tests/control/test_orders.py b/src/tests/control/test_orders.py
index 2a88e481b..952c6a54a 100644
--- a/src/tests/control/test_orders.py
+++ b/src/tests/control/test_orders.py
@@ -42,6 +42,8 @@ from django.core import mail
from django.utils.timezone import now
from django_countries.fields import Country
from django_scopes import scopes_disabled
+
+from pretix.base.services.tax import VATIDFinalError, VATIDTemporaryError
from tests.base import SoupTest
from tests.plugins.stripe.test_provider import MockedCharge
@@ -1563,8 +1565,8 @@ def test_check_vatid(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='ATU1234567', country=Country('AT'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
assert 'alert-success' in response.content.decode()
ia.refresh_from_db()
@@ -1576,8 +1578,8 @@ def test_check_vatid_no_entered(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, country=Country('AT'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
assert 'alert-danger' in response.content.decode()
ia.refresh_from_db()
@@ -1589,12 +1591,10 @@ def test_check_vatid_invalid_country(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='ATU1234567', country=Country('FR'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
- response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
- assert 'alert-danger' in response.content.decode()
- ia.refresh_from_db()
- assert not ia.vat_id_validated
+ response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
+ assert 'alert-danger' in response.content.decode()
+ ia.refresh_from_db()
+ assert not ia.vat_id_validated
@pytest.mark.django_db
@@ -1602,8 +1602,8 @@ def test_check_vatid_noneu_country(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='CHU1234567', country=Country('CH'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
assert 'alert-danger' in response.content.decode()
ia.refresh_from_db()
@@ -1615,8 +1615,8 @@ def test_check_vatid_no_country(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='ATU1234567')
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
assert 'alert-danger' in response.content.decode()
ia.refresh_from_db()
@@ -1626,8 +1626,8 @@ def test_check_vatid_no_country(client, env):
@pytest.mark.django_db
def test_check_vatid_no_invoiceaddress(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
assert 'alert-danger' in response.content.decode()
@@ -1637,10 +1637,9 @@ def test_check_vatid_invalid(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='ATU1234567', country=Country('AT'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
def raiser(*args, **kwargs):
- import vat_moss.errors
- raise vat_moss.errors.InvalidError('Fail')
+ raise VATIDFinalError('Fail')
mock_validate.side_effect = raiser
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
@@ -1654,10 +1653,9 @@ def test_check_vatid_unavailable(client, env):
client.login(email='dummy@dummy.dummy', password='dummy')
with scopes_disabled():
ia = InvoiceAddress.objects.create(order=env[2], is_business=True, vat_id='ATU1234567', country=Country('AT'))
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
def raiser(*args, **kwargs):
- import vat_moss.errors
- raise vat_moss.errors.WebServiceUnavailableError('Fail')
+ raise VATIDTemporaryError('Fail')
mock_validate.side_effect = raiser
response = client.post('/control/event/dummy/dummy/orders/FOO/checkvatid', {}, follow=True)
diff --git a/src/tests/presale/test_checkout.py b/src/tests/presale/test_checkout.py
index 2e53ba828..e802d11dc 100644
--- a/src/tests/presale/test_checkout.py
+++ b/src/tests/presale/test_checkout.py
@@ -47,6 +47,7 @@ from pretix.base.models.items import (
ItemAddOn, ItemBundle, ItemVariation, SubEventItem, SubEventItemVariation,
)
from pretix.base.services.orders import OrderError, _perform_order
+from pretix.base.services.tax import VATIDTemporaryError, VATIDFinalError
from pretix.testutils.scope import classscope
from pretix.testutils.sessions import get_cart_session_key
@@ -139,8 +140,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -163,8 +164,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
def test_reverse_charge_enable_then_disable(self):
self.test_reverse_charge()
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'individual',
'name': 'Bar',
@@ -195,10 +196,9 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
def raiser(*args, **kwargs):
- import vat_moss.errors
- raise vat_moss.errors.InvalidError()
+ raise VATIDFinalError('final')
mock_validate.side_effect = raiser
resp = self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
@@ -229,7 +229,7 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
mock_validate.return_value = ('AU', 'AU123456', 'Foo')
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
@@ -263,8 +263,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -296,20 +296,18 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
- resp = self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
- 'is_business': 'business',
- 'company': 'Foo',
- 'name': 'Bar',
- 'street': 'Baz',
- 'zipcode': '12345',
- 'city': 'Here',
- 'country': 'FR',
- 'vat_id': 'AT123456',
- 'email': 'admin@localhost'
- }, follow=True)
- assert 'alert-danger' in resp.content.decode()
+ resp = self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
+ 'is_business': 'business',
+ 'company': 'Foo',
+ 'name': 'Bar',
+ 'street': 'Baz',
+ 'zipcode': '12345',
+ 'city': 'Here',
+ 'country': 'FR',
+ 'vat_id': 'AT123456',
+ 'email': 'admin@localhost'
+ }, follow=True)
+ assert 'alert-danger' in resp.content.decode()
cr1.refresh_from_db()
assert cr1.price == Decimal('23.00')
@@ -326,10 +324,9 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
def raiser(*args, **kwargs):
- import vat_moss.errors
- raise vat_moss.errors.WebServiceUnavailableError('Fail')
+ raise VATIDTemporaryError('temp')
mock_validate.side_effect = raiser
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
@@ -364,8 +361,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -401,8 +398,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -418,7 +415,7 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
cr1.refresh_from_db()
assert cr1.price == Decimal('19.33')
- with mock.patch('vat_moss.id.validate') as mock_validate:
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
mock_validate.return_value = ('DE', 'DE123456', 'Foo')
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
@@ -465,8 +462,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
cr1.refresh_from_db()
assert cr1.price == Decimal('23.00')
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
r = self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -525,8 +522,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
price=23, expires=now() + timedelta(minutes=10)
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'individual',
'name': 'Bar',
@@ -577,8 +574,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
voucher=self.event.vouchers.create()
)
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'individual',
'name': 'Bar',
@@ -605,8 +602,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
def test_country_taxing_switch(self):
self._test_country_taxing()
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'individual',
'name': 'Bar',
@@ -657,8 +654,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
assert cr1.price == Decimal('28.56')
assert cr1.tax_rate == Decimal('19.00')
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',
@@ -721,8 +718,8 @@ class CheckoutTestCase(BaseCheckoutTestCase, TestCase):
assert cr1.price == Decimal('47.60')
assert cr1.tax_rate == Decimal('19.00')
- with mock.patch('vat_moss.id.validate') as mock_validate:
- mock_validate.return_value = ('AT', 'AT123456', 'Foo')
+ with mock.patch('pretix.base.services.tax._validate_vat_id_EU') as mock_validate:
+ mock_validate.return_value = 'AT123456'
self.client.post('/%s/%s/checkout/questions/' % (self.orga.slug, self.event.slug), {
'is_business': 'business',
'company': 'Foo',