mirror of
https://github.com/bringout/oca-ocb-sale.git
synced 2026-04-27 06:12:06 +02:00
Initial commit: Sale packages
This commit is contained in:
commit
14e3d26998
6469 changed files with 2479670 additions and 0 deletions
|
|
@ -0,0 +1,5 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
from . import test_pos_sale_flow
|
||||
from . import test_pos_sale_report
|
||||
|
|
@ -0,0 +1,663 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
import odoo
|
||||
|
||||
from odoo.addons.point_of_sale.tests.test_frontend import TestPointOfSaleHttpCommon
|
||||
from odoo.tests.common import Form
|
||||
from odoo import fields
|
||||
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class TestPoSSale(TestPointOfSaleHttpCommon):
|
||||
def test_settle_order_with_kit(self):
|
||||
if not self.env["ir.module.module"].search([("name", "=", "mrp"), ("state", "=", "installed")]):
|
||||
self.skipTest("mrp module is required for this test")
|
||||
|
||||
self.kit = self.env['product.product'].create({
|
||||
'name': 'Pizza Chicken',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
})
|
||||
|
||||
self.component_a = self.env['product.product'].create({
|
||||
'name': 'Chicken',
|
||||
'type': 'product',
|
||||
'available_in_pos': True,
|
||||
'uom_id': self.env.ref('uom.product_uom_gram').id,
|
||||
'uom_po_id': self.env.ref('uom.product_uom_gram').id,
|
||||
'lst_price': 10.0,
|
||||
})
|
||||
self.location = self.env['stock.location'].create({
|
||||
'name': 'Test location',
|
||||
'usage': 'internal',
|
||||
})
|
||||
|
||||
self.env['stock.quant']._update_available_quantity(self.component_a, self.location, 100000)
|
||||
|
||||
bom_product_form = Form(self.env['mrp.bom'])
|
||||
bom_product_form.product_id = self.kit
|
||||
bom_product_form.product_tmpl_id = self.kit.product_tmpl_id
|
||||
bom_product_form.product_qty = 1.0
|
||||
bom_product_form.type = 'phantom'
|
||||
with bom_product_form.bom_line_ids.new() as bom_line:
|
||||
bom_line.product_id = self.component_a
|
||||
bom_line.product_qty = 300.0
|
||||
self.bom_a = bom_product_form.save()
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.kit.id,
|
||||
'name': self.kit.name,
|
||||
'product_uom_qty': 10,
|
||||
'product_uom': self.kit.uom_id.id,
|
||||
'price_unit': self.kit.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
picking = sale_order.picking_ids
|
||||
picking.move_ids.quantity_done = 300
|
||||
action = picking.button_validate()
|
||||
wizard = Form(self.env[action['res_model']].with_context(action['context']))
|
||||
wizard.save().process()
|
||||
|
||||
self.assertEqual(sale_order.order_line.qty_delivered, 1)
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrder', login="accountman")
|
||||
|
||||
#assert that sales order qty are correctly updated
|
||||
self.assertEqual(sale_order.order_line.qty_delivered, 3)
|
||||
self.assertEqual(sale_order.picking_ids[0].move_ids.product_qty, 2100) # 7 left to deliver => 300 * 7 = 2100
|
||||
self.assertEqual(sale_order.picking_ids[0].move_ids.quantity_done, 0)
|
||||
self.assertEqual(sale_order.picking_ids[1].move_ids.product_qty, 300)
|
||||
self.assertEqual(sale_order.picking_ids[1].move_ids.quantity_done, 300) # 1 delivered => 300 * 2 = 600
|
||||
|
||||
def test_settle_order_with_incompatible_partner(self):
|
||||
""" If the partner of the sale order is not compatible with the current pos order,
|
||||
then a new pos order should be to settle the newly selected sale order.
|
||||
"""
|
||||
|
||||
product1 = self.env['product.product'].create({
|
||||
'name': 'product1',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10,
|
||||
'taxes_id': [odoo.Command.clear()],
|
||||
})
|
||||
product2 = self.env['product.product'].create({
|
||||
'name': 'product2',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 11,
|
||||
'taxes_id': [odoo.Command.clear()],
|
||||
})
|
||||
partner_1 = self.env['res.partner'].create({'name': 'Test Partner 1'})
|
||||
partner_2 = self.env['res.partner'].create({'name': 'Test Partner 2'})
|
||||
self.env['sale.order'].create({
|
||||
'partner_id': partner_1.id,
|
||||
'partner_shipping_id': partner_2.id,
|
||||
'order_line': [(0, 0, {'product_id': product1.id})],
|
||||
})
|
||||
self.env['sale.order'].create({
|
||||
'partner_id': partner_1.id,
|
||||
'partner_shipping_id': partner_1.id,
|
||||
'order_line': [(0, 0, {'product_id': product2.id})],
|
||||
})
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrderIncompatiblePartner', login="accountman")
|
||||
|
||||
def test_settle_order_with_different_product(self):
|
||||
"""This test create an order and settle it in the PoS. But only one of the product is delivered.
|
||||
And we need to make sure the quantity are correctly updated on the sale order.
|
||||
"""
|
||||
#create 2 products
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
'default_code': 'A001',
|
||||
})
|
||||
product_b = self.env['product.product'].create({
|
||||
'name': 'Product B',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
'default_code': 'A002',
|
||||
})
|
||||
#create a sale order with 2 lines
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
}), (0, 0, {
|
||||
'product_id': product_b.id,
|
||||
'name': product_b.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_b.uom_id.id,
|
||||
'price_unit': product_b.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 0)
|
||||
self.assertEqual(sale_order.order_line[1].qty_delivered, 0)
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrder2', login="accountman")
|
||||
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 1)
|
||||
self.assertEqual(sale_order.order_line[1].qty_delivered, 0)
|
||||
orderline_product_a = sale_order.order_line.filtered(lambda l: l.product_id.id == product_a.id)
|
||||
orderline_product_b = sale_order.order_line.filtered(lambda l: l.product_id.id == product_b.id)
|
||||
# nothing to deliver for product a because already handled in pos.
|
||||
self.assertEqual(orderline_product_a.move_ids.product_uom_qty, 0)
|
||||
# 1 item to deliver for product b.
|
||||
self.assertEqual(orderline_product_b.move_ids.product_uom_qty, 1)
|
||||
|
||||
def test_settle_order_unreserve_order_lines(self):
|
||||
#create a product category that use the closest location for the removal strategy
|
||||
self.removal_strategy = self.env['product.removal'].search([('method', '=', 'closest')], limit=1)
|
||||
self.product_category = self.env['product.category'].create({
|
||||
'name': 'Product Category',
|
||||
'removal_strategy_id': self.removal_strategy.id,
|
||||
})
|
||||
|
||||
self.product = self.env['product.product'].create({
|
||||
'name': 'Product',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
'taxes_id': False,
|
||||
'categ_id': self.product_category.id,
|
||||
})
|
||||
|
||||
#create 2 stock location Shelf 1 and Shelf 2
|
||||
self.warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1)
|
||||
self.shelf_1 = self.env['stock.location'].create({
|
||||
'name': 'Shelf 1',
|
||||
'usage': 'internal',
|
||||
'location_id': self.warehouse.lot_stock_id.id,
|
||||
})
|
||||
self.shelf_2 = self.env['stock.location'].create({
|
||||
'name': 'Shelf 2',
|
||||
'usage': 'internal',
|
||||
'location_id': self.warehouse.lot_stock_id.id,
|
||||
})
|
||||
|
||||
quants = self.env['stock.quant'].with_context(inventory_mode=True).create({
|
||||
'product_id': self.product.id,
|
||||
'inventory_quantity': 2,
|
||||
'location_id': self.shelf_1.id,
|
||||
})
|
||||
quants |= self.env['stock.quant'].with_context(inventory_mode=True).create({
|
||||
'product_id': self.product.id,
|
||||
'inventory_quantity': 5,
|
||||
'location_id': self.shelf_2.id,
|
||||
})
|
||||
quants.action_apply_inventory()
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.product.id,
|
||||
'name': self.product.name,
|
||||
'product_uom_qty': 4,
|
||||
'price_unit': self.product.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.assertEqual(sale_order.order_line.move_ids.move_line_ids[0].reserved_qty, 2)
|
||||
self.assertEqual(sale_order.order_line.move_ids.move_line_ids[0].location_id.id, self.shelf_1.id)
|
||||
self.assertEqual(sale_order.order_line.move_ids.move_line_ids[1].reserved_qty, 2)
|
||||
self.assertEqual(sale_order.order_line.move_ids.move_line_ids[1].location_id.id, self.shelf_2.id)
|
||||
|
||||
self.config = self.env['res.config.settings'].create({
|
||||
'update_stock_quantities': 'real',
|
||||
})
|
||||
self.config.execute()
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrderRealTime', login="accountman")
|
||||
|
||||
pos_order = self.env['pos.order'].search([], order='id desc', limit=1)
|
||||
self.assertEqual(pos_order.picking_ids.move_line_ids[0].qty_done, 2)
|
||||
self.assertEqual(pos_order.picking_ids.move_line_ids[0].location_id.id, self.shelf_1.id)
|
||||
self.assertEqual(pos_order.picking_ids.move_line_ids[1].qty_done, 2)
|
||||
self.assertEqual(pos_order.picking_ids.move_line_ids[1].location_id.id, self.shelf_2.id)
|
||||
self.assertEqual(sale_order.order_line.move_ids.move_lines_count, 0)
|
||||
|
||||
def test_downpayment_refund(self):
|
||||
#create a sale order
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.product_a.id,
|
||||
'name': self.product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'price_unit': 100,
|
||||
'product_uom': self.product_a.uom_id.id
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
#set downpayment product in pos config
|
||||
self.downpayment_product = self.env['product.product'].create({
|
||||
'name': 'Down Payment',
|
||||
'available_in_pos': True,
|
||||
'type': 'service',
|
||||
})
|
||||
self.main_pos_config.write({
|
||||
'down_payment_product_id': self.downpayment_product.id,
|
||||
})
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosRefundDownpayment', login="accountman")
|
||||
self.assertEqual(len(sale_order.order_line), 3)
|
||||
self.assertEqual(sale_order.order_line[1].qty_invoiced, 1)
|
||||
self.assertEqual(sale_order.order_line[2].qty_invoiced, -1)
|
||||
|
||||
def test_settle_order_with_multistep_delivery(self):
|
||||
"""This test create an order and settle it in the PoS. It also uses multistep delivery
|
||||
and we need to make sure that all the picking are cancelled if the order is fully delivered.
|
||||
"""
|
||||
|
||||
#get the warehouse
|
||||
warehouse = self.env['stock.warehouse'].search([('company_id', '=', self.env.company.id)], limit=1)
|
||||
warehouse.delivery_steps = 'pick_pack_ship'
|
||||
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
})
|
||||
self.env['stock.quant']._update_available_quantity(product_a, warehouse.lot_stock_id, 1)
|
||||
|
||||
#create a sale order with 2 lines
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 0)
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrder3', login="accountman")
|
||||
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 1)
|
||||
self.assertEqual(sale_order.picking_ids.mapped('state'), ['cancel', 'cancel', 'cancel'])
|
||||
|
||||
def test_pos_not_groupable_product(self):
|
||||
#Create a UoM Category that is not pos_groupable
|
||||
uom_category = self.env['uom.category'].create({
|
||||
'name': 'Test',
|
||||
'is_pos_groupable': False,
|
||||
})
|
||||
uom = self.env['uom.uom'].create({
|
||||
'name': 'Test',
|
||||
'category_id': uom_category.id,
|
||||
'uom_type': 'reference',
|
||||
'rounding': 0.01
|
||||
})
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
'uom_id': uom.id,
|
||||
'uom_po_id': uom.id,
|
||||
})
|
||||
#create a sale order with product_a
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 3.5,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': 8, # manually set a different price than the lst_price
|
||||
'discount': 10,
|
||||
})],
|
||||
})
|
||||
self.assertEqual(sale_order.amount_total, 28.98) # 3.5 * 8 * 1.15 * 90%
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrderNotGroupable', login="accountman")
|
||||
|
||||
def test_customer_notes(self):
|
||||
"""This test create an order and settle it in the PoS. It also uses multistep delivery
|
||||
and we need to make sure that all the picking are cancelled if the order is fully delivered.
|
||||
"""
|
||||
|
||||
#create a sale order with 2 customer notes
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'note': 'Customer note 1',
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.whiteboard_pen.id,
|
||||
'name': self.whiteboard_pen.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': self.whiteboard_pen.uom_id.id,
|
||||
'price_unit': self.whiteboard_pen.lst_price,
|
||||
}), (0, 0, {
|
||||
'name': 'Customer note 2',
|
||||
'display_type': 'line_note',
|
||||
}), (0, 0, {
|
||||
'name': 'Customer note 3',
|
||||
'display_type': 'line_note',
|
||||
})],
|
||||
})
|
||||
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrderWithNote', login="accountman")
|
||||
|
||||
def test_untaxed_invoiced_amount(self):
|
||||
"""Make sure that orders invoiced in the pos gets their untaxed invoiced
|
||||
amount updated accordingly"""
|
||||
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
'taxes_id': [],
|
||||
})
|
||||
|
||||
product_b = self.env['product.product'].create({
|
||||
'name': 'Product B',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 5.0,
|
||||
'taxes_id': [],
|
||||
})
|
||||
|
||||
partner_test = self.env['res.partner'].create({'name': 'Test Partner'})
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_test.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
}), (0, 0, {
|
||||
'product_id': product_b.id,
|
||||
'name': product_b.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_b.uom_id.id,
|
||||
'price_unit': product_b.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
self.main_pos_config.open_ui()
|
||||
current_session = self.main_pos_config.current_session_id
|
||||
|
||||
pos_order = {'data':
|
||||
{'amount_paid': 10,
|
||||
'amount_return': 0,
|
||||
'amount_tax': 0,
|
||||
'amount_total': 10,
|
||||
'creation_date': fields.Datetime.to_string(fields.Datetime.now()),
|
||||
'fiscal_position_id': False,
|
||||
'to_invoice': True,
|
||||
'partner_id': partner_test.id,
|
||||
'pricelist_id': self.main_pos_config.available_pricelist_ids[0].id,
|
||||
'lines': [[0,
|
||||
0,
|
||||
{'discount': 0,
|
||||
'pack_lot_ids': [],
|
||||
'price_unit': 10,
|
||||
'product_id': product_a.id,
|
||||
'price_subtotal': 10,
|
||||
'price_subtotal_incl': 10,
|
||||
'sale_order_line_id': sale_order.order_line[0],
|
||||
'sale_order_origin_id': sale_order,
|
||||
'qty': 1,
|
||||
'tax_ids': []}]],
|
||||
'name': 'Order 00044-003-0014',
|
||||
'pos_session_id': current_session.id,
|
||||
'sequence_number': self.main_pos_config.journal_id.id,
|
||||
'statement_ids': [[0,
|
||||
0,
|
||||
{'amount': 10,
|
||||
'name': fields.Datetime.now(),
|
||||
'payment_method_id': self.main_pos_config.payment_method_ids[0].id}]],
|
||||
'uid': '00044-003-0014',
|
||||
'user_id': self.env.uid},
|
||||
}
|
||||
|
||||
self.env['pos.order'].create_from_ui([pos_order])
|
||||
self.assertEqual(sale_order.order_line[0].untaxed_amount_invoiced, 10, "Untaxed invoiced amount should be 10")
|
||||
self.assertEqual(sale_order.order_line[1].untaxed_amount_invoiced, 0, "Untaxed invoiced amount should be 0")
|
||||
|
||||
def test_order_does_not_remain_in_list(self):
|
||||
"""Verify that a paid order doesn't remain in the orders list"""
|
||||
|
||||
# Create a sale order
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.whiteboard_pen.id,
|
||||
'name': self.whiteboard_pen.name,
|
||||
'product_uom_qty': 1,
|
||||
'price_unit': 100,
|
||||
'product_uom': self.whiteboard_pen.uom_id.id
|
||||
})],
|
||||
})
|
||||
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosOrderDoesNotRemainInList', login="accountman")
|
||||
|
||||
def test_settle_order_change_customer(self):
|
||||
"""
|
||||
When settling an order, the price set on the sol shouldn't reset to
|
||||
the sale price of the product when changing customer.
|
||||
"""
|
||||
self.product_a.lst_price = 150
|
||||
self.product_a.taxes_id = None
|
||||
self.product_a.available_in_pos = True
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': self.env['res.partner'].create({'name': 'Test Partner'}).id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': self.product_a.id,
|
||||
'name': self.product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'price_unit': 100,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleCustomPrice', login="accountman")
|
||||
|
||||
def test_settle_draft_order_service_product(self):
|
||||
"""
|
||||
Checks that, when settling a draft order (quotation), the quantity set on the corresponding
|
||||
PoS order, for service products, is set correctly.
|
||||
"""
|
||||
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Test service product',
|
||||
'available_in_pos': True,
|
||||
'type': 'service',
|
||||
'invoice_policy': 'order',
|
||||
'lst_price': 50.0,
|
||||
'taxes_id': [],
|
||||
})
|
||||
|
||||
partner_test = self.env['res.partner'].create({'name': 'Test Partner'})
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_test.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
})],
|
||||
})
|
||||
|
||||
self.assertEqual(sale_order.state, 'draft')
|
||||
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleDraftOrder', login="accountman")
|
||||
|
||||
def test_settle_order_ship_later_delivered_qty(self):
|
||||
"""This test create an order, settle it in the PoS and ship it later.
|
||||
We need to make sure that the quantity delivered on the original sale is updated correctly
|
||||
"""
|
||||
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 10.0,
|
||||
})
|
||||
|
||||
partner_test = self.env['res.partner'].create({
|
||||
'name': 'Test Partner',
|
||||
'city': 'San Francisco',
|
||||
'state_id': self.env.ref('base.state_us_5').id,
|
||||
'country_id': self.env.ref('base.us').id,
|
||||
'zip': '94134',
|
||||
'street': 'Rue du burger',
|
||||
})
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_test.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 0)
|
||||
|
||||
self.main_pos_config.ship_later = True
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PosSettleOrderShipLater', login="accountman")
|
||||
|
||||
# The pos order is being shipped later so the qty_delivered should still be 0
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 0)
|
||||
|
||||
# We validate the delivery of the order, now the qty_delivered should be 1
|
||||
pickings = sale_order.pos_order_line_ids.order_id.picking_ids
|
||||
pickings.move_ids.quantity_done = 1
|
||||
pickings.button_validate()
|
||||
self.assertEqual(sale_order.order_line[0].qty_delivered, 1)
|
||||
|
||||
def test_downpayment_amount_to_invoice(self):
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 100.0,
|
||||
'taxes_id': [],
|
||||
})
|
||||
partner_test = self.env['res.partner'].create({'name': 'Test Partner'})
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_test.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
self.downpayment_product = self.env['product.product'].create({
|
||||
'name': 'Down Payment',
|
||||
'available_in_pos': True,
|
||||
'type': 'service',
|
||||
'taxes_id': [],
|
||||
})
|
||||
self.main_pos_config.write({
|
||||
'down_payment_product_id': self.downpayment_product.id,
|
||||
})
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PoSDownPaymentAmount', login="accountman")
|
||||
|
||||
self.assertEqual(sale_order.order_line[1].price_unit, 10)
|
||||
|
||||
# Update delivered quantity of SO line
|
||||
sale_order.order_line[0].write({'qty_delivered': 1.0})
|
||||
context = {
|
||||
'active_model': 'sale.order',
|
||||
'active_ids': [sale_order.id],
|
||||
'active_id': sale_order.id,
|
||||
'default_journal_id': self.company_data['default_journal_sale'].id,
|
||||
}
|
||||
|
||||
# Let's do the invoice for the remaining amount
|
||||
payment = self.env['sale.advance.payment.inv'].with_context(context).create({
|
||||
'deposit_account_id': self.company_data['default_account_revenue'].id
|
||||
})
|
||||
payment.create_invoices()
|
||||
|
||||
# Confirm all invoices
|
||||
sale_order.invoice_ids.action_post()
|
||||
self.assertEqual(sale_order.order_line[1].price_unit, 10)
|
||||
|
||||
def test_downpayment_line_name(self):
|
||||
product_a = self.env['product.product'].create({
|
||||
'name': 'Product A',
|
||||
'available_in_pos': True,
|
||||
'type': 'product',
|
||||
'lst_price': 100.0,
|
||||
'taxes_id': [],
|
||||
})
|
||||
partner_test = self.env['res.partner'].create({'name': 'Test Partner'})
|
||||
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_test.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_a.id,
|
||||
'name': product_a.name,
|
||||
'product_uom_qty': 1,
|
||||
'product_uom': product_a.uom_id.id,
|
||||
'price_unit': product_a.lst_price,
|
||||
})],
|
||||
})
|
||||
sale_order.action_confirm()
|
||||
|
||||
downpayment_product = self.env['product.product'].create({
|
||||
'name': 'Down Payment',
|
||||
'available_in_pos': True,
|
||||
'type': 'service',
|
||||
'taxes_id': [],
|
||||
})
|
||||
self.main_pos_config.write({
|
||||
'down_payment_product_id': downpayment_product.id,
|
||||
})
|
||||
self.main_pos_config.open_ui()
|
||||
self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PoSDownPaymentAmount', login="accountman")
|
||||
|
||||
downpayment_line_pos = sale_order.order_line.filtered('is_downpayment')
|
||||
self.assertTrue(downpayment_line_pos)
|
||||
self.assertNotIn('(draft)', downpayment_line_pos.name.lower())
|
||||
self.assertNotIn('(canceled)', downpayment_line_pos.name.lower())
|
||||
|
|
@ -0,0 +1,107 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
import odoo
|
||||
|
||||
from odoo.addons.point_of_sale.tests.common import TestPoSCommon
|
||||
|
||||
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class TestPoSSaleReport(TestPoSCommon):
|
||||
|
||||
def setUp(self):
|
||||
super(TestPoSSaleReport, self).setUp()
|
||||
self.config = self.basic_config
|
||||
self.product0 = self.create_product('Product 0', self.categ_basic, 0.0, 0.0)
|
||||
# Ensure that adding a uom to the product with a factor != 1
|
||||
# does not cause an error in weight and volume calculation
|
||||
self.product0.uom_id = self.env['uom.uom'].search([('name', '=', 'Dozens')], limit=1)
|
||||
|
||||
def test_weight_and_volume(self):
|
||||
self.product0.product_tmpl_id.weight = 3
|
||||
self.product0.product_tmpl_id.volume = 4
|
||||
|
||||
self.open_new_session()
|
||||
session = self.pos_session
|
||||
orders = []
|
||||
|
||||
# Process two orders
|
||||
orders.append(self.create_ui_order_data([(self.product0, 3)]))
|
||||
orders.append(self.create_ui_order_data([(self.product0, 1)]))
|
||||
self.env['pos.order'].create_from_ui(orders)
|
||||
# Duplicate the first line of the first order
|
||||
session.order_ids[0].lines.copy()
|
||||
|
||||
session.action_pos_session_closing_control()
|
||||
|
||||
# PoS Orders have negative IDs to avoid conflict, so reports[0] will correspond to the newest order
|
||||
reports = self.env['sale.report'].sudo().search([('product_id', '=', self.product0.id)], order='id', limit=2)
|
||||
self.assertEqual(reports[0].weight, 3)
|
||||
self.assertEqual(reports[0].volume, 4)
|
||||
self.assertEqual(reports[1].weight, 18)
|
||||
self.assertEqual(reports[1].volume, 24)
|
||||
|
||||
def test_weight_and_volume_product_variant(self):
|
||||
colors = ['red', 'blue']
|
||||
prod_attr = self.env['product.attribute'].create({'name': 'Color', 'create_variant': 'dynamic'})
|
||||
prod_attr_values = self.env['product.attribute.value'].create([{'name': color, 'attribute_id': prod_attr.id, 'sequence': 1} for color in colors])
|
||||
|
||||
uom_unit = self.env.ref('uom.product_uom_unit')
|
||||
product_template = self.env['product.template'].create({
|
||||
'name': 'Sofa',
|
||||
'uom_id': uom_unit.id,
|
||||
'uom_po_id': uom_unit.id,
|
||||
'attribute_line_ids': [(0, 0, {
|
||||
'attribute_id': prod_attr.id,
|
||||
'value_ids': [(6, 0, prod_attr_values.ids)]
|
||||
})]
|
||||
})
|
||||
prod_tmpl_attrs = self.env['product.template.attribute.value'].search([
|
||||
('attribute_line_id', '=', product_template.attribute_line_ids.id),
|
||||
('product_attribute_value_id', 'in', prod_attr_values.ids)
|
||||
])
|
||||
|
||||
product_1 = product_template._create_product_variant(prod_tmpl_attrs[0])
|
||||
product_1.weight = 1
|
||||
product_1.volume = 1
|
||||
|
||||
product_2 = product_template._create_product_variant(prod_tmpl_attrs[1])
|
||||
product_2.weight = 2
|
||||
product_2.volume = 2
|
||||
|
||||
self.open_new_session()
|
||||
session = self.pos_session
|
||||
|
||||
order = self.create_ui_order_data([(product_1, 3), (product_2, 3)])
|
||||
self.env['pos.order'].create_from_ui([order])
|
||||
|
||||
session.action_pos_session_closing_control()
|
||||
|
||||
report = self.env['sale.report'].sudo().search([('product_id', '=', product_1.id)], order='id', limit=1)
|
||||
self.assertEqual(report.weight, 3)
|
||||
self.assertEqual(report.weight, 3)
|
||||
report = self.env['sale.report'].sudo().search([('product_id', '=', product_2.id)], order='id', limit=1)
|
||||
self.assertEqual(report.weight, 6)
|
||||
self.assertEqual(report.weight, 6)
|
||||
|
||||
def test_different_shipping_address(self):
|
||||
product_0 = self.create_product('Product 0', self.categ_basic, 0.0, 0.0)
|
||||
partner_1 = self.env['res.partner'].create({'name': 'Test Partner 1'})
|
||||
partner_2 = self.env['res.partner'].create({'name': 'Test Partner 2'})
|
||||
sale_order = self.env['sale.order'].create({
|
||||
'partner_id': partner_1.id,
|
||||
'partner_shipping_id': partner_2.id,
|
||||
'order_line': [(0, 0, {
|
||||
'product_id': product_0.id,
|
||||
})],
|
||||
})
|
||||
self.open_new_session()
|
||||
|
||||
data = self.create_ui_order_data([(product_0, 1)], partner_1, True)
|
||||
data['data']['lines'][0][2]['sale_order_origin_id'] = sale_order.read()[0]
|
||||
data['data']['lines'][0][2]['sale_order_line_id'] = sale_order.order_line[0].read()[0]
|
||||
order_ids = self.env['pos.order'].create_from_ui([data])
|
||||
|
||||
move_id = self.env['account.move'].browse(order_ids[0]['account_move'])
|
||||
self.assertEqual(move_id.partner_id.id, partner_1.id)
|
||||
self.assertEqual(move_id.partner_shipping_id.id, partner_2.id)
|
||||
Loading…
Add table
Add a link
Reference in a new issue