mirror of
https://github.com/odoo/documentation.git
synced 2026-01-02 17:59:20 +07:00
This commit replaces the outdated QUnit documentation with links to the
freshly written Hoot/web testing documentation.
closes odoo/documentation#14924
X-original-commit: 33d7058113
Signed-off-by: Julien Mougenot (jum) <jum@odoo.com>
756 lines
25 KiB
ReStructuredText
756 lines
25 KiB
ReStructuredText
.. _reference/testing:
|
|
|
|
============
|
|
Testing Odoo
|
|
============
|
|
|
|
There are many ways to test an application. In Odoo, we have three kinds of
|
|
tests
|
|
|
|
- Python unit tests (see `Testing Python code`_): useful for testing model business logic
|
|
- JS unit tests (see `Testing JS code`_): useful to test the javascript code in isolation
|
|
- Tours (see `Integration Testing`_): tours simulate a real situation. They ensures that the
|
|
python and the javascript parts properly talk to each other.
|
|
|
|
.. _testing/python:
|
|
|
|
Testing Python code
|
|
===================
|
|
|
|
Odoo provides support for testing modules using `Python's unittest library
|
|
<https://docs.python.org/3/library/unittest.html>`_.
|
|
|
|
To write tests, simply define a ``tests`` sub-package in your module, it will
|
|
be automatically inspected for test modules. Test modules should have a name
|
|
starting with ``test_`` and should be imported from ``tests/__init__.py``,
|
|
e.g.
|
|
|
|
.. code-block:: text
|
|
|
|
your_module
|
|
├── ...
|
|
├── tests
|
|
| ├── __init__.py
|
|
| ├── test_bar.py
|
|
| └── test_foo.py
|
|
|
|
and ``__init__.py`` contains::
|
|
|
|
from . import test_foo, test_bar
|
|
|
|
.. warning::
|
|
|
|
test modules which are not imported from ``tests/__init__.py`` will not be
|
|
run
|
|
|
|
The test runner will simply run any test case, as described in the official
|
|
`unittest documentation`_, but Odoo provides a number of utilities and helpers
|
|
related to testing Odoo content (modules, mainly):
|
|
|
|
.. autoclass:: odoo.tests.TransactionCase
|
|
:members: browse_ref, ref
|
|
|
|
.. autoclass:: odoo.tests.SingleTransactionCase
|
|
:members: browse_ref, ref
|
|
|
|
.. autoclass:: odoo.tests.HttpCase
|
|
:members: browse_ref, ref, url_open, browser_js
|
|
|
|
.. autofunction:: odoo.tests.tagged
|
|
|
|
By default, tests are run once right after the corresponding module has been
|
|
installed. Test cases can also be configured to run after all modules have
|
|
been installed, and not run right after the module installation::
|
|
|
|
# coding: utf-8
|
|
from odoo.tests import HttpCase, tagged
|
|
|
|
# This test should only be executed after all modules have been installed.
|
|
@tagged('-at_install', 'post_install')
|
|
class WebsiteVisitorTests(HttpCase):
|
|
def test_create_visitor_on_tracked_page(self):
|
|
Page = self.env['website.page']
|
|
|
|
The most common situation is to use
|
|
:class:`~odoo.tests.TransactionCase` and test a property of a model
|
|
in each method::
|
|
|
|
class TestModelA(TransactionCase):
|
|
def test_some_action(self):
|
|
record = self.env['model.a'].create({'field': 'value'})
|
|
record.some_action()
|
|
self.assertEqual(
|
|
record.field,
|
|
expected_field_value)
|
|
|
|
# other tests...
|
|
|
|
.. note::
|
|
|
|
Test methods must start with ``test_``
|
|
|
|
.. autoclass:: odoo.tests.Form
|
|
:members:
|
|
|
|
.. autoclass:: odoo.tests.M2MProxy
|
|
:members: add, remove, clear
|
|
|
|
.. autoclass:: odoo.tests.O2MProxy
|
|
:members: new, edit, remove
|
|
|
|
Running tests
|
|
-------------
|
|
|
|
Tests are automatically run when installing or updating modules if
|
|
:option:`--test-enable <odoo-bin --test-enable>` was enabled when starting the
|
|
Odoo server.
|
|
|
|
.. _unittest documentation: https://docs.python.org/3/library/unittest.html
|
|
|
|
.. _developer/reference/testing/selection:
|
|
|
|
Test selection
|
|
--------------
|
|
|
|
In Odoo, Python tests can be tagged to facilitate the test selection when
|
|
running tests.
|
|
|
|
Subclasses of :class:`odoo.tests.BaseCase` (usually through
|
|
:class:`~odoo.tests.TransactionCase` or
|
|
:class:`~odoo.tests.HttpCase`) are automatically tagged with
|
|
``standard`` and ``at_install`` by default.
|
|
|
|
Invocation
|
|
~~~~~~~~~~
|
|
|
|
:option:`--test-tags <odoo-bin --test-tags>` can be used to select/filter tests
|
|
to run on the command-line. It implies :option:`--test-enable <odoo-bin --test-enable>`,
|
|
so it's not necessary to specify :option:`--test-enable <odoo-bin --test-enable>`
|
|
when using :option:`--test-tags <odoo-bin --test-tags>`.
|
|
|
|
This option defaults to ``+standard`` meaning tests tagged ``standard``
|
|
(explicitly or implicitly) will be run by default when starting Odoo
|
|
with :option:`--test-enable <odoo-bin --test-enable>`.
|
|
|
|
When writing tests, the :func:`~odoo.tests.tagged` decorator can be
|
|
used on **test classes** to add or remove tags.
|
|
|
|
The decorator's arguments are tag names, as strings.
|
|
|
|
.. danger:: :func:`~odoo.tests.tagged` is a class decorator, it has no
|
|
effect on functions or methods
|
|
|
|
Tags can be prefixed with the minus (``-``) sign, to *remove* them instead of
|
|
add or select them e.g. if you don't want your test to be executed by
|
|
default you can remove the ``standard`` tag:
|
|
|
|
.. code-block:: python
|
|
|
|
from odoo.tests import TransactionCase, tagged
|
|
|
|
@tagged('-standard', 'nice')
|
|
class NiceTest(TransactionCase):
|
|
...
|
|
|
|
This test will not be selected by default, to run it the relevant tag will
|
|
have to be selected explicitly:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags nice
|
|
|
|
Note that only the tests tagged ``nice`` are going to be executed. To run
|
|
*both* ``nice`` and ``standard`` tests, provide multiple values to
|
|
:option:`--test-tags <odoo-bin --test-tags>`: on the command-line, values
|
|
are *additive* (you're selecting all tests with *any* of the specified tags)
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags nice,standard
|
|
|
|
The config switch parameter also accepts the ``+`` and ``-`` prefixes. The
|
|
``+`` prefix is implied and therefore, totally optional. The ``-`` (minus)
|
|
prefix is made to deselect tests tagged with the prefixed tags, even if they
|
|
are selected by other specified tags e.g. if there are ``standard`` tests which
|
|
are also tagged as ``slow`` you can run all standard tests *except* the slow
|
|
ones:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags 'standard,-slow'
|
|
|
|
When you write a test that does not inherit from the
|
|
:class:`~odoo.tests.BaseCase`, this test will not have the default tags,
|
|
you have to add them explicitly to have the test included in the default test
|
|
suite. This is a common issue when using a simple ``unittest.TestCase`` as
|
|
they're not going to get run:
|
|
|
|
.. code-block:: python
|
|
|
|
import unittest
|
|
from odoo.tests import tagged
|
|
|
|
@tagged('standard', 'at_install')
|
|
class SmallTest(unittest.TestCase):
|
|
...
|
|
|
|
Besides tags you can also specify specific modules, classes or functions to
|
|
test. The full syntax of the format accepted by :option:`--test-tags <odoo-bin --test-tags>`
|
|
is:
|
|
|
|
.. code-block:: text
|
|
|
|
[-][tag][/module][:class][.method]
|
|
|
|
So if you want to test the `stock_account` module, you can use:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags /stock_account
|
|
|
|
If you want to test a specific function with a unique name, it can be specified
|
|
directly:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags .test_supplier_invoice_forwarded_by_internal_user_without_supplier
|
|
|
|
This is equivalent to
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags /account:TestAccountIncomingSupplierInvoice.test_supplier_invoice_forwarded_by_internal_user_without_supplier
|
|
|
|
if the name of the test is unambiguous. Multiple modules, classes and functions
|
|
can be specified at once separated by a `,` like with regular tags.
|
|
|
|
.. _reference/testing/tags:
|
|
|
|
Special tags
|
|
~~~~~~~~~~~~
|
|
|
|
- ``standard``: All Odoo tests that inherit from
|
|
:class:`~odoo.tests.BaseCase` are implicitly tagged standard.
|
|
:option:`--test-tags <odoo-bin --test-tags>` also defaults to ``standard``.
|
|
|
|
That means untagged test will be executed by default when tests are enabled.
|
|
- ``at_install``: Means that the test will be executed right after the module
|
|
installation and before other modules are installed. This is a default
|
|
implicit tag.
|
|
- ``post_install``: Means that the test will be executed after all the modules
|
|
are installed. This is what you want for HttpCase tests most of the time.
|
|
|
|
Note that this is *not exclusive* with ``at_install``, however since you
|
|
will generally not want both ``post_install`` is usually paired with
|
|
``-at_install`` when tagging a test class.
|
|
|
|
Examples
|
|
~~~~~~~~
|
|
|
|
.. important::
|
|
|
|
Tests will be executed only in installed modules. If you're starting from
|
|
a clean database, you'll need to install the modules with the
|
|
:option:`-i <odoo-bin -i>` switch at least once. After that it's no longer
|
|
needed, unless you need to upgrade the module, in which case
|
|
:option:`-u <odoo-bin -u>` can be used. For simplicity, those switches are
|
|
not specified in the examples below.
|
|
|
|
Run only the tests from the sale module:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags /sale
|
|
|
|
Run the tests from the sale module but not the ones tagged as slow:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags '/sale,-slow'
|
|
|
|
Run only the tests from stock or tagged as slow:
|
|
|
|
.. code-block:: console
|
|
|
|
$ odoo-bin --test-tags '-standard, slow, /stock'
|
|
|
|
.. note:: ``-standard`` is implicit (not required), and present for clarity
|
|
|
|
Testing JS code
|
|
===============
|
|
|
|
Testing a complex system is an important safeguard to prevent regressions and to
|
|
guarantee that some basic functionality still works. Since Odoo has a non trivial
|
|
codebase in Javascript, it is necessary to test it.
|
|
|
|
See the :doc:`Unit testing <../frontend/unit_testing>` to learn about the
|
|
various aspect of the front-end testing framework, or jump directly to one of the
|
|
sub-sections:
|
|
|
|
- :doc:`Hoot <../frontend/unit_testing/hoot>`
|
|
|
|
- :doc:`Web test helpers <../frontend/unit_testing/web_helpers>`
|
|
|
|
- :doc:`Mock server <../frontend/unit_testing/mock_server>`
|
|
|
|
.. _reference/testing/integration-testing:
|
|
|
|
Integration Testing
|
|
===================
|
|
|
|
Testing Python code and JS code separately is very useful, but it does not prove that the web client
|
|
and the server work together. In order to do that, we can write another kind of test: tours. A
|
|
tour is a mini scenario of some interesting business flow. It explains a sequence of steps that
|
|
should be followed. The test runner will then create a PhantomJs browser, point it to the proper
|
|
url and simulate the click and inputs, according to the scenario.
|
|
|
|
Writing a test tour
|
|
-------------------
|
|
|
|
Structure
|
|
~~~~~~~~~
|
|
|
|
To write a test tour for `your_module`, start with creating the required files:
|
|
|
|
.. code-block:: text
|
|
|
|
your_module
|
|
├── ...
|
|
├── static
|
|
| └── tests
|
|
| └── tours
|
|
| └── your_tour.js
|
|
├── tests
|
|
| ├── __init__.py
|
|
| └── test_calling_the_tour.py
|
|
└── __manifest__.py
|
|
|
|
You can then:
|
|
|
|
- update :file:`__manifest__.py` to add :file:`your_tour.js` in the assets.
|
|
|
|
.. code-block:: python
|
|
|
|
'assets': {
|
|
'web.assets_tests': [
|
|
'your_module/static/tests/tours/your_tour.js',
|
|
],
|
|
},
|
|
|
|
- update :file:`__init__.py` in the folder :file:`tests` to import :file:`test_calling_the_tour`.
|
|
|
|
.. seealso::
|
|
- :ref:`Assets Bundle <reference/assets_bundle>`
|
|
- :ref:`testing/python`
|
|
|
|
.. _testing/javascript/test:
|
|
|
|
Javascript
|
|
~~~~~~~~~~
|
|
|
|
#. Setup your tour by registering it.
|
|
|
|
.. code-block:: javascript
|
|
|
|
import tour from 'web_tour.tour';
|
|
tour.register('rental_product_configurator_tour', {
|
|
url: '/web', // Here, you can specify any other starting url
|
|
}, [
|
|
// Your sequence of steps
|
|
]);
|
|
|
|
#. Add any step you want.
|
|
|
|
Every step contains at least a trigger. You can either use the `predefined steps
|
|
<{GITHUB_PATH}/addons/web_tour/static/src/tour_service/tour_utils.js#L426>`_ or write your own personalized
|
|
step.
|
|
|
|
Here are some example of steps:
|
|
|
|
.. example::
|
|
|
|
.. code-block:: javascript
|
|
|
|
// First step
|
|
tour.stepUtils.showAppsMenuItem(),
|
|
// Second step
|
|
{
|
|
trigger: '.o_app[data-menu-xmlid="your_module.maybe_your_module_menu_root"]',
|
|
isActive: ['community'], // Optional
|
|
run: "click",
|
|
}, {
|
|
// Third step
|
|
},
|
|
|
|
.. example::
|
|
|
|
.. code-block:: javascript
|
|
|
|
{
|
|
trigger: '.js_product:has(strong:contains(Chair floor protection)) .js_add',
|
|
run: "click",
|
|
},
|
|
|
|
.. example::
|
|
|
|
.. code-block:: javascript
|
|
|
|
{
|
|
isActive: ["mobile", "enterprise"],
|
|
content: "Click on Add a product link",
|
|
trigger: 'a:contains("Add a product")',
|
|
tooltipPosition: "bottom",
|
|
async run(helpers) { //Exactly the same as run: "click"
|
|
helpers.click();
|
|
}
|
|
},
|
|
|
|
Here are some possible arguments for your personalized steps:
|
|
|
|
- **trigger**: Required, Selector/element to ``run`` an action on. The tour will
|
|
wait until the element exists and is visible before ``run``-ing the
|
|
action *on it*.
|
|
- **run**: Optional, Action to perform on the *trigger* element. If no ``run``,
|
|
no action.
|
|
|
|
The action can be:
|
|
|
|
- A function, asynchronous, executed with the trigger's ``Tip`` as
|
|
context (``this``) and the action helpers as parameter.
|
|
- The name of one of the action helpers, which will be run on the
|
|
trigger element:
|
|
|
|
.. rst-class:: o-definition-list
|
|
|
|
``check``
|
|
Ensures that the **trigger** element is checked. This helper is intended
|
|
for `<input[type=checkbox]>` elements only.
|
|
``clear``
|
|
Clears the value of the **trigger** element. This helper is
|
|
intended for `<input>` or `<textarea>` elements only.
|
|
``click``
|
|
Clicks the **trigger** element, performing all the relevant intermediate
|
|
events.
|
|
``dblclick``,
|
|
Same as ``click`` with two repetitions.
|
|
:samp:`drag_and_drop {target}`
|
|
Simulates the dragging of the **trigger** element over to the ``target``.
|
|
:samp:`edit {content}`
|
|
``clear`` the element and then ``fill`` the ``content``.
|
|
:samp:`editor {content}`
|
|
Focus the **trigger** element (wysiwyg) and then ``press`` the ``content``.
|
|
:samp:`fill {content}`
|
|
Focus the **trigger** element and then ``press`` the ``content``. This helper is
|
|
intended for `<input>` or `<textarea>` elements only.
|
|
``hover``
|
|
Performs a hover sequence on the **trigger** element.
|
|
:samp:`press {content}`
|
|
Performs a keyboard event sequence.
|
|
:samp:`range {content}`
|
|
Focus the **trigger** element and set ``content`` as value. This helper is intended
|
|
for `<input[type=range]>` elements only.
|
|
:samp:`select {value}`
|
|
Performs a selection event sequence on **trigger** element. Select the option by its
|
|
``value``. This helper is intended for `<select>` elements only.
|
|
:samp:`selectByIndex {index}`
|
|
Same as ``select`` but select the option by its ``index``. Note that first option has
|
|
index 0.
|
|
:samp:`selectByLabel {label}`
|
|
Same as ``select`` but select the option by its ``label``.
|
|
``uncheck``
|
|
Ensures that the **trigger** element is unchecked. This helper is intended
|
|
for `<input[type=checkbox]>` elements only.
|
|
|
|
|
|
- **isActive**: Optional,
|
|
Activates the step only if all conditions of isActive array are met.
|
|
- Browser is in either **desktop** or **mobile** mode.
|
|
- The tour concerns either **community** or **enterprise** edition.
|
|
- The tour is run in either **auto** (runbot) or **manual** (onboarding) mode.
|
|
- **tooltipPosition**: Optional, ``"top"``, ``"right"``, ``"bottom"``, or
|
|
``"left"``. Where to position the tooltip relative to the **target**
|
|
when running interactive tours.
|
|
- **content**: Optional but recommended, the content of the tooltip in
|
|
interactive tours, also logged to the console so very useful to
|
|
trace and debug automated tours.
|
|
- **timeout**: How long to wait until the step can ``run``, in
|
|
milliseconds, 10000 (10 seconds).
|
|
|
|
.. important::
|
|
|
|
The last step(s) of a tour should always return the client to a
|
|
"stable" state (e.g. no ongoing editions) and ensure all
|
|
side-effects (network requests) have finished running to avoid race
|
|
conditions or errors during teardown.
|
|
|
|
.. seealso::
|
|
- `jQuery documentation about find <https://api.jquery.com/find/>`_
|
|
|
|
Python
|
|
~~~~~~
|
|
|
|
To start a tour from a python test, make the class inherit from
|
|
:class:`~odoo.tests.HTTPCase`, and call `start_tour`:
|
|
|
|
.. code-block:: python
|
|
|
|
def test_your_test(self):
|
|
# Optional Setup
|
|
self.start_tour("/web", "your_tour_name", login="admin")
|
|
# Optional verifications
|
|
|
|
Writing an onboarding tour
|
|
--------------------------
|
|
|
|
Structure
|
|
~~~~~~~~~
|
|
|
|
To write an onboarding tour for `your_module`, start with creating the required files:
|
|
|
|
.. code-block:: text
|
|
|
|
your_module
|
|
├── ...
|
|
├── data
|
|
| └── your_tour.xml
|
|
├── static/src/js/tours/your_tour.js
|
|
└── __manifest__.py
|
|
|
|
You can then update :file:`__manifest__.py` to add :file:`your_tour.js` in the assets and :file:`your_tour.xml` in the data.
|
|
|
|
.. code-block:: python
|
|
|
|
'data': [
|
|
'data/your_tour.xml',
|
|
],
|
|
'assets': {
|
|
'web.assets_backend': [
|
|
'your_module/static/src/js/tours/your_tour.js',
|
|
],
|
|
},
|
|
|
|
Javascript
|
|
~~~~~~~~~~
|
|
|
|
The javascript part is the same as for :ref: `the test tour <testing/javascript/test>`.
|
|
|
|
XML
|
|
~~~
|
|
|
|
When you have your tour in the javascript registry, you can create a record `web_tour.tour` in the xml, like that:
|
|
|
|
.. code-block:: xml
|
|
|
|
<?xml version="1.0" encoding="utf-8"?>
|
|
<odoo>
|
|
<record id="your_tour" model="web_tour.tour">
|
|
<field name="name">your_tour</field>
|
|
<field name="sequence">10</field>
|
|
<field name="rainbow_man_message">Congrats, that was a great tour</field>
|
|
</record>
|
|
</odoo>
|
|
|
|
- `name`: Required, the name must be the same as the one in the
|
|
javascript registry.
|
|
- `sequence`: Optional; determines the order to execute the
|
|
onboarding tours. Defaults to 1000.
|
|
- `url`: Optional; the url where to start the tour. If ``url`` is ``False``,
|
|
take the url from the registry. Defaults to "/odoo".
|
|
- `rainbow_man_message`: Optional; will show the message in the
|
|
rainbow man effect at the completion of the tour. If ``rainbow_man_message`` is ``False``,
|
|
there is no rainbow effect. Defaults to ``<b>Good job!</b> You went through all steps of this tour.``
|
|
|
|
Running onboarding tours
|
|
~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
They can all be started in their sequence order by toggling the :guilabel:`Onboarding` option in the user menu.
|
|
You can run specific onboarding tours by going to the :menuselection:`Settings --> Technical --> User Interface --> Tours`
|
|
and clicking on :guilabel:`Onboarding` or :guilabel:`Testing`.
|
|
|
|
- **Onboarding**: will execute the tour in interactive mode. That means the tour will show what to do and
|
|
wait for interactions from the user.
|
|
- **Testing**: will execute the tour automatically. That means the tour will be executing all the step in
|
|
front of the user.
|
|
|
|
Tour recorder
|
|
~~~~~~~~~~~~~
|
|
|
|
You can also create tours easily with the tour recorder. To do so, click on :guilabel:`Record` on the
|
|
onboarding tours view. When started, this tool will record all your interactions in Odoo.
|
|
|
|
The created tours are flagged in the onboarding tours view as **Custom**. These tours can also
|
|
be exported to a javascript file, ready to be put in your module.
|
|
|
|
Debugging tips
|
|
--------------
|
|
|
|
Observing test tours in a browser
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
There are three ways with different tradeoffs:
|
|
|
|
``watch=True``
|
|
**************
|
|
|
|
When running a tour locally via the test suite, the ``watch=True``
|
|
parameter can be added to the ``browser_js`` or ``start_tour``
|
|
call::
|
|
|
|
self.start_tour("/web", "your_tour_name", watch=True)
|
|
|
|
This will automatically open a Chrome window with the tour being
|
|
run inside it.
|
|
|
|
**Advantages**
|
|
- always works if the tour has Python setup / surrounding code, or multiple steps
|
|
- runs entirely automatically (just select the test which launches the tour)
|
|
- transactional (*should* always be runnable multiple times)
|
|
**Drawbacks**
|
|
- only works locally
|
|
- only works if the test / tour can run correctly locally
|
|
|
|
``debug=True``
|
|
**************
|
|
|
|
When running a tour locally via the test suite, the ``debug=True``
|
|
parameter can be added to the ``browser_js`` or ``start_tour``
|
|
call::
|
|
|
|
self.start_tour("/web", "your_tour_name", debug=True)
|
|
|
|
This will automatically open a fullscreen Chrome window with opened
|
|
devtools and a debugger breakpoint set at the start of the tour. The tour
|
|
is ran with the debug=assets query parameter. When an error is thrown, the
|
|
debugger stops on the exception.
|
|
|
|
**Advantages**
|
|
- Same advantages as mode `watch=True`
|
|
- Easier to debug steps
|
|
**Drawbacks**
|
|
- only works locally
|
|
- only works if the test / tour can run correctly locally
|
|
|
|
Run via browser
|
|
***************
|
|
|
|
Test tours can also be launched via the browser UI by calling
|
|
|
|
.. code-block:: javascript
|
|
|
|
odoo.startTour("tour_name");
|
|
|
|
in the javascript console, or by enabling :ref:`tests mode
|
|
<frontend/framework/tests_debug_mode>` by setting ``?debug=tests`` in
|
|
the URL.
|
|
|
|
**Advantages**
|
|
- easier to run
|
|
- can be used on production or test sites, not just local instances
|
|
- allows running in "Onboarding" mode (manual steps)
|
|
**Drawbacks**
|
|
- harder to use with test tours involving Python setup
|
|
- may not work multiple times depending on tour side-effects
|
|
|
|
.. tip::
|
|
|
|
It's possible to use this method to observe or interact with tours
|
|
which require Python setup:
|
|
|
|
- add a *python* breakpoint before the relevant tour is started
|
|
(``start_tour`` or ``browser_js`` call)
|
|
- when the breakpoint is hit, open the instance in your browser
|
|
- run the tour
|
|
|
|
At this point the Python setup will be visible to the browser, and
|
|
the tour will be able to run.
|
|
|
|
You may want to comment the ``start_tour`` or ``browser_js`` call
|
|
if you also want the test to continue afterwards, depending on the
|
|
tour's side-effects.
|
|
|
|
Screenshots and screencasts during browser_js tests
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
When running tests that use ``HttpCase.browser_js`` from the command line, the Chrome
|
|
browser is used in headless mode. By default, if a test fails, a PNG screenshot is
|
|
taken at the moment of the failure and written in
|
|
|
|
.. code-block:: console
|
|
|
|
'/tmp/odoo_tests/{db_name}/screenshots/'
|
|
|
|
Two new command line arguments were added since Odoo 13.0 to control this behavior:
|
|
:option:`--screenshots <odoo-bin --screenshots>` and :option:`--screencasts <odoo-bin --screencasts>`
|
|
|
|
Introspecting / debugging steps
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
When trying to fix / debug a tour, the screenshots (on failure) are
|
|
not necessarily sufficient. In that case it can be useful to see
|
|
what's happening at some or each step.
|
|
|
|
While this is pretty easy when in an "onboarding" (as they're mostly
|
|
driven explicitly by the user) it's more complicated when running
|
|
"test" tours, or when running tours through the test suite. In that
|
|
case there are two main tricks:
|
|
|
|
- A step property ``break: true,`` in debug mode (debug=True).
|
|
|
|
This adds a debugger breakpoint at the start of the step.
|
|
You can then add your own wherever you need.
|
|
|
|
**Advantages**
|
|
- very simple
|
|
- the tour continues as soon as you resume execution
|
|
**Drawbacks**
|
|
- page interaction is limited as all javascript is blocked
|
|
|
|
- A step property ``pause: true,`` in debug mode (debug=True).
|
|
|
|
The tour will stop at the end of the step. This allows inspecting
|
|
and interacting with the page until the developer is ready to
|
|
resume by typing **play();** in the browser console.
|
|
|
|
**Advantages**
|
|
- allows interacting with the page
|
|
- no useless (for this situation) debugger UI
|
|
|
|
- A step with a ``run() { debugger; }`` action.
|
|
|
|
This can be added to an existing step, or can be a new dedicated
|
|
step. Once the step's **trigger** is matched, the execution will
|
|
stop all javascript execution.
|
|
|
|
**Advantages**
|
|
- simple
|
|
- the tour continues as soon as you resume execution
|
|
**Drawbacks**
|
|
- page interaction is limited as all javascript is blocked
|
|
- the debugger is triggered after trying to find targeted
|
|
element defined in the step.
|
|
|
|
Performance Testing
|
|
===================
|
|
|
|
Query counts
|
|
------------
|
|
|
|
One of the ways to test performance is to measure database queries. Manually, this can be tested with the
|
|
`--log-sql` CLI parameter. If you want to establish the maximum number of queries for an operation,
|
|
you can use the :meth:`~odoo.tests.BaseCase.assertQueryCount` method, integrated in Odoo test classes.
|
|
|
|
.. code-block:: python
|
|
|
|
with self.assertQueryCount(11):
|
|
do_something()
|
|
|
|
.. _qunit: https://qunitjs.com/
|
|
.. _qunit_config.js: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/static/tests/helpers/qunit_config.js#L49
|
|
.. _web.tests_assets: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/views/webclient_templates.xml#L594
|
|
.. _controller: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/controllers/main.py#L637
|
|
.. _test_js.py: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/tests/test_js.py#L13
|
|
.. _test_utils.js: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/static/tests/helpers/test_utils.js
|
|
.. _mock server: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/static/tests/helpers/mock_server.js
|
|
.. _qunit assertions: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/static/tests/helpers/qunit_asserts.js
|
|
.. _createView: https://github.com/odoo/odoo/blob/51ee0c3cb59810449a60dae0b086b49b1ed6f946/addons/web/static/tests/helpers/test_utils_create.js#L267
|