Python client for Appium
Project description
Appium Python Client
An extension library for adding WebDriver Protocol and Appium commands to the Selenium Python language binding for use with the mobile testing framework Appium.
Getting the Appium Python client
There are three ways to install and use the Appium Python client.
-
Install from PyPi, as 'Appium-Python-Client'.
pip install Appium-Python-Client
You can see the history from here
-
Install from source, via PyPi. From 'Appium-Python-Client', download and unarchive the source tarball (Appium-Python-Client-X.X.tar.gz).
tar -xvf Appium-Python-Client-X.X.tar.gz cd Appium-Python-Client-X.X python setup.py install
-
Install from source via GitHub.
git clone git@github.com:appium/python-client.git cd python-client python setup.py install
Compatibility Matrix
| Appium Python Client | Selenium binding | Python version |
|---|---|---|
5.1.1+ |
4.26.0+ |
3.9+ |
4.5.0 - 5.1.0 |
4.26.0 - 4.31.0 |
3.9+ |
4.3.0 - 4.4.0 |
4.26.0 - 4.31.0 |
3.8+ |
3.0.0 - 4.2.1 |
4.12.0 - 4.25.0 |
3.8+ |
2.10.0 - 2.11.1 |
4.1.0 - 4.11.2 |
3.7+ |
2.2.0 - 2.9.0 |
4.1.0 - 4.9.0 |
3.7+ |
2.0.0 - 2.1.4 |
4.0.0 |
3.7+ |
1.0.0 - 1.3.0 |
3.x |
3.7+ |
0.52 and below |
3.x |
2.7, 3.4 - 3.7 |
The Appium Python Client depends on Selenium Python binding, thus the Selenium Python binding update might affect the Appium Python Client behavior. For example, some changes in the Selenium binding could break the Appium client.
Note We strongly recommend you manage dependencies with version management tools such as uv to keep compatible version combinations.
Quick migration guide from v4 to v5
- This change affects only for users who specify
keep_alive,direct_connectionandstrict_sslarguments forwebdriver.Remote:- Please use
AppiumClientConfigasclient_configargument similar to how it is specified below:SERVER_URL_BASE = 'http://127.0.0.1:4723' # before driver = webdriver.Remote( SERVER_URL_BASE, options=UiAutomator2Options().load_capabilities(desired_caps), direct_connection=True, keep_alive=False, strict_ssl=False ) # after from appium.webdriver.client_config import AppiumClientConfig client_config = AppiumClientConfig( remote_server_addr=SERVER_URL_BASE, direct_connection=True, keep_alive=False, ignore_certificates=True, ) driver = webdriver.Remote( options=UiAutomator2Options().load_capabilities(desired_caps), client_config=client_config )
- Note that you can keep using
webdriver.Remote(url, options=options, client_config=client_config)format as well. In such case theremote_server_addrargument ofAppiumClientConfigconstructor would have priority over theurlargument ofwebdriver.Remoteconstructor.
- Note that you can keep using
- Please use
- Use
http://127.0.0.1:4723as the default server url instead ofhttp://127.0.0.1:4444/wd/hub
Quick migration guide from v3 to v4
- Removal
MultiActionandTouchActionare removed. Please use W3C WebDriver actions ormobile:extensions- appium/webdriver/extensions/action_helpers.py
- https://www.selenium.dev/documentation/webdriver/actions_api/
- https://www.youtube.com/watch?v=oAJ7jwMNFVU
- https://appiumpro.com/editions/30-ios-specific-touch-action-methods
- https://appiumpro.com/editions/29-automating-complex-gestures-with-the-w3c-actions-api
- Deprecated
AppiumBy.WINDOWS_UI_AUTOMATION, which has no usage right now.
Quick migration guide from v2 to v3
optionskeyword argument in thewebdriver.Remoteconstructor such asXCUITestOptionsinstead ofdesired_capabilities- Available options are https://github.com/appium/python-client/tree/master/appium/options
- Please check the Usage below as an example.
- Not a "new" change, but the
desired_capabilitiesargument has been removed since v3.
- Available options are https://github.com/appium/python-client/tree/master/appium/options
- Replacement
start_activitymethod: Please usemobile: startActivitylaunch_app,close_appandresetmethods: Please refer to https://github.com/appium/appium/issues/15807available_ime_engines,is_ime_active,activate_ime_engine,deactivate_ime_engineandactive_ime_enginemethods: Please usemobile: shellset_valueandset_textmethods: Please useelement.send_keysorsend_keysby W3C Actions
- Removal
end_test_coveragemethod is no longer availablesessionproperty is no longer availableall_sessionsproperty is no longer available
Quick migration guide from v1 to v2
- Enhancement
- Updated base Selenium Python binding version to v4
- Removed
forceMjsonwpsince Selenium v4 and Appium Python client v2 expect only W3C WebDriver protocol
- Removed
- Methods
ActionHelpers#scroll,ActionHelpers#drag_and_drop,ActionHelpers#tap,ActionHelpers#swipeandActionHelpers#flicknow call W3C actions as its backend- Please check each behavior. Their behaviors could slightly differ.
- Added
strict_sslto relax SSL errors such as self-signed ones
- Updated base Selenium Python binding version to v4
- Deprecated
MultiActionandTouchActionare deprecated. Please use W3C WebDriver actions ormobile:extensionslaunch_app,close_app, andresetare deprecated. Please read issues#15807 for more details
MultiAction/TouchAction to W3C actions
Some elements can be handled with touch pointer action instead of the default mouse pointer action in the Selenium Python client.
For example, the below action builder is to replace the default one with the touch pointer action.
from selenium.webdriver import ActionChains
from selenium.webdriver.common.actions import interaction
from selenium.webdriver.common.actions.action_builder import ActionBuilder
from selenium.webdriver.common.actions.pointer_input import PointerInput
actions = ActionChains(driver)
# override as 'touch' pointer action
actions.w3c_actions = ActionBuilder(driver, mouse=PointerInput(interaction.POINTER_TOUCH, "touch"))
actions.w3c_actions.pointer_action.move_to_location(start_x, start_y)
actions.w3c_actions.pointer_action.pointer_down()
actions.w3c_actions.pointer_action.pause(2)
actions.w3c_actions.pointer_action.move_to_location(end_x, end_y)
actions.w3c_actions.pointer_action.release()
actions.perform()
- appium/webdriver/extensions/action_helpers.py
- https://www.selenium.dev/documentation/webdriver/actions_api/
Usage
The Appium Python Client is fully compliant with the WebDriver Protocol including several helpers to make mobile testing in Python easier.
To use the new functionality now, and to use the superset of functions, instead of
including the Selenium webdriver module in your test code, use that from
Appium instead.
from appium import webdriver
From there much of your test code will work with no change.
As a base for the following code examples, the following set up the UnitTest environment:
# Python/Pytest
import pytest
from appium import webdriver
# Options are only available since client version 2.3.0
# If you use an older client then switch to desired_capabilities
# instead: https://github.com/appium/python-client/pull/720
from appium.options.android import UiAutomator2Options
from appium.options.ios import XCUITestOptions
from appium.webdriver.appium_service import AppiumService
from appium.webdriver.common.appiumby import AppiumBy
APPIUM_PORT = 4723
APPIUM_HOST = '127.0.0.1'
# HINT: fixtures below could be extracted into conftest.py
# HINT: and shared across all tests in the suite
@pytest.fixture(scope='session')
def appium_service():
service = AppiumService()
service.start(
# Check the output of `appium server --help` for the complete list of
# server command line arguments
args=['--address', APPIUM_HOST, '-p', str(APPIUM_PORT)],
timeout_ms=20000,
)
yield service
service.stop()
def create_ios_driver(custom_opts = None):
options = XCUITestOptions()
options.platformVersion = '13.4'
options.udid = '123456789ABC'
if custom_opts is not None:
options.load_capabilities(custom_opts)
# Appium1 points to http://127.0.0.1:4723/wd/hub by default
return webdriver.Remote(f'http://{APPIUM_HOST}:{APPIUM_PORT}', options=options)
def create_android_driver(custom_opts = None):
options = UiAutomator2Options()
options.platformVersion = '10'
options.udid = '123456789ABC'
if custom_opts is not None:
options.load_capabilities(custom_opts)
# Appium1 points to http://127.0.0.1:4723/wd/hub by default
return webdriver.Remote(f'http://{APPIUM_HOST}:{APPIUM_PORT}', options=options)
@pytest.fixture
def ios_driver_factory():
return create_ios_driver
@pytest.fixture
def ios_driver():
# prefer this fixture if there is no need to customize driver options in tests
driver = create_ios_driver()
yield driver
driver.quit()
@pytest.fixture
def android_driver_factory():
return create_android_driver
@pytest.fixture
def android_driver():
# prefer this fixture if there is no need to customize driver options in tests
driver = create_android_driver()
yield driver
driver.quit()
def test_ios_click(appium_service, ios_driver_factory):
# Usage of the context manager ensures the driver session is closed properly
# after the test completes. Otherwise, make sure to call `driver.quit()` on teardown.
with ios_driver_factory({
'appium:app': '/path/to/app/UICatalog.app.zip'
}) as driver:
el = driver.find_element(by=AppiumBy.ACCESSIBILITY_ID, value='item')
el.click()
def test_android_click(appium_service, android_driver_factory):
# Usage of the context manager ensures the driver session is closed properly
# after the test completes. Otherwise, make sure to call `driver.quit()` on teardown.
with android_driver_factory({
'appium:app': '/path/to/app/test-app.apk',
'appium:udid': '567890',
}) as driver:
el = driver.find_element(by=AppiumBy.ACCESSIBILITY_ID, value='item')
el.click()
Available options
Appium Python Client has a common options class named AppiumOptions but the available commands are minimal.
It does not have driver/automationName specific commands unless adding commands with add_command method.
Available options for each automation name below will help to check what options are already defined. Please use proper options for your automaiton usage.
automationName |
Package path |
|---|---|
| any | appium.options.common.base.AppiumOptions |
uiautomator2 |
appium.options.android.Uiautomator2Options |
espresso |
appium.options.android.EspressoOptions |
xcuitest |
appium.options.ios.XCUITestOptions |
safari |
appium.options.ios.SafariOptions |
mac2 |
appium.options.mac.Mac2Options |
windows |
appium.options.WindowsOptions |
gecko |
appium.options.GeckoOptions |
flutterintegration |
appium.options.flutter_integration.FlutterOptions |
Direct Connect URLs
If your Selenium/Appium server decorates the new session capabilities response with the following keys:
directConnectProtocoldirectConnectHostdirectConnectPortdirectConnectPath
Then python client will switch its endpoint to the one specified by the values of those keys.
from appium import webdriver
# Options are only available since client version 2.3.0
# If you use an older client then switch to desired_capabilities
# instead: https://github.com/appium/python-client/pull/720
from appium.options.ios import XCUITestOptions
from appium.webdriver.client_config import AppiumClientConfig
# load_capabilities API could be used to
# load options mapping stored in a dictionary
options = XCUITestOptions().load_capabilities({
'platformVersion': '13.4',
'deviceName': 'iPhone Simulator',
'app': '/full/path/to/app/UICatalog.app.zip',
})
client_config = AppiumClientConfig(
remote_server_addr='http://127.0.0.1:4723',
direct_connection=True
)
driver = webdriver.Remote(
# Appium1 points to http://127.0.0.1:4723/wd/hub by default
'http://127.0.0.1:4723',
options=options,
client_config=client_config
)
Relax SSL validation
strict_ssl option allows you to send commands to an invalid certificate host like a self-signed one.
from appium import webdriver
# Options are only available since client version 2.3.0
# If you use an older client then switch to desired_capabilities
# instead: https://github.com/appium/python-client/pull/720
from appium.options.common import AppiumOptions
options = AppiumOptions()
options.platform_name = 'mac'
options.automation_name = 'safari'
# set_capability API allows to provide any custom option
# calls to it could be chained
options.set_capability('browser_name', 'safari')
# Appium1 points to http://127.0.0.1:4723/wd/hub by default
driver = webdriver.Remote('http://127.0.0.1:4723', options=options, strict_ssl=False)
Since Appium Python client v4.3.0, we recommend using selenium.webdriver.remote.client_config.ClientConfig
instead of giving strict_ssl as an argument of webdriver.Remote below to configure the validation.
from appium import webdriver
from selenium.webdriver.remote.client_config import ClientConfig
client_config = ClientConfig(
remote_server_addr='http://127.0.0.1:4723',
ignore_certificates=True
)
driver = webdriver.Remote(client_config.remote_server_addr, options=options, client_config=client_config)
Set custom AppiumConnection
The first argument of webdriver.Remote can set an arbitrary command executor for you.
- Set init arguments for the pool manager Appium Python client uses to manage HTTP requests.
from appium import webdriver
from appium.options.ios import XCUITestOptions
import urllib3
from appium.webdriver.appium_connection import AppiumConnection
# Retry connection error up to 3 times.
init_args_for_pool_manage = {
'retries': urllib3.util.retry.Retry(total=3, connect=3, read=False)
}
appium_executor = AppiumConnection(
remote_server_addr='http://127.0.0.1:4723',
init_args_for_pool_manage=init_args_for_pool_manage
)
options = XCUITestOptions()
options.platformVersion = '13.4'
options.udid = '123456789ABC'
options.app = '/full/path/to/app/UICatalog.app.zip'
driver = webdriver.Remote(appium_executor, options=options)
- Define a subclass of
AppiumConnection
from appium import webdriver
from appium.options.ios import XCUITestOptions
from appium.webdriver.appium_connection import AppiumConnection
class CustomAppiumConnection(AppiumConnection):
# Can add your own methods for the custom class
pass
custom_executor = CustomAppiumConnection(remote_server_addr='http://127.0.0.1:4723')
options = XCUITestOptions().load_capabilities({
'platformVersion': '13.4',
'deviceName': 'iPhone Simulator',
'app': '/full/path/to/app/UICatalog.app.zip',
})
driver = webdriver.Remote(custom_executor, options=options)
The AppiumConnection can set selenium.webdriver.remote.client_config.ClientConfig as well.
Relaxing HTTP request read timeout
Appium Python Client has 120 seconds read timeout on each HTTP request since the version v4.3.0 because of
the corresponding selenium binding version.
You have two methods to extend the read timeout.
- Recommend Configure timeout via
appium.webdriver.client_config.AppiumClientConfigorselenium.webdriver.remote.client_config.ClientConfigtimeoutargument, orinit_args_for_pool_managerargument forurllib3.PoolManager
- Set
GLOBAL_DEFAULT_TIMEOUTenvironment variable- This env var will be removed from the selenium binding (issue)
Documentation
- https://appium.github.io/python-client-sphinx/ is detailed documentation
- functional tests also may help to see concrete examples.
Development
- Code Style: PEP-0008
- Apply
ruffas pre commit hook - Run
makecommand for development. Seemake helpoutput for details
- Apply
- Docstring style: Google Style
Setup
make install-uv
exec $SHELL
make sync-dev
Running above commands should automatically setup the virtual environment for the project
using the default system Python version and put it into the .venv folder under the project root.
If you'd like to customize the Python version then run the following command before make sync-dev:
uv venv --python <V>
where <V> is the actual Python version, for example 3.12.
If you want to customize the folder where uv stores the virtual environment by default
(e.g. .venv) then add an argument containing the destination folder path to the above command:
uv venv /venv/root/folder
In order to activate the newly created virtual environment you may either source it:
source /venv/root/folder/bin/activate
or add it to PATH:
export "PATH=/venv/root/folder/bin:$PATH"
Linting And Formatting
Run linter and format checks
make check
Address autofixable linter and formatting issues
make fix
Testing
Unit
make unittest
Run in parallel (2 threads)
make unittest ARGS="-n 2"
Functional
uv run pytest test/functional/ios/search_context/find_by_ios_class_chain_tests.py
In parallel for iOS
- Create simulators named 'iPhone X - 8100' and 'iPhone X - 8101'
- Run tests
uv run pytest -n 2 test/functional/ios/search_context/find_by_ios_class_chain_tests.py
Release
Follow the below steps.
Set GH_TOKEN env var to update the GitHub contents.
rm -rf dist
# bumping the version, building a package and creating a tag.
uv run semantic-release version --patch|--minor|--major
# To publish the version's module to pypi
UV_PUBLISH_TOKEN=<token to publish via pypi> uv publish
# To push built modules in 'dist' directory to the GH release page.
uv run semantic-release publish
License
Apache License v2
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 appium_python_client-5.2.4.tar.gz.
File metadata
- Download URL: appium_python_client-5.2.4.tar.gz
- Upload date:
- Size: 212.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eafeee9ff1ad8ac3052d2f706dca5cb52ebf51b5335f1f7102c71a663ee18c13
|
|
| MD5 |
63dadef8643f4f806f7f5a1281485177
|
|
| BLAKE2b-256 |
6ac527f5768feb757a9e8a77b4c1681bae4a2be3117e654ba4fe0e1fc3c45cee
|
File details
Details for the file appium_python_client-5.2.4-py3-none-any.whl.
File metadata
- Download URL: appium_python_client-5.2.4-py3-none-any.whl
- Upload date:
- Size: 348.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.8.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1c8ccdc1b6e5c88e9835f4d81d338cae574e74114a85d5ffaa049d9b0fb4c32a
|
|
| MD5 |
460793d16406dfa47a3b258a5f1c72bc
|
|
| BLAKE2b-256 |
5048758bea06efad9c32eab0ade089d5fea230bb5009921a7d6de8564b98585e
|