813 lines
45 KiB
Python
813 lines
45 KiB
Python
from flectra.addons.account.tests.account_test_classes import AccountingTestCase
|
|
import time
|
|
import unittest
|
|
|
|
|
|
class TestReconciliation(AccountingTestCase):
|
|
|
|
"""Tests for reconciliation (account.tax)
|
|
|
|
Test used to check that when doing a sale or purchase invoice in a different currency,
|
|
the result will be balanced.
|
|
"""
|
|
|
|
def setUp(self):
|
|
super(TestReconciliation, self).setUp()
|
|
self.account_invoice_model = self.env['account.invoice']
|
|
self.account_invoice_line_model = self.env['account.invoice.line']
|
|
self.acc_bank_stmt_model = self.env['account.bank.statement']
|
|
self.acc_bank_stmt_line_model = self.env['account.bank.statement.line']
|
|
self.res_currency_model = self.registry('res.currency')
|
|
self.res_currency_rate_model = self.registry('res.currency.rate')
|
|
|
|
partner_agrolait = self.env.ref("base.res_partner_2")
|
|
self.partner_agrolait_id = partner_agrolait.id
|
|
self.currency_swiss_id = self.env.ref("base.CHF").id
|
|
self.currency_usd_id = self.env.ref("base.USD").id
|
|
self.currency_euro_id = self.env.ref("base.EUR").id
|
|
company = self.env.ref('base.main_company')
|
|
self.cr.execute("UPDATE res_company SET currency_id = %s WHERE id = %s", [self.currency_euro_id, company.id])
|
|
self.account_rcv = partner_agrolait.property_account_receivable_id or self.env['account.account'].search([('user_type_id', '=', self.env.ref('account.data_account_type_receivable').id)], limit=1)
|
|
self.account_rsa = partner_agrolait.property_account_payable_id or self.env['account.account'].search([('user_type_id', '=', self.env.ref('account.data_account_type_payable').id)], limit=1)
|
|
self.product = self.env.ref("product.product_product_4")
|
|
|
|
self.bank_journal_euro = self.env['account.journal'].create({'name': 'Bank', 'type': 'bank', 'code': 'BNK67'})
|
|
self.account_euro = self.bank_journal_euro.default_debit_account_id
|
|
|
|
self.bank_journal_usd = self.env['account.journal'].create({'name': 'Bank US', 'type': 'bank', 'code': 'BNK68', 'currency_id': self.currency_usd_id})
|
|
self.account_usd = self.bank_journal_usd.default_debit_account_id
|
|
|
|
self.diff_income_account = self.env['res.users'].browse(self.env.uid).company_id.income_currency_exchange_account_id
|
|
self.diff_expense_account = self.env['res.users'].browse(self.env.uid).company_id.expense_currency_exchange_account_id
|
|
|
|
self.inbound_payment_method = self.env['account.payment.method'].create({
|
|
'name': 'inbound',
|
|
'code': 'IN',
|
|
'payment_type': 'inbound',
|
|
})
|
|
|
|
def create_invoice(self, type='out_invoice', invoice_amount=50, currency_id=None):
|
|
#we create an invoice in given currency
|
|
invoice = self.account_invoice_model.create({'partner_id': self.partner_agrolait_id,
|
|
'reference_type': 'none',
|
|
'currency_id': currency_id,
|
|
'name': type == 'out_invoice' and 'invoice to client' or 'invoice to vendor',
|
|
'account_id': self.account_rcv.id,
|
|
'type': type,
|
|
'date_invoice': time.strftime('%Y') + '-07-01',
|
|
})
|
|
self.account_invoice_line_model.create({'product_id': self.product.id,
|
|
'quantity': 1,
|
|
'price_unit': invoice_amount,
|
|
'invoice_id': invoice.id,
|
|
'name': 'product that cost ' + str(invoice_amount),
|
|
'account_id': self.env['account.account'].search([('user_type_id', '=', self.env.ref('account.data_account_type_revenue').id)], limit=1).id,
|
|
})
|
|
|
|
#validate invoice
|
|
invoice.action_invoice_open()
|
|
return invoice
|
|
|
|
def make_payment(self, invoice_record, bank_journal, amount=0.0, amount_currency=0.0, currency_id=None):
|
|
bank_stmt = self.acc_bank_stmt_model.create({
|
|
'journal_id': bank_journal.id,
|
|
'date': time.strftime('%Y') + '-07-15',
|
|
'name': 'payment' + invoice_record.number
|
|
})
|
|
|
|
bank_stmt_line = self.acc_bank_stmt_line_model.create({'name': 'payment',
|
|
'statement_id': bank_stmt.id,
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': amount,
|
|
'amount_currency': amount_currency,
|
|
'currency_id': currency_id,
|
|
'date': time.strftime('%Y') + '-07-15',})
|
|
|
|
#reconcile the payment with the invoice
|
|
for l in invoice_record.move_id.line_ids:
|
|
if l.account_id.id == self.account_rcv.id:
|
|
line_id = l
|
|
break
|
|
amount_in_widget = currency_id and amount_currency or amount
|
|
bank_stmt_line.process_reconciliation(counterpart_aml_dicts=[{
|
|
'move_line': line_id,
|
|
'debit': amount_in_widget < 0 and -amount_in_widget or 0.0,
|
|
'credit': amount_in_widget > 0 and amount_in_widget or 0.0,
|
|
'name': line_id.name,
|
|
}])
|
|
return bank_stmt
|
|
|
|
def check_results(self, move_line_recs, aml_dict):
|
|
#we check that the line is balanced (bank statement line)
|
|
self.assertEquals(len(move_line_recs), len(aml_dict))
|
|
for move_line in move_line_recs:
|
|
self.assertEquals(round(move_line.debit, 2), aml_dict[move_line.account_id.id]['debit'])
|
|
self.assertEquals(round(move_line.credit, 2), aml_dict[move_line.account_id.id]['credit'])
|
|
self.assertEquals(round(move_line.amount_currency, 2), aml_dict[move_line.account_id.id]['amount_currency'])
|
|
self.assertEquals(move_line.currency_id.id, aml_dict[move_line.account_id.id]['currency_id'])
|
|
if 'currency_diff' in aml_dict[move_line.account_id.id]:
|
|
currency_diff_move = move_line.full_reconcile_id.exchange_move_id
|
|
for currency_diff_line in currency_diff_move.line_ids:
|
|
if aml_dict[move_line.account_id.id].get('currency_diff') == 0:
|
|
if currency_diff_line.account_id.id == move_line.account_id.id:
|
|
self.assertAlmostEquals(currency_diff_line.amount_currency, aml_dict[move_line.account_id.id].get('amount_currency_diff'))
|
|
if aml_dict[move_line.account_id.id].get('currency_diff') == 0:
|
|
if currency_diff_line.account_id.id == move_line.account_id.id:
|
|
self.assertAlmostEquals(currency_diff_line.amount_currency, aml_dict[move_line.account_id.id].get('amount_currency_diff'))
|
|
if aml_dict[move_line.account_id.id].get('currency_diff') > 0:
|
|
if currency_diff_line.account_id.id == move_line.account_id.id:
|
|
self.assertAlmostEquals(currency_diff_line.debit, aml_dict[move_line.account_id.id].get('currency_diff'))
|
|
else:
|
|
self.assertAlmostEquals(currency_diff_line.credit, aml_dict[move_line.account_id.id].get('currency_diff'))
|
|
self.assertIn(currency_diff_line.account_id.id, [self.diff_expense_account.id, self.diff_income_account.id], 'The difference accounts should be used correctly. ')
|
|
else:
|
|
if currency_diff_line.account_id.id == move_line.account_id.id:
|
|
self.assertAlmostEquals(currency_diff_line.credit, abs(aml_dict[move_line.account_id.id].get('currency_diff')))
|
|
else:
|
|
self.assertAlmostEquals(currency_diff_line.debit, abs(aml_dict[move_line.account_id.id].get('currency_diff')))
|
|
self.assertIn(currency_diff_line.account_id.id, [self.diff_expense_account.id, self.diff_income_account.id], 'The difference accounts should be used correctly. ')
|
|
|
|
def make_customer_and_supplier_flows(self, invoice_currency_id, invoice_amount, bank_journal, amount, amount_currency, transaction_currency_id):
|
|
#we create an invoice in given invoice_currency
|
|
invoice_record = self.create_invoice(type='out_invoice', invoice_amount=invoice_amount, currency_id=invoice_currency_id)
|
|
#we encode a payment on it, on the given bank_journal with amount, amount_currency and transaction_currency given
|
|
bank_stmt = self.make_payment(invoice_record, bank_journal, amount=amount, amount_currency=amount_currency, currency_id=transaction_currency_id)
|
|
customer_move_lines = bank_stmt.move_line_ids
|
|
|
|
#we create a supplier bill in given invoice_currency
|
|
invoice_record = self.create_invoice(type='in_invoice', invoice_amount=invoice_amount, currency_id=invoice_currency_id)
|
|
#we encode a payment on it, on the given bank_journal with amount, amount_currency and transaction_currency given
|
|
bank_stmt = self.make_payment(invoice_record, bank_journal, amount=-amount, amount_currency=-amount_currency, currency_id=transaction_currency_id)
|
|
supplier_move_lines = bank_stmt.move_line_ids
|
|
return customer_move_lines, supplier_move_lines
|
|
|
|
def test_statement_usd_invoice_eur_transaction_eur(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_euro_id, 30, self.bank_journal_usd, 42, 30, self.currency_euro_id)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_usd.id: {'debit': 30.0, 'credit': 0.0, 'amount_currency': 42, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 30.0, 'amount_currency': -42, 'currency_id': self.currency_usd_id},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_usd.id: {'debit': 0.0, 'credit': 30.0, 'amount_currency': -42, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 30.0, 'credit': 0.0, 'amount_currency': 42, 'currency_id': self.currency_usd_id},
|
|
})
|
|
|
|
def test_statement_usd_invoice_usd_transaction_usd(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_usd_id, 50, self.bank_journal_usd, 50, 0, False)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_usd.id: {'debit': 32.70, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 32.70, 'amount_currency': -50, 'currency_id': self.currency_usd_id},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_usd.id: {'debit': 0.0, 'credit': 32.70, 'amount_currency': -50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 32.70, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id},
|
|
})
|
|
|
|
def test_statement_usd_invoice_usd_transaction_eur(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_usd_id, 50, self.bank_journal_usd, 50, 40, self.currency_euro_id)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_usd.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': -50, 'currency_id': self.currency_usd_id, 'currency_diff': 7.30},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_usd.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': -50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id, 'currency_diff': -7.30},
|
|
})
|
|
|
|
def test_statement_usd_invoice_chf_transaction_chf(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_swiss_id, 50, self.bank_journal_usd, 42, 50, self.currency_swiss_id)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_usd.id: {'debit': 27.47, 'credit': 0.0, 'amount_currency': 42, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 27.47, 'amount_currency': -50, 'currency_id': self.currency_swiss_id, 'currency_diff': -10.74},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_usd.id: {'debit': 0.0, 'credit': 27.47, 'amount_currency': -42, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 27.47, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_swiss_id, 'currency_diff': 10.74},
|
|
})
|
|
|
|
def test_statement_eur_invoice_usd_transaction_usd(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_usd_id, 50, self.bank_journal_euro, 40, 50, self.currency_usd_id)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_euro.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': -50, 'currency_id': self.currency_usd_id, 'currency_diff': 7.30},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_euro.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': -50, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_usd_id, 'currency_diff': -7.30},
|
|
})
|
|
|
|
def test_statement_eur_invoice_usd_transaction_eur(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_usd_id, 50, self.bank_journal_euro, 40, 0.0, False)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_euro.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 0.0, 'currency_id': False},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': -61.16, 'currency_id': self.currency_usd_id},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_euro.id: {'debit': 0.0, 'credit': 40.0, 'amount_currency': 0.0, 'currency_id': False},
|
|
self.account_rcv.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 61.16, 'currency_id': self.currency_usd_id},
|
|
})
|
|
|
|
def test_statement_euro_invoice_usd_transaction_chf(self):
|
|
customer_move_lines, supplier_move_lines = self.make_customer_and_supplier_flows(self.currency_usd_id, 50, self.bank_journal_euro, 42, 50, self.currency_swiss_id)
|
|
self.check_results(customer_move_lines, {
|
|
self.account_euro.id: {'debit': 42.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_swiss_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 42.0, 'amount_currency': -50, 'currency_id': self.currency_swiss_id},
|
|
})
|
|
self.check_results(supplier_move_lines, {
|
|
self.account_euro.id: {'debit': 0.0, 'credit': 42.0, 'amount_currency': -50, 'currency_id': self.currency_swiss_id},
|
|
self.account_rcv.id: {'debit': 42.0, 'credit': 0.0, 'amount_currency': 50, 'currency_id': self.currency_swiss_id},
|
|
})
|
|
|
|
def test_statement_euro_invoice_usd_transaction_euro_full(self):
|
|
#we create an invoice in given invoice_currency
|
|
invoice_record = self.create_invoice(type='out_invoice', invoice_amount=50, currency_id=self.currency_usd_id)
|
|
#we encode a payment on it, on the given bank_journal with amount, amount_currency and transaction_currency given
|
|
bank_stmt = self.acc_bank_stmt_model.create({
|
|
'journal_id': self.bank_journal_euro.id,
|
|
'date': time.strftime('%Y') + '-01-01',
|
|
})
|
|
|
|
bank_stmt_line = self.acc_bank_stmt_line_model.create({'name': 'payment',
|
|
'statement_id': bank_stmt.id,
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 40,
|
|
'date': time.strftime('%Y') + '-01-01',})
|
|
|
|
#reconcile the payment with the invoice
|
|
for l in invoice_record.move_id.line_ids:
|
|
if l.account_id.id == self.account_rcv.id:
|
|
line_id = l
|
|
break
|
|
bank_stmt_line.process_reconciliation(counterpart_aml_dicts=[{
|
|
'move_line': line_id,
|
|
'debit': 0.0,
|
|
'credit': 32.7,
|
|
'name': line_id.name,
|
|
}], new_aml_dicts=[{
|
|
'debit': 0.0,
|
|
'credit': 7.3,
|
|
'name': 'exchange difference',
|
|
'account_id': self.diff_income_account.id
|
|
}])
|
|
self.check_results(bank_stmt.move_line_ids, {
|
|
self.account_euro.id: {'debit': 40.0, 'credit': 0.0, 'amount_currency': 0, 'currency_id': False},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 32.7, 'amount_currency': -41.97, 'currency_id': self.currency_usd_id, 'currency_diff': 0, 'amount_currency_diff': -8.03},
|
|
self.diff_income_account.id: {'debit': 0.0, 'credit': 7.3, 'amount_currency': -9.37, 'currency_id': self.currency_usd_id},
|
|
})
|
|
|
|
# The invoice should be paid, as the payments totally cover its total
|
|
self.assertEquals(invoice_record.state, 'paid', 'The invoice should be paid by now')
|
|
invoice_rec_line = invoice_record.move_id.line_ids.filtered(lambda x: x.account_id.reconcile)
|
|
self.assertTrue(invoice_rec_line.reconciled, 'The invoice should be totally reconciled')
|
|
self.assertEquals(invoice_rec_line.amount_residual, 0, 'The invoice should be totally reconciled')
|
|
self.assertEquals(invoice_rec_line.amount_residual_currency, 0, 'The invoice should be totally reconciled')
|
|
|
|
@unittest.skip('adapt to new accounting')
|
|
def test_balanced_exchanges_gain_loss(self):
|
|
# The point of this test is to show that we handle correctly the gain/loss exchanges during reconciliations in foreign currencies.
|
|
# For instance, with a company set in EUR, and a USD rate set to 0.033,
|
|
# the reconciliation of an invoice of 2.00 USD (60.61 EUR) and a bank statement of two lines of 1.00 USD (30.30 EUR)
|
|
# will lead to an exchange loss, that should be handled correctly within the journal items.
|
|
env = api.Environment(self.cr, self.uid, {})
|
|
# We update the currency rate of the currency USD in order to force the gain/loss exchanges in next steps
|
|
rateUSDbis = env.ref("base.rateUSDbis")
|
|
rateUSDbis.write({
|
|
'name': time.strftime('%Y-%m-%d') + ' 00:00:00',
|
|
'rate': 0.033,
|
|
})
|
|
# We create a customer invoice of 2.00 USD
|
|
invoice = self.account_invoice_model.create({
|
|
'partner_id': self.partner_agrolait_id,
|
|
'currency_id': self.currency_usd_id,
|
|
'name': 'Foreign invoice with exchange gain',
|
|
'account_id': self.account_rcv_id,
|
|
'type': 'out_invoice',
|
|
'date_invoice': time.strftime('%Y-%m-%d'),
|
|
'journal_id': self.bank_journal_usd_id,
|
|
'invoice_line': [
|
|
(0, 0, {
|
|
'name': 'line that will lead to an exchange gain',
|
|
'quantity': 1,
|
|
'price_unit': 2,
|
|
})
|
|
]
|
|
})
|
|
invoice.action_invoice_open()
|
|
# We create a bank statement with two lines of 1.00 USD each.
|
|
statement = self.acc_bank_stmt_model.create({
|
|
'journal_id': self.bank_journal_usd_id,
|
|
'date': time.strftime('%Y-%m-%d'),
|
|
'line_ids': [
|
|
(0, 0, {
|
|
'name': 'half payment',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 1.0,
|
|
'date': time.strftime('%Y-%m-%d')
|
|
}),
|
|
(0, 0, {
|
|
'name': 'second half payment',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 1.0,
|
|
'date': time.strftime('%Y-%m-%d')
|
|
})
|
|
]
|
|
})
|
|
|
|
# We process the reconciliation of the invoice line with the two bank statement lines
|
|
line_id = None
|
|
for l in invoice.move_id.line_id:
|
|
if l.account_id.id == self.account_rcv_id:
|
|
line_id = l
|
|
break
|
|
for statement_line in statement.line_ids:
|
|
statement_line.process_reconciliation([
|
|
{'counterpart_move_line_id': line_id.id, 'credit': 1.0, 'debit': 0.0, 'name': line_id.name}
|
|
])
|
|
|
|
# The invoice should be paid, as the payments totally cover its total
|
|
self.assertEquals(invoice.state, 'paid', 'The invoice should be paid by now')
|
|
reconcile = None
|
|
for payment in invoice.payment_ids:
|
|
reconcile = payment.reconcile_id
|
|
break
|
|
# The invoice should be reconciled (entirely, not a partial reconciliation)
|
|
self.assertTrue(reconcile, 'The invoice should be totally reconciled')
|
|
result = {}
|
|
exchange_loss_line = None
|
|
for line in reconcile.line_id:
|
|
res_account = result.setdefault(line.account_id, {'debit': 0.0, 'credit': 0.0, 'count': 0})
|
|
res_account['debit'] = res_account['debit'] + line.debit
|
|
res_account['credit'] = res_account['credit'] + line.credit
|
|
res_account['count'] += 1
|
|
if line.credit == 0.01:
|
|
exchange_loss_line = line
|
|
# We should be able to find a move line of 0.01 EUR on the Debtors account, being the cent we lost during the currency exchange
|
|
self.assertTrue(exchange_loss_line, 'There should be one move line of 0.01 EUR in credit')
|
|
# The journal items of the reconciliation should have their debit and credit total equal
|
|
# Besides, the total debit and total credit should be 60.61 EUR (2.00 USD)
|
|
self.assertEquals(sum(res['debit'] for res in result.values()), 60.61)
|
|
self.assertEquals(sum(res['credit'] for res in result.items()), 60.61)
|
|
counterpart_exchange_loss_line = None
|
|
for line in exchange_loss_line.move_id.line_id:
|
|
if line.account_id.id == self.account_fx_expense_id:
|
|
counterpart_exchange_loss_line = line
|
|
# We should be able to find a move line of 0.01 EUR on the Foreign Exchange Loss account
|
|
self.assertTrue(counterpart_exchange_loss_line, 'There should be one move line of 0.01 EUR on account "Foreign Exchange Loss"')
|
|
|
|
def test_manual_reconcile_wizard_opw678153(self):
|
|
|
|
def create_move(name, amount, amount_currency, currency_id):
|
|
debit_line_vals = {
|
|
'name': name,
|
|
'debit': amount > 0 and amount or 0.0,
|
|
'credit': amount < 0 and -amount or 0.0,
|
|
'account_id': self.account_rcv.id,
|
|
'amount_currency': amount_currency,
|
|
'currency_id': currency_id,
|
|
}
|
|
credit_line_vals = debit_line_vals.copy()
|
|
credit_line_vals['debit'] = debit_line_vals['credit']
|
|
credit_line_vals['credit'] = debit_line_vals['debit']
|
|
credit_line_vals['account_id'] = self.account_rsa.id
|
|
credit_line_vals['amount_currency'] = -debit_line_vals['amount_currency']
|
|
vals = {
|
|
'journal_id': self.bank_journal_euro.id,
|
|
'line_ids': [(0,0, debit_line_vals), (0, 0, credit_line_vals)]
|
|
}
|
|
return self.env['account.move'].create(vals).id
|
|
move_list_vals = [
|
|
('1', -1.83, 0, self.currency_swiss_id),
|
|
('2', 728.35, 795.05, self.currency_swiss_id),
|
|
('3', -4.46, 0, self.currency_swiss_id),
|
|
('4', 0.32, 0, self.currency_swiss_id),
|
|
('5', 14.72, 16.20, self.currency_swiss_id),
|
|
('6', -737.10, -811.25, self.currency_swiss_id),
|
|
]
|
|
move_ids = []
|
|
for name, amount, amount_currency, currency_id in move_list_vals:
|
|
move_ids.append(create_move(name, amount, amount_currency, currency_id))
|
|
aml_recs = self.env['account.move.line'].search([('move_id', 'in', move_ids), ('account_id', '=', self.account_rcv.id)])
|
|
wizard = self.env['account.move.line.reconcile'].with_context(active_ids=[x.id for x in aml_recs]).create({})
|
|
wizard.trans_rec_reconcile_full()
|
|
for aml in aml_recs:
|
|
self.assertTrue(aml.reconciled, 'The journal item should be totally reconciled')
|
|
self.assertEquals(aml.amount_residual, 0, 'The journal item should be totally reconciled')
|
|
self.assertEquals(aml.amount_residual_currency, 0, 'The journal item should be totally reconciled')
|
|
|
|
move_list_vals = [
|
|
('2', 728.35, 795.05, self.currency_swiss_id),
|
|
('3', -4.46, 0, False),
|
|
('4', 0.32, 0, False),
|
|
('5', 14.72, 16.20, self.currency_swiss_id),
|
|
('6', -737.10, -811.25, self.currency_swiss_id),
|
|
]
|
|
move_ids = []
|
|
for name, amount, amount_currency, currency_id in move_list_vals:
|
|
move_ids.append(create_move(name, amount, amount_currency, currency_id))
|
|
aml_recs = self.env['account.move.line'].search([('move_id', 'in', move_ids), ('account_id', '=', self.account_rcv.id)])
|
|
wizard = self.env['account.move.line.reconcile.writeoff'].with_context(active_ids=[x.id for x in aml_recs]).create({'journal_id': self.bank_journal_usd.id, 'writeoff_acc_id': self.account_rsa.id})
|
|
wizard.trans_rec_reconcile()
|
|
for aml in aml_recs:
|
|
self.assertTrue(aml.reconciled, 'The journal item should be totally reconciled')
|
|
self.assertEquals(aml.amount_residual, 0, 'The journal item should be totally reconciled')
|
|
self.assertEquals(aml.amount_residual_currency, 0, 'The journal item should be totally reconciled')
|
|
|
|
def test_reconcile_bank_statement_with_payment_and_writeoff(self):
|
|
# Use case:
|
|
# Company is in EUR, create a bill for 80 USD and register payment of 80 USD.
|
|
# create a bank statement in USD bank journal with a bank statement line of 85 USD
|
|
# Reconcile bank statement with payment and put the remaining 5 USD in bank fees or another account.
|
|
|
|
invoice = self.create_invoice(type='out_invoice', invoice_amount=80, currency_id=self.currency_usd_id)
|
|
# register payment on invoice
|
|
payment = self.env['account.payment'].create({'payment_type': 'inbound',
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_in').id,
|
|
'partner_type': 'customer',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 80,
|
|
'currency_id': self.currency_usd_id,
|
|
'payment_date': time.strftime('%Y') + '-07-15',
|
|
'journal_id': self.bank_journal_usd.id,
|
|
})
|
|
payment.post()
|
|
payment_move_line = False
|
|
bank_move_line = False
|
|
for l in payment.move_line_ids:
|
|
if l.account_id.id == self.account_rcv.id:
|
|
payment_move_line = l
|
|
else:
|
|
bank_move_line = l
|
|
invoice.register_payment(payment_move_line)
|
|
|
|
# create bank statement
|
|
bank_stmt = self.acc_bank_stmt_model.create({
|
|
'journal_id': self.bank_journal_usd.id,
|
|
'date': time.strftime('%Y') + '-07-15',
|
|
})
|
|
|
|
bank_stmt_line = self.acc_bank_stmt_line_model.create({'name': 'payment',
|
|
'statement_id': bank_stmt.id,
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 85,
|
|
'date': time.strftime('%Y') + '-07-15',})
|
|
|
|
#reconcile the statement with invoice and put remaining in another account
|
|
bank_stmt_line.process_reconciliation(payment_aml_rec= bank_move_line, new_aml_dicts=[{
|
|
'account_id': self.diff_income_account.id,
|
|
'debit': 0,
|
|
'credit': 5,
|
|
'name': 'bank fees',
|
|
}])
|
|
|
|
# Check that move lines associated to bank_statement are correct
|
|
bank_stmt_aml = self.env['account.move.line'].search([('statement_id', '=', bank_stmt.id)])
|
|
bank_stmt_aml |= bank_stmt_aml.mapped('move_id').mapped('line_ids')
|
|
self.assertEquals(len(bank_stmt_aml), 4, "The bank statement should have 4 moves lines")
|
|
lines = {
|
|
self.account_usd.id: [
|
|
{'debit': 3.27, 'credit': 0.0, 'amount_currency': 5, 'currency_id': self.currency_usd_id},
|
|
{'debit': 52.33, 'credit': 0, 'amount_currency': 80, 'currency_id': self.currency_usd_id}
|
|
],
|
|
self.diff_income_account.id: {'debit': 0.0, 'credit': 3.27, 'amount_currency': -5, 'currency_id': self.currency_usd_id},
|
|
self.account_rcv.id: {'debit': 0.0, 'credit': 52.33, 'amount_currency': -80, 'currency_id': self.currency_usd_id},
|
|
}
|
|
for aml in bank_stmt_aml:
|
|
line = lines[aml.account_id.id]
|
|
if type(line) == list:
|
|
# find correct line inside the list
|
|
if line[0]['debit'] == round(aml.debit, 2):
|
|
line = line[0]
|
|
else:
|
|
line = line[1]
|
|
self.assertEquals(round(aml.debit, 2), line['debit'])
|
|
self.assertEquals(round(aml.credit, 2), line['credit'])
|
|
self.assertEquals(round(aml.amount_currency, 2), line['amount_currency'])
|
|
self.assertEquals(aml.currency_id.id, line['currency_id'])
|
|
|
|
def test_partial_reconcile_currencies(self):
|
|
# client Account (payable, rsa)
|
|
# Debit Credit
|
|
# --------------------------------------------------------
|
|
# Pay a : 25/0.5 = 50 | Inv a : 50/0.5 = 100
|
|
# Pay b: 50/0.75 = 66.66 | Inv b : 50/0.75 = 66.66
|
|
# Pay c: 25/0.8 = 31.25 |
|
|
#
|
|
# Debit_currency = 100 | Credit currency = 100
|
|
# Debit = 147.91 | Credit = 166.66
|
|
# Balance Debit = 18.75
|
|
# Counterpart Credit goes in Exchange diff
|
|
|
|
dest_journal_id = self.env['account.journal'].search([('type', '=', 'purchase'), ('company_id', '=', self.env.ref('base.main_company').id)], limit=1)
|
|
account_expenses = self.env['account.account'].search([('user_type_id', '=', self.env.ref('account.data_account_type_expenses').id)], limit=1)
|
|
|
|
self.bank_journal_euro.write({'default_debit_account_id': self.account_rsa.id,
|
|
'default_credit_account_id': self.account_rsa.id})
|
|
dest_journal_id.write({'default_debit_account_id': self.account_rsa.id,
|
|
'default_credit_account_id': self.account_rsa.id})
|
|
# Setting up rates for USD (main_company is in EUR)
|
|
self.env['res.currency.rate'].create({'name': time.strftime('%Y') + '-' + '07' + '-01',
|
|
'rate': 0.5,
|
|
'currency_id': self.currency_usd_id,
|
|
'company_id': self.env.ref('base.main_company').id})
|
|
|
|
self.env['res.currency.rate'].create({'name': time.strftime('%Y') + '-' + '08' + '-01',
|
|
'rate': 0.75,
|
|
'currency_id': self.currency_usd_id,
|
|
'company_id': self.env.ref('base.main_company').id})
|
|
|
|
self.env['res.currency.rate'].create({'name': time.strftime('%Y') + '-' + '09' + '-01',
|
|
'rate': 0.80,
|
|
'currency_id': self.currency_usd_id,
|
|
'company_id': self.env.ref('base.main_company').id})
|
|
|
|
# Preparing Invoices (from vendor)
|
|
invoice_a = self.account_invoice_model.create({'partner_id': self.partner_agrolait_id,
|
|
'reference_type': 'none',
|
|
'currency_id': self.currency_usd_id,
|
|
'name': 'invoice to vendor',
|
|
'account_id': self.account_rsa.id,
|
|
'type': 'in_invoice',
|
|
'date_invoice': time.strftime('%Y') + '-' + '07' + '-01',
|
|
})
|
|
self.account_invoice_line_model.create({'product_id': self.product.id,
|
|
'quantity': 1,
|
|
'price_unit': 50,
|
|
'invoice_id': invoice_a.id,
|
|
'name': 'product that cost ' + str(50),
|
|
'account_id': account_expenses.id,
|
|
})
|
|
|
|
invoice_b = self.account_invoice_model.create({'partner_id': self.partner_agrolait_id,
|
|
'reference_type': 'none',
|
|
'currency_id': self.currency_usd_id,
|
|
'name': 'invoice to vendor',
|
|
'account_id': self.account_rsa.id,
|
|
'type': 'in_invoice',
|
|
'date_invoice': time.strftime('%Y') + '-' + '08' + '-01',
|
|
})
|
|
self.account_invoice_line_model.create({'product_id': self.product.id,
|
|
'quantity': 1,
|
|
'price_unit': 50,
|
|
'invoice_id': invoice_b.id,
|
|
'name': 'product that cost ' + str(50),
|
|
'account_id': account_expenses.id,
|
|
})
|
|
|
|
invoice_a.action_invoice_open()
|
|
invoice_b.action_invoice_open()
|
|
|
|
# Preparing Payments
|
|
# One partial for invoice_a (fully assigned to it)
|
|
payment_a = self.env['account.payment'].create({'payment_type': 'outbound',
|
|
'amount': 25,
|
|
'currency_id': self.currency_usd_id,
|
|
'journal_id': self.bank_journal_euro.id,
|
|
'company_id': self.env.ref('base.main_company').id,
|
|
'payment_date': time.strftime('%Y') + '-' + '07' + '-01',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_out').id,
|
|
'destination_journal_id': dest_journal_id.id,
|
|
'partner_type': 'supplier'})
|
|
|
|
# One that will complete the payment of a, the rest goes to b
|
|
payment_b = self.env['account.payment'].create({'payment_type': 'outbound',
|
|
'amount': 50,
|
|
'currency_id': self.currency_usd_id,
|
|
'journal_id': self.bank_journal_euro.id,
|
|
'company_id': self.env.ref('base.main_company').id,
|
|
'payment_date': time.strftime('%Y') + '-' + '08' + '-01',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_out').id,
|
|
'destination_journal_id': dest_journal_id.id,
|
|
'partner_type': 'supplier'})
|
|
|
|
# The last one will complete the payment of b
|
|
payment_c = self.env['account.payment'].create({'payment_type': 'outbound',
|
|
'amount': 25,
|
|
'currency_id': self.currency_usd_id,
|
|
'journal_id': self.bank_journal_euro.id,
|
|
'company_id': self.env.ref('base.main_company').id,
|
|
'payment_date': time.strftime('%Y') + '-' + '09' + '-01',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_out').id,
|
|
'destination_journal_id': dest_journal_id.id,
|
|
'partner_type': 'supplier'})
|
|
|
|
payment_a.post()
|
|
payment_b.post()
|
|
payment_c.post()
|
|
|
|
# Assigning payments to invoices
|
|
debit_line_a = payment_a.move_line_ids.filtered(lambda l: l.debit and l.account_id == dest_journal_id.default_debit_account_id)
|
|
debit_line_b = payment_b.move_line_ids.filtered(lambda l: l.debit and l.account_id == dest_journal_id.default_debit_account_id)
|
|
debit_line_c = payment_c.move_line_ids.filtered(lambda l: l.debit and l.account_id == dest_journal_id.default_debit_account_id)
|
|
|
|
invoice_a.assign_outstanding_credit(debit_line_a.id)
|
|
invoice_a.assign_outstanding_credit(debit_line_b.id)
|
|
invoice_b.assign_outstanding_credit(debit_line_b.id)
|
|
invoice_b.assign_outstanding_credit(debit_line_c.id)
|
|
|
|
# Asserting correctness (only in the payable account)
|
|
full_reconcile = False
|
|
for inv in (invoice_a + invoice_b):
|
|
self.assertTrue(inv.reconciled)
|
|
for aml in (inv.payment_move_line_ids + inv.move_id.line_ids).filtered(lambda l: l.account_id == self.account_rsa):
|
|
self.assertEqual(aml.amount_residual, 0.0)
|
|
self.assertEqual(aml.amount_residual_currency, 0.0)
|
|
self.assertTrue(aml.reconciled)
|
|
if not full_reconcile:
|
|
full_reconcile = aml.full_reconcile_id
|
|
else:
|
|
self.assertTrue(aml.full_reconcile_id == full_reconcile)
|
|
|
|
full_rec_move = full_reconcile.exchange_move_id
|
|
# Globally check whether the amount is correct
|
|
self.assertEqual(full_rec_move.amount, 18.75)
|
|
|
|
# Checking if the direction of the move is correct
|
|
full_rec_payable = full_rec_move.line_ids.filtered(lambda l: l.account_id == self.account_rsa)
|
|
self.assertEqual(full_rec_payable.balance, 18.75)
|
|
|
|
def test_unreconcile(self):
|
|
# Use case:
|
|
# 2 invoices paid with a single payment. Unreconcile the payment with one invoice, the
|
|
# other invoice should remain reconciled.
|
|
inv1 = self.create_invoice(invoice_amount=10, currency_id=self.currency_usd_id)
|
|
inv2 = self.create_invoice(invoice_amount=20, currency_id=self.currency_usd_id)
|
|
payment = self.env['account.payment'].create({
|
|
'payment_type': 'inbound',
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_in').id,
|
|
'partner_type': 'customer',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 100,
|
|
'currency_id': self.currency_usd_id,
|
|
'journal_id': self.bank_journal_usd.id,
|
|
})
|
|
payment.post()
|
|
credit_aml = payment.move_line_ids.filtered('credit')
|
|
|
|
# Check residual before assignation
|
|
self.assertAlmostEquals(inv1.residual, 10)
|
|
self.assertAlmostEquals(inv2.residual, 20)
|
|
|
|
# Assign credit and residual
|
|
inv1.assign_outstanding_credit(credit_aml.id)
|
|
inv2.assign_outstanding_credit(credit_aml.id)
|
|
self.assertAlmostEquals(inv1.residual, 0)
|
|
self.assertAlmostEquals(inv2.residual, 0)
|
|
|
|
# Unreconcile one invoice at a time and check residual
|
|
credit_aml.with_context(invoice_id=inv1.id).remove_move_reconcile()
|
|
self.assertAlmostEquals(inv1.residual, 10)
|
|
self.assertAlmostEquals(inv2.residual, 0)
|
|
credit_aml.with_context(invoice_id=inv2.id).remove_move_reconcile()
|
|
self.assertAlmostEquals(inv1.residual, 10)
|
|
self.assertAlmostEquals(inv2.residual, 20)
|
|
|
|
def test_unreconcile_exchange(self):
|
|
# Use case:
|
|
# - Company currency in EUR
|
|
# - Create 2 rates for USD:
|
|
# 1.0 on 2018-01-01
|
|
# 0.5 on 2018-02-01
|
|
# - Create an invoice on 2018-01-02 of 111 USD
|
|
# - Register a payment on 2018-02-02 of 111 USD
|
|
# - Unreconcile the payment
|
|
|
|
self.env['res.currency.rate'].create({
|
|
'name': time.strftime('%Y') + '-07-01',
|
|
'rate': 1.0,
|
|
'currency_id': self.currency_usd_id,
|
|
'company_id': self.env.ref('base.main_company').id
|
|
})
|
|
self.env['res.currency.rate'].create({
|
|
'name': time.strftime('%Y') + '-08-01',
|
|
'rate': 0.5,
|
|
'currency_id': self.currency_usd_id,
|
|
'company_id': self.env.ref('base.main_company').id
|
|
})
|
|
inv = self.create_invoice(invoice_amount=111, currency_id=self.currency_usd_id)
|
|
payment = self.env['account.payment'].create({
|
|
'payment_type': 'inbound',
|
|
'payment_method_id': self.env.ref('account.account_payment_method_manual_in').id,
|
|
'partner_type': 'customer',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'amount': 111,
|
|
'currency_id': self.currency_usd_id,
|
|
'journal_id': self.bank_journal_usd.id,
|
|
'payment_date': time.strftime('%Y') + '-08-01',
|
|
})
|
|
payment.post()
|
|
credit_aml = payment.move_line_ids.filtered('credit')
|
|
|
|
# Check residual before assignation
|
|
self.assertAlmostEquals(inv.residual, 111)
|
|
|
|
# Assign credit, check exchange move and residual
|
|
inv.assign_outstanding_credit(credit_aml.id)
|
|
self.assertEqual(len(payment.move_line_ids.mapped('full_reconcile_id').exchange_move_id), 1)
|
|
self.assertAlmostEquals(inv.residual, 0)
|
|
|
|
# Unreconcile invoice and check residual
|
|
credit_aml.with_context(invoice_id=inv.id).remove_move_reconcile()
|
|
self.assertAlmostEquals(inv.residual, 111)
|
|
|
|
def test_revert_payment_and_reconcile(self):
|
|
payment = self.env['account.payment'].create({
|
|
'payment_method_id': self.inbound_payment_method.id,
|
|
'payment_type': 'inbound',
|
|
'partner_type': 'customer',
|
|
'partner_id': self.partner_agrolait_id,
|
|
'journal_id': self.bank_journal_usd.id,
|
|
'payment_date': '2018-06-04',
|
|
'amount': 666,
|
|
})
|
|
payment.post()
|
|
|
|
self.assertEqual(len(payment.move_line_ids), 2)
|
|
|
|
bank_line = payment.move_line_ids.filtered(lambda l: l.account_id.id == self.bank_journal_usd.default_debit_account_id.id)
|
|
customer_line = payment.move_line_ids - bank_line
|
|
|
|
self.assertEqual(len(bank_line), 1)
|
|
self.assertEqual(len(customer_line), 1)
|
|
self.assertNotEqual(bank_line.id, customer_line.id)
|
|
|
|
self.assertEqual(bank_line.move_id.id, customer_line.move_id.id)
|
|
move = bank_line.move_id
|
|
|
|
# Reversing the payment's move
|
|
reversed_move_list = move.reverse_moves('2018-06-04')
|
|
self.assertEqual(len(reversed_move_list), 1)
|
|
reversed_move = self.env['account.move'].browse(reversed_move_list[0])
|
|
|
|
self.assertEqual(len(reversed_move.line_ids), 2)
|
|
|
|
# Testing the reconciliation matching between the move lines and their reversed counterparts
|
|
reversed_bank_line = reversed_move.line_ids.filtered(lambda l: l.account_id.id == self.bank_journal_usd.default_debit_account_id.id)
|
|
reversed_customer_line = reversed_move.line_ids - reversed_bank_line
|
|
|
|
self.assertEqual(len(reversed_bank_line), 1)
|
|
self.assertEqual(len(reversed_customer_line), 1)
|
|
self.assertNotEqual(reversed_bank_line.id, reversed_customer_line.id)
|
|
self.assertEqual(reversed_bank_line.move_id.id, reversed_customer_line.move_id.id)
|
|
|
|
self.assertEqual(reversed_bank_line.full_reconcile_id.id, bank_line.full_reconcile_id.id)
|
|
self.assertEqual(reversed_customer_line.full_reconcile_id.id, customer_line.full_reconcile_id.id)
|
|
|
|
def create_invoice_partner(self, type='out_invoice', invoice_amount=50, currency_id=None, partner_id=False):
|
|
#we create an invoice in given currency
|
|
invoice = self.account_invoice_model.create({'partner_id': partner_id,
|
|
'reference_type': 'none',
|
|
'currency_id': currency_id,
|
|
'name': type == 'out_invoice' and 'invoice to client' or 'invoice to vendor',
|
|
'account_id': self.account_rcv.id,
|
|
'type': type,
|
|
'date_invoice': time.strftime('%Y') + '-07-01',
|
|
})
|
|
self.account_invoice_line_model.create({'product_id': self.product.id,
|
|
'quantity': 1,
|
|
'price_unit': invoice_amount,
|
|
'invoice_id': invoice.id,
|
|
'name': 'product that cost ' + str(invoice_amount),
|
|
'account_id': self.env['account.account'].search([('user_type_id', '=', self.env.ref('account.data_account_type_revenue').id)], limit=1).id,
|
|
})
|
|
|
|
#validate invoice
|
|
invoice.action_invoice_open()
|
|
return invoice
|
|
|
|
def test_aged_report(self):
|
|
AgedReport = self.env['report.account.report_agedpartnerbalance'].with_context(include_nullified_amount=True)
|
|
account_type = ['receivable']
|
|
report_date_to = time.strftime('%Y') + '-07-15'
|
|
partner = self.env['res.partner'].create({'name': 'AgedPartner'})
|
|
currency = self.env.user.company_id.currency_id
|
|
|
|
invoice = self.create_invoice_partner(currency_id=currency.id, partner_id=partner.id)
|
|
journal = self.env['account.journal'].create({'name': 'Bank', 'type': 'bank', 'code': 'THE', 'currency_id': currency.id})
|
|
|
|
statement = self.make_payment(invoice, journal, 50)
|
|
|
|
# Case 1: The invoice and payment are reconciled: Nothing should appear
|
|
report_lines, total, amls = AgedReport._get_partner_move_lines(account_type, report_date_to, 'posted', 30, self.env.user.company_id.branch_id)
|
|
partner_lines = [line for line in report_lines if line['partner_id'] == partner.id]
|
|
self.assertEqual(partner_lines, [], 'The aged receivable shouldn\'t have lines at this point')
|
|
self.assertFalse(partner.id in amls, 'The aged receivable should not have amls either')
|
|
|
|
# Case 2: The invoice and payment are not reconciled: we should have one line on the report
|
|
# and 2 amls
|
|
invoice.move_id.line_ids.with_context(invoice_id=invoice.id).remove_move_reconcile()
|
|
report_lines, total, amls = AgedReport._get_partner_move_lines(account_type, report_date_to, 'posted', 30, self.env.user.company_id.branch_id)
|
|
|
|
partner_lines = [line for line in report_lines if line['partner_id'] == partner.id]
|
|
self.assertEqual(partner_lines, [{'trust': 'normal', '1': 0.0, '0': 0.0, 'direction': 0.0, 'partner_id': partner.id, '3': 0.0, 'total': 0.0, 'name': 'AgedPartner', '4': 0.0, '2': 0.0}],
|
|
'We should have a line in the report for the partner')
|
|
self.assertEqual(len(amls[partner.id]), 2, 'We should have 2 account move lines for the partner')
|
|
|
|
positive_line = [line for line in amls[partner.id] if line['line'].balance > 0]
|
|
negative_line = [line for line in amls[partner.id] if line['line'].balance < 0]
|
|
|
|
self.assertEqual(positive_line[0]['amount'], 50.0, 'The amount of the amls should be 50')
|
|
self.assertEqual(negative_line[0]['amount'], -50.0, 'The amount of the amls should be -50')
|