mirror of
https://github.com/bringout/oca-ocb-core.git
synced 2026-04-20 00:52:07 +02:00
vanilla 19.0
This commit is contained in:
parent
991d2234ca
commit
d1963a3c3a
3066 changed files with 1651266 additions and 922560 deletions
|
|
@ -2,13 +2,22 @@
|
|||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
import re
|
||||
from contextlib import suppress
|
||||
|
||||
import odoo.tests
|
||||
from odoo.tools.misc import file_open
|
||||
from werkzeug.urls import url_quote_plus
|
||||
|
||||
RE_FORBIDDEN_STATEMENTS = re.compile(r'test.*\.(only|debug)\(')
|
||||
RE_ONLY = re.compile(r'QUnit\.(only|debug)\(')
|
||||
|
||||
|
||||
def unit_test_error_checker(message):
|
||||
return '[HOOT]' not in message
|
||||
|
||||
|
||||
def qunit_error_checker(message):
|
||||
# ! DEPRECATED
|
||||
# We don't want to stop qunit if a qunit is breaking.
|
||||
|
||||
# '%s/%s test failed.' case: end message when all tests are finished
|
||||
|
|
@ -22,47 +31,62 @@ def qunit_error_checker(message):
|
|||
return True # in other cases, always stop (missing dependency, ...)
|
||||
|
||||
|
||||
def _get_filters(test_params):
|
||||
filters = []
|
||||
for sign, param in test_params:
|
||||
parts = param.split(',')
|
||||
for part in parts:
|
||||
part = part.strip()
|
||||
if not part:
|
||||
continue
|
||||
part_sign = sign
|
||||
if part.startswith('-'):
|
||||
part = part[1:]
|
||||
part_sign = '-' if sign == '+' else '+'
|
||||
filters.append((part_sign, part))
|
||||
return sorted(filters)
|
||||
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class WebsuiteCommon(odoo.tests.HttpCase):
|
||||
def get_filter(self, test_params):
|
||||
positive = []
|
||||
negative = []
|
||||
for sign, param in test_params:
|
||||
filters = param.split(',')
|
||||
for filter in filters:
|
||||
filter = filter.strip()
|
||||
if not filter:
|
||||
continue
|
||||
negate = sign == '-'
|
||||
if filter.startswith('-'):
|
||||
negate = not negate
|
||||
filter = filter[1:]
|
||||
if negate:
|
||||
negative.append(f'({re.escape(filter)}.*)')
|
||||
else:
|
||||
positive.append(f'({re.escape(filter)}.*)')
|
||||
class QunitCommon(odoo.tests.HttpCase):
|
||||
|
||||
def setUp(self):
|
||||
super().setUp()
|
||||
self.qunit_filters = self.get_qunit_filters()
|
||||
|
||||
def get_qunit_regex(self, test_params):
|
||||
filters = _get_filters(test_params)
|
||||
positive = [f'({re.escape(f)}.*)' for sign, f in filters if sign == '+']
|
||||
negative = [f'({re.escape(f)}.*)' for sign, f in filters if sign == '-']
|
||||
filter = ''
|
||||
if positive or negative:
|
||||
if filters:
|
||||
positive_re = '|'.join(positive) or '.*'
|
||||
negative_re = '|'.join(negative)
|
||||
negative_re = f'(?!{negative_re})' if negative_re else ''
|
||||
filter = f'^({negative_re})({positive_re})$'
|
||||
return filter
|
||||
|
||||
def test_get_filter(self):
|
||||
f1 = self.get_filter([('+', 'utils,mail,-utils > bl1,-utils > bl2')])
|
||||
f2 = self.get_filter([('+', 'utils'), ('-', 'utils > bl1,utils > bl2'), ('+', 'mail')])
|
||||
for f in (f1, f2):
|
||||
self.assertRegex('utils', f)
|
||||
self.assertRegex('mail', f)
|
||||
self.assertRegex('utils > something', f)
|
||||
def get_qunit_filters(self):
|
||||
filter_param = ''
|
||||
filter = self.get_qunit_regex(self._test_params)
|
||||
if filter:
|
||||
url_filter = url_quote_plus(filter)
|
||||
filter_param = f'&filter=/{url_filter}/'
|
||||
return filter_param
|
||||
|
||||
self.assertNotRegex('utils > bl1', f)
|
||||
self.assertNotRegex('utils > bl2', f)
|
||||
self.assertNotRegex('web', f)
|
||||
def test_get_qunit_regex(self):
|
||||
f = self.get_qunit_regex([('+', 'utils,mail,-utils > bl1,-utils > bl2')])
|
||||
f2 = self.get_qunit_regex([('+', 'utils'), ('-', 'utils > bl1,utils > bl2'), ('+', 'mail')])
|
||||
self.assertEqual(f, f2)
|
||||
self.assertRegex('utils', f)
|
||||
self.assertRegex('mail', f)
|
||||
self.assertRegex('utils > something', f)
|
||||
|
||||
f2 = self.get_filter([('+', '-utils > bl1,-utils > bl2')])
|
||||
f3 = self.get_filter([('-', 'utils > bl1,utils > bl2')])
|
||||
self.assertNotRegex('utils > bl1', f)
|
||||
self.assertNotRegex('utils > bl2', f)
|
||||
self.assertNotRegex('web', f)
|
||||
|
||||
f2 = self.get_qunit_regex([('+', '-utils > bl1,-utils > bl2')])
|
||||
f3 = self.get_qunit_regex([('-', 'utils > bl1,utils > bl2')])
|
||||
for f in (f2, f3):
|
||||
self.assertRegex('utils', f)
|
||||
self.assertRegex('mail', f)
|
||||
|
|
@ -72,33 +96,95 @@ class WebsuiteCommon(odoo.tests.HttpCase):
|
|||
self.assertNotRegex('utils > bl1', f)
|
||||
self.assertNotRegex('utils > bl2', f)
|
||||
|
||||
def get_filter_param(self):
|
||||
filter_param = ''
|
||||
filter = self.get_filter(self._test_params)
|
||||
if filter:
|
||||
url_filter = url_quote_plus(filter)
|
||||
filter_param = f'&filter=/{url_filter}/'
|
||||
return filter_param
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class HOOTCommon(odoo.tests.HttpCase):
|
||||
|
||||
def setUp(self):
|
||||
super().setUp()
|
||||
self.hoot_filters = self.get_hoot_filters()
|
||||
|
||||
def _generate_hash(self, test_string):
|
||||
hash = 0
|
||||
for char in test_string:
|
||||
hash = (hash << 5) - hash + ord(char)
|
||||
hash = hash & 0xFFFFFFFF
|
||||
return f'{hash:08x}'
|
||||
|
||||
def get_hoot_filters(self):
|
||||
filters = _get_filters(self._test_params)
|
||||
filter = ''
|
||||
for sign, f in filters:
|
||||
h = self._generate_hash(f)
|
||||
if sign == '-':
|
||||
h = f'-{h}'
|
||||
# Since we don't know if the descriptor we have is a test or a suite, we need to provide the hash for a generic "job"
|
||||
filter += f'&id={h}'
|
||||
return filter
|
||||
|
||||
def test_generate_hoot_hash(self):
|
||||
self.assertEqual(self._generate_hash('@web/core'), 'e39ce9ba')
|
||||
self.assertEqual(self._generate_hash('@web/core/autocomplete'), '69a6561d') # suite
|
||||
self.assertEqual(self._generate_hash('@web/core/autocomplete/open dropdown on input'), 'ee565d54') # test
|
||||
|
||||
def test_get_hoot_filter(self):
|
||||
self._test_params = []
|
||||
self.assertEqual(self.get_hoot_filters(), '')
|
||||
expected = '&id=e39ce9ba&id=-69a6561d'
|
||||
self._test_params = [('+', '@web/core,-@web/core/autocomplete')]
|
||||
self.assertEqual(self.get_hoot_filters(), expected)
|
||||
self._test_params = [('+', '@web/core'), ('-', '@web/core/autocomplete')]
|
||||
self.assertEqual(self.get_hoot_filters(), expected)
|
||||
self._test_params = [('+', '-@web/core/autocomplete,-@web/core/autocomplete2')]
|
||||
self.assertEqual(self.get_hoot_filters(), '&id=-69a6561d&id=-cb246db5')
|
||||
self._test_params = [('-', '-@web/core/autocomplete,-@web/core/autocomplete2')]
|
||||
self.assertEqual(self.get_hoot_filters(), '&id=69a6561d&id=cb246db5')
|
||||
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class WebSuite(WebsuiteCommon):
|
||||
class WebSuite(QunitCommon, HOOTCommon):
|
||||
|
||||
@odoo.tests.no_retry
|
||||
def test_js(self):
|
||||
filter_param = self.get_filter_param()
|
||||
# webclient desktop test suite
|
||||
self.browser_js('/web/tests?mod=web%s' % filter_param, "", "", login='admin', timeout=1800, error_checker=qunit_error_checker)
|
||||
def test_unit_desktop(self):
|
||||
# Unit tests suite (desktop)
|
||||
self.browser_js(f'/web/tests?headless&loglevel=2&preset=desktop&timeout=15000{self.hoot_filters}', "", "", login='admin', timeout=3000, success_signal="[HOOT] Test suite succeeded", error_checker=unit_test_error_checker)
|
||||
|
||||
@odoo.tests.no_retry
|
||||
def test_hoot(self):
|
||||
# HOOT tests suite
|
||||
self.browser_js(f'/web/static/lib/hoot/tests/index.html?headless&loglevel=2{self.hoot_filters}', "", "", login='admin', timeout=1800, success_signal="[HOOT] Test suite succeeded", error_checker=unit_test_error_checker)
|
||||
|
||||
@odoo.tests.no_retry
|
||||
def test_qunit_desktop(self):
|
||||
# ! DEPRECATED
|
||||
self.browser_js(f'/web/tests/legacy?mod=web{self.qunit_filters}', "", "", login='admin', timeout=1800, success_signal="QUnit test suite done.", error_checker=qunit_error_checker)
|
||||
|
||||
def test_check_suite(self):
|
||||
# verify no js test is using `QUnit.only` as it forbid any other test to be executed
|
||||
self._check_forbidden_statements('web.assets_unit_tests')
|
||||
# Checks that no test is using `only` or `debug` as it prevents other tests to be run
|
||||
self._check_only_call('web.qunit_suite_tests')
|
||||
self._check_only_call('web.qunit_mobile_suite_tests')
|
||||
|
||||
def _check_forbidden_statements(self, bundle):
|
||||
# As we currently are not in a request context, we cannot render `web.layout`.
|
||||
# We then re-define it as a minimal proxy template.
|
||||
self.env.ref('web.layout').write({'arch_db': '<t t-name="web.layout"><html><head><meta charset="utf-8"/><link/><script id="web.layout.odooscript"/><meta/><t t-esc="head"/></head><body><t t-out="0"/></body></html></t>'})
|
||||
|
||||
assets = self.env['ir.qweb']._get_asset_content(bundle)[0]
|
||||
if len(assets) == 0:
|
||||
self.fail("No assets found in the given test bundle")
|
||||
|
||||
for asset in assets:
|
||||
filename = asset['filename']
|
||||
if not filename.endswith('.test.js'):
|
||||
continue
|
||||
with suppress(FileNotFoundError):
|
||||
with file_open(filename, 'rb', filter_ext=('.js',)) as fp:
|
||||
if RE_FORBIDDEN_STATEMENTS.search(fp.read().decode('utf-8')):
|
||||
self.fail("`only()` or `debug()` used in file %r" % asset['url'])
|
||||
|
||||
def _check_only_call(self, suite):
|
||||
# ! DEPRECATED
|
||||
# As we currently aren't in a request context, we can't render `web.layout`.
|
||||
# redefinied it as a minimal proxy template.
|
||||
self.env.ref('web.layout').write({'arch_db': '<t t-name="web.layout"><head><meta charset="utf-8"/><t t-esc="head"/></head></t>'})
|
||||
self.env.ref('web.layout').write({'arch_db': '<t t-name="web.layout"><html><head><meta charset="utf-8"/><link/><script id="web.layout.odooscript"/><meta/><t t-esc="head"/></head><body><t t-out="0"/></body></html></t>'})
|
||||
|
||||
assets = self.env['ir.qweb']._get_asset_content(suite)[0]
|
||||
if len(assets) == 0:
|
||||
|
|
@ -106,19 +192,20 @@ class WebSuite(WebsuiteCommon):
|
|||
|
||||
for asset in assets:
|
||||
filename = asset['filename']
|
||||
if not filename or asset['atype'] != 'text/javascript':
|
||||
if not filename.endswith('.js'):
|
||||
continue
|
||||
with open(filename, 'rb') as fp:
|
||||
if RE_ONLY.search(fp.read().decode('utf-8')):
|
||||
self.fail("`QUnit.only()` or `QUnit.debug()` used in file %r" % asset['url'])
|
||||
with suppress(FileNotFoundError):
|
||||
with file_open(filename, 'rb', filter_ext=('.js',)) as fp:
|
||||
if RE_ONLY.search(fp.read().decode('utf-8')):
|
||||
self.fail("`QUnit.only()` or `QUnit.debug()` used in file %r" % asset['url'])
|
||||
|
||||
|
||||
@odoo.tests.tagged('post_install', '-at_install')
|
||||
class MobileWebSuite(WebsuiteCommon):
|
||||
class MobileWebSuite(QunitCommon, HOOTCommon):
|
||||
browser_size = '375x667'
|
||||
touch_enabled = True
|
||||
|
||||
def test_mobile_js(self):
|
||||
filter_param = self.get_filter_param()
|
||||
# webclient mobile test suite
|
||||
self.browser_js('/web/tests/mobile?mod=web%s' % filter_param, "", "", login='admin', timeout=1800, error_checker=qunit_error_checker)
|
||||
@odoo.tests.no_retry
|
||||
def test_unit_mobile(self):
|
||||
# Unit tests suite (mobile)
|
||||
self.browser_js(f'/web/tests?headless&loglevel=2&preset=mobile&tag=-headless&timeout=15000{self.hoot_filters}', "", "", login='admin', timeout=2100, success_signal="[HOOT] Test suite succeeded", error_checker=unit_test_error_checker)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue