[tryton-debian-vcs] tryton-modules-account-payment-sepa branch debian updated. debian/3.4.1-1-3-g86d62a3

Mathias Behrle tryton-debian-vcs at alioth.debian.org
Thu Apr 23 15:56:51 UTC 2015


The following commit has been merged in the debian branch:
https://alioth.debian.org/plugins/scmgit/cgi-bin/gitweb.cgi/?p=tryton/tryton-modules-account-payment-sepa.git;a=commitdiff;h=debian/3.4.1-1-3-g86d62a3

commit 86d62a33c5dff08e7ca6888d836ecf49662ab80e
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Thu Apr 23 17:32:37 2015 +0200

    Updating Depends.

diff --git a/debian/control b/debian/control
index 999b56a..1c5a2e3 100644
--- a/debian/control
+++ b/debian/control
@@ -18,9 +18,11 @@ Package: tryton-modules-account-payment-sepa
 Architecture: all
 Depends:
  python-genshi,
+ python-dateutil,
  python-lxml,
  python-pkg-resources,
  tryton-modules-account-payment (>= ${version:major}),
+ tryton-modules-account-payment-clearing (>= ${version:major}),
  tryton-modules-bank (>= ${version:major}),
  tryton-modules-company (>= ${version:major}),
  tryton-modules-party (>= ${version:major}),
commit 50857c88b8e511a5821997c6a0b19a73843b0598
Author: Mathias Behrle <mathiasb at m9s.biz>
Date:   Thu Apr 23 16:59:51 2015 +0200

    Merging upstream version 3.6.0.

diff --git a/CHANGELOG b/CHANGELOG
index 39d44f3..7dff6f2 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,5 +1,10 @@
-Version 3.4.1 - 2015-03-01
+Version 3.6.0 - 2015-04-20
 * Bug fixes (see mercurial logs for details)
+* Add support for PyPy
+* Use value date of camt.054 as clearing date
+* Allow to re-generate message of a payment group
+* Use right sequence type in case of previous rejected payments
+* Add pain.001.003.03 and pain.008.003.02 flavors
 
 Version 3.4.0 - 2014-10-20
 * Bug fixes (see mercurial logs for details)
diff --git a/INSTALL b/INSTALL
index 1341054..523986f 100644
--- a/INSTALL
+++ b/INSTALL
@@ -6,6 +6,7 @@ Prerequisites
 
  * Python 2.7 or later (http://www.python.org/)
  * Genshi (http://genshi.edgewall.org/)
+ * python-dateutil (https://dateutil.readthedocs.org/)
  * trytond (http://www.tryton.org/)
  * trytond_party (http://www.tryton.org/)
  * trytond_account_payment (http://www.tryton.org/)
diff --git a/PKG-INFO b/PKG-INFO
index 5a270b1..4a40a90 100644
--- a/PKG-INFO
+++ b/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.1
 Name: trytond_account_payment_sepa
-Version: 3.4.1
+Version: 3.6.0
 Summary: Tryton module for SEPA payment
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: issue_tracker at tryton.org
 License: GPL-3
-Download-URL: http://downloads.tryton.org/3.4/
+Download-URL: http://downloads.tryton.org/3.6/
 Description: trytond_account_payment_sepa
         ============================
         
@@ -64,5 +64,7 @@ Classifier: Natural Language :: Slovenian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
 Classifier: Programming Language :: Python :: 2.7
+Classifier: Programming Language :: Python :: Implementation :: CPython
+Classifier: Programming Language :: Python :: Implementation :: PyPy
 Classifier: Topic :: Office/Business
 Classifier: Topic :: Office/Business :: Financial :: Accounting
diff --git a/__init__.py b/__init__.py
index 6b0570d..5ac6d23 100644
--- a/__init__.py
+++ b/__init__.py
@@ -1,5 +1,5 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 from trytond.pool import Pool
 from .payment import *
 from .party import *
diff --git a/account.py b/account.py
index 5e3f57b..da88600 100644
--- a/account.py
+++ b/account.py
@@ -1,7 +1,8 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 from trytond.pool import PoolMeta
 from trytond.model import fields
+from trytond.pyson import Eval
 
 __all__ = ['Configuration']
 __metaclass__ = PoolMeta
@@ -12,4 +13,6 @@ class Configuration:
     sepa_mandate_sequence = fields.Property(fields.Many2One('ir.sequence',
             'SEPA Mandate Sequence', domain=[
                 ('code', '=', 'account.payment.sepa.mandate'),
+                ('company', 'in', [Eval('context', {}).get('company', -1),
+                        None]),
                 ]))
diff --git a/account.xml b/account.xml
index ec6a1d9..035654e 100644
--- a/account.xml
+++ b/account.xml
@@ -8,7 +8,9 @@ this repository contains the full copyright notices and license terms. -->
             id="act_move_line_form_domain_receivable_mandate">
             <field name="name">Receivable with Mandate</field>
             <field name="sequence" eval="25"/>
-            <field name="domain">['AND', ['OR', ('debit', '>', 0), ('credit', '<', 0)], ('party.sepa_mandates', '!=', None)]</field>
+            <field name="domain"
+                eval="['AND', ['OR', ('debit', '>', 0), ('credit', '<', 0)], ('party.sepa_mandates', '!=', None)]"
+                pyson="1"/>
             <field name="act_window" ref="account_payment.act_move_line_form"/>
         </record>
 
diff --git a/locale/ca_ES.po b/locale/ca_ES.po
index 7964207..d3cf77a 100644
--- a/locale/ca_ES.po
+++ b/locale/ca_ES.po
@@ -233,7 +233,7 @@ msgstr "Cancel·lat"
 
 msgctxt "model:ir.action.act_window.domain,name:act_message_form_domain_done"
 msgid "Done"
-msgstr "Realitzat"
+msgstr "Finalitzat"
 
 msgctxt "model:ir.action.act_window.domain,name:act_message_form_domain_draft"
 msgid "Draft"
@@ -451,7 +451,7 @@ msgstr "Cancel·lat"
 
 msgctxt "selection:account.payment.sepa.message,state:"
 msgid "Done"
-msgstr "Realitzat"
+msgstr "Finalitzat"
 
 msgctxt "selection:account.payment.sepa.message,state:"
 msgid "Draft"
@@ -469,6 +469,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "Sortida"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Missatge generat"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
diff --git a/locale/de_DE.po b/locale/de_DE.po
index e3580a5..0688785 100644
--- a/locale/de_DE.po
+++ b/locale/de_DE.po
@@ -466,6 +466,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "OUT"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Nachricht erzeugen"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
@@ -500,7 +504,7 @@ msgstr "Prüfen"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "Cancel"
-msgstr "Annulliert"
+msgstr "Annullieren"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "Do"
diff --git a/locale/es_AR.po b/locale/es_AR.po
index 45c6fd2..28909a8 100644
--- a/locale/es_AR.po
+++ b/locale/es_AR.po
@@ -465,6 +465,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "SALIDA"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Generar mensaje"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "Área Única de Pagos en Euros (SEPA)"
diff --git a/locale/es_CO.po b/locale/es_CO.po
index 20fe05c..e9d49f4 100644
--- a/locale/es_CO.po
+++ b/locale/es_CO.po
@@ -102,7 +102,7 @@ msgstr "Identificación de Solo Lectura"
 
 msgctxt "field:account.payment.sepa.mandate,party:"
 msgid "Party"
-msgstr "Terceros"
+msgstr "Tercero"
 
 msgctxt "field:account.payment.sepa.mandate,payments:"
 msgid "Payments"
@@ -178,7 +178,7 @@ msgstr "Tipo"
 
 msgctxt "field:account.payment.sepa.message,write_date:"
 msgid "Write Date"
-msgstr "Modificado por Usuario"
+msgstr "Fecha de Modificación"
 
 msgctxt "field:account.payment.sepa.message,write_uid:"
 msgid "Write User"
@@ -463,6 +463,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "SALIDA"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Generar Mensaje"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
diff --git a/locale/es_EC.po b/locale/es_EC.po
index 0d938ef..d7793e7 100644
--- a/locale/es_EC.po
+++ b/locale/es_EC.po
@@ -465,6 +465,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "Salida"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Generar mensaje"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
diff --git a/locale/es_ES.po b/locale/es_ES.po
index 9313192..ac4e247 100644
--- a/locale/es_ES.po
+++ b/locale/es_ES.po
@@ -233,7 +233,7 @@ msgstr "Cancelado"
 
 msgctxt "model:ir.action.act_window.domain,name:act_message_form_domain_done"
 msgid "Done"
-msgstr "Realizado"
+msgstr "Finalizado"
 
 msgctxt "model:ir.action.act_window.domain,name:act_message_form_domain_draft"
 msgid "Draft"
@@ -449,7 +449,7 @@ msgstr "Cancelado"
 
 msgctxt "selection:account.payment.sepa.message,state:"
 msgid "Done"
-msgstr "Realizado"
+msgstr "Finalizado"
 
 msgctxt "selection:account.payment.sepa.message,state:"
 msgid "Draft"
@@ -467,6 +467,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "Salida"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Generar mensaje"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
diff --git a/locale/fr_FR.po b/locale/fr_FR.po
index 6e21205..5499bb9 100644
--- a/locale/fr_FR.po
+++ b/locale/fr_FR.po
@@ -465,6 +465,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "Sortant"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Générer le message"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
diff --git a/locale/sl_SI.po b/locale/sl_SI.po
index 5db1d7f..34c380e 100644
--- a/locale/sl_SI.po
+++ b/locale/sl_SI.po
@@ -18,7 +18,7 @@ msgstr "Soglasja \"%s\" ni možno zbrisati, ker ni v pripravi ali preklicano."
 
 msgctxt "field:account.configuration,sepa_mandate_sequence:"
 msgid "SEPA Mandate Sequence"
-msgstr "Štetje soglasij SEPA"
+msgstr "Štetje SEPA soglasij"
 
 msgctxt "field:account.payment,sepa_end_to_end_id:"
 msgid "SEPA End To End ID"
@@ -26,7 +26,7 @@ msgstr "ID SEPA End-to-End "
 
 msgctxt "field:account.payment,sepa_instruction_id:"
 msgid "SEPA Instruction ID"
-msgstr "ID navodila SEPA"
+msgstr "ID SEPA navodila"
 
 msgctxt "field:account.payment,sepa_mandate:"
 msgid "Mandate"
@@ -46,7 +46,7 @@ msgstr "Podrobnosti razloga zavrnitve"
 
 msgctxt "field:account.payment.group,sepa_messages:"
 msgid "SEPA Messages"
-msgstr "Sporočila SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "field:account.payment.journal,company_party:"
 msgid "Company Party"
@@ -126,7 +126,7 @@ msgstr "Stanje"
 
 msgctxt "field:account.payment.sepa.mandate,type:"
 msgid "Type"
-msgstr "Vrsta"
+msgstr "Tip"
 
 msgctxt "field:account.payment.sepa.mandate,write_date:"
 msgid "Write Date"
@@ -174,7 +174,7 @@ msgstr "Stanje"
 
 msgctxt "field:account.payment.sepa.message,type:"
 msgid "Type"
-msgstr "Vrsta"
+msgstr "Tip"
 
 msgctxt "field:account.payment.sepa.message,write_date:"
 msgid "Write Date"
@@ -186,27 +186,27 @@ msgstr "Zapisal"
 
 msgctxt "field:party.party,sepa_creditor_identifier:"
 msgid "SEPA Creditor Identifier"
-msgstr "Identifikacija upnika SEPA"
+msgstr "Identifikacija  SEPA upnika"
 
 msgctxt "field:party.party,sepa_creditor_identifier_used:"
 msgid "SEPA Creditor Identifier Used"
-msgstr "Uporabljena Identifikacija upnika SEPA"
+msgstr "Uporabljena Identifikacija SEPA upnika"
 
 msgctxt "field:party.party,sepa_mandates:"
 msgid "SEPA Mandates"
-msgstr "Soglasja SEPA"
+msgstr "SEPA soglasja"
 
 msgctxt "model:account.payment.sepa.mandate,name:"
 msgid "SEPA Mandate"
-msgstr "Soglasje SEPA"
+msgstr "SEPA soglasje"
 
 msgctxt "model:account.payment.sepa.message,name:"
 msgid "SEPA Message"
-msgstr "Sporočilo SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "model:ir.action,name:act_mandate_form"
 msgid "SEPA Mandates"
-msgstr "Soglasja SEPA"
+msgstr "SEPA soglasja"
 
 msgctxt "model:ir.action,name:act_mandate_form2"
 msgid "Mandates"
@@ -214,7 +214,7 @@ msgstr "Soglasja"
 
 msgctxt "model:ir.action,name:act_message_form"
 msgid "SEPA Messages"
-msgstr "Sporočila SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "model:ir.action,name:report_mandate"
 msgid "Mandate"
@@ -235,7 +235,7 @@ msgstr "Zaključeno"
 
 msgctxt "model:ir.action.act_window.domain,name:act_message_form_domain_draft"
 msgid "Draft"
-msgstr "V pripravi"
+msgstr "Priprava"
 
 msgctxt ""
 "model:ir.action.act_window.domain,name:act_message_form_domain_waiting"
@@ -249,15 +249,15 @@ msgstr "Terjatve s soglasjem"
 
 msgctxt "model:ir.sequence.type,name:sequence_type_mandate"
 msgid "SEPA Mandate"
-msgstr "Soglasje SEPA"
+msgstr "SEPA soglasje"
 
 msgctxt "model:ir.ui.menu,name:menu_mandate_form"
 msgid "SEPA Mandates"
-msgstr "Soglasja SEPA"
+msgstr "SEPA soglasja"
 
 msgctxt "model:ir.ui.menu,name:menu_message_form"
 msgid "SEPA Messages"
-msgstr "Sporočila SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "odt:account.payment.sepa.mandate:"
 msgid "."
@@ -312,7 +312,7 @@ msgstr "Datum"
 
 msgctxt "odt:account.payment.sepa.mandate:"
 msgid "Direct Debit MANDATE"
-msgstr "Soglasje direktne obremenitve"
+msgstr "Soglasje direktne bremenitve"
 
 msgctxt "odt:account.payment.sepa.mandate:"
 msgid "E-Mail:"
@@ -462,6 +462,10 @@ msgctxt "selection:account.payment.sepa.message,type:"
 msgid "OUT"
 msgstr "Izhod"
 
+msgctxt "view:account.payment.group:"
+msgid "Generate Message"
+msgstr "Izdelava sporočila"
+
 msgctxt "view:account.payment.journal:"
 msgid "SEPA"
 msgstr "SEPA"
@@ -504,15 +508,15 @@ msgstr "Izvedi"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "Draft"
-msgstr "V pripravi"
+msgstr "Priprava"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "SEPA Message"
-msgstr "Sporočilo SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "SEPA Messages"
-msgstr "Sporočila SEPA"
+msgstr "SEPA sporočilo"
 
 msgctxt "view:account.payment.sepa.message:"
 msgid "Wait"
diff --git a/mandate.odt b/mandate.odt
index ac7e364..44937ce 100644
Binary files a/mandate.odt and b/mandate.odt differ
diff --git a/party.py b/party.py
index 5154440..944635e 100644
--- a/party.py
+++ b/party.py
@@ -1,5 +1,5 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 from trytond.pool import PoolMeta
 from trytond.model import fields
 
diff --git a/party.xml b/party.xml
index 8783a35..688d00c 100644
--- a/party.xml
+++ b/party.xml
@@ -11,7 +11,8 @@ this repository contains the full copyright notices and license terms. -->
         <record model="ir.action.act_window" id="act_mandate_form2">
             <field name="name">Mandates</field>
             <field name="res_model">account.payment.sepa.mandate</field>
-            <field name="domain">[("party", "=", Eval('active_id'))]</field>
+            <field name="domain"
+                eval="[('party', '=', Eval('active_id'))]" pyson="1"/>
         </record>
         <record model="ir.action.act_window.view" id="act_mandate_form2_view1">
             <field name="sequence" eval="10"/>
diff --git a/payment.py b/payment.py
index 6ce3d7c..4d6da05 100644
--- a/payment.py
+++ b/payment.py
@@ -1,7 +1,8 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 import datetime
 import os
+import unicodedata
 from itertools import groupby
 from io import BytesIO
 
@@ -11,7 +12,7 @@ from lxml import etree
 from sql import Literal
 
 from trytond.pool import PoolMeta, Pool
-from trytond.model import ModelSQL, ModelView, Workflow, fields
+from trytond.model import ModelSQL, ModelView, Workflow, fields, dualmethod
 from trytond.pyson import Eval, If
 from trytond.transaction import Transaction
 from trytond.tools import reduce_ids, grouped_slice
@@ -43,6 +44,7 @@ class Journal:
             (None, ''),
             ('pain.001.001.03', 'pain.001.001.03'),
             ('pain.001.001.05', 'pain.001.001.05'),
+            ('pain.001.003.03', 'pain.001.003.03'),
             ], 'Payable Flavor', states={
             'required': Eval('process_method') == 'sepa',
             'invisible': Eval('process_method') != 'sepa',
@@ -53,6 +55,7 @@ class Journal:
             (None, ''),
             ('pain.008.001.02', 'pain.008.001.02'),
             ('pain.008.001.04', 'pain.008.001.04'),
+            ('pain.008.003.02', 'pain.008.003.02'),
             ], 'Receivable Flavor', states={
             'required': Eval('process_method') == 'sepa',
             'invisible': Eval('process_method') != 'sepa',
@@ -127,6 +130,9 @@ class Group:
         cls._error_messages.update({
                 'no_mandate': 'No valid mandate for payment "%s"',
                 })
+        cls._buttons.update({
+                'generate_message': {},
+                })
 
     def get_sepa_template(self):
         if self.kind == 'payable':
@@ -137,11 +143,9 @@ class Group:
     def process_sepa(self):
         pool = Pool()
         Payment = pool.get('account.payment')
-        Message = pool.get('account.payment.sepa.message')
         if self.kind == 'receivable':
-            payments = [p for p in self.payments if not p.sepa_mandate]
-            mandates = Payment.get_sepa_mandates(payments)
-            for payment, mandate in zip(payments, mandates):
+            mandates = Payment.get_sepa_mandates(self.payments)
+            for payment, mandate in zip(self.payments, mandates):
                 if not mandate:
                     self.raise_user_error('no_mandate', payment.rec_name)
                 # Write one by one becasue mandate.sequence_type must be
@@ -150,15 +154,27 @@ class Group:
                         'sepa_mandate': mandate,
                         'sepa_mandate_sequence_type': mandate.sequence_type,
                         })
-        tmpl = self.get_sepa_template()
-        if not tmpl:
-            raise NotImplementedError
-        if not self.sepa_messages:
-            self.sepa_messages = ()
-        message = tmpl.generate(group=self,
-            datetime=datetime).filter(remove_comment).render()
-        message = Message(message=message, type='out', state='waiting')
-        self.sepa_messages += (message,)
+        self.generate_message(_save=False)
+
+    @dualmethod
+    @ModelView.button
+    def generate_message(cls, groups, _save=True):
+        pool = Pool()
+        Message = pool.get('account.payment.sepa.message')
+        for group in groups:
+            tmpl = group.get_sepa_template()
+            if not tmpl:
+                raise NotImplementedError
+            if not group.sepa_messages:
+                group.sepa_messages = ()
+            message = tmpl.generate(group=group,
+                datetime=datetime, normalize=unicodedata.normalize,
+                ).filter(remove_comment).render()
+            message = Message(message=message, type='out', state='waiting',
+                company=group.company)
+            group.sepa_messages += (message,)
+        if _save:
+            cls.save(groups)
 
     @property
     def sepa_initiating_party(self):
@@ -192,8 +208,9 @@ class Payment:
         ondelete='RESTRICT',
         domain=[
             ('party', '=', Eval('party', -1)),
+            ('company', '=', Eval('company', -1)),
             ],
-        depends=['party'])
+        depends=['party', 'company'])
     sepa_mandate_sequence_type = fields.Char('Mandate Sequence Type',
         readonly=True)
     sepa_return_reason_code = fields.Char('Return Reason Code', readonly=True,
@@ -215,14 +232,27 @@ class Payment:
         'get_sepa_instruction_id', searcher='search_sepa_instruction_id')
 
     @classmethod
+    def copy(cls, payments, default=None):
+        if default is None:
+            default = {}
+        default.setdefault('sepa_mandate_sequence_type', None)
+        return super(Payment, cls).copy(payments, default=default)
+
+    @classmethod
     def get_sepa_mandates(cls, payments):
         mandates = []
         for payment in payments:
-            for mandate in payment.party.sepa_mandates:
-                if mandate.is_valid:
-                    break
+            if payment.sepa_mandate:
+                if payment.sepa_mandate.is_valid:
+                    mandate = payment.sepa_mandate
+                else:
+                    mandate = None
             else:
-                mandate = None
+                for mandate in payment.party.sepa_mandates:
+                    if mandate.is_valid:
+                        break
+                else:
+                    mandate = None
             mandates.append(mandate)
         return mandates
 
@@ -233,7 +263,7 @@ class Payment:
     def search_end_to_end_id(cls, name, domain):
         table = cls.__table__()
         _, operator, value = domain
-        cast = cls.sepa_end_to_end_id._field.sql_type().base
+        cast = cls.sepa_end_to_end_id.sql_type().base
         Operator = fields.SQL_OPERATORS[operator]
         query = table.select(table.id,
             where=Operator(table.id.cast(cast), value))
@@ -260,6 +290,17 @@ class Payment:
                     if number.type == 'iban':
                         return number
 
+    @property
+    def rejected(self):
+        return (self.state == 'failed'
+            and self.sepa_return_reason_code
+            and self.sepa_return_reason_information == '/RTYP/RJCT')
+
+    def create_clearing_move(self, date=None):
+        if not date:
+            date = Transaction().context.get('date_value')
+        return super(Payment, self).create_clearing_move(date=date)
+
 
 class Mandate(Workflow, ModelSQL, ModelView):
     'SEPA Mandate'
@@ -271,6 +312,7 @@ class Mandate(Workflow, ModelSQL, ModelView):
             },
         depends=['state'])
     account_number = fields.Many2One('bank.account.number', 'Account Number',
+        ondelete='RESTRICT',
         states={
             'readonly': Eval('state').in_(['validated', 'canceled']),
             'required': Eval('state') == 'validated',
@@ -443,7 +485,9 @@ class Mandate(Workflow, ModelSQL, ModelView):
     def sequence_type(self):
         if self.type == 'one-off':
             return 'OOFF'
-        elif not self.payments:
+        elif (not self.payments
+                or all(not p.sepa_mandate_sequence_type for p in self.payments)
+                or all(p.rejected for p in self.payments)):
             return 'FRST'
         # TODO manage FNAL
         else:
diff --git a/payment.xml b/payment.xml
index 751007e..87b796c 100644
--- a/payment.xml
+++ b/payment.xml
@@ -55,10 +55,19 @@ this repository contains the full copyright notices and license terms. -->
             <field name="global_p" eval="True"/>
         </record>
         <record model="ir.rule" id="rule_payment1">
-            <field name="domain">[('company', '=', user.company.id if user.company else None)]</field>
+            <field name="domain"
+                eval="[('company', '=', Eval('user', {}).get('company', None))]"
+                pyson="1"/>
             <field name="rule_group" ref="rule_group_mandate"/>
         </record>
 
+        <record model="ir.model.access" id="access_mandate">
+            <field name="model" search="[('model', '=', 'account.payment.sepa.mandate')]"/>
+            <field name="perm_read" eval="False"/>
+            <field name="perm_write" eval="False"/>
+            <field name="perm_create" eval="False"/>
+            <field name="perm_delete" eval="False"/>
+        </record>
         <record model="ir.model.access" id="access_mandate_account_admin">
             <field name="model" search="[('model', '=', 'account.payment.sepa.mandate')]"/>
             <field name="group" ref="account.group_account_admin"/>
@@ -121,25 +130,25 @@ this repository contains the full copyright notices and license terms. -->
         <record model="ir.action.act_window.domain" id="act_message_form_domain_draft">
             <field name="name">Draft</field>
             <field name="sequence" eval="10"/>
-            <field name="domain">[('state', '=', 'draft')]</field>
+            <field name="domain" eval="[('state', '=', 'draft')]" pyson="1"/>
             <field name="act_window" ref="act_message_form"/>
         </record>
         <record model="ir.action.act_window.domain" id="act_message_form_domain_waiting">
             <field name="name">Waiting</field>
             <field name="sequence" eval="20"/>
-            <field name="domain">[('state', '=', 'waiting')]</field>
+            <field name="domain" eval="[('state', '=', 'waiting')]" pyson="1"/>
             <field name="act_window" ref="act_message_form"/>
         </record>
         <record model="ir.action.act_window.domain" id="act_message_form_domain_done">
             <field name="name">Done</field>
             <field name="sequence" eval="30"/>
-            <field name="domain">[('state', '=', 'done')]</field>
+            <field name="domain" eval="[('state', '=', 'done')]" pyson="1"/>
             <field name="act_window" ref="act_message_form"/>
         </record>
         <record model="ir.action.act_window.domain" id="act_message_form_domain_canceled">
             <field name="name">Canceled</field>
             <field name="sequence" eval="40"/>
-            <field name="domain">[('state', '=', 'canceled')]</field>
+            <field name="domain" eval="[('state', '=', 'canceled')]" pyson="1"/>
             <field name="act_window" ref="act_message_form"/>
         </record>
         <record model="ir.action.act_window.domain" id="act_message_form_domain_all">
@@ -157,10 +166,20 @@ this repository contains the full copyright notices and license terms. -->
             <field name="global_p" eval="True"/>
         </record>
         <record model="ir.rule" id="rule_message1">
-            <field name="domain">[('company', '=', user.company.id if user.company else None)]</field>
+            <field name="domain"
+                eval="[('company', '=', Eval('user', {}).get('company', None))]"
+                pyson="1"/>
             <field name="rule_group" ref="rule_group_message"/>
         </record>
 
+        <record model="ir.model.access" id="access_message">
+            <field name="model"
+                search="[('model', '=', 'account.payment.sepa.message')]"/>
+            <field name="perm_read" eval="False"/>
+            <field name="perm_write" eval="False"/>
+            <field name="perm_create" eval="False"/>
+            <field name="perm_delete" eval="False"/>
+        </record>
         <record model="ir.model.access" id="access_message_account_admin">
             <field name="model" search="[('model', '=', 'account.payment.sepa.message')]"/>
             <field name="group" ref="account.group_account_admin"/>
diff --git a/sepa_handler.py b/sepa_handler.py
index 7c4271c..2360b1d 100644
--- a/sepa_handler.py
+++ b/sepa_handler.py
@@ -1,6 +1,9 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 from lxml import etree
+from dateutil.parser import parse
+
+from trytond.transaction import Transaction
 
 __all__ = ['SEPAHandler', 'CAMT054']
 
@@ -32,13 +35,16 @@ class CAMT054(SEPAHandler):
             element.clear()
 
     def handle_entry(self, element):
-        payment = self.get_payment(element)
+        payments = self.get_payments(element)
         if self.is_returned(element):
-            self.set_return_information(payment, element)
-            payment.save()
-            self.Payment.fail([payment])
+            for payment in payments:
+                self.set_return_information(payment, element)
+            self.Payment.save(payments)
+            self.Payment.fail(payments)
         else:
-            self.Payment.succeed([payment])
+            date_value = self.date_value(element)
+            with Transaction().set_context(date_value=date_value):
+                self.Payment.succeed(payments)
 
     def get_payment_kind(self, element):
         tag = etree.QName(element)
@@ -49,7 +55,7 @@ class CAMT054(SEPAHandler):
         'DBIT': 'receivable',
         }
 
-    def get_payment(self, element):
+    def get_payments(self, element):
         tag = etree.QName(element)
         details = element.find('./{%s}NtryDtls' % tag.namespace)
         if details is None:
@@ -59,18 +65,30 @@ class CAMT054(SEPAHandler):
             return
         instr_id = details.find('.//{%s}InstrId' % tag.namespace)
         if instr_id is not None:
-            payment, = self.Payment.search([
+            payments = self.Payment.search([
                     ('sepa_instuction_id', '=', instr_id.text),
                     ('kind', '=', self.get_payment_kind(element)),
                     ])
-            return payment
+            return payments
         end_to_end_id = details.find('.//{%s}EndToEndId' % tag.namespace)
         if end_to_end_id is not None:
-            payment, = self.Payment.search([
+            payments = self.Payment.search([
                     ('sepa_end_to_end_id', '=', end_to_end_id.text),
                     ('kind', '=', self.get_payment_kind(element)),
                     ])
-            return payment
+            return payments
+
+    def date_value(self, element):
+        tag = etree.QName(element)
+        date = element.find('./{%(ns)s}ValDt/{%(ns)s}Dt'
+            % {'ns': tag.namespace})
+        if date is not None:
+            return parse(date.text)
+        else:
+            datetime = element.find('./{%(ns)s}ValDt/{%(ns)s}DtTm'
+                % {'ns': tag.namespace})
+            if datetime:
+                return parse(datetime.text).date()
 
     def is_returned(self, element):
         tag = etree.QName(element)
diff --git a/setup.py b/setup.py
index 545faeb..854e49b 100644
--- a/setup.py
+++ b/setup.py
@@ -1,6 +1,6 @@
 #!/usr/bin/env python
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 
 from setuptools import setup
 import re
@@ -41,7 +41,7 @@ if minor_version % 2:
         'hg+http://hg.tryton.org/modules/%s#egg=%s-%s' % (
             name[8:], name, version))
 
-requires = ['Genshi', 'lxml']
+requires = ['Genshi', 'lxml', 'python-dateutil']
 for dep in info.get('depends', []):
     if not re.match(r'(ir|res|webdav)(\W|$)', dep):
         requires.append(get_require_version('trytond_%s' % dep))
@@ -88,6 +88,8 @@ setup(name=name,
         'Natural Language :: Spanish',
         'Operating System :: OS Independent',
         'Programming Language :: Python :: 2.7',
+        'Programming Language :: Python :: Implementation :: CPython',
+        'Programming Language :: Python :: Implementation :: PyPy',
         'Topic :: Office/Business',
         'Topic :: Office/Business :: Financial :: Accounting',
         ],
diff --git a/template/base.003.xml b/template/base.003.xml
new file mode 100644
index 0000000..39555e0
--- /dev/null
+++ b/template/base.003.xml
@@ -0,0 +1,53 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!-- This file is part of Tryton.  The COPYRIGHT file at the top level of
+this repository contains the full copyright notices and license terms. -->
+<py:strip xmlns:py="http://genshi.edgewall.org/">
+    <!-- only ASCII -->
+    <py:def function="PartyIdentification(party, id=True, with_name=True, with_address=True)">
+        <!-- EPC limits to 70 instead of 140 -->
+        <Nm py:if="with_name">${normalize('NFKD', party.name).encode('ascii', 'replace')[:70]}</Nm>
+        <py:if test="with_address">
+            <py:with vars="address = party.address_get()">
+                <PstlAdr py:if="address">
+                    ${PostalAddress(address)}
+                </PstlAdr>
+            </py:with>
+        </py:if>
+        <Id py:if="id and party.sepa_creditor_identifier_used">
+            <PrvtId>
+                <Othr>
+                    <Id>${party.sepa_creditor_identifier_used}</Id>
+                    <SchmeNm>
+                        <Prtry>SEPA</Prtry>
+                    </SchmeNm>
+                    <!-- Issr -->
+                </Othr>
+            </PrvtId>
+        </Id>
+        <!-- CtryOfRes -->
+        <!-- CtctDtls -->
+    </py:def>
+    <py:def function="PostalAddress(address)">
+        <Ctry py:if="address.country">${normalize('NFKD', address.country.code).encode('ascii', 'replace')}</Ctry>
+        <AdrLine py:if="address.street">${normalize('NFKD', address.street).encode('ascii', 'replace')[:70]}</AdrLine>
+        <AdrLine py:if="address.streetbis">${normalize('NFKD', address.streetbis).encode('ascii', 'replace')[:70]}</AdrLine>
+    </py:def>
+    <py:def function="Account(account_number, with_ccy=True)">
+        <Id>
+            <!-- EPC only IBAN is allowed -->
+            <IBAN>${account_number.compact_iban}</IBAN>
+        </Id>
+        <!-- Tp -->
+        <Ccy py:if="with_ccy and account_number.account.currency">${account_number.account.currency.code}</Ccy>
+        <!-- Nm -->
+    </py:def>
+    <!-- uses BIC instead of BICFI -->
+    <py:def function="FinancialInstitution(bank, only_bic=False)">
+        <FinInstnId>
+            <BIC py:if="bank.bic">${bank.bic}</BIC>
+            <Othr py:if="not bank.bic">
+                <Id>NOTPROVIDED</Id>
+            </Othr>
+        </FinInstnId>
+    </py:def>
+</py:strip>
diff --git a/template/base.xml b/template/base.xml
index c6830af..0441aa4 100644
--- a/template/base.xml
+++ b/template/base.xml
@@ -36,13 +36,13 @@ this repository contains the full copyright notices and license terms. -->
         <Ctry py:if="address.country">${address.country.code}</Ctry>
         <!-- AdrLine -->
     </py:def>
-    <py:def function="Account(account_number)">
+    <py:def function="Account(account_number, currency=True)">
         <Id>
             <!-- EPC only IBAN is allowed -->
             <IBAN>${account_number.compact_iban}</IBAN>
         </Id>
         <!-- Tp -->
-        <Ccy py:if="account_number.account.currency">${account_number.account.currency.code}</Ccy>
+        <Ccy py:if="currency and account_number.account.currency">${account_number.account.currency.code}</Ccy>
         <!-- Nm -->
     </py:def>
     <py:def function="FinancialInstitution(bank, only_bic=False)">
diff --git a/template/pain.001.001.03.xml b/template/pain.001.001.03.xml
index 29ea6d6..25073f6 100644
--- a/template/pain.001.001.03.xml
+++ b/template/pain.001.001.03.xml
@@ -31,7 +31,7 @@ this repository contains the full copyright notices and license terms. -->
             <MsgId>${group.reference[:35]}</MsgId>
             <CreDtTm>${datetime.datetime.now().isoformat()}</CreDtTm>
             <!-- Authstn -->
-            <NbOfTxs>${len(group.payments)}</NbOfTxs>
+            <NbOfTxs>${sum(len(payments) for _, payments in group.sepa_payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in group.payments)}</CtrlSum>
             <!-- PmtTpInf -->
             <!-- ReqdColltnDt -->
diff --git a/template/pain.001.001.05.xml b/template/pain.001.001.05.xml
index db275a5..319afcb 100644
--- a/template/pain.001.001.05.xml
+++ b/template/pain.001.001.05.xml
@@ -12,7 +12,7 @@ this repository contains the full copyright notices and license terms. -->
             <MsgId>${group.reference[:35]}</MsgId>
             <CreDtTm>${datetime.datetime.now().isoformat()}</CreDtTm>
             <!-- Authstn -->
-            <NbOfTxs>${len(group.payments)}</NbOfTxs>
+            <NbOfTxs>${sum(len(payments) for _, payments in group.sepa_payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in group.payments)}</CtrlSum>
             <!-- PmtTpInf -->
             <!-- ReqdColltnDt -->
diff --git a/template/pain.001.001.05.xml b/template/pain.001.003.03.xml
similarity index 69%
copy from template/pain.001.001.05.xml
copy to template/pain.001.003.03.xml
index db275a5..9b613b4 100644
--- a/template/pain.001.001.05.xml
+++ b/template/pain.001.003.03.xml
@@ -1,25 +1,24 @@
 <?xml version="1.0" encoding="UTF-8"?>
 <!-- This file is part of Tryton.  The COPYRIGHT file at the top level of
 this repository contains the full copyright notices and license terms. -->
-<Document xmlns="urn:iso:std:iso:20022:tech:xsd:pain.001.001.05"
+<Document xmlns="urn:iso:std:iso:20022:tech:xsd:pain.001.003.03"
     xmlns:xsd="http://www.w3.org/2001/XMLSchema"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+    xsi:schemaLocation="urn:iso:std:iso:20022:tech:xsd:pain.001.003.03
+    pain.001.003.03.xsd"
     xmlns:xi="http://www.w3.org/2001/XInclude"
     xmlns:py="http://genshi.edgewall.org/">
     <xi:include href="base.xml"/>
+    <xi:include href="base.003.xml"/>
     <CstmrCdtTrfInitn>
         <GrpHdr>
             <MsgId>${group.reference[:35]}</MsgId>
             <CreDtTm>${datetime.datetime.now().isoformat()}</CreDtTm>
-            <!-- Authstn -->
-            <NbOfTxs>${len(group.payments)}</NbOfTxs>
+            <NbOfTxs>${sum(len(payments) for _, payments in group.sepa_payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in group.payments)}</CtrlSum>
-            <!-- PmtTpInf -->
-            <!-- ReqdColltnDt -->
             <InitgPty>
-                ${PartyIdentification(group.sepa_initiating_party)}
+                ${PartyIdentification(group.sepa_initiating_party, with_address=False)}
             </InitgPty>
-            <!-- FwdgAgt -->
         </GrpHdr>
         <PmtInf py:for="key, payments in group.sepa_payments">
             <PmtInfId>${group.sepa_group_payment_id(key)[:35]}</PmtInfId>
@@ -32,11 +31,9 @@ this repository contains the full copyright notices and license terms. -->
                 <SvcLvl>
                     <Cd>SEPA</Cd>
                 </SvcLvl>
-                <!-- LclInstrm -->
                 <!-- CtgyPurp -->
             </PmtTpInf>
             <ReqdExctnDt>${key['date'].isoformat()}</ReqdExctnDt>
-            <!-- PoolgAdjstmntDt -->
             <Dbtr>
                 ${PartyIdentification(group.company.party)}
             </Dbtr>
@@ -46,12 +43,8 @@ this repository contains the full copyright notices and license terms. -->
             <DbtrAgt>
                 ${FinancialInstitution(group.journal.sepa_bank_account_number.account.bank, only_bic=True)}
             </DbtrAgt>
-            <!-- DbtrAgtAcct -->
-            <!-- InstrForDbtrAgt -->
             <!-- UltmtDbtr -->
-            <ChrgBr>${group.journal.sepa_charge_bearer}</ChrgBr>
-            <!-- ChrgsAcct -->
-            <!-- ChrgsAcctAgt -->
+            <ChrgBr>SLEV</ChrgBr>
             <CdtTrfTxInf py:for="payment in payments">
                 <PmtId>
                     <InstrId>${payment.sepa_instruction_id}</InstrId>
@@ -60,41 +53,24 @@ this repository contains the full copyright notices and license terms. -->
                 <!-- PmtTpInf --> <!-- EPC only at payment information level -->
                 <Amt>
                     <InstdAmt py:attrs="{'Ccy': payment.currency.code}">${payment.amount}</InstdAmt>
-                    <!-- EqvtAmt -->
                 </Amt>
-                <!-- XchgRateInf -->
                 <!-- ChrgBr --> <!-- EPC only at payment information level -->
-                <!-- ChqInstr -->
                 <!-- UltmtDbtr -->
-                <!-- IntrmyAgt1 -->
-                <!-- IntrmyAgt1Acct -->
-                <!-- IntrmyAgt2 -->
-                <!-- IntrmyAgt2Acct -->
-                <!-- IntrmyAgt3 -->
-                <!-- IntrmyAgt3Acct -->
                 <CdtrAgt>
                     ${FinancialInstitution(payment.sepa_bank_account_number.account.bank, only_bic=True)}
                 </CdtrAgt>
-                <!-- CdtrAgtAcct -->
                 <Cdtr>
                     ${PartyIdentification(payment.party)}
                 </Cdtr>
                 <CdtrAcct>
-                    ${Account(payment.sepa_bank_account_number)}
+                    ${Account(payment.sepa_bank_account_number, with_ccy=False)}
                 </CdtrAcct>
                 <!-- UltmtCdtr -->
-                <!-- InstrForCdtrAgt -->
-                <!-- InstrForDbtrAgt -->
                 <!-- Purp -->
-                <!-- RgltryRptg -->
-                <!-- Tax -->
-                <!-- RltdRmtInf -->
                 <RmtInf py:if="payment.sepa_remittance_information">
-                    <Ustrd>${payment.sepa_remittance_information[:140]}</Ustrd>
+                    <Ustrd>${normalize('NFKD', payment.sepa_remittance_information).encode('ascii', 'replace')[:140]}</Ustrd>
                 </RmtInf>
-                <!-- SplmtryData -->
             </CdtTrfTxInf>
         </PmtInf>
-        <!-- SplmtryData -->
     </CstmrCdtTrfInitn>
 </Document>
diff --git a/template/pain.008.001.02.xml b/template/pain.008.001.02.xml
index 002db1a..86a28ef 100644
--- a/template/pain.008.001.02.xml
+++ b/template/pain.008.001.02.xml
@@ -31,7 +31,7 @@ this repository contains the full copyright notices and license terms. -->
             <MsgId>${group.reference[:35]}</MsgId>
             <CreDtTm>${datetime.datetime.now().isoformat()}</CreDtTm>
             <!-- Authstn -->
-            <NbOfTxs>${len(group.payments)}</NbOfTxs>
+            <NbOfTxs>${sum(len(payments) for _, payments in group.sepa_payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in group.payments)}</CtrlSum>
             <!-- PmtTpInf -->
             <!-- ReqdColltnDt -->
@@ -109,7 +109,7 @@ this repository contains the full copyright notices and license terms. -->
                     ${PartyIdentification(payment.party)}
                 </Dbtr>
                 <DbtrAcct>
-                    ${Account(payment.sepa_bank_account_number)}
+                    ${Account(payment.sepa_bank_account_number, currency=False)}
                 </DbtrAcct>
                 <!-- UltmtDbtr -->
                 <!-- InstrForCdtrAgt -->
diff --git a/template/pain.008.001.04.xml b/template/pain.008.001.04.xml
index 2ec8914..57c680e 100644
--- a/template/pain.008.001.04.xml
+++ b/template/pain.008.001.04.xml
@@ -90,7 +90,7 @@ this repository contains the full copyright notices and license terms. -->
                     ${PartyIdentification(payment.party)}
                 </Dbtr>
                 <DbtrAcct>
-                    ${Account(payment.sepa_bank_account_number)}
+                    ${Account(payment.sepa_bank_account_number, currency=False)}
                 </DbtrAcct>
                 <!-- UltmtDbtr -->
                 <!-- InstrForCdtrAgt -->
diff --git a/template/pain.008.001.04.xml b/template/pain.008.003.02.xml
similarity index 74%
copy from template/pain.008.001.04.xml
copy to template/pain.008.003.02.xml
index 2ec8914..7fdb45d 100644
--- a/template/pain.008.001.04.xml
+++ b/template/pain.008.003.02.xml
@@ -1,25 +1,24 @@
 <?xml version="1.0" encoding="UTF-8"?>
 <!-- This file is part of Tryton.  The COPYRIGHT file at the top level of
 this repository contains the full copyright notices and license terms. -->
-<Document xmlns="urn:iso:std:iso:20022:tech:xsd:pain.008.001.04"
+<Document xmlns="urn:iso:std:iso:20022:tech:xsd:pain.008.003.02"
     xmlns:xsd="http://www.w3.org/2001/XMLSchema"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+    xsi:schemaLocation="urn:iso:std:iso:20022:tech:xsd:pain.008.003.02
+    pain.008.003.02.xsd"
     xmlns:xi="http://www.w3.org/2001/XInclude"
     xmlns:py="http://genshi.edgewall.org/">
     <xi:include href="base.xml"/>
+    <xi:include href="base.003.xml"/>
     <CstmrDrctDbtInitn>
         <GrpHdr>
             <MsgId>${group.reference[:35]}</MsgId>
             <CreDtTm>${datetime.datetime.now().isoformat()}</CreDtTm>
-            <!-- Authstn -->
-            <NbOfTxs>${len(group.payments)}</NbOfTxs>
+            <NbOfTxs>${sum(len(payments) for _, payments in group.sepa_payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in group.payments)}</CtrlSum>
-            <!-- PmtTpInf -->
-            <!-- ReqdColltnDt -->
             <InitgPty>
-                ${PartyIdentification(group.sepa_initiating_party)}
+                ${PartyIdentification(group.sepa_initiating_party, with_address=False)}
             </InitgPty>
-            <!-- FwdgAgt -->
         </GrpHdr>
         <PmtInf py:for="key, payments in group.sepa_payments">
             <PmtInfId>${group.sepa_group_payment_id(key)[:35]}</PmtInfId>
@@ -28,14 +27,11 @@ this repository contains the full copyright notices and license terms. -->
             <NbOfTxs>${len(payments)}</NbOfTxs>
             <CtrlSum>${sum(p.amount for p in payments)}</CtrlSum>
             <PmtTpInf>
-                <!-- InstrPrty -->
                 <SvcLvl>
                     <Cd>SEPA</Cd>
-                    <!-- Prtry -->
                 </SvcLvl>
                 <LclInstrm>
                     <Cd>${key['scheme']}</Cd>
-                    <!-- Prtry -->
                 </LclInstrm>
                 <SeqTp>${key['sequence_type']}</SeqTp>
                 <!-- CtgyPurp -->
@@ -50,20 +46,16 @@ this repository contains the full copyright notices and license terms. -->
             <CdtrAgt>
                 ${FinancialInstitution(group.journal.sepa_bank_account_number.account.bank, only_bic=True)}
             </CdtrAgt>
-            <!-- CdtrAgtAcct -->
             <!-- UltmtCdtr -->
-            <ChrgBr>${group.journal.sepa_charge_bearer}</ChrgBr>
-            <!-- ChrgsAcct -->
-            <!-- ChrgsAcctAgt -->
+            <ChrgBr>SLEV</ChrgBr>
             <CdtrSchmeId>
-                ${PartyIdentification(group.company.party)}
+                ${PartyIdentification(group.company.party, with_name=False, with_address=False)}
             </CdtrSchmeId>
             <DrctDbtTxInf py:for="payment in payments">
                 <PmtId>
                     <InstrId>${payment.sepa_instruction_id}</InstrId>
                     <EndToEndId>${payment.sepa_end_to_end_id}</EndToEndId>
                 </PmtId>
-                <!-- PmtTpInf -->
                 <InstdAmt py:attrs="{'Ccy': payment.currency.code}">${payment.amount}</InstdAmt>
                 <!-- ChrgBr --> <!-- EPC only at payment information level -->
                 <DrctDbtTx>
@@ -73,37 +65,26 @@ this repository contains the full copyright notices and license terms. -->
                         <!-- AmdmntInd -->
                         <!-- AmdmntInfDtls -->
                         <!-- ElctrncSgntr -->
-                        <!-- FrstColltnDt -->
-                        <!-- FnlColltnDt -->
-                        <!-- Frqcy -->
                     </MndtRltdInf>
                     <!-- CdtrSchmeId -->
-                    <!-- PreNtfctnId -->
-                    <!-- PreNtfctnDt -->
                 </DrctDbtTx>
                 <!-- UltmtCdtr -->
                 <DbtrAgt>
                     ${FinancialInstitution(payment.sepa_bank_account_number.account.bank, only_bic=True)}
                 </DbtrAgt>
-                <!-- DbtrAgtAcct -->
                 <Dbtr>
                     ${PartyIdentification(payment.party)}
                 </Dbtr>
                 <DbtrAcct>
-                    ${Account(payment.sepa_bank_account_number)}
+                    ${Account(payment.sepa_bank_account_number, with_ccy=False)}
                 </DbtrAcct>
                 <!-- UltmtDbtr -->
-                <!-- InstrForCdtrAgt -->
                 <!-- Purp -->
-                <!-- RgltryRptg -->
-                <!-- Tax -->
-                <!-- RltdRmtInf -->
                 <RmtInf py:if="payment.sepa_remittance_information">
-                    <Ustrd>${payment.sepa_remittance_information[:140]}</Ustrd>
+                    <Ustrd>${normalize('NFKD', payment.sepa_remittance_information).encode('ascii', 'replace')[:140]}</Ustrd>
+                    <!-- Strd -->
                 </RmtInf>
-                <!-- SplmtryData -->
             </DrctDbtTxInf>
         </PmtInf>
-        <!-- SplmtryData -->
     </CstmrDrctDbtInitn>
 </Document>
diff --git a/tests/__init__.py b/tests/__init__.py
index 84e2248..9e31e2c 100644
--- a/tests/__init__.py
+++ b/tests/__init__.py
@@ -1,5 +1,5 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 
 from .test_account_payment_sepa import suite
 
diff --git a/tests/pain.001.003.03.xsd b/tests/pain.001.003.03.xsd
new file mode 100644
index 0000000..a90ed99
--- /dev/null
+++ b/tests/pain.001.003.03.xsd
@@ -0,0 +1,474 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Version gemäß DFÜ-Abkommen Anlage 3, Version 2.7, gültig ab November 2013 mit Umsetzung von IBAN Only gemäß EPC SCT 7.0, zudem Erweiterung Service Level auf Externe Codeliste-->
+<!-- Mit XMLSpy v2008 am 29.11.2012 von der SIZ GmbH bearbeitet -->
+<xs:schema xmlns="urn:iso:std:iso:20022:tech:xsd:pain.001.003.03" xmlns:xs="http://www.w3.org/2001/XMLSchema" targetNamespace="urn:iso:std:iso:20022:tech:xsd:pain.001.003.03" elementFormDefault="qualified">
+	<xs:element name="Document" type="Document"/>
+	<xs:complexType name="AccountIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="IBAN" type="IBAN2007Identifier"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyAndAmount_SimpleTypeSEPA">
+		<xs:restriction base="xs:decimal">
+			<xs:minInclusive value="0.01"/>
+			<xs:maxInclusive value="999999999.99"/>
+			<xs:fractionDigits value="2"/>
+			<xs:totalDigits value="11"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyCode">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{3,3}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="ActiveOrHistoricCurrencyAndAmountSEPA">
+		<xs:simpleContent>
+			<xs:extension base="ActiveOrHistoricCurrencyAndAmount_SimpleTypeSEPA">
+				<xs:attribute name="Ccy" type="ActiveOrHistoricCurrencyCodeEUR" use="required"/>
+			</xs:extension>
+		</xs:simpleContent>
+	</xs:complexType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyCodeEUR">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="EUR"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="AmountTypeSEPA">
+		<xs:sequence>
+			<xs:element name="InstdAmt" type="ActiveOrHistoricCurrencyAndAmountSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="AnyBICIdentifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{6,6}[A-Z2-9][A-NP-Z0-9]([A-Z0-9]{3,3}){0,1}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="BICIdentifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{6,6}[A-Z2-9][A-NP-Z0-9]([A-Z0-9]{3,3}){0,1}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="BatchBookingIndicator">
+		<xs:restriction base="xs:boolean"/>
+	</xs:simpleType>
+	<xs:complexType name="BranchAndFinancialInstitutionIdentificationSEPA1">
+		<xs:sequence>
+			<xs:element name="FinInstnId" type="FinancialInstitutionIdentificationSEPA1"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="BranchAndFinancialInstitutionIdentificationSEPA3">
+		<xs:sequence>
+			<xs:element name="FinInstnId" type="FinancialInstitutionIdentificationSEPA3"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CashAccountSEPA1">
+		<xs:sequence>
+			<xs:element name="Id" type="AccountIdentificationSEPA"/>
+			<xs:element name="Ccy" type="ActiveOrHistoricCurrencyCode" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CashAccountSEPA2">
+		<xs:sequence>
+			<xs:element name="Id" type="AccountIdentificationSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CategoryPurposeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalCategoryPurpose1Code"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="ChargeBearerTypeSEPACode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SLEV"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="CountryCode">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{2,2}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="CreditTransferTransactionInformationSCT">
+		<xs:sequence>
+			<xs:element name="PmtId" type="PaymentIdentificationSEPA"/>
+			<xs:element name="PmtTpInf" type="PaymentTypeInformationSCT2" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If used, it is recommended to be used at ‘Payment Information’ level and not at ‘Credit Transfer Transaction Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="Amt" type="AmountTypeSEPA"/>
+			<xs:element name="ChrgBr" type="ChargeBearerTypeSEPACode" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that this element be specified at ‘Payment Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="UltmtDbtr" type="PartyIdentificationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>This data element may be present either at ‘Payment Information’ or at ‘Credit Transfer Transaction Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="CdtrAgt" type="BranchAndFinancialInstitutionIdentificationSEPA1" minOccurs="0"/>
+			<xs:element name="Cdtr" type="PartyIdentificationSEPA2"/>
+			<xs:element name="CdtrAcct" type="CashAccountSEPA2"/>
+			<xs:element name="UltmtCdtr" type="PartyIdentificationSEPA1" minOccurs="0"/>
+			<xs:element name="Purp" type="PurposeSEPA" minOccurs="0"/>
+			<xs:element name="RmtInf" type="RemittanceInformationSEPA1Choice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CreditorReferenceInformationSEPA1">
+		<xs:sequence>
+			<xs:element name="Tp" type="CreditorReferenceTypeSEPA"/>
+			<xs:element name="Ref" type="Max35Text">
+				<xs:annotation>
+					<xs:documentation>If a Creditor Reference contains a check digit, the receiving bank is not required to validate this.
+If the receiving bank validates the check digit and if this validation fails, the bank may continue its processing and send the transaction to the next party in the chain.
+RF Creditor Reference may be used (ISO 11649).</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CreditorReferenceTypeSEPA">
+		<xs:sequence>
+			<xs:element name="CdOrPrtry" type="CreditorReferenceTypeCodeSEPA"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CreditorReferenceTypeCodeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="DocumentType3CodeSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CustomerCreditTransferInitiationV03">
+		<xs:sequence>
+			<xs:element name="GrpHdr" type="GroupHeaderSCT"/>
+			<xs:element name="PmtInf" type="PaymentInstructionInformationSCT" maxOccurs="unbounded"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="DateAndPlaceOfBirth">
+		<xs:sequence>
+			<xs:element name="BirthDt" type="ISODate"/>
+			<xs:element name="PrvcOfBirth" type="Max35Text" minOccurs="0"/>
+			<xs:element name="CityOfBirth" type="Max35Text"/>
+			<xs:element name="CtryOfBirth" type="CountryCode"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="DecimalNumber">
+		<xs:restriction base="xs:decimal">
+			<xs:fractionDigits value="17"/>
+			<xs:totalDigits value="18"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="Document">
+		<xs:sequence>
+			<xs:element name="CstmrCdtTrfInitn" type="CustomerCreditTransferInitiationV03"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="DocumentType3CodeSEPA">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SCOR"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalCategoryPurpose1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalOrganisationIdentification1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalPersonIdentification1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalPurpose1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalServiceLevel1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="FinancialInstitutionIdentificationSEPA1">
+		<xs:sequence>
+			<xs:element name="BIC" type="BICIdentifier"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="FinancialInstitutionIdentificationSEPA3">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="BIC" type="BICIdentifier"/>
+				<xs:element name="Othr" type="OthrIdentification"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="OthrIdentification">
+		<xs:sequence>
+			<xs:element name="Id" type="OthrIdentificationCode"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="OthrIdentificationCode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="NOTPROVIDED"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="GenericOrganisationIdentification1">
+		<xs:sequence>
+			<xs:element name="Id" type="Max35Text"/>
+			<xs:element name="SchmeNm" type="OrganisationIdentificationSchemeName1Choice" minOccurs="0"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="GenericPersonIdentification1">
+		<xs:sequence>
+			<xs:element name="Id" type="Max35Text"/>
+			<xs:element name="SchmeNm" type="PersonIdentificationSchemeName1Choice" minOccurs="0"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="GroupHeaderSCT">
+		<xs:sequence>
+			<xs:element name="MsgId" type="RestrictedIdentificationSEPA1"/>
+			<xs:element name="CreDtTm" type="ISODateTime"/>
+			<xs:element name="NbOfTxs" type="Max15NumericText"/>
+			<xs:element name="CtrlSum" type="DecimalNumber" minOccurs="0"/>
+			<xs:element name="InitgPty" type="PartyIdentificationSEPA1"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="IBAN2007Identifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{2,2}[0-9]{2,2}[a-zA-Z0-9]{1,30}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ISODate">
+		<xs:restriction base="xs:date"/>
+	</xs:simpleType>
+	<xs:simpleType name="ISODateTime">
+		<xs:restriction base="xs:dateTime"/>
+	</xs:simpleType>
+	<xs:simpleType name="Max140Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="140"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max15NumericText">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[0-9]{1,15}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max35Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="35"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max70Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="70"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="OrganisationIdentificationSEPAChoice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="BICOrBEI" type="AnyBICIdentifier"/>
+				<xs:element name="Othr" type="GenericOrganisationIdentification1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="OrganisationIdentificationSchemeName1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Cd" type="ExternalOrganisationIdentification1Code"/>
+				<xs:element name="Prtry" type="Max35Text"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartySEPAChoice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="OrgId" type="OrganisationIdentificationSEPAChoice">
+					<xs:annotation>
+						<xs:documentation>Either ‘BIC or BEI’ or one
+occurrence of ‘Other’ is allowed.</xs:documentation>
+					</xs:annotation>
+				</xs:element>
+				<xs:element name="PrvtId" type="PersonIdentificationSEPA1Choice">
+					<xs:annotation>
+						<xs:documentation>Either ‘Date and Place of Birth’ or one occurrence of ‘Other’ is allowed.</xs:documentation>
+					</xs:annotation>
+				</xs:element>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA1">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>‘Name’ is limited to 70 characters
+in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="Id" type="PartySEPAChoice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA2">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text">
+				<xs:annotation>
+					<xs:documentation>‘Name’ is limited to 70 characters
+in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="PstlAdr" type="PostalAddressSEPA" minOccurs="0"/>
+			<xs:element name="Id" type="PartySEPAChoice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PaymentIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="InstrId" type="RestrictedIdentificationSEPA1" minOccurs="0"/>
+			<xs:element name="EndToEndId" type="RestrictedIdentificationSEPA1"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PaymentInstructionInformationSCT">
+		<xs:sequence>
+			<xs:element name="PmtInfId" type="RestrictedIdentificationSEPA1"/>
+			<xs:element name="PmtMtd" type="PaymentMethodSCTCode">
+				<xs:annotation>
+					<xs:documentation>Only ‘TRF’ is allowed.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="BtchBookg" type="BatchBookingIndicator" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If present and contains ‘true’, batch booking is requested. If present and contains ‘false’, booking per transaction is requested. If element is not present, pre-agreed customer-to-bank conditions apply.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="NbOfTxs" type="Max15NumericText" minOccurs="0"/>
+			<xs:element name="CtrlSum" type="DecimalNumber" minOccurs="0"/>
+			<xs:element name="PmtTpInf" type="PaymentTypeInformationSCT1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If used, it is recommended to be used only at ‘Payment Information’ level and not at Credit Transfer Transaction Information’ level.
+When Instruction Priority is to be used, ‘Payment Type Information’ must be present at ‘Payment Information’ level. </xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="ReqdExctnDt" type="ISODate"/>
+			<xs:element name="Dbtr" type="PartyIdentificationSEPA2"/>
+			<xs:element name="DbtrAcct" type="CashAccountSEPA1"/>
+			<xs:element name="DbtrAgt" type="BranchAndFinancialInstitutionIdentificationSEPA3"/>
+			<xs:element name="UltmtDbtr" type="PartyIdentificationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>This data element may be present either at ‘Payment Information’ or at ‘Credit Transfer Transaction Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="ChrgBr" type="ChargeBearerTypeSEPACode" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that this element be specified at ‘Payment Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="CdtTrfTxInf" type="CreditTransferTransactionInformationSCT" maxOccurs="unbounded"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="PaymentMethodSCTCode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="TRF"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="PaymentTypeInformationSCT1">
+		<xs:sequence>
+			<xs:element name="InstrPrty" type="Priority2Code" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If present, pre-agreed customer-to-bank conditions apply.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="SvcLvl" type="ServiceLevelSEPA"/>
+			<xs:element name="CtgyPurp" type="CategoryPurposeSEPA" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Depending on the agreement between the Originator and the Originator Bank, ‘Category Purpose’ may be forwarded to the Beneficiary Bank.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PaymentTypeInformationSCT2">
+		<xs:sequence>
+			<xs:element name="SvcLvl" type="ServiceLevelSEPA"/>
+			<xs:element name="CtgyPurp" type="CategoryPurposeSEPA" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Depending on the agreement between the Originator and the Originator Bank, ‘Category Purpose’ may be forwarded to the Beneficiary Bank.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PersonIdentificationSEPA1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="DtAndPlcOfBirth" type="DateAndPlaceOfBirth"/>
+				<xs:element name="Othr" type="GenericPersonIdentification1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PersonIdentificationSchemeName1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Cd" type="ExternalPersonIdentification1Code"/>
+				<xs:element name="Prtry" type="Max35Text"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PostalAddressSEPA">
+		<xs:sequence>
+			<xs:element name="Ctry" type="CountryCode" minOccurs="0"/>
+			<xs:element name="AdrLine" type="Max70Text" minOccurs="0" maxOccurs="2"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="Priority2Code">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="HIGH"/>
+			<xs:enumeration value="NORM"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="PurposeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalPurpose1Code">
+				<xs:annotation>
+					<xs:documentation>Only codes from the ISO 20022 ExternalPurposeCode list are allowed.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="RemittanceInformationSEPA1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Ustrd" type="Max140Text"/>
+				<xs:element name="Strd" type="StructuredRemittanceInformationSEPA1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="ServiceLevelSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalServiceLevel1Code"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="StructuredRemittanceInformationSEPA1">
+		<xs:sequence>
+			<xs:element name="CdtrRefInf" type="CreditorReferenceInformationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>When present, the receiving bank is not obliged to validate the the reference information. </xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="RestrictedIdentificationSEPA1">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="([A-Za-z0-9]|[\+|\?|/|\-|:|\(|\)|\.|,|'| ]){1,35}"/>
+		</xs:restriction>
+	</xs:simpleType>
+</xs:schema>
diff --git a/tests/pain.008.003.02.xsd b/tests/pain.008.003.02.xsd
new file mode 100644
index 0000000..99992b5
--- /dev/null
+++ b/tests/pain.008.003.02.xsd
@@ -0,0 +1,614 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Mit XMLSpy v2008 rel. 2 (http://www.altova.com) von Wenzel (SIZ Bonn) bearbeitet -->
+<!-- Version gemäß DFÜ-Abkommen Anlage 3, Version 2.7, gültig ab November 2013 mit Umsetzung von IBAN Only gemäß EPC SDD Core IG 7.0 bzw. SDD B2B IG 5.0 , zudem Einbau der COR1-Option durch Erweiterung Local Instrument und Erweiterung Service Level auf Externe Codeliste-->
+<!-- Mit XMLSpy v2008 rel. 2 sp2 (http://www.altova.com) am 29.11.2012 von der SIZ GmbH bearbeitet -->
+<xs:schema xmlns="urn:iso:std:iso:20022:tech:xsd:pain.008.003.02" xmlns:xs="http://www.w3.org/2001/XMLSchema" targetNamespace="urn:iso:std:iso:20022:tech:xsd:pain.008.003.02" elementFormDefault="qualified">
+	<xs:element name="Document" type="Document"/>
+	<xs:complexType name="AccountIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="IBAN" type="IBAN2007Identifier"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyAndAmount_SimpleTypeSEPA">
+		<xs:restriction base="xs:decimal">
+			<xs:minInclusive value="0.01"/>
+			<xs:maxInclusive value="999999999.99"/>
+			<xs:fractionDigits value="2"/>
+			<xs:totalDigits value="11"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="ActiveOrHistoricCurrencyAndAmountSEPA">
+		<xs:simpleContent>
+			<xs:extension base="ActiveOrHistoricCurrencyAndAmount_SimpleTypeSEPA">
+				<xs:attribute name="Ccy" type="ActiveOrHistoricCurrencyCodeEUR" use="required"/>
+			</xs:extension>
+		</xs:simpleContent>
+	</xs:complexType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyCodeEUR">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="EUR"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ActiveOrHistoricCurrencyCode">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{3,3}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="AmendmentInformationDetailsSDD">
+		<xs:sequence>
+			<xs:element name="OrgnlMndtId" type="RestrictedIdentificationSEPA2" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Mandatory if changes occur in ‘Mandate Identification’, otherwise not to be used.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="OrgnlCdtrSchmeId" type="PartyIdentificationSEPA4" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Mandatory if changes occur in 'Creditor Scheme Identification', otherwise not to be used.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="OrgnlDbtrAcct" type="CashAccountSEPA2" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>To be used only for changes of accounts within the same bank.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="OrgnlDbtrAgt" type="BranchAndFinancialInstitutionIdentificationSEPA2" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>To use 'Identification’ under 'Other' under 'Financial Institution Identifier with code ‘SMNDA’ to indicate same mandate with new Debtor Agent. To be used with the ‘FRST’ indicator in the ‘Sequence Type’.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="AnyBICIdentifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{6,6}[A-Z2-9][A-NP-Z0-9]([A-Z0-9]{3,3}){0,1}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="BICIdentifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{6,6}[A-Z2-9][A-NP-Z0-9]([A-Z0-9]{3,3}){0,1}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="BatchBookingIndicator">
+		<xs:restriction base="xs:boolean"/>
+	</xs:simpleType>
+	<xs:complexType name="BranchAndFinancialInstitutionIdentificationSEPA3">
+		<xs:sequence>
+			<xs:element name="FinInstnId" type="FinancialInstitutionIdentificationSEPA3"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="BranchAndFinancialInstitutionIdentificationSEPA2">
+		<xs:sequence>
+			<xs:element name="FinInstnId" type="FinancialInstitutionIdentificationSEPA2"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CashAccountSEPA1">
+		<xs:sequence>
+			<xs:element name="Id" type="AccountIdentificationSEPA"/>
+			<xs:element name="Ccy" type="ActiveOrHistoricCurrencyCode" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CashAccountSEPA2">
+		<xs:sequence>
+			<xs:element name="Id" type="AccountIdentificationSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CategoryPurposeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalCategoryPurpose1Code"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="ChargeBearerTypeSEPACode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SLEV"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="CountryCode">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{2,2}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="CreditorReferenceInformationSEPA1">
+		<xs:sequence>
+			<xs:element name="Tp" type="CreditorReferenceTypeSEPA"/>
+			<xs:element name="Ref" type="Max35Text">
+				<xs:annotation>
+					<xs:documentation>If a Creditor Reference contains a check digit, the receiving bank is not required to validate this.
+If the receiving bank validates the check digit and if this validation fails, the bank may continue its processing and send the transaction to the next party in the chain.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CreditorReferenceTypeSEPA">
+		<xs:sequence>
+			<xs:element name="CdOrPrtry" type="CreditorReferenceTypeCodeSEPA"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CreditorReferenceTypeCodeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="DocumentType3CodeSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="CustomerDirectDebitInitiationV02">
+		<xs:sequence>
+			<xs:element name="GrpHdr" type="GroupHeaderSDD"/>
+			<xs:element name="PmtInf" type="PaymentInstructionInformationSDD" maxOccurs="unbounded"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="DateAndPlaceOfBirth">
+		<xs:sequence>
+			<xs:element name="BirthDt" type="ISODate"/>
+			<xs:element name="PrvcOfBirth" type="Max35Text" minOccurs="0"/>
+			<xs:element name="CityOfBirth" type="Max35Text"/>
+			<xs:element name="CtryOfBirth" type="CountryCode"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="DecimalNumber">
+		<xs:restriction base="xs:decimal">
+			<xs:fractionDigits value="17"/>
+			<xs:totalDigits value="18"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="DirectDebitTransactionSDD">
+		<xs:sequence>
+			<xs:element name="MndtRltdInf" type="MandateRelatedInformationSDD"/>
+			<xs:element name="CdtrSchmeId" type="PartyIdentificationSEPA3" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that all transactions within the same ‘Payment Information’ block have the same ‘Creditor Scheme Identification’.
+This data element must be present at either ‘Payment Information’ or ‘Direct Debit
+Transaction’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="DirectDebitTransactionInformationSDD">
+		<xs:sequence>
+			<xs:element name="PmtId" type="PaymentIdentificationSEPA"/>
+			<xs:element name="InstdAmt" type="ActiveOrHistoricCurrencyAndAmountSEPA"/>
+			<xs:element name="ChrgBr" type="ChargeBearerTypeSEPACode" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that this element be specified at ‘Payment Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="DrctDbtTx" type="DirectDebitTransactionSDD"/>
+			<xs:element name="UltmtCdtr" type="PartyIdentificationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>This data element may be present either at ‘Payment Information’ or at ‘Direct Debit Transaction Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="DbtrAgt" type="BranchAndFinancialInstitutionIdentificationSEPA3"/>
+			<xs:element name="Dbtr" type="PartyIdentificationSEPA2"/>
+			<xs:element name="DbtrAcct" type="CashAccountSEPA2"/>
+			<xs:element name="UltmtDbtr" type="PartyIdentificationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Mandatory if provided by the debtor in the mandate.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="Purp" type="PurposeSEPA" minOccurs="0"/>
+			<xs:element name="RmtInf" type="RemittanceInformationSEPA1Choice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="Document">
+		<xs:sequence>
+			<xs:element name="CstmrDrctDbtInitn" type="CustomerDirectDebitInitiationV02"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="DocumentType3CodeSEPA">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SCOR"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalCategoryPurpose1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalLocalInstrument1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="35"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalOrganisationIdentification1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalPersonIdentification1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalPurpose1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ExternalServiceLevel1Code">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="4"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="FinancialInstitutionIdentificationSEPA3">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="BIC" type="BICIdentifier"/>
+				<xs:element name="Othr" type="OthrIdentification"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="OthrIdentification">
+		<xs:sequence>
+			<xs:element name="Id" type="OthrIdentificationCode"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="OthrIdentificationCode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="NOTPROVIDED"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="FinancialInstitutionIdentificationSEPA2">
+		<xs:sequence>
+			<xs:element name="Othr" type="RestrictedFinancialIdentificationSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="RestrictedFinancialIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="Id" type="RestrictedSMNDACode"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="RestrictedSMNDACode">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SMNDA"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="GenericOrganisationIdentification1">
+		<xs:sequence>
+			<xs:element name="Id" type="Max35Text"/>
+			<xs:element name="SchmeNm" type="OrganisationIdentificationSchemeName1Choice" minOccurs="0"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="GenericPersonIdentification1">
+		<xs:sequence>
+			<xs:element name="Id" type="Max35Text"/>
+			<xs:element name="SchmeNm" type="PersonIdentificationSchemeName1Choice" minOccurs="0"/>
+			<xs:element name="Issr" type="Max35Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="RestrictedPersonIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="Id" type="RestrictedPersonIdentifierSEPA"/>
+			<xs:element name="SchmeNm" type="RestrictedPersonIdentificationSchemeNameSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="RestrictedPersonIdentifierSEPA">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[a-zA-Z]{2,2}[0-9]{2,2}([A-Za-z0-9]|[\+|\?|/|\-|:|\(|\)|\.|,|']){3,3}([A-Za-z0-9]|[\+|\?|/|\-|:|\(|\)|\.|,|']){1,28}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="GroupHeaderSDD">
+		<xs:sequence>
+			<xs:element name="MsgId" type="RestrictedIdentificationSEPA1"/>
+			<xs:element name="CreDtTm" type="ISODateTime"/>
+			<xs:element name="NbOfTxs" type="Max15NumericText"/>
+			<xs:element name="CtrlSum" type="DecimalNumber" minOccurs="0"/>
+			<xs:element name="InitgPty" type="PartyIdentificationSEPA1"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="IBAN2007Identifier">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[A-Z]{2,2}[0-9]{2,2}[a-zA-Z0-9]{1,30}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="ISODate">
+		<xs:restriction base="xs:date"/>
+	</xs:simpleType>
+	<xs:simpleType name="ISODateTime">
+		<xs:restriction base="xs:dateTime"/>
+	</xs:simpleType>
+	<xs:complexType name="LocalInstrumentSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalLocalInstrument1Code"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="MandateRelatedInformationSDD">
+		<xs:sequence>
+			<xs:element name="MndtId" type="RestrictedIdentificationSEPA2"/>
+			<xs:element name="DtOfSgntr" type="ISODate"/>
+			<xs:element name="AmdmntInd" type="TrueFalseIndicator" minOccurs="0"/>
+			<xs:element name="AmdmntInfDtls" type="AmendmentInformationDetailsSDD" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Mandatory if 'Amendment Indicator' is 'TRUE'
+The reason code from the Rulebook is indicated using one of the following message subelements.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="ElctrncSgntr" type="Max1025Text" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="Max1025Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="1025"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max140Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="140"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max15NumericText">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="[0-9]{1,15}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max35Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="35"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="Max70Text">
+		<xs:restriction base="xs:string">
+			<xs:minLength value="1"/>
+			<xs:maxLength value="70"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="OrganisationIdentificationSEPAChoice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="BICOrBEI" type="AnyBICIdentifier"/>
+				<xs:element name="Othr" type="GenericOrganisationIdentification1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="OrganisationIdentificationSchemeName1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Cd" type="ExternalOrganisationIdentification1Code"/>
+				<xs:element name="Prtry" type="Max35Text"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartySEPAChoice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="OrgId" type="OrganisationIdentificationSEPAChoice">
+					<xs:annotation>
+						<xs:documentation>Either ‘BIC or BEI’ or one
+occurrence of ‘Other’ is allowed.</xs:documentation>
+					</xs:annotation>
+				</xs:element>
+				<xs:element name="PrvtId" type="PersonIdentificationSEPA1Choice">
+					<xs:annotation>
+						<xs:documentation>Either ‘Date and Place of Birth’ or one occurrence of ‘Other’ is allowed</xs:documentation>
+					</xs:annotation>
+				</xs:element>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartySEPA2">
+		<xs:sequence>
+			<xs:element name="PrvtId" type="PersonIdentificationSEPA2">
+				<xs:annotation>
+					<xs:documentation>Private Identification is used to identify either an organisation or a private
+person.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA1">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>‘Name’ is limited to 70 characters in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="Id" type="PartySEPAChoice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA2">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text">
+				<xs:annotation>
+					<xs:documentation>‘Name’ is limited to 70 characters in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="PstlAdr" type="PostalAddressSEPA" minOccurs="0"/>
+			<xs:element name="Id" type="PartySEPAChoice" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA3">
+		<xs:sequence>
+			<xs:element name="Id" type="PartySEPA2"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA4">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If present the new’ Name’ must be specified under ‘Creditor’. ‘Name’ is limited to 70 characters in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="Id" type="PartySEPA2" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PartyIdentificationSEPA5">
+		<xs:sequence>
+			<xs:element name="Nm" type="Max70Text">
+				<xs:annotation>
+					<xs:documentation>‘Name’ is limited to 70 characters in length.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="PstlAdr" type="PostalAddressSEPA" minOccurs="0"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PaymentIdentificationSEPA">
+		<xs:sequence>
+			<xs:element name="InstrId" type="RestrictedIdentificationSEPA1" minOccurs="0"/>
+			<xs:element name="EndToEndId" type="RestrictedIdentificationSEPA1"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PaymentInstructionInformationSDD">
+		<xs:sequence>
+			<xs:element name="PmtInfId" type="RestrictedIdentificationSEPA1"/>
+			<xs:element name="PmtMtd" type="PaymentMethod2Code"/>
+			<xs:element name="BtchBookg" type="BatchBookingIndicator" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>If present and contains ‘true’, batch booking is requested. If present and contains ‘false’, booking per transaction is requested. If element is not present, pre-agreed customer-to-bank conditions apply.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="NbOfTxs" type="Max15NumericText" minOccurs="0"/>
+			<xs:element name="CtrlSum" type="DecimalNumber" minOccurs="0"/>
+			<xs:element name="PmtTpInf" type="PaymentTypeInformationSDD"/>
+			<xs:element name="ReqdColltnDt" type="ISODate"/>
+			<xs:element name="Cdtr" type="PartyIdentificationSEPA5"/>
+			<xs:element name="CdtrAcct" type="CashAccountSEPA1"/>
+			<xs:element name="CdtrAgt" type="BranchAndFinancialInstitutionIdentificationSEPA3"/>
+			<xs:element name="UltmtCdtr" type="PartyIdentificationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>This data element may be present either at ‘Payment Information’ or at ‘Direct Debit Transaction Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="ChrgBr" type="ChargeBearerTypeSEPACode" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that this element be specified at ‘Payment Information’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="CdtrSchmeId" type="PartyIdentificationSEPA3" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>It is recommended that all transactions within the same ‘Payment Information’ block have the same ‘Creditor Scheme Identification’.
+This data element must be present at either ‘Payment Information’ or ‘Direct Debit
+Transaction’ level.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="DrctDbtTxInf" type="DirectDebitTransactionInformationSDD" maxOccurs="unbounded"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="PaymentMethod2Code">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="DD"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="PaymentTypeInformationSDD">
+		<xs:sequence>
+			<xs:element name="SvcLvl" type="ServiceLevelSEPA"/>
+			<xs:element name="LclInstrm" type="LocalInstrumentSEPA">
+				<xs:annotation>
+					<xs:documentation>Only ‘B2B’, 'CORE' or 'COR1' is allowed. The mixing of different Local Instrument values is not allowed in the same message.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="SeqTp" type="SequenceType1Code">
+				<xs:annotation>
+					<xs:documentation>If 'Amendment Indicator' is 'true' and 'Original Debtor Agent' is set to 'SMNDA' this message element must indicate 'FRST'</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+			<xs:element name="CtgyPurp" type="CategoryPurposeSEPA" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>Depending on the agreement between the Creditor and the Creditor Bank, ‘Category Purpose’ may be forwarded to the Debtor Bank.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PersonIdentificationSEPA1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="DtAndPlcOfBirth" type="DateAndPlaceOfBirth"/>
+				<xs:element name="Othr" type="GenericPersonIdentification1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PersonIdentificationSEPA2">
+		<xs:sequence>
+			<xs:element name="Othr" type="RestrictedPersonIdentificationSEPA">
+				<xs:annotation>
+					<xs:documentation>Only one occurrence of ‘Other’ is allowed, and no other sub-elements are allowed.
+Identification must be used with an identifier described in General Message Element Specifications, Chapter 1.5.2 of the Implementation Guide.
+Scheme Name’ under ‘Other’ must specify ‘SEPA’ under ‘Proprietary</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PersonIdentificationSchemeName1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Cd" type="ExternalPersonIdentification1Code"/>
+				<xs:element name="Prtry" type="Max35Text"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="RestrictedPersonIdentificationSchemeNameSEPA">
+		<xs:sequence>
+			<xs:element name="Prtry" type="IdentificationSchemeNameSEPA"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="IdentificationSchemeNameSEPA">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="SEPA"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="PostalAddressSEPA">
+		<xs:sequence>
+			<xs:element name="Ctry" type="CountryCode" minOccurs="0"/>
+			<xs:element name="AdrLine" type="Max70Text" minOccurs="0" maxOccurs="2"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="PurposeSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalPurpose1Code">
+				<xs:annotation>
+					<xs:documentation>Only codes from the ISO 20022 ExternalPurposeCode list are allowed.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="RemittanceInformationSEPA1Choice">
+		<xs:sequence>
+			<xs:choice>
+				<xs:element name="Ustrd" type="Max140Text"/>
+				<xs:element name="Strd" type="StructuredRemittanceInformationSEPA1"/>
+			</xs:choice>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="SequenceType1Code">
+		<xs:restriction base="xs:string">
+			<xs:enumeration value="FRST"/>
+			<xs:enumeration value="RCUR"/>
+			<xs:enumeration value="FNAL"/>
+			<xs:enumeration value="OOFF"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:complexType name="ServiceLevelSEPA">
+		<xs:sequence>
+			<xs:element name="Cd" type="ExternalServiceLevel1Code"/>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:complexType name="StructuredRemittanceInformationSEPA1">
+		<xs:sequence>
+			<xs:element name="CdtrRefInf" type="CreditorReferenceInformationSEPA1" minOccurs="0">
+				<xs:annotation>
+					<xs:documentation>When present, the receiving bank is not obliged to validate the reference information.</xs:documentation>
+				</xs:annotation>
+			</xs:element>
+		</xs:sequence>
+	</xs:complexType>
+	<xs:simpleType name="TrueFalseIndicator">
+		<xs:restriction base="xs:boolean"/>
+	</xs:simpleType>
+	<xs:simpleType name="RestrictedIdentificationSEPA1">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="([A-Za-z0-9]|[\+|\?|/|\-|:|\(|\)|\.|,|'| ]){1,35}"/>
+		</xs:restriction>
+	</xs:simpleType>
+	<xs:simpleType name="RestrictedIdentificationSEPA2">
+		<xs:restriction base="xs:string">
+			<xs:pattern value="([A-Za-z0-9]|[\+|\?|/|\-|:|\(|\)|\.|,|']){1,35}"/>
+		</xs:restriction>
+	</xs:simpleType>
+</xs:schema>
diff --git a/tests/test_account_payment_sepa.py b/tests/test_account_payment_sepa.py
index 9b1658a..f688c65 100644
--- a/tests/test_account_payment_sepa.py
+++ b/tests/test_account_payment_sepa.py
@@ -1,5 +1,5 @@
-#This file is part of Tryton.  The COPYRIGHT file at the top level of
-#this repository contains the full copyright notices and license terms.
+# This file is part of Tryton.  The COPYRIGHT file at the top level of
+# this repository contains the full copyright notices and license terms.
 import os
 import unittest
 import doctest
@@ -11,165 +11,191 @@ from io import BytesIO
 from mock import Mock, patch
 
 import trytond.tests.test_tryton
-from trytond.tests.test_tryton import test_view, test_depends
-from trytond.tests.test_tryton import POOL, DB_NAME, USER, CONTEXT
+from trytond.tests.test_tryton import ModuleTestCase
+from trytond.tests.test_tryton import DB_NAME, USER, CONTEXT
 from trytond.transaction import Transaction
 from trytond.exceptions import UserError
 from trytond.modules.account_payment_sepa.payment import CAMT054
-
-
-class AccountPaymentSepaTestCase(unittest.TestCase):
+from trytond.pool import Pool
+
+
+def setup_environment():
+    pool = Pool()
+    Address = pool.get('party.address')
+    Company = pool.get('company.company')
+    Currency = pool.get('currency.currency')
+    Party = pool.get('party.party')
+    Bank = pool.get('bank')
+
+    company, = Company.search([
+            ('rec_name', '=', 'Dunder Mifflin'),
+            ])
+    euro, = Currency.create([{
+                'name': 'Euro',
+                'symbol': 'EUR',
+                'code': 'EUR',
+                }])
+    company.currency = euro
+    company.party.sepa_creditor_identifier = 'BE68539007547034'
+    company.party.save()
+    company.save()
+    bank_party = Party(name='European Bank')
+    bank_party.save()
+    bank = Bank(party=bank_party, bic='BICODEBBXXX')
+    bank.save()
+    customer = Party(name='Customer')
+    address = Address(street='street', streetbis='street bis',
+        zip='1234', city='City')
+    customer.addresses = [address]
+    customer.save()
+    return {
+        'company': company,
+        'bank': bank,
+        'customer': customer,
+        }
+
+
+def setup_accounts(bank, company, customer):
+    pool = Pool()
+    Account = pool.get('bank.account')
+    return Account.create([{
+                'bank': bank,
+                'owners': [('add', [company.party])],
+                'currency': company.currency.id,
+                'numbers': [('create', [{
+                                'type': 'iban',
+                                'number': 'ES8200000000000000000000',
+                                }])]}, {
+                'bank': bank,
+                'owners': [('add', [customer])],
+                'currency': company.currency.id,
+                'numbers': [('create', [{
+                                'type': 'iban',
+                                'number': 'ES3600000000050000000001',
+                                }])]}])
+
+
+def setup_mandate(company, customer, account):
+    pool = Pool()
+    Mandate = pool.get('account.payment.sepa.mandate')
+    Date = pool.get('ir.date')
+    return Mandate.create([{
+                'company': company,
+                'party': customer,
+                'account_number': account.numbers[0],
+                'identification': 'MANDATE',
+                'type': 'recurrent',
+                'signature_date': Date.today(),
+                'state': 'validated',
+                }])[0]
+
+
+def setup_journal(flavor, kind, company, account):
+    pool = Pool()
+    Journal = pool.get('account.payment.journal')
+    journal = Journal()
+    journal.name = flavor
+    journal.company = company
+    journal.currency = company.currency
+    journal.process_method = 'sepa'
+    journal.sepa_bank_account_number = account.numbers[0]
+    journal.sepa_payable_flavor = 'pain.001.001.03'
+    journal.sepa_receivable_flavor = 'pain.008.001.02'
+    setattr(journal, 'sepa_%s_flavor' % kind, flavor)
+    journal.save()
+    return journal
+
+
+def validate_file(flavor, kind, xsd=None):
+    'Test generated files are valid'
+    pool = Pool()
+    Payment = pool.get('account.payment')
+    PaymentGroup = pool.get('account.payment.group')
+    Date = pool.get('ir.date')
+    ProcessPayment = pool.get('account.payment.process', type='wizard')
+
+    if xsd is None:
+        xsd = flavor
+
+    environment = setup_environment()
+    company = environment['company']
+    bank = environment['bank']
+    customer = environment['customer']
+    company_account, customer_account = setup_accounts(
+        bank, company, customer)
+    setup_mandate(company, customer, customer_account)
+    journal = setup_journal(flavor, kind, company, company_account)
+
+    payment, = Payment.create([{
+                'company': company,
+                'party': customer,
+                'journal': journal,
+                'kind': kind,
+                'amount': Decimal('1000.0'),
+                'state': 'approved',
+                'description': 'PAYMENT',
+                'date': Date.today(),
+                }])
+
+    session_id, _, _ = ProcessPayment.create()
+    process_payment = ProcessPayment(session_id)
+    with Transaction().set_context(active_ids=[payment.id]):
+        _, data = process_payment.do_process(None)
+    group, = PaymentGroup.browse(data['res_id'])
+    message, = group.sepa_messages
+    assert message.type == 'out', message.type
+    assert message.state == 'waiting', message.state
+    sepa_string = message.message.encode('utf-8')
+    sepa_xml = etree.fromstring(sepa_string)
+    schema_file = os.path.join(os.path.dirname(__file__),
+        '%s.xsd' % xsd)
+    schema = etree.XMLSchema(etree.parse(schema_file))
+    schema.assertValid(sepa_xml)
+
+
+class AccountPaymentSepaTestCase(ModuleTestCase):
     'Test Account Payment SEPA module'
-
-    def setUp(self):
-        trytond.tests.test_tryton.install_module('account_payment_sepa')
-        self.bank = POOL.get('bank')
-        self.bank_account = POOL.get('bank.account')
-        self.bank_number = POOL.get('bank.account.number')
-        self.company = POOL.get('company.company')
-        self.currency = POOL.get('currency.currency')
-        self.date = POOL.get('ir.date')
-        self.mandate = POOL.get('account.payment.sepa.mandate')
-        self.party = POOL.get('party.party')
-        self.payment = POOL.get('account.payment')
-        self.payment_group = POOL.get('account.payment.group')
-        self.payment_journal = POOL.get('account.payment.journal')
-        self.process_payment = POOL.get('account.payment.process',
-            type='wizard')
-        self.user = POOL.get('res.user')
-
-    def test0005views(self):
-        'Test views'
-        test_view('account_payment_sepa')
-
-    def test0006depends(self):
-        'Test depends'
-        test_depends()
-
-    def setup_environment(self):
-        company, = self.company.search([
-                ('rec_name', '=', 'Dunder Mifflin'),
-                ])
-        euro, = self.currency.create([{
-                    'name': 'Euro',
-                    'symbol': 'EUR',
-                    'code': 'EUR',
-                    }])
-        company.currency = euro
-        company.party.sepa_creditor_identifier = 'BE68539007547034'
-        company.party.save()
-        company.save()
-        bank_party = self.party(name='European Bank')
-        bank_party.save()
-        bank = self.bank(party=bank_party, bic='BICODEBBXXX')
-        bank.save()
-        customer = self.party(name='Customer')
-        customer.save()
-        return {
-            'company': company,
-            'bank': bank,
-            'customer': customer,
-            }
-
-    def setup_accounts(self, bank, company, customer):
-        return self.bank_account.create([{
-                    'bank': bank,
-                    'owners': [('add', [company.party])],
-                    'numbers': [('create', [{
-                                    'type': 'iban',
-                                    'number': 'ES8200000000000000000000',
-                                    }])]}, {
-                    'bank': bank,
-                    'owners': [('add', [customer])],
-                    'numbers': [('create', [{
-                                    'type': 'iban',
-                                    'number': 'ES3600000000050000000001',
-                                    }])]}])
-
-    def setup_mandate(self, company, customer, account):
-        return self.mandate.create([{
-                    'company': company,
-                    'party': customer,
-                    'account_number': account.numbers[0],
-                    'identification': 'MANDATE',
-                    'type': 'recurrent',
-                    'signature_date': self.date.today(),
-                    'state': 'validated',
-                    }])[0]
-
-    def setup_journal(self, flavor, kind, company, account):
-        journal = self.payment_journal()
-        journal.name = flavor
-        journal.company = company
-        journal.currency = company.currency
-        journal.process_method = 'sepa'
-        journal.sepa_bank_account_number = account.numbers[0]
-        journal.sepa_payable_flavor = 'pain.001.001.03'
-        journal.sepa_receivable_flavor = 'pain.008.001.02'
-        setattr(journal, 'sepa_%s_flavor' % kind, flavor)
-        journal.save()
-        return journal
-
-    def validate_file(self, flavor, kind):
-        'Test generated files are valid'
-        with Transaction().start(DB_NAME, USER, context=CONTEXT):
-            environment = self.setup_environment()
-            company = environment['company']
-            bank = environment['bank']
-            customer = environment['customer']
-            company_account, customer_account = self.setup_accounts(bank,
-                company, customer)
-            self.setup_mandate(company, customer, customer_account)
-            journal = self.setup_journal(flavor, kind, company,
-                company_account)
-
-            payment, = self.payment.create([{
-                        'company': company,
-                        'party': customer,
-                        'journal': journal,
-                        'kind': kind,
-                        'amount': Decimal('1000.0'),
-                        'state': 'approved',
-                        'description': 'PAYMENT',
-                        'date': self.date.today(),
-                        }])
-
-            session_id, _, _ = self.process_payment.create()
-            process_payment = self.process_payment(session_id)
-            with Transaction().set_context(active_ids=[payment.id]):
-                _, data = process_payment.do_process(None)
-            group, = self.payment_group.browse(data['res_id'])
-            message, = group.sepa_messages
-            sepa_string = message.message.encode('utf-8')
-            sepa_xml = etree.fromstring(sepa_string)
-            schema_file = os.path.join(os.path.dirname(__file__),
-                '%s.xsd' % flavor)
-            schema = etree.XMLSchema(etree.parse(schema_file))
-            schema.assertValid(sepa_xml)
+    module = 'account_payment_sepa'
 
     def test_pain001_001_03(self):
         'Test pain001.001.03 xsd validation'
-        self.validate_file('pain.001.001.03', 'payable')
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.001.001.03', 'payable')
 
     def test_pain001_001_05(self):
         'Test pain001.001.05 xsd validation'
-        self.validate_file('pain.001.001.05', 'payable')
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.001.001.05', 'payable')
+
+    def test_pain001_003_03(self):
+        'Test pain001.003.03 xsd validation'
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.001.003.03', 'payable')
 
     def test_pain008_001_02(self):
         'Test pain008.001.02 xsd validation'
-        self.validate_file('pain.008.001.02', 'receivable')
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.008.001.02', 'receivable')
 
     def test_pain008_001_04(self):
         'Test pain008.001.04 xsd validation'
-        self.validate_file('pain.008.001.04', 'receivable')
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.008.001.04', 'receivable')
+
+    def test_pain008_003_02(self):
+        'Test pain008.003.02 xsd validation'
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            validate_file('pain.008.003.02', 'receivable')
 
     def test_sepa_mandate_sequence(self):
         'Test SEPA mandate sequence'
-        Configuration = POOL.get('account.configuration')
-        Sequence = POOL.get('ir.sequence')
-        Party = POOL.get('party.party')
-        Mandate = POOL.get('account.payment.sepa.mandate')
         with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            pool = Pool()
+            Configuration = pool.get('account.configuration')
+            Sequence = pool.get('ir.sequence')
+            Party = pool.get('party.party')
+            Mandate = pool.get('account.payment.sepa.mandate')
+
             party = Party(name='Test')
             party.save()
             mandate = Mandate(party=party)
@@ -189,10 +215,13 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
 
     def test_identification_unique(self):
         'Test unique identification constraint'
-        Party = POOL.get('party.party')
-        Mandate = POOL.get('account.payment.sepa.mandate')
-        same_id = '1'
         with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            pool = Pool()
+            Party = pool.get('party.party')
+            Mandate = pool.get('account.payment.sepa.mandate')
+
+            same_id = '1'
+
             party = Party(name='Test')
             party.save()
             mandate = Mandate(party=party, identification=same_id)
@@ -219,11 +248,12 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
     def test_payment_sepa_bank_account_number(self):
         'Test Payment.sepa_bank_account_number'
         with Transaction().start(DB_NAME, USER, context=CONTEXT):
-            Payment = POOL.get('account.payment')
-            Mandate = POOL.get('account.payment.sepa.mandate')
-            AccountNumber = POOL.get('bank.account.number')
-            Party = POOL.get('party.party')
-            BankAccount = POOL.get('bank.account')
+            pool = Pool()
+            Payment = pool.get('account.payment')
+            Mandate = pool.get('account.payment.sepa.mandate')
+            AccountNumber = pool.get('bank.account.number')
+            Party = pool.get('party.party')
+            BankAccount = pool.get('bank.account')
 
             account_number = AccountNumber()
             mandate = Mandate(account_number=account_number)
@@ -246,17 +276,22 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
     def test_payment_sequence_type(self):
         'Test payment sequence type'
         with Transaction().start(DB_NAME, USER, context=CONTEXT):
-            environment = self.setup_environment()
+            pool = Pool()
+            Date = pool.get('ir.date')
+            Payment = pool.get('account.payment')
+            ProcessPayment = pool.get('account.payment.process', type='wizard')
+
+            environment = setup_environment()
             company = environment['company']
             bank = environment['bank']
             customer = environment['customer']
-            company_account, customer_account = self.setup_accounts(bank,
-                company, customer)
-            self.setup_mandate(company, customer, customer_account)
-            journal = self.setup_journal('pain.008.001.02', 'receivable',
+            company_account, customer_account = setup_accounts(
+                bank, company, customer)
+            setup_mandate(company, customer, customer_account)
+            journal = setup_journal('pain.008.001.02', 'receivable',
                 company, company_account)
 
-            payment, = self.payment.create([{
+            payment, = Payment.create([{
                         'company': company,
                         'party': customer,
                         'journal': journal,
@@ -264,17 +299,17 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
                         'amount': Decimal('1000.0'),
                         'state': 'approved',
                         'description': 'PAYMENT',
-                        'date': self.date.today(),
+                        'date': Date.today(),
                         }])
 
-            session_id, _, _ = self.process_payment.create()
-            process_payment = self.process_payment(session_id)
+            session_id, _, _ = ProcessPayment.create()
+            process_payment = ProcessPayment(session_id)
             with Transaction().set_context(active_ids=[payment.id]):
                 _, data = process_payment.do_process(None)
 
             self.assertEqual(payment.sepa_mandate_sequence_type, 'FRST')
 
-            payments = self.payment.create([{
+            payments = Payment.create([{
                         'company': company,
                         'party': customer,
                         'journal': journal,
@@ -282,7 +317,7 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
                         'amount': Decimal('2000.0'),
                         'state': 'approved',
                         'description': 'PAYMENT',
-                        'date': self.date.today(),
+                        'date': Date.today(),
                         }, {
                         'company': company,
                         'party': customer,
@@ -291,12 +326,12 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
                         'amount': Decimal('3000.0'),
                         'state': 'approved',
                         'description': 'PAYMENT',
-                        'date': self.date.today(),
+                        'date': Date.today(),
                         },
                     ])
 
-            session_id, _, _ = self.process_payment.create()
-            process_payment = self.process_payment(session_id)
+            session_id, _, _ = ProcessPayment.create()
+            process_payment = ProcessPayment(session_id)
             payment_ids = [p.id for p in payments]
             with Transaction().set_context(active_ids=payment_ids):
                 _, data = process_payment.do_process(None)
@@ -307,7 +342,8 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
     def handle_camt054(self, flavor):
         'Handle camt.054'
         with Transaction().start(DB_NAME, USER, context=CONTEXT):
-            Message = POOL.get('account.payment.sepa.message')
+            pool = Pool()
+            Message = pool.get('account.payment.sepa.message')
 
             message_file = os.path.join(os.path.dirname(__file__),
                 '%s.xml' % flavor)
@@ -335,7 +371,7 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
                 is_returned.return_value = True
                 handler = CAMT054(BytesIO(message), Payment)
 
-                payment.save.assert_called_with()
+                Payment.save.assert_called_with([payment])
                 Payment.fail.assert_called_with([payment])
 
     def test_camt054_001_01(self):
@@ -354,6 +390,24 @@ class AccountPaymentSepaTestCase(unittest.TestCase):
         'Test camt.054.001.04 handling'
         self.handle_camt054('camt.054.001.04')
 
+    def test_sepa_mandate_report(self):
+        'Test sepa mandate report'
+        with Transaction().start(DB_NAME, USER, context=CONTEXT):
+            pool = Pool()
+            Report = pool.get('account.payment.sepa.mandate', type='report')
+
+            environment = setup_environment()
+            company = environment['company']
+            bank = environment['bank']
+            customer = environment['customer']
+            company_account, customer_account = setup_accounts(
+                bank, company, customer)
+            mandate = setup_mandate(company, customer, customer_account)
+
+            oext, content, _, _ = Report.execute([mandate.id], {})
+            self.assertEqual(oext, 'odt')
+            self.assertTrue(content)
+
 
 def suite():
     suite = trytond.tests.test_tryton.suite()
diff --git a/tryton.cfg b/tryton.cfg
index ff9390b..d842ac8 100644
--- a/tryton.cfg
+++ b/tryton.cfg
@@ -1,11 +1,13 @@
 [tryton]
-version=3.4.1
+version=3.6.0
 depends:
     account_payment
     company
     ir
     bank
     party
+extra_depends:
+    account_payment_clearing
 xml:
     payment.xml
     party.xml
diff --git a/trytond_account_payment_sepa.egg-info/PKG-INFO b/trytond_account_payment_sepa.egg-info/PKG-INFO
index b2371b2..0d8e740 100644
--- a/trytond_account_payment_sepa.egg-info/PKG-INFO
+++ b/trytond_account_payment_sepa.egg-info/PKG-INFO
@@ -1,12 +1,12 @@
 Metadata-Version: 1.1
 Name: trytond-account-payment-sepa
-Version: 3.4.1
+Version: 3.6.0
 Summary: Tryton module for SEPA payment
 Home-page: http://www.tryton.org/
 Author: Tryton
 Author-email: issue_tracker at tryton.org
 License: GPL-3
-Download-URL: http://downloads.tryton.org/3.4/
+Download-URL: http://downloads.tryton.org/3.6/
 Description: trytond_account_payment_sepa
         ============================
         
@@ -64,5 +64,7 @@ Classifier: Natural Language :: Slovenian
 Classifier: Natural Language :: Spanish
 Classifier: Operating System :: OS Independent
 Classifier: Programming Language :: Python :: 2.7
+Classifier: Programming Language :: Python :: Implementation :: CPython
+Classifier: Programming Language :: Python :: Implementation :: PyPy
 Classifier: Topic :: Office/Business
 Classifier: Topic :: Office/Business :: Financial :: Accounting
diff --git a/trytond_account_payment_sepa.egg-info/SOURCES.txt b/trytond_account_payment_sepa.egg-info/SOURCES.txt
index 317be22..bff80a2 100644
--- a/trytond_account_payment_sepa.egg-info/SOURCES.txt
+++ b/trytond_account_payment_sepa.egg-info/SOURCES.txt
@@ -28,11 +28,14 @@ tryton.cfg
 ./locale/es_ES.po
 ./locale/fr_FR.po
 ./locale/sl_SI.po
+./template/base.003.xml
 ./template/base.xml
 ./template/pain.001.001.03.xml
 ./template/pain.001.001.05.xml
+./template/pain.001.003.03.xml
 ./template/pain.008.001.02.xml
 ./template/pain.008.001.04.xml
+./template/pain.008.003.02.xml
 ./tests/__init__.py
 ./tests/camt.054.001.01.xsd
 ./tests/camt.054.001.02.xsd
@@ -40,8 +43,10 @@ tryton.cfg
 ./tests/camt.054.001.04.xsd
 ./tests/pain.001.001.03.xsd
 ./tests/pain.001.001.05.xsd
+./tests/pain.001.003.03.xsd
 ./tests/pain.008.001.02.xsd
 ./tests/pain.008.001.04.xsd
+./tests/pain.008.003.02.xsd
 ./tests/test_account_payment_sepa.py
 ./view/configuration_form.xml
 ./view/mandate_form.xml
@@ -61,19 +66,24 @@ locale/es_EC.po
 locale/es_ES.po
 locale/fr_FR.po
 locale/sl_SI.po
+template/base.003.xml
 template/base.xml
 template/pain.001.001.03.xml
 template/pain.001.001.05.xml
+template/pain.001.003.03.xml
 template/pain.008.001.02.xml
 template/pain.008.001.04.xml
+template/pain.008.003.02.xml
 tests/camt.054.001.01.xsd
 tests/camt.054.001.02.xsd
 tests/camt.054.001.03.xsd
 tests/camt.054.001.04.xsd
 tests/pain.001.001.03.xsd
 tests/pain.001.001.05.xsd
+tests/pain.001.003.03.xsd
 tests/pain.008.001.02.xsd
 tests/pain.008.001.04.xsd
+tests/pain.008.003.02.xsd
 trytond_account_payment_sepa.egg-info/PKG-INFO
 trytond_account_payment_sepa.egg-info/SOURCES.txt
 trytond_account_payment_sepa.egg-info/dependency_links.txt
diff --git a/trytond_account_payment_sepa.egg-info/requires.txt b/trytond_account_payment_sepa.egg-info/requires.txt
index d610120..643da8e 100644
--- a/trytond_account_payment_sepa.egg-info/requires.txt
+++ b/trytond_account_payment_sepa.egg-info/requires.txt
@@ -1,7 +1,8 @@
 Genshi
 lxml
-trytond_account_payment >= 3.4, < 3.5
-trytond_company >= 3.4, < 3.5
-trytond_bank >= 3.4, < 3.5
-trytond_party >= 3.4, < 3.5
-trytond >= 3.4, < 3.5
\ No newline at end of file
+python-dateutil
+trytond_account_payment >= 3.6, < 3.7
+trytond_company >= 3.6, < 3.7
+trytond_bank >= 3.6, < 3.7
+trytond_party >= 3.6, < 3.7
+trytond >= 3.6, < 3.7
\ No newline at end of file
diff --git a/view/payment_group_form.xml b/view/payment_group_form.xml
index 2a2a169..e2409eb 100644
--- a/view/payment_group_form.xml
+++ b/view/payment_group_form.xml
@@ -4,5 +4,6 @@ this repository contains the full copyright notices and license terms. -->
 <data>
     <xpath expr="/form/field[@name='kind']" position="after">
         <field name="sepa_messages" colspan="4"/>
+        <button name="generate_message" string="Generate Message" colspan="4"/>
     </xpath>
 </data>
diff --git a/view/payment_journal_form.xml b/view/payment_journal_form.xml
index 0f2e1b1..c14b08c 100644
--- a/view/payment_journal_form.xml
+++ b/view/payment_journal_form.xml
@@ -3,8 +3,7 @@
 this repository contains the full copyright notices and license terms. -->
 <data>
     <xpath expr="/form/field[@name='process_method']" position="after">
-        <separator id="sepa" string="SEPA" colspan="4"
-            states="{'invisible': Eval('process_method') != 'sepa'}"/>
+        <separator name="sepa_bank_account_number" string="SEPA" colspan="4"/>
         <label name="sepa_bank_account_number"/>
         <field name="sepa_bank_account_number"/>
         <newline/>
-- 
tryton-modules-account-payment-sepa



More information about the tryton-debian-vcs mailing list