Python Automation Testing With Examples
This blog explores Python automation testing for the frontend, with insights to leverage Python for automating both simple and complex scenarios effectively.
Join the DZone community and get the full member experience.
Join For FreeWhen it comes to automating front-end tests, choosing the ideal programming language becomes extremely crucial. Python is one such language that tops the list, owing to ease of use and extensive community support.
Moreover, Python automation testing lets you harness the capabilities offered by popular libraries and frameworks like Selenium, pytest, and Robot, amongst others. Using Selenium with Python helps in making the most of Selenium’s powerful browser automation capabilities and Python’s simplicity and extensibility. Overall, Python automation testing is widely used by QAs all over, especially with Selenium WebDriver.
In this blog, we will dive deep into the nuances of Python from a front-end perspective. The learnings of this blog will be useful in leveraging Python capabilities for automating simple as well as complex front-end scenarios.
What Is Python Automation Testing?
As the name suggests, Python automation testing is the process of using Python scripts for automating tests. It is one of the most sought-after programming languages for the automation of manual and repetitive tasks.
A simple example of Python automation testing involves leveraging the pytest framework with Selenium to automate front-end tests on an eCommerce platform. You can verify the registration functionality of an eCommerce website and use the same session to navigate through the site and add required items to the shopping cart. By the end of this test, you will have verified the registration, login, and cart functionalities of the eCommerce website.
Lastly, you can also harness Python for automating system administration, automating emails, and data processing tasks. Since Python is a prerequisite for automated tests, please refer to the video below, which dives deeply into the installation of Python.
Why Python for Automated Testing?
Now that we know why testers prefer Selenium with Python, let’s look at some of the essential reasons to choose Python for automation testing:
1. Wide Range of Libraries and Frameworks
PyUnit (or unittest) is the default test framework for performing unit testing with Python. Though PyUnit is available out of the box, Python supports other popular frameworks like pytest, Behave, Robot, Lettuce, and Nose2.
All of them can be used exhaustively with Selenium and Playwright frameworks for automating web browser tests.
2. Super Easy Parallel Test Execution
Parallel testing in Selenium and Python can be used extensively to perform web browser automation tests across different combinations of browsers and platforms. Though all Selenium-supported languages support parallel test execution, it is darn easy to use with Python.
3. Multi-Paradigm Programming Language
Python is a multi-paradigm programming language. Hence, there is full-fledged support for object-oriented programming and structured programming. A majority of the features in Python support functional programming and aspect-oriented programming. Python, along with Selenium, can also be used to test websites and web applications functionally.
4. Dynamic Typing
The Python language uses dynamic typing and late binding (or dynamic name resolution) that binds the methods and variable names during execution. This feature is super handy for Python test automation.
Python also offers options like Pyre (a performant type checker for Python 3) and Mypy, which are popular static type checkers. With these checkers, Python lets you combine the power of dynamic and static typing.
5. Web Scraping
Web scraping with Python is the process of extracting meaningful and useful information/data from websites. It is primarily used for academic research, competitor analysis, content aggregation, and more.
Python offers a number of libraries and frameworks, namely BeautifulSoup (bs4), Selenium, Puppeteer, and Pyppeteer, that ease the task of scraping content from websites.
6. Powerful and Hassle-Free Reporting
Reporting in test automation provides greater visibility into the nuances of test execution (i.e., percentage of tests passed/failed, test environment, screenshots, etc.). Powerful reports that give the right information in a concise and understandable form can be sent to necessary stakeholders (in the team) so that they are aware of the progress on the testing front.
How to Perform Python Automation Testing?
Now that we have looked into the importance of Python automation testing, let’s get our hands dirty by running some tests. Our discussion will mainly center around front-testing automated testing with Python.
Before we get started with the tests, let’s set up the virtual environment (venv
), which helps manage the environment and dependencies better. venv
plays an instrumental role in providing isolation from the packages installed in the base environment.
Run the commands virtualenv venv
and source venv
/bin
/activate
on the terminal to create the virtual environment. All the dependencies (or Python packages), like pytest
, selenium
, etc., required for project execution are available in the requirements.txt file.
pytest-selenium
pytest-xdist
selenium>=4.6.0
urllib3==1.26.12
requests
py
Dependencies
The dependencies can be installed by triggering pip install -r requirements.txt
on the terminal. Selenium v4.6.0 (or above) is installed as a part of the installation procedure.
For demonstration, we would execute simple Selenium Python tests with pytest and PyUnit (or unittest) frameworks. If you are well-versed in either of these frameworks, it is recommended that you use fixtures in Python and the Page Object Model in Selenium Python for improved maintenance of the tests. The Selenium package itself doesn’t provide a testing tool or framework. Hence, we would be using Selenium with pytest and PyUnit for automating interactions with the elements on the web page.
Test Scenario
- Navigate to the LambdaTest Selenium Playground.
- Locate the Input Form Submit link on the page.
- Input the required information on the page.
- Submit details, and assert if the information has not been submitted successfully.
Implementation (pytest Framework)
Below is the test script for the above test scenario:
import os
import pytest
from os import environ
import time
from selenium.webdriver import ChromeOptions
from selenium.webdriver.support.ui import WebDriverWait
from selenium import webdriver
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.ui import Select
from selenium.webdriver.common.by import By
exec_platform = os.getenv('EXEC_PLATFORM')
time_sleep = 2
########################## Locators #########################
xSubmitForm = "//a[.='Input Form Submit']"
xInpName = "//input[@id='name']"
cInpName = "#name"
xInpEmail = "//form[@id='seleniumform']//input[@name='email']"
xInpPassword = "//input[@name='password']"
cssCompany = "#company"
cWebName = "#websitename"
xInpCountry = "//select[@name='country']"
xInpCity = "//input[@id='inputCity']"
cssAddress1 = "[placeholder='Address 1']"
cssAddress2 = "[placeholder='Address 2']"
cssInpState = "#inputState"
cssInpZip = "#inputZip"
cssInpButton = ".bg-lambda-900"
nameSearchBox = "search"
class TestFormInput:
def setup_method(self):
if exec_platform == 'cloud':
username = environ.get('LT_USERNAME', None)
access_key = environ.get('LT_ACCESS_KEY', None)
ch_options = webdriver.ChromeOptions()
lt_options = {}
lt_options["build"] = "Build: Getting Started with Selenium PyTest"
lt_options["project"] = "Project: Getting Started with Selenium PyTest"
lt_options["name"] = "Test: Getting Started with Selenium PyTest"
lt_options["browserName"] = "Chrome"
lt_options["browserVersion"] = "latest"
lt_options["platformName"] = "macOS Sonoma"
lt_options["geoLocation"] = "US"
lt_options["console"] = "error"
lt_options["w3c"] = True
lt_options["headless"] = False
ch_options.set_capability('LT:Options', lt_options)
gridURL = "https://{}:{}@hub.lambdatest.com/wd/hub".format(username, access_key)
self.driver = webdriver.Remote(
command_executor = gridURL,
options = ch_options
)
elif exec_platform == 'local':
ch_options = ChromeOptions()
self.driver = webdriver.Chrome(options=ch_options)
def test_enter_form_details(self):
resultant_str = "Thanks for contacting us, we will get back to you shortly."
driver = self.driver
driver.get("https://www.lambdatest.com/selenium-playground/")
# Commented once the tests are executed in non-headless mode
driver.maximize_window()
wait = WebDriverWait(driver, 5)
try:
element = driver.find_element(By.XPATH, xSubmitForm)
element.click()
elem_name = driver.find_element(By.XPATH, xInpName)
elem_name.send_keys("Testing")
time.sleep(time_sleep)
elem_email = driver.find_element(By.XPATH, xInpEmail)
elem_email.send_keys("testing@testing.com")
time.sleep(time_sleep)
elem_pass = driver.find_element(By.XPATH, xInpPassword)
elem_pass.send_keys("password")
time.sleep(time_sleep)
elem_comp = driver.find_element(By.CSS_SELECTOR, cssCompany)
elem_comp.send_keys("LambdaTest")
elem = driver.find_element(By.CSS_SELECTOR, cWebName)
elem.send_keys("https://wwww.lambdatest.com")
country_dropdown = Select(driver.find_element(By.XPATH, xInpCountry))
country_dropdown.select_by_visible_text("United States")
time.sleep(time_sleep)
elem = driver.find_element(By.XPATH, xInpCity)
elem.send_keys("San Jose")
time.sleep(time_sleep)
elem = driver.find_element(By.CSS_SELECTOR, cssAddress1)
elem.send_keys("Googleplex, 1600 Amphitheatre Pkwy")
time.sleep(time_sleep)
elem = driver.find_element(By.CSS_SELECTOR, cssAddress2)
elem.send_keys("Mountain View, CA 94043")
time.sleep(time_sleep)
elem = driver.find_element(By.CSS_SELECTOR, cssInpState)
elem.send_keys("California")
time.sleep(time_sleep)
elem = driver.find_element(By.CSS_SELECTOR, cssInpZip)
elem.send_keys("94088")
time.sleep(time_sleep)
# Click on the Submit button
submit_button = driver.find_element(By.CSS_SELECTOR, cssInpButton)
submit_button.click()
time.sleep(2)
try:
element = wait.until(
EC.presence_of_element_located((By.CSS_SELECTOR, ".success-msg"))
)
assert resultant_str in element.text, f"'{resultant_str}' not found in the specified element."
except Exception as e:
if exec_platform == 'cloud':
driver.execute_script("lambda-status=failed")
pytest.fail(f"Text '{resultant_str}' not found: {str(e)}")
time.sleep(2)
except Exception as e:
# Catch other exceptions
print(f"Failed: Input Form Demo, generic exception - {e}")
if exec_platform == 'cloud':
driver.execute_script("lambda-status=failed")
if exec_platform == 'cloud':
driver.execute_script("lambda-status=passed")
print(f"PyTest Demo: Test Passed")
def teardown_method(self):
if (self.driver != None):
# self.driver.close()
self.driver.quit()
if __name__ == "__main__":
pytest.main()
Code Walkthrough
To get started, we first import the necessary modules required for the implementation of tests. Since we are using the pytest, the pytest
module is also imported into the code.
The WebDriverWait
class from the selenium.webdriver.support.ui module is imported so that we can use explicit waits for scenarios where WebElements are located dynamically. The expected_conditions
module provides a set of pre-defined ExpectedConditions in Selenium that can be used with explicit waits.
As the elements need to be located before we perform any action, we first define the element locators of the required elements. ID, Name, XPath, Link Text, Partial Link Text, etc., are some of the widely used web locators that help you find elements within the Document Object Model (DOM).
You can either use the inspect tools available natively with the web browser or a plugin (or add-on) like POM Builder that makes it easy to find the XPath/CSS Selector of WebElements.
Once the web locator is identified, the respective WebElement(s) is/are located by combining the locator with the find_element()
or find_elements()
method of Selenium Python. The find_element()
method returns a single WebElement, whereas find_elements()
returns a list of WebElements matching the locator criteria.
As stated earlier, the setup_method()
is a pytest fixture that is a part of the initialization. The method is called before every test function is implemented under the said test class.
The tests being implemented can be run on Selenium installed on the local machine as well as on the online Selenium grid offered by cloud testing. LambdaTest is an AI-powered test execution platform that lets you run Python automated tests at scale across different browsers and operating systems. It comes with a number of benefits, the major ones being reduced maintenance, lower costs, and accelerated test execution.
As far as the implementation is concerned, the only change is related to Selenium WebDriver, where Remote WebDriver in Selenium is instantiated when running tests on a cloud grid. The Automation Capabilities Generator helps generate capabilities for the test combination intended to be used in the tests.
In Selenium 4, browser options are used in place of Desired Capabilities. You can check out Selenium 3 vs Selenium 4 differences blog to know more about what’s deprecated and what’s deprecated in Selenium 4.
The environment variables LT_USERNAME
and LT_ACCESS_KEY
can be obtained from your LambdaTest Account Settings > Password & Security. The combination, when passed along with the LambdaTest grid URL, helps in executing tests on the cloud grid. In the test_enter_form_details()
method, we first navigate to the test URL by invoking the driver.get()
method.
Next, the instantiated browser window is maximized since it is considered one of the best practices of Selenium.
Next, the Input Form Submit element is located using the find_element()
method along with the XPath locator in Selenium. Once located, the button click in Selenium is invoked to simulate click action on the button element located in the earlier step.
Since all the test steps involve locating elements and performing actions, we focus on only a few methods. The approach used for locating the company element via the CSS Selector in Selenium is shown below. The send_keys()
in Selenium WebDriver helps in sending text input in the located element.
Dropdowns are widely used on websites; hence, automating interactions with dropdowns using Selenium becomes an absolute must for your test cases. The Select
class of the selenium.webdriver.support.ui module provides methods that let you handle dropdowns with Selenium.
Here, an object (i.e., country_dropdown
) of the Select
class is created with the input set to the dropdown WebElement located with XPath. The select_by_visible_text()
method of the Select
class helps select items with visible text matching the given string (i.e., United States).
Once all the information in the form is entered and the Submit button is clicked, we wait till the success message string (default: hidden) is visible on the page. An explicit wait with the ExpectedCondition (i.e., presence of located element) is performed until the success message is not on the page.
An assert is raised if the resultant string is not present on the page. For execution on the cloud, the lambda-status
variable is marked as passed/failed depending on the status of execution.
The teardown_method()
fixture contains implementation for cleaning up resources or state after execution of the tests in the class. The if __name__ == "__main__":
construct ensures that code execution is performed only when the script is run directly. The pytest.main()
calls the pytest framework that further discovers and executes all the enabled (i.e., not marked as skipped) tests in the script.
Test Execution (pytest Framework)
After setting EXEC_PLATFORM
to cloud, invoke the following command on the terminal to run pytest tests on the LambdaTest cloud grid:
pytest --verbose --capture=no tests/pytest/pytest_selenium_demo.py
Below is the LambdaTest Web Automation dashboard screenshot that indicates that the test execution was successful:
Implementation (PyUnit framework)
The test script for the above-mentioned test scenario using the PyUnit framework is located in the tests/pyunit/pyunit_selenium_demo.py.
The core logic of the test remains unchanged when migrating from pytest to PyUnit (or unittest) framework. Instead of the pytest
module, the unittest
module is imported into the code. The test case class is inherited from the unittest.TestCase
informs the unittest module that this is a test case.
pytest fixtures setup_method()
/teardown()
are similar to the setUp()
/tearDown()
methods of the PyUnit framework. The setUp()
and tearDown()
methods consist of implementation whose responsibility is initialization and deinitialization, respectively.
Shown below is the boilerplate code to execute the test suite:
if __name__ == "__main__":
unittest.main()
Test Execution (PyUnit Framework)
After setting EXEC_PLATFORM
to cloud, invoke the following command on the terminal to run PyUnit tests on the cloud grid:
python tests/pyunit/pyunit_selenium_demo.py
Shown below is the LambdaTest Web Automation dashboard screenshot that indicates that the test execution was successful:
In case you want to run the above tests on Selenium installed on the local machine, simply set EXEC_PLATFORM
to local, and you are all set with the local execution.
Top Python Testing Frameworks
Since Python supports a number of test automation frameworks, choosing the right framework becomes extremely crucial for your project. The choice actually lays the foundation for efficient testing. Apart from the framework’s capabilities, you also need to have a look at the internal expertise with the said framework. Below are some of the best Python testing frameworks:
PyUnit (unittest)
It is the default framework available in Python. As the name suggests, it is primarily used for unit testing. PyUnit is inspired by the JUnit framework for Java and shares a similar structure and functionality.
The unittest framework also supports test automation sharing of setup and shutdown code for tests, independence of the tests from the reporting framework, and more. It also supports test suites and test cases that too in an object-oriented way. It also has a test runner which is a component that is responsible for orchestrating the execution of tests.
pytest
It is one of the most popular test automation frameworks for Python. It uses a less verbose and user-friendly syntax for implementing tests. pytest can be leveraged for implementing unit tests as well as complex functional tests for testing websites and web applications.
Tests written using pytest are a lot more compact, as the framework does not require boilerplate code. pytest has built-in features that help with the auto-discovery of test modules and functions.
Robot
It is a keyword driven open-source Python framework that is majorly used for Robot Process Automation (RPA) and test automation. Akin to the pytest framework, Robot is also extensible. The usage of human-readable syntax/keywords minimizes the learning curve involved in learning Robot.
Tests written in Robot are saved with a .robot
extension. In case you plan to use Robot for front-end tests, you can do so with the SeleniumLibrary, which is a web testing library for Robot Framework. The library supports an exhaustive range of keywords (Click Button, Click Image, Open Browser, Drag and Drop, among others).
Nose2
It is the successor to Nose and is a Python testing framework that extends the capabilities of the PyUnit (or unittest) framework. It is relatively easy to get started with Nose2 if you have prior working experience with unittest.
The major upside of Nose2 over PyUnit is the availability of a large number of built-in Nose plugins that make testing easier and faster. Plugins in Nose2 help in test parameterization, better organization of tests, fixtures support, test discovery, and more.
Behave
It is a Python framework that is used for Behavior-Driven Development (BDD). The tests are based on the Gherkin syntax, which is based on the Given-When-Then format.
Since the tests are implemented in scenario and feature files, even non-technical personnel can be a part of the QA process. SpecFlow (C#) and Cucumber (Java, JS, Ruby) are some of the other popular BDD frameworks.
Conclusion
As seen so far, Python is by far the best scripting language for test automation. It is relatively easy to get started with Python automation testing. Its wide range of test frameworks can be used for unit testing, cross-browser testing, and more. Let us know below your preferred programming language for Selenium testing and how you rate it against Python, the undisputed king for automation testing. Happy testing!
Published at DZone with permission of Himanshu Sheth. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments