Skip to main content

Supported pytest and run-orchestration helpers for GridFleet integrations

Project description

GridFleet Testkit

testkit/ is the supported Python integration surface for external pytest/Appium suites that run through GridFleet.

What This Package Owns

  • Stable import root: gridfleet_testkit
  • Supported pytest plugin: gridfleet_testkit.pytest_plugin
  • Supported pytest fixtures: appium_driver, gridfleet_client, device_config, device_test_data, device_handle, gridfleet_worker_id
  • Supported public Appium helpers:
    • build_appium_options
    • create_appium_driver
    • get_connection_target_from_driver
    • get_device_config_for_driver
    • get_device_test_data_for_driver
  • Supported public client helpers:
    • GridFleetClient
    • HeartbeatThread
    • register_run_cleanup
  • Supported public allocation/session helpers:
    • AllocatedDevice
    • UnavailableInclude
    • build_error_session_payload
    • hydrate_allocated_device
    • hydrate_allocated_device_from_driver
    • resolve_device_handle_from_driver
  • Supported public result types:
    • CooldownResult
    • CooldownSetResult
    • CooldownEscalatedResult
  • Supported public exceptions:
    • UnknownIncludeError
    • ReserveCapabilitiesUnsupportedError
  • Manual hardware examples under testkit/examples/

What It Does Not Own

  • Appium server installation or host-level driver setup
  • Selenium Grid lifecycle
  • Device registration, verification, or readiness setup
  • CI orchestration beyond the documented client helpers

The supported contract is the installable package and documented import pattern. The example scripts are onboarding aids, not CI-backed conformance tests.

Install

From PyPI:

pip install "gridfleet-testkit"

From a local checkout:

uv pip install -e ./testkit

From a copied testkit/ directory inside another repository:

uv pip install -e ./testkit

From a Git checkout or VCS URL that contains this package:

uv pip install "git+https://github.com/<org>/<repo>.git#subdirectory=testkit"

The package supports Python 3.10 and newer. Appium-Python-Client is installed as a runtime dependency because the pytest fixtures create real Appium sessions.

Environment

Variable Default Meaning
GRID_URL http://localhost:4444 Selenium Grid hub URL used by the pytest Appium fixture
GRIDFLEET_API_URL http://localhost:8000/api GridFleet API base used for session reporting, config lookup, run helpers, and driver-pack catalog lookup
GRIDFLEET_TESTKIT_USERNAME unset Machine-auth username sent as HTTP Basic auth on every API call. Required when the manager runs with GRIDFLEET_AUTH_ENABLED=true. Use the same value as the manager's GRIDFLEET_MACHINE_AUTH_USERNAME.
GRIDFLEET_TESTKIT_PASSWORD unset Machine-auth password sent as HTTP Basic auth on every API call. Required when the manager runs with GRIDFLEET_AUTH_ENABLED=true. Use the same value as the manager's GRIDFLEET_MACHINE_AUTH_PASSWORD.
GRIDFLEET_TESTKIT_PACK_ID unset Optional default driver pack id for Appium option building
GRIDFLEET_TESTKIT_PLATFORM_ID unset Optional default platform id for Appium option building
GRIDFLEET_RUN_ID free Optional run id injected into Appium capabilities as gridfleet:run_id. The pytest plugin sets this automatically when sessions are created inside a reserved run.

The package assumes a running GridFleet API, a reachable Selenium Grid hub, and platform-specific Appium driver setup on the registered hosts. When auth is disabled on the manager, leave GRIDFLEET_TESTKIT_USERNAME / GRIDFLEET_TESTKIT_PASSWORD unset and the testkit will send no Authorization header.

Pytest Plugin

Load the supported plugin from your test project:

pytest_plugins = ["gridfleet_testkit.pytest_plugin"]

Minimal usage:

import pytest

@pytest.mark.parametrize(
    "appium_driver",
    [{"pack_id": "appium-uiautomator2", "platform_id": "android_mobile"}],
    indirect=True,
)
def test_session_starts(appium_driver):
    assert appium_driver.session_id is not None

The plugin resolves pack_id and platform_id against the enabled driver-pack catalog, then injects Appium platformName, appium:automationName, appium:platform, and gridfleet:testName.

When exactly one enabled pack provides a platform id, platform_id alone is accepted. For environment-portable tests, set GRIDFLEET_TESTKIT_PACK_ID and GRIDFLEET_TESTKIT_PLATFORM_ID, then parametrize with {}.

If you need raw Appium control instead, omit pack_id and platform_id, then pass platformName as a normal capability key.

Plugin Lifecycle

  • Creates an Appium session through GRID_URL
  • Injects gridfleet:testName with the pytest test name
  • Injects gridfleet:run_id when a GRIDFLEET_RUN_ID environment variable is present (for example, inside a reserved run)
  • Reports final session status back to GRIDFLEET_API_URL
  • Exposes device_config for post-session config lookup using the runtime connection target
  • Exposes device_test_data for post-session operator-attached test data using the runtime connection target
  • Exposes device_handle for fetching the canonical manager device row using the runtime connection target
  • Exposes gridfleet_worker_id which returns the pytest-xdist worker id, or "controller" for non-worker processes
  • Relies on manager-owned runtime isolation for Appium driver sub-ports and XCUITest build paths

If Appium driver creation fails before a Grid session exists, the pytest fixture registers a device-less terminal error session with an error-<uuid> session id, attempted capabilities, requested pack/platform metadata when available, and exception details, then re-raises the original exception. These rows make setup failures visible in the GridFleet Sessions view.

Direct Appium Usage

If you need to create a driver outside pytest, use the public Appium helpers:

from gridfleet_testkit import create_appium_driver, get_device_config_for_driver

driver = create_appium_driver(
    pack_id="appium-uiautomator2",
    platform_id="firetv_real",
    test_name="manual-smoke",
)

try:
    assert driver.session_id is not None
    device_config = get_device_config_for_driver(driver)
finally:
    driver.quit()

create_appium_driver(...) reuses the same driver-pack catalog resolver as the pytest fixture. Managed nodes still get their host-scoped runtime allocations from the manager, so callers should not hard-code systemPort, chromedriverPort, mjpegServerPort, wdaLocalPort, or derivedDataPath. get_device_config_for_driver(...) is the non-pytest equivalent of the device_config fixture. If you only need the options object, use build_appium_options(...).

Client Helpers

Helper Purpose
GridFleetClient.list_devices(*, pack_id=None, status=None, host_id=None, ...) List devices using backend keyword filters (pack_id, platform_id, status, host_id, connection_target, tags, ...)
GridFleetClient.get_device(device_id) Fetch one full device detail row by backend device id
GridFleetClient.get_device_config(connection_target) Look up a device by runtime connection target and fetch its config
GridFleetClient.get_device_by_connection_target(connection_target) Fetch one device detail row by runtime connection target
GridFleetClient.get_device_capabilities(device_id) Fetch current Appium capability metadata for a device
GridFleetClient.get_device_test_data(device_id) Fetch operator-attached free-form test_data for a device
GridFleetClient.get_run(run_id) Fetch one run detail row by backend run id
GridFleetClient.replace_device_test_data(device_id, body) Replace test_data with the supplied object
GridFleetClient.merge_device_test_data(device_id, body) Deep-merge into device test_data
GridFleetClient.resolve_device_id_by_connection_target(connection_target) Resolve the backend device id for a runtime connection target
GridFleetClient.get_driver_pack_catalog() Fetch enabled driver-pack catalog data for Appium platform selection
GridFleetClient.reserve_devices(...) Create a run/reservation and return the manager response
GridFleetClient.signal_ready(run_id) Signal that a run is ready
GridFleetClient.signal_active(run_id) Move a run to active
GridFleetClient.heartbeat(run_id) Send a run heartbeat and read current state
GridFleetClient.report_preparation_failure(run_id, device_id, message, source="ci_preparation") Exclude one reserved device after setup fails
GridFleetClient.register_session(fields) Register a Grid/Appium session with optional requested capability metadata
GridFleetClient.register_session_from_driver(driver, fields) Extract session id and capabilities from an Appium driver and register the session
GridFleetClient.notify_session_finished(session_id) Tell the manager the WebDriver session has ended
GridFleetClient.update_session_status(session_id, status) Report final session status
GridFleetClient.complete_run(run_id) Complete a run
GridFleetClient.cancel_run(run_id) Cancel a run
GridFleetClient.cooldown_device(run_id, device_id, reason=..., ttl_seconds=...) Exclude a reserved device from the run with a cooldown TTL
GridFleetClient.start_heartbeat(run_id, interval=30) Start a background heartbeat thread
build_error_session_payload(fields) Build a /api/sessions payload for driver-creation failures without importing pytest
hydrate_allocated_device(device_handle, run_id, client) Combine a device handle with optional device config and live capabilities
hydrate_allocated_device_from_driver(allocated, driver, client) Return a new allocated-device object with capabilities from a running driver
resolve_device_handle_from_driver(driver, client) Resolve the assigned manager device row from a running Appium session
get_device_test_data_for_driver(driver, gridfleet_client=None) Fetch test_data for a live Appium driver
register_run_cleanup(client, run_id, heartbeat_thread=None) Register atexit cleanup callable and return it; stops the heartbeat thread on exit but does not complete or cancel the run by default

Targeting Devices by Tag

GridFleet injects device tags into Grid node stereotypes as appium:gridfleet:tag:<key> capabilities, so Selenium Grid can route sessions to devices matching specific tags.

@pytest.mark.parametrize(
    "appium_driver",
    [
        {
            "pack_id": "appium-uiautomator2",
            "platform_id": "android_mobile",
            "appium:gridfleet:tag:screen_type": "4k",
        }
    ],
    indirect=True,
)
def test_4k_display(appium_driver):
    ...

The same capability works for free sessions:

driver = create_appium_driver(
    pack_id="appium-uiautomator2",
    platform_id="android_mobile",
    capabilities={"appium:gridfleet:tag:screen_type": "4k"},
)

When an operator edits device tags, GridFleet marks the device for re-verification. The next verification restarts the Appium node and re-registers it with the updated Grid stereotype.

Worker Identity

worker_id is an arbitrary string used for reservation telemetry and run attribution. For pytest-xdist, pass request.config.workerinput["workerid"] from worker processes; values are normally gw0, gw1, and so on. For controller-only flows, use "controller" or a stable hostname. For custom schedulers, use a UUID or job-specific worker name.

Reservation Flow

GridFleet runs are grid-routed: once devices are reserved, the manager tags matching Grid nodes with the run id, and Selenium Grid routes new Appium sessions to those nodes automatically via the gridfleet:run_id capability. There are no per-worker claim or release calls.

from gridfleet_testkit import GridFleetClient, register_run_cleanup

client = GridFleetClient("http://manager-ip:8000/api")

run = client.reserve_devices(
    name="my-test-run",
    requirements=[
        {
            "pack_id": "appium-uiautomator2",
            "platform_id": "firetv_real",
            "os_version": "8",
            "allocation": "all_available",
            "min_count": 1,
        }
    ],
    ttl_minutes=45,
    created_by="local-dev",
)

run_id = run["id"]
worker_count = len(run["devices"])
heartbeat_thread = client.start_heartbeat(run_id, interval=30)
cleanup = register_run_cleanup(client, run_id, heartbeat_thread)
# cleanup() runs at process exit; call client.complete_run(run_id) on success
# or client.cancel_run(run_id) on failure to set the run state explicitly.

# If one reserved device fails setup:
client.report_preparation_failure(
    run_id,
    device_id="device-123",
    message="Driver bootstrap timed out during CI setup",
    source="local-dev",
)

client.signal_ready(run_id)
client.signal_active(run_id)

Use count for exact reservations. Use allocation: "all_available" when CI should reserve every currently eligible matching device and size its worker pool from len(run["devices"]).

For pytest-xdist controller/worker orchestration, see Testkit xdist recipe. The recipe is copyable guidance, not a public testkit abstraction.

Cooling Down an Unstable Device

If a reserved device becomes unstable during a test, you can put it on cooldown so it is excluded from the run for a TTL. If the same device is cooled down too many times in the same run, it is escalated to maintenance automatically.

from gridfleet_testkit import GridFleetClient

client = GridFleetClient()

result = client.cooldown_device(
    run_id="run-123",
    device_id="device-456",
    reason="Connection dropped mid-test",
    ttl_seconds=120,
)

if result["status"] == "cooldown_set":
    print(f"Device on cooldown until {result['excluded_until']}")
elif result["status"] == "maintenance_escalated":
    print(f"Escalated after {result['cooldown_count']} cooldowns")

The manager enforces a maximum TTL via the general.device_cooldown_max_sec setting. The default is 3600 seconds. An httpx.HTTPStatusError with status 422 is raised if ttl_seconds exceeds the maximum.

Allocated Device Hydration

Use hydrate_allocated_device(...) immediately after reserving a run when a custom plugin needs a stable object instead of raw device-handle JSON.

from gridfleet_testkit import GridFleetClient, hydrate_allocated_device, resolve_device_handle_from_driver

client = GridFleetClient()
run_id = "run-123"

# After creating an Appium session, resolve the assigned device handle
device_handle = resolve_device_handle_from_driver(driver, client=client)
allocated = hydrate_allocated_device(device_handle, run_id=run_id, client=client)

assert allocated.device_id == device_handle["device_id"]
assert allocated.platform_name in {"Android", "iOS", "tvOS", "Roku"}

The helper fetches static device config by default when connection_target is present. It fetches live capabilities only when fetch_capabilities=True. Pass fetch_test_data=True to also populate allocated.test_data. The test_data field is also available directly from the reserve response when the manager inlines it.

Run Cleanup Policy

register_run_cleanup(...) registers an atexit cleanup callable and returns it. By default it stops the heartbeat thread but does not complete or cancel the run, because process exit alone does not prove test success. Prefer explicit client.complete_run(run_id) after successful orchestration and client.cancel_run(run_id) for known failures. Pass on_exit="complete" or on_exit="cancel" only when that policy is correct for your script. Signal handlers are opt-in with install_signal_handlers=True; signal cleanup defaults to cancellation.

Device Test Data

The device_test_data fixture returns the operator-attached free-form test_data for the device assigned to the current test:

def test_uses_operator_data(appium_driver, device_test_data):
    assert "account" in device_test_data

Outside of pytest, use the client directly:

test_data = client.get_device_test_data(device_id)

Or use the driver helper:

from gridfleet_testkit import get_device_test_data_for_driver

test_data = get_device_test_data_for_driver(driver)

Errors and Result Types

  • UnknownIncludeError(ValueError): raised when the backend rejects one or more ?include= keys. Exposes values with the rejected key names. The ValueError base is part of the contract.
  • ReserveCapabilitiesUnsupportedError(ValueError): raised when a reserve-time include request contains "capabilities", which is not supported at reserve time. The ValueError base is part of the contract.
  • CooldownResult: union response type from cooldown_device, with status equal to "cooldown_set" or "maintenance_escalated". CooldownSetResult and CooldownEscalatedResult are the concrete TypedDict variants.

Reduced HTTP Round-trips on Reserve

The manager can inline device config and live capabilities into the reserve_devices response, eliminating per-worker follow-up GETs.

client = GridFleetClient()
run = client.reserve_devices(
    name="my-test-run",
    requirements=[...],
    include=("config", "capabilities"),
)
for device_handle in run["devices"]:
    allocated = hydrate_allocated_device(device_handle, run_id=run["id"], client=client)
    # zero follow-up GETs; allocated.config / allocated.live_capabilities populated inline

device_config and inline config payloads are returned verbatim from the manager. The testkit does not perform client-side secret masking or reveal toggles. Protect device config with manager authentication, operator access control, and your lab's secret-handling policy.

reserve_devices accepts include=("config",) only — include=("capabilities",) raises ReserveCapabilitiesUnsupportedError client-side because reserve-time capabilities are not yet device-bound. Pass include= on the per-worker hydrate_allocated_device call instead if you need capabilities after sessions are running.

include= must be a sequence of strings (tuple or list) — order is preserved in the emitted query parameter. Passing a bare string like include="config" raises TypeError to avoid silently splitting the value into characters.

hydrate_allocated_device accepts device-handle payloads such as reserve_response["devices"] entries or rows returned by get_device_by_connection_target.

Examples

Baseline screenshot examples:

  • examples/test_android_mobile_screenshot.py
  • examples/test_android_tv_screenshot.py
  • examples/test_firetv_screenshot.py
  • examples/test_ios_simulator_screenshot.py
  • examples/test_tvos_screenshot.py
  • examples/test_roku_screenshot.py

Advanced example:

  • examples/test_roku_sideload_screenshot.py

The baseline examples share the same flow:

  1. Create a session through Selenium Grid
  2. Print the resolved connection context
  3. Save a screenshot
  4. Assert that the screenshot file exists and is non-empty

Platform Notes

  • Android Mobile / Android TV / Fire TV:
    • require the UiAutomator2 driver
    • rely on Grid routing hints generated from GridFleet metadata
  • Fire TV:
    • baseline example supports optional appium:os_version filtering when you need a specific Fire OS release
  • iOS simulator:
    • baseline example intentionally targets the simulator lane with appium:device_type=simulator
  • tvOS:
    • baseline example intentionally targets a real device and assumes the host already satisfies XCUITest and WebDriverAgent prerequisites
  • Roku:
    • screenshot examples install and activate the bundled sample dev app before capture
    • both Roku examples depend on Roku dev credentials

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

gridfleet_testkit-0.9.1.tar.gz (170.8 kB view details)

Uploaded Source

Built Distribution

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

gridfleet_testkit-0.9.1-py3-none-any.whl (23.2 kB view details)

Uploaded Python 3

File details

Details for the file gridfleet_testkit-0.9.1.tar.gz.

File metadata

  • Download URL: gridfleet_testkit-0.9.1.tar.gz
  • Upload date:
  • Size: 170.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for gridfleet_testkit-0.9.1.tar.gz
Algorithm Hash digest
SHA256 1a39d90369b765b2b782f2f1126043d16ab3aea639c8ccb5568c0ed40578d327
MD5 aeee8ab44156afa37320e8041e8e4bf1
BLAKE2b-256 7886c132374f79755edc0f89caf66d6676335c75397503d1555916e1fdf6d89e

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridfleet_testkit-0.9.1.tar.gz:

Publisher: release-please.yml on quidow/gridfleet

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

File details

Details for the file gridfleet_testkit-0.9.1-py3-none-any.whl.

File metadata

File hashes

Hashes for gridfleet_testkit-0.9.1-py3-none-any.whl
Algorithm Hash digest
SHA256 67da3e1f04173f6f60d8a6a79d19df9f97a3a927d669be661068cf166c6c9fef
MD5 9bc2234bf5380950663cda6879df533d
BLAKE2b-256 e3f3114c8186cd3afbe768c78e7869c9a8ae9aeb207b62d929bf1a20fc52c8ab

See more details on using hashes here.

Provenance

The following attestation bundles were made for gridfleet_testkit-0.9.1-py3-none-any.whl:

Publisher: release-please.yml on quidow/gridfleet

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