odoo18/addons/mrp_subcontracting_dropship.../tests/test_sale_dropshipping.py

401 lines
18 KiB
Python

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.
from odoo.tests import Form
from odoo.addons.mrp_subcontracting.tests.common import TestMrpSubcontractingCommon
class TestSaleDropshippingFlows(TestMrpSubcontractingCommon):
@classmethod
def setUpClass(cls):
super().setUpClass()
cls.supplier = cls.env["res.partner"].create({"name": "Supplier"})
cls.customer = cls.env["res.partner"].create({"name": "Customer"})
cls.dropship_route = cls.env.ref('stock_dropshipping.route_drop_shipping')
def test_dropship_with_different_suppliers(self):
"""
Suppose a kit with 3 components supplied by 3 vendors
When dropshipping this kit, if 2 components are delivered and if the last
picking is cancelled, we should consider the kit as fully delivered.
"""
partners = self.env['res.partner'].create([{'name': 'Vendor %s' % i} for i in range(4)])
compo01, compo02, compo03, kit = self.env['product.product'].create([{
'name': name,
'type': 'consu',
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': seller.id})],
} for name, seller in zip(['Compo01', 'Compo02', 'Compo03', 'Kit'], partners)])
self.env['mrp.bom'].create({
'product_tmpl_id': kit.product_tmpl_id.id,
'product_qty': 1,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': compo01.id, 'product_qty': 1}),
(0, 0, {'product_id': compo02.id, 'product_qty': 1}),
(0, 0, {'product_id': compo03.id, 'product_qty': 1}),
],
})
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': kit.name, 'product_id': kit.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0)
purchase_orders = self.env['purchase.order'].search([('partner_id', 'in', partners.ids)])
purchase_orders.button_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0)
# Deliver the first one
picking = sale_order.picking_ids.filtered(lambda p: p.partner_id == partners[0])
picking.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 0)
# Deliver the third one
picking = sale_order.picking_ids.filtered(lambda p: p.partner_id == partners[2])
picking.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 0)
# Cancel the second one
sale_order.picking_ids[1].action_cancel()
self.assertEqual(sale_order.order_line.qty_delivered, 1)
def test_return_kit_and_delivered_qty(self):
"""
Sell a kit thanks to the dropshipping route, return it then deliver it again
The delivered quantity should be correctly computed
"""
compo, kit = self.env['product.product'].create([{
'name': n,
'type': 'consu',
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': self.supplier.id})],
} for n in ['Compo', 'Kit']])
self.env['mrp.bom'].create({
'product_tmpl_id': kit.product_tmpl_id.id,
'product_qty': 1,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': compo.id, 'product_qty': 1}),
],
})
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': kit.name, 'product_id': kit.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
sale_order._get_purchase_orders().button_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0)
picking = sale_order.picking_ids
picking.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 1.0)
for case in ['return', 'deliver again']:
delivered_before_case = 1.0 if case == 'return' else 0.0
delivered_after_case = 0.0 if case == 'return' else 1.0
return_form = Form(self.env['stock.return.picking'].with_context(active_ids=[picking.id], active_id=picking.id, active_model='stock.picking'))
with return_form.product_return_moves.edit(0) as line_form:
line_form.quantity = 1.0
return_wizard = return_form.save()
action = return_wizard.action_create_returns()
picking = self.env['stock.picking'].browse(action['res_id'])
self.assertEqual(sale_order.order_line.qty_delivered, delivered_before_case, "Incorrect delivered qty for case '%s'" % case)
picking.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, delivered_after_case, "Incorrect delivered qty for case '%s'" % case)
def test_partial_return_kit_and_delivered_qty(self):
"""
Suppose a kit with 4x the same dropshipped component
Suppose a complex delivery process:
- Deliver 2 (with backorder)
- Return 2
- Deliver 1 (with backorder)
- Deliver 1 (process "done")
- Deliver 1 (from the return)
- Deliver 1 (from the return)
The test checks the all-or-nothing policy of the delivered quantity
This quantity should be 1.0 after the last delivery
"""
compo, kit = self.env['product.product'].create([{
'name': n,
'type': 'consu',
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': self.supplier.id})],
} for n in ['Compo', 'Kit']])
self.env['mrp.bom'].create({
'product_tmpl_id': kit.product_tmpl_id.id,
'product_qty': 1,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': compo.id, 'product_qty': 4}),
],
})
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': kit.name, 'product_id': kit.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
sale_order._get_purchase_orders().button_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 0/4")
picking01 = sale_order.picking_ids
picking01.move_ids.quantity = 2
picking01.move_ids.picked = True
Form.from_action(self.env, picking01.button_validate()).save().process()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 2/4")
# Create a return of picking01 (with both components)
return_form = Form(self.env['stock.return.picking'].with_context(active_id=picking01.id, active_model='stock.picking'))
wizard = return_form.save()
wizard.product_return_moves.write({'quantity': 2.0})
res = wizard.action_create_returns()
return01 = self.env['stock.picking'].browse(res['res_id'])
return01.move_ids.quantity = 2
return01.move_ids.picked = True
return01.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 0/4")
picking02 = picking01.backorder_ids
picking02.move_ids.quantity = 1
picking02.move_ids.picked = True
Form.from_action(self.env, picking02.button_validate()).save().process()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 1/4")
picking03 = picking02.backorder_ids
picking03.move_ids.quantity = 1
picking03.move_ids.picked = True
picking03.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 2/4")
# Create a return of return01 (with 1 component)
return_form = Form(self.env['stock.return.picking'].with_context(active_id=return01.id, active_model='stock.picking'))
wizard = return_form.save()
wizard.product_return_moves.write({'quantity': 1.0})
res = wizard.action_create_returns()
picking04 = self.env['stock.picking'].browse(res['res_id'])
picking04.move_ids.quantity = 1
picking04.move_ids.picked = True
picking04.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0, "Delivered components: 3/4")
# Create a second return of return01 (with 1 component, the last one)
return_form = Form(self.env['stock.return.picking'].with_context(active_id=return01.id, active_model='stock.picking'))
wizard = return_form.save()
wizard.product_return_moves.write({'quantity': 1.0})
res = wizard.action_create_returns()
picking04 = self.env['stock.picking'].browse(res['res_id'])
picking04.move_ids.quantity = 1
picking04.move_ids.picked = True
picking04.button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 1, "Delivered components: 4/4")
def test_cancelled_picking_and_delivered_qty(self):
"""
The delivered quantity should be zero if all SM are cancelled
"""
compo, kit = self.env['product.product'].create([{
'name': n,
'type': 'consu',
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': self.supplier.id})],
} for n in ['Compo', 'Kit']])
self.env['mrp.bom'].create({
'product_tmpl_id': kit.product_tmpl_id.id,
'product_qty': 1,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': compo.id, 'product_qty': 1}),
],
})
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': kit.name, 'product_id': kit.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
self.env['purchase.order'].search([], order='id desc', limit=1).button_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0)
sale_order.picking_ids.action_cancel()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0)
def test_sale_kit_with_dropshipped_component(self):
"""
The test checks the delivered quantity of a kit when one of the
components is dropshipped
"""
compo01, compo02, kit = self.env['product.product'].create([{
'name': n,
'type': 'consu',
} for n in ['compo01', 'compo02', 'super kit']])
compo02.write({
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': self.supplier.id})],
})
self.env['mrp.bom'].create({
'product_tmpl_id': kit.product_tmpl_id.id,
'product_qty': 1,
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': compo01.id, 'product_qty': 1}),
(0, 0, {'product_id': compo02.id, 'product_qty': 1}),
],
})
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': kit.name, 'product_id': kit.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
self.env['purchase.order'].search([], order='id desc', limit=1).button_confirm()
sale_order.picking_ids.move_ids.quantity = 1
sale_order.picking_ids.move_ids.picked = True
sale_order.picking_ids[0].button_validate()
sale_order.picking_ids[1].button_validate()
self.assertEqual(sale_order.order_line.qty_delivered, 1.0)
def test_kit_dropshipped_change_qty_SO(self):
# Create BoM
product_a, product_b, final_product = self.env['product.product'].create([{
'name': p_name,
'type': 'consu',
'is_storable': True,
'seller_ids': [(0, 0, {
'partner_id': self.supplier.id,
})],
} for p_name in ['Comp 1', 'Comp 2', 'Final Product']])
product_a.route_ids = self.env.ref('stock_dropshipping.route_drop_shipping')
product_b.route_ids = self.env.ref('stock_dropshipping.route_drop_shipping')
self.env['mrp.bom'].create({
'product_id': final_product.id,
'product_tmpl_id': final_product.product_tmpl_id.id,
'product_qty': 1,
'consumption': 'flexible',
'type': 'phantom',
'bom_line_ids': [
(0, 0, {'product_id': product_a.id, 'product_qty': 1}),
(0, 0, {'product_id': product_b.id, 'product_qty': 1}),
]
})
# Create sale order
partner = self.env['res.partner'].create({'name': 'Testing Man'})
so = self.env['sale.order'].create({
'partner_id': partner.id,
})
sol = self.env['sale.order.line'].create({
'name': "Order line",
'product_id': final_product.id,
'order_id': so.id,
'product_uom_qty': 25,
})
so.action_confirm()
user_admin = self.env['res.users'].search([('login', '=', 'admin')])
sol.with_user(user_admin).write({'product_uom_qty': 10})
self.assertEqual(sol.purchase_line_ids.mapped('product_uom_qty'), [10, 10])
def test_dropship_standard_perpetual_anglosaxon_ordered_return_internal_aml(self):
"""
test that, with sbc installed, the return to an internal location of a dropshipped move
(perpetual and anglosaxon) creates the correct account move (debitting stock valuation)
"""
grp_multi_loc = self.env.ref('stock.group_stock_multi_locations')
self.env.user.write({'groups_id': [(4, grp_multi_loc.id)]})
self.env.company.anglo_saxon_accounting = True
product = self.env['product.product'].create({
'name': 'product',
'type': 'consu',
'is_storable': True,
'standard_price': 10,
'route_ids': [(6, 0, [self.dropship_route.id])],
'seller_ids': [(0, 0, {'partner_id': self.supplier.id})],
})
product.product_tmpl_id.categ_id = self.env.ref('product.product_category_all')
self._setup_category_stock_journals()
product.product_tmpl_id.categ_id.property_cost_method = 'standard'
product.product_tmpl_id.categ_id.property_valuation = 'real_time'
product.product_tmpl_id.invoice_policy = 'order'
sale_order = self.env['sale.order'].create({
'partner_id': self.customer.id,
'picking_policy': 'direct',
'order_line': [
(0, 0, {'name': product.name, 'product_id': product.id, 'product_uom_qty': 1}),
],
})
sale_order.action_confirm()
self.env['purchase.order'].search([], order='id desc', limit=1).button_confirm()
self.assertEqual(sale_order.order_line.qty_delivered, 0.0)
picking = sale_order.picking_ids
picking.button_validate()
stock_return_picking_form = Form(self.env['stock.return.picking']
.with_context(active_ids=sale_order.picking_ids.ids, active_id=sale_order.picking_ids.ids[0],
active_model='stock.picking'))
stock_return_picking = stock_return_picking_form.save()
stock_return_picking.product_return_moves.write({'quantity': 1.0})
stock_return_picking_action = stock_return_picking.action_create_returns()
return_pick = self.env['stock.picking'].browse(stock_return_picking_action['res_id'])
return_pick.location_dest_id = self.subcontractor_partner1.property_stock_subcontractor
return_pick.move_ids[0].move_line_ids[0].quantity = 1.0
return_pick.move_ids[0].picked = True
return_pick._action_done()
self.assertEqual(return_pick.move_ids._is_dropshipped_returned(), True)
stock_valuation_account = product.categ_id.property_stock_valuation_account_id
stock_interim_delivered = product.categ_id.property_stock_account_output_categ_id
stock_interim_received = product.categ_id.property_stock_account_input_categ_id
original_move_in_svl_amls = picking.move_ids.stock_valuation_layer_ids.filtered(lambda svl: svl.value >= 0).account_move_id.line_ids.sorted('debit')
original_move_out_svl_amls = picking.move_ids.stock_valuation_layer_ids.filtered(lambda svl: svl.value < 0).account_move_id.line_ids.sorted('debit')
return_move_amls = return_pick.move_ids.stock_valuation_layer_ids.account_move_id.line_ids.sorted('debit')
self.assertRecordValues(original_move_in_svl_amls, [
{'credit': 10, 'debit': 0, 'account_id': stock_interim_received.id},
{'credit': 0, 'debit': 10, 'account_id': stock_valuation_account.id},
])
self.assertRecordValues(original_move_out_svl_amls, [
{'credit': 10, 'debit': 0, 'account_id': stock_valuation_account.id},
{'credit': 0, 'debit': 10, 'account_id': stock_interim_delivered.id},
])
# it's a return to an internal location so it should even out the amls of the outgoing svl of the original move
self.assertRecordValues(return_move_amls, [
{'credit': 10, 'debit': 0, 'account_id': stock_interim_delivered.id},
{'credit': 0, 'debit': 10, 'account_id': stock_valuation_account.id},
])