| 1 |
""" |
|---|
| 2 |
USA-specific Form helpers |
|---|
| 3 |
""" |
|---|
| 4 |
|
|---|
| 5 |
from django.forms import ValidationError |
|---|
| 6 |
from django.forms.fields import Field, RegexField, Select, EMPTY_VALUES |
|---|
| 7 |
from django.utils.encoding import smart_unicode |
|---|
| 8 |
from django.utils.translation import ugettext_lazy as _ |
|---|
| 9 |
import re |
|---|
| 10 |
|
|---|
| 11 |
phone_digits_re = re.compile(r'^(?:1-?)?(\d{3})[-\.]?(\d{3})[-\.]?(\d{4})$') |
|---|
| 12 |
ssn_re = re.compile(r"^(?P<area>\d{3})[-\ ]?(?P<group>\d{2})[-\ ]?(?P<serial>\d{4})$") |
|---|
| 13 |
|
|---|
| 14 |
class USZipCodeField(RegexField): |
|---|
| 15 |
default_error_messages = { |
|---|
| 16 |
'invalid': _('Enter a zip code in the format XXXXX or XXXXX-XXXX.'), |
|---|
| 17 |
} |
|---|
| 18 |
|
|---|
| 19 |
def __init__(self, *args, **kwargs): |
|---|
| 20 |
super(USZipCodeField, self).__init__(r'^\d{5}(?:-\d{4})?$', |
|---|
| 21 |
max_length=None, min_length=None, *args, **kwargs) |
|---|
| 22 |
|
|---|
| 23 |
class USPhoneNumberField(Field): |
|---|
| 24 |
default_error_messages = { |
|---|
| 25 |
'invalid': u'Phone numbers must be in XXX-XXX-XXXX format.', |
|---|
| 26 |
} |
|---|
| 27 |
|
|---|
| 28 |
def clean(self, value): |
|---|
| 29 |
super(USPhoneNumberField, self).clean(value) |
|---|
| 30 |
if value in EMPTY_VALUES: |
|---|
| 31 |
return u'' |
|---|
| 32 |
value = re.sub('(\(|\)|\s+)', '', smart_unicode(value)) |
|---|
| 33 |
m = phone_digits_re.search(value) |
|---|
| 34 |
if m: |
|---|
| 35 |
return u'%s-%s-%s' % (m.group(1), m.group(2), m.group(3)) |
|---|
| 36 |
raise ValidationError(self.error_messages['invalid']) |
|---|
| 37 |
|
|---|
| 38 |
class USSocialSecurityNumberField(Field): |
|---|
| 39 |
""" |
|---|
| 40 |
A United States Social Security number. |
|---|
| 41 |
|
|---|
| 42 |
Checks the following rules to determine whether the number is valid: |
|---|
| 43 |
|
|---|
| 44 |
* Conforms to the XXX-XX-XXXX format. |
|---|
| 45 |
* No group consists entirely of zeroes. |
|---|
| 46 |
* The leading group is not "666" (block "666" will never be allocated). |
|---|
| 47 |
* The number is not in the promotional block 987-65-4320 through |
|---|
| 48 |
987-65-4329, which are permanently invalid. |
|---|
| 49 |
* The number is not one known to be invalid due to otherwise widespread |
|---|
| 50 |
promotional use or distribution (e.g., the Woolworth's number or the |
|---|
| 51 |
1962 promotional number). |
|---|
| 52 |
""" |
|---|
| 53 |
default_error_messages = { |
|---|
| 54 |
'invalid': _('Enter a valid U.S. Social Security number in XXX-XX-XXXX format.'), |
|---|
| 55 |
} |
|---|
| 56 |
|
|---|
| 57 |
def clean(self, value): |
|---|
| 58 |
super(USSocialSecurityNumberField, self).clean(value) |
|---|
| 59 |
if value in EMPTY_VALUES: |
|---|
| 60 |
return u'' |
|---|
| 61 |
match = re.match(ssn_re, value) |
|---|
| 62 |
if not match: |
|---|
| 63 |
raise ValidationError(self.error_messages['invalid']) |
|---|
| 64 |
area, group, serial = match.groupdict()['area'], match.groupdict()['group'], match.groupdict()['serial'] |
|---|
| 65 |
|
|---|
| 66 |
# First pass: no blocks of all zeroes. |
|---|
| 67 |
if area == '000' or \ |
|---|
| 68 |
group == '00' or \ |
|---|
| 69 |
serial == '0000': |
|---|
| 70 |
raise ValidationError(self.error_messages['invalid']) |
|---|
| 71 |
|
|---|
| 72 |
# Second pass: promotional and otherwise permanently invalid numbers. |
|---|
| 73 |
if area == '666' or \ |
|---|
| 74 |
(area == '987' and group == '65' and 4320 <= int(serial) <= 4329) or \ |
|---|
| 75 |
value == '078-05-1120' or \ |
|---|
| 76 |
value == '219-09-9999': |
|---|
| 77 |
raise ValidationError(self.error_messages['invalid']) |
|---|
| 78 |
return u'%s-%s-%s' % (area, group, serial) |
|---|
| 79 |
|
|---|
| 80 |
class USStateField(Field): |
|---|
| 81 |
""" |
|---|
| 82 |
A form field that validates its input is a U.S. state name or abbreviation. |
|---|
| 83 |
It normalizes the input to the standard two-leter postal service |
|---|
| 84 |
abbreviation for the given state. |
|---|
| 85 |
""" |
|---|
| 86 |
default_error_messages = { |
|---|
| 87 |
'invalid': u'Enter a U.S. state or territory.', |
|---|
| 88 |
} |
|---|
| 89 |
|
|---|
| 90 |
def clean(self, value): |
|---|
| 91 |
from us_states import STATES_NORMALIZED |
|---|
| 92 |
super(USStateField, self).clean(value) |
|---|
| 93 |
if value in EMPTY_VALUES: |
|---|
| 94 |
return u'' |
|---|
| 95 |
try: |
|---|
| 96 |
value = value.strip().lower() |
|---|
| 97 |
except AttributeError: |
|---|
| 98 |
pass |
|---|
| 99 |
else: |
|---|
| 100 |
try: |
|---|
| 101 |
return STATES_NORMALIZED[value.strip().lower()].decode('ascii') |
|---|
| 102 |
except KeyError: |
|---|
| 103 |
pass |
|---|
| 104 |
raise ValidationError(self.error_messages['invalid']) |
|---|
| 105 |
|
|---|
| 106 |
class USStateSelect(Select): |
|---|
| 107 |
""" |
|---|
| 108 |
A Select widget that uses a list of U.S. states/territories as its choices. |
|---|
| 109 |
""" |
|---|
| 110 |
def __init__(self, attrs=None): |
|---|
| 111 |
from us_states import STATE_CHOICES |
|---|
| 112 |
super(USStateSelect, self).__init__(attrs, choices=STATE_CHOICES) |
|---|