2007-04-26 20:41:34 +08:00
|
|
|
"""
|
|
|
|
Chile specific form helpers.
|
|
|
|
"""
|
2007-04-26 22:59:27 +08:00
|
|
|
|
2011-10-18 08:47:49 +08:00
|
|
|
from __future__ import absolute_import
|
|
|
|
|
|
|
|
from django.contrib.localflavor.cl.cl_regions import REGION_CHOICES
|
2010-01-05 11:56:19 +08:00
|
|
|
from django.core.validators import EMPTY_VALUES
|
2008-07-19 09:22:26 +08:00
|
|
|
from django.forms import ValidationError
|
2010-01-05 11:56:19 +08:00
|
|
|
from django.forms.fields import RegexField, Select
|
2008-06-18 21:10:05 +08:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 20:11:04 +08:00
|
|
|
from django.utils.encoding import smart_unicode
|
2007-04-26 20:41:34 +08:00
|
|
|
|
2007-07-12 20:44:05 +08:00
|
|
|
|
|
|
|
class CLRegionSelect(Select):
|
|
|
|
"""
|
|
|
|
A Select widget that uses a list of Chilean Regions (Regiones)
|
|
|
|
as its choices.
|
|
|
|
"""
|
|
|
|
def __init__(self, attrs=None):
|
|
|
|
super(CLRegionSelect, self).__init__(attrs, choices=REGION_CHOICES)
|
|
|
|
|
2007-04-26 20:41:34 +08:00
|
|
|
class CLRutField(RegexField):
|
|
|
|
"""
|
|
|
|
Chilean "Rol Unico Tributario" (RUT) field. This is the Chilean national
|
|
|
|
identification number.
|
|
|
|
|
|
|
|
Samples for testing are available from
|
|
|
|
https://palena.sii.cl/cvc/dte/ee_empresas_emisoras.html
|
|
|
|
"""
|
2007-12-17 16:05:27 +08:00
|
|
|
default_error_messages = {
|
2008-06-18 21:10:05 +08:00
|
|
|
'invalid': _('Enter a valid Chilean RUT.'),
|
|
|
|
'strict': _('Enter a valid Chilean RUT. The format is XX.XXX.XXX-X.'),
|
|
|
|
'checksum': _('The Chilean RUT is not valid.'),
|
2007-12-17 16:05:27 +08:00
|
|
|
}
|
|
|
|
|
2007-04-26 20:41:34 +08:00
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
if 'strict' in kwargs:
|
|
|
|
del kwargs['strict']
|
|
|
|
super(CLRutField, self).__init__(r'^(\d{1,2}\.)?\d{3}\.\d{3}-[\dkK]$',
|
2007-12-17 16:05:27 +08:00
|
|
|
error_message=self.default_error_messages['strict'], *args, **kwargs)
|
2007-04-26 20:41:34 +08:00
|
|
|
else:
|
|
|
|
# In non-strict mode, accept RUTs that validate but do not exist in
|
|
|
|
# the real world.
|
2007-12-17 16:05:27 +08:00
|
|
|
super(CLRutField, self).__init__(r'^[\d\.]{1,11}-?[\dkK]$', *args, **kwargs)
|
2007-04-26 20:41:34 +08:00
|
|
|
|
|
|
|
def clean(self, value):
|
|
|
|
"""
|
2007-04-26 22:59:27 +08:00
|
|
|
Check and clean the Chilean RUT.
|
2007-04-26 20:41:34 +08:00
|
|
|
"""
|
|
|
|
super(CLRutField, self).clean(value)
|
|
|
|
if value in EMPTY_VALUES:
|
|
|
|
return u''
|
|
|
|
rut, verificador = self._canonify(value)
|
|
|
|
if self._algorithm(rut) == verificador:
|
|
|
|
return self._format(rut, verificador)
|
|
|
|
else:
|
2007-12-17 16:05:27 +08:00
|
|
|
raise ValidationError(self.error_messages['checksum'])
|
2007-04-26 20:41:34 +08:00
|
|
|
|
|
|
|
def _algorithm(self, rut):
|
|
|
|
"""
|
|
|
|
Takes RUT in pure canonical form, calculates the verifier digit.
|
|
|
|
"""
|
|
|
|
suma = 0
|
|
|
|
multi = 2
|
|
|
|
for r in rut[::-1]:
|
|
|
|
suma += int(r) * multi
|
|
|
|
multi += 1
|
|
|
|
if multi == 8:
|
|
|
|
multi = 2
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 20:11:04 +08:00
|
|
|
return u'0123456789K0'[11 - suma % 11]
|
2007-04-26 20:41:34 +08:00
|
|
|
|
|
|
|
def _canonify(self, rut):
|
|
|
|
"""
|
|
|
|
Turns the RUT into one normalized format. Returns a (rut, verifier)
|
|
|
|
tuple.
|
|
|
|
"""
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 20:11:04 +08:00
|
|
|
rut = smart_unicode(rut).replace(' ', '').replace('.', '').replace('-', '')
|
2011-02-08 19:59:38 +08:00
|
|
|
return rut[:-1], rut[-1].upper()
|
2007-04-26 20:41:34 +08:00
|
|
|
|
|
|
|
def _format(self, code, verifier=None):
|
|
|
|
"""
|
|
|
|
Formats the RUT from canonical form to the common string representation.
|
|
|
|
If verifier=None, then the last digit in 'code' is the verifier.
|
|
|
|
"""
|
|
|
|
if verifier is None:
|
|
|
|
verifier = code[-1]
|
|
|
|
code = code[:-1]
|
|
|
|
while len(code) > 3 and '.' not in code[:3]:
|
|
|
|
pos = code.find('.')
|
|
|
|
if pos == -1:
|
|
|
|
new_dot = -3
|
|
|
|
else:
|
|
|
|
new_dot = pos - 3
|
|
|
|
code = code[:new_dot] + '.' + code[new_dot:]
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 20:11:04 +08:00
|
|
|
return u'%s-%s' % (code, verifier)
|
2007-04-26 20:41:34 +08:00
|
|
|
|