3dac701075562efc78f1c66be6ce582eb86c2ed2
[odoo/odoo.git] / addons / base_iban / base_iban.py
1 # -*- coding: utf-8 -*-
2 ##############################################################################
3 #
4 #    OpenERP, Open Source Management Solution
5 #    Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
6 #
7 #    This program is free software: you can redistribute it and/or modify
8 #    it under the terms of the GNU Affero General Public License as
9 #    published by the Free Software Foundation, either version 3 of the
10 #    License, or (at your option) any later version.
11 #
12 #    This program is distributed in the hope that it will be useful,
13 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
14 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 #    GNU Affero General Public License for more details.
16 #
17 #    You should have received a copy of the GNU Affero General Public License
18 #    along with this program.  If not, see <http://www.gnu.org/licenses/>.
19 #
20 ##############################################################################
21 import string
22
23 import netsvc
24 from osv import fields, osv
25 from tools.translate import _
26
27 # Length of IBAN
28 _iban_len = {'al':28, 'ad':24, 'at':20, 'be': 16, 'ba': 20, 'bg': 22, 'hr': 21, 'cy': 28,
29 'cz': 24, 'dk': 18, 'ee': 20, 'fo':18, 'fi': 18, 'fr': 27, 'ge': 22, 'de': 22, 'gi': 23,
30 'gr': 27, 'gl': 18, 'hu': 28, 'is':26, 'ie': 22, 'il': 23, 'it': 27, 'kz': 20, 'lv': 21,
31 'lb': 28, 'li': 21, 'lt': 20, 'lu':20 ,'mk': 19, 'mt': 31, 'mu': 30, 'mc': 27, 'gb': 22,
32 'me': 22, 'nl': 18, 'no': 15, 'pl':28, 'pt': 25, 'ro': 24, 'sm': 27, 'sa': 24, 'rs': 22,
33 'sk': 24, 'si': 19, 'es': 24, 'se':24, 'ch': 21, 'tn': 24, 'tr': 26}
34
35 # Reference Examples of IBAN
36 _ref_iban = { 'al':'ALkk BBBS SSSK CCCC CCCC CCCC CCCC', 'ad':'ADkk BBBB SSSS CCCC CCCC CCCC',
37 'at':'ATkk BBBB BCCC CCCC CCCC', 'be': 'BEkk BBBC CCCC CCKK', 'ba': 'BAkk BBBS SSCC CCCC CoKK',
38 'bg': 'BGkk BBBB SSSS DDCC CCCC CC', 'hr': 'HRkk BBBB BBBC CCCC CCCC C',
39 'cy': 'CYkk BBBS SSSS CCCC CCCC CCCC CCCC',
40 'cz': 'CZkk BBBB SSSS SSCC CCCC CCCC', 'dk': 'DKkk BBBB CCCC CCCC CC',
41  'ee': 'EEkk BBSS CCCC CCCC CCCK', 'fo': 'FOkk CCCC CCCC CCCC CC',
42  'fi': 'FIkk BBBB BBCC CCCC CK', 'fr': 'FRkk BBBB BGGG GGCC CCCC CCCC CKK',
43  'ge': 'GEkk BBCC CCCC CCCC CCCC CC', 'de': 'DEkk BBBB BBBB CCCC CCCC CC',
44  'gi': 'GIkk BBBB CCCC CCCC CCCC CCC', 'gr': 'GRkk BBBS SSSC CCCC CCCC CCCC CCC',
45  'gl': 'GLkk BBBB CCCC CCCC CC', 'hu': 'HUkk BBBS SSSC CCCC CCCC CCCC CCCC',
46  'is':'ISkk BBBB SSCC CCCC XXXX XXXX XX', 'ie': 'IEkk AAAA BBBB BBCC CCCC CC',
47  'il': 'ILkk BBBN NNCC CCCC CCCC CCC', 'it': 'ITkk KAAA AABB BBBC CCCC CCCC CCC',
48  'kz': 'KZkk BBBC CCCC CCCC CCCC', 'lv': 'LVkk BBBB CCCC CCCC CCCC C',
49 'lb': 'LBkk BBBB AAAA AAAA AAAA AAAA AAAA', 'li': 'LIkk BBBB BCCC CCCC CCCC C',
50 'lt': 'LTkk BBBB BCCC CCCC CCCC', 'lu': 'LUkk BBBC CCCC CCCC CCCC' ,
51 'mk': 'MKkk BBBC CCCC CCCC CKK', 'mt': 'MTkk BBBB SSSS SCCC CCCC CCCC CCCC CCC',
52 'mu': 'MUkk BBBB BBSS CCCC CCCC CCCC CCCC CC', 'mc': 'MCkk BBBB BGGG GGCC CCCC CCCC CKK',
53 'gb': 'GBkk BBBB SSSS SSCC CCCC CC', 'me': 'MEkk BBBC CCCC CCCC CCCC KK',
54 'nl': 'NLkk BBBB CCCC CCCC CC', 'no': 'NOkk BBBB CCCC CCK', 'pl':'PLkk BBBS SSSK CCCC CCCC CCCC CCCC',
55 'pt': 'PTkk BBBB SSSS CCCC CCCC CCCK K', 'ro': 'ROkk BBBB CCCC CCCC CCCC CCCC',
56 'sm': 'SMkk KAAA AABB BBBC CCCC CCCC CCC', 'sa': 'SAkk BBCC CCCC CCCC CCCC CCCC',
57 'rs': 'RSkk BBBC CCCC CCCC CCCC KK', 'sk': 'SKkk BBBB SSSS SSCC CCCC CCCC',
58 'si': 'SIkk BBSS SCCC CCCC CKK', 'es': 'ESkk BBBB GGGG KKCC CCCC CCCC',
59 'se': 'SEkk BBBB CCCC CCCC CCCC CCCC', 'ch': 'CHkk BBBB BCCC CCCC CCCC C',
60 'tn': 'TNkk BBSS SCCC CCCC CCCC CCCC', 'tr': 'TRkk BBBB BRCC CCCC CCCC CCCC CC'
61 }
62
63 def _format_iban(string):
64     '''
65     This function removes all characters from given 'string' that isn't a alpha numeric and converts it to upper case.
66     '''
67     res = ""
68     for char in string:
69         if char.isalnum():
70             res += char.upper()
71     return res
72
73 def _pretty_iban(string):
74     "return string in groups of four characters separated by a single space"
75     res = []
76     while string:
77         res.append(string[:4])
78         string = string[4:]
79     return ' '.join(res)
80
81 class res_partner_bank(osv.osv):
82     _inherit = "res.partner.bank"
83
84     def create(self, cr, uid, vals, context=None):
85         #overwrite to format the iban number correctly
86         if (vals.get('state',False)=='iban') and vals.get('acc_number', False):
87             vals['acc_number'] = _format_iban(vals['acc_number'])
88             vals['acc_number'] = _pretty_iban(vals['acc_number'])
89         return super(res_partner_bank, self).create(cr, uid, vals, context)
90
91     def write(self, cr, uid, ids, vals, context=None):
92         #overwrite to format the iban number correctly
93         if (vals.get('state',False)=='iban') and vals.get('acc_number', False):
94             vals['acc_number'] = _format_iban(vals['acc_number'])
95             vals['acc_number'] = _pretty_iban(vals['acc_number'])
96         return super(res_partner_bank, self).write(cr, uid, ids, vals, context)
97
98     def check_iban(self, cr, uid, ids, context=None):
99         '''
100         Check the IBAN number
101         '''
102         for bank_acc in self.browse(cr, uid, ids, context=context):
103             if bank_acc.state not in ('iban', 'iban-rib'):
104                 continue
105             iban = _format_iban(bank_acc.acc_number).lower()
106             if iban[:2] in _iban_len and len(iban) != _iban_len[iban[:2]]:
107                 return False
108             #the four first digits have to be shifted to the end
109             iban = iban[4:] + iban[:4]
110             #letters have to be transformed into numbers (a = 10, b = 11, ...)
111             iban2 = ""
112             for char in iban:
113                 if char.isalpha():
114                     iban2 += str(ord(char)-87)
115                 else:
116                     iban2 += char
117             #iban is correct if modulo 97 == 1
118             if not int(iban2) % 97 == 1:
119                 return False
120         return True
121
122     def _construct_constraint_msg(self, cr, uid, ids, context=None):
123
124         def default_iban_check(iban_cn):
125             return iban_cn[0] in string.ascii_lowercase and iban_cn[1] in string.ascii_lowercase
126
127         iban_country = self.browse(cr, uid, ids)[0].acc_number[:2].lower()
128         if default_iban_check(iban_country):
129             iban_example = iban_country in _ref_iban and _ref_iban[iban_country] + ' \nWhere A = Account number, B = National bank code, S = Branch code, C = account No, N = branch No, K = National check digits....' or ''
130             return _('The IBAN does not seem to be correct. You should have entered something like this %s'), (iban_example)
131         return _('The IBAN is invalid, it should begin with the country code'), ()
132
133     def _check_bank(self, cr, uid, ids, context=None):
134         for partner_bank in self.browse(cr, uid, ids, context=context):
135             if partner_bank.state == 'iban' and not partner_bank.bank.bic:
136                 return False
137         return True
138
139     def get_bban_from_iban(self, cr, uid, ids, context=None):
140         '''
141         This function returns the bank account number computed from the iban account number, thanks to the mapping_list dictionary that contains the rules associated to its country.
142         '''
143         res = {}
144         mapping_list = {
145          #TODO add rules for others countries
146             'be': lambda x: x[4:],
147             'fr': lambda x: x[14:],
148             'ch': lambda x: x[9:],
149             'gb': lambda x: x[14:],
150         }
151         for record in self.browse(cr, uid, ids, context=context):
152             if not record.acc_number:
153                 res[record.id] = False
154                 continue
155             res[record.id] = False
156             for code, function in mapping_list.items():
157                 if record.acc_number.lower().startswith(code):
158                     res[record.id] = function(record.acc_number)
159                     break
160         return res
161
162     _columns = {
163         # Deprecated: we keep it for backward compatibility, to be removed in v7
164         # We use acc_number instead of IBAN since v6.1, but we keep this field
165         # to not break community modules.
166         'iban': fields.related('acc_number', string='IBAN', size=34, readonly=True, help="International Bank Account Number", type="char"),
167     }
168     _constraints = [
169         (check_iban, _construct_constraint_msg, ["iban"]),
170         (_check_bank, '\nPlease define BIC/Swift code on bank for bank type IBAN Account to make valid payments', ['bic'])
171     ]
172
173 res_partner_bank()
174
175 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: