mirror of
https://github.com/OCA/account-reconcile.git
synced 2025-01-20 12:27:39 +02:00
Add a selection of currencies for the 'Currencies' rules
It allows to have a different setup for different currencies (the amounts won't be the same).
This commit is contained in:
committed by
Cyril Gaudin
parent
fd2430747e
commit
cb488c6329
@@ -48,6 +48,12 @@ class AccountStatementOperationRule(models.Model):
|
||||
string='Max. Amount',
|
||||
digits=dp.get_precision('Account'),
|
||||
)
|
||||
currencies = fields.Many2many(
|
||||
comodel_name='res.currency',
|
||||
string='Currencies',
|
||||
help="For 'Currencies' rules, you can choose for which currencies "
|
||||
"the rule will be applicable.",
|
||||
)
|
||||
sequence = fields.Integer(
|
||||
default=20,
|
||||
help="If several rules match, the first one is used.",
|
||||
@@ -88,15 +94,24 @@ class AccountStatementOperationRule(models.Model):
|
||||
|
||||
@api.multi
|
||||
def _is_valid_multicurrency(self, statement_line, move_lines, balance):
|
||||
# FIXME: surely wrong
|
||||
if self._is_multicurrency(statement_line):
|
||||
""" Check if the multi-currency rule can be applied
|
||||
|
||||
The rule is applied if and only if:
|
||||
* The currency is not company's one
|
||||
* The currency of the statement line and all the lines is the same
|
||||
* The balance of the amount currencies is 0
|
||||
* The balance is between the bounds configured on the rule
|
||||
"""
|
||||
if not self._is_multicurrency(statement_line):
|
||||
return False
|
||||
currency = statement_line.currency_for_rules()
|
||||
if currency not in self.currencies:
|
||||
return False
|
||||
amount_currency = statement_line.amount_currency
|
||||
for move_line in move_lines:
|
||||
if move_line.currency_id != statement_line.currency_id:
|
||||
# use case not supported, no rule found
|
||||
return self.browse()
|
||||
return False
|
||||
amount_currency -= move_line.amount_currency
|
||||
|
||||
# amount in currency is the same, so the balance is
|
||||
|
||||
@@ -20,3 +20,4 @@
|
||||
##############################################################################
|
||||
|
||||
from . import test_rule_rounding
|
||||
from . import test_rule_currency
|
||||
|
||||
@@ -22,13 +22,15 @@
|
||||
|
||||
def prepare_statement(test, difference,
|
||||
statement_line_currency=None,
|
||||
move_line_currency=None):
|
||||
move_line_currency=None,
|
||||
amount_currency_difference=0):
|
||||
""" Prepare a bank statement line and a move line
|
||||
|
||||
The difference is applied on the bank statement line relatively to
|
||||
the move line.
|
||||
"""
|
||||
amount = 100
|
||||
amount_currency = 120
|
||||
statement_obj = test.env['account.bank.statement']
|
||||
statement_line_obj = test.env['account.bank.statement.line']
|
||||
move_obj = test.env['account.move']
|
||||
@@ -43,7 +45,11 @@ def prepare_statement(test, difference,
|
||||
'statement_id': statement.id,
|
||||
}
|
||||
if statement_line_currency:
|
||||
line_vals['currency_id'] = statement_line_currency.id
|
||||
line_vals.update({
|
||||
'currency_id': statement_line_currency.id,
|
||||
'amount_currency': amount_currency + amount_currency_difference,
|
||||
})
|
||||
|
||||
statement_line = statement_line_obj.create(line_vals)
|
||||
move = move_obj.create({
|
||||
'journal_id': test.ref('account.sales_journal')
|
||||
@@ -55,7 +61,10 @@ def prepare_statement(test, difference,
|
||||
'debit': amount,
|
||||
}
|
||||
if move_line_currency:
|
||||
line_vals['currency_id'] = move_line_currency.id
|
||||
line_vals.update({
|
||||
'currency_id': move_line_currency.id,
|
||||
'amount_currency': amount_currency,
|
||||
})
|
||||
move_line = move_line_obj.create(line_vals)
|
||||
line_vals = {
|
||||
'move_id': move.id,
|
||||
|
||||
164
account_statement_operation_rule/tests/test_rule_currency.py
Normal file
164
account_statement_operation_rule/tests/test_rule_currency.py
Normal file
@@ -0,0 +1,164 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
##############################################################################
|
||||
#
|
||||
# Author: Guewen Baconnier
|
||||
# Copyright 2014 Camptocamp SA
|
||||
#
|
||||
# This program is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU Affero General Public License as
|
||||
# published by the Free Software Foundation, either version 3 of the
|
||||
# License, or (at your option) any later version.
|
||||
#
|
||||
# This program is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU Affero General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU Affero General Public License
|
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
##############################################################################
|
||||
|
||||
from openerp.tests import common
|
||||
|
||||
from .common import prepare_statement
|
||||
|
||||
|
||||
class TestRuleCurrency(common.TransactionCase):
|
||||
|
||||
def setUp(self):
|
||||
super(TestRuleCurrency, self).setUp()
|
||||
self.operation_obj = self.env['account.statement.operation.template']
|
||||
self.rule_obj = self.env['account.statement.operation.rule']
|
||||
self.aed = self.browse_ref('base.AED')
|
||||
self.afn = self.browse_ref('base.AFN')
|
||||
self.all = self.browse_ref('base.ALL')
|
||||
self.amd = self.browse_ref('base.AMD')
|
||||
self.aoa = self.browse_ref('base.AOA')
|
||||
self.operation_currency_1 = self.operation_obj.create({
|
||||
'name': 'Currency AED, AFR, ALL -1.0 to 0.0',
|
||||
'label': 'Currency',
|
||||
'account_id': self.ref('account.rsa'),
|
||||
'amount_type': 'percentage_of_total',
|
||||
'amount': 100.0,
|
||||
})
|
||||
self.rule_currency_1 = self.rule_obj.create({
|
||||
'name': 'Currency AED, AFR, ALL -1.0 to 0.0',
|
||||
'rule_type': 'currency',
|
||||
'operations': [(6, 0, (self.operation_currency_1.id, ))],
|
||||
'amount_min': -1.0,
|
||||
'amount_max': 0,
|
||||
'sequence': 1,
|
||||
'currencies': [(6, 0, [self.aed.id, self.afn.id, self.all.id])],
|
||||
})
|
||||
self.operation_currency_2 = self.operation_obj.create({
|
||||
'name': 'Currency AED, AFR, ALL -2.0 to -1.0',
|
||||
'label': 'Currency',
|
||||
'account_id': self.ref('account.rsa'),
|
||||
'amount_type': 'percentage_of_total',
|
||||
'amount': 100.0,
|
||||
})
|
||||
self.rule_currency_2 = self.rule_obj.create({
|
||||
'name': 'Currency AED, AFR, ALL -2.0 to 1.0',
|
||||
'rule_type': 'currency',
|
||||
'operations': [(6, 0, (self.operation_currency_2.id, ))],
|
||||
'amount_min': -2.0,
|
||||
'amount_max': -1.0,
|
||||
'sequence': 2,
|
||||
'currencies': [(6, 0, [self.aed.id, self.afn.id, self.all.id])],
|
||||
})
|
||||
self.operation_currency_3 = self.operation_obj.create({
|
||||
'name': 'Currency AMD, AOA -2.0 to 0.0',
|
||||
'label': 'Currency',
|
||||
'account_id': self.ref('account.rsa'),
|
||||
'amount_type': 'percentage_of_total',
|
||||
'amount': 100.0,
|
||||
|
||||
})
|
||||
self.rule_currency_3 = self.rule_obj.create({
|
||||
'name': 'Currency AMD, AOA -2.0 to 0.0',
|
||||
'rule_type': 'currency',
|
||||
'operations': [(6, 0, (self.operation_currency_3.id, ))],
|
||||
'amount_min': -2,
|
||||
'amount_max': 0,
|
||||
'sequence': 2,
|
||||
'currencies': [(6, 0, [self.amd.id, self.aoa.id])],
|
||||
})
|
||||
|
||||
def test_no_currency_match(self):
|
||||
"""No rules for the current currency"""
|
||||
sek = self.browse_ref('base.SEK')
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -0.5,
|
||||
statement_line_currency=sek,
|
||||
move_line_currency=sek)
|
||||
ops = self.rule_obj.operations_for_reconciliation(statement_line.id,
|
||||
move_line.ids)
|
||||
self.assertFalse(ops)
|
||||
|
||||
def test_rounding_lines(self):
|
||||
"""No Currencies rules on lines with company currency"""
|
||||
statement_line, move_line = prepare_statement(self, -0.5)
|
||||
ops = self.rule_obj.operations_for_reconciliation(statement_line.id,
|
||||
move_line.ids)
|
||||
self.assertFalse(ops)
|
||||
|
||||
def test_currency_rule_1(self):
|
||||
"""Rule 1 is found with -0.5 AED"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -0.5,
|
||||
statement_line_currency=self.aed,
|
||||
move_line_currency=self.aed,
|
||||
amount_currency_difference=0)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertEquals(rule, self.rule_currency_1)
|
||||
|
||||
def test_currency_rule_2(self):
|
||||
"""Rule 2 is found with -2 AED"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -2,
|
||||
statement_line_currency=self.aed,
|
||||
move_line_currency=self.aed,
|
||||
amount_currency_difference=0)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertEquals(rule, self.rule_currency_2)
|
||||
|
||||
def test_currency_rule_3(self):
|
||||
"""Rule 3 is found with -2 AOA"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -2,
|
||||
statement_line_currency=self.aoa,
|
||||
move_line_currency=self.aoa,
|
||||
amount_currency_difference=0)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertEquals(rule, self.rule_currency_3)
|
||||
|
||||
def test_currency_rule_not_in_bounds(self):
|
||||
"""No rule is found with -3 AOA"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -3,
|
||||
statement_line_currency=self.aoa,
|
||||
move_line_currency=self.aoa,
|
||||
amount_currency_difference=0)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertFalse(rule)
|
||||
|
||||
def test_no_rule_amount_currency_different(self):
|
||||
"""No rule when amount currency is different"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -0.5,
|
||||
statement_line_currency=self.aed,
|
||||
move_line_currency=self.aed,
|
||||
amount_currency_difference=0.5)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertFalse(rule)
|
||||
|
||||
def test_rule_amount_currency_difference_near_zero(self):
|
||||
"""Apply the rule when the difference is near 0"""
|
||||
statement_line, move_line = prepare_statement(
|
||||
self, -0.5,
|
||||
statement_line_currency=self.aed,
|
||||
move_line_currency=self.aed,
|
||||
amount_currency_difference=-0.001)
|
||||
rule = self.rule_obj.find_first_rule(statement_line, [move_line])
|
||||
self.assertEquals(rule, self.rule_currency_1)
|
||||
@@ -19,7 +19,6 @@
|
||||
#
|
||||
##############################################################################
|
||||
|
||||
import unittest2
|
||||
from openerp.tests import common
|
||||
|
||||
from .common import prepare_statement
|
||||
|
||||
@@ -23,9 +23,13 @@
|
||||
<field name="amount_min" class="oe_inline" /> And
|
||||
<field name="amount_max" class="oe_inline" />
|
||||
</div>
|
||||
<label for="currencies" string="And the currency is one of" attrs="{'invisible': [('rule_type', '!=', 'currency')]}"/>
|
||||
<div attrs="{'invisible': [('rule_type', '!=', 'currency')]}">
|
||||
<field name="currencies" class="oe_inline" widget="many2many_tags"/>
|
||||
</div>
|
||||
</group>
|
||||
<group name="operations">
|
||||
<label string="Then the following operations will be applied:" colspan="2"/>
|
||||
<label for="operations" string="Then the following operations will be applied:" colspan="2"/>
|
||||
<field name="operations" nolabel="1"/>
|
||||
</group>
|
||||
</sheet>
|
||||
|
||||
Reference in New Issue
Block a user