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_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-воркеров, а уже после этого запускать обычный прогон с отправкой результатов.

Доказательства падений: вложения и ссылки

import testy

def test_login(page):
    page.screenshot(path="fail.png")
    testy.attach("fail.png")
    ...

Когда загружать вложения, задаётся через TESTY_ATTACH:

failure / always / never

По умолчанию используется failure, то есть вложения отправляются только на падениях.

Также адаптер сохраняет в атрибутах результата ссылки из 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.0.tar.gz (23.9 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.0-py3-none-any.whl (21.7 kB view details)

Uploaded Python 3

File details

Details for the file testy_pytest_adapter-1.0.0.tar.gz.

File metadata

  • Download URL: testy_pytest_adapter-1.0.0.tar.gz
  • Upload date:
  • Size: 23.9 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.0.tar.gz
Algorithm Hash digest
SHA256 8593eb4492315be8d91a8c4d402ccfdb9fe4777284fa01a97a71f712bae1ce51
MD5 259f90c327873745dbda9455c90ecb3b
BLAKE2b-256 8316877244d37a0cbddd536f0214f383f9ba1d4829082304d49bb6d70e486d9d

See more details on using hashes here.

Provenance

The following attestation bundles were made for testy_pytest_adapter-1.0.0.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.0-py3-none-any.whl.

File metadata

File hashes

Hashes for testy_pytest_adapter-1.0.0-py3-none-any.whl
Algorithm Hash digest
SHA256 709735922199098200762b6d46dd6faf2e1e43745a390317b8c6ae92e6405961
MD5 1046bf568f1b1b756f81f4ac6c8ecd67
BLAKE2b-256 988476dce800d96bb2be45f55006b95118b838701b694beefa4bf958a365852f

See more details on using hashes here.

Provenance

The following attestation bundles were made for testy_pytest_adapter-1.0.0-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