[ADD] base_vat_optional_vies
* Disable VIES test * Fixes to avoid exception when using default _construct_constraint_msg method
This commit is contained in:
parent
90eb5b8081
commit
6ba2b16969
78
base_vat_optional_vies/README.rst
Normal file
78
base_vat_optional_vies/README.rst
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg
|
||||||
|
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||||
|
:alt: License: AGPL-3
|
||||||
|
|
||||||
|
===================================
|
||||||
|
Optional validation of VAT via VIES
|
||||||
|
===================================
|
||||||
|
|
||||||
|
This module extends base_vat module features allowing to know if VIES
|
||||||
|
validation was passed or not.
|
||||||
|
|
||||||
|
Then you can use "VIES validation passed" field in order to show VAT ID with
|
||||||
|
or without country preffix in invoices, for instance.
|
||||||
|
|
||||||
|
*NOTE*: Altought VIES validation is actived in your company, this validation
|
||||||
|
will not block VAT ID write (main different to Odoo standard behavior) if this
|
||||||
|
VAT ID is valid in its country.
|
||||||
|
|
||||||
|
|
||||||
|
Configuration
|
||||||
|
=============
|
||||||
|
|
||||||
|
In order to activate VIES validation, you must set this option in your company:
|
||||||
|
Settings > Companies > Companies > Your Company > Configuration > Accounting > VIES VAT Check
|
||||||
|
|
||||||
|
|
||||||
|
Usage
|
||||||
|
=====
|
||||||
|
|
||||||
|
When VIES VAT Check is activated:
|
||||||
|
|
||||||
|
* Odoo will try to validate VAT against VIES online service
|
||||||
|
* If passed, then "VIES validation passed" field will be True
|
||||||
|
* If not passed, then try to validate using country validation method
|
||||||
|
* If validated, then "VIES validation passed" field will be False
|
||||||
|
* If not validated, then a ValidationError will be shown to user
|
||||||
|
|
||||||
|
When VIES VAT Check is not activated:
|
||||||
|
|
||||||
|
* "VIES validation passed" field will be always False
|
||||||
|
|
||||||
|
You must preffix VAT with country code (ISO 3166-1 alpha-2) and if you want to
|
||||||
|
bypass country validation you can use "EU" code
|
||||||
|
|
||||||
|
|
||||||
|
Bug Tracker
|
||||||
|
===========
|
||||||
|
|
||||||
|
Bugs are tracked on `GitHub Issues <https://github.com/OCA/account-financial-tools/issues>`_.
|
||||||
|
In case of trouble, please check there if your issue has already been reported.
|
||||||
|
If you spotted it first, help us smashing it by providing a detailed and welcomed feedback
|
||||||
|
`here <https://github.com/OCA/account-financial-tools/issues/new?body=module:%20base_vat_optional_vies%0Aversion:%208.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
||||||
|
|
||||||
|
|
||||||
|
Credits
|
||||||
|
=======
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
------------
|
||||||
|
|
||||||
|
* Rafael Blasco <rafabn@antiun.com>
|
||||||
|
* Antonio Espinosa <antonioea@antiun.com>
|
||||||
|
|
||||||
|
|
||||||
|
Maintainer
|
||||||
|
----------
|
||||||
|
|
||||||
|
.. image:: https://odoo-community.org/logo.png
|
||||||
|
:alt: Odoo Community Association
|
||||||
|
:target: https://odoo-community.org
|
||||||
|
|
||||||
|
This module is maintained by the OCA.
|
||||||
|
|
||||||
|
OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||||
|
mission is to support the collaborative development of Odoo features and
|
||||||
|
promote its widespread use.
|
||||||
|
|
||||||
|
To contribute to this module, please visit http://odoo-community.org.
|
5
base_vat_optional_vies/__init__.py
Normal file
5
base_vat_optional_vies/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
from . import models
|
24
base_vat_optional_vies/__openerp__.py
Normal file
24
base_vat_optional_vies/__openerp__.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
{
|
||||||
|
'name': "Optional validation of VAT via VIES",
|
||||||
|
'category': 'Accounting',
|
||||||
|
'version': '8.0.1.0.0',
|
||||||
|
'depends': [
|
||||||
|
'base_vat',
|
||||||
|
],
|
||||||
|
'external_dependencies': {
|
||||||
|
'python': ['vatnumber'],
|
||||||
|
},
|
||||||
|
'data': [
|
||||||
|
'views/res_partner_view.xml',
|
||||||
|
],
|
||||||
|
'author': 'Antiun Ingeniería S.L., '
|
||||||
|
'Odoo Community Association (OCA)',
|
||||||
|
'website': 'http://www.antiun.com',
|
||||||
|
'license': 'AGPL-3',
|
||||||
|
'images': [],
|
||||||
|
'installable': True,
|
||||||
|
}
|
0
base_vat_optional_vies/i18n/.gitkeep
Normal file
0
base_vat_optional_vies/i18n/.gitkeep
Normal file
5
base_vat_optional_vies/models/__init__.py
Normal file
5
base_vat_optional_vies/models/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
from . import res_partner
|
137
base_vat_optional_vies/models/res_partner.py
Normal file
137
base_vat_optional_vies/models/res_partner.py
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
import logging
|
||||||
|
import re
|
||||||
|
from openerp import models, fields, api
|
||||||
|
from openerp.exceptions import ValidationError
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
try:
|
||||||
|
import vatnumber
|
||||||
|
except ImportError:
|
||||||
|
_logger.warning(
|
||||||
|
"VAT validation partially unavailable because the `vatnumber` Python "
|
||||||
|
"library cannot be found. Install it to support more countries, "
|
||||||
|
"for example with `easy_install vatnumber` or "
|
||||||
|
"`pip install vatnumber`.")
|
||||||
|
vatnumber = None
|
||||||
|
|
||||||
|
|
||||||
|
class ResPartner(models.Model):
|
||||||
|
_inherit = 'res.partner'
|
||||||
|
|
||||||
|
vies_passed = fields.Boolean(
|
||||||
|
string="VIES validation passed", readonly=True)
|
||||||
|
|
||||||
|
def __init__(self, pool, cr):
|
||||||
|
super(ResPartner, self).__init__(pool, cr)
|
||||||
|
self._constraints = []
|
||||||
|
|
||||||
|
@api.constrains('vat')
|
||||||
|
def check_vat(self):
|
||||||
|
for partner in self:
|
||||||
|
if (not self.env.context.get('avoid_check_vat') and
|
||||||
|
not partner.parent_id):
|
||||||
|
if not partner.validate_vat():
|
||||||
|
raise ValidationError(partner._construct_constraint_msg())
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def button_check_vat(self):
|
||||||
|
if not self.validate_vat():
|
||||||
|
raise ValidationError(self._construct_constraint_msg())
|
||||||
|
return True
|
||||||
|
|
||||||
|
def _split_vat(self, vat, country=False):
|
||||||
|
"""
|
||||||
|
@summary: Split Partner vat into country_code and number
|
||||||
|
@result: (vat_country, vat_number)
|
||||||
|
"""
|
||||||
|
vat_country = 'XX'
|
||||||
|
vat_number = vat
|
||||||
|
if vat and re.match(r'[A-Za-z]{2}', vat):
|
||||||
|
vat_country = vat[:2].upper()
|
||||||
|
vat_number = vat[2:].replace(' ', '')
|
||||||
|
elif country:
|
||||||
|
vat_country = country
|
||||||
|
return vat_country, vat_number
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def validate_vat(self):
|
||||||
|
self.ensure_one()
|
||||||
|
if self.company_id.vat_check_vies:
|
||||||
|
# VIES online check
|
||||||
|
check_func = self.vies_vat_optional_check
|
||||||
|
else:
|
||||||
|
# quick and partial off-line checksum validation
|
||||||
|
check_func = self.simple_vat_optional_check
|
||||||
|
vat_country, vat_number = self._split_vat(self.vat)
|
||||||
|
if vat_number and vat_country == 'XX':
|
||||||
|
_logger.info("VAT country not found!")
|
||||||
|
raise ValidationError(self._construct_constraint_msg())
|
||||||
|
if vat_number and not check_func(vat_country, vat_number):
|
||||||
|
_logger.info("VAT Number [%s] is not valid !" % vat_number)
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def simple_vat_optional_check(self, country_code, vat_number):
|
||||||
|
"""
|
||||||
|
Check the VAT number depending of the country.
|
||||||
|
http://sima-pc.com/nif.php
|
||||||
|
"""
|
||||||
|
self.ensure_one()
|
||||||
|
res = self.simple_vat_check(country_code.lower(), vat_number)
|
||||||
|
data = {}
|
||||||
|
if res and self.vies_passed and not self.company_id.vat_check_vies:
|
||||||
|
# Can not be sure that this VAT is signed up in VIES
|
||||||
|
data['vies_passed'] = False
|
||||||
|
if res:
|
||||||
|
vat = country_code + vat_number
|
||||||
|
if self.vat != vat:
|
||||||
|
data['vat'] = vat
|
||||||
|
if data:
|
||||||
|
self.with_context(avoid_check_vat=True).write(data)
|
||||||
|
return res
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def vies_vat_optional_check(self, country_code, vat_number):
|
||||||
|
self.ensure_one()
|
||||||
|
data = {}
|
||||||
|
res = False
|
||||||
|
try:
|
||||||
|
# Validate against VAT Information Exchange System (VIES)
|
||||||
|
# see also http://ec.europa.eu/taxation_customs/vies/
|
||||||
|
vat = country_code + vat_number
|
||||||
|
res = vatnumber.check_vies(vat)
|
||||||
|
if res and not self.vies_passed:
|
||||||
|
data['vies_passed'] = True
|
||||||
|
except Exception:
|
||||||
|
# See:
|
||||||
|
# http://ec.europa.eu/taxation_customs/vies/checkVatService.wsdl
|
||||||
|
# Fault code may contain INVALID_INPUT, SERVICE_UNAVAILABLE,
|
||||||
|
# MS_UNAVAILABLE, TIMEOUT or SERVER_BUSY. There is no way we can
|
||||||
|
# validate the input with VIES if any of these arise, including
|
||||||
|
# the first one (it means invalid country code or empty
|
||||||
|
# VAT number), so we fall back to the simple check.
|
||||||
|
pass
|
||||||
|
|
||||||
|
if not res:
|
||||||
|
res = self.simple_vat_optional_check(country_code, vat_number)
|
||||||
|
if self.vies_passed:
|
||||||
|
data['vies_passed'] = False
|
||||||
|
if res:
|
||||||
|
vat = country_code + vat_number
|
||||||
|
if self.vat != vat:
|
||||||
|
data['vat'] = vat
|
||||||
|
if data:
|
||||||
|
self.with_context(avoid_check_vat=True).write(data)
|
||||||
|
return res
|
||||||
|
|
||||||
|
# Delete old api constraint defined in base_vat addon
|
||||||
|
@api.multi
|
||||||
|
def _validate_fields(self, field_names):
|
||||||
|
self._constraints = [x for x in self._constraints if 'vat' not in x[2]]
|
||||||
|
super(ResPartner, self)._validate_fields(field_names)
|
BIN
base_vat_optional_vies/static/description/icon.png
Normal file
BIN
base_vat_optional_vies/static/description/icon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 9.2 KiB |
5
base_vat_optional_vies/tests/__init__.py
Normal file
5
base_vat_optional_vies/tests/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
from . import test_res_partner
|
70
base_vat_optional_vies/tests/test_res_partner.py
Normal file
70
base_vat_optional_vies/tests/test_res_partner.py
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3: Antiun Ingenieria S.L. - Antonio Espinosa
|
||||||
|
# See README.rst file on addon root folder for more details
|
||||||
|
|
||||||
|
from openerp.tests.common import TransactionCase
|
||||||
|
|
||||||
|
|
||||||
|
class TestResPartner(TransactionCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestResPartner, self).setUp()
|
||||||
|
self.m_partner = self.env['res.partner']
|
||||||
|
self.m_company = self.env['res.company']
|
||||||
|
self.company = self.m_company.browse(self.ref('base.main_company'))
|
||||||
|
self.partner = self.m_partner.browse(self.ref('base.res_partner_1'))
|
||||||
|
|
||||||
|
def test_split_vat(self):
|
||||||
|
cases = (
|
||||||
|
# vat, country, => vat_country, vat_number
|
||||||
|
('ESB12345678', False, 'ES', 'B12345678'),
|
||||||
|
('B12345678', False, 'XX', 'B12345678'),
|
||||||
|
('1EB12345678', False, 'XX', '1EB12345678'),
|
||||||
|
('ESB12345678', 'DE', 'ES', 'B12345678'),
|
||||||
|
('B12345678', 'ES', 'ES', 'B12345678'),
|
||||||
|
)
|
||||||
|
for vat, country, vat_country, vat_number in cases:
|
||||||
|
res_country, res_number = self.m_partner._split_vat(vat, country)
|
||||||
|
self.assertEqual(res_country, vat_country)
|
||||||
|
self.assertEqual(res_number, vat_number)
|
||||||
|
|
||||||
|
def _test_validate_vat(self, cases):
|
||||||
|
for vat, res_vat, res_vies in cases:
|
||||||
|
self.partner.write({
|
||||||
|
'vat': vat,
|
||||||
|
})
|
||||||
|
self.assertEqual(self.partner.vat, res_vat)
|
||||||
|
self.assertEqual(self.partner.vies_passed, res_vies)
|
||||||
|
|
||||||
|
# AEA: Can't use this test in Travis, VIES checking returns always False
|
||||||
|
# because of timeout
|
||||||
|
# def test_validate_vat_vies(self):
|
||||||
|
# """
|
||||||
|
# Validate VAT when company 'vat_check_vies' option is True
|
||||||
|
# All VATs are valid, but some are not signed up in VIES database
|
||||||
|
# """
|
||||||
|
# self.company.vat_check_vies = True
|
||||||
|
# cases = (
|
||||||
|
# # vat => vat, vies_passed
|
||||||
|
# # VATs signed up in VIES
|
||||||
|
# ('ESB84718550', 'ESB84718550', True),
|
||||||
|
# ('de222070543', 'DE222070543', True),
|
||||||
|
# # Valid VATs don't signed up in VIES
|
||||||
|
# ('DE253130868', 'DE253130868', False),
|
||||||
|
# ('esB87286357', 'ESB87286357', False),
|
||||||
|
# )
|
||||||
|
# self._test_validate_vat(cases)
|
||||||
|
|
||||||
|
def test_validate_vat_no_vies(self):
|
||||||
|
"""
|
||||||
|
Validate VAT when company 'vat_check_vies' option is False
|
||||||
|
"""
|
||||||
|
self.company.vat_check_vies = False
|
||||||
|
cases = (
|
||||||
|
# vat => vat, vies_passed
|
||||||
|
('ESB84718550', 'ESB84718550', False),
|
||||||
|
('de222070543', 'DE222070543', False),
|
||||||
|
('DE253130868', 'DE253130868', False),
|
||||||
|
('esB87286357', 'ESB87286357', False),
|
||||||
|
)
|
||||||
|
self._test_validate_vat(cases)
|
18
base_vat_optional_vies/views/res_partner_view.xml
Normal file
18
base_vat_optional_vies/views/res_partner_view.xml
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<openerp>
|
||||||
|
<data>
|
||||||
|
|
||||||
|
<record id="view_partner_property_form" model="ir.ui.view">
|
||||||
|
<field name="name">Add VAT country and VIES passed fields</field>
|
||||||
|
<field name="model">res.partner</field>
|
||||||
|
<field name="inherit_id" ref="account.view_partner_property_form"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<field name="property_account_position" position="after">
|
||||||
|
<field name="vies_passed"/>
|
||||||
|
</field>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</openerp>
|
Loading…
Reference in New Issue
Block a user