# Part of Odoo. See LICENSE file for full copyright and licensing details. import random import re from unittest.mock import patch import textwrap from datetime import datetime from lxml import etree import logging import odoo from odoo.tests.common import BaseCase, HttpCase, tagged from odoo.tools import topological_sort from odoo.addons.base.models.assetsbundle import AssetsBundle, WebAsset _logger = logging.getLogger(__name__) class TestStaticInheritanceCommon(odoo.tests.TransactionCase): def setUp(self): super().setUp() self.template_files = { '/module_1/static/xml/file_1.xml': """
Ho !
At first I was afraid
Kept thinking I could never live without you by my side
And I grew strong
""", '/module_2/static/xml/file_1.xml': """
I was petrified
Scary screams
But then I spent so many nights thinking how you did me wrong
And I learned how to get along
And I learned how to get along
""", } self._patch = patch.object(WebAsset, '_fetch_content', lambda asset: self.template_files[asset.url]) self.startPatcher(self._patch) def renderBundle(self, debug=False): files = [] for url in self.template_files: atype = 'text/xml' if '.js' in url: atype = 'text/javascript' files.append({ 'atype': atype, 'url': url, 'filename': url, 'content': None, 'media': None, }) asset = AssetsBundle('web.test_bundle', files, env=self.env, css=False, js=True) # to_node return the files descriptions and generate attachments. asset.to_node(css=False, js=False, debug=debug and 'assets' or '') content = asset.xml(show_inherit_info=debug) return f'\n{content}\n' # Custom Assert def assertXMLEqual(self, output, expected): self.assertTrue(output) self.assertTrue(expected) self.assertEqual(etree.fromstring(output), etree.fromstring(expected)) @tagged('assets_bundle', 'static_templates') class TestStaticInheritance(TestStaticInheritanceCommon): # Actual test cases def test_static_with_debug_mode(self): expected = """
Ho !
At first I was afraid
Kept thinking I could never live without you by my side
And I grew strong
And I learned how to get along
Ho !
At first I was afraid
I was petrified
But then I spent so many nights thinking how you did me wrong
Kept thinking I could never live without you by my side
And I learned how to get along
""" self.assertXMLEqual(self.renderBundle(debug=True), expected) def test_static_inheritance_01(self): expected = """
Ho !
At first I was afraid
Kept thinking I could never live without you by my side
And I grew strong
And I learned how to get along
Ho !
At first I was afraid
I was petrified
But then I spent so many nights thinking how you did me wrong
Kept thinking I could never live without you by my side
And I learned how to get along
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_inheritance_02(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
Kept thinking I could never live without you by my side
I was petrified
""", } expected = """
At first I was afraid
Kept thinking I could never live without you by my side
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_inheritance_03(self): self.template_files = { '/module_1/static/xml/file_1.xml': '''
At first I was afraid
Kept thinking I could never live without you by my side
I was petrified
''' } expected = """
At first I was afraid
Kept thinking I could never live without you by my side
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
At first I was afraid
Kept thinking I could never live without you by my side
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_inheritance_in_same_module(self): self.template_files = { '/module_1/static/xml/file_1.xml': '''
At first I was afraid
Kept thinking I could never live without you by my side
''', '/module_1/static/xml/file_2.xml': '''
I was petrified
''' } expected = """
At first I was afraid
Kept thinking I could never live without you by my side
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_inheritance_in_same_file(self): self.template_files = { '/module_1/static/xml/file_1.xml': '''
At first I was afraid
Kept thinking I could never live without you by my side
I was petrified
''', } expected = """
At first I was afraid
Kept thinking I could never live without you by my side
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_inherit_extended_template(self): self.template_files = { '/module_1/static/xml/file_1.xml': '''
At first I was afraid
Kept thinking I could never live without you by my side
I was petrified
But then I spent so many nights thinking how you did me wrong
''', } expected = """
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
At first I was afraid
I was petrified
Kept thinking I could never live without you by my side
But then I spent so many nights thinking how you did me wrong
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_sibling_extension(self): self.template_files = { '/module_1/static/xml/file_1.xml': '''
I am a man of constant sorrow
I've seen trouble all my days
''', '/module_2/static/xml/file_1.xml': '''
In constant sorrow all through his days
''', '/module_3/static/xml/file_1.xml': '''
Oh Brother !
''' } expected = """
I am a man of constant sorrow
In constant sorrow all through his days
Oh Brother !
I've seen trouble all my days
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_static_misordered_modules(self): files = self.template_files self.template_files = { '/module_2/static/xml/file_1.xml': files['/module_2/static/xml/file_1.xml'], '/module_1/static/xml/file_1.xml': files['/module_1/static/xml/file_1.xml'], } with self.assertRaises(ValueError) as ve: self.renderBundle(debug=False) self.assertEqual( str(ve.exception), "Module 'module_1' not loaded or inexistent (try to inherit 'template_1_1'), or templates of addon being loaded 'module_2' are misordered (template 'template_2_1')" ) def test_static_misordered_templates(self): self.template_files['/module_2/static/xml/file_1.xml'] = """
I was petrified
And I learned how to get along
""" with self.assertRaises(ValueError) as ve: self.renderBundle(debug=False) self.assertEqual( str(ve.exception), "Cannot create 'module_2.template_2_1' because the template to inherit 'module_2.template_2_2' is not found.", ) def test_replace_in_debug_mode(self): """ Replacing a template's meta definition in place doesn't keep the original attrs of the template """ self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong
""", } expected = """
And I grew strong
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_replace_in_debug_mode2(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

And so you're back
""", } expected = """
And I grew strong

And I learned how to get along

And so you're back
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_replace_in_debug_mode3(self): """Text outside of a div which will replace a whole template becomes outside of the template This doesn't mean anything in terms of the business of template inheritance But it is in the XPATH specs""" self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

And so you're back
""", } expected = """
And I grew strong

And I learned how to get along

And so you're back
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_replace_root_node_tag(self): """ Root node IS targeted by //NODE_TAG in xpath """ self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
Inner Form
Form replacer
""", } expected = """
Form replacer
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_replace_root_node_tag_in_primary(self): """ Root node IS targeted by //NODE_TAG in xpath """ self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
Inner Form
Form replacer
""", } expected = """
At first I was afraid
Inner Form
Form replacer
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_primary_replace_debug(self): """ The inheriting template has got both its own defining attrs and new ones if one is to replace its defining root node """ self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

""", } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_replace_in_nodebug_mode1(self): """Comments already in the arch are ignored""" self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

And so you're back
""", } expected = """
And I grew strong

And I learned how to get along

And so you're back
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_from_dotted_tname_1(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

""", } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_from_dotted_tname_2(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

""", } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_from_dotted_tname_2bis(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

""", } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_from_dotted_tname_2ter(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
And I grew strong

And I learned how to get along

""", } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_from_dotted_tname_3(self): self.template_files = { '/module_1/static/xml/file_1.xml': """
At first I was afraid
""", '/module_2/static/xml/file_1.xml': """
And I grew strong

And I learned how to get along

""" } expected = """
At first I was afraid
And I grew strong

And I learned how to get along

""" self.assertXMLEqual(self.renderBundle(debug=False), expected) def test_inherit_and_qweb_extend(self): self.template_files['/module_1/static/xml/file_2.xml'] = """
111
!!!
222
333
""" expected = """
!!!
At first I was afraid
Kept thinking I could never live without you by my side
And I grew strong
And I learned how to get along
111
222
333
Ho !
At first I was afraid
I was petrified
But then I spent so many nights thinking how you did me wrong
Kept thinking I could never live without you by my side
And I learned how to get along
""" self.assertXMLEqual(self.renderBundle(debug=False), expected) @tagged('-standard', 'assets_bundle', 'static_templates_performance') class TestStaticInheritancePerformance(TestStaticInheritanceCommon): def _sick_script(self, nMod, nFilePerMod, nTemplatePerFile, stepInheritInModule=2, stepInheritPreviousModule=3): """ Make a sick amount of templates to test perf nMod modules each module: has nFilesPerModule files, each of which contains nTemplatePerFile templates """ self.asset_paths = [] self.template_files = {} number_templates = 0 for m in range(nMod): for f in range(nFilePerMod): mname = 'mod_%s' % m fname = 'mod_%s/folder/file_%s.xml' % (m, f) self.asset_paths.append((fname, mname, 'bundle_1')) _file = '' for t in range(nTemplatePerFile): _template = '' if t % stepInheritInModule or t % stepInheritPreviousModule or t == 0: _template += """
Parent
""" elif not t % stepInheritInModule and t >= 1: _template += """
Sick XPath
""" elif not t % stepInheritPreviousModule and m >= 1: _template += """
Mental XPath
""" if _template: number_templates += 1 _template_number = 1000 * f + t _file += _template % { 't_number': _template_number, 'm_number': m, 't_inherit': _template_number - 1, 't_module_inherit': _template_number, 'm_module_inherit': m - 1, } _file += '
' self.template_files[fname] = _file self.assertEqual(number_templates, nMod * nFilePerMod * nTemplatePerFile) def test_static_templates_treatment_linearity(self): # With 2500 templates for starters nMod, nFilePerMod, nTemplatePerFile = 50, 5, 10 self._sick_script(nMod, nFilePerMod, nTemplatePerFile) before = datetime.now() contents = self.renderBundle(debug=False) after = datetime.now() delta2500 = after - before _logger.runbot('Static Templates Inheritance: 2500 templates treated in %s seconds' % delta2500.total_seconds()) whole_tree = etree.fromstring(contents) self.assertEqual(len(whole_tree), nMod * nFilePerMod * nTemplatePerFile) # With 25000 templates next nMod, nFilePerMod, nTemplatePerFile = 50, 5, 100 self._sick_script(nMod, nFilePerMod, nTemplatePerFile) before = datetime.now() self.renderBundle(debug=False) after = datetime.now() delta25000 = after - before time_ratio = delta25000.total_seconds() / delta2500.total_seconds() _logger.runbot('Static Templates Inheritance: 25000 templates treated in %s seconds' % delta25000.total_seconds()) _logger.runbot('Static Templates Inheritance: Computed linearity ratio: %s' % time_ratio) self.assertLessEqual(time_ratio, 14)