Skip to main content

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


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

testy_pytest_adapter-1.0.2.tar.gz (25.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

testy_pytest_adapter-1.0.2-py3-none-any.whl (23.1 kB view details)

Uploaded Python 3

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

Hashes for testy_pytest_adapter-1.0.2.tar.gz
Algorithm Hash digest
SHA256 8b40e9af247a4817ce268a49cdc4a3550558ddff2b5762c7ecb7458bea1ff4d1
MD5 dfe22217bf8e64a5be3591ffd91150c9
BLAKE2b-256 55adf5631c75ef5c0766e67636ba469e47ebdbbfdad571e122e60c0ce1368c54

See more details on using hashes here.

Provenance

The following attestation bundles were made for testy_pytest_adapter-1.0.2.tar.gz:

Publisher: publish.yml on TheGreatPepix/testy-pytest-adapter

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file testy_pytest_adapter-1.0.2-py3-none-any.whl.

File metadata

File hashes

Hashes for testy_pytest_adapter-1.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 0b9278e14e757d2d8e1e18e76aac4ab06f0d50951b2e59f09f0211febd051569
MD5 214df59806657ccf4132c518a883007c
BLAKE2b-256 5542e2eb79a957185f12e83ab33484c64f9d26d8e3fce0a34ae04713904f06a2

See more details on using hashes here.

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

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page