Захист свого коду за допомогою модульних тестів¶
Важливо
This tutorial is an extension of the Server framework 101 tutorial. Make sure you have
completed it and use the estate
module you have built as a base for the exercises in this
tutorial.
Посилання: Test Framework Odoo: ознайомтеся з передовими методами (Odoo Experience 2020) на YouTube.
Написання тестів є необхідністю з кількох причин. Ось неповний список:
Ensure code will not be broken in the future
Визначте область дії вашого коду
Наведіть приклади випадків використання
Це один із способів технічно задокументувати код
Help your coding by defining your goal before working towards it
Виконання тестів¶
Перш ніж знати, як писати тести, нам потрібно знати, як їх запускати.
$ odoo-bin -h
Usage: odoo-bin [options]
Options:
--version show program's version number and exit
-h, --help show this help message and exit
[...]
Testing Configuration:
--test-file=TEST_FILE
Launch a python test file.
--test-enable Enable unit tests.
--test-tags=TEST_TAGS
Comma-separated list of specs to filter which tests to
execute. Enable unit tests if set. A filter spec has
the format: [-][tag][/module][:class][.method] The '-'
specifies if we want to include or exclude tests
matching this spec. The tag will match tags added on a
class with a @tagged decorator (all Test classes have
'standard' and 'at_install' tags until explicitly
removed, see the decorator documentation). '*' will
match all tags. If tag is omitted on include mode, its
value is 'standard'. If tag is omitted on exclude
mode, its value is '*'. The module, class, and method
will respectively match the module name, test class
name and test method name. Example: --test-tags
:TestClass.test_func,/test_module,external Filtering
and executing the tests happens twice: right after
each module installation/update and at the end of the
modules loading. At each stage tests are filtered by
--test-tags specs and additionally by dynamic specs
'at_install' and 'post_install' correspondingly.
--screencasts=DIR Screencasts will go in DIR/{db_name}/screencasts.
--screenshots=DIR Screenshots will go in DIR/{db_name}/screenshots.
Defaults to /tmp/odoo_tests.
$ # run all the tests of account, and modules installed by account
$ # the dependencies already installed are not tested
$ # this takes some time because you need to install the modules, but at_install
$ # and post_install are respected
$ odoo-bin -i account --test-enable
$ # run all the tests in this file
$ odoo-bin --test-file=addons/account/tests/test_account_move_entry.py
$ # test tags can help you filter quite easily
$ odoo-bin --test-tags=/account:TestAccountMove.test_custom_currency_on_account_1
Інтеграційні боти¶
Примітка
This section is only for Odoo employees and people that are contributing to
github.com/odoo
. We highly recommend having your own CI otherwise.
Під час написання тесту важливо переконатися, що він завжди проходить, коли до вихідного коду вносяться зміни. Щоб автоматизувати це завдання, ми використовуємо практику розробки під назвою Continuous Integration (CI). Ось чому у нас є боти, які виконують усі тести в різні моменти. Незалежно від того, працюєте ви в Odoo чи ні, якщо ви намагаєтесь об’єднати щось усередині odoo/odoo
, odoo/enterprise
, odoo/upgrade
або на odoo.sh, вам доведеться пройти через CI. Якщо ви працюєте над іншим проектом, вам варто подумати про додавання власного CI.
Runbot¶
Довідка: документацію, пов’язану з цією темою, можна знайти в розділі Поширені запитання щодо Runbot.
Більшість тестів виконується на Runbot кожного разу, коли на GitHub надсилається комміт.
Ви можете побачити стан коміту/розгалуження, відфільтрувавши на інформаційній панелі Runbot.
A bundle is created for each branch. A bundle consists of a configuration and batches.
Комплект - це набір збірок, залежно від параметрів комплекту. Комплект вважається зеленим (тобто проходить тести), якщо всі збірки зелені.
Збірка – це коли ми запускаємо сервер. Його можна розділити на саб-збірки. Зазвичай існують збірки для версії спільноти, корпоративної версії (тільки якщо є корпоративна філія, але ви можете примусово створити збірку) і міграція гілки. Збірка зелена, якщо кожна дочірня саб-збірка зелена.
Саб-збірка виконує лише деякі частини того, що виконує повна збірка. Він використовується для прискорення процесу CI. Зазвичай він використовується для розділення тестів після встановлення на 4 паралельні екземпляри. Допоміжна збірка зелена, якщо всі тести проходять успішно та не зареєстровано жодних помилок/попереджень.
Примітка
All tests are run regardless of the modifications done. Correcting a typo in an error message or refactoring a whole module triggers the same tests. All modules will be installed as well. This means something might not work even if the Runbot is green, i.e. your changes depend on a module that the module the changes are in doesn’t depend on.
The localization modules (i.e. country-specific modules) are not installed on Runbot (except the generic one). Some modules with external dependencies can also be excluded.
There is a nightly build running additional tests: module operations, localization, single module installs, multi-builds for nondeterministic bugs, etc. These are not kept in the standard CI to shorten the time of execution.
You can also login to a build built by Runbot. There are 3 users usable: admin
, demo
and
portal
. The password is the same as the login. This is useful to quickly test things on different
versions without having to build it locally. The full logs are also available; these are used for
monitoring.
Robodoo¶
Швидше за все, вам доведеться набратися трохи більше досвіду, перш ніж отримати права викликати robodoo, але все ж є кілька зауважень.
Robodoo is the guy spamming the CI status as tags on your PRs, but he is also the guy that kindly integrates your commits into the main repositories.
When the last batch is green, the reviewer can ask robodoo to merge your PR (it is more
a rebase
than a merge
). It will then go to the mergebot.
Mergebot¶
Mergebot є останнім етапом тестування перед об’єднанням PR.
Він візьме коміти у вашій гілці, яких ще немає на цільовій платформі, ініціює це та повторно запустить тести ще раз, включаючи enterprise версію, навіть якщо ви змінюєте лише щось у community.
Цей крок може бути невдалим із повідомленням про помилку Staging failed
. Це може бути через
недетермінована помилка, яка вже є цільовою. Якщо ви є співробітником Odoo, ви можете перевірити їх тут: https://runbot.odoo.com/runbot/errors
недетерміновану помилку, яку ви представили, але не виявили раніше в CI
несумісність з іншим комітом, об’єднаним прямо перед тим, і тим, що ви намагаєтесь об’єднати
несумісність із enterprise репозиторієм, якщо ви внесли зміни лише в сховище community
Завжди перевіряйте, чи проблема не походить від вас, перш ніж попросити бота для злиття повторити спробу: перебазуйте свою гілку на ціль і повторно запустіть тести локально.
Модулі¶
Оскільки Odoo є модульним, тести також мають бути модульними. Це означає, що тести визначаються в модулі, який додає функціональність, яку ви додаєте, і тести не можуть залежати від функціональності, що надходить із модулів, від яких ваш модуль не залежить.
Довідка: документацію, пов’язану з цією темою, можна знайти в Спеціальні теги.
from odoo.tests.common import TransactionCase
from odoo.tests import tagged
# The CI will run these tests after all the modules are installed,
# not right after installing the one defining it.
@tagged('post_install', '-at_install') # add `post_install` and remove `at_install`
class PostInstallTestCase(TransactionCase):
def test_01(self):
...
@tagged('at_install') # this is the default
class AtInstallTestCase(TransactionCase):
def test_01(self):
...
Якщо поведінку, яку ви хочете перевірити, можна змінити встановленням іншого модуля, вам потрібно переконатися, що встановлено тег at_install
; інакше ви можете використати тег post_install
, щоб пришвидшити CI та переконатися, що він не змінюється, якщо це не повинно бути.
Написання тесту¶
Довідка: документацію, пов’язану з цією темою, можна знайти в Python unittest та Testing Odoo.
Ось кілька речей, які слід взяти до уваги перед написанням тесту
Тести повинні бути незалежними від даних, які зараз містяться в базі даних (включаючи демонстраційні дані)
Тести не повинні впливати на базу даних, залишаючи/змінюючи залишкові дані. Зазвичай це робить тестовий фреймворк, виконуючи відкат. Тому ви ніколи не повинні викликати
cr.commit
у тесті (ані будь-де в бізнес-коді).Для виправлення помилки тест має бути невдалим перед застосуванням виправлення та пройти після.
Не тестуйте те, що вже перевірено деінде; ви можете довіряти ORM. Більшість тестів у бізнес-модулях мають перевіряти лише бізнес-потоки.
Вам не потрібно скидати дані в базу даних.
Примітка
Пам’ятайте, що onchange
застосовується лише у представленнях форм, а не шляхом зміни атрибутів у python. Це також стосується і тестів. Якщо ви хочете емулювати представлення форми, ви можете використовувати odoo.tests.common.Form
.
Тести мають бути в папці tests
у корені вашого модуля. Ім’я кожного тестового файлу має починатися з test_
і імпортуватися в __init__.py
папки test. Ви не повинні імпортувати тестову папку/модуль у __init__.py
модуля.
estate
├── models
│ ├── *.py
│ └── __init__.py
├── tests
│ ├── test_*.py
│ └── __init__.py
├── __init__.py
└── __manifest__.py
All the tests should extend odoo.tests.common.TransactionCase
. You usually define a
setUpClass
and the tests. After writing the setUpClass
, you have an env
available in the
class and can start interacting with the ORM.
Ці тестові класи створено на основі модуля python unittest
.
from odoo.tests.common import TransactionCase
from odoo.exceptions import UserError
from odoo.tests import tagged
# The CI will run these tests after all the modules are installed,
# not right after installing the one defining it.
@tagged('post_install', '-at_install')
class EstateTestCase(TransactionCase):
@classmethod
def setUpClass(cls):
# add env on cls and many other things
super(EstateTestCase, cls).setUpClass()
# create the data for each tests. By doing it in the setUpClass instead
# of in a setUp or in each test case, we reduce the testing time and
# the duplication of code.
cls.properties = cls.env['estate.property'].create([...])
def test_creation_area(self):
"""Test that the total_area is computed like it should."""
self.properties.living_area = 20
self.assertRecordValues(self.properties, [
{'name': ..., 'total_area': ...},
{'name': ..., 'total_area': ...},
])
def test_action_sell(self):
"""Test that everything behaves like it should when selling a property."""
self.properties.action_sold()
self.assertRecordValues(self.properties, [
{'name': ..., 'state': ...},
{'name': ..., 'state': ...},
])
with self.assertRaises(UserError):
self.properties.forbidden_action_on_sold_property()
Примітка
For better readability, split your tests into multiple files depending on the scope of the tests. You can also have a Common class that most of the tests should inherit from; this common class can define the whole setup for the module. For instance, in account.
Exercise
Update the code so no one can:
Create an offer for a sold property
Sell a property with no accepted offers on it
and create tests for both of these cases. Additionally check that selling a property that can be sold is correctly marked as sold after selling it.
Exercise
Хтось постійно порушує скидання Garden Area та Orientation, коли ви знімаєте прапорець у полі Garden. Слідкуйте, щоб це не повторилося.
Порада
Порада: запам’ятайте примітку про Форму
трохи вище.