summaryrefslogtreecommitdiff
path: root/addons/point_of_sale/tests/test_frontend.py
diff options
context:
space:
mode:
authorstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
committerstephanchrst <stephanchrst@gmail.com>2022-05-10 21:51:50 +0700
commit3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch)
treea44932296ef4a9b71d5f010906253d8c53727726 /addons/point_of_sale/tests/test_frontend.py
parent0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff)
initial commit 2
Diffstat (limited to 'addons/point_of_sale/tests/test_frontend.py')
-rw-r--r--addons/point_of_sale/tests/test_frontend.py525
1 files changed, 525 insertions, 0 deletions
diff --git a/addons/point_of_sale/tests/test_frontend.py b/addons/point_of_sale/tests/test_frontend.py
new file mode 100644
index 00000000..40e31e1c
--- /dev/null
+++ b/addons/point_of_sale/tests/test_frontend.py
@@ -0,0 +1,525 @@
+# -*- coding: utf-8 -*-
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+
+from odoo import tools
+from odoo.api import Environment
+from odoo.tools import DEFAULT_SERVER_DATE_FORMAT
+from datetime import date, timedelta
+
+import odoo.tests
+
+
+class TestPointOfSaleHttpCommon(odoo.tests.HttpCase):
+
+ def setUp(self):
+ super().setUp()
+ env = self.env(user=self.env.ref('base.user_admin'))
+
+ journal_obj = env['account.journal']
+ account_obj = env['account.account']
+ main_company = env.ref('base.main_company')
+ self.main_pos_config = env.ref('point_of_sale.pos_config_main')
+
+ env['res.partner'].create({
+ 'name': 'Deco Addict',
+ })
+
+ account_receivable = account_obj.create({'code': 'X1012',
+ 'name': 'Account Receivable - Test',
+ 'user_type_id': env.ref('account.data_account_type_receivable').id,
+ 'reconcile': True})
+ self.env.company.account_default_pos_receivable_account_id = account_receivable
+
+ self.env['ir.property']._set_default('property_account_receivable_id', 'res.partner', account_receivable, main_company)
+
+ cash_journal = journal_obj.create({
+ 'name': 'Cash Test',
+ 'type': 'cash',
+ 'company_id': main_company.id,
+ 'code': 'CSH',
+ 'sequence': 10,
+ })
+
+ # Archive all existing product to avoid noise during the tours
+ all_pos_product = self.env['product.product'].search([('available_in_pos', '=', True)])
+ discount = self.env.ref('point_of_sale.product_product_consumable')
+ self.tip = self.env.ref('point_of_sale.product_product_tip')
+ (all_pos_product - discount - self.tip)._write({'active': False})
+
+ # In DESKS categ: Desk Pad
+ pos_categ_desks = env.ref('point_of_sale.pos_category_desks')
+
+ # In DESKS categ: Whiteboard Pen
+ pos_categ_misc = env.ref('point_of_sale.pos_category_miscellaneous')
+
+ # In CHAIR categ: Letter Tray
+ pos_categ_chairs = env.ref('point_of_sale.pos_category_chairs')
+
+ # test an extra price on an attribute
+ pear = env['product.product'].create({
+ 'name': 'Whiteboard Pen',
+ 'available_in_pos': True,
+ 'list_price': 1.20,
+ 'taxes_id': False,
+ 'weight': 0.01,
+ 'to_weight': True,
+ 'pos_categ_id': pos_categ_misc.id,
+ })
+ wall_shelf = env['product.product'].create({
+ 'name': 'Wall Shelf Unit',
+ 'available_in_pos': True,
+ 'list_price': 1.98,
+ 'taxes_id': False,
+ })
+ small_shelf = env['product.product'].create({
+ 'name': 'Small Shelf',
+ 'available_in_pos': True,
+ 'list_price': 2.83,
+ 'taxes_id': False,
+ })
+ magnetic_board = env['product.product'].create({
+ 'name': 'Magnetic Board',
+ 'available_in_pos': True,
+ 'list_price': 1.98,
+ 'taxes_id': False,
+ })
+ monitor_stand = env['product.product'].create({
+ 'name': 'Monitor Stand',
+ 'available_in_pos': True,
+ 'list_price': 3.19,
+ 'taxes_id': False,
+ })
+ desk_pad = env['product.product'].create({
+ 'name': 'Desk Pad',
+ 'available_in_pos': True,
+ 'list_price': 1.98,
+ 'taxes_id': False,
+ 'pos_categ_id': pos_categ_desks.id,
+ })
+ letter_tray = env['product.product'].create({
+ 'name': 'Letter Tray',
+ 'available_in_pos': True,
+ 'list_price': 4.80,
+ 'taxes_id': False,
+ 'pos_categ_id': pos_categ_chairs.id,
+ })
+ desk_organizer = env['product.product'].create({
+ 'name': 'Desk Organizer',
+ 'available_in_pos': True,
+ 'list_price': 5.10,
+ 'taxes_id': False,
+ })
+ configurable_chair = env['product.product'].create({
+ 'name': 'Configurable Chair',
+ 'available_in_pos': True,
+ 'list_price': 10,
+ 'taxes_id': False,
+ })
+
+ attribute = env['product.attribute'].create({
+ 'name': 'add 2',
+ })
+ attribute_value = env['product.attribute.value'].create({
+ 'name': 'add 2',
+ 'attribute_id': attribute.id,
+ })
+ line = env['product.template.attribute.line'].create({
+ 'product_tmpl_id': pear.product_tmpl_id.id,
+ 'attribute_id': attribute.id,
+ 'value_ids': [(6, 0, attribute_value.ids)]
+ })
+ line.product_template_value_ids[0].price_extra = 2
+
+ chair_color_attribute = env['product.attribute'].create({
+ 'name': 'Color',
+ 'display_type': 'color',
+ 'create_variant': 'no_variant',
+ })
+ chair_color_red = env['product.attribute.value'].create({
+ 'name': 'Red',
+ 'attribute_id': chair_color_attribute.id,
+ 'html_color': '#ff0000',
+ })
+ chair_color_blue = env['product.attribute.value'].create({
+ 'name': 'Blue',
+ 'attribute_id': chair_color_attribute.id,
+ 'html_color': '#0000ff',
+ })
+ chair_color_line = env['product.template.attribute.line'].create({
+ 'product_tmpl_id': configurable_chair.product_tmpl_id.id,
+ 'attribute_id': chair_color_attribute.id,
+ 'value_ids': [(6, 0, [chair_color_red.id, chair_color_blue.id])]
+ })
+ chair_color_line.product_template_value_ids[0].price_extra = 1
+
+ chair_legs_attribute = env['product.attribute'].create({
+ 'name': 'Chair Legs',
+ 'display_type': 'select',
+ 'create_variant': 'no_variant',
+ })
+ chair_legs_metal = env['product.attribute.value'].create({
+ 'name': 'Metal',
+ 'attribute_id': chair_legs_attribute.id,
+ })
+ chair_legs_wood = env['product.attribute.value'].create({
+ 'name': 'Wood',
+ 'attribute_id': chair_legs_attribute.id,
+ })
+ chair_legs_line = env['product.template.attribute.line'].create({
+ 'product_tmpl_id': configurable_chair.product_tmpl_id.id,
+ 'attribute_id': chair_legs_attribute.id,
+ 'value_ids': [(6, 0, [chair_legs_metal.id, chair_legs_wood.id])]
+ })
+
+ chair_fabrics_attribute = env['product.attribute'].create({
+ 'name': 'Fabrics',
+ 'display_type': 'radio',
+ 'create_variant': 'no_variant',
+ })
+ chair_fabrics_leather = env['product.attribute.value'].create({
+ 'name': 'Leather',
+ 'attribute_id': chair_fabrics_attribute.id,
+ })
+ chair_fabrics_other = env['product.attribute.value'].create({
+ 'name': 'Other',
+ 'attribute_id': chair_fabrics_attribute.id,
+ 'is_custom': True,
+ })
+ chair_fabrics_line = env['product.template.attribute.line'].create({
+ 'product_tmpl_id': configurable_chair.product_tmpl_id.id,
+ 'attribute_id': chair_fabrics_attribute.id,
+ 'value_ids': [(6, 0, [chair_fabrics_leather.id, chair_fabrics_other.id])]
+ })
+ chair_color_line.product_template_value_ids[1].is_custom = True
+
+ fixed_pricelist = env['product.pricelist'].create({
+ 'name': 'Fixed',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ 'applied_on': '0_product_variant',
+ 'product_id': wall_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 13.95, # test for issues like in 7f260ab517ebde634fc274e928eb062463f0d88f
+ 'applied_on': '0_product_variant',
+ 'product_id': small_shelf.id,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Percentage',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'percentage',
+ 'percent_price': 100,
+ 'applied_on': '0_product_variant',
+ 'product_id': wall_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'percentage',
+ 'percent_price': 99,
+ 'applied_on': '0_product_variant',
+ 'product_id': small_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'percentage',
+ 'percent_price': 0,
+ 'applied_on': '0_product_variant',
+ 'product_id': magnetic_board.id,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Formula',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'formula',
+ 'price_discount': 6,
+ 'price_surcharge': 5,
+ 'applied_on': '0_product_variant',
+ 'product_id': wall_shelf.id,
+ }), (0, 0, {
+ # .99 prices
+ 'compute_price': 'formula',
+ 'price_surcharge': -0.01,
+ 'price_round': 1,
+ 'applied_on': '0_product_variant',
+ 'product_id': small_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'formula',
+ 'price_min_margin': 10,
+ 'price_max_margin': 100,
+ 'applied_on': '0_product_variant',
+ 'product_id': magnetic_board.id,
+ }), (0, 0, {
+ 'compute_price': 'formula',
+ 'price_surcharge': 10,
+ 'price_max_margin': 5,
+ 'applied_on': '0_product_variant',
+ 'product_id': monitor_stand.id,
+ }), (0, 0, {
+ 'compute_price': 'formula',
+ 'price_discount': -100,
+ 'price_min_margin': 5,
+ 'price_max_margin': 20,
+ 'applied_on': '0_product_variant',
+ 'product_id': desk_pad.id,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'min_quantity ordering',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ 'applied_on': '0_product_variant',
+ 'min_quantity': 2,
+ 'product_id': wall_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ 'applied_on': '0_product_variant',
+ 'min_quantity': 1,
+ 'product_id': wall_shelf.id,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ 'applied_on': '0_product_variant',
+ 'min_quantity': 2,
+ 'product_id': env.ref('point_of_sale.product_product_consumable').id,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Product template',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ 'applied_on': '1_product',
+ 'product_tmpl_id': wall_shelf.product_tmpl_id.id,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ })],
+ })
+
+ product_category_3 = env['product.category'].create({
+ 'name': 'Services',
+ 'parent_id': env.ref('product.product_category_1').id,
+ })
+
+ env['product.pricelist'].create({
+ # no category has precedence over category
+ 'name': 'Category vs no category',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ 'applied_on': '2_product_category',
+ 'categ_id': product_category_3.id, # All / Saleable / Services
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ })],
+ })
+
+ p = env['product.pricelist'].create({
+ 'name': 'Category',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ 'applied_on': '2_product_category',
+ 'categ_id': env.ref('product.product_category_all').id,
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ 'applied_on': '2_product_category',
+ 'categ_id': product_category_3.id, # All / Saleable / Services
+ })],
+ })
+
+ today = date.today()
+ one_week_ago = today - timedelta(weeks=1)
+ two_weeks_ago = today - timedelta(weeks=2)
+ one_week_from_now = today + timedelta(weeks=1)
+ two_weeks_from_now = today + timedelta(weeks=2)
+
+ public_pricelist = env['product.pricelist'].create({
+ 'name': 'Public Pricelist',
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Dates',
+ 'item_ids': [(0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 1,
+ 'date_start': two_weeks_ago.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ 'date_end': one_week_ago.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 2,
+ 'date_start': today.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ 'date_end': one_week_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ }), (0, 0, {
+ 'compute_price': 'fixed',
+ 'fixed_price': 3,
+ 'date_start': one_week_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ 'date_end': two_weeks_from_now.strftime(DEFAULT_SERVER_DATE_FORMAT),
+ })],
+ })
+
+ cost_base_pricelist = env['product.pricelist'].create({
+ 'name': 'Cost base',
+ 'item_ids': [(0, 0, {
+ 'base': 'standard_price',
+ 'compute_price': 'percentage',
+ 'percent_price': 55,
+ })],
+ })
+
+ pricelist_base_pricelist = env['product.pricelist'].create({
+ 'name': 'Pricelist base',
+ 'item_ids': [(0, 0, {
+ 'base': 'pricelist',
+ 'base_pricelist_id': cost_base_pricelist.id,
+ 'compute_price': 'percentage',
+ 'percent_price': 15,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Pricelist base 2',
+ 'item_ids': [(0, 0, {
+ 'base': 'pricelist',
+ 'base_pricelist_id': pricelist_base_pricelist.id,
+ 'compute_price': 'percentage',
+ 'percent_price': 3,
+ })],
+ })
+
+ env['product.pricelist'].create({
+ 'name': 'Pricelist base rounding',
+ 'item_ids': [(0, 0, {
+ 'base': 'pricelist',
+ 'base_pricelist_id': fixed_pricelist.id,
+ 'compute_price': 'percentage',
+ 'percent_price': 0.01,
+ })],
+ })
+
+ excluded_pricelist = env['product.pricelist'].create({
+ 'name': 'Not loaded'
+ })
+ res_partner_18 = self.env['res.partner'].create({
+ 'name': 'Lumber Inc',
+ 'is_company': True,
+ })
+ res_partner_18.property_product_pricelist = excluded_pricelist
+
+ partner = self.env['res.partner'].create({
+ 'name': 'TEST PARTNER',
+ 'email': 'test@partner.com',
+ })
+
+ # set the company currency to USD, otherwise it will assume
+ # euro's. this will cause issues as the sales journal is in
+ # USD, because of this all products would have a different
+ # price
+ main_company.currency_id = env.ref('base.USD')
+
+ test_sale_journal = journal_obj.create({'name': 'Sales Journal - Test',
+ 'code': 'TSJ',
+ 'type': 'sale',
+ 'company_id': main_company.id})
+
+ all_pricelists = env['product.pricelist'].search([('id', '!=', excluded_pricelist.id)])
+ all_pricelists.write(dict(currency_id=main_company.currency_id.id))
+
+ src_tax = env['account.tax'].create({'name': "SRC", 'amount': 10})
+ dst_tax = env['account.tax'].create({'name': "DST", 'amount': 5})
+
+ letter_tray.taxes_id = [(6, 0, [src_tax.id])]
+
+ self.main_pos_config.write({
+ 'tax_regime_selection': True,
+ 'fiscal_position_ids': [(0, 0, {
+ 'name': "FP-POS-2M",
+ 'tax_ids': [
+ (0,0,{'tax_src_id': src_tax.id,
+ 'tax_dest_id': src_tax.id}),
+ (0,0,{'tax_src_id': src_tax.id,
+ 'tax_dest_id': dst_tax.id})]
+ })],
+ 'journal_id': test_sale_journal.id,
+ 'invoice_journal_id': test_sale_journal.id,
+ 'payment_method_ids': [(0, 0, { 'name': 'Cash',
+ 'is_cash_count': True,
+ 'cash_journal_id': cash_journal.id,
+ 'receivable_account_id': account_receivable.id,
+ })],
+ 'use_pricelist': True,
+ 'pricelist_id': public_pricelist.id,
+ 'available_pricelist_ids': [(4, pricelist.id) for pricelist in all_pricelists],
+ 'module_pos_loyalty': False,
+ })
+
+ # Change the default sale pricelist of customers,
+ # so the js tests can expect deterministically this pricelist when selecting a customer.
+ env['ir.property']._set_default(
+ "property_product_pricelist",
+ "res.partner",
+ public_pricelist,
+ )
+
+
+@odoo.tests.tagged('post_install', '-at_install')
+class TestUi(TestPointOfSaleHttpCommon):
+ def test_01_pos_basic_order(self):
+
+ self.main_pos_config.write({
+ 'iface_tipproduct': True,
+ 'tip_product_id': self.tip.id,
+ })
+
+ # open a session, the /pos/ui controller will redirect to it
+ self.main_pos_config.open_session_cb(check_coa=False)
+
+ # needed because tests are run before the module is marked as
+ # installed. In js web will only load qweb coming from modules
+ # that are returned by the backend in module_boot. Without
+ # this you end up with js, css but no qweb.
+ self.env['ir.module.module'].search([('name', '=', 'point_of_sale')], limit=1).state = 'installed'
+
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'pos_pricelist', login="admin")
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'pos_basic_order', login="admin")
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'ProductScreenTour', login="admin")
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'PaymentScreenTour', login="admin")
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'ReceiptScreenTour', login="admin")
+
+ for order in self.env['pos.order'].search([]):
+ self.assertEqual(order.state, 'paid', "Validated order has payment of " + str(order.amount_paid) + " and total of " + str(order.amount_total))
+
+ # check if email from ReceiptScreenTour is properly sent
+ email_count = self.env['mail.mail'].search_count([('email_to', '=', 'test@receiptscreen.com')])
+ self.assertEqual(email_count, 1)
+
+ def test_02_pos_with_invoiced(self):
+ self.main_pos_config.open_session_cb(check_coa=False)
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'ChromeTour', login="admin")
+ n_invoiced = self.env['pos.order'].search_count([('state', '=', 'invoiced')])
+ n_paid = self.env['pos.order'].search_count([('state', '=', 'paid')])
+ self.assertEqual(n_invoiced, 1, 'There should be 1 invoiced order.')
+ self.assertEqual(n_paid, 2, 'There should be 2 paid order.')
+
+ def test_03_order_management(self):
+ self.main_pos_config.write({ 'manage_orders': True, 'module_account': True })
+ self.main_pos_config.open_session_cb(check_coa=False)
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'OrderManagementScreenTour', login="admin")
+
+ def test_04_product_configurator(self):
+ self.main_pos_config.write({ 'product_configurator': True })
+ self.main_pos_config.open_session_cb(check_coa=False)
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config, 'ProductConfiguratorTour', login="admin")
+
+ def test_05_ticket_screen(self):
+ self.main_pos_config.open_session_cb(check_coa=False)
+ self.start_tour("/pos/ui?config_id=%d" % self.main_pos_config.id, 'TicketScreenTour', login="admin")