You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

99 lines
4.3 KiB

  1. # -*- coding: utf-8 -*-
  2. """Implement parser for MT940 files - Rabobank dialect."""
  3. ##############################################################################
  4. #
  5. # OpenERP, Open Source Management Solution
  6. # Copyright (C) 2015 Therp BV <http://therp.nl>.
  7. #
  8. # This program is free software: you can redistribute it and/or modify
  9. # it under the terms of the GNU Affero General Public License as
  10. # published by the Free Software Foundation, either version 3 of the
  11. # License, or (at your option) any later version.
  12. #
  13. # This program is distributed in the hope that it will be useful,
  14. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. # GNU Affero General Public License for more details.
  17. #
  18. # You should have received a copy of the GNU Affero General Public License
  19. # along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. #
  21. ##############################################################################
  22. import re
  23. from string import printable
  24. from datetime import datetime
  25. from openerp.addons.bank_statement_parse_mt940.mt940 import (
  26. MT940, str2amount, get_subfields, handle_common_subfields)
  27. class MT940Parser(MT940):
  28. """Implement parser for MT940 files - Rabobank dialect."""
  29. tag_61_regex = re.compile(
  30. r'^(?P<date>\d{6})(?P<sign>[CD])(?P<amount>\d+,\d{2})N(?P<type>.{3})'
  31. r'(?P<reference>\w{1,16})')
  32. def __init__(self):
  33. """Initialize parser - override at least header_regex."""
  34. super(MT940Parser, self).__init__()
  35. self.header_lines = 1 # Number of lines to skip
  36. # Do not user $ for end of string below: line contains much
  37. # more data than just the first line.
  38. self.header_regex = '^:940:' # Start of relevant data
  39. def parse(self, data):
  40. """Filter Unprintable characters from file data.
  41. The file contents of the Rabobank tend to contain unprintable
  42. characters that prevent proper parsing. These will be removed.
  43. """
  44. data = ''.join([x for x in data if x in printable])
  45. return super(MT940Parser, self).parse(data)
  46. def handle_tag_60F(self, data):
  47. """get start balance and currency"""
  48. # For the moment only first 60F record
  49. # The alternative would be to split the file and start a new
  50. # statement for each 20: tag encountered.
  51. stmt = self.current_statement
  52. if not stmt.local_currency:
  53. stmt.local_currency = data[7:10]
  54. stmt.date = datetime.strptime(data[1:7], '%y%m%d')
  55. stmt.start_balance = str2amount(data[0], data[10:])
  56. stmt.statement_id = '%s-%s' % (
  57. self.current_statement.date.strftime('%Y-%m-%d'),
  58. self.current_statement.statement_id)
  59. def handle_tag_61(self, data):
  60. """Handle tag 61: transaction data."""
  61. super(MT940Parser, self).handle_tag_61(data)
  62. parsed_data = self.tag_61_regex.match(data).groupdict()
  63. self.current_transaction.transferred_amount = (
  64. str2amount(parsed_data['sign'], parsed_data['amount']))
  65. self.current_transaction.eref = parsed_data['reference']
  66. def handle_tag_86(self, data):
  67. """Handle tag 86: transaction details"""
  68. if not self.current_transaction:
  69. return
  70. codewords = ['RTRN', 'BENM', 'ORDP', 'CSID', 'BUSP', 'MARF', 'EREF',
  71. 'PREF', 'REMI', 'ID', 'PURP', 'ULTB', 'ULTD',
  72. 'CREF', 'IREF', 'NAME', 'ADDR', 'ULTC', 'EXCH', 'CHGS']
  73. subfields = get_subfields(data, codewords)
  74. transaction = self.current_transaction
  75. # If we have no subfields, set message to whole of data passed:
  76. if not subfields:
  77. transaction.message = data
  78. else:
  79. handle_common_subfields(transaction, subfields)
  80. # Use subfields for transaction details:
  81. if 'NAME' in subfields:
  82. transaction.remote_owner = ' '.join(subfields['NAME'])
  83. if 'ADDR' in subfields:
  84. # Do NOT join address fields, array is expected on other code!
  85. transaction.remote_owner_address = subfields['ADDR']
  86. # Prevent handling tag 86 later for non transaction details:
  87. self.current_transaction = None
  88. # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: