Pytest plugin for sync Tests with TestY TMS
Project description
testy-pytest-adapter
Pytest-плагин, который после прогона отправляет результаты автотестов в TestY TMS: в нужный тест-план, рядом с ручными тестами, с историей запусков и связкой «кейс ↔ автотест».
Зачем это нужно
Обычно автотесты и TMS живут отдельно. Результаты остаются в Allure или JUnit, каждый новый прогон перетирает предыдущий, а статусы в TMS приходится обновлять руками.
Этот адаптер решает эту проблему:
- сопоставляет pytest-тест с тест-кейсом в TestY и отправляет результат
(
passed,failed,skipped,broken) с комментарием и трейсбеком; - записывает результаты в конкретный тест-план и сохраняет историю прогонов;
- может создавать недостающие кейсы, наборы и планы из кода по разметке Allure;
- добавляет к результату логи и ссылки на CI-джобу.
Это именно мост между автотестами и TMS, а не замена Allure. Allure по-прежнему можно использовать для подробного разбора прогона.
Установка
pip install testy-pytest-adapter
Плагин подхватывается pytest автоматически. Пока не передан флаг --testy
или не включён TESTY_ENABLED / testy_enabled, он ничего не делает.
Быстрый старт
pytest --testy \
--testy-url="https://testy.example.com" \
--testy-token="$TESTY_TOKEN" \
--testy-project=1 \
--testy-root-name="Autotests"
После прогона адаптер найдёт подходящие кейсы и запишет результаты в план 1234.
Как сопоставляются тест и кейс
Адаптер берёт стабильный идентификатор теста из pytest nodeid.
По умолчанию суффикс параметризации [...] отрезается.
Пример nodeid:
tests/api/login_test.py::TestLogin::test_login
Дальше адаптер ищет кейс, у которого в атрибутах есть automation_id с таким же значением:
cases/?case_attributes={"automation_id":"..."}
То есть id кейса в коде прописывать не нужно. Достаточно, чтобы у кейса в TestY был атрибут
automation_id с nodeid теста.
Заполнить этот атрибут можно вручную или автоматически через --testy-sync.
Название ключа настраивается через testy_automation_key.
Конфигурация
Значения берутся в таком порядке:
CLI → переменные окружения → pytest.ini → дефолт
Токен намеренно не читается из pytest.ini. Его лучше передавать через переменную окружения
или CLI:
--testy-token / TESTY_TOKEN
Несекретные настройки удобно хранить в pytest.ini или в [tool.pytest.ini_options]
в pyproject.toml.
[pytest]
testy_enabled = true
testy_url = https://testy.example.com
testy_project_id = 4
testy_root_name = Autotests
; самоподписанный сертификат на стенде:
testy_insecure = true
; имена статусов в проекте могут быть локализованы:
testy_status_passed = Пройден
testy_status_failed = Провален
testy_status_skipped = Пропущен
testy_status_broken = Сломан
testy_status_untested = Untested
Тогда для запуска достаточно передать токен через окружение:
TESTY_TOKEN=... pytest --testy --testy-root-name=Autotests
Основные опции
| CLI | Переменная окружения | pytest.ini | Назначение |
|---|---|---|---|
--testy |
TESTY_ENABLED |
testy_enabled |
включить отправку в TestY |
--testy-url |
TESTY_URL |
testy_url |
базовый адрес TestY |
--testy-token |
TESTY_TOKEN |
- | токен доступа |
--testy-project |
TESTY_PROJECT_ID |
testy_project_id |
id проекта |
--testy-plan |
TESTY_PLAN_ID |
testy_plan_id |
id корневого тест-плана |
--testy-root-name |
TESTY_ROOT_NAME |
testy_root_name |
имя корня, если план нужно найти или создать по имени |
--testy-suite |
TESTY_SUITE_ID |
testy_suite_id |
корневой набор для автосоздания кейсов |
--testy-sync |
- | - | создать кейсы и структуру перед прогоном |
| - | TESTY_AUTOMATION_KEY |
testy_automation_key |
ключ атрибута для матчинга, по умолчанию automation_id |
| - | TESTY_KEEP_PARAMS |
testy_keep_params |
не срезать [param], каждая параметризация будет отдельным кейсом |
| - | TESTY_OVERRIDE_CASES |
testy_override_cases |
перезаписывать шаги существующего кейса, по умолчанию true |
| - | TESTY_ATTACH |
testy_attach |
когда загружать вложения: failure, always, never |
| - | TESTY_ATTACH_BYTES |
testy_attach_bytes |
захватывать allure.attach из памяти (скриншоты), по умолчанию true |
| - | TESTY_AUTH_SCHEME |
testy_auth_scheme |
схема авторизации: Token или Bearer |
| - | TESTY_INSECURE |
testy_insecure |
отключить проверку TLS-сертификата |
| - | TESTY_STATUS_PASSED и остальные статусы |
testy_status_* |
реальные имена статусов проекта |
Авторизация и статусы
Для авторизации используется TTL-токен из TestY:
POST /api/token/obtain/
Имена статусов в TestY могут отличаться от стандартных. Например, в проекте они могут
называться Пройден, Провален, Пропущен.
Адаптер сначала приводит результат pytest к одному из канонических статусов:
Passed / Failed / Skipped / Broken / Untested
Затем эти статусы можно замапить на реальные имена проекта:
TESTY_STATUS_PASSED=Пройден
TESTY_STATUS_FAILED=Провален
Или задать их в pytest.ini:
testy_status_passed = Пройден
testy_status_failed = Провален
Автосоздание кейсов и структуры
Чтобы не заполнять automation_id вручную, можно запустить отдельный sync-шаг.
Он создаст недостающие кейсы и добавит их в план.
Тесты при этом не выполняются, запуск идёт через --collect-only:
pytest --collect-only --testy --testy-sync \
--testy-root-name=Autotests \
--testy-url="$TESTY_URL" \
--testy-token="$TESTY_TOKEN" \
--testy-project=4
Дерево суитов и тест-планов в TestY строится из разметки Allure:
| Allure в коде | В TestY |
|---|---|
@allure.parent_suite("API") / @allure.suite("/api/login") / @allure.sub_suite(...) |
вложенное дерево TestSuite и TestPlan под корнем |
@allure.title("GET /api/login") |
имя TestCase |
Без Allure тоже работает. Allure не обязателен и даже не должен быть установлен.
Если у теста нет @allure.title, имя кейса берётся из имени pytest-теста:
test_login
test_login[case1]
Если нет @allure.suite, @allure.parent_suite или @allure.sub_suite, дерево не строится.
Кейсы создаются плоско под корнем, который задан через --testy-suite или --testy-root-name.
Матчинг при этом всё равно работает по nodeid → automation_id.
Allure нужен только для более читаемых имён и структуры.
Sync идемпотентный: повторный запуск находит уже созданные сущности и не плодит дубли. Лучше запускать его отдельным шагом, без xdist-воркеров, а уже после этого запускать обычный прогон с отправкой результатов.
Доказательства падений: вложения и ссылки
Вложения к результату можно добавить двумя способами.
Через testy.attach — явно указать локальный файл:
import testy
def test_login(page):
page.screenshot(path="fail.png")
testy.attach("fail.png")
...
Через Allure — если установлен allure-pytest, адаптер сам подхватывает любые
allure.attach(...) и allure.attach.file(...) и грузит их к результату. Отдельно
вызывать testy.attach не нужно — в том числе если скриншот прикрепляется из хука
pytest_runtest_makereport (автоскриншот на падении):
@pytest.hookimpl(hookwrapper=True)
def pytest_runtest_makereport(item):
outcome = yield
report = outcome.get_result()
if report.when == "call" and not report.passed:
allure.attach(page.screenshot(), attachment_type=allure.attachment_type.PNG)
Картинки (image/*) дополнительно встраиваются прямо в текст комментария результата.
Когда загружать вложения, задаётся через TESTY_ATTACH:
failure / always / never
По умолчанию используется failure, то есть вложения отправляются только на падениях.
Захват allure.attach() можно отключить, оставив только файловые вложения:
TESTY_ATTACH_BYTES=false # или testy_attach_bytes = false в pytest.ini
Также адаптер сохраняет в атрибутах результата ссылки из GitLab CI:
CI_PIPELINE_URL
CI_JOB_URL
Шаги Allure
Если установлен allure-pytest, адаптер забирает дерево выполненных allure.step
и синхронизирует его в кейс как структурные шаги.
Перезапись шагов существующего кейса управляется настройкой:
testy_override_cases
По умолчанию значение true: кейсы поддерживаются в актуальном состоянии относительно кода.
Если Allure не установлен, этот механизм просто не используется.
Параметризованные тесты
По умолчанию все варианты одного параметризованного теста схлопываются в один кейс.
Суффикс параметризации [...] отрезается.
Итоговый статус выбирается как худший из всех вариантов. Например, если один вариант упал,
весь кейс будет отмечен как failed. В комментарий добавляется краткая сводка.
Если нужно, чтобы каждый параметр стал отдельным кейсом, включите:
TESTY_KEEP_PARAMS=1
Пример для GitLab CI
testy_run:
stage: test
rules:
- if: '$CI_PIPELINE_SOURCE == "trigger" && $TESTY_PLAN_ID'
script:
- pip install -r requirements.txt testy-pytest-adapter
- >
pytest -v
--testy
--testy-url="$TESTY_URL"
--testy-token="$TESTY_TOKEN"
--testy-project="$TESTY_PROJECT_ID"
--testy-plan="$TESTY_PLAN_ID"
--alluredir=./allure-results
tests
artifacts:
when: always
paths:
- ./allure-results
TESTY_* удобно передавать как переменные пайплайна.
TESTY_TOKEN лучше хранить как masked CI/CD-переменную репозитория.
Поведение и гарантии
- Ошибки отправки результатов только логируются.
- Прогон тестов не падает из-за проблем с TestY.
- При запуске через
pytest-xdistворкеры собирают результаты, а запись в TestY выполняет только контроллер в конце прогона.
Лицензия
MIT - см. файл LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file testy_pytest_adapter-1.0.2.tar.gz.
File metadata
- Download URL: testy_pytest_adapter-1.0.2.tar.gz
- Upload date:
- Size: 25.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8b40e9af247a4817ce268a49cdc4a3550558ddff2b5762c7ecb7458bea1ff4d1
|
|
| MD5 |
dfe22217bf8e64a5be3591ffd91150c9
|
|
| BLAKE2b-256 |
55adf5631c75ef5c0766e67636ba469e47ebdbbfdad571e122e60c0ce1368c54
|
Provenance
The following attestation bundles were made for testy_pytest_adapter-1.0.2.tar.gz:
Publisher:
publish.yml on TheGreatPepix/testy-pytest-adapter
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
testy_pytest_adapter-1.0.2.tar.gz -
Subject digest:
8b40e9af247a4817ce268a49cdc4a3550558ddff2b5762c7ecb7458bea1ff4d1 - Sigstore transparency entry: 1732637462
- Sigstore integration time:
-
Permalink:
TheGreatPepix/testy-pytest-adapter@214804b436691d8f9f457b35f739ed1739583909 -
Branch / Tag:
refs/tags/v1.0.2 - Owner: https://github.com/TheGreatPepix
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@214804b436691d8f9f457b35f739ed1739583909 -
Trigger Event:
push
-
Statement type:
File details
Details for the file testy_pytest_adapter-1.0.2-py3-none-any.whl.
File metadata
- Download URL: testy_pytest_adapter-1.0.2-py3-none-any.whl
- Upload date:
- Size: 23.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
0b9278e14e757d2d8e1e18e76aac4ab06f0d50951b2e59f09f0211febd051569
|
|
| MD5 |
214df59806657ccf4132c518a883007c
|
|
| BLAKE2b-256 |
5542e2eb79a957185f12e83ab33484c64f9d26d8e3fce0a34ae04713904f06a2
|
Provenance
The following attestation bundles were made for testy_pytest_adapter-1.0.2-py3-none-any.whl:
Publisher:
publish.yml on TheGreatPepix/testy-pytest-adapter
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
testy_pytest_adapter-1.0.2-py3-none-any.whl -
Subject digest:
0b9278e14e757d2d8e1e18e76aac4ab06f0d50951b2e59f09f0211febd051569 - Sigstore transparency entry: 1732637532
- Sigstore integration time:
-
Permalink:
TheGreatPepix/testy-pytest-adapter@214804b436691d8f9f457b35f739ed1739583909 -
Branch / Tag:
refs/tags/v1.0.2 - Owner: https://github.com/TheGreatPepix
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@214804b436691d8f9f457b35f739ed1739583909 -
Trigger Event:
push
-
Statement type: