2018-01-16 06:58:15 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
2018-01-16 11:34:37 +01:00
|
|
|
# Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details.
|
2018-01-16 06:58:15 +01:00
|
|
|
|
2018-01-16 11:34:37 +01:00
|
|
|
from flectra.api import Environment
|
|
|
|
from flectra.tools import DEFAULT_SERVER_DATE_FORMAT
|
2018-01-16 06:58:15 +01:00
|
|
|
from datetime import date, timedelta
|
|
|
|
|
2018-01-16 11:34:37 +01:00
|
|
|
import flectra.tests
|
2018-01-16 06:58:15 +01:00
|
|
|
|
|
|
|
|
2018-01-16 11:34:37 +01:00
|
|
|
class TestUi(flectra.tests.HttpCase):
|
2018-01-16 06:58:15 +01:00
|
|
|
def test_01_pos_basic_order(self):
|
|
|
|
cr = self.registry.cursor()
|
|
|
|
assert cr == self.registry.test_cr
|
|
|
|
env = Environment(cr, self.uid, {})
|
|
|
|
|
|
|
|
# By default parent_store computation is deferred until end of
|
|
|
|
# tests. Pricelist items however are sorted based on these
|
|
|
|
# fields, so they need to be computed.
|
|
|
|
env['product.category']._parent_store_compute()
|
|
|
|
|
|
|
|
journal_obj = env['account.journal']
|
|
|
|
account_obj = env['account.account']
|
|
|
|
main_company = env.ref('base.main_company')
|
|
|
|
main_pos_config = env.ref('point_of_sale.pos_config_main')
|
|
|
|
|
|
|
|
account_receivable = account_obj.create({'code': 'X1012',
|
|
|
|
'name': 'Account Receivable - Test',
|
|
|
|
'user_type_id': env.ref('account.data_account_type_receivable').id,
|
|
|
|
'reconcile': True})
|
|
|
|
field = self.env['ir.model.fields'].search([('name', '=', 'property_account_receivable_id'),
|
|
|
|
('model', '=', 'res.partner'),
|
|
|
|
('relation', '=', 'account.account')], limit=1)
|
|
|
|
env['ir.property'].create({'name': 'property_account_receivable_id',
|
|
|
|
'company_id': main_company.id,
|
|
|
|
'fields_id': field.id,
|
|
|
|
'value': 'account.account,' + str(account_receivable.id)})
|
|
|
|
|
|
|
|
# test an extra price on an attribute
|
|
|
|
pear = env.ref('point_of_sale.poire_conference')
|
|
|
|
attribute_value = env['product.attribute.value'].create({
|
|
|
|
'name': 'add 2',
|
|
|
|
'product_ids': [(6, 0, [pear.id])],
|
|
|
|
'attribute_id': env['product.attribute'].create({
|
|
|
|
'name': 'add 2',
|
|
|
|
}).id,
|
|
|
|
})
|
|
|
|
env['product.attribute.price'].create({
|
|
|
|
'product_tmpl_id': pear.product_tmpl_id.id,
|
|
|
|
'price_extra': 2,
|
|
|
|
'value_id': attribute_value.id,
|
|
|
|
})
|
|
|
|
|
|
|
|
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': env.ref('point_of_sale.boni_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'fixed',
|
|
|
|
'fixed_price': 13.95, # test for issues like in 7f260ab517ebde634fc274e928eb062463f0d88f
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.papillon_orange').id,
|
|
|
|
})],
|
|
|
|
})
|
|
|
|
|
|
|
|
env['product.pricelist'].create({
|
|
|
|
'name': 'Percentage',
|
|
|
|
'item_ids': [(0, 0, {
|
|
|
|
'compute_price': 'percentage',
|
|
|
|
'percent_price': 100,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.boni_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'percentage',
|
|
|
|
'percent_price': 99,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.papillon_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'percentage',
|
|
|
|
'percent_price': 0,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.citron').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': env.ref('point_of_sale.boni_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
# .99 prices
|
|
|
|
'compute_price': 'formula',
|
|
|
|
'price_surcharge': -0.01,
|
|
|
|
'price_round': 1,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.papillon_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'formula',
|
|
|
|
'price_min_margin': 10,
|
|
|
|
'price_max_margin': 100,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.citron').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'formula',
|
|
|
|
'price_surcharge': 10,
|
|
|
|
'price_max_margin': 5,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.limon').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'formula',
|
|
|
|
'price_discount': -100,
|
|
|
|
'price_min_margin': 5,
|
|
|
|
'price_max_margin': 20,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'product_id': env.ref('point_of_sale.pamplemousse_rouge_pamplemousse').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': env.ref('point_of_sale.boni_orange').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'fixed',
|
|
|
|
'fixed_price': 2,
|
|
|
|
'applied_on': '0_product_variant',
|
|
|
|
'min_quantity': 1,
|
|
|
|
'product_id': env.ref('point_of_sale.boni_orange').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': env.ref('point_of_sale.boni_orange_product_template').id,
|
|
|
|
}), (0, 0, {
|
|
|
|
'compute_price': 'fixed',
|
|
|
|
'fixed_price': 2,
|
|
|
|
})],
|
|
|
|
})
|
|
|
|
|
|
|
|
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': env.ref('product.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': env.ref('product.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)
|
|
|
|
|
|
|
|
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'
|
|
|
|
})
|
|
|
|
env.ref('base.res_partner_18').property_product_pricelist = excluded_pricelist
|
|
|
|
|
|
|
|
# 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))
|
|
|
|
|
|
|
|
main_pos_config.write({
|
|
|
|
'journal_id': test_sale_journal.id,
|
|
|
|
'invoice_journal_id': test_sale_journal.id,
|
|
|
|
'journal_ids': [(0, 0, {'name': 'Cash Journal - Test',
|
|
|
|
'code': 'TSC',
|
|
|
|
'type': 'cash',
|
|
|
|
'company_id': main_company.id,
|
|
|
|
'journal_user': True})],
|
|
|
|
'available_pricelist_ids': [(4, pricelist.id) for pricelist in all_pricelists],
|
|
|
|
})
|
|
|
|
|
|
|
|
# open a session, the /pos/web controller will redirect to it
|
|
|
|
main_pos_config.open_session_cb()
|
|
|
|
|
|
|
|
# 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.
|
|
|
|
env['ir.module.module'].search([('name', '=', 'point_of_sale')], limit=1).state = 'installed'
|
|
|
|
cr.release()
|
|
|
|
|
|
|
|
self.phantom_js("/pos/web",
|
2018-01-16 11:34:37 +01:00
|
|
|
"flectra.__DEBUG__.services['web_tour.tour'].run('pos_pricelist')",
|
|
|
|
"flectra.__DEBUG__.services['web_tour.tour'].tours.pos_pricelist.ready",
|
2018-01-16 06:58:15 +01:00
|
|
|
login="admin")
|
|
|
|
|
|
|
|
self.phantom_js("/pos/web",
|
2018-01-16 11:34:37 +01:00
|
|
|
"flectra.__DEBUG__.services['web_tour.tour'].run('pos_basic_order')",
|
|
|
|
"flectra.__DEBUG__.services['web_tour.tour'].tours.pos_basic_order.ready",
|
2018-01-16 06:58:15 +01:00
|
|
|
login="admin")
|
|
|
|
|
|
|
|
for order in 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))
|