From 3f777d437c95bfb9643e2f39bc55f27a2cb628df Mon Sep 17 00:00:00 2001 From: Brice bib Bartoletti <bib@odoo.com> Date: Thu, 11 Nov 2021 18:26:27 +0000 Subject: [PATCH] [FIX] stock_account:make cogs multi_company safe The aim of this commit is to make the cogs generation multi-company safe. Before this commit: When posting an invoice from company B with both company A and B selected and A being selected as "main", the cost from company A could be selected instead of cost from company A to create the anglosaxon lines. After this commit: Whatever the selected companies are, the cost selected will always be from the company that created the invoice. closes odoo/odoo#79674 Task: #2686104 Signed-off-by: Laurent Smet <las@odoo.com> --- addons/point_of_sale/models/pos_order.py | 2 +- addons/sale_mrp/models/account_move.py | 2 +- addons/sale_stock/models/account_move.py | 5 +- addons/sale_stock/tests/__init__.py | 1 + .../tests/test_anglosaxon_account.py | 78 +++++++++++++++++++ addons/stock_account/models/account_move.py | 2 +- .../tests/test_stockvaluation.py | 2 +- 7 files changed, 85 insertions(+), 7 deletions(-) create mode 100644 addons/sale_stock/tests/test_anglosaxon_account.py diff --git a/addons/point_of_sale/models/pos_order.py b/addons/point_of_sale/models/pos_order.py index 71947793694e..3de53f52b774 100644 --- a/addons/point_of_sale/models/pos_order.py +++ b/addons/point_of_sale/models/pos_order.py @@ -194,7 +194,7 @@ class PosOrder(models.Model): .mapped('picking_id.move_lines')\ .filtered(lambda m: m.product_id.id == product.id)\ .sorted(lambda x: x.date) - price_unit = product._compute_average_price(0, quantity, moves) + price_unit = product.with_context(force_company=self.company_id.id)._compute_average_price(0, quantity, moves) return - price_unit name = fields.Char(string='Order Ref', required=True, readonly=True, copy=False, default='/') diff --git a/addons/sale_mrp/models/account_move.py b/addons/sale_mrp/models/account_move.py index 9ecbb7c15b2f..7812fa805397 100644 --- a/addons/sale_mrp/models/account_move.py +++ b/addons/sale_mrp/models/account_move.py @@ -26,7 +26,7 @@ class AccountMoveLine(models.Model): prod_moves = moves.filtered(lambda m: m.product_id == product) prod_qty_invoiced = factor * qty_invoiced prod_qty_to_invoice = factor * qty_to_invoice - average_price_unit += factor * product._compute_average_price(prod_qty_invoiced, prod_qty_to_invoice, prod_moves) + average_price_unit += factor * product.with_context(force_company=self.company_id.id)._compute_average_price(prod_qty_invoiced, prod_qty_to_invoice, prod_moves) price_unit = average_price_unit / bom.product_qty or price_unit price_unit = self.product_id.uom_id._compute_price(price_unit, self.product_uom_id) return price_unit diff --git a/addons/sale_stock/models/account_move.py b/addons/sale_stock/models/account_move.py index d9f9d78e27a5..65c59770493e 100644 --- a/addons/sale_stock/models/account_move.py +++ b/addons/sale_stock/models/account_move.py @@ -117,9 +117,8 @@ class AccountMoveLine(models.Model): if so_line: qty_to_invoice = self.product_uom_id._compute_quantity(self.quantity, self.product_id.uom_id) qty_invoiced = sum([x.product_uom_id._compute_quantity(x.quantity, x.product_id.uom_id) for x in so_line.invoice_lines if x.move_id.state == 'posted']) - average_price_unit = self.product_id._compute_average_price(qty_invoiced, qty_to_invoice, so_line.move_ids) + average_price_unit = self.product_id.with_context(force_company=self.company_id.id)._compute_average_price(qty_invoiced, qty_to_invoice, so_line.move_ids) price_unit = average_price_unit or price_unit - price_unit = self.product_id.uom_id._compute_price(price_unit, self.product_uom_id) + price_unit = self.product_id.uom_id.with_context(force_company=self.company_id.id)._compute_price(price_unit, self.product_uom_id) return price_unit - diff --git a/addons/sale_stock/tests/__init__.py b/addons/sale_stock/tests/__init__.py index 017499f740f3..5dc7952cd3f1 100644 --- a/addons/sale_stock/tests/__init__.py +++ b/addons/sale_stock/tests/__init__.py @@ -2,6 +2,7 @@ # Part of Odoo. See LICENSE file for full copyright and licensing details. from . import test_anglo_saxon_valuation from . import test_anglo_saxon_valuation_reconciliation +from . import test_anglosaxon_account from . import test_sale_stock from . import test_sale_stock_lead_time from . import test_sale_order_dates diff --git a/addons/sale_stock/tests/test_anglosaxon_account.py b/addons/sale_stock/tests/test_anglosaxon_account.py new file mode 100644 index 000000000000..8fa748852f91 --- /dev/null +++ b/addons/sale_stock/tests/test_anglosaxon_account.py @@ -0,0 +1,78 @@ +# -*- coding: utf-8 -*- +from odoo.addons.account.tests.account_test_savepoint import AccountTestInvoicingCommon +from odoo.tests import tagged + +@tagged('post_install', '-at_install') +class TestAngloSaxonAccounting(AccountTestInvoicingCommon): + + @classmethod + def setUpClass(cls): + super().setUpClass() + (cls.company_data['company'] + cls.company_data_2['company']).write({'anglo_saxon_accounting': True}) + product_category = cls.env['product.category'].create({ + 'name': 'a random storable product category', + 'property_cost_method': 'fifo', + 'property_valuation': 'real_time', + }) + cls.storable_product = cls.env['product.product'].create({ + 'name': 'storable product a', + 'type': 'product', + 'categ_id': product_category.id, + }) + # Those values are company dependent and need to be explicitly set for both companies + product_category.with_context(force_company=cls.company_data_2['company'].id).write({ + 'property_cost_method': 'fifo', + 'property_valuation': 'real_time', + }) + + def test_cogs_should_use_price_from_the_right_company(self): + """ + Reproduce the flow of creating an invoice from a sale order with company A + and posting the invoice with both companies selected and company B as the main. + """ + company_a_data = self.company_data + company_b_data = self.company_data_2 + product = self.storable_product + + # set different cost price for the same product in the 2 companies + company_a_standard_price = 20.0 + product.with_context(force_company=company_a_data['company'].id).standard_price = company_a_standard_price + company_b_standard_price = 10.0 + product.with_context(force_company=company_b_data['company'].id).standard_price = company_b_standard_price + + # create sale order with company A in draft (by default, self.env.user.company_id is company A) + company_a_order = self.env['sale.order'].create({ + 'name': 'testing sale order', + 'partner_id': self.partner_a.id, + 'order_line': [ + (0, 0, { + 'name': 'product storable product a', + 'product_id': product.id, + 'product_uom_qty': 1, + 'product_uom': product.uom_id.id, + 'price_unit': 40.0, + }) + ], + }) + company_a_order.action_confirm() + + # Create an invoice from the sale order using the sale wizard + self.env['sale.advance.payment.inv'].with_context({ + 'active_model': 'sale.order', + 'active_ids': [company_a_order.id], + 'active_id': company_a_order.id, + 'default_journal_id': company_a_data['default_journal_sale'].id + }).create({ + 'advance_payment_method': 'delivered' + }).create_invoices() + company_a_invoice = company_a_order.invoice_ids + + # Post the invoice from company A with company B + company_a_invoice.with_context(force_company=company_b_data['company'].id).action_post() + + # check cost used for anglo_saxon_line is from company A + anglo_saxon_lines = company_a_invoice.line_ids.filtered('is_anglo_saxon_line') + self.assertRecordValues(anglo_saxon_lines, [ + {'debit': 0.0, 'credit': company_a_standard_price}, + {'debit': company_a_standard_price, 'credit': 0.0}, + ]) diff --git a/addons/stock_account/models/account_move.py b/addons/stock_account/models/account_move.py index 9633e3b2241a..f611b1f45531 100644 --- a/addons/stock_account/models/account_move.py +++ b/addons/stock_account/models/account_move.py @@ -239,4 +239,4 @@ class AccountMoveLine(models.Model): self.ensure_one() if not self.product_id: return self.price_unit - return self.product_id._stock_account_get_anglo_saxon_price_unit(uom=self.product_uom_id) + return self.product_id.with_context(force_company=self.company_id.id)._stock_account_get_anglo_saxon_price_unit(uom=self.product_uom_id) diff --git a/addons/stock_account/tests/test_stockvaluation.py b/addons/stock_account/tests/test_stockvaluation.py index 3ba8bc1c0fd6..0e0991fbe51f 100644 --- a/addons/stock_account/tests/test_stockvaluation.py +++ b/addons/stock_account/tests/test_stockvaluation.py @@ -2364,7 +2364,7 @@ class TestStockValuation(SavepointCase): self.assertAlmostEqual(self.product1.standard_price, 10.0) - def test_average_perpetual_8(self): + def test_average_perpetual_9(self): """ When a product has an available quantity of -5, edit an incoming shipment and increase the received quantity by 5 units. """ -- GitLab